EditSpanImpl.mesa; written by Bill Paxton, June 1981
edited by McGregor, February 14, 1983 1:52 pm
edited by Bill Paxton, December 23, 1982 2:47 pm
edited by Maxwell, January 5, 1983 12:46 pm
DIRECTORY
EditSpan,
EditSpanSupport,
EditNotify,
Rope,
UndoEvent,
TextEdit,
TreeSlice,
TiogaLooks,
TiogaNode,
TiogaNodeOps,
TiogaTreeOps;
EditSpanImpl: CEDAR PROGRAM
IMPORTS TextEdit, TiogaNodeOps, TiogaTreeOps, UndoEvent, EditSpan, EditSpanSupport, TreeSlice, EditNotify
EXPORTS EditSpan
SHARES TiogaNode =
BEGIN
OPEN EditSpan, EditSpanSupport, TreeSlice, EditNotify;
CannotDoEdit: PUBLIC ERROR = CODE;
afterMoved1, afterMoved2: PUBLIC TreeLoc;
Slice: TYPE = TreeSlice.Slice;
nullSpan: TreeSpan = TiogaTreeOps.nullSpan;
nullBranchSpan: BranchSpan = TiogaTreeOps.nullBranchSpan;
***** Editing operations on spans
Move: PUBLIC PROC [destRoot, sourceRoot: RefBranchNode, dest: TreeLoc, source: TreeSpan,
where: Place ← after, nesting: INTEGER ← 0, event: Event ← NIL]
RETURNS [result: TreeSpan] = {
dest cannot be within source
result is moved span
sBefore, sAfter, sTop, sBottom, dBefore, dAfter: Slice;
sNesting, dNesting: INTEGER;
sDepth: NAT;
beforeSource, afterSource: RefBranchNode; -- nodes adjacent to source after do splits
afterLoc: TreeLoc;
d: BranchLoc;
s: BranchSpan;
FreeSlices: PROC = {
FreeSlice[sBefore]; FreeSlice[sAfter]; FreeSlice[sTop]; FreeSlice[sBottom];
FreeSlice[dBefore]; FreeSlice[dAfter];
sBefore ← sAfter ← sTop ← sBottom ← dBefore ← dAfter ← NIL };
ForcedUnNest: PROC [after: Ref] = { OPEN TiogaTreeOps;
afterNode: RefBranchNode = TiogaNodeOps.NarrowToBranchNode[after];
span: TreeSpan = MakeTreeSpan[afterNode, LastBranchWithin[afterNode]];
IF afterNode=NIL THEN ERROR;
FreeSlices[];
[] ← UnNest[Root[afterNode], span, event] };
ForcedNest: PROC = { OPEN TiogaTreeOps;
bottom: RefBranchNode = TiogaNodeOps.NarrowToBranchNode[sBottom[sDepth]];
span: TreeSpan ← MakeTreeSpan[sTop[sDepth], LastBranchWithin[bottom]];
IF bottom=NIL THEN ERROR;
IF TreeSlice.SliceLength[sTop] = sDepth+1 THEN { -- can't do it by calling Nest
IF nesting >= 1 THEN ERROR;
nesting ← nesting+1; -- move to a deeper position
FreeSlices[];
RETURN };
FreeSlices[];
[] ← Nest[Root[span.start.node], span, event] };
IF CheckForNil[source] THEN RETURN [nullSpan];
IF TextLocationAndSpan[dest, source] THEN -- pure text move
RETURN [DestSpanText[destRoot, sourceRoot, dest, source, TextEdit.MoveText, event]];
IF where=child THEN { where ← after; nesting ← nesting+1 }
ELSE IF where=sibling THEN {
parentBr, childBr: RefBranchNode;
parentBr ← TiogaTreeOps.ContainingStatement[dest.node];
childBr ← TiogaTreeOps.LastBranchWithin[parentBr];
dest ← [childBr, NodeItself];
where ← after;
nesting ← nesting + TiogaTreeOps.Level[parentBr] - TiogaTreeOps.Level[childBr] };
split source and dest, if necessary, so can deal with entire nodes
[d, s] ← DoSplitsForMove[dest, source, where, event];
beforeSource ← TiogaTreeOps.Backward[s.start.node].back;
afterSource ← TiogaTreeOps.Forward[s.end.node].nx;
afterLoc ← [afterSource, NodeItself];
{ -- for exits
check for d already in correct position
IF d # TiogaTreeOps.nullBranchLoc AND
((where = after AND (d.node = beforeSource OR d.node = s.end.node))
OR (where = before AND (d.node = afterSource OR d.node = s.start.node)))
THEN { -- not going anywhere, but might be changing nesting
IF nesting > 0 THEN
FOR i:INTEGER IN [0..nesting) DO
[] ← Nest[sourceRoot, NodeSpan[s], event];
ENDLOOP
ELSE IF nesting < 0 THEN
FOR i:INTEGER IN [nesting..0) DO
[] ← UnNest[sourceRoot, NodeSpan[s], event];
ENDLOOP }
ELSEDO -- repeat this loop only if have forced nest or unnest or s and d in same tree
[sBefore, sAfter, sTop, sBottom, sNesting, sDepth] ←
DescribeBand[s.start.node, s.end.node ! BadBand => { s ← nullBranchSpan; GOTO ErrorReturn } ];
IF d = TiogaTreeOps.nullBranchLoc THEN { -- moving to limbo
d ← CreateDest[TreeSlice.SliceLength[sTop]-sDepth];
destRoot ← TiogaTreeOps.Root[d.node];
destRoot.deleted ← TRUE; -- so will free this when it falls off the edit history list
where ← after; nesting ← 1 };
[dBefore, dAfter, dNesting] ← DestSlices[d.node, where];
IF TreeSlice.CompareSliceOrder[dBefore, sBefore]=after AND
TreeSlice.CompareSliceOrder[dBefore, sBottom]=before THEN GOTO ErrorReturn; -- d inside s
IF ~TiogaNodeOps.IsBranch[dBefore[0]] AND BadDest[dBefore[0], sBefore, sBottom]
THEN GOTO ErrorReturn;
Dest is inside a list or box node. Check to see if statement is inside s
IF dBefore[0] = sBefore[0] THEN { -- s and d in same tree; too hard, so move s out first.
span: TreeSpan = NodeSpan[s];
FreeSlices[];
[] ← MoveToLimbo[sourceRoot, span, event];
LOOP };
dNesting ← dNesting + nesting;
SELECT NeedNestingChange[dBefore, dAfter, sTop, sBottom, dNesting, sDepth] FROM
needUnNest => { ForcedUnNest[TreeSlice.LastOfSlice[dAfter]]; LOOP };
needNest => { ForcedNest[]; LOOP };
ENDCASE;
IF TreeSlice.SliceLength[sAfter] > TreeSlice.SliceLength[sBefore]+1 THEN {
ForcedUnNest[TreeSlice.LastOfSlice[sAfter]]; LOOP }
ELSE { -- do it
notify: REF MovingNodes Change;
notify ← NEW[MovingNodes Change ← [MovingNodes[
destRoot, sourceRoot, d.node, s.start.node, s.end.node,
TreeSlice.LastOfSlice[sBefore], sNesting, (where # before)]]];
Notify[notify, before];
DeletePrefix[sTop, sDepth]; DeletePrefix[sBottom, sDepth];
ReplaceBand[dBefore, dAfter, sTop, sBottom, dNesting, event];
Splice[sBefore, sAfter];
FreeSlices[];
Notify[notify, after];
UndoEvent.Note[event, UndoMoveNodes, notify] };
EXIT;
ENDLOOP;
IF d.where # NodeItself THEN { -- undo prior splits
start, end: BOOLEANFALSE;
IF s.start.where # NodeItself THEN { -- merge start of s with front of dest
start ← TRUE;
[s, ] ← ReMerge[s, nullBranchSpan, s.start.node, event] };
IF s.end.where # NodeItself THEN { -- merge end of s with tail of dest
end ← TRUE;
[s, ] ← ReMerge[s, nullBranchSpan, s.end.node, event, TRUE] };
IF start AND end THEN { -- merge before s with after s
afterLoc ← JoinStatements[TiogaTreeOps.Root[afterSource], afterSource, event] }}
ELSE IF s.start.where # NodeItself AND s.end.where # NodeItself THEN {
afterLoc ← JoinStatements[TiogaTreeOps.Root[afterSource], afterSource, event] };
afterMoved2 ← afterMoved1; -- save previous hint
afterMoved1 ← IF afterSource=NIL THEN [beforeSource, 0] ELSE afterLoc; -- hint for repaint
source.start.node ← s.start.node; source.end.node ← s.end.node;
source.start.where ← source.end.where ← NodeItself;
RETURN [source];
EXITS ErrorReturn => {
FreeSlices[];
UndoSplitsForMove[d, s, where, event];
ERROR CannotDoEdit }}};
UndoMoveNodes: PROC [undoRef: REF, currentEvent: Event] = TRUSTED {
saved: REF Change ← NARROW[undoRef];
WITH x:saved SELECT FROM
MovingNodes => {
[] ← Move[x.sourceRoot, x.destRoot,
TiogaTreeOps.MakeTreeLoc[x.pred], TiogaTreeOps.MakeTreeSpan[x.first, x.last],
after, x.nesting, currentEvent] };
ENDCASE => ERROR };
BadDest: PROC [top: Ref, sBefore, sBottom: Slice] RETURNS [error: BOOL] = {
DO -- check for dest inside source
s: RefBranchNode ← TiogaTreeOps.ContainingStatement[top];
before, after: Slice;
[before, after] ← MakeSlices[s];
top ← before[0];
error TreeSlice.CompareSliceOrder[before, sBefore]=after AND
TreeSlice.CompareSliceOrder[before, sBottom]=before;
FreeSlice[before]; FreeSlice[after];
IF error THEN RETURN;
IF TiogaNodeOps.IsBranch[top] THEN RETURN;
ENDLOOP };
Transpose: PUBLIC PROC [alphaRoot, betaRoot: RefBranchNode, alpha, beta: TreeSpan, event: Event ← NIL]
RETURNS [newAlpha, newBeta: TreeSpan] = {
newAlpha is new location of alpha span; ditto for newBeta
aBefore, aAfter, aTop, aBottom, bBefore, bAfter, bTop, bBottom: Slice;
aNesting, bNesting: INTEGER;
aDepth, bDepth: NAT;
beforeAlpha, afterAlpha, beforeBeta, afterBeta: RefBranchNode; -- nodes adjacent after do splits
afterAlphaLoc, afterBetaLoc: TreeLoc;
a, b: BranchSpan;
FreeSlices: PROC = {
FreeSlice[aBefore]; FreeSlice[aAfter]; FreeSlice[aTop]; FreeSlice[aBottom];
FreeSlice[bBefore]; FreeSlice[bAfter]; FreeSlice[bTop]; FreeSlice[bBottom];
aBefore ← aAfter ← aTop ← aBottom ← bBefore ← bAfter ← bTop ← bBottom ← NIL };
{ -- for exit
IF CheckForNil[alpha] OR CheckForNil[beta] THEN
RETURN [nullSpan, nullSpan];
IF TextSpans[alpha, beta] THEN { -- pure text transpose
alphaNode: RefTextNode ← TiogaNodeOps.NarrowToTextNode[alpha.start.node];
betaNode: RefTextNode ← TiogaNodeOps.NarrowToTextNode[beta.start.node];
alphaStart: Offset ← alpha.start.where;
alphaLen: Offset ← alpha.end.where - alphaStart + 1;
betaStart: Offset ← beta.start.where;
betaLen: Offset ← beta.end.where - betaStart + 1;
alphaResultStart, alphaResultLen, betaResultStart, betaResultLen: Offset;
[alphaResultStart, alphaResultLen, betaResultStart, betaResultLen] ← TextEdit.TransposeText[
alphaRoot, betaRoot, alphaNode, alphaStart, alphaLen, betaNode, betaStart, betaLen, event];
newAlpha ← [[betaNode, alphaResultStart], [betaNode, alphaResultStart+alphaResultLen-1]];
newBeta ← [[alphaNode, betaResultStart], [alphaNode, betaResultStart+betaResultLen-1]];
RETURN };
[a, b] ← DoSplits[alpha, beta, event]; -- so can deal with entire nodes
beforeAlpha ← TiogaTreeOps.Backward[a.start.node].back;
afterAlpha ← TiogaTreeOps.Forward[a.end.node].nx;
afterAlphaLoc ← [afterAlpha, 0];
beforeBeta ← TiogaTreeOps.Backward[b.start.node].back;
afterBeta ← TiogaTreeOps.Forward[b.end.node].nx;
afterBetaLoc ← [afterBeta, 0];
now check for a b adjacent or overlapping as special cases
IF afterAlpha = b.start.node THEN { -- a just before b
move b nodes to before a nodes
[] ← Move[alphaRoot, betaRoot, NodeLoc[a.start], NodeSpan[b], before, 0, event] }
ELSE IF afterBeta = a.start.node THEN { -- b just before a
move a nodes to before b nodes
[] ← Move[betaRoot, alphaRoot, NodeLoc[b.start], NodeSpan[a], before, 0, event] }
ELSE  { -- get slices describing the bands of nodes to be transposed
overlap: BOOLEAN;
head, tail: BranchSpan; -- sections of a or b before and after the overlap
startOrder, endOrder: Order;
[aBefore, aAfter, aTop, aBottom, aNesting, aDepth] ←
DescribeBand[a.start.node, a.end.node ! BadBand =>
{ a ← b ← nullBranchSpan; GOTO ErrorReturn }];
[bBefore, bAfter, bTop, bBottom, bNesting, bDepth] ←
DescribeBand[b.start.node, b.end.node ! BadBand =>
{ a ← b ← nullBranchSpan; GOTO ErrorReturn }];
IF ~TiogaNodeOps.IsBranch[aBefore[0]] AND BadDest[aBefore[0], bBefore, bBottom]
THEN GOTO ErrorReturn; -- a inside statement inside b
IF ~TiogaNodeOps.IsBranch[bBefore[0]] AND BadDest[bBefore[0], aBefore, aBottom]
THEN GOTO ErrorReturn; -- b inside statement inside a
check for overlap
[overlap, head, tail, startOrder, endOrder] ←
SliceOrder[a, b, aBefore, aBottom, bBefore, bBottom];
IF overlap THEN { -- bands overlap
FreeSlices[];
IF head = nullBranchSpan AND tail = nullBranchSpan THEN NULL
ELSE IF head = nullBranchSpan THEN { --move tail to before alphastart
[] ← Move[alphaRoot, betaRoot, NodeLoc[a.start], NodeSpan[tail], before, 0, event];
IF endOrder=before THEN { -- a end before b end
b.start ← tail.start; b.end ← a.end }
ELSE { -- b end before a end
a.start ← tail.start; a.end ← b.end }}
ELSE IF tail = nullBranchSpan THEN { --move head to after alphaend
[] ← Move[alphaRoot, betaRoot, NodeLoc[a.end], NodeSpan[head], after, 0, event];
IF startOrder=before THEN { -- a start before b start
a.start ← b.start; a.end ← head.end }
ELSE { -- b start before a start
b.start ← a.start; b.end ← head.end }}
ELSE IF startOrder # endOrder THEN NULL -- one contained in the other
ELSE { --transpose head and tail
[] ← Transpose[alphaRoot, betaRoot, NodeSpan[head], NodeSpan[tail], event];
IF startOrder=before THEN { -- a start before b start
a.start ← b.start; a.end ← head.end;
b.start ← tail.start; b.end ← a.end }
ELSE { -- b start before a start
b.start ← a.start; b.end ← head.end;
a.start ← tail.start; a.end ← b.end }}}
ELSE { -- do transpose as two moves
aSpan, bSpan: TreeSpan;
after1, after2: TreeLoc;
bLoc: TreeLoc ← TiogaTreeOps.MakeTreeLoc[TreeSlice.LastOfSlice[bBefore]];
aLoc: TreeLoc ← TiogaTreeOps.MakeTreeLoc[TreeSlice.LastOfSlice[aBefore]];
FreeSlices[];
aSpan ← TiogaTreeOps.MakeTreeSpan[a.start.node, a.end.node];
bSpan ← TiogaTreeOps.MakeTreeSpan[b.start.node, b.end.node];
[] ← MoveToLimbo[alphaRoot, aSpan, event]; after1 ← afterMoved1; -- repaint hints
[] ← MoveToLimbo[betaRoot, bSpan, event]; after2 ← afterMoved1;
[] ← Move[betaRoot, TiogaTreeOps.Root[aSpan.start.node],
bLoc, aSpan, after, bNesting, event];
[] ← Move[alphaRoot, TiogaTreeOps.Root[bSpan.start.node],
aLoc, bSpan, after, aNesting, event];
afterMoved1 ← after1; afterMoved2 ← after2 }};
IF a.start.where # NodeItself AND b.start.where # NodeItself THEN { -- remerge starts
[a, b] ← ReMerge[a, b, a.start.node, event];
[a, b] ← ReMerge[a, b, b.start.node, event] };
IF a.end.where # NodeItself AND b.end.where # NodeItself THEN { -- remerge ends
[a, b] ← ReMerge[a, b, a.end.node, event, TRUE];
[a, b] ← ReMerge[a, b, b.end.node, event, TRUE];
afterAlphaLoc.node ← b.end.node; afterBetaLoc.node ← a.end.node };
afterMoved1 ← IF afterAlphaLoc.node=NIL THEN [beforeAlpha, 0] ELSE afterAlphaLoc; -- hint for repaint
afterMoved2 ← IF afterBetaLoc.node=NIL THEN [beforeBeta, 0] ELSE afterBetaLoc;
alpha.start ← [a.start.node, NodeItself];
alpha.end ← [a.end.node, NodeItself];
beta.start ← [b.start.node, NodeItself];
beta.end ← [b.end.node, NodeItself];
RETURN [alpha, beta]; 
EXITS ErrorReturn => {
FreeSlices[]; UndoSplits[a, b, event];
ERROR CannotDoEdit }}};
MoveOnto: PUBLIC PROC [destRoot, sourceRoot: RefBranchNode, dest, source: TreeSpan,
saveForPaste: BOOLEANTRUE, event: Event ← NIL]
RETURNS [result: TreeSpan] = {
like Replace, but moves source instead of copying it
result is moved span
overlap: BOOLEAN;
newDest: TreeSpan;
head, tail: BranchSpan; -- sections of alpha or beta before and after the overlap
startOrder, endOrder: Order;
aBefore, aAfter, aTop, aBottom, bBefore, bAfter, bTop, bBottom: Slice;
aNesting, bNesting: INTEGER;
aDepth, bDepth: NAT;
d, s: BranchSpan;
FreeSlices: PROC = {
FreeSlice[aBefore]; FreeSlice[aAfter]; FreeSlice[aTop]; FreeSlice[aBottom];
FreeSlice[bBefore]; FreeSlice[bAfter]; FreeSlice[bTop]; FreeSlice[bBottom];
aBefore ← aAfter ← aTop ← aBottom ← bBefore ← bAfter ← bTop ← bBottom ← NIL };
{ -- for exit
IF CheckForNil[source] OR CheckForNil[dest] THEN RETURN [nullSpan];
IF TextSpans[dest, source] THEN { -- pure text move
RETURN [TwoSpanText[destRoot, sourceRoot, dest, source, saveForPaste, TextEdit.MoveTextOnto, event]] };
[d, s] ← DoSplits[dest, source, event];
get slices for d and s
[aBefore, aAfter, aTop, aBottom, aNesting, aDepth] ←
DescribeBand[d.start.node, d.end.node ! BadBand => { d ← s ← nullBranchSpan; GOTO ErrorReturn }];
[bBefore, bAfter, bTop, bBottom, bNesting, bDepth] ←
DescribeBand[s.start.node, s.end.node ! BadBand => { d ← s ← nullBranchSpan; GOTO ErrorReturn }];
IF ~TiogaNodeOps.IsBranch[aBefore[0]] AND BadDest[aBefore[0], bBefore, bBottom]
THEN GOTO ErrorReturn; -- d inside statement inside s
[overlap, head, tail, startOrder, endOrder] ← SliceOrder[d, s, aBefore, aBottom, bBefore, bBottom];
FreeSlices[];
check for overlap
IF overlap THEN { -- bands overlap. modify d so doesn't overlap
IF head = nullBranchSpan AND tail = nullBranchSpan THEN GOTO ErrorReturn;
IF head = nullBranchSpan THEN { -- s start = d start
IF endOrder=before THEN GOTO ErrorReturn -- d end before s end
ELSE d.start ← tail.start }
ELSE IF tail = nullBranchSpan THEN { --s end = d end
IF startOrder=before THEN d.end ← head.end -- d start before s start
ELSE GOTO ErrorReturn } -- s start before d start
ELSE { -- have both head and tail
IF startOrder=before AND endOrder=after THEN {
[] ← Delete[destRoot, NodeSpan[tail], event]; d.end ← head.end }
ELSE IF startOrder=before THEN d.end ← head.end
ELSE IF endOrder=after THEN d.start ← tail.start
ELSE GOTO ErrorReturn }};
source ← MoveToLimbo[sourceRoot, NodeSpan[s], event];
sourceRoot ← TiogaTreeOps.Root[source.start.node];
[result, newDest] ← Transpose[sourceRoot, destRoot, source, NodeSpan[d], event];
IF saveForPaste THEN SaveSpanForPaste[newDest, event];
RETURN;
EXITS ErrorReturn => { UndoSplits[d, s, event]; ERROR CannotDoEdit }}};
Replace: PUBLIC PROC [
destRoot, sourceRoot: RefBranchNode, dest, source: TreeSpan,
saveForPaste: BOOLTRUE, event: Event ← NIL]
RETURNS [result: TreeSpan] = {
replace dest span by copy of source span
result is the new copy of source
newDest: TreeSpan;
IF TextSpans[dest, source] THEN { -- pure text replace
RETURN [TwoSpanText[destRoot, sourceRoot, dest, source, saveForPaste, TextEdit.ReplaceText, event]] };
source ← CopySpan[source];
sourceRoot ← TiogaTreeOps.Root[source.start.node];
sourceRoot.deleted ← TRUE;
[result, newDest] ← Transpose[sourceRoot, destRoot, source, dest, event];
IF saveForPaste THEN SaveSpanForPaste[newDest, event] };
TextSpans: PROC [dest, source: TreeSpan] RETURNS [BOOLEAN] = {
RETURN [dest.start.node = dest.end.node AND
dest.start.where # NodeItself AND dest.end.where # NodeItself AND
TiogaNodeOps.IsText[dest.start.node] AND
source.start.node = source.end.node AND
source.start.where # NodeItself AND source.end.where # NodeItself AND
TiogaNodeOps.IsText[source.start.node]] };
TwoSpanText: PROC [destRoot, sourceRoot: RefBranchNode, dest, source: TreeSpan,
saveForPaste: BOOLEAN, textProc: TextEdit.TwoSpanProc, event: Event]
RETURNS [result: TreeSpan] = {
destNode: RefTextNode ← TiogaNodeOps.NarrowToTextNode[dest.start.node];
sourceNode: RefTextNode ← TiogaNodeOps.NarrowToTextNode[source.start.node];
destStart: Offset ← dest.start.where;
destLen: Offset ← dest.end.where - destStart + 1;
sourceStart: Offset ← source.start.where;
sourceLen: Offset ← source.end.where - sourceStart + 1;
resultStart, resultLen: Offset;
IF saveForPaste THEN SaveTextForPaste[destNode, destStart, destLen, event];
[resultStart, resultLen] ← textProc[destRoot, sourceRoot,
destNode, destStart, destLen, sourceNode, sourceStart, sourceLen, event];
RETURN [[[destNode, resultStart], [destNode, resultStart+resultLen-1]]] };
SaveForPaste: PUBLIC PROC [span: TreeSpan, event: Event ← NIL] = {
node: RefTextNode;
IF CheckForNil[span] THEN RETURN;
IF span.start.node = span.end.node AND
span.start.where # NodeItself AND span.end.where # NodeItself AND
(node ← TiogaNodeOps.NarrowToTextNode[span.start.node]) # NIL THEN { -- pure text
start: Offset ← span.start.where;
len: Offset ← span.end.where-span.start.where+1;
SaveTextForPaste[node, start, len, event] }
ELSE SaveSpanForPaste[CopySpan[span], event] };
Delete: PUBLIC PROC [
root: RefBranchNode, del: TreeSpan, event: Event ← NIL, saveForPaste: BOOLTRUE] = {
node: RefTextNode;
IF CheckForNil[del] THEN RETURN;
IF del.start.node = del.end.node AND
del.start.where # NodeItself AND del.end.where # NodeItself AND
(node ← TiogaNodeOps.NarrowToTextNode[del.start.node]) # NIL THEN { -- pure text
start: Offset ← del.start.where;
len: Offset ← del.end.where-del.start.where+1;
IF saveForPaste THEN SaveTextForPaste[node, start, len, event];
TextEdit.DeleteText[root, node, start, len, event] }
ELSE {
d: TreeSpan ← MoveToLimbo[root, del, event];
IF saveForPaste THEN SaveSpanForPaste[d, event] }};
paste: REF Change;
--really REF ChangingTextForPaste/ChangingSpanForPaste Change
SaveTextForPaste: PROC [node: RefTextNode, start, len: Offset, event: Event] = {
SaveOldPaste[event];
paste ← NEW[ChangingTextForPaste Change ←
[ChangingTextForPaste[node.rope, node.runs, start, len]]] };
SaveSpanForPaste: PROC [span: TreeSpan, event: Event] = {
SaveOldPaste[event];
paste ← NEW[ChangingSpanForPaste Change ← [ChangingSpanForPaste[span]]] };
RecordGroupForPaste: PUBLIC PROC [start, end: Ref, event: Event] = {
span: TreeSpan;
span.start ← [start, NodeItself];
span.end ← [TiogaTreeOps.LastWithin[end], NodeItself];
SaveSpanForPaste[span, event] };
SaveOldPaste: PROC [event: Event] = {
UndoEvent.Note[event, RestorePaste, paste]; paste ← NIL };
RestorePaste: PROC [undoRef: REF Change, currentEvent: UndoEvent.Ref] = {
SaveOldPaste[currentEvent]; paste ← undoRef };
SavedForPaste: PUBLIC PROC RETURNS [span: TreeSpan] = TRUSTED {
savedPaste: REF Change ← paste;
IF savedPaste=NIL THEN RETURN [nullSpan];
WITH x: savedPaste SELECT FROM
ChangingTextForPaste => { -- convert saved text to a span
node: RefTextNode;
root: RefBranchNode;
span: TreeSpan;
IF x.len <= 0 THEN RETURN [nullSpan];
node ← TiogaNodeOps.NewTextNode[];
root ← TiogaNodeOps.DocFromNode[node];
[] ← TextEdit.ReplaceByText[root: root, dest: node,
sourceRope: x.rope, sourceRuns: x.runs,
sourceStart: x.start, sourceLen: x.len];
span ← [[node, 0], [node, x.len-1]];
paste ← NEW[ChangingSpanForPaste Change ← [ChangingSpanForPaste[span]]];
RETURN [span] };
ChangingSpanForPaste => RETURN [x.span];
ENDCASE => ERROR;
};
MoveToLimbo: PROC [root: RefBranchNode, span: TreeSpan, event: Event] RETURNS [result: TreeSpan] =
INLINE { RETURN [Move[NIL, root, TiogaTreeOps.nullLoc, span, after, 1, event]] };
Copy: PUBLIC PROC [
destRoot, sourceRoot: RefBranchNode, dest: TreeLoc, source: TreeSpan,
where: Place ← after, nesting: INTEGER ← 0, event: Event ← NIL]
RETURNS [result: TreeSpan] = {
result is the new copy of source
IF CheckForNil[source] OR dest.node=NIL THEN RETURN [nullSpan];
IF TextLocationAndSpan[dest, source] THEN -- pure text copy
RETURN [DestSpanText[destRoot, sourceRoot, dest, source, TextEdit.CopyText, event]];
source ← CopySpan[source];
sourceRoot ← TiogaTreeOps.Root[source.start.node];
sourceRoot.deleted ← TRUE;
result ← Move[destRoot, sourceRoot, dest, source, where, nesting, event] };
TextLocationAndSpan: PROC [dest: TreeLoc, source: TreeSpan] RETURNS [BOOLEAN] = {
RETURN [dest.where # NodeItself AND
source.start.node = source.end.node AND
source.start.where # NodeItself AND source.end.where # NodeItself
AND TiogaNodeOps.NarrowToTextNode[source.start.node] # NIL] };
DestSpanText: PROC [
destRoot, sourceRoot: RefBranchNode, dest: TreeLoc, source: TreeSpan,
textProc: TextEdit.DestSpanProc, event: Event]
RETURNS [result: TreeSpan] = {
destNode: RefTextNode ← TiogaNodeOps.NarrowToTextNode[dest.node];
sourceNode: RefTextNode ← TiogaNodeOps.NarrowToTextNode[source.start.node];
destLoc: Offset ← dest.where;
sourceStart: Offset ← source.start.where;
sourceLen: Offset ← source.end.where - sourceStart + 1;
resultStart, resultLen: Offset;
[resultStart, resultLen] ← textProc[destRoot, sourceRoot,
destNode, destLoc, sourceNode, sourceStart, sourceLen, event];
RETURN [[[destNode, resultStart], [destNode, resultStart+resultLen-1]]] };
***** Nesting
ChangeNesting: PUBLIC PROC [
root: RefBranchNode, span: TreeSpan, change: INTEGER, event: Event ← NIL]
RETURNS [new: TreeSpan] = {
moves span to a deeper nesting level in tree
before, after, top, bottom: Slice;
nesting: INTEGER;
depth: NAT;
s: BranchSpan;
FreeSlices: PROC = {
FreeSlice[before]; FreeSlice[after]; FreeSlice[top]; FreeSlice[bottom];
before ← after ← top ← bottom ← NIL };
{ -- for exit
IF CheckForNil[span] THEN RETURN [nullSpan];
[s, ] ← DoSplits[span, nullSpan, event]; -- so can deal with entire nodes
DO -- only repeat this loop if have forced nest/unnest
[before, after, top, bottom, nesting, depth] ←
DescribeBand[s.start.node, s.end.node ! BadBand => { s ← nullBranchSpan; GOTO ErrorReturn } ];
IF nesting+change > 1 THEN GOTO ErrorReturn; -- cannot do it
SELECT NeedNestingChange[before, after, top, bottom, nesting+change, depth] FROM
needUnNest => {
afterNode: RefBranchNode = TiogaNodeOps.NarrowToBranchNode[TreeSlice.LastOfSlice[after]];
last: RefBranchNode = TiogaTreeOps.LastBranchWithin[afterNode];
s: TreeSpan = TiogaTreeOps.MakeTreeSpan[afterNode, last];
IF afterNode=NIL THEN ERROR;
FreeSlices[];
[] ← UnNest[root, s, event];
LOOP };
needNest => {
last: RefBranchNode = TiogaTreeOps.LastBranchWithin[
TiogaNodeOps.NarrowToBranchNode[bottom[depth]]];
s: TreeSpan = TiogaTreeOps.MakeTreeSpan[top[depth], last];
IF last=NIL THEN ERROR;
IF TreeSlice.SliceLength[top] = depth+1 THEN GOTO ErrorReturn;
FreeSlices[];
[] ← Nest[root, s, event];
LOOP };
ENDCASE => { -- do it
notify: REF NodeNesting Change;
notify ← NEW[NodeNesting Change ← [NodeNesting[root, s.start.node, s.end.node, change]]];
Notify[notify, before];
DeletePrefix[top, depth];
DeletePrefix[bottom, depth];
ReplaceBand[before, after, top, bottom, nesting+change, event];
FreeSlices[];
Notify[notify, after];
UndoEvent.Note[event, UndoChangeNesting, notify] };
EXIT;
ENDLOOP;
span.start ← [s.start.node, NodeItself];
span.end ← [s.end.node, NodeItself];
RETURN [span];
EXITS ErrorReturn => {
FreeSlices[]; UndoSplits[s, nullBranchSpan, event]; ERROR CannotDoEdit }}};
UndoChangeNesting: PROC [undoRef: REF, currentEvent: Event] = TRUSTED {
saved: REF Change ← NARROW[undoRef];
WITH x:saved SELECT FROM
NodeNesting => {
[] ← ChangeNesting[x.root, TiogaTreeOps.MakeTreeSpan[x.first, x.last], -x.change, currentEvent] };
ENDCASE => ERROR };
***** Miscellaneous
CheckForNil: PROC [span: TreeSpan] RETURNS [BOOLEAN] = INLINE {
RETURN [span.start.node = NIL OR span.end.node = NIL] };
NodeSpan: PROC [span: BranchSpan] RETURNS [TreeSpan] = INLINE {
RETURN [[[span.start.node, NodeItself], [span.end.node, NodeItself]]] };
NodeLoc: PROC [loc: BranchLoc] RETURNS [TreeLoc] = INLINE { RETURN [[loc.node, NodeItself]] };
END....