DIRECTORY Menus USING [ChangeNumberOfLines], Process USING [Detach], Rope USING [Concat, Equal, Map, ROPE, Size], TEditDocumentsCache USING [], TEditDocumentPrivate USING [], TEditInput USING [FreeTree], TEditProfile USING [showUnsavedDocumentsList, unsavedDocumentCacheSize], TextNode USING [Ref], ViewerClasses USING [Viewer], ViewerOps USING [CreateViewer, DestroyViewer, EstablishViewerPosition, PaintViewer]; TEditDocumentsCacheImpl: CEDAR MONITOR IMPORTS Menus, Process, Rope, TEditInput, TEditProfile, --TiogaAccess, --ViewerOps EXPORTS TEditDocumentsCache, TEditDocumentPrivate = BEGIN ROPE: TYPE ~ Rope.ROPE; UnsavedDocumentInfo: TYPE = REF UnsavedDocumentInfoRec; UnsavedDocumentInfoRec: TYPE = RECORD [fileName: ROPE, root: TextNode.Ref]; list: LIST OF UnsavedDocumentInfo; viewer: ViewerClasses.Viewer; GetList: PUBLIC ENTRY PROC RETURNS [ROPE] = { RETURN [GetListInternal[]] }; GetListInternal: INTERNAL PROC RETURNS [rope: ROPE] = { FOR l: LIST OF UnsavedDocumentInfo _ list, l.rest UNTIL l=NIL DO rope _ Rope.Concat[rope, Rope.Concat["\n", l.first.fileName]]; ENDLOOP; }; RecordUnsavedDocument: PUBLIC ENTRY PROC [fileName: ROPE, root: TextNode.Ref] = { ENABLE UNWIND => NULL; Size: PROC RETURNS [size: INTEGER] = { size _ 0; IF list=NIL THEN RETURN; FOR l: LIST OF UnsavedDocumentInfo _ list, l.rest UNTIL l=NIL DO size _ size+1; ENDLOOP; }; info: UnsavedDocumentInfo; oldRoot: TextNode.Ref; IF (oldRoot _ SearchList[fileName]) # NIL THEN -- discard previous entry, if any TEditInput.FreeTree[oldRoot]; info _ NEW[UnsavedDocumentInfoRec _ [fileName, root]]; list _ CONS[info, list]; WHILE Size[] > MAX[TEditProfile.unsavedDocumentCacheSize, 0] DO DiscardOldest[]; ENDLOOP; TRUSTED { Process.Detach[FORK Update[]] }; }; FindUnsavedDocument: PUBLIC ENTRY PROC [fileName: ROPE] RETURNS [root: TextNode.Ref] = { ENABLE UNWIND => NULL; IF (root _ SearchList[fileName]) # NIL THEN TRUSTED {Process.Detach[FORK Update[]] } }; DiscardOldest: PROC = { IF list=NIL THEN RETURN; FOR l: LIST OF UnsavedDocumentInfo _ list, l.rest DO IF l.rest # NIL THEN LOOP; -- spin until get to the oldest entry TEditInput.FreeTree[SearchList[l.first.fileName]]; RETURN; ENDLOOP; }; SearchList: PROC [fileName: ROPE] RETURNS [root: TextNode.Ref] = { -- if find, then remove prev: LIST OF UnsavedDocumentInfo; IF list=NIL THEN RETURN [NIL]; IF Rope.Equal[list.first.fileName, fileName, FALSE] THEN { root _ list.first.root; list _ list.rest; RETURN; }; prev _ list; FOR l: LIST OF UnsavedDocumentInfo _ list.rest, l.rest UNTIL l=NIL DO IF Rope.Equal[l.first.fileName, fileName, FALSE] THEN { root _ l.first.root; prev.rest _ l.rest; RETURN; }; prev _ l; ENDLOOP; RETURN [NIL]; }; Update: PUBLIC ENTRY PROC = { ENABLE UNWIND => NULL; rope: ROPE _ GetListInternal[]; -- this is monitored, so get consistent list viewerName: ROPE = "Unsaved Documents List"; ViewerOk: PROC RETURNS [BOOL] = { RETURN [viewer # NIL AND NOT viewer.destroyed AND viewer.file=NIL AND Rope.Equal[viewer.name, viewerName]] }; format: ATOM _ NIL; comment: BOOL _ FALSE; deltaLevel: INT _ 1; IF NOT TEditProfile.showUnsavedDocumentsList THEN RETURN; IF Rope.Size[rope] = 0 THEN { IF ViewerOk[] THEN ViewerOps.DestroyViewer[viewer]; RETURN }; IF NOT ViewerOk[] THEN { viewer _ ViewerOps.CreateViewer[flavor: $Text, info: [name: viewerName, iconic: TRUE], paint: FALSE]; viewer.class.set[self: viewer, data: NIL, op: $ReadOnly]; Menus.ChangeNumberOfLines[viewer.menu, 0]; -- don't let user mess with it ViewerOps.EstablishViewerPosition[viewer, viewer.wx, viewer.wy, viewer.ww, viewer.wh]; ViewerOps.PaintViewer[viewer, all] }; format _ $note; format _ $lead; deltaLevel _ 0; viewer.newVersion _ TRUE; -- set edited bit ViewerOps.PaintViewer[viewer, caption] }; -- repaint the caption END. TEditDocumentsCacheImpl.mesa Copyright Σ 1985, 1988 by Xerox Corporation. All rights reserved. Edited by Paxton on October 20, 1982 8:39 am Last Edited by: Maxwell, January 6, 1983 11:10 am Last Edited by: Plass, April 10, 1985 10:17:55 am PST Michael Plass, October 19, 1987 10:37:13 am PDT Doug Wyatt, February 19, 1988 11:12:04 am PST TiogaAccess USING [Create, Looks, Put, Writer, WriteViewer], writer: TiogaAccess.Writer ~ TiogaAccess.Create[]; looks: TiogaAccess.Looks _ ALL[FALSE]; Put: PROC [rope: ROPE] ~ { action: PROC [c: CHAR] RETURNS [quit: BOOL _ FALSE] ~ { TiogaAccess.Put[writer, [ charSet: 0, char: c, looks: looks, format: format, comment: comment, endOfNode: (c = '\n), deltaLevel: deltaLevel, propList: NIL ]]; }; [] _ Rope.Map[base: rope, action: action]; }; Put["The following files were edited but not saved. They may still be restored with edits intact simply by loading them. If you really want to get rid of the edits, load the file and hit Reset.\n"]; Put[rope]; Put["\n"]; TiogaAccess.WriteViewer[writer, viewer]; Κ΄˜šœ™IcodešœB™BJšœ,™,Jšœ1™1Jšœ5™5K™/Jšœ-™-J™—šΟk ˜ Jšœœ˜"Jšœœ ˜Jšœœœ˜,Jšœœ˜Jšœœ˜Jšœ œ ˜Kšœ œ6˜HJšœ œ˜Kšœ œ+™Jšœ˜—J˜J˜—š Ÿœœœœ œ˜QJšœœœ˜šŸœœœœ˜&J˜ Jšœœœœ˜š œœœ$œœ˜@J˜Jšœ˜—J˜—J˜J˜šœ$œœΟc!˜PJ˜—Jšœœ,˜6Jšœœ ˜šœ œ+˜?Jšœ˜Jšœ˜—Jšœœ ˜*J˜J˜—š Ÿœœœœ œœ˜XJšœœœ˜Jš œ!œœœœ ˜TJ˜J˜—šŸ œœ˜Jšœœœœ˜šœœœ$˜4Jš œ œœœ %˜@J˜2Jšœ˜Jšœ˜—J˜J˜—š Ÿ œœ œœ ˜ZJšœœœ˜"Jš œœœœœ˜šœ+œœ˜:J˜J˜Jšœ˜J˜—J˜ š œœœ)œœ˜Ešœ(œœ˜7J˜J˜Jšœ˜J˜—J˜ Jšœ˜—Jšœœ˜ J˜—J˜šŸœœœœ˜Kšœœœ˜Kšœœ ,˜LKšœ œ˜,šŸœœœœ˜!š œ œœœœ œ˜EK˜%—K˜—Kšœ2™2Kšœœœ™&Kšœœœ˜Kšœ œœ˜Kšœ œ˜šŸœœœ™š œœœœœœ™7šœ™Jšœ ™ Jšœ™Jšœ ™ Jšœ™Jšœ™Jšœ™Jšœ™Jšœ ™ Jšœ™—Jšœ™—J•StartOfExpansionT[base: ROPE, start: INT _ 0, len: INT _ 2147483647, action: Rope.ActionType]šœ*™*Jšœ™—Kšœœ'œœ˜9šœœ˜Kšœ œ!˜3Kšœ˜K˜—šœœ œ˜˜GKšœœ œ˜—Kšœ%œ˜9Kšœ+ ˜IK˜VK˜#K˜—K˜KšœΘ™ΘK˜Kšœ˜Kšœ ™ Kšœ ™ Kšœ(™(Kšœœ ˜+K˜'Kšœ ˜K˜—Jšœ˜—J˜—…—Έˆ