Advertisement
MudkipTheEpic

MudDev [WIP]

May 4th, 2013
206
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
text 21.40 KB | None | 0 0
  1. --[[
  2. MudDev: An extension of shell made to appeal to developers.
  3. Credit to Espen for his epic tab completing read.
  4. Credit to the ComputerCraft team for making the original shell.
  5. ]]
  6. if not term.isColor then printError("Cannot run without color support.") error("",0) end
  7. local mX,mY=term.getSize()
  8. local function centerPrint(text, ny, tC, bC)
  9. if tC then term.setTextColor(tC) end
  10. if bC then term.setBackgroundColor(bC) end
  11. if type(text) == "table" then for _, v in pairs(text) do centerPrint(v) end
  12. else
  13. local x, y = term.getCursorPos()
  14. local w, h = term.getSize()
  15. term.setCursorPos(w/2 - text:len()/2 + (#text % 2 == 0 and 1 or 0), ny or y)
  16. print(text)
  17. end
  18. end
  19.  
  20. local ideas={
  21. firstword={
  22. "Rednet",
  23. "Turtle",
  24. "Bacon",
  25. "File",
  26. "Working",
  27. "Advanced",
  28. "GUI",
  29. },
  30. secondword={
  31. "Bootloader",
  32. "Packaging",
  33. "Cheesecake",
  34. "Encryption",
  35. "Sandbox",
  36. "Anti-Virus",
  37. "Role-Playing",
  38. },
  39. thirdword={
  40. "OS",
  41. "API",
  42. "Game",
  43. "Browser",
  44. "Program",
  45. "Derpcake",
  46. "Dev Tool",
  47. },
  48. }
  49.  
  50. local fw=ideas.firstword
  51. local sw=ideas.secondword
  52. local tw=ideas.thirdword
  53.  
  54. local function getIdea()
  55. local promptColour, textColour, bgColour
  56. if term.isColour() then
  57. promptColour = colours.orange
  58. textColour = colours.lightBlue
  59. bgColour = colours.black
  60. else
  61. promptColour = colours.white
  62. textColour = colours.white
  63. bgColour = colours.black
  64. end
  65. while true do
  66. term.setBackgroundColor(bgColour)
  67. term.clear()
  68. paintutils.drawLine(1, 1, 1, mY, promptColour)
  69. paintutils.drawLine(1, 1, mX, 1, promptColour)
  70. paintutils.drawLine(1, mY, mX, mY, promptColour)
  71. paintutils.drawLine(mX, 1, mX, mY, promptColour)
  72. term.setTextColor(promptColour)
  73. term.setBackgroundColor(bgColour)
  74. centerPrint("Your program idea is....",math.floor(mY/2)-1)
  75. centerPrint(fw[math.random(#fw)].." "..sw[math.random(#sw)].." "..tw[math.random(#tw)],math.floor(mY/2))
  76. centerPrint("Press Space to exit",mY-2)
  77. centerPrint("Press any other key to generate a new idea.",mY-4)
  78. local e,key=os.pullEventRaw("key")
  79. if key==keys.space then return end
  80. end
  81. end
  82.  
  83.  
  84. local function drawMenu()
  85. local promptColour, textColour, bgColour
  86. if term.isColour() then
  87. promptColour = colours.orange
  88. textColour = colours.lightBlue
  89. bgColour = colours.black
  90. else
  91. promptColour = colours.white
  92. textColour = colours.white
  93. bgColour = colours.black
  94. end
  95.  
  96. local promptColor=promptColour
  97. local textColor=textColour
  98. local bgColor=bgColour
  99. local mX,mY=term.getSize()
  100. term.setBackgroundColor(bgColor)
  101. term.clear()
  102. paintutils.drawLine(1, 1, 1, mY, promptColour)
  103. paintutils.drawLine(1, 1, mX, 1, promptColour)
  104. paintutils.drawLine(1, mY, mX, mY, promptColour)
  105. paintutils.drawLine(mX, 1, mX, mY, promptColour)
  106. centerPrint("MudDev Controls",2,promptColor,bgColor)
  107. centerPrint("TAB: Toggle Tab-Completion",4)
  108. centerPrint("R: Reboot",6)
  109. centerPrint("S: Shutdown",8)
  110. centerPrint("I: Program Ideas",10)
  111. centerPrint("ALT: Exit Menu",11)
  112. end
  113.  
  114. local cTable={
  115. ["tab"]=function() tabAuto=not tabAuto print("test") end,
  116. ["r"]=function() os.reboot() end,
  117. ["s"]=function() os.shutdown() end,
  118. ["i"]=function() getIdea() drawMenu() end,
  119. }
  120.  
  121. local tabAuto=true
  122. local parentShell = shell
  123.  
  124. local bExit = false
  125. local sDir = (parentShell and parentShell.dir()) or ""
  126. local sPath = (parentShell and parentShell.path()) or ".:/rom/programs"
  127. local tAliases = (parentShell and parentShell.aliases()) or {}
  128. local tProgramStack = {}
  129. local shell = {}
  130. local tEnv = {
  131. ["shell"] = shell,
  132. }
  133. local mX,mY=term.getSize()
  134.  
  135.  
  136. -- Install shell API
  137. function shell.run( ... )
  138. return runLine( table.concat( { ... }, " " ) )
  139. end
  140.  
  141. function shell.exit()
  142. bExit = true
  143. end
  144.  
  145. function shell.dir()
  146. return sDir
  147. end
  148.  
  149. function shell.setDir( _sDir )
  150. sDir = _sDir
  151. end
  152.  
  153. function shell.path()
  154. return sPath
  155. end
  156.  
  157. function shell.setPath( _sPath )
  158. sPath = _sPath
  159. end
  160.  
  161. function shell.resolve( _sPath )
  162. local sStartChar = string.sub( _sPath, 1, 1 )
  163. if sStartChar == "/" or sStartChar == "\\" then
  164. return fs.combine( "", _sPath )
  165. else
  166. return fs.combine( sDir, _sPath )
  167. end
  168. end
  169.  
  170. function shell.resolveProgram( _sCommand )
  171. -- Substitute aliases firsts
  172. if tAliases[ _sCommand ] ~= nil then
  173. _sCommand = tAliases[ _sCommand ]
  174. end
  175.  
  176. -- If the path is a global path, use it directly
  177. local sStartChar = string.sub( _sCommand, 1, 1 )
  178. if sStartChar == "/" or sStartChar == "\\" then
  179. local sPath = fs.combine( "", _sCommand )
  180. if fs.exists( sPath ) and not fs.isDir( sPath ) then
  181. return sPath
  182. end
  183. return nil
  184. end
  185.  
  186. -- Otherwise, look on the path variable
  187. for sPath in string.gmatch(sPath, "[^:]+") do
  188. sPath = fs.combine( shell.resolve( sPath ), _sCommand )
  189. if fs.exists( sPath ) and not fs.isDir( sPath ) then
  190. return sPath
  191. end
  192. end
  193.  
  194. -- Not found
  195. return nil
  196. end
  197.  
  198. function shell.programs( _bIncludeHidden )
  199. local tItems = {}
  200.  
  201. -- Add programs from the path
  202. for sPath in string.gmatch(sPath, "[^:]+") do
  203. sPath = shell.resolve( sPath )
  204. if fs.isDir( sPath ) then
  205. local tList = fs.list( sPath )
  206. for n,sFile in pairs( tList ) do
  207. if not fs.isDir( fs.combine( sPath, sFile ) ) and
  208. (_bIncludeHidden or string.sub( sFile, 1, 1 ) ~= ".") then
  209. tItems[ sFile ] = true
  210. end
  211. end
  212. end
  213. end
  214.  
  215. -- Sort and return
  216. local tItemList = {}
  217. for sItem, b in pairs( tItems ) do
  218. table.insert( tItemList, sItem )
  219. end
  220. table.sort( tItemList )
  221. return tItemList
  222. end
  223.  
  224. function shell.getRunningProgram()
  225. if #tProgramStack > 0 then
  226. return tProgramStack[#tProgramStack]
  227. end
  228. return nil
  229. end
  230.  
  231. function shell.setAlias( _sCommand, _sProgram )
  232. tAliases[ _sCommand ] = _sProgram
  233. end
  234.  
  235. function shell.clearAlias( _sCommand )
  236. tAliases[ _sCommand ] = nil
  237. end
  238.  
  239. function shell.aliases()
  240. -- Add aliases
  241. local tCopy = {}
  242. for sAlias, sCommand in pairs( tAliases ) do
  243. tCopy[sAlias] = sCommand
  244. end
  245. return tCopy
  246. end
  247.  
  248. local function nRead( _sReplaceChar, _tHistory )
  249. term.setCursorBlink( true )
  250.  
  251. local sLine = ""
  252. local nHistoryPos = nil
  253. local nPos = 0
  254. if _sReplaceChar then
  255. _sReplaceChar = string.sub( _sReplaceChar, 1, 1 )
  256. end
  257.  
  258. local w, h = term.getSize()
  259. local sx, sy = term.getCursorPos()
  260.  
  261. local function redraw( _sCustomReplaceChar )
  262. local nScroll = 0
  263. if sx + nPos >= w then
  264. nScroll = (sx + nPos) - w
  265. end
  266.  
  267. term.setCursorPos( sx, sy )
  268. local sReplace = _sCustomReplaceChar or _sReplaceChar
  269. if sReplace then
  270. term.write( string.rep(sReplace, string.len(sLine) - nScroll) )
  271. else
  272. term.write( string.sub( sLine, nScroll + 1 ) )
  273. end
  274. term.setCursorPos( sx + nPos - nScroll, sy )
  275. end
  276.  
  277. while true do
  278. local sEvent, param = os.pullEvent()
  279. if sEvent == "char" then
  280. sLine = string.sub( sLine, 1, nPos ) .. param .. string.sub( sLine, nPos + 1 )
  281. nPos = nPos + 1
  282. redraw()
  283.  
  284. elseif sEvent == "key" then
  285. if param == keys.enter then
  286. -- Enter
  287. break
  288.  
  289. elseif param == keys.left then
  290. -- Left
  291. if nPos > 0 then
  292. nPos = nPos - 1
  293. redraw()
  294. end
  295.  
  296. elseif param == keys.right then
  297. -- Right
  298. if nPos < string.len(sLine) then
  299. nPos = nPos + 1
  300. redraw()
  301. end
  302.  
  303. elseif param == keys.up or param == keys.down then
  304. -- Up or down
  305. if _tHistory then
  306. redraw(" ");
  307. if param == keys.up then
  308. -- Up
  309. if nHistoryPos == nil then
  310. if #_tHistory > 0 then
  311. nHistoryPos = #_tHistory
  312. end
  313. elseif nHistoryPos > 1 then
  314. nHistoryPos = nHistoryPos - 1
  315. end
  316. else
  317. -- Down
  318. if nHistoryPos == #_tHistory then
  319. nHistoryPos = nil
  320. elseif nHistoryPos ~= nil then
  321. nHistoryPos = nHistoryPos + 1
  322. end
  323. end
  324.  
  325. if nHistoryPos then
  326. sLine = _tHistory[nHistoryPos]
  327. nPos = string.len( sLine )
  328. else
  329. sLine = ""
  330. nPos = 0
  331. end
  332. redraw()
  333. end
  334. elseif param == keys.backspace then
  335. -- Backspace
  336. if nPos > 0 then
  337. redraw(" ");
  338. sLine = string.sub( sLine, 1, nPos - 1 ) .. string.sub( sLine, nPos + 1 )
  339. nPos = nPos - 1
  340. redraw()
  341. end
  342. elseif param == keys.home then
  343. -- Home
  344. nPos = 0
  345. redraw()
  346. elseif param == keys.delete then
  347. if nPos < string.len(sLine) then
  348. redraw(" ");
  349. sLine = string.sub( sLine, 1, nPos ) .. string.sub( sLine, nPos + 2 )
  350. redraw()
  351. end
  352. elseif param == keys["end"] then
  353. -- End
  354. nPos = string.len(sLine)
  355. redraw()
  356. elseif (param == keys["leftAlt"]) or (param == keys["rightAlt"]) then
  357. return 1337
  358. end
  359. end
  360. end
  361.  
  362. term.setCursorBlink( false )
  363. term.setCursorPos( w + 1, sy )
  364. print()
  365.  
  366. return sLine
  367. end
  368.  
  369. --[[
  370. Modified read() with Tab-Completion v1.2 by Espen
  371. Original from shell-file of ComputerCraft v1.46
  372.  
  373. Changelog:
  374. v1.2 - Added a separate program to to allow runtime-injection of the custom read() into the running shell of an individual computer.
  375. Thanks @BigSHinyToys for making me aware of the global table's dropped protection. ^_^
  376. v1.12 - Fixed: Pressing Tab without entering anything threw an error.
  377. v1.11 - Removed previously introduced, but now unnecessary parameter from read()
  378. v1.1 - Path traversal implemented
  379. v1.0 - Initial release
  380. --]]
  381.  
  382. local function mRead( _sReplaceChar, _tHistory )
  383. term.setCursorBlink( true )
  384.  
  385. local sLine = ""
  386. local tMatches = { pointer = 0, get = function( self ) return self[ self.pointer ] end } -- tMatches:get() == tMatches[ tMatches.pointer ]
  387. local nHistoryPos = nil
  388. local nPos = 0
  389. if _sReplaceChar then
  390. _sReplaceChar = string.sub( _sReplaceChar, 1, 1 )
  391. end
  392.  
  393. local w, h = term.getSize()
  394. local sx, sy = term.getCursorPos()
  395.  
  396. local function redraw( _sCustomReplaceChar )
  397. local nScroll = 0
  398. if sx + nPos >= w then
  399. nScroll = (sx + nPos) - w
  400. end
  401.  
  402. term.setCursorPos( sx, sy )
  403. local sReplace = _sCustomReplaceChar or _sReplaceChar
  404. if sReplace then
  405. term.write( string.rep(sReplace, string.len(sLine) - nScroll) )
  406. else
  407. term.write( string.sub( sLine, nScroll + 1 ) )
  408. end
  409. term.setCursorPos( sx + nPos - nScroll, sy )
  410. end
  411.  
  412. while true do
  413. local sEvent, param = os.pullEvent()
  414. if sEvent == "char" then
  415. tMatches = { pointer = 0, get = function( self ) return self[ self.pointer ] end } -- Reset completion match-table.
  416. sLine = string.sub( sLine, 1, nPos ) .. param .. string.sub( sLine, nPos + 1 )
  417. nPos = nPos + 1
  418. redraw()
  419.  
  420. elseif sEvent == "key" then
  421. if not (param == keys.tab) then
  422. tMatches = { pointer = 0, get = function( self ) return self[ self.pointer ] end } -- Reset completion match-table.
  423. end
  424. if param == keys.enter then
  425. -- Enter
  426. break
  427.  
  428. elseif param == keys.tab then
  429. -- Tab
  430. if #tMatches > 0 then -- tab was pressed before.
  431. -- [[ We already have matches, show the next one ]]
  432. local nLastMatchSize = string.len( tMatches:get() )
  433.  
  434. -- Shift pointer to next match.
  435. tMatches.pointer = tMatches.pointer + 1
  436. if tMatches.pointer > #tMatches then tMatches.pointer = 1 end -- Wrap around if the pointer has gone past the end.
  437.  
  438. -- Clear the line if the new match is smaller than the previous.
  439. if string.len(tMatches:get()) < nLastMatchSize then redraw(" ") end
  440. -- Assemble the new line.
  441. sLine = string.sub( sLine, 1, nPos - nLastMatchSize ) .. tMatches:get()
  442. nPos = string.len(sLine)
  443. redraw()
  444. else
  445. -- [[ No matches yet, look for some now. ]]
  446. local sLastPar = string.match( sLine, "[^%s]+$" ) or "" -- Get last entered parameter.
  447. local sToMatch = string.match( sLastPar, "[^%s/\.]+$" ) or "" -- Get filename-part of sLastPar.
  448.  
  449. -- Only look for matches if there was something afters the last slash.
  450. if sToMatch and sToMatch ~= "" then
  451. -- Get absolute path of the entered location.
  452. local sAbsPath = shell.resolve( sLastPar )
  453. -- Cut off filename, e.g. /some/test/path -> /some/test/
  454. if not fs.isDir( sAbsPath ) then
  455. sAbsPath = string.match( sAbsPath, "^([^%s]+[/\])[^%s/\]+[/\]?$" )
  456. end
  457. if not sAbsPath then sAbsPath = shell.resolve( "/" ) end
  458.  
  459. -- Search for matches in the resolved folder.
  460. local ok, tFileList = pcall( fs.list, sAbsPath )
  461. if ok then
  462. local match = nil
  463. -- Populate table with all matches.
  464. for k, v in ipairs(tFileList) do
  465. match = string.match( v, "^"..sToMatch..".*$" )
  466. if match then
  467. table.insert( tMatches, match )
  468. end
  469. end
  470. end
  471.  
  472. -- Show first match.
  473. if #tMatches > 0 then
  474. tMatches.pointer = 1
  475. local partialMatch = string.gsub( tMatches:get(), "^"..sToMatch, "" )
  476. sLine = sLine..partialMatch
  477. nPos = nPos + string.len( partialMatch )
  478. redraw()
  479. end
  480. end
  481. end
  482.  
  483. elseif param == keys.left then
  484. -- Left
  485. if nPos > 0 then
  486. nPos = nPos - 1
  487. redraw()
  488. end
  489.  
  490. elseif param == keys.right then
  491. -- Right
  492. if nPos < string.len(sLine) then
  493. nPos = nPos + 1
  494. redraw()
  495. end
  496.  
  497. elseif param == keys.up or param == keys.down then
  498. -- Up or down
  499. if _tHistory then
  500. redraw(" ");
  501. if param == keys.up then
  502. -- Up
  503. if nHistoryPos == nil then
  504. if #_tHistory > 0 then
  505. nHistoryPos = #_tHistory
  506. end
  507. elseif nHistoryPos > 1 then
  508. nHistoryPos = nHistoryPos - 1
  509. end
  510. else
  511. -- Down
  512. if nHistoryPos == #_tHistory then
  513. nHistoryPos = nil
  514. elseif nHistoryPos ~= nil then
  515. nHistoryPos = nHistoryPos + 1
  516. end
  517. end
  518.  
  519. if nHistoryPos then
  520. sLine = _tHistory[nHistoryPos]
  521. nPos = string.len( sLine )
  522. else
  523. sLine = ""
  524. nPos = 0
  525. end
  526. redraw()
  527. end
  528. elseif param == keys.backspace then
  529. -- Backspace
  530. if nPos > 0 then
  531. redraw(" ");
  532. sLine = string.sub( sLine, 1, nPos - 1 ) .. string.sub( sLine, nPos + 1 )
  533. nPos = nPos - 1
  534. redraw()
  535. end
  536. elseif param == keys.home then
  537. -- Home
  538. nPos = 0
  539. redraw()
  540. elseif param == keys.delete then
  541. if nPos < string.len(sLine) then
  542. redraw(" ");
  543. sLine = string.sub( sLine, 1, nPos ) .. string.sub( sLine, nPos + 2 )
  544. redraw()
  545. end
  546. elseif param == keys["end"] then
  547. -- End
  548. nPos = string.len(sLine)
  549. redraw()
  550. elseif param==keys.leftAlt or param==keys.rightAlt then
  551. return 1337
  552. end
  553. end
  554. end
  555.  
  556. term.setCursorBlink( false )
  557. term.setCursorPos( w + 1, sy )
  558. print()
  559.  
  560. return sLine
  561. end
  562.  
  563. -- Colours
  564. local promptColour, textColour, bgColour
  565. if term.isColour() then
  566. promptColour = colours.orange
  567. textColour = colours.lightBlue
  568. bgColour = colours.black
  569. else
  570. promptColour = colours.white
  571. textColour = colours.white
  572. bgColour = colours.black
  573. end
  574.  
  575. local promptColor=promptColour
  576. local textColor=textColour
  577. local bgColor=bgColour
  578.  
  579.  
  580. local function run( _sCommand, ... )
  581. local sPath = shell.resolveProgram( _sCommand )
  582. if sPath ~= nil then
  583. tProgramStack[#tProgramStack + 1] = sPath
  584. local result = os.run( tEnv, sPath, ... )
  585. tProgramStack[#tProgramStack] = nil
  586. return result
  587. else
  588. printError( "No such program" )
  589. return false
  590. end
  591. end
  592.  
  593. local function runLine( _sLine )
  594. local tWords = {}
  595. for match in string.gmatch( _sLine, "[^ \t]+" ) do
  596. table.insert( tWords, match )
  597. end
  598.  
  599. local sCommand = tWords[1]
  600. if sCommand then
  601. return run( sCommand, unpack( tWords, 2 ) )
  602. end
  603. return false
  604. end
  605.  
  606. term.setBackgroundColor( bgColour )
  607. term.setTextColour( promptColour )
  608. print( "MudDev: "..os.version() )
  609. term.setTextColour( textColour )
  610.  
  611. -- If this is the toplevel shell, run the startup programs
  612. if parentShell == nil then
  613. -- Run the startup from the ROM first
  614. local sRomStartup = shell.resolveProgram( "/rom/startup" )
  615. if sRomStartup then
  616. shell.run( sRomStartup )
  617. end
  618.  
  619. -- Then run the user created startup, from the disks or the root
  620. local sUserStartup = shell.resolveProgram( "/startup" )
  621. for n,sSide in pairs( peripheral.getNames() ) do
  622. --for n,sSide in pairs( redstone.getSides() ) do
  623. if disk.isPresent( sSide ) and disk.hasData( sSide ) then
  624. local sDiskStartup = shell.resolveProgram( fs.combine(disk.getMountPath( sSide ), "startup") )
  625. if sDiskStartup then
  626. sUserStartup = sDiskStartup
  627. break
  628. end
  629. end
  630. end
  631.  
  632. if sUserStartup then
  633. shell.run( sUserStartup )
  634. end
  635. end
  636.  
  637. -- Run any programs passed in as arguments
  638. local tArgs = { ... }
  639. if #tArgs > 0 then
  640. shell.run( ... )
  641. end
  642.  
  643. -- Read commands and execute them
  644. local tCommandHistory = {}
  645. while not bExit do
  646. local sRead=tabAuto and mRead or nRead
  647. term.setBackgroundColor( bgColour )
  648. term.setTextColour( promptColour )
  649. write( shell.dir() .. "> " )
  650. term.setTextColour( textColour )
  651.  
  652. local sLine = sRead( nil, tCommandHistory )
  653. if sLine ~= 1337 then
  654. table.insert( tCommandHistory, sLine )
  655. runLine( sLine )
  656. else
  657. drawMenu()
  658. while true do
  659. local e,param=os.pullEventRaw("key")
  660. if (param == keys["leftAlt"]) or (param == keys["rightAlt"]) then
  661. break
  662. elseif param==keys["tab"] then tabAuto=not tabAuto
  663. elseif cTable[keys.getName(param)] then
  664. cTable[keys.getName(param)]()
  665. end
  666. end
  667. term.setBackgroundColor(bgColor)
  668. term.clear()
  669. term.setCursorPos(1,1)
  670. end
  671. end
  672.  
  673. -- If this is the toplevel shell, run the shutdown program
  674.  
  675. term.setTextColor(promptColor)
  676. print("Goodbye")
  677. sleep(1)
  678.  
  679. os.shutdown() -- just in case
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement