:: :: :: :::: /hoon/drum/lib :: :: :: :: :: /? 310 :: version /- sole /+ sole [. ^sole] !: :: :: :::: :: :: :: :: :: |% :: :: ++ drum-part {$drum $1 drum-pith-1} :: ++ drum-part-old {$drum $0 drum-pith-0} :: ++ drum-pith-0 _!! :: forgotten ++ drum-pith-1 :: $: sys/(unit bone) :: local console eel/(set gill) :: connect to ray/(set well) :: fur/(map dude (unit server)) :: servers bin/(map bone source) :: terminals == :: :: :: :: :::: :: :: :: :: :: ++ server :: running server $: syd/desk :: app identity cas/case :: boot case == :: ++ kill :: kill ring $: pos/@ud :: ring position num/@ud :: number of entries max/_60 :: max entries old/(list (list @c)) :: entries proper == :: ++ source :: input device $: edg/_80 :: terminal columns off/@ud :: window offset kil/kill :: kill buffer inx/@ud :: ring index fug/(map gill (unit target)) :: connections mir/(pair @ud (list @c)) :: mirrored terminal == :: ++ history :: past input $: pos/@ud :: input position num/@ud :: number of entries lay/(map @ud (list @c)) :: editing overlay old/(list (list @c)) :: entries proper == :: ++ search :: reverse-i-search $: pos/@ud :: search position str/(list @c) :: search string == :: ++ target :: application target $: blt/(pair (unit dill-belt) (unit dill-belt)) :: curr & prev belts ris/(unit search) :: reverse-i-search hit/history :: all past input pom/sole-prompt :: static prompt inp/sole-command :: input state == :: -- :: :: :: :::: :: :: :: :: :: |% ++ deft-apes :: default servers |= our/ship %- ~(gas in *(set well)) ^- (list well) =+ myr=(clan our) ?: ?=($pawn myr) [[%base %talk] [%base %dojo] ~] ?: ?=($earl myr) [[%home %dojo] ~] [[%home %talk] [%home %dojo] ~] :: ++ deft-fish :: default connects |= our/ship %- ~(gas in *(set gill)) ^- (list gill) ?: ?=($earl (clan our)) [[(sein our) %talk] [our %dojo] ~] [[our %talk] [our %dojo] ~] :: ++ drum-make :: initial part |= our/ship ^- drum-part :* %drum %1 ~ :: sys (deft-fish our) :: eel (deft-apes our) :: ray ~ :: fur ~ :: bin == :: :: ++ drum-port |= old/?(drum-part drum-part-old) ^- drum-part ?- &2.old $1 old $0 !! :: XX unreachable, see issue #242 == :: ++ drum-path :: encode path |= gyl/gill ^- wire [%drum %phat (scot %p p.gyl) q.gyl ~] :: ++ drum-phat :: decode path |= way/wire ^- gill ?>(?=({@ @ $~} way) [(slav %p i.way) i.t.way]) -- !: :::: :: |= {bowl drum-part} :: main drum work =+ (fall (~(get by bin) ost) *source) =* dev - => |% :: arvo structures ++ pear :: request $% {$sole-action p/sole-action} :: {$talk-command command:talk} :: == :: ++ lime :: update $% {$dill-blit dill-blit} :: == :: ++ card :: general card $% {$conf wire dock $load ship term} :: {$diff lime} :: {$peer wire dock path} :: {$poke wire dock pear} :: {$pull wire dock $~} :: == :: ++ move (pair bone card) :: user-level move -- |_ {moz/(list move) biz/(list dill-blit)} ++ diff-sole-effect-phat :: app event |= {way/wire fec/sole-effect} =< se-abet =< se-view =+ gyl=(drum-phat way) ?: (se-aint gyl) +>.$ (se-diff gyl fec) :: ++ peer :: |= pax/path ~| [%drum-unauthorized our+our src+src] :: ourself ?> (team our src) :: or our own moon =< se-abet =< se-view (se-text "[{}, driving {}]") :: ++ poke-dill-belt :: terminal event |= bet/dill-belt =< se-abet =< se-view (se-belt bet) :: ++ poke-start :: start app |= wel/well =< se-abet =< se-view (se-born wel) :: ++ poke-link :: connect app |= gyl/gill =< se-abet =< se-view (se-link gyl) :: ++ poke-unlink :: disconnect app |= gyl/gill =< se-abet =< se-view (se-klin gyl) :: ++ poke-exit :: shutdown |=($~ se-abet:(se-blit-sys `dill-blit`[%qit ~])) :: :: ++ poke-put :: write file |= {pax/path txt/@} se-abet:(se-blit-sys [%sav pax txt]) :: :: ++ reap-phat :: ack connect |= {way/wire saw/(unit tang)} =< se-abet =< se-view =+ gyl=(drum-phat way) ?~ saw (se-join gyl) (se-dump:(se-drop & gyl) u.saw) :: ++ take-coup-phat :: ack poke |= {way/wire saw/(unit tang)} =< se-abet =< se-view ?~ saw +> =+ gyl=(drum-phat way) ?: (se-aint gyl) +>.$ %- se-dump:(se-drop & gyl) :_ u.saw >[%drum-coup-fail src ost gyl]< :: ++ take-onto :: ack start |= {way/wire saw/(each suss tang)} =< se-abet =< se-view ?> ?=({@ @ $~} way) ?> (~(has by fur) i.t.way) =/ wel/well [i.way i.t.way] ?- saw {$| *} (se-dump p.saw) {$& *} ?> =(q.wel p.p.saw) :: =. +>.$ (se-text "live {}") +>.$(fur (~(put by fur) q.wel `[p.wel %da r.p.saw])) == :: ++ quit-phat :: |= way/wire =< se-abet =< se-view =+ gyl=(drum-phat way) ~& [%drum-quit src ost gyl] (se-drop %| gyl) :: :: :: :::: :: :: :: :: :: ++ se-abet :: resolve ^- (quip move *drum-part) =* pith +>+>+<+ ?. se-ably =. . se-adit [(flop moz) pith] =. sys ?^(sys sys `ost) =. . se-subze:se-adze:se-adit :_ pith(bin (~(put by bin) ost dev)) %- flop ^- (list move) ?~ biz moz :_ moz [ost %diff %dill-blit ?~(t.biz i.biz [%mor (flop biz)])] :: ++ se-ably (~(has by sup) ost) :: caused by console :: ++ se-adit :: update servers ^+ . %+ roll (~(tap in ray)) =< .(con +>) |= {wel/well con/_..se-adit} ^+ con =. +>.$ con =+ hig=(~(get by fur) q.wel) ?: &(?=(^ hig) |(?=($~ u.hig) =(p.wel syd.u.u.hig))) +>.$ =. +>.$ (se-text "activated app {(trip p.wel)}/{(trip q.wel)}") %- se-emit(fur (~(put by fur) q.wel ~)) [ost %conf [%drum p.wel q.wel ~] [our q.wel] %load our p.wel] :: ++ se-adze :: update connections ^+ . %+ roll (~(tap in eel)) =< .(con +>) |= {gil/gill con/_.} ^+ con =. +>.$ con ?: (~(has by fug) gil) +>.$ (se-peer gil) :: ++ se-subze :: downdate connections =< .(dev (~(got by bin) ost)) =. bin (~(put by bin) ost dev) ^+ . %- ~(rep by bin) =< .(con +>) |= {{ost/bone dev/source} con/_.} ^+ con =+ xeno=se-subze-local:%_(con ost ost, dev dev) xeno(ost ost.con, dev dev.con, bin (~(put by bin) ost dev.xeno)) :: ++ se-subze-local ^+ . %- ~(rep by fug) =< .(con +>) |= {{gil/gill *} con/_.} ^+ con =. +>.$ con ?: (~(has in eel) gil) +>.$ (se-nuke gil) :: ++ se-aint :: ignore result |= gyl/gill ^- ? ?. (~(has by bin) ost) & =+ gyr=(~(get by fug) gyl) |(?=($~ gyr) ?=($~ u.gyr)) :: ++ se-alas :: recalculate index |= gyl/gill =+ [xin=0 wag=se-amor] |- ^+ +>.^$ ?~ wag +>.^$(inx 0) ?: =(i.wag gyl) +>.^$(inx xin) $(wag t.wag, xin +(xin)) :: ++ se-amor :: live targets ^- (list gill) %+ skim (~(tap in eel)) |=(a/gill ?=({$~ $~ *} (~(get by fug) a))) :: ++ se-anon :: rotate index =+ wag=se-amor ?~ wag + :: ~& [%se-anon inx+inx wag+wag nex+(mod +(inx) (lent se-amor))] +(inx (mod +(inx) (lent wag))) :: ++ se-agon :: current gill ^- (unit gill) =+ wag=se-amor ?~ wag ~ `(snag inx `(list gill)`wag) :: ++ se-belt :: handle input |= bet/dill-belt ^+ +> ?: ?=({?($cru $hey $rez $yow) *} bet) :: target-agnostic ?- bet {$cru *} (se-dump:(se-text (trip p.bet)) q.bet) {$hey *} +>(mir [0 ~]) :: refresh {$rez *} +>(edg (dec p.bet)) :: resize window {$yow *} ~&([%no-yow -.bet] +>) == =+ gul=se-agon ?: |(?=($~ gul) (se-aint u.gul)) (se-blit %bel ~) ta-abet:(ta-belt:(se-tame u.gul) bet) :: ++ se-born :: new server |= wel/well ^+ +> ?: (~(has in ray) wel) (se-text "[already running {}/{}]") %= +> ray (~(put in ray) wel) eel (~(put in eel) [our q.wel]) == :: ++ se-drop :: disconnect |= {pej/? gyl/gill} ^+ +> =+ lag=se-agon ?. (~(has by fug) gyl) +>.$ =. fug (~(del by fug) gyl) =. eel ?.(pej eel (~(del in eel) gyl)) =. +>.$ ?. &(?=(^ lag) !=(gyl u.lag)) +>.$(inx 0) (se-alas u.lag) =. +>.$ (se-text "[unlinked from {}]") ?: =(gyl [our %dojo]) :: undead dojo (se-link gyl) +>.$ :: ++ se-dump :: print tanks |= tac/(list tank) ^+ +> ?. se-ably (se-talk tac) =/ wol/wall (zing (turn (flop tac) |=(a/tank (~(win re a) [0 edg])))) |- ^+ +>.^$ ?~ wol +>.^$ $(wol t.wol, +>.^$ (se-blit %out (tuba i.wol))) :: ++ se-join :: confirm connection |= gyl/gill ^+ +> =. +> (se-text "[linked to {}]") ?> ?=($~ (~(got by fug) gyl)) (se-alas(fug (~(put by fug) gyl `*target)) gyl) :: ++ se-nuke :: teardown connection |= gyl/gill ^+ +> (se-drop:(se-pull gyl) & gyl) :: ++ se-klin :: disconnect app |= gyl/gill +>(eel (~(del in eel) gyl)) :: ++ se-link :: connect to app |= gyl/gill +>(eel (~(put in eel) gyl)) :: ++ se-blit :: give output |= bil/dill-blit +>(biz [bil biz]) :: ++ se-blit-sys :: output to system console |= bil/dill-blit ^+ +> ?~ sys ~&(%se-blit-no-sys +>) (se-emit [u.sys %diff %dill-blit bil]) :: ++ se-show :: show buffer, raw |= lin/(pair @ud (list @c)) ^+ +> ?: =(mir lin) +> =. +> ?:(=(q.mir q.lin) +> (se-blit %pro q.lin)) =. +> ?:(=(p.mir p.lin) +> (se-blit %hop p.lin)) +>(mir lin) :: ++ se-just :: adjusted buffer |= lin/(pair @ud (list @c)) ^+ +> =. off ?:((lth p.lin edg) 0 (sub p.lin edg)) (se-show (sub p.lin off) (scag edg (slag off q.lin))) :: ++ se-view :: flush buffer ^+ . =+ gul=se-agon ?: |(?=($~ gul) (se-aint u.gul)) + (se-just ta-vew:(se-tame u.gul)) :: ++ se-emit :: emit move |= mov/move %_(+> moz [mov moz]) :: ++ se-talk |= tac/(list tank) ^+ +> :: XX talk should be usable for stack traces, see urbit#584 which this change :: closed for the problems there ((slog (flop tac)) +>) ::(se-emit 0 %poke /drum/talk [our %talk] (said:talk our %drum now eny tac)) :: ++ se-text :: return text |= txt/tape ^+ +> ?. se-ably (se-talk [%leaf txt]~) (se-blit %out (tuba txt)) :: ++ se-poke :: send a poke |= {gyl/gill par/pear} (se-emit [ost %poke (drum-path gyl) gyl par]) :: ++ se-peer :: send a peer |= gyl/gill %- se-emit(fug (~(put by fug) gyl ~)) [ost %peer (drum-path gyl) gyl /sole] :: ++ se-pull :: cancel subscription |= gyl/gill (se-emit [ost %pull (drum-path gyl) gyl ~]) :: ++ se-tame :: switch connection |= gyl/gill ^+ ta ~(. ta [& gyl] (need (~(got by fug) gyl))) :: ++ se-diff :: receive results |= {gyl/gill fec/sole-effect} ^+ +> ta-abet:(ta-fec:(se-tame gyl) fec) :: ++ ta :: per target |_ $: $: liv/? :: don't delete gyl/gill :: target app == :: target :: target state == :: ++ ta-abet :: resolve ^+ ..ta ?. liv ?: (~(has in (deft-fish our)) gyl) (se-blit qit+~) (se-nuke gyl) ..ta(fug (~(put by fug) gyl ``target`+<+)) :: ++ ta-poke |=(a/pear +>(..ta (se-poke gyl a))) :: poke gyl :: ++ ta-act :: send action |= act/sole-action ^+ +> (ta-poke %sole-action act) :: ++ ta-aro :: hear arrow |= key/?($d $l $r $u) ^+ +> =. ris ~ ?- key $d ?. =(num.hit pos.hit) (ta-mov +(pos.hit)) ?: =(0 (lent buf.say.inp)) ta-bel (ta-hom:ta-nex %set ~) $l ?: =(0 pos.inp) ta-bel +>(pos.inp (dec pos.inp)) $r ?: =((lent buf.say.inp) pos.inp) ta-bel +>(pos.inp +(pos.inp)) $u ?:(=(0 pos.hit) ta-bel (ta-mov (dec pos.hit))) == :: ++ ta-bel :: beep .(..ta (se-blit %bel ~), q.blt ~) :: forget belt :: ++ ta-belt :: handle input |= bet/dill-belt ^+ +> ?< ?=({?($cru $hey $rez $yow) *} bet) :: target-specific =. blt [q.blt `bet] :: remember belt ?- bet {$aro *} (ta-aro p.bet) {$bac *} ta-bac {$ctl *} (ta-ctl p.bet) {$del *} ta-del {$met *} (ta-met p.bet) {$ret *} ta-ret {$txt *} (ta-txt p.bet) == :: ++ ta-det :: send edit |= ted/sole-edit ^+ +> (ta-act %det [[his.ven.say.inp own.ven.say.inp] (sham buf.say.inp) ted]) :: ++ ta-bac :: hear backspace ^+ . ?^ ris ?: =(~ str.u.ris) ta-bel .(str.u.ris (scag (dec (lent str.u.ris)) str.u.ris)) ?: =(0 pos.inp) ?~ buf.say.inp (ta-act %clr ~) ta-bel (ta-hom %del (dec pos.inp)) :: ++ ta-ctl :: hear control |= key/@ud ^+ +> =. ris ?.(?=(?($g $r) key) ~ ris) ?+ key ta-bel $a +>(pos.inp 0) $b (ta-aro %l) $c ta-bel $d ?: &(=(0 pos.inp) =(0 (lent buf.say.inp))) +>(liv |) ta-del $e +>(pos.inp (lent buf.say.inp)) $f (ta-aro %r) $g ?~ ris ta-bel (ta-hom(pos.hit num.hit, ris ~) [%set ~]) $k =+ len=(lent buf.say.inp) ?: =(pos.inp len) ta-bel (ta-kil %r [pos.inp (sub len pos.inp)]) $l +>(..ta (se-blit %clr ~)) $n (ta-aro %d) $p (ta-aro %u) $r ?~ ris +>(ris `[pos.hit ~]) ?: =(0 pos.u.ris) ta-bel (ta-ser ~) $t =+ len=(lent buf.say.inp) ?: |(=(0 pos.inp) (lth len 2)) ta-bel =+ sop=(sub pos.inp ?:(=(len pos.inp) 2 1)) (ta-hom (rep:edit [sop 2] (flop (swag [sop 2] buf.say.inp)))) $u ?: =(0 pos.inp) ta-bel (ta-kil %l [0 pos.inp]) $v ta-bel $w ?: =(0 pos.inp) ta-bel =+ sop=(ta-off %l %ace pos.inp) (ta-kil %l [(sub pos.inp sop) sop]) $x +>(..ta se-anon) $y ?: =(0 num.kil) ta-bel (ta-hom (cat:edit pos.inp (snag (sub num.kil pos.kil) old.kil))) == :: ++ ta-del :: hear delete ^+ . ?: =((lent buf.say.inp) pos.inp) .(..ta (se-blit %bel ~)) (ta-hom %del pos.inp) :: ++ ta-erl :: hear local error |= pos/@ud ta-bel(pos.inp (min pos (lent buf.say.inp))) :: ++ ta-err :: hear remote error |= pos/@ud (ta-erl (~(transpose sole say.inp) pos)) :: ++ ta-fec :: apply effect |= fec/sole-effect ^+ +> ?- fec {$bel *} ta-bel {$blk *} +> {$clr *} +>(..ta (se-blit fec)) {$det *} (ta-got +.fec) {$err *} (ta-err p.fec) {$mor *} |- ^+ +>.^$ ?~ p.fec +>.^$ $(p.fec t.p.fec, +>.^$ ^$(fec i.p.fec)) {$nex *} ta-nex {$pro *} (ta-pro +.fec) {$tan *} +>(..ta (se-dump p.fec)) {$sag *} +>(..ta (se-blit fec)) {$sav *} +>(..ta (se-blit fec)) {$txt *} +>(..ta (se-text p.fec)) {$url *} +>(..ta (se-blit fec)) == :: ++ ta-dog :: change cursor |= ted/sole-edit %_ +> pos.inp =+ len=(lent buf.say.inp) %+ min len |- ^- @ud ?- ted {$del *} ?:((gth pos.inp p.ted) (dec pos.inp) pos.inp) {$ins *} ?:((gte pos.inp p.ted) +(pos.inp) pos.inp) {$mor *} |- ^- @ud ?~ p.ted pos.inp $(p.ted t.p.ted, pos.inp ^$(ted i.p.ted)) {$nop *} pos.inp {$set *} len == == :: ++ ta-got :: apply change |= cal/sole-change =^ ted say.inp (~(receive sole say.inp) cal) (ta-dog ted) :: ++ ta-hom :: local edit |= ted/sole-edit ^+ +> =. +> (ta-det ted) (ta-dog(say.inp (~(commit sole say.inp) ted)) ted) :: ++ ta-jump :: buffer pos |= {dir/?($l $r) til/?($ace $edg $wrd) pos/@ud} ^- @ud %- ?:(?=($l dir) sub add) [pos (ta-off dir til pos)] :: ++ ta-kil :: kill selection |= {dir/?($l $r) sel/{@ @}} ^+ +> =+ buf=(swag sel buf.say.inp) %. (cut:edit sel) %= ta-hom kil ?. ?& ?=(^ old.kil) ?=(^ p.blt) ?| ?=({$ctl ?($k $u $w)} u.p.blt) ?=({$met ?($d $bac)} u.p.blt) == == %= kil :: prepend num +(num.kil) pos +(num.kil) old (scag max.kil `(list (list @c))`[buf old.kil]) == %= kil :: cumulative yanks pos num.kil old :_ t.old.kil ?- dir $l (welp buf i.old.kil) $r (welp i.old.kil buf) == == == :: ++ lcas :: lowercase |* a/(list @) ^+ a %+ turn a |=(a/@ ?.(&((gte a 'A') (lte a 'Z')) a (add 32 a))) :: ++ ucas :: uppercase |* a/(list @) ^+ a %+ turn a |=(a/@ ?.(&((gte a 'a') (lte a 'z')) a (sub a 32))) :: ++ ta-met :: meta key |= key/@ud ^+ +> =. ris ~ ?+ key ta-bel $dot ?. &(?=(^ old.hit) ?=(^ i.old.hit)) :: last "arg" from hist ta-bel =+ old=`(list @c)`i.old.hit =+ sop=(ta-jump(buf.say.inp old) %l %ace (lent old)) (ta-hom (cat:edit pos.inp (slag sop old))) :: $bac ?: =(0 pos.inp) :: kill left-word ta-bel =+ sop=(ta-off %l %edg pos.inp) (ta-kil %l [(sub pos.inp sop) sop]) :: $b ?: =(0 pos.inp) :: jump left-word ta-bel +>(pos.inp (ta-jump %l %edg pos.inp)) :: $c ?: =(pos.inp (lent buf.say.inp)) :: capitalize ta-bel =+ sop=(ta-jump %r %wrd pos.inp) %- ta-hom(pos.inp (ta-jump %r %edg sop)) (rep:edit [sop 1] (ucas (swag [sop 1] buf.say.inp))) :: $d ?: =(pos.inp (lent buf.say.inp)) :: kill right-word ta-bel (ta-kil %r [pos.inp (ta-off %r %edg pos.inp)]) :: $f ?: =(pos.inp (lent buf.say.inp)) :: jump right-word ta-bel +>(pos.inp (ta-jump %r %edg pos.inp)) :: $r %- ta-hom(lay.hit (~(put by lay.hit) pos.hit ~)) :- %set :: revert hist edit ?: =(pos.hit num.hit) ~ (snag (sub num.hit +(pos.hit)) old.hit) :: $t =+ a=(ta-jump %r %edg pos.inp) :: transpose words =+ b=(ta-jump %l %edg a) =+ c=(ta-jump %l %edg b) ?: =(b c) ta-bel =+ next=[b (sub a b)] =+ prev=[c (ta-off %r %edg c)] %- ta-hom(pos.inp a) :~ %mor (rep:edit next (swag prev buf.say.inp)) (rep:edit prev (swag next buf.say.inp)) == :: ?($u $l) :: upper/lower case ?: =(pos.inp (lent buf.say.inp)) ta-bel =+ case=?:(?=($u key) ucas lcas) =+ sop=(ta-jump %r %wrd pos.inp) =+ sel=[sop (ta-off %r %edg sop)] (ta-hom (rep:edit sel (case (swag sel buf.say.inp)))) :: $y ?. ?& (gth num.kil 0) :: rotate & yank ?=(^ p.blt) ?| ?=({$ctl $y} u.p.blt) ?=({$met $y} u.p.blt) == == ta-bel =+ las=(lent (snag (sub num.kil pos.kil) old.kil)) =+ pos=?:(=(1 pos.kil) num.kil (dec pos.kil)) %- ta-hom(pos.kil pos) %+ rep:edit [(sub pos.inp las) las] (snag (sub num.kil pos) old.kil) == :: ++ ta-mov :: move in history |= sop/@ud ^+ +> ?: =(sop pos.hit) +> %- %= ta-hom pos.hit sop lay.hit (~(put by lay.hit) pos.hit buf.say.inp) == :- %set %. (~(get by lay.hit) sop) (bond |.((snag (sub num.hit +(sop)) old.hit))) :: ++ ta-nex :: advance history ^+ . =. ris ~ =. lay.hit ~ ?: ?| ?=($~ buf.say.inp) &(?=(^ old.hit) =(buf.say.inp i.old.hit)) == .(pos.hit num.hit) %_ . num.hit +(num.hit) pos.hit +(num.hit) old.hit [buf.say.inp old.hit] == :: ++ ta-off :: buffer pos offset |= {dir/?($l $r) til/?($ace $edg $wrd) pos/@ud} ^- @ud %- ?- til $ace ace:offset $edg edg:offset $wrd wrd:offset == ?- dir $l (flop (scag pos buf.say.inp)) $r (slag pos buf.say.inp) == :: ++ ta-pro :: set prompt |= pom/sole-prompt +>(pom pom(cad :(welp (scow %p p.gyl) ":" (trip q.gyl) cad.pom))) :: ++ ta-ret :: hear return (ta-act %ret ~) :: ++ ta-ser :: reverse search |= ext/(list @c) ^+ +> ?: |(?=($~ ris) =(0 pos.u.ris)) ta-bel =+ sop=?~(ext (dec pos.u.ris) pos.u.ris) =+ tot=(weld str.u.ris ext) =+ dol=(slag (sub num.hit sop) old.hit) =/ sup |- ^- (unit @ud) ?~ dol ~ ?^ (find tot i.dol) `sop $(sop (dec sop), dol t.dol) ?~ sup ta-bel (ta-mov(str.u.ris tot, pos.u.ris u.sup) (dec u.sup)) :: ++ ta-txt :: hear text |= txt/(list @c) ^+ +> ?^ ris (ta-ser txt) (ta-hom (cat:edit pos.inp txt)) :: ++ ta-vew :: computed prompt |- ^- (pair @ud (list @c)) =; vew/(pair (list @c) tape) [(add pos.inp (lent q.vew)) (weld (tuba q.vew) p.vew)] ?: vis.pom :- buf.say.inp :: default prompt ?~ ris cad.pom :(welp "(reverse-i-search)'" (tufa str.u.ris) "': ") :- (reap (lent buf.say.inp) `@c`'*') :: hidden input %+ welp cad.pom ?~ buf.say.inp ~ :(welp "<" (scow %p (end 4 1 (sham buf.say.inp))) "> ") -- ++ edit :: produce sole-edits |% ++ cat :: mass insert |= {pos/@ud txt/(list @c)} ^- sole-edit :- %mor |- ^- (list sole-edit) ?~ txt ~ [[%ins pos i.txt] $(pos +(pos), txt t.txt)] :: ++ cut :: mass delete |= {pos/@ud num/@ud} ^- sole-edit :- %mor |- ^- (list sole-edit) ?: =(0 num) ~ [[%del pos] $(num (dec num))] :: ++ rep :: mass replace |= {{pos/@ud num/@ud} txt/(list @c)} ^- sole-edit :~ %mor (cut pos num) (cat pos txt) == -- ++ offset :: calculate offsets |% ++ alnm :: alpha-numeric |= a/@ ^- ? ?| &((gte a '0') (lte a '9')) &((gte a 'A') (lte a 'Z')) &((gte a 'a') (lte a 'z')) == :: ++ ace :: next whitespace |= a/(list @) =| {b/_| i/@ud} |- ^- @ud ?~ a i =/ c !=(32 i.a) =. b |(b c) ?: &(b !|(=(0 i) c)) i $(i +(i), a t.a) :: ++ edg :: next word boundary |= a/(list @) =| {b/_| i/@ud} |- ^- @ud ?~ a i =/ c (alnm i.a) =. b |(b c) ?: &(b !|(=(0 i) c)) i $(i +(i), a t.a) :: ++ wrd :: next or current word |= a/(list @) =| i/@ud |- ^- @ud ?: |(?=($~ a) (alnm i.a)) i $(i +(i), a t.a) -- --