Advertisement
Guest User

Untitled

a guest
Jun 26th, 2013
67
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Lua 11.57 KB | None | 0 0
  1. local colorStringsTree
  2. local breakerChars = {}
  3. local captureChars = {}
  4.  
  5. local nDefaultColor = colors.white
  6. local nNumberColor = colors.orange
  7. local nStringColor = colors.lightGray
  8. local nCommentColor = colors.lime
  9. local nBackgroundColor = colors.black
  10.  
  11. --RESPONSE PROTOTYPE--
  12. local tokenResponsePrototype = {}
  13. tokenResponsePrototype.__index = tokenResponsePrototype
  14.  
  15. setmetatable(tokenResponsePrototype, {
  16.     __call = function (cls, ...)
  17.     return cls.new(...)
  18.     end,
  19. })
  20.  
  21. --we check each token added to see if it is actually a number
  22. function tokenResponsePrototype:add(args)
  23.     self[#self+1] = args
  24.     --if the text is a number but is not a hyphen (because tonumber will consider "-" a number) then set its type to number
  25.     if tonumber(args.text) and args.text ~= "-" then
  26.         args.type = "number"
  27.     end
  28. end
  29.  
  30. function tokenResponsePrototype.new()
  31.     local self = setmetatable({}, tokenResponsePrototype)
  32.     return self
  33. end
  34. ----------------------
  35.  
  36. --COLORED STRING PROTOTYPE--
  37. local coloredStringProtoype = {}
  38. coloredStringProtoype.__index = coloredStringProtoype
  39.  
  40. setmetatable(coloredStringProtoype, {
  41.   __call = function (cls, ...)
  42.     return cls.new(...)
  43.   end,
  44. })
  45.  
  46. function coloredStringProtoype.new()
  47.   local self = setmetatable({}, coloredStringProtoype)
  48.   self.strings = {}
  49.   return self
  50. end
  51.  
  52. function coloredStringProtoype:addSubString(args)
  53.   if args.text ~= nil and args.text ~= "" then
  54.     self.strings[#self.strings+1] = {}
  55.     for key,value in pairs(args) do
  56.         self.strings[#self.strings][key] = value
  57.     end
  58.     return self.strings[#self.strings]
  59.   end
  60. end
  61.  
  62. function coloredStringProtoype:clearHighlights()
  63.     for i = 1, #self.strings do
  64.         self.strings[i]["highlights"] = nil
  65.     end
  66. end
  67. -------------------------
  68.  
  69. --TREE SEARCH PROTOTYPE--
  70. local searchBranchProtoype = {}
  71. searchBranchProtoype.__index = searchBranchProtoype
  72.  
  73. setmetatable(searchBranchProtoype, {
  74.   __call = function (cls, ...)
  75.     return cls.new(...)
  76.   end,
  77. })
  78.  
  79. function searchBranchProtoype.new(char)
  80.   local self = setmetatable({}, searchBranchProtoype)
  81.   self.char = char
  82.   self.branches = {}
  83.   return self
  84. end
  85.  
  86. function searchBranchProtoype:addBranch(args) -- char
  87.   if args.char ~= nil and args.char ~= "" then
  88.     self.branches[#self.branches+1] = args
  89.     return self.branches[#self.branches]
  90.   end
  91. end
  92.  
  93. function searchBranchProtoype:setEndBranch(args) -- color, highlight
  94.   if args.color ~= nil and args.color ~= "" then
  95.     self.color = args.color
  96.     self.endpoint = true
  97.     if args.highlight ~= nil and args.highlight ~= "" then
  98.         self.highlight = args.highlight
  99.     end
  100.   end
  101. end
  102. -------------------------
  103.  
  104. --TREE BUILDING AND SEARCHING--
  105. function buildTree(listOfStrings)
  106.     local tree = searchBranchProtoype("root")
  107.     for key,value in pairs(listOfStrings) do
  108.         addBranchtoTree(tree, key, value)
  109.     end
  110.     return tree
  111. end
  112.  
  113. function addBranchtoTree(tree, stringToAdd, finalColor)
  114.     local found = false
  115.     if tree.branches ~= nil then
  116.         for i = 1, #tree.branches do --foreach branch in the tree already there
  117.             if string.sub(stringToAdd,1,1) == tree.branches[i].char and #stringToAdd > 1 then --if the first letter of our string matches one of the branches, and it has additional letters
  118.                 addBranchtoTree(tree.branches[i], string.sub(stringToAdd, 2), finalColor) --then check inside to see if another matches
  119.                 found = true
  120.             elseif string.sub(stringToAdd,1,1) == tree.branches[i].char and #stringToAdd == 1 then --first letter matches and its also the last letter of the string.. even though a branch continues... set an endpoint here
  121.                 tree.branches[i]:setEndBranch{["color"] = finalColor}
  122.             end
  123.         end
  124.     end
  125.     if found == false then --didnt find a match.. add branch
  126.         local temp = searchBranchProtoype(string.sub(stringToAdd, 1, 1))
  127.         local newbranch = tree:addBranch(temp) -- the char should be equal to the first char of the stringToAdd
  128.         if #stringToAdd == 1 then --if its the last char in the string
  129.             newbranch:setEndBranch{["color"] = finalColor}
  130.         else --if its not hte last char, go into the next one
  131.             addBranchtoTree(newbranch, string.sub(stringToAdd, 2, #stringToAdd), finalColor) --then
  132.         end
  133.     end
  134. end
  135.  
  136. function searchTree(branch, stringToSearch) --0 is no match, 1 is partial match, 2 is complete match
  137.     if branch.branches ~= nil then
  138.         for i = 1, #branch.branches do --foreach branch in the tree already there
  139.             if string.sub(stringToSearch,1,1) == branch.branches[i].char then --if the first letter of our string matches one of the branches
  140.                 if #stringToSearch ~= 1 then
  141.                     if #branch.branches[i].branches > 0 then --if the next branch is not an endpoint
  142.                         return searchTree(branch.branches[i], string.sub(stringToSearch, 2)) --then check inside to see if another matches
  143.                     else --if the next branch is an endpoint but the string continues
  144.                         return {["result"] = 0}
  145.                     end
  146.                 elseif #stringToSearch == 1 and #branch.branches[i].branches == 0 then --if this is the last char and the branch it matches is an endpoint
  147.                     return {["result"] = 2, ["color"] = branch.branches[i].color}
  148.                 else --string has 1 char but branch is not an endpoint so this is a partial match
  149.                     return {["result"] = 1}
  150.                 end
  151.             end
  152.         end
  153.         return {["result"] = 0}
  154.     end
  155. end
  156. -------------------------
  157.  
  158. --TOP LEVEL FUNCTIONS----
  159.  
  160. --return a table that is indexed
  161. --entries contain the keys "text" and "type"
  162. --"text" is the text inside the token
  163. --type is either "token", "capture" (a string), "comment" (a comment string), "number" (a number)
  164. function tokenizeString(stringToT)
  165.     --where we store the tokens
  166.     response = tokenResponsePrototype()
  167.     --used to check the next character after a hyphen (for comment detection)
  168.     local waitForHyphen = false
  169.     --used to wait for a specific character and capture all characters inbetween. For double quote and single quotes
  170.     local currentCapture = ""
  171.     --the current token we're building
  172.     local token = ""
  173.     --loop through each character in the string we were asked to tokenize
  174.     for i = 1, #stringToT do
  175.         --get the character
  176.         local char = string.sub(stringToT, i, i)
  177.         --if we're waiting for a hyphen
  178.         if waitForHyphen then
  179.             waitForHyphen = false
  180.             --if we found the hyphen, then set the character we should wait for to "EOL", representing End Of Line, and add the hyphen to the current token
  181.             if char == "-" then
  182.                 token = token..char
  183.                 currentCapture = "EOL"
  184.             --if the character is not a hyphen then add a hyphen to the response list to make up for the first hyphen we withheld
  185.             else
  186.                 response:add{["text"] = "-", ["type"] = "token"}
  187.             end
  188.         end
  189.         --if its the last character
  190.         if i == #stringToT then
  191.             --if either the character is not a break character, or we're capturing anything
  192.             if breakerChars[char] == nil or currentCapture ~= "" then
  193.                 --if we're capturing for a comment then add the token and the character to the response with a 'type' of comment
  194.                 if currentCapture == "EOL" then
  195.                     response:add{["text"] = token..char, ["type"] = "comment"}
  196.                 --if we're not capturing for a comment then add token and character to response with 'type' of token
  197.                 else
  198.                     response:add{["text"] = token..char, ["type"] = "token"}
  199.                 end
  200.             --if char is a break character and we're not capturing anything then add token and char to response, but seperatley
  201.             else
  202.                 response:add{["text"] = token, ["type"] = "token"}
  203.                 response:add{["text"] = char, ["type"] = "token"}
  204.             end
  205.         --if char is a capture character
  206.         elseif captureChars[char] ~= nil then
  207.             --if char is the capture character we're looking for to end the current capture, and the previous character isnt a "\"
  208.             --then add token..char to response, empty token, and stop the capture
  209.             if char == currentCapture and string.sub(token, #token, #token) ~= "\\" then
  210.                 response:add{["text"] = token..char, ["type"] = "capture"}
  211.                 token = ""
  212.                 currentCapture = ""
  213.             --if there is no current capture then start a new one, and add the current token to response
  214.             elseif currentCapture == "" then
  215.                 currentCapture = char
  216.                 if token ~= "" then
  217.                     response:add{["text"] = token, ["type"] = "token"}
  218.                 end
  219.                 token = char
  220.             --if the current character is a capture character but not the one we're looking for then add it to the token
  221.             else
  222.                 token = token..char
  223.             end
  224.         --if the character is a break character and we're not in the middle of a capture
  225.         elseif breakerChars[char] ~= nil and currentCapture == "" then
  226.             --add token to response
  227.             if token ~= "" then
  228.                 response:add{["text"] = token, ["type"] = "token"}
  229.                 token = ""
  230.             end
  231.             --if the character is a hyphen then withold doing anything and set waitForHyphen in order to check the next character
  232.             if char == "-" then
  233.                 waitForHyphen = true
  234.             --if the character is a normal break character then add it to the response
  235.             else
  236.                 response:add{["text"] = char, ["type"] = "token"}
  237.             end
  238.         --if the character is not a break character, or capture character then add it to the token
  239.         else
  240.             token = token..char
  241.         end
  242.     end
  243.     return response
  244. end
  245.  
  246. --converts regular string into a colored string
  247. function stringToColorString(stringToWrite)
  248.     local response = coloredStringProtoype()
  249.     response["text"] = stringToWrite
  250.     local tokens = tokenizeString(stringToWrite)
  251.     for i = 1, #tokens do
  252.         local color
  253.         if tokens[i]["type"] == "comment" then
  254.             color = nCommentColor
  255.         elseif tokens[i]["type"] == "capture" then
  256.             color = nStringColor
  257.         elseif tokens[i]["type"] == "number" then
  258.             color = nNumberColor
  259.         elseif tokens[i]["type"] == "token" then
  260.             color = searchTree(colorStringsTree, tokens[i].text)["color"] or nDefaultColor
  261.         end
  262.         response:addSubString{["text"] = tokens[i].text, ["color"] = color}
  263.     end
  264.     return response
  265. end
  266.  
  267. function writeColorString(coloredString)
  268.     for i=1, #coloredString.strings do
  269.         if coloredString.strings[i]["highlight"] ~= nil then
  270.             term.setBackgroundColor(coloredString.strings[i]["highlight"])
  271.         else
  272.             term.setBackgroundColor(nBackgroundColor)
  273.         end
  274.         term.setTextColor(coloredString.strings[i]["color"])
  275.         term.write(coloredString.strings[i]["text"])
  276.     end
  277.     term.setTextColor(nDefaultColor)
  278.     term.setBackgroundColor(nBackgroundColor)
  279. end
  280.  
  281. ---------------------------
  282.  
  283. --initalize everything
  284. function initialize()
  285.  
  286.     local colorStrings = {}
  287.    
  288.     local keywordColor = colors.blue
  289.     local osAPIcolor = colors.red
  290.    
  291.     colorStrings["local"] = keywordColor
  292.     colorStrings["function"] = keywordColor
  293.     colorStrings["if"] = keywordColor
  294.     colorStrings["os.unloadAPI"] = osAPIcolor
  295.     colorStrings["os.pullEvent"] = osAPIcolor
  296.    
  297.     colorStringsTree = buildTree(colorStrings)
  298.    
  299.     ----------------------------
  300.  
  301.     breakerChars[" "] = nDefaultColor
  302.     breakerChars["-"] = nDefaultColor
  303.     breakerChars["+"] = nDefaultColor
  304.     breakerChars["/"] = nDefaultColor
  305.     breakerChars["*"] = nDefaultColor
  306.     breakerChars["#"] = nDefaultColor
  307.     breakerChars["("] = nDefaultColor
  308.     breakerChars[")"] = nDefaultColor
  309.     breakerChars["{"] = nDefaultColor
  310.     breakerChars["}"] = nDefaultColor
  311.     breakerChars["["] = nDefaultColor
  312.     breakerChars["]"] = nDefaultColor
  313.     breakerChars["~"] = nDefaultColor
  314.     breakerChars["+"] = nDefaultColor
  315.     breakerChars[">"] = nDefaultColor
  316.     breakerChars["<"] = nDefaultColor
  317.     breakerChars[","] = nDefaultColor
  318.  
  319.     captureChars["\""] = colors.gray
  320.     captureChars["'"] = colors.gray
  321.     captureChars["EOL"] = colors.green
  322. end
  323.  
  324. initialize()
  325. --tokenize the tests string
  326. local coloredTokens = stringToColorString("local something = \"test\\\" string\" --test comment")
  327.  
  328. if term.isColor() then
  329.     writeColorString(coloredTokens)
  330. else
  331.     local file = fs.open("tableSaved","w")
  332.     file.write(textutils.serialize(coloredTokens))
  333.     file.close()
  334. end
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement