Advertisement
Guest User

WYSIwrite Latest

a guest
Apr 14th, 2015
1,799
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Lua 69.84 KB | None | 0 0
  1. --[[
  2.     WYSIwrite
  3.     A WYSIWYG text editor for printing documents in ComputerCraft
  4.  
  5.     Written by: Nitrogen Fingers
  6. ]]--
  7. doc = {}
  8.  
  9. --The width and height of the screen
  10. local w,h = term.getSize()
  11. --Whether or not the program is still running
  12. local finished = false
  13.  
  14. --The length allowed for an individual line
  15. local linelen = 25
  16. --The lines allowed for an individual page
  17. local pagelen = 21
  18.  
  19. --The currently viewed page
  20. local currpage = 1
  21. --The currently edited line
  22. local currline = 1
  23. --The currently index of the line edit
  24. local currind = 1
  25.  
  26. --The amount of scrolling applied to the page on the Y.
  27. local yscroll = 0
  28. --The drawing offset
  29. local _xoff,_yoff = math.floor(w/2 - linelen/2),4
  30. --Whether or not a scroll or page change has occurred
  31. local lscroll = true
  32. --Whether or not the player is currently in a drop down menu
  33. local inDropDown = false
  34.  
  35. --Whether or not whitespace characters are displayed
  36. local whitespaceVis = false
  37. --Whether or not a change has been made to the document since last save
  38. local sChange = false
  39. --The path currently being used to save to
  40. local sPath = nil
  41. --The file's name
  42. local sName = nil
  43.  
  44. --The default directory for the write backup; this can be changed if you don't like it
  45. local bPath = "/.writebackup"
  46. --The backup timer, and its default length
  47. local bTimer, bLength = nil, 5
  48.  
  49. --Alignment options, and a global test variable
  50. local ALIGNLEFT, ALIGNCENTER, ALIGNRIGHT = 0,1,2
  51. local testAlign = ALIGNLEFT
  52.  
  53. --The list of all possible actions
  54. local ACTIONS = {
  55.     ADDCHAR = 0;
  56.     DELETECHAR = 1;
  57.     ADDBREAK = 2;
  58. }
  59. --The list of all actions performed. They contain:
  60.     --page,line,index as the location of the event
  61.     --type from the ACTIONS table
  62.     --value is the character or characters that were added/removed
  63. local undolist = {}
  64. local redolist = {}
  65. --The timer that determines which 'actions' are clumped together with undo and redo
  66. local actionTimer = nil
  67. --How many seconds of inactivity before the action is classified as separate by undo
  68. local actionDelay = 0.5
  69.  
  70. --Menu Bindings, initialized at bottom of program
  71.  
  72. local _printers = { name = "Print" }
  73. local _iChoices = { name = "Insert" }
  74. local _aChoices = { name = "Align" }
  75. local _mChoices = { name = "File", x = 2, y = 1 }
  76. local _eChoices = { name = "Edit", x = 4 + #_mChoices.name, y = 1 }
  77.    
  78. --For me
  79. local testMode = false
  80. local _tro = false
  81. local _suppress = false
  82.  
  83. --For my testing purposes.
  84. function testPrint(_str)
  85.     if testMode then
  86.         if term.getCursorPos() ~= 1 then term.setCursorPos(1,1) end
  87.         term.setBackgroundColour(colours.white)
  88.         term.setTextColour(colours.red)
  89.         if not _suppress then print(_str) end
  90.         lscroll = true
  91.         _tro = true
  92.     end
  93. end
  94. function testWait()
  95.     if testMode and _tro then os.pullEvent("key") end
  96.     _tro = false
  97. end
  98. function testSuppress() _suppress = true end
  99. function testRelax() _suppress = false end
  100.  
  101. --Returns whether or not a token is considered a newline for printing and pagination purposes
  102. function nl(str) return str == "\n" or str == string.char(17) end
  103.  
  104. --I'm getting a lot of mileage out of this split function      
  105. function split(str, pattern)
  106.   local t = { }
  107.   local fpat = "(.-)" .. pattern
  108.   local last_end = 1
  109.   local s, e, cap = str:find(fpat, 1)
  110.   while s do
  111.     if s ~= 1 or cap ~= "" then
  112.       table.insert(t,cap)
  113.     end
  114.     last_end = e+1
  115.     s, e, cap = str:find(fpat, last_end)
  116.   end
  117.   if last_end <= #str then
  118.     cap = str:sub(last_end)
  119.     table.insert(t, cap)
  120.   end
  121.   return t
  122. end
  123.  
  124. --Returns whether or not the given string is break terminated.
  125. local function hasBreakEnd(_line)
  126.     return _line and _line:find("[\n\f]") ~= nil
  127. end
  128.  
  129.  
  130. --[[ Gets the previous file in the path, if there is one ]]--
  131. local function getPreviousDir(_path)
  132.     if _path == "" or _path == "/" then return path
  133.     else
  134.         _path = _path:reverse()
  135.         return _path:sub(_path:find("/.*")+1):reverse()
  136.     end
  137. end
  138.  
  139. --[[ Constructs an empty document that can be edited. Empty documents contain a
  140.      single page, with one line that cannot be deleted.
  141.      Parameters: none
  142.      Returns: none
  143. ]]--
  144. local function constructNewDocument(_sName, _sPath)
  145.     --pages
  146.     doc = {
  147.         --lines
  148.         {
  149.             ""
  150.         }
  151.     }
  152.     currpage = 1
  153.     currline = 1
  154.     currind = 1
  155.     paginateOverflow()
  156.     sPath = _sPath
  157.     sName = _sName
  158.     sChange = false
  159. end
  160.  
  161. --[[ Constructs a document from an existing source
  162.      Parameters: none
  163.      Returns: none
  164. ]]--
  165. local function loadDocument(_path)
  166.     doc = {
  167.         { }
  168.     }
  169.     assert(fs.exists(_path), _path)
  170.     local _file = fs.open(_path, "r")
  171.     local _line = _file.readLine()
  172.     while _line do
  173.         --Form feed check
  174.         _lineEndIndex = 1
  175.         while true do
  176.             local _fco = _line:find("\f", _lineEndIndex)
  177.             if not _fco then break end
  178.             if #doc[#doc] < pagelen then
  179.                 if _lineEndIndex == 1 then
  180.                     table.insert(doc[#doc], _line:sub(_lineEndIndex, _fco))
  181.                 else
  182.                     doc[#doc][#doc[#doc]] = _line:sub(_lineEndIndex, _fco)
  183.                 end
  184.             else
  185.                 table.insert(doc, { _line:sub(_lineEndIndex, _fco)})
  186.             end
  187.             _lineEndIndex = _fco + 1
  188.             currpage = #doc
  189.             currline = #doc[#doc]
  190.             paginateOverflow()
  191.         end
  192.         --Rest of line
  193.         if #doc[#doc] < pagelen then
  194.             if _lineEndIndex == 1 then
  195.                 table.insert(doc[#doc], _line:sub(_lineEndIndex).."\n")
  196.             else
  197.                 --Replacing the empty strings added by the paginator on form feeds.
  198.                 doc[#doc][#doc[#doc]] = _line:sub(_lineEndIndex).."\n"
  199.             end
  200.         else
  201.             table.insert(doc, { _line:sub(_lineEndIndex).."\n" })
  202.         end
  203.         currpage = #doc
  204.         currline = #doc[#doc]
  205.         paginateOverflow()
  206.        
  207.         _line = _file.readLine()
  208.         if not _line then
  209.             local _od = doc[#doc][#doc[#doc]]
  210.             doc[#doc][#doc[#doc]] = _od:sub(1,#_od-1)
  211.         end
  212.     end
  213.     _file:close()
  214.     if not doc[1][1] then doc[1][1] = "" end
  215.     currpage = 1
  216.     currline = 1
  217.     currind = 1
  218.     sChange = false
  219. end
  220.  
  221. --[[ "Depaginates" the current document into straight lines and saves the
  222.      output to a file location for standard viewing
  223.     Parameters: _spath:string= the location to save the file
  224.     Returns: none
  225. ]]--
  226. local function saveDocument(_path)
  227.     if not _path then _path = sPath end
  228.     local _lines = { "" }
  229.    
  230.     local _opi,_oli = currpage, currline
  231.     currpage = 1
  232.     currline = 0
  233.     while nextLine() do
  234.         local _cline = doc[currpage][currline]
  235.         if _cline:sub(#_cline) == "\n" then
  236.             _lines[#_lines] = _lines[#_lines].._cline:sub(1,#_cline-1)
  237.             table.insert(_lines, "")
  238.         else
  239.             _lines[#_lines] = _lines[#_lines].._cline
  240.         end
  241.     end
  242.     --Remove that empty 'newline' we include
  243.     if _lines[#_lines] == "" then table.remove(_lines, #_lines) end
  244.     local _file = fs.open(_path, "w")
  245.     for i=1,#_lines do
  246.         _l = _lines[i]
  247.         _file.writeLine(_l)
  248.     end
  249.     _file.close()
  250.     currpage, currline = _opi, _oli
  251.     if _path ~= bPath then sChange = false end
  252.     testWait()
  253. end
  254.  
  255. --A quick function to let the program know there have been changes to the file
  256. local function setUpdated()
  257.     sChange = true
  258.     bTimer = os.startTimer(bLength)
  259. end
  260.  
  261. --[[ Updates the document scroll to match the movement of the cursor.
  262.      Parameters: none
  263.      Returns: none
  264. ]]--
  265. local function checkYScroll()
  266.     if yscroll + currline + _yoff > h - 3 then
  267.         yscroll = h - 3 - currline - _yoff
  268.         lscroll = true
  269.     elseif yscroll + currline + _yoff < 3 then
  270.         yscroll = 3 - currline - _yoff
  271.         lscroll = true
  272.     end
  273. end
  274.  
  275. --[[ Returns the current amount of whitespace added to the front of the line
  276.      (in addition to spaces etc.) as required by the alignment of the paragraph.
  277.     Params: _pagenum = the page number being printed
  278.             _linenum = the linenum being printed
  279.     Returns:int = the number of additional spaces from the left before the line starts
  280. ]]--
  281. function getWhitespaceLength(_pagenum, _linenum)
  282.     local _page = doc[_pagenum]
  283.     _line = _page[_linenum]:gsub("[\n\f]", "")
  284.     if testAlign == ALIGNRIGHT then
  285.         return linelen-#_line
  286.     elseif testAlign == ALIGNCENTER then
  287.         return math.floor(linelen/2) - math.floor(#_line/2)
  288.     else
  289.         return 0
  290.     end
  291. end
  292.  
  293. --[[ Lowers the current line by 1, if possible
  294.      Parameters: int:bookean = whether or not to include null lines
  295.      Returns: true if there is a next line; false otherwise
  296. ]]--
  297. function nextLine(_inc)
  298.     --Lines are checked if we are below the pagelen and not on the last page
  299.     if (_inc and (currline < pagelen and currpage < #doc) or (currline < #doc[currpage]
  300.             and currpage == #doc)) or (not _inc and currline < #doc[currpage]) then
  301.         currline = currline + 1
  302.     elseif currpage < #doc then
  303.         currpage = currpage + 1
  304.         currline = 1
  305.     else
  306.         return false
  307.     end
  308.     return true
  309. end
  310.  
  311. --[[ Ensures the cursor is in a valid position. Being outside the size of
  312.      a string or before a forbidden character like a newline or tab are fixed.
  313.      Parameters: none
  314.      Returns: none
  315. ]]--
  316. local function fixCursorPos()
  317.     if not doc[currpage] then
  318.         currpage = #doc
  319.         currline = #doc[#doc]
  320.     end
  321.     if not doc[currpage][currline] then
  322.         if currline > pagelen then
  323.             currline = pagelen
  324.         else
  325.             --I don't know if this often occurs...
  326.             currline = #doc[currpage]
  327.         end
  328.         currind = #doc[currpage][currline] + 1
  329.     end
  330.     if currind < 1 then currind = 1
  331.     elseif currind > #doc[currpage][currline] + 1 then
  332.         currind = #doc[currpage][currline] + 1
  333.     end
  334.     if currind > #doc[currpage][currline] and hasBreakEnd(doc[currpage][currline]) then
  335.         currind = currind - 1
  336.     end
  337. end
  338.  
  339. --[[ Goes to the previous line, if possible
  340.      Parameters: none
  341.      Returns: true if there is a previous line; false otherwise
  342. ]]--
  343. local function lastLine()
  344.     if currline > 1 then
  345.         currline = currline - 1
  346.     elseif currpage > 1 then
  347.         currpage = currpage - 1
  348.         currline = pagelen
  349.     else return false end
  350.     return true
  351. end
  352.  
  353. --[[ Gets the previous line in the document to the one provided
  354.      Parameters: _page: the page in question (default to currpage)
  355.                  _line: the line in question (default to currline)
  356.      Returns: string=The previous line, or nil if on the first line
  357. ]]--
  358. local function getPrevLine(_page, _line)
  359.     if not _page or not _line then _page = currpage _line = currline end
  360.     if _page == 1 and _line == 1 then return nil
  361.     elseif _line == 1 then _page = _page - 1 _line = #doc[_page]
  362.     else _line = _line - 1 end
  363.     return doc[_page][_line]
  364. end
  365.  
  366. --[[ Gets the next line in the document to the one provided
  367.      Parameters: _page: the page in question (default to currpage)
  368.                  _line: the line in question (default to currline)
  369.      Returns: string=The next line, or nil if on the last line
  370. ]]--
  371. local function getNextLine(_page, _line)
  372.     if not _page or not _line then _page = currpage _line = currline end
  373.     if _page == #doc and _line == #doc[#doc] then return nil
  374.     elseif _line == #doc[_page] then _line = 1 _page = _page + 1
  375.     else _line = _line + 1 end
  376.     return doc[_page][_line]
  377. end
  378.  
  379. --[[ A helper that returns whether or not the input line is the last in the
  380.      document (the last line has a few special rules associated with it)
  381.      Parameters: _page: the page in question (default to currpage)
  382.                  _line: the line in question (default to currline)
  383.      Returns: true if the last line in document; false otherwise
  384. ]]--
  385. local function isLastLine(_page, _line)
  386.     if not _page or not _line then _page = currpage _line = currline end
  387.     return _page == #doc and _line == #doc[#doc]
  388. end
  389.  
  390. --[[Inserts a character to function as a line break. These can be any form of
  391.     character which should end the line and disallow further edits past it.
  392.     By current implementation, the break is inserted at the point of the cursor.
  393.     Params: _sym:character = the kind of break to insert. Currently, \n and
  394.                 \f are supported.
  395. ]]--
  396. function insertLineBreak(_sym)
  397.     --First we put whatever follows the cursor in a carry buffer
  398.     local _cbuff = doc[currpage][currline]:sub(currind)
  399.     doc[currpage][currline] = doc[currpage][currline]:sub(1,currind-1).._sym
  400.     _desl,_desp = -1,-1
  401.     --We then push the carry buffer down each line in the document
  402.     while nextLine(true) do
  403.         --This remembers where our cursor needs to be when we're done
  404.         if _desl == -1 or _desp == -1 then
  405.             _desl, _desp = currline, currpage
  406.         end
  407.         --First line form feed test
  408.         if _cbuff:find("\f") and currline == 1 then
  409.             table.insert(doc, currpage, { _cbuff })
  410.             _cbuff = nil
  411.             break
  412.         end
  413.         --We then swap out our buffer for the current line, to push down.
  414.         local _ctemp = doc[currpage][currline]
  415.         doc[currpage][currline] = _cbuff
  416.         _cbuff = _ctemp
  417.         --Special form feed test
  418.         if doc[currpage][currline]:find("\f") then
  419.             break
  420.         end
  421.     end
  422.     --We then drop in our last line
  423.     if _cbuff then
  424.         if currline < pagelen then
  425.             doc[currpage][currline+1] = _cbuff
  426.         else
  427.             doc[currpage+1] = { _cbuff }
  428.         end
  429.     end
  430.     --This updates our cursor back to where it should be
  431.     if _desl ~= -1 and _desp ~= -1 then
  432.         currind, currline, currpage = 1, _desl, _desp
  433.     else
  434.         --Called when we hit enter at the end of a document
  435.         nextLine()
  436.         currind = 1
  437.     end
  438.     testWait()
  439.     paginateUnderflow()
  440.     testWait()
  441.     checkYScroll()
  442.     setUpdated()
  443. end
  444.  
  445. --[[ Adds a sequence of text to the current line and index
  446.      Parameters: val:String = the character to be inserted
  447.      Returns: none
  448. ]]--
  449. local function addCharacter(_val)
  450.     local _cltext = doc[currpage][currline]
  451.     if currind > #_cltext then
  452.         _cltext = _cltext.._val
  453.     else
  454.         _cltext = _cltext:sub(1, currind-1).._val.._cltext:sub(currind)
  455.     end
  456.     doc[currpage][currline] = _cltext
  457.     currind = currind + #_val
  458.     paginateOverflow()
  459.     -- Space actions must also be underflowed; they may have broken a word.
  460.     if _val == " " then paginateUnderflow() end
  461.     checkYScroll()
  462.     setUpdated()
  463. end
  464.  
  465. --[[ Removes a character from the current line and index
  466.      Parameters: none
  467.      Returns:String = the character that was deleted
  468. ]]--
  469. function deleteCharacter()
  470.     --We first move our cursor to the character we want to delete
  471.     if currind == 1 then
  472.         if currline > 1 then currline = currline - 1
  473.         elseif currpage > 1 then
  474.             currpage = currpage - 1
  475.             currline = #doc[currpage]
  476.         else return end
  477.         currind = #doc[currpage][currline]
  478.     else currind = currind - 1 end
  479.     --Then create 2 'snips'; before and after the character
  480.     local _cline = doc[currpage][currline]
  481.     local _fsnip,_asnip = "",""
  482.     if currind > 1 then _fsnip = _cline:sub(1,currind-1) end
  483.     if currind < #_cline then _asnip = _cline:sub(currind+1) end
  484.     --We patch our snips, remove dead lines and do a cleanup
  485.     doc[currpage][currline] = _fsnip.._asnip
  486.     if doc[currpage][currline] == "" then
  487.         _lline = getPrevLine()
  488.         if not _lline or not hasBreakEnd(_lline) then
  489.             removeLine(currpage, currline)
  490.         end
  491.     end
  492.     paginateUnderflow()
  493.     fixCursorPos()
  494.     checkYScroll()
  495.     setUpdated()
  496.     return _cline:sub(currind,currind)
  497. end
  498.  
  499. --[[Undoes the first action in the provided list, and pushes it's inverse to the second.
  500.     Parameters: _remList: the list to reverse the action
  501.                 _addList: the list to push the move to
  502.     Returns: nil
  503. ]]--
  504. local function undoAction(_remList,_addList)
  505.     if #_remList == 0 then return end
  506.     local _actionList,_newList = table.remove(_remList, #_remList), {}
  507.     testPrint("List has "..#_actionList.." elements")
  508.     for _,_action in ipairs(_actionList) do
  509.         testPrint("At "..currind.." doing ".._action.type.." <".._action.value..">")
  510.         currpage,currline,currind = _action.page, _action.line, _action.index
  511.         if _action.type == ACTIONS.ADDCHAR then
  512.             currind = currind + 1
  513.             deleteCharacter()
  514.             _action.type = ACTIONS.DELETECHAR
  515.         elseif _action.type == ACTIONS.DELETECHAR then
  516.             if _action.value == "\n" or _action.value == "\f" then
  517.                 insertLineBreak(_action.value)
  518.                 _action.type = ACTIONS.ADDBREAK
  519.             else
  520.                 addCharacter(_action.value)
  521.                 _action.type = ACTIONS.ADDCHAR
  522.             end
  523.         elseif _action.type == ACTIONS.ADDBREAK then
  524.             currind = currind + 1
  525.             deleteCharacter()
  526.             _action.type = ACTIONS.DELETECHAR
  527.         end
  528.         table.insert(_newList, 1, _action)
  529.     end
  530.     table.insert(_addList, _newList)
  531. end
  532.  
  533. --[[Adds a new undoable action to the current list
  534.     Parameters: _ind: the index the cursor needs to be at
  535.                 _type: the type of action
  536.                 _value: the character impacted by the action
  537.  
  538. ]]--
  539. local function addUndoableAction(_ind, _type, _value)
  540.     if not actionTimer then
  541.         table.insert(undolist, {})
  542.     end
  543.     actionTimer = os.startTimer(actionDelay)
  544.     table.insert(undolist[#undolist], 1, {
  545.         page = currpage;
  546.         line = currline;
  547.         index = _ind;
  548.         type = _type;
  549.         value = _value;
  550.     })
  551. end
  552.  
  553. --[[Returns the page and line by adding the given linecount. Allows going past
  554.     the bounds of the document, and can be used in reverse (but always stops at 1,1)
  555.     Params: _page:int = the page number to start at
  556.             _line:int = the line number to start at
  557.             _linecount:int = the number of lines to move forwards or backwards
  558.     Returns:int = the new page
  559.             int = the new line
  560. ]]--
  561. function moveBy(_page, _line, _linecount)
  562.     _line = _line + _linecount
  563.     while _line > pagelen do
  564.         _line = _line - pagelen
  565.         _page = _page + 1
  566.     end
  567.     return _page, _line
  568. end
  569.  
  570. --[[Takes a line with a page break terminator and determines whether or
  571.     not the document complies with the check. Only once per update now :)
  572.     Params: _pagenum:int = the page number of the line
  573.             _linenum:int = the line number
  574.     Returns: true if a change was made, false otherwise
  575.              int,int: the new page and line number of the NEXT page break;
  576.              this will need to be evaluated.
  577. ]]--
  578. function performPageBreakCheck(_pagenum, _linenum)
  579.     if not _pagenum and not _linenum then _pagenum, _linenum = 1, 1 end
  580.    
  581.     --For all subsequent pages, a reverse parse to ensure there are no null lines
  582.     --between them and the next page (except where there are page breaks of course)
  583.     local _docEnd = #doc
  584.     local _nullSpace = {}
  585.     local _floatingLines = {}
  586.     _p = _pagenum
  587.     while _p <= #doc do
  588.         local _breakPassed = false
  589.         local _textOnPage = true
  590.         for _l=_linenum,pagelen do
  591.             local _line = doc[_p][_l]
  592.             if not _line then
  593.                 if #_floatingLines > 0 and not _breakPassed then
  594.                     doc[_p][_l] = table.remove(_floatingLines, 1)
  595.                     if type(_floatingLines[1]) == "number" then
  596.                         currind = table.remove(_floatingLines, 1)
  597.                         currpage, currline = _p, _l
  598.                     end
  599.                     _textOnPage = true
  600.                     if doc[_p][_l]:sub(#doc[_p][_l]) == "\f" then
  601.                         _breakPassed = true
  602.                         _nullSpace = {}
  603.                     end
  604.                 elseif not _breakPassed then
  605.                     table.insert(_nullSpace, {_p, _l})
  606.                 end
  607.             else
  608.                 if _breakPassed then
  609.                     table.insert(_floatingLines, _line)
  610.                     if _p == currpage and _l == currline then
  611.                         table.insert(_floatingLines, currind)
  612.                     end
  613.                     doc[_p][_l] = nil
  614.                 elseif #_floatingLines > 0 then
  615.                     table.insert(_floatingLines, _line)
  616.                     doc[_p][_l] = table.remove(_floatingLines, 1)
  617.                     if type(_floatingLines[1]) == number then
  618.                         currind = table.remove(_floatingLines[1])
  619.                         currpage, currline = _p, _l
  620.                     end
  621.                 elseif #_nullSpace > 0 then
  622.                     local _ns = table.remove(_nullSpace, 1)
  623.                     doc[_ns[1]][_ns[2]] = _line
  624.                     doc[_p][_l] = nil
  625.                     if currpage == _p and _currline == _l then
  626.                         currpage, currline = _ns[1], _ns[2]
  627.                     end
  628.                     --Now this MAY be unsafe, but I can't recreate the issue on paper
  629.                     if _line:sub(#_line) == "\f" then
  630.                         _breakPassed = true
  631.                         _nullSpaces = {}
  632.                     end
  633.                     if _ns[1] == _p then _textOnPage = true end
  634.                 elseif _line:sub(#_line) == "\f" then
  635.                     _breakPassed = true
  636.                     _textOnPage = true
  637.                     _nullSpaces = {}
  638.                 end
  639.             end
  640.         end
  641.        
  642.         if not _textOnPage and _p > 1 then
  643.             table.remove(doc, _p)
  644.             _p = _p - 1
  645.         end
  646.         if _p == #doc and #_floatingLines > 0 then
  647.             table.insert(doc, {})
  648.         end
  649.         if _p == #doc and _breakPassed and #_floatingLines == 0 then
  650.             table.insert(doc, {})
  651.             table.insert(_floatingLines, "")
  652.         end
  653.        
  654.         testWait()
  655.         _docEnd = #doc
  656.         _p = _p + 1
  657.         _linenum = 1
  658.     end
  659.     lscroll = true
  660. end
  661.  
  662. --[[ Removes an entire line from the document, and shifts all previous
  663.      lines up
  664.      Parameters: pagenum:int = the page the line is on
  665.                  linenum:int = the line to remove
  666.      Returns: none
  667. ]]--
  668. function removeLine(_pagenum, _linenum)
  669.     _linenum = _linenum + 1
  670.     if _linenum > pagelen then _linenum = 1 _pagenum = _pagenum + 1 end
  671.     local _lp = ""
  672.     --To delete, we copy each line to replace the previous
  673.     while _pagenum <= #doc and _linenum <= #doc[_pagenum] do
  674.         local _cbuff = doc[_pagenum][_linenum]
  675.         if _linenum == 1 then
  676.             doc[_pagenum-1][pagelen] = _cbuff
  677.             --Nothing follows a form feed, so the last page was empty. Delete!
  678.             if _cbuff:find("\f") then
  679.                 table.remove(doc, _pagenum)
  680.                 assert(doc[_pagenum], "Deleted last page (ff) in remove line")
  681.                 return
  682.             end
  683.         else
  684.             doc[_pagenum][_linenum-1] = _cbuff
  685.             --Nothing follows a form feed so we stop removing here.
  686.             if _cbuff:find("\f") then doc[_pagenum][_linenum] = nil return end
  687.         end
  688.         if _linenum == pagelen then _linenum = 1 _pagenum = _pagenum + 1
  689.         else _linenum = _linenum + 1 end
  690.         _lp = _cbuff
  691.     end
  692.     --And black out the very last so we know nothing is there.
  693.     _linenum = _linenum - 1
  694.     if (_pagenum == 1 and _linenum == 1) or _lp:sub(#_lp) == "\n" then
  695.         doc[#doc][#doc[#doc]] = ""
  696.     else doc[#doc][#doc[#doc]] = nil end
  697.     if #doc[#doc] == 0 and #doc ~= 1 then
  698.         table.remove(doc, #doc)
  699.     end
  700.     --So long as we still have something in the doc
  701.     if #doc[1] == 0 then doc[1][1] = "" end
  702. end
  703.  
  704. --[[ Inserts a line of text into a specified index. The replaced line and
  705.      all subsequent lines are pushed down the document
  706.      Parameters: _pagenum:int = the page the line is one
  707.                  _linenum:int = the line in which to insert
  708.                  _text:string = the text to insert. Defaults to a newline.
  709. ]]--
  710. function insertLine(_pagenum, _linenum, _text)
  711.     if not _text then _text = "\n" end
  712.     local _pi,_li = #doc, #doc[#doc] + 1
  713.     if _li > linelen then _pi = _pi + 1 _li = 1 end
  714.     if not doc[_pi] then doc[_pi] = { } end
  715.     local _icount = 0
  716.     while _pi > _pagenum or _li > _linenum do
  717.         doc[_pi][_li] = getPrevLine(_pi,_li)
  718.         _li = _li - 1
  719.         if _li == 0 then _li = pagelen _pi = _pi - 1 end
  720.         _icount = _icount + 1
  721.     end
  722.     doc[_pagenum][_linenum] = _text
  723. end
  724.  
  725.  
  726. --[[ Updates pagination at the point following the cursor for overflow as
  727.      a result of an insertion edit action. The overflow by default only
  728.      checks so far as the previous insertion but can be set to check the entire
  729.      program through a switch.
  730.      Returns: none
  731. ]]--
  732. function paginateOverflow()
  733.     --Whether or not we need to keep paginating
  734.     local _pfollow = false
  735.     --The line we start at on this page
  736.     local _pi,_nli = currpage, currline
  737.     --Whether or not we've done an insertion, and need to perform another check
  738.     local _icheck = false
  739.    
  740.     while _pi <= #doc do
  741.         _li = _nli
  742.         while _li <= #doc[_pi] do
  743.             local _cltext = doc[_pi][_li]
  744.             --First we check to see if it's a carry-on line
  745.             local _conn = 0
  746.             local _nl = getNextLine(_pi,_li)
  747.             if _nl and _cltext:sub(#_cltext) ~= " " and
  748.                     _nl:sub(1,1) ~= " " then
  749.                 _conn = _nl:find(" ")
  750.                 if not _conn then _conn = _nl:find("[\n\f]") end
  751.                 if not _conn then _conn = #_nl
  752.                 else
  753.                     _conn = _conn - 1
  754.                 end
  755.             end
  756.             --And a bit of leniency for newlines (they don't print anyway)
  757.             if hasBreakEnd(_cltext) then _conn = -1 end
  758.             --If the line is shorter than the line length, we are done
  759.             if (#_cltext + _conn <= linelen and not _icheck)
  760.                     or #_cltext == 0 then
  761.                 _pfollow = true
  762.                 break
  763.             end
  764.             _icheck = false
  765.  
  766.             --Next check is for form feeds stuck between lines, we space them out
  767.             local _nsp = -1
  768.             local _ffline = _cltext:find("\f")
  769.             if _ffline and _ffline <= linelen then _nsp = _ffline end
  770.            
  771.             --Otherwise we find the nearest space to the edge of the document
  772.             if _nsp == -1 then
  773.                 for _nsi = linelen, 1, -1 do
  774.                     if _cltext:sub(_nsi,_nsi) == " " then
  775.                         _nsp = _nsi
  776.                         break
  777.                     end
  778.                 end
  779.             end
  780.             --The excess string is carried to the front of the next line or page
  781.             --Notice the space stays if there is more than one word on that line
  782.             if _nsp == -1 then _nsp = linelen end
  783.             doc[_pi][_li] = doc[_pi][_li]:sub(1,_nsp)
  784.             _carrystr = _cltext:sub(_nsp+1)
  785.             if _li == pagelen then
  786.                 if _pi == #doc then
  787.                     doc[_pi+1] = { _carrystr }
  788.                 --Both newlines and form feeds must be carried, and cannot overflow.
  789.                 elseif hasBreakEnd(_carrystr) then
  790.                     insertLine(_pi+1,1,_carrystr)
  791.                     _icheck = true
  792.                 else
  793.                     doc[_pi+1][1] = _carrystr..doc[_pi+1][1]
  794.                 end
  795.             elseif _li == #doc[_pi] then
  796.                 doc[_pi][_li+1] = _carrystr
  797.                 --Newlines must have a following space in addition to this.
  798.             elseif _carrystr:sub(#_carrystr) == "\n" then
  799.                 insertLine(_pi,_li+1,_carrystr)
  800.                 _icheck = true
  801.             elseif _carrystr:sub(#_carrystr) == "\f" then
  802.                 assert(false, "Carry form feed at ".._pi..",".._li)
  803.             else
  804.                 doc[_pi][_li+1] = _carrystr..doc[_pi][_li+1]
  805.             end
  806.             --If this pagination affected our cursor, we move down a line
  807.             if _pi == currpage and _li == currline and
  808.                      currind > #doc[currpage][currline] then
  809.                 currind = currind - #doc[currpage][currline]
  810.                 if currline == pagelen then
  811.                     currline = 1
  812.                     currpage = currpage + 1
  813.                 else
  814.                     currline = currline + 1
  815.                 end
  816.             end
  817.             _li = _li + 1
  818.             testWait()
  819.         end
  820.         if _pfollow then break end
  821.         _nli = 1
  822.         _pi = _pi + 1
  823.     end
  824.     performPageBreakCheck()
  825. end
  826.  
  827. --[[ Updates pagination at the point following the trailing line from the
  828.      cursor for underflow as a result of a delete edit action.
  829.      The pagination is optimized to only go so far as it has to when making
  830.      changes, but can cover the entire document through a switch.
  831.      Parameters: _cflag:bool= set to true to check entire document
  832.      Returns: none
  833. ]]--
  834. function paginateUnderflow(_cflag)
  835.     --Whether or not we need to keep paginating
  836.     local _pfollow = false
  837.     --Whether or not a vulnerable underflow occurred. These are characterized
  838.     --as underflows that leave a partial word behind.
  839.     local _vocc = false
  840.     --How many pagination checks we need to run. It's a minimum of 2
  841.     --if we're on the same line
  842.     local _pcount = 3
  843.    
  844.     --First step, we check out how the page breaks have modified things
  845.     performPageBreakCheck()
  846.    
  847.     --The line we start at on this page
  848.     local _pi,_nli = currpage, currline
  849.     if _cflag then _pi,_nli = 1,1 end
  850.     --Our comparataor line
  851.     local _cpline = doc[_pi][_nli]
  852.     --The last character of the second to last line
  853.     local _llsnip = nil
  854.     --Where do we start, here or next line?
  855.     if _pi == 1 and _nli == 1 then _nli = _nli + 1 _pcount = _pcount - 1
  856.     elseif _nli == 1 then
  857.         _cpline = doc[_pi-1][#doc[_pi-1]]
  858.     else _cpline = doc[_pi][_nli-1] end
  859.    
  860.     while _pi <= #doc do
  861.         local _li = _nli
  862.         while _li <= #doc[_pi] do
  863.             _llsnip = _cpline:sub(#_cpline)
  864.             local _cline = doc[_pi][_li]
  865.             local _nsp = -1
  866.             local _conn = 0
  867.             if hasBreakEnd(_cline) then _conn = -1 end
  868.             --We search for a word that will fit on the previous line
  869.             for _nsi = linelen - #_cpline,1,-1 do
  870.                 if _cline:sub(_nsi,_nsi) == " " then
  871.                     _nsp = _nsi
  872.                     --Because we don't use it here
  873.                     _conn = 0
  874.                     break
  875.                 end
  876.             end
  877.             --Single words short enough can be moved too
  878.             if #_cline+_conn <= linelen - #_cpline then
  879.                 _nsp = #_cline+_conn
  880.                 --A little exception in the event we leave our cursor behind
  881.             end
  882.             --And if there's no space at the end of the line, it moves back too
  883.             if _nsp == -1 and _cpline:sub(#_cpline) ~= " "
  884.                     and not hasBreakEnd(_cpline) then
  885.                 _nsp = linelen - #_cpline
  886.                 --Also unused here
  887.                 _conn = 0
  888.                 --This however now requires an overflow check
  889.                 _vocc = true
  890.             end
  891.            
  892.             --If no such word exists, we're done
  893.             --Also, if our comparator line has a break, underflow is disallowed
  894.             --NOTE: Hideous code. This could be done far better.
  895.             if (_nsp == -1 or hasBreakEnd(_cpline)) then
  896.                 if _pi ~= currpage and _li ~= currline and _pcount == 0 then
  897.                     _pfollow = true
  898.                     break
  899.                 end
  900.             else
  901.                 --Otherwise we snip our words from the current line,
  902.                 local _snip = _cline:sub(1,_nsp-_conn)
  903.                 doc[_pi][_li] = _cline:sub(_nsp+1-_conn)
  904.                 --then stick the snip on the next line (lazy coding but meh)
  905.                 if _li > 1 then doc[_pi][_li-1] = doc[_pi][_li-1].._snip
  906.                 else
  907.                     doc[_pi-1][#doc[_pi-1]] = doc[_pi-1][#doc[_pi-1]].._snip
  908.                 end
  909.                 --If we performed a deletion on the cursor line line and the
  910.                 --cursor is within the clip, we must move it
  911.                 if _pi == currpage and _li == currline and currind <= _nsp-_conn then
  912.                     currind = currind + #_cpline
  913.                     if _li == 1 then
  914.                         currpage = currpage -1
  915.                         currline = pagelen
  916.                     else
  917.                         currline = currline - 1
  918.                     end
  919.                 end
  920.             end
  921.             --Remove any excess lines
  922.             local _slsnip = getPrevLine(#doc, #doc[#doc])
  923.             if doc[_pi][_li] == "" and not(isLastLine(_pi, _li)
  924.                     and hasBreakEnd(_slsnip)) then
  925.                 --Remove line was not moving cursor. This needs to be fixed in later versions.
  926.                 if _pi > currpage or (_pi == currpage and currline >= _li) then
  927.                     --Our exception case; cursor is at EOF
  928.                     if _pi == #doc and _li == #doc[_pi] then
  929.                         removeLine(_pi,_li)
  930.                         fixCursorPos()
  931.                     else
  932.                         removeLine(_pi,_li)
  933.                     end
  934.                 else
  935.                     removeLine(_pi,_li)
  936.                 end
  937.                 _li = _li - 1
  938.             end
  939.             --Update our variables, and we're done here.
  940.             if not doc[_pi] or not doc[_pi][_li] then break end
  941.             _cpline = doc[_pi][_li]
  942.             _li = _li + 1
  943.             _pcount = math.max(_pcount-1, 0)
  944.         end
  945.         if _pfollow then break end
  946.         _nli = 1
  947.         _pi = _pi + 1
  948.     end
  949.     --Unsafe carryovers have to check for overflow pagination.
  950.     if _vocc then
  951.         paginateOverflow()
  952.     end
  953. end
  954.  
  955. --[[ Draws the 'whitespace' surrounding the page and the menu. This includes both
  956.      page-display features like the margins around the page and borders to make it more
  957.      distinct.
  958.      Parameters: none
  959.      Returns: none
  960. ]]--
  961. local function displayClearPage()
  962.     --Drawing the background
  963.     term.setBackgroundColour(colours.lightBlue)
  964.     term.clear()
  965.     term.setBackgroundColour(colours.grey)
  966.     --Adding a little border to the page to make it more distinct
  967.     local _ydpos = _yoff + yscroll - 1
  968.     if _ydpos > 0 and _ydpos <= h then
  969.         term.setCursorPos(_xoff + 1, _ydpos)
  970.         term.write(string.rep(" ", 2 + linelen))
  971.         term.setBackgroundColour(colours.white)
  972.         term.setCursorPos(_xoff, _ydpos + 1)
  973.         term.write(string.rep(" ", 2 + linelen))
  974.     end
  975.     if _ydpos + pagelen + 2 > 0 and _ydpos + pagelen + 2 <= h then
  976.         term.setBackgroundColour(colours.white)
  977.         term.setCursorPos(_xoff, _ydpos + pagelen + 2)
  978.         term.write(string.rep(" ", 2 + linelen))
  979.     end
  980.     for i = math.max(0, _ydpos), math.min(h, pagelen + yscroll + _yoff) do
  981.         if i > _ydpos or i == 0 then
  982.             term.setBackgroundColour(colours.white)
  983.             term.setCursorPos(_xoff, i)
  984.             term.write(" ")
  985.             term.setCursorPos(_xoff + linelen + 1, i)
  986.             term.write(" ")
  987.         else
  988.             term.setCursorPos(_xoff + linelen + 2, i)
  989.         end
  990.         term.setBackgroundColour(colours.grey)
  991.         term.write(" ")
  992.     end
  993. end
  994.  
  995. --[[ Draws the currently selected page. This prints exactly the contents of the page and
  996.      whitespace to fill out the gaps as necessary.
  997.      Parameters: page:The page of the document to print
  998.                  _out: The output construct
  999.                  _x:int = the x offset for the page to be printed at
  1000.                  _y:int = the y offset for the page to be printed at
  1001.      Returns: none
  1002. ]]--
  1003. function printPage(pagenum, _out, _x, _y)
  1004.     page = doc[pagenum]
  1005.     if not _x or not _y then _x = 1 _y = 0 end 
  1006.     --Displaying the page itself, both text and no text
  1007.     for i=1,pagelen do
  1008.         local _ydpos = i + _y
  1009.         --This little optimization only works if _out == term, so it's not in for now.
  1010.         --if _out == term and _ydpos > 1 and _ydpos <= h then
  1011.         _out.setCursorPos(_x, i + _y)
  1012.         if i > 0 and i <= #page then
  1013.             local _wsEnd = hasBreakEnd(page[i])
  1014.             local _disp = string.gsub(page[i], "\n"," ")
  1015.             --Depending on the alignment, we add whitespace to make up the difference.
  1016.             local _wchar = " "
  1017.             if testMode then
  1018.                 term.setTextColour(colours.red)
  1019.                 _wchar = "-"
  1020.             end
  1021.             term.write(string.rep(_wchar, getWhitespaceLength(pagenum,i)))
  1022.             -- Our whitespace display function
  1023.             if (whitespaceVis or testMode) and _out == term then
  1024.                 for _c=1,#page[i] do
  1025.                     local _chr = page[i]:sub(_c,_c)
  1026.                     if _chr == " " then
  1027.                         term.setTextColour(colours.red)
  1028.                         term.write(".")
  1029.                     elseif _chr == "\n" then
  1030.                         term.setTextColour(colours.red)
  1031.                         term.write(">")
  1032.                     elseif _chr == "\f" then
  1033.                         term.setTextColour(colours.red)
  1034.                         term.write("V")
  1035.                     else
  1036.                         term.setTextColour(colours.black)
  1037.                         term.write(_chr)
  1038.                     end
  1039.                 end
  1040.                 term.setTextColour(colours.black)
  1041.             else
  1042.                 --Bah. This has to change. Forget nice code. This program is one enormous, hideous hack.
  1043.                 _out.write(_disp)
  1044.             end
  1045.             _out.write(string.rep(" ", math.max(linelen - #_disp -
  1046.                     getWhitespaceLength(pagenum, i), 0)))
  1047.             --Another hack. Exactly what I am....
  1048.             if _out == term and not _wsEnd then _out.write(" ") end
  1049.         else
  1050.             _out.write(string.rep(" ", linelen))
  1051.         end
  1052. --      end
  1053.     end
  1054. end
  1055.  
  1056.  
  1057.         --[[ Interface malarky ]]--
  1058.  
  1059. --[[ A presently mostly empty function that displays tools and information about the
  1060.      current document for the user
  1061.      Parameters: none
  1062.      Returns: none
  1063. ]]--
  1064. local function displayInterface()
  1065.     term.setBackgroundColour(colours.lightBlue)
  1066.     term.setTextColour(colours.black)
  1067.     --A little interface feature to tell the user what page they're viewing
  1068.     local _ppos = string.rep(" ",math.floor(math.log10(#doc)) - math.floor(math.log10(currpage)))
  1069.             ..currpage.." of "..#doc
  1070.     term.setCursorPos(w-#_ppos-1,h)
  1071.     term.setTextColour(colours.black)
  1072.     term.write(_ppos)
  1073.     term.setCursorPos(1,1)
  1074.     term.setBackgroundColour(colours.lightGrey)
  1075.     if testMode then term.setBackgroundColour(colours.red) end
  1076.     term.setTextColour(colours.black)
  1077.     term.clearLine()
  1078.    
  1079.     term.setCursorPos(_mChoices.x, _mChoices.y)
  1080.     term.write(_mChoices.name)
  1081.     term.setCursorPos(_eChoices.x, _eChoices.y)
  1082.     term.write(_eChoices.name)
  1083.    
  1084.     local _t = sPath or "Unnamed"
  1085.     if #_t > w - 14 then _t = _t:sub(1,#_t-math.max(sName,14)-3).."..."..sName end
  1086.     term.setTextColour(colours.grey)
  1087.     term.setCursorPos(w/2-#_t/2,1)
  1088.     term.write(_t)
  1089.     if sChange then
  1090.         term.setTextColour(colours.red)
  1091.         term.write("*")
  1092.     end
  1093. end
  1094.  
  1095. --[[ Clears and redraws the screen ]]--
  1096. local function clearAndRedraw()
  1097.     displayClearPage()
  1098.     term.setBackgroundColour(colours.white)
  1099.     term.setTextColour(colours.black)
  1100.     printPage(currpage, term, _xoff + 1, _yoff + yscroll)
  1101.     displayInterface()
  1102. end
  1103.  
  1104. --[[ Draws the WYSIwrite logo. ENORMOUSLY lazy but I'm over this. ]]--
  1105. local function drawLogo(x,y)
  1106.     term.setBackgroundColour(colours.grey)
  1107.     for i=2,3 do
  1108.         term.setCursorPos(x,y+i)
  1109.         term.write(string.rep(" ", 5))
  1110.     end
  1111.     term.setBackgroundColour(colors.white)
  1112.     for i=0,3 do if i ~= 2 then
  1113.         term.setCursorPos(x + 1, y+i)
  1114.         term.write(string.rep(" ",3))
  1115.     end end
  1116.     term.setCursorPos(x+2,y+1)
  1117.     term.write("W")
  1118. end
  1119.  
  1120. --[[ Shows a quick "about" page for those curious ]]--
  1121. local function displayAbout()
  1122.     local _dw,_dh = w/2+3,6
  1123.     local _l,_t = math.floor(w/2-_dw/2),6
  1124.    
  1125.     clearAndRedraw()
  1126.     term.setBackgroundColour(colours.lightGrey)
  1127.     term.setTextColour(colours.black)
  1128.     for _y=0,_dh do
  1129.         term.setCursorPos(_l,_t+_y)
  1130.         term.write(string.rep(" ", _dw))
  1131.     end
  1132.     term.setCursorPos(_l + 7, _t + 2)
  1133.     term.write("WYSIwrite")
  1134.     term.setCursorPos(_l + 7, _t + 4)
  1135.     term.write("By Nitrogen Fingers")
  1136.     drawLogo(_l + 1, _t + 1)
  1137.     while true do
  1138.         local id,_key = os.pullEvent()
  1139.         if id == "timer" and _key == bTimer then saveDocument(bPath)
  1140.         else break end
  1141.     end
  1142.     lscroll = true
  1143. end
  1144.  
  1145. --[[ A recycled centre print function I've gotten a LOT of mileage out of
  1146.      Parameters: msg:string= the message to display
  1147.                  height:int= the y position of the start of the message
  1148.                  width:int= the allowed width of the message
  1149.                  offset:int= the leftmost border of the message
  1150.      Returns:int= the number of lines printed
  1151. ]]--
  1152. local function wprintOffCenter(msg, height, width, offset)
  1153.     local inc = 0
  1154.     local ops = 1
  1155.     while #msg - ops > width do
  1156.         local nextspace = 0
  1157.         while string.find(msg, " ", ops + nextspace) and
  1158.                 string.find(msg, " ", ops + nextspace) - ops < width do
  1159.             nextspace = string.find(msg, " ", nextspace + ops) + 1 - ops
  1160.         end
  1161.         local ox,oy = term.getCursorPos()
  1162.         term.setCursorPos(width/2 - (nextspace)/2 + offset, height + inc)
  1163.         inc = inc + 1
  1164.         term.write(string.sub(msg, ops, nextspace + ops - 1))
  1165.         ops = ops + nextspace
  1166.     end
  1167.     term.setCursorPos(width/2 - #string.sub(msg, ops)/2 + offset, height + inc)
  1168.     term.write(string.sub(msg, ops))
  1169.    
  1170.     return inc + 1
  1171. end
  1172.  
  1173. --[[Produces a nice dropdown menu based on a table of strings. Depending on the position, this will auto-adjust the position
  1174.         of the menu drawn, and allows nesting of menus and sub menus. Clicking anywhere outside the menu will cancel and return nothing
  1175.         Params: x:int = the x position the menu should be displayed at
  1176.                 y:int = the y position the menu should be displayed at
  1177.                 options:table = the list of options available to the user, as strings or submenus (tables of strings, with a name parameter)
  1178.         Returns:string the selected menu option.
  1179. ]]--
  1180. local function displayDropDown(x, y, options, noTitle)
  1181.     inDropDown = true
  1182.     if noTitle then y = y - 1 end
  1183.     --Figures out the dimensions of our thing
  1184.     local longestX = #options.name
  1185.     for i=1,#options do
  1186.         local currVal = options[i]
  1187.         if type(currVal) == "table" then currVal = currVal.name end
  1188.         longestX = math.max(longestX, #currVal)
  1189.     end
  1190.     local xOffset = math.max(0, longestX - ((w-2) - x) + 1)
  1191.     local yOffset = math.max(0, #options - ((h-1) - y))
  1192.    
  1193.     local clickTimes = 0
  1194.     local tid = nil
  1195.     local selection = nil
  1196.     while clickTimes < 4 do
  1197.         if not noTitle then
  1198.             term.setCursorPos(x-xOffset,y-yOffset)
  1199.             term.setBackgroundColour(colours.blue)
  1200.             term.setTextColour(colours.white)
  1201.             term.write(options.name.." ")
  1202.         end
  1203.  
  1204.         for i=1,#options do
  1205.             term.setCursorPos(x-xOffset, y-yOffset+i)
  1206.             local currVal = options[i]
  1207.             if type(currVal.value) == "table" then
  1208.                 currVal.enabled = #currVal.value > 0
  1209.             end
  1210.            
  1211.             if i==selection and clickTimes % 2 == 0 then
  1212.                 term.setBackgroundColour(colours.blue)
  1213.                 if options[selection].enabled then term.setTextColour(colours.white)
  1214.                 else term.setTextColour(colours.grey) end
  1215.             else
  1216.                 term.setBackgroundColour(colours.lightGrey)
  1217.                 local _tcol = colours.black
  1218.                 if not currVal.enabled then _tcol = colours.grey end
  1219.                 term.setTextColour(_tcol)
  1220.             end
  1221.             if type(currVal.value) == "table" then
  1222.                 term.write(currVal.name..string.rep(" ", longestX-#currVal.name))
  1223.                 term.setBackgroundColour(colours.blue)
  1224.                 term.setTextColour(colours.white)
  1225.                 term.write(">")
  1226.             else
  1227.                 term.write(currVal.name..string.rep(" ", longestX-#currVal.name + 1))
  1228.             end
  1229.            
  1230.             if (i~= selection or clickTimes %2 == 1) and currVal.key and currVal.enabled then
  1231.                 term.setTextColour(colours.blue)
  1232.                 term.setBackgroundColour(colours.lightGrey)
  1233.                 local co = currVal.name:find(currVal.key)
  1234.                 if not co then co = longestX else co = co - 1 end
  1235.                 term.setCursorPos(x-xOffset+co, y-yOffset + i)
  1236.                 term.write(currVal.key)
  1237.             end
  1238.         end
  1239.        
  1240.         local id, p1, p2, p3 = os.pullEvent()
  1241.         if id == "timer" then
  1242.             if p1 == tid then
  1243.                 clickTimes = clickTimes + 1
  1244.                 if clickTimes > 2 then
  1245.                     break
  1246.                 else
  1247.                     tid = os.startTimer(0.1)
  1248.                 end
  1249.             elseif p1 == bTimer then saveDocument(bPath) end
  1250.         elseif id == "key" and not tid then
  1251.             if p1 == keys.leftCtrl then
  1252.                 selection = ""
  1253.                 break
  1254.             elseif p1 == keys.down and (not selection or selection < #options) then
  1255.                 selection = selection or 0
  1256.                 _os = selection
  1257.                 repeat selection = selection + 1
  1258.                 until selection == #options + 1 or options[selection].enabled
  1259.                 --if selection == #options + 1 then selection = _os end
  1260.             elseif p1 == keys.up and (not selection or selection > 1) then
  1261.                 selection = selection or #options + 1
  1262.                 _os = selection
  1263.                 repeat selection = selection - 1
  1264.                 until selection == 0 or options[selection].enabled
  1265.                 if selection == 0 then selection = _os end
  1266.             elseif p1 == keys.enter and selection and options[selection].enabled then
  1267.                 tid = os.startTimer(0.1)
  1268.                 clickTimes = clickTimes - 1
  1269.             end
  1270.         elseif id == "mouse_click" and not tid then
  1271.             local _xp, _yp = x - xOffset, y - yOffset
  1272.             if p2 >= _xp and p2 <= _xp + longestX + 1 and
  1273.             p3 >= _yp+1 and p3 <= _yp+#options then
  1274.                 if options[p3 - _yp].enabled then
  1275.                     selection = p3-(_yp)
  1276.                     tid = os.startTimer(0.1)
  1277.                 end
  1278.             else
  1279.                 selection = ""
  1280.                 break
  1281.             end
  1282.         elseif id == "char" and not tid then
  1283.             for k,v in ipairs(options) do
  1284.                 if v.key and v.key:lower() == p1:lower() and options[k].enabled then
  1285.                     selection = k
  1286.                     tid = os.startTimer(0.1)
  1287.                     break
  1288.                 end
  1289.             end
  1290.         end
  1291.     end
  1292.    
  1293.     local _val = selection
  1294.     if type(selection) == "number" then
  1295.         selection = options[selection].value
  1296.     end
  1297.    
  1298.     if type(selection) == "string" then
  1299.         inDropDown = false
  1300.         return selection
  1301.     elseif type(selection) == "table" then
  1302.         return displayDropDown(x + longestX + 1, y + _val, selection, true)
  1303.     elseif type(selection) == "function" then
  1304.         local _rval = selection()
  1305.         if not _rval then return "" else return _rval end
  1306.     end
  1307. end--[[ Copied out of other menus, simply provides a display to indicate something has happened.
  1308.      Parameters: ctitle:string = the title of the dialogue
  1309.                  msg:string = the message in the dialogue
  1310.                  arg:{string, colour} = pairs of buttons with strings and their text colour (background is default)
  1311.      Returns: none
  1312. ]]--
  1313. local function displayConfirmDialogue(ctitle, msg, ...)
  1314.     local _doffX, _doffY = 8, 5
  1315.     --We actually print twice- once to get the lines, second time to print proper. Easier this way.
  1316.     local lines = wprintOffCenter(msg, _doffY, w - (_doffX+2) * 2, _doffX + 2)
  1317.    
  1318.     term.setCursorPos(_doffX, 3)
  1319.     term.setBackgroundColour(colours.blue)
  1320.     term.setTextColour(colours.white)
  1321.     term.write(string.rep(" ", w - _doffX * 2))
  1322.     term.setCursorPos(_doffX + (w - _doffX * 2)/2 - #ctitle/2, 3)
  1323.     term.write(ctitle)
  1324.     term.setTextColour(colours.black)
  1325.     term.setBackgroundColour(colours.lightGrey)
  1326.     term.setCursorPos(_doffX, 4)
  1327.     term.write(string.rep(" ", w - _doffX * 2))
  1328.     for i = _doffY, _doffY + lines do
  1329.         term.setCursorPos(_doffX, i)
  1330.         term.write(" "..string.rep(" ", w - (_doffX) * 2 - 2).." ")
  1331.     end
  1332.     wprintOffCenter(msg, _doffY, w - (_doffX+2) * 2, _doffX + 2)
  1333.    
  1334.     if arg then
  1335.         term.setCursorPos(_doffX, _doffY + lines + 1)
  1336.         term.write(string.rep(" ", w - _doffX * 2))
  1337.         term.setCursorPos(_doffX, _doffY + lines + 2)
  1338.         term.write(string.rep(" ", w - _doffX * 2))
  1339.    
  1340.         local _sspace = 0
  1341.         for _k,_v in ipairs(arg) do _sspace = _sspace + #_v[1] end
  1342.         _sspace = (w - (_doffX * 2) - _sspace - (2 * #arg))/(#arg)
  1343.         if _sspace <= #arg - 1 then assert(false, "Too little space: needed "..(#arg - 1)..", got ".._sspace) end
  1344.         term.setBackgroundColour(colours.grey)
  1345.         term.setCursorPos(_doffX + 1, _doffY + lines + 1)
  1346.         local _spart = false
  1347.         for i=1,#arg do
  1348.             _v = arg[i]
  1349.             arg[i][3] = term.getCursorPos()
  1350.             term.setTextColour(_v[2])
  1351.             term.write(" ".._v[1].." ")
  1352.             local _vspace = math.floor(_sspace)
  1353.             if i >= #arg/2 and not _spart then _vspace = math.ceil(_sspace) _spart = true end
  1354.             local _x,_y = term.getCursorPos()
  1355.             term.setCursorPos(_x + _vspace, _y)
  1356.         end
  1357.     end
  1358.    
  1359.     --In the event of a message, the player hits anything to continue
  1360.     while true do
  1361.         local _id,_p1,_p2,_p3 = os.pullEvent()
  1362.         if (not arg or #arg == 0) and (id == "key" or id == "mouse_click" or id == "mouse_drag") then
  1363.             break
  1364.         elseif _id == "key" then
  1365.             if _p1 == keys.enter then return 1 end
  1366.             if _p1 == keys.backspace then return 2 end
  1367.         elseif _id == "mouse_click" and _p3 == _doffY + lines + 1 then
  1368.             for i=1,#arg do
  1369.                 _v = arg[i]
  1370.                 if _p2 >= _v[3] and _p2 <= _v[3] + #_v[1] + 1 then return i end
  1371.             end
  1372.         elseif _id == "timer" and _p1 == bTimer then saveDocument(bPath) end
  1373.     end
  1374. end
  1375.  
  1376. --[[ Runs a printing interface that lets users select the pages and number of copies to print
  1377.      Also displays toner and paper levels, and returns the start, end and copy number, or null.
  1378. ]]--
  1379. local function displayPrint(_printer,_w,_h,_y,_x)
  1380.     _w = math.floor(_w)
  1381.     if not _x then _x = math.floor(w/2-_w/2) else _x = math.floor(_x) end
  1382.     if not _y then _y = math.floor(h/2-_h/2) end
  1383.     local _bc,_tc = colours.lightGrey, colours.black
  1384.     local _btc,_ttc = colours.blue, colours.white
  1385.     local _bfc,_tfc,_tdc = colours.black, colours.white, colours.green
  1386.     --The position of the text boxes (which can vary)
  1387.     local _cXoff,_cYoff = 0,0
  1388.     local _pX,_pY,_cX,_cY = 0,0,0,0
  1389.     --The selected pages and copies
  1390.     local _pstart, _pend, _copies = 1, #doc, 1
  1391.     --The amount of paper and ink required
  1392.     local _pageCount = (_pend - _pstart + 1) * _copies
  1393.     local _ilX,_ilY = 0,0
  1394.    
  1395.     local _wsel = 0
  1396.    
  1397.     local function _drawWindow()
  1398.         --Window title
  1399.         term.setBackgroundColour(_btc)
  1400.         term.setTextColour(_ttc)
  1401.         term.setCursorPos(_x,_y)
  1402.         _title = "Print"
  1403.         term.write(string.rep(" ", math.floor(_w/2) - math.floor(#_title/2))
  1404.                 .._title..string.rep(" ", math.ceil(_w/2) - math.ceil(#_title/2)-1))
  1405.         term.setBackgroundColour(colours.red)
  1406.         term.setTextColour(colours.white)
  1407.         term.write("x")
  1408.         --Body and main options
  1409.         term.setBackgroundColour(_bc)
  1410.         term.setTextColour(_tc)
  1411.         for i=1,_h do
  1412.             term.setCursorPos(_x,_y+i)
  1413.             term.write(string.rep(" ", _w))
  1414.         end
  1415.        
  1416.         --Hardcoded value. To change.
  1417.         local _plen = math.log10(#doc) + 1
  1418.         term.setCursorPos(_x + 1, _y + 2)
  1419.         term.write ("Print: ")
  1420.         _pX,_pY = term.getCursorPos()
  1421.         term.setBackgroundColour(colours.white)
  1422.         term.write(string.rep(" ", _plen))
  1423.         term.setBackgroundColour(_bc)
  1424.         term.write("-")
  1425.         term.setBackgroundColour(colours.white)
  1426.         term.write(string.rep(" ", _plen))
  1427.         term.setBackgroundColour(_bc)
  1428.        
  1429.         local _ctoken = "Copies: "
  1430.         local _clen,_cx,_cy = 2, term.getCursorPos()
  1431.         if _cx + #_ctoken + _clen + 2 > _x + _w - 1 then
  1432.             _cXoff = 1
  1433.             _cYoff = 2
  1434.             _ilY = _cy + 4
  1435.             term.setCursorPos(_x + _cXoff, _cy + _cYoff)
  1436.         else
  1437.             _cXoff = _cx + #_ctoken + 3
  1438.             _ilY = _cy + 2
  1439.             term.write("  ")
  1440.         end
  1441.         term.write(_ctoken)
  1442.         _cX, _cY = term.getCursorPos()
  1443.         term.setBackgroundColour(colours.white)
  1444.         term.write("  ")
  1445.        
  1446.         term.setBackgroundColour(_bc)
  1447.         term.setCursorPos(_x + 1, _ilY)
  1448.         term.write("Toner level: ")
  1449.         term.setCursorPos(_x + 1, _ilY + 1)
  1450.         term.write("Paper level: ")
  1451.         _ilX = term.getCursorPos()
  1452.        
  1453.         --Cancel button
  1454.         term.setBackgroundColour(colours.grey)
  1455.         term.setTextColour(colours.red)
  1456.         term.setCursorPos(_x + _w - 10, _y + _h - 1)
  1457.         term.write(" Cancel ")
  1458.     end
  1459.    
  1460.     local function _drawPrintOptions()
  1461.         --Numbers in the pages and copies box
  1462.         if _wsel ~= 1 and _pstart == nil then _pstart = 1 end
  1463.         if _wsel ~= 2 and _pend == nil then _pend = 1 end
  1464.         if _wsel ~= 3 and _copies == nil then _copies = 1 end
  1465.        
  1466.         term.setCursorPos(_pX, _pY)
  1467.         term.setTextColour(colours.black)
  1468.         if _wsel == 1 then term.setBackgroundColour(colours.orange)
  1469.         else term.setBackgroundColour(colours.white) end
  1470.         if _pstart == nil then term.write(string.rep(" ", math.log10(#doc) + 1))
  1471.         else term.write(string.rep(" ", math.floor(math.log10(#doc)) - math.floor(math.log10(_pstart))).._pstart) end
  1472.         if _wsel == 2 then term.setBackgroundColour(colours.orange)
  1473.         else term.setBackgroundColour(colours.white) end
  1474.         term.setCursorPos(term.getCursorPos() + 1, _pY)
  1475.         if _pend == nil then term.write(string.rep(" ", math.log10(#doc) + 1))
  1476.         else term.write(string.rep(" ", math.floor(math.log10(#doc)) - math.floor(math.log10(_pend))).._pend) end
  1477.         if _wsel == 3 then term.setBackgroundColour(colours.orange)
  1478.         else term.setBackgroundColour(colours.white) end
  1479.         term.setCursorPos(_cX, _cY)
  1480.         if _copies == nil then term.write(string.rep(" ", 2))
  1481.         else term.write(string.rep(" ", 1 - math.floor(math.log10(_copies))).._copies) end
  1482.        
  1483.         --Ink and paper levels
  1484.         local _iLvl, _pLvl = _printer.getInkLevel(), _printer.getPaperLevel()
  1485.         if not _pstart or not _pend or not _copies then _pageCount = -1
  1486.         else _pageCount = (_pend - _pstart + 1) * _copies end
  1487.         if _pageCount <= 0 then _pageCount = -1 end
  1488.         term.setBackgroundColour(_bc)
  1489.         if _iLvl < _pageCount then term.setTextColour(colours.red)
  1490.         else term.setTextColour(colours.black) end
  1491.         term.setCursorPos(_ilX, _ilY)
  1492.         term.write(_iLvl.."")
  1493.         term.setTextColour(colours.black)
  1494.         if _pageCount == -1 then term.write(" (--)")
  1495.         else term.write(" (".._pageCount..")") end
  1496.         term.write(string.rep(" ", _x + _w - term.getCursorPos()))
  1497.         term.setCursorPos(_ilX, _ilY + 1)
  1498.         if _pLvl < _pageCount then term.setTextColour(colours.red)
  1499.         else term.setTextColour(colours.black) end
  1500.         term.write(_pLvl.."")
  1501.        
  1502.         --Print/Invalid button
  1503.         term.setCursorPos(_x + 1, _y + _h - 1)
  1504.         term.setBackgroundColour(colours.grey)
  1505.         if not _pstart or not _pend or _pstart > _pend then
  1506.             term.setTextColour(colours.lightGrey)
  1507.             term.write(" Invalid ")
  1508.         else
  1509.             term.setTextColour(colours.green)
  1510.             term.write(" Print ")
  1511.             term.setBackgroundColour(colours.lightGrey)
  1512.             term.write("  ")
  1513.         end
  1514.     end
  1515.    
  1516.     local function _runInput()
  1517.         while true do
  1518.             local _id, _p1, _p2, _p3 = os.pullEvent()
  1519.             if _id == "mouse_click" then
  1520.                 if _p2 >= _pX and _p2 <= _pX + math.floor(math.log10(#doc)) and _p3 == _pY then
  1521.                     _wsel = 1
  1522.                     _pstart = nil
  1523.                     _drawPrintOptions()
  1524.                 elseif _p2 >= _pX + math.floor(math.log10(#doc)) + 2 and _p2 <= _pX + (2*math.floor(math.log10(#doc))) + 2 and
  1525.                         _p3 == _pY then
  1526.                     _pend = nil
  1527.                     _wsel = 2
  1528.                     _drawPrintOptions()
  1529.                 elseif _p2 >= _cX and _p2 <= _cX + 1 and _p3 == _cY then
  1530.                     _copies = nil
  1531.                     _wsel = 3
  1532.                     _drawPrintOptions()
  1533.                 elseif _p2 >= _x + 1 and _p2 <= _x + 8 and _p3 == _y + _h - 1 and _pstart <= _pend then
  1534.                     if _pstart == nil then _pstart = 1 end
  1535.                     if _pend == nil then _pend = #doc end
  1536.                     if _copies == nil then _copies = 1 end
  1537.                     return true
  1538.                 elseif (_p2 >= _x + _w - 9 and _p2 <= _x + _w - 1 and _p3 == _y + _h - 1) or
  1539.                         (_p2 == _x + _w - 1 and _p3 == _y) then
  1540.                     return false
  1541.                 else
  1542.                     _wsel = 0
  1543.                     _drawPrintOptions()
  1544.                 end
  1545.             elseif _id == "key" then
  1546.                 if _p1 == keys.delete or _p1 == keys.backspace then
  1547.                     if _wsel == 1 and _pstart then _pstart = math.floor(_pstart / 10) if _pstart == 0 then _pstart = nil end
  1548.                     elseif _wsel == 2 and _pend then _pend = math.floor(_pend / 10) if _pend == 0 then _pend = nil end
  1549.                     elseif _wsel == 3 and _copies then _copies = math.floor(_copies / 10) if _copies == 0 then _copies = nil end
  1550.                     end
  1551.                     _drawPrintOptions()
  1552.                 elseif _p1 == keys.tab then
  1553.                     _wsel = (_wsel % 3) + 1
  1554.                     _drawPrintOptions()
  1555.                 elseif _p1 == keys.enter and _pstart <= _pend then
  1556.                     if _pstart == nil then _pstart = 1 end
  1557.                     if _pend == nil then _pend = #doc end
  1558.                     if _copies == nil then _copies = 1 end
  1559.                     return true
  1560.                 end
  1561.             elseif _id == "char" and _wsel > 0 and tonumber(_p1) then
  1562.                 if _wsel == 1 then
  1563.                     if not _pstart then _pstart = 0 end
  1564.                     _pstart = (10 * _pstart) + tonumber(_p1)
  1565.                     if _pstart > #doc then _pstart = #doc
  1566.                     elseif _pstart == 0 then _pstart = nil end
  1567.                     _drawPrintOptions()
  1568.                 elseif _wsel == 2 then
  1569.                     if not _pend then _pend = 0 end
  1570.                     _pend = (10 * _pend) + tonumber(_p1)
  1571.                     if _pend > #doc then _pend = #doc
  1572.                     elseif _pend == 0 then _pend = nil end
  1573.                     _drawPrintOptions()
  1574.                 elseif _wsel == 3 then
  1575.                     if not _copies then _copies = 0 end
  1576.                     _copies = (10 * _copies) + tonumber(_p1)
  1577.                     if _copies > 99 then _copies = 99
  1578.                     elseif _copies == 0 then _copies = nil end
  1579.                     _drawPrintOptions()
  1580.                 end
  1581.             elseif _id == "timer" and _p1 == bTimer then saveDocument(bPath) end
  1582.         end
  1583.     end
  1584.    
  1585.     _drawWindow()
  1586.     _drawPrintOptions()
  1587.     if _runInput() then return _pstart, _pend, _copies end
  1588. end
  1589.  
  1590. --[[ A more elaborate save menu than my first effort. Directory navigation and
  1591.      selection. It works well enough.
  1592. ]]--
  1593. local function displayFileBrowser(_flag,_h,_w,_y,_x)
  1594.     local _mt = { ["-s"] = "Save As", ["-b"] = "Browse Files", ["-l"] = "Load File" }
  1595.     if not _h then _h = math.floor(h/2) else _h = math.floor(_h) end
  1596.     if not _w then _w = math.floor(w/2) else _w = math.floor(_w) end
  1597.     if not _x then _x = math.floor(w/2-_w/2) + 1 else _x = math.floor(_x) end
  1598.     if not _y then _y = math.floor(h/2-_h/2) end
  1599.     local _bc,_tc = colours.lightGrey, colours.black
  1600.     local _btc,_ttc = colours.blue, colours.white
  1601.     local _bfc,_tfc,_tdc = colours.black, colours.white, colours.green
  1602.     local _fname = sName or ""
  1603.     --This is a nasty timesaver.
  1604.     local _cpath = sPath or "/"..shell.resolve(".")
  1605.     if not _cpath:find("/") then
  1606.         _cpath = ""
  1607.     elseif sPath then
  1608.         _cpath = "/"..getPreviousDir(_cpath)
  1609.     end
  1610.     local _rlist = fs.list(_cpath)
  1611.     if _cpath ~= "/" and _cpath ~= "" then table.insert(_rlist, 1, "..") end
  1612.    
  1613.     local _scr = 0
  1614.     local _abmsg = { ["-l"] = " Open ", ["-s"] = " Save As ",
  1615.             [1] = " Invalid ", [2] = " Overwrite " }
  1616.     local _labmsg = ""
  1617.     local _winh = _h - 7
  1618.     local _cpos = 0
  1619.    
  1620.     clearAndRedraw()
  1621.     --Some dedicated internal draw functions (to speed things up)
  1622.     local function _drawWindow()
  1623.         --Permanent parts of the window
  1624.         term.setBackgroundColour(_btc)
  1625.         term.setTextColour(_ttc)
  1626.         term.setCursorPos(_x,_y)
  1627.         term.write(string.rep(" ", math.floor(_w/2) - math.floor(#_mt[_flag]/2))
  1628.                 .._mt[_flag]..string.rep(" ", math.ceil(_w/2) -
  1629.                 math.ceil(#_mt[_flag]/2)-1))
  1630.         term.setBackgroundColour(colours.red)
  1631.         term.setTextColour(colours.white)
  1632.         term.write("x")
  1633.         term.setBackgroundColour(_bc)
  1634.         term.setTextColour(_tc)
  1635.         for i=1,_h do
  1636.             term.setCursorPos(_x,_y+i)
  1637.             term.write(string.rep(" ", _w))
  1638.         end
  1639.        
  1640.         term.setBackgroundColour(colours.grey)
  1641.         term.setTextColour(colours.red)
  1642.         term.setCursorPos(_x + _w - 10, _y + _h - 1)
  1643.         term.write(" Cancel ")
  1644.     end
  1645.    
  1646.     local function _drawBrowser()
  1647.         term.setBackgroundColour(_bc)
  1648.         term.setTextColour(_tc)
  1649.         local _dpath = _cpath
  1650.         if #_dpath > _w-4 then
  1651.             while _dpath:find("/") do
  1652.                 local _ind = _dpath:find("/") + 1
  1653.                 _dpath = _dpath:sub(_ind)
  1654.                 if #_dpath < _w-7 then
  1655.                     _dpath = "...".._dpath
  1656.                     break
  1657.                 end
  1658.             end
  1659.         end
  1660.         if #_dpath > _w-4 then _dpath = "...".._dpath:sub(_w-4-#_dpath) end
  1661.         term.setCursorPos(_x+2,_y+2)
  1662.         term.write(_dpath..string.rep(" ", _w-4-#_dpath))
  1663.        
  1664.         term.setBackgroundColour(_bfc)
  1665.         for i = 1 + _scr, _winh + _scr do
  1666.             _pth = _rlist[i] or ""
  1667.             term.setCursorPos(_x + 2, _y + 2 + i - _scr)
  1668.             if fs.isDir(_cpath.."/".._pth) then
  1669.                 term.setTextColour(_tdc)
  1670.             else term.setTextColour(_tfc) end
  1671.             term.write(" ".._pth..string.rep(" ", _w - 5 - #_pth))
  1672.         end
  1673.     end
  1674.    
  1675.     local function _drawActivationButton()
  1676.         _owrite = 0
  1677.         local _val = _cpath.."/".._fname
  1678.         if (not fs.exists(_val) and _flag == "-l") or
  1679.                 (fs.exists(_val) and fs.isDir(_val)) then
  1680.             _owrite = 1
  1681.         elseif fs.exists(_val) and _flag == "-s" then _owrite = 2 end
  1682.         term.setBackgroundColour(colours.grey)
  1683.         term.setCursorPos(_x + 2, _y + _h - 1)
  1684.         if _owrite == 1 and _flag ~= "-b" then
  1685.             term.setTextColour(colours.lightGrey)
  1686.             _labmsg = " Invalid "
  1687.         elseif _owrite == 2 then
  1688.             term.setTextColour(colours.orange)
  1689.             _labmsg = " Overwrite "
  1690.         elseif _flag == "-s" then
  1691.             term.setTextColour(colours.green)
  1692.             _labmsg = " Save "
  1693.         elseif _flag == "-l" then
  1694.             term.setTextColour(colours.green)
  1695.             _labmsg = " Open "
  1696.         end
  1697.         term.write(_labmsg)
  1698.         term.setBackgroundColour(_bc)
  1699.         term.write(string.rep(" ", #" Overwrite " - #_labmsg))
  1700.     end
  1701.    
  1702.     local function _drawWriteBar()
  1703.         term.setCursorPos(_x + 2, _y + _h - 3)
  1704.         term.setTextColour(colours.black)
  1705.         term.setBackgroundColour(colours.lightGrey)
  1706.         local _pm = "Save As "
  1707.         if _flag == "-l" then _pm = "Open " end
  1708.         term.write(_pm)
  1709.        
  1710.         local _msg = _fname
  1711.         if #_msg > _w - 8 - #_pm then _msg = _msg:sub(#_msg - (_w - 8 - #_pm)) end
  1712.         term.setBackgroundColour(colours.white)
  1713.         term.write(" ".._msg)
  1714.         _cpos = term.getCursorPos()
  1715.         term.write(string.rep(" ", math.max(_w - 5 - #_pm - #_msg, 1)))
  1716.     end
  1717.    
  1718.     _drawWindow()
  1719.     _drawActivationButton()
  1720.     _drawBrowser()
  1721.     _drawWriteBar()
  1722.    
  1723.     while true do
  1724.         term.setTextColour(colours.black)
  1725.         term.setCursorPos(_cpos, _y + _h - 3)
  1726.         term.setCursorBlink(true)
  1727.         local _id,_p1,_p2,_p3 = os.pullEvent()
  1728.        
  1729.         if _id == "key" then
  1730.             if _p1 == keys.backspace and #_fname > 0 then
  1731.                 _fname = _fname:sub(1,#_fname-1)
  1732.                 _drawActivationButton()
  1733.                 _drawWriteBar()
  1734.             elseif _p1 == keys.up and _scr > 0 then
  1735.                 _scr = _scr - 1
  1736.                 _drawBrowser()
  1737.             elseif _p1 == keys.down and _scr < #_rlist - _winh then
  1738.                 _scr = _scr + 1
  1739.                 _drawBrowser()
  1740.             elseif _p1 == keys.enter then
  1741.                 local _val = _cpath.."/".._fname
  1742.                 if (_flag == "-l" and fs.exists(_val) and not fs.isDir(_val))
  1743.                         or(_flag == "-s" and not fs.isDir(_val)) then
  1744.                     break
  1745.                 end
  1746.             elseif _p1 == keys.leftCtrl or _p == keys.rightCtrl then
  1747.                 _fname = nil
  1748.                 break
  1749.             end
  1750.         elseif _id == "char" then
  1751.             _fname = _fname.._p1
  1752.             _drawActivationButton()
  1753.             _drawWriteBar()
  1754.         elseif _id == "mouse_click" then
  1755.             if _p2 == _x + _w - 1 and _p3 == _y then
  1756.                 _fname = nil
  1757.                 break
  1758.             elseif _p2 >= _x + 2 and _p2 <= _x + _w - 2 and _p3 >= _y + 3 and
  1759.                     _p3 < _y + 3 + _winh then
  1760.                 _p3 = _p3 - _y - 2
  1761.                 local _val = _rlist[_p3 + _scr]
  1762.                 if _val == ".." then
  1763.                     --Wow why is there no reverse find
  1764.                     _cpath = getPreviousDir(_cpath)
  1765.                     _rlist = fs.list(_cpath)
  1766.                     if _cpath ~= "/" and _cpath ~= "" then table.insert(_rlist, 1, "..") end
  1767.                     _scr = 0
  1768.                     _drawBrowser()
  1769.                     _drawActivationButton()
  1770.                 elseif fs.isDir(_cpath.."/".._val) then
  1771.                     _cpath = _cpath.."/".._val
  1772.                     _rlist = fs.list(_cpath)
  1773.                     if _cpath ~= "/" and  _cpath ~= "" then table.insert(_rlist, 1, "..") end
  1774.                     _scr = 0
  1775.                     _drawBrowser()
  1776.                     _drawActivationButton()
  1777.                 else
  1778.                     _fname = _val or _fname
  1779.                     _drawActivationButton()
  1780.                     _drawWriteBar()
  1781.                 end
  1782.             elseif _p3 == _y + _h - 1 and _p2 >= _x + 2 and _p2 < _x + 2 + #        _labmsg and _labmsg ~= _abmsg[1] then
  1783.                 break
  1784.             elseif _p3 == _y + _h - 1 and _p2 >= _x + _w - 2 - #" Cancel " and
  1785.                     _p2 < _x + _w - 2 then
  1786.                 _fname = nil
  1787.                 return false
  1788.             end
  1789.         elseif _id == "mouse_scroll" then
  1790.             _scr = math.min(_scr, #_rlist - _winh - 1)
  1791.             _scr = math.max(_scr + _p1, 0)
  1792.             _drawBrowser()
  1793.         elseif _id == "timer" and _p1 == bTimer then saveDocument(bPath) end
  1794.     end
  1795.     if _fname then return (_cpath.."/".._fname):sub(2),_fname else return nil end
  1796. end
  1797.  
  1798. local function menuSaveAs()
  1799.     local _sp,sn = displayFileBrowser("-s", 14, 24, 3)
  1800.     if _sp then
  1801.         sPath = _sp
  1802.         sName = _sn
  1803.         saveDocument()
  1804.     end
  1805. end
  1806.  
  1807. local function checkSave()
  1808.     if not sChange then return true end
  1809.     clearAndRedraw()
  1810.     local _val = displayConfirmDialogue("Unsaved Changes", "Your document contains unsaved changes. Are you sure you want to    quit?", { [1] = "Save"; [2] = colours.green }, { [1] = "Discard"; [2] = colors.orange; }, { [1] = "Cancel";
  1811.     [2] = colors.red; })
  1812.     if _val == 1 then
  1813.         if sPath then saveDocument()
  1814.         else menuSaveAs() end
  1815.         return true
  1816.     elseif _val == 2 then
  1817.         return true
  1818.     else
  1819.         return false
  1820.     end
  1821. end
  1822.  
  1823. --[[ Updates the list of currently available printers and other menu options
  1824.      Params: none
  1825.      Returns: none
  1826. ]]--
  1827. local function updateMenu()
  1828.     --Again, horrible. My coding sucks.
  1829.     for i=1,6 do _printers[i] = nil end
  1830.     for _,_s in pairs(rs.getSides()) do
  1831.         if peripheral.isPresent(_s) and peripheral.getType(_s) == "printer" then
  1832.             table.insert(_printers, { name = _s, enabled = true, value = _s })
  1833.         end
  1834.     end
  1835.     _mChoices[3].enabled = sChange and sPath
  1836.     _mChoices[4].enabled = sChange and sPath
  1837.     _eChoices[1].enabled = #undolist > 0
  1838.     _eChoices[2].enabled = #redolist > 0
  1839. end
  1840.  
  1841. --Performs a print with a given printer
  1842. local function performPrint(_val)
  1843.     _p = peripheral.wrap(_val)
  1844.     clearAndRedraw()
  1845.     local _ps,_pe,_cp = displayPrint(_p, math.ceil(w/2), 8)
  1846.     if not _ps then
  1847.         lscroll = true
  1848.         return
  1849.     end
  1850.     clearAndRedraw()
  1851.     for c=1,_cp do
  1852.         for i=_ps,_pe do
  1853.             while _p.getPaperLevel() == 0 do
  1854.                 displayConfirmDialogue("Page "..i..": Check paper levels", "The paper tray is currently empty. Fill with paper and click here to continue.")
  1855.             end
  1856.             while _p.getInkLevel() == 0 do
  1857.                 displayConfirmDialogue("Page "..i..": Check ink levels", "The printer is out of ink. Insert more ink, and click here to continue.")
  1858.             end
  1859.             --There is no out-tray check... that I can tell at least.
  1860.             while not _p:newPage() do
  1861.                 displayConfirmDialogue("Page "..i..": Check out tray", "The out tray may be full. Remove all pages and click here to continue.")
  1862.             end
  1863.             _p.setPageTitle((sName or "").." page "..i)
  1864.             printPage(i, _p)
  1865.             _p:endPage()
  1866.         end
  1867.         if c<_cp then displayConfirmDialogue("Copy "..c.." Complete!", "Clear out tray and click here to continue") end
  1868.     end
  1869.    
  1870.     clearAndRedraw()
  1871.     displayConfirmDialogue("Print Complete!", "Printed ".._cp.." copies (total "..#doc.." page(s).)")
  1872. end
  1873.  
  1874. --Performs the menu
  1875. local function performMenu(_menu)
  1876.     updateMenu()
  1877.     local _val = displayDropDown(_menu.x, _menu.y, _menu)
  1878.    
  1879.     if peripheral.getType(_val) == "printer" then performPrint(_val) end
  1880.     lscroll = true
  1881. end
  1882.  
  1883. --[[ A series of functions binded to keyboard shortcuts
  1884.      Parameters: none
  1885.      Returns: none
  1886. ]]--
  1887. local bindings = {
  1888.     --Moves us up. Hitting the top moves the cursor to the start of the line.
  1889.     --Most editors do this. I don't know why. Convention?
  1890.     [keys.up] = function()
  1891.         if currline > 1 then currline = currline - 1
  1892.         elseif currpage > 1 then
  1893.             currpage = currpage - 1
  1894.             currline = #doc[currpage]
  1895.             lscroll = true
  1896.         else
  1897.             currind = 1
  1898.             return
  1899.         end
  1900.         if currind > #doc[currpage][currline] then
  1901.             currind = math.max(1, #doc[currpage][currline] + 1)
  1902.         end
  1903.         fixCursorPos()
  1904.         checkYScroll()
  1905.     end,
  1906.     --Moves us down. Hitting the bottom moves cursor to end of the line.
  1907.     [keys.down] = function()
  1908.         if currline < #doc[currpage] then currline = currline + 1
  1909.         elseif currpage < #doc then
  1910.             currpage = currpage + 1
  1911.             currline = 1
  1912.             lscroll = true
  1913.         else
  1914.             currind = #doc[currpage][currline] + 1
  1915.             return
  1916.         end
  1917.         if currind > #doc[currpage][currline] then
  1918.             currind = math.max(1, #doc[currpage][currline] + 1)
  1919.         end
  1920.         fixCursorPos()
  1921.         checkYScroll()
  1922.     end,
  1923.     --Moves us left. Hitting and edge moves to the end of the last line.
  1924.     [keys.left] = function()
  1925.         if currind <= 1 then
  1926.             if currline > 1 then
  1927.                 currline = currline - 1
  1928.                 currind = #doc[currpage][currline] + 1
  1929.             elseif currpage > 1 then
  1930.                 currpage = currpage - 1
  1931.                 currline = #doc[currpage]
  1932.                 currind = #doc[currpage][currline] + 1
  1933.             end
  1934.         else
  1935.             currind = currind - 1
  1936.         end
  1937.         fixCursorPos()
  1938.         checkYScroll()
  1939.     end,
  1940.     --Moves us right. Hitting the edge moves to the start of the next line.
  1941.     [keys.right] = function()
  1942.         local _c = doc[currpage][currline]:sub(currind)
  1943.         if currind >= #doc[currpage][currline] + 1 or _c == "\n" or _c == "\f" then
  1944.             if currline < #doc[currpage] then
  1945.                 currline = currline + 1
  1946.                 currind = 1
  1947.             elseif currpage < #doc then
  1948.                 currpage = currpage + 1
  1949.                 currline = 1
  1950.                 currind = 1
  1951.             end
  1952.         else
  1953.             currind = currind + 1
  1954.         end
  1955.         fixCursorPos()
  1956.         checkYScroll()
  1957.     end,
  1958.     --[[ Adds a newline character to the end of the line, and pushes everything down ]]
  1959.     [keys.enter] = function()
  1960.         addUndoableAction(currind, ACTIONS.ADDBREAK, "\n")
  1961.         insertLineBreak("\n")
  1962.     end,
  1963.     --Some more simple cursor manipulation keys
  1964.     [keys.home] = function() currind = 1 end,
  1965.     [keys["end"]] = function()
  1966.         currind = #doc[currpage][currline] + 1
  1967.         fixCursorPos()
  1968.     end,
  1969.     [keys.pageDown] = function()
  1970.         if currpage < #doc then
  1971.             currpage = currpage + 1
  1972.             currline = 1 currind = 1
  1973.             checkYScroll()
  1974.         end
  1975.         lscroll = true
  1976.     end,
  1977.     [keys.pageUp] = function()
  1978.         if currpage > 1 then
  1979.             currpage = currpage - 1
  1980.             currline = 1 currind = 1
  1981.             checkYScroll()
  1982.         end
  1983.         lscroll = true
  1984.     end,
  1985.     -- Some other core functions in the program
  1986.     [keys.backspace] = function ()
  1987.         local _val = deleteCharacter()
  1988.         addUndoableAction(currind, ACTIONS.DELETECHAR, _val)
  1989.     end;
  1990.     [keys.leftCtrl] = function() performMenu(_mChoices) end,
  1991.     [keys.grave] = function() testMode = not testMode end,
  1992.     [keys.tab] = function()
  1993.         local _len = #doc[currpage][currline]
  1994.         local _add = math.min(4 - (_len % 4), linelen - currind + 1)
  1995.         for i=1,_add do addCharacter(" ") end
  1996.     end
  1997. }
  1998.  
  1999.  
  2000. --[[ Performs actions based on keyboard and mouse input
  2001.      Parameters: none
  2002.      Returns: none
  2003. ]]--
  2004. local function handleInput()
  2005.     local _,_cpy = term.getCursorPos()
  2006.     if _cpy > 1 then term.setCursorBlink(true) end
  2007.     local _id,_p1,_p2,_p3 = os.pullEvent()
  2008.     term.setCursorBlink(false)
  2009.     local _oUndoLen,_oRedoLen = #undolist,#redolist
  2010.     if _id == "char" then
  2011.         addCharacter(_p1)
  2012.         addUndoableAction(currind - 1, ACTIONS.ADDCHAR, _p1)
  2013.     elseif _id == "key" then
  2014.         if bindings[_p1] then bindings[_p1]()
  2015.         elseif _p1 == keys.leftCtrl then finished = true end
  2016.     elseif _id == "mouse_scroll" then
  2017.         yscroll = yscroll - _p1
  2018.         if yscroll > 4 - _yoff then
  2019.             if currpage > 1 then
  2020.                 yscroll = h - pagelen - _yoff - 3
  2021.                 currpage = currpage - 1
  2022.                 currline = 1
  2023.                 currind = 1
  2024.             else yscroll = 4 - _yoff end
  2025.         elseif yscroll < h - pagelen - _yoff - 3 then
  2026.             if currpage < #doc then
  2027.                 yscroll = 4 - _yoff
  2028.                 currpage = currpage + 1
  2029.                 currline = 1
  2030.                 currind = 1
  2031.             else yscroll = h - pagelen - _yoff - 3 end
  2032.         end
  2033.         lscroll = true
  2034.     elseif _id == "mouse_click" then
  2035.         if _p2 >= _mChoices.x and _p2 <= _mChoices.x + #_mChoices.name
  2036.                 and _p3 == _mChoices.y then
  2037.             performMenu(_mChoices)
  2038.         elseif _p2 >= _eChoices.x and _p2 <= _eChoices.x + #_eChoices.name
  2039.                 and _p3 == _eChoices.y then
  2040.             performMenu(_eChoices)
  2041.         else
  2042.             currline = math.max(_p3 - _yoff - yscroll, 1)
  2043.             currline = math.min(currline, #doc[currpage])
  2044.             local _ws = getWhitespaceLength(currpage, currline)
  2045.             currind = math.max(_p2 - _xoff - _ws, 1)
  2046.             currind = math.min(currind, #doc[currpage][currline] + 1)
  2047.             if currind > #doc[currpage][currline] and hasBreakEnd(doc[currpage][currline])
  2048.                 then currind = currind - 1 end
  2049.             checkYScroll()
  2050.         end
  2051.     elseif _id == "timer" then
  2052.         if _p1 == bTimer then
  2053.             saveDocument(bPath)
  2054.         elseif _p1 == actionTimer then
  2055.             actionTimer = nil
  2056.         end
  2057.     end
  2058.     --Clearing the redo list
  2059.     if _oUndoLen < #undolist and _oRedoLen == #redolist then redolist = {}  end
  2060.     testWait()
  2061. end
  2062.  
  2063. --Menu initialization
  2064. table.insert(_aChoices, { name = "Left", key = "L", enabled = true, value = function()
  2065.         testAlign = ALIGNLEFT end})
  2066. table.insert(_aChoices, { name = "Center", key = "C", enabled = true, value = function()
  2067.         testAlign = ALIGNCENTER end})
  2068. table.insert(_aChoices, { name = "Right", key = "R", enabled = true, value = function()
  2069.         testAlign = ALIGNRIGHT end})
  2070. table.insert(_iChoices, { name = "Page Break", key = "P", enabled = true, value = function()
  2071.         addUndoableAction(currind, ACTIONS.ADDBREAK, "\f")
  2072.         insertLineBreak("\f")
  2073.     end })
  2074. table.insert(_iChoices, { name = "Section Break", key = "S", enabled = false,
  2075.     value = function() testPrint("unimplemented") end })
  2076. table.insert(_mChoices, { name = "New File", key = "N", enabled = true, value = function()
  2077.     if checkSave() then constructNewDocument() end end })
  2078. table.insert(_mChoices, { name = "Open", key = "O", enabled = true, value = function()
  2079.     if checkSave() then
  2080.         local _sp,_sn = displayFileBrowser("-l", 14, 24, 3)
  2081.         if _sp then
  2082.             loadDocument(_sp)
  2083.             sPath = _sp
  2084.             sName = _sn
  2085.     end end end })
  2086. table.insert(_mChoices, { name = "Revert", key = "R", enabled = false, value = function()
  2087.     loadDocument(sPath)
  2088. end })
  2089. table.insert(_mChoices, { name = "Save", key = "S", enabled = false, value = saveDocument })
  2090. table.insert(_mChoices, { name = "Save As", key = "A", enabled = true, value = menuSaveAs })
  2091. table.insert(_mChoices, { name = "Print", key = "P", enabled = false, value = _printers })
  2092. table.insert(_mChoices, { name = "--------", enabled = false, value = nil })
  2093. table.insert(_mChoices, { name = "About", key = "b", enabled = true, value = displayAbout })
  2094. table.insert(_mChoices, { name = "Quit", key = "Q", enabled = true, value = function()
  2095.     if checkSave() then finished = true end end })
  2096. table.insert(_eChoices, { name = "Undo", key = "z", enabled = false, value = function()
  2097.     undoAction(undolist, redolist) end })
  2098. table.insert(_eChoices, { name = "Redo", key = "y", enabled = false, value = function()
  2099.     undoAction(redolist, undolist) end })
  2100. --table.insert(_eChoices, { name = "Cut", key = "x", enabled = false, value = function()
  2101. --  testPrint("unimplemented") end })
  2102. --table.insert(_eChoices, { name = "Copy", key = "c", enabled = false, value = function()
  2103. --  testPrint("unimplemented") end })
  2104. --table.insert(_eChoices, { name = "Paste", key = "v", enabled = false, value = function()
  2105. --  testPrint("unimplemented") end })
  2106. table.insert(_eChoices, { name = "-------", enabled = false, value = nil })
  2107. table.insert(_eChoices, { name = "Insert", key = "I", enabled = true, value = _iChoices })
  2108. table.insert(_eChoices, { name = "Align", key = "A", enabled = true, value = _aChoices })
  2109.  
  2110. if not term.isColour() then
  2111.     print("Currently, WYSIwrite is only supported on advanced computers")
  2112. end
  2113.  
  2114. local _resp = 0
  2115. --If there is a backup, we ask if this is to be restored
  2116. if (fs.exists(bPath)) then
  2117.     term.setBackgroundColour(colours.lightBlue)
  2118.     shell.run("clear")
  2119.     _resp = displayConfirmDialogue("Restore last document?", [[WYSIwrite unexpectedly closed last session. Would you like to restore your last document?]],
  2120.     {"Restore", colours.green}, {"Discard", colours.red})
  2121.     if _resp == 1 then
  2122.         loadDocument(bPath)
  2123.     end
  2124.     fs.delete(bPath)
  2125. end
  2126.  
  2127. local _tArgs = {...}
  2128. if _tArgs[1] and _resp ~= 1 then
  2129.     sName = _tArgs[1]:gsub("\\","/")
  2130.     sPath = shell.resolve(sName)
  2131.     local _pcts = split(sName, "/")
  2132.     sName = _pcts[#_pcts]
  2133.     if sName == ".writebackup" then
  2134.         print("Unsafe to construct/overwrite backup with this application. If file exists, it can be restored by running write without a path.")
  2135.         return
  2136.     end
  2137.     if fs.exists(sPath) then loadDocument(sPath)
  2138.     else constructNewDocument(sName, sPath) end
  2139. elseif _resp ~= 1 then constructNewDocument() end
  2140.  
  2141. checkYScroll()
  2142. while not finished do
  2143.     --Basic printing stuff
  2144.     if lscroll then
  2145.         displayClearPage()
  2146.         lscroll = false
  2147.     end
  2148.     term.setBackgroundColour(colours.white)
  2149.     term.setTextColour(colours.black)
  2150.     printPage(currpage, term, _xoff + 1, _yoff + yscroll)
  2151.     displayInterface()
  2152.     --Repositioning the cursor
  2153.     term.setTextColour(colours.black)
  2154.     term.setCursorPos(currind + _xoff + getWhitespaceLength(currpage,currline),
  2155.             currline + _yoff + yscroll)
  2156.    
  2157.     handleInput()
  2158. end
  2159. --remove backup (this was a safe quit)
  2160. fs.delete(bPath)
  2161. term.setBackgroundColour(colours.black)
  2162. shell.run("clear")
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement