Added missing text files.
This commit is contained in:
parent
5864354739
commit
965624714e
293
text.lua
Normal file
293
text.lua
Normal file
@ -0,0 +1,293 @@
|
||||
local List, Dict
|
||||
do
|
||||
local _obj_0 = require("containers")
|
||||
List, Dict = _obj_0.List, _obj_0.Dict
|
||||
end
|
||||
local reverse, upper, lower, find, byte, match, gmatch, gsub, sub, format, rep, char
|
||||
do
|
||||
local _obj_0 = string
|
||||
reverse, upper, lower, find, byte, match, gmatch, gsub, sub, format, rep, char = _obj_0.reverse, _obj_0.upper, _obj_0.lower, _obj_0.find, _obj_0.byte, _obj_0.match, _obj_0.gmatch, _obj_0.gsub, _obj_0.sub, _obj_0.format, _obj_0.rep, _obj_0.char
|
||||
end
|
||||
local isplit
|
||||
isplit = function(self, sep)
|
||||
if sep == nil then
|
||||
sep = '%s+'
|
||||
end
|
||||
local step
|
||||
step = function(self, i)
|
||||
local start = self.pos
|
||||
if not (start) then
|
||||
return
|
||||
end
|
||||
i = i + 1
|
||||
local nl = find(self.str, self.sep, start)
|
||||
self.pos = nl and (nl + 1) or nil
|
||||
local line = sub(self.str, start, nl and (nl - 1) or #self.str)
|
||||
return i, line, start, (nl and (nl - 1) or #self.str)
|
||||
end
|
||||
return step, {
|
||||
str = self,
|
||||
pos = 1,
|
||||
sep = sep
|
||||
}, 0
|
||||
end
|
||||
local lua_keywords = {
|
||||
["and"] = true,
|
||||
["break"] = true,
|
||||
["do"] = true,
|
||||
["else"] = true,
|
||||
["elseif"] = true,
|
||||
["end"] = true,
|
||||
["false"] = true,
|
||||
["for"] = true,
|
||||
["function"] = true,
|
||||
["goto"] = true,
|
||||
["if"] = true,
|
||||
["in"] = true,
|
||||
["local"] = true,
|
||||
["nil"] = true,
|
||||
["not"] = true,
|
||||
["or"] = true,
|
||||
["repeat"] = true,
|
||||
["return"] = true,
|
||||
["then"] = true,
|
||||
["true"] = true,
|
||||
["until"] = true,
|
||||
["while"] = true
|
||||
}
|
||||
local is_lua_id
|
||||
is_lua_id = function(str)
|
||||
return match(str, "^[_a-zA-Z][_a-zA-Z0-9]*$") and not lua_keywords[str]
|
||||
end
|
||||
local as_lua_id
|
||||
as_lua_id = function(str)
|
||||
str = gsub(str, "x([0-9A-F][0-9A-F])", "x78%1")
|
||||
str = gsub(str, "%W", function(c)
|
||||
if c == ' ' then
|
||||
return '_'
|
||||
else
|
||||
return format("x%02X", byte(c))
|
||||
end
|
||||
end)
|
||||
if not (is_lua_id(match(str, "^_*(.*)$"))) then
|
||||
str = "_" .. str
|
||||
end
|
||||
return str
|
||||
end
|
||||
local from_lua_id
|
||||
from_lua_id = function(str)
|
||||
if not (is_lua_id(match(str, "^_*(.*)$"))) then
|
||||
str = sub(str, 2, -1)
|
||||
end
|
||||
str = gsub(str, "_", " ")
|
||||
str = gsub(str, "x([0-9A-F][0-9A-F])", function(hex)
|
||||
return char(tonumber(hex, 16))
|
||||
end)
|
||||
return str
|
||||
end
|
||||
local Text = {
|
||||
isplit = isplit,
|
||||
uppercase = upper,
|
||||
lowercase = lower,
|
||||
reversed = reverse,
|
||||
is_lua_id = is_lua_id,
|
||||
capitalized = function(self)
|
||||
return (gsub(self, '%l', upper, 1))
|
||||
end,
|
||||
byte = byte,
|
||||
as_a_number = tonumber,
|
||||
as_a_base_1_number = tonumber,
|
||||
bytes = function(self, i, j)
|
||||
return List({
|
||||
byte(self, i or 1, j or -1)
|
||||
})
|
||||
end,
|
||||
split = function(self, sep)
|
||||
return List((function()
|
||||
local _accum_0 = { }
|
||||
local _len_0 = 1
|
||||
for i, chunk in isplit(self, sep) do
|
||||
_accum_0[_len_0] = chunk
|
||||
_len_0 = _len_0 + 1
|
||||
end
|
||||
return _accum_0
|
||||
end)())
|
||||
end,
|
||||
starts_with = function(self, s)
|
||||
return sub(self, 1, #s) == s
|
||||
end,
|
||||
ends_with = function(self, s)
|
||||
return #self >= #s and sub(self, #self - #s, -1) == s
|
||||
end,
|
||||
lines = function(self)
|
||||
return List((function()
|
||||
local _accum_0 = { }
|
||||
local _len_0 = 1
|
||||
for i, line in isplit(self, '\n') do
|
||||
_accum_0[_len_0] = line
|
||||
_len_0 = _len_0 + 1
|
||||
end
|
||||
return _accum_0
|
||||
end)())
|
||||
end,
|
||||
line = function(self, line_num)
|
||||
for i, line, start in isplit(self, '\n') do
|
||||
if i == line_num then
|
||||
return line
|
||||
end
|
||||
end
|
||||
end,
|
||||
line_info_at = function(self, pos)
|
||||
assert(type(pos) == 'number', "Invalid string position")
|
||||
for i, line, start, stop in isplit(self, '\n') do
|
||||
if stop + 1 >= pos then
|
||||
return line, i, (pos - start + 1)
|
||||
end
|
||||
end
|
||||
end,
|
||||
indented = function(self, indent)
|
||||
if indent == nil then
|
||||
indent = " "
|
||||
end
|
||||
return indent .. (gsub(self, "\n", "\n" .. indent))
|
||||
end,
|
||||
as_lua = function(self)
|
||||
local escaped = gsub(self, "\\", "\\\\")
|
||||
escaped = gsub(escaped, "\n", "\\n")
|
||||
escaped = gsub(escaped, '"', '\\"')
|
||||
escaped = gsub(escaped, "[^ %g]", function(c)
|
||||
return format("\\%03d", byte(c, 1))
|
||||
end)
|
||||
return '"' .. escaped .. '"'
|
||||
end,
|
||||
as_nomsu = function(self)
|
||||
return self:as_lua()
|
||||
end,
|
||||
formatted_with = format,
|
||||
byte = byte,
|
||||
position_of = (function(...)
|
||||
return (find(...))
|
||||
end),
|
||||
position_of_1_after = (function(...)
|
||||
return (find(...))
|
||||
end),
|
||||
as_a_lua_identifier = as_lua_id,
|
||||
is_a_lua_identifier = is_lua_id,
|
||||
as_lua_id = as_lua_id,
|
||||
as_a_lua_id = as_lua_id,
|
||||
is_a_lua_id = is_lua_id,
|
||||
is_lua_id = is_lua_id,
|
||||
from_lua_id = from_lua_id,
|
||||
bytes_1_to = function(self, start, stop)
|
||||
return List({
|
||||
byte(self, start, stop)
|
||||
})
|
||||
end,
|
||||
[as_lua_id("with 1 ->")] = function(...)
|
||||
return (gsub(...))
|
||||
end,
|
||||
bytes = function(self)
|
||||
return List({
|
||||
byte(self, 1, -1)
|
||||
})
|
||||
end,
|
||||
wrapped_to = function(self, maxlen, margin)
|
||||
if maxlen == nil then
|
||||
maxlen = 80
|
||||
end
|
||||
if margin == nil then
|
||||
margin = 8
|
||||
end
|
||||
local lines = { }
|
||||
local _list_0 = self:lines()
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local line = _list_0[_index_0]
|
||||
while #line > maxlen do
|
||||
local chunk = sub(line, 1, maxlen)
|
||||
local split = find(chunk, ' ', maxlen - margin, true) or maxlen
|
||||
chunk = sub(line, 1, split)
|
||||
line = sub(line, split + 1, -1)
|
||||
lines[#lines + 1] = chunk
|
||||
end
|
||||
lines[#lines + 1] = line
|
||||
end
|
||||
return table.concat(lines, "\n")
|
||||
end,
|
||||
line_at = function(self, i)
|
||||
return (self:line_info_at(i))
|
||||
end,
|
||||
line_number_at = function(self, i)
|
||||
return select(2, self:line_info_at(i))
|
||||
end,
|
||||
line_position_at = function(self, i)
|
||||
return select(3, self:line_info_at(i))
|
||||
end,
|
||||
matches = function(self, patt)
|
||||
return match(self, patt) and true or false
|
||||
end,
|
||||
matching = function(self, patt)
|
||||
return (match(self, patt))
|
||||
end,
|
||||
matching_groups = function(self, patt)
|
||||
return List({
|
||||
match(self, patt)
|
||||
})
|
||||
end,
|
||||
[as_lua_id("* 1")] = function(self, n)
|
||||
return rep(self, n)
|
||||
end,
|
||||
all_matches_of = function(self, patt)
|
||||
local result = { }
|
||||
local stepper, x, i = gmatch(self, patt)
|
||||
while true do
|
||||
local tmp = List({
|
||||
stepper(x, i)
|
||||
})
|
||||
if #tmp == 0 then
|
||||
break
|
||||
end
|
||||
i = tmp[1]
|
||||
result[#result + 1] = (#tmp == 1) and tmp[1] or tmp
|
||||
end
|
||||
return List(result)
|
||||
end,
|
||||
from_1_to = sub,
|
||||
from = sub,
|
||||
character = function(self, i)
|
||||
return sub(self, i, i)
|
||||
end
|
||||
}
|
||||
for k, v in pairs(string) do
|
||||
Text[k] = Text[k] or v
|
||||
end
|
||||
local _1_as_text
|
||||
_1_as_text = function(x)
|
||||
if x == true then
|
||||
return "yes"
|
||||
end
|
||||
if x == false then
|
||||
return "no"
|
||||
end
|
||||
return tostring(x)
|
||||
end
|
||||
setmetatable(Text, {
|
||||
__call = function(self, ...)
|
||||
local ret = {
|
||||
...
|
||||
}
|
||||
for i = 1, select("#", ...) do
|
||||
ret[i] = _1_as_text(ret[i])
|
||||
end
|
||||
return table.concat(ret)
|
||||
end
|
||||
})
|
||||
debug.setmetatable("", {
|
||||
__type = "Text",
|
||||
__index = function(self, k)
|
||||
return Text[k] or (type(k) == 'number' and sub(self, k, k) or nil)
|
||||
end,
|
||||
__add = function(self, x)
|
||||
return _1_as_text(self) .. _1_as_text(x)
|
||||
end
|
||||
})
|
||||
return Text
|
141
text.moon
Normal file
141
text.moon
Normal file
@ -0,0 +1,141 @@
|
||||
-- Expand the capabilities of the built-in strings
|
||||
{:List, :Dict} = require "containers"
|
||||
{:reverse, :upper, :lower, :find, :byte, :match, :gmatch, :gsub, :sub, :format, :rep, :char} = string
|
||||
|
||||
isplit = (sep='%s+')=>
|
||||
step = (i)=>
|
||||
start = @pos
|
||||
return unless start
|
||||
i += 1
|
||||
nl = find(@str, @sep, start)
|
||||
@pos = nl and (nl+1) or nil
|
||||
line = sub(@str, start, nl and (nl-1) or #@str)
|
||||
return i, line, start, (nl and (nl-1) or #@str)
|
||||
return step, {str:@, pos:1, :sep}, 0
|
||||
|
||||
lua_keywords = {
|
||||
["and"]:true, ["break"]:true, ["do"]:true, ["else"]:true, ["elseif"]:true, ["end"]:true,
|
||||
["false"]:true, ["for"]:true, ["function"]:true, ["goto"]:true, ["if"]:true,
|
||||
["in"]:true, ["local"]:true, ["nil"]:true, ["not"]:true, ["or"]:true, ["repeat"]:true,
|
||||
["return"]:true, ["then"]:true, ["true"]:true, ["until"]:true, ["while"]:true
|
||||
}
|
||||
is_lua_id = (str)->
|
||||
match(str, "^[_a-zA-Z][_a-zA-Z0-9]*$") and not lua_keywords[str]
|
||||
|
||||
-- Convert an arbitrary text into a valid Lua identifier. This function is injective,
|
||||
-- but not idempotent. In logic terms: (x != y) => (as_lua_id(x) != as_lua_id(y)),
|
||||
-- but not (as_lua_id(a) == b) => (as_lua_id(b) == b).
|
||||
as_lua_id = (str)->
|
||||
-- Escape 'x' (\x78) when it precedes something that looks like an uppercase hex sequence.
|
||||
-- This way, all Lua IDs can be unambiguously reverse-engineered, but normal usage
|
||||
-- of 'x' won't produce ugly Lua IDs.
|
||||
-- i.e. "x" -> "x", "oxen" -> "oxen", but "Hex2Dec" -> "Hex782Dec" and "He-ec" -> "Hex2Dec"
|
||||
str = gsub str, "x([0-9A-F][0-9A-F])", "x78%1"
|
||||
-- Map spaces to underscores, and everything else non-alphanumeric to hex escape sequences
|
||||
str = gsub str, "%W", (c)->
|
||||
if c == ' ' then '_'
|
||||
else format("x%02X", byte(c))
|
||||
|
||||
unless is_lua_id(match(str, "^_*(.*)$"))
|
||||
str = "_"..str
|
||||
return str
|
||||
|
||||
-- from_lua_id(as_lua_id(str)) == str, but behavior is unspecified for inputs that
|
||||
-- did not come from as_lua_id()
|
||||
from_lua_id = (str)->
|
||||
unless is_lua_id(match(str, "^_*(.*)$"))
|
||||
str = sub(str,2,-1)
|
||||
str = gsub(str, "_", " ")
|
||||
str = gsub(str, "x([0-9A-F][0-9A-F])", (hex)-> char(tonumber(hex, 16)))
|
||||
return str
|
||||
|
||||
Text = {
|
||||
:isplit, uppercase:upper, lowercase:lower, reversed:reverse, :is_lua_id
|
||||
capitalized: => (gsub(@, '%l', upper, 1))
|
||||
byte: byte, as_a_number: tonumber, as_a_base_1_number: tonumber,
|
||||
bytes: (i, j)=> List{byte(@, i or 1, j or -1)}
|
||||
split: (sep)=> List[chunk for i,chunk in isplit(@, sep)]
|
||||
starts_with: (s)=> sub(@, 1, #s) == s
|
||||
ends_with: (s)=> #@ >= #s and sub(@, #@-#s, -1) == s
|
||||
lines: => List[line for i,line in isplit(@, '\n')]
|
||||
line: (line_num)=>
|
||||
for i, line, start in isplit(@, '\n')
|
||||
return line if i == line_num
|
||||
|
||||
line_info_at: (pos)=>
|
||||
assert(type(pos) == 'number', "Invalid string position")
|
||||
for i, line, start, stop in isplit(@, '\n')
|
||||
if stop+1 >= pos
|
||||
return line, i, (pos-start+1)
|
||||
|
||||
indented: (indent=" ")=>
|
||||
indent..(gsub(@, "\n", "\n"..indent))
|
||||
|
||||
as_lua: =>
|
||||
escaped = gsub(@, "\\", "\\\\")
|
||||
escaped = gsub(escaped, "\n", "\\n")
|
||||
escaped = gsub(escaped, '"', '\\"')
|
||||
escaped = gsub(escaped, "[^ %g]", (c)-> format("\\%03d", byte(c, 1)))
|
||||
return '"'..escaped..'"'
|
||||
|
||||
as_nomsu: => @as_lua!
|
||||
|
||||
formatted_with:format, byte:byte,
|
||||
position_of:((...)->(find(...))), position_of_1_after:((...)->(find(...))),
|
||||
as_a_lua_identifier: as_lua_id, is_a_lua_identifier: is_lua_id,
|
||||
as_lua_id: as_lua_id, as_a_lua_id: as_lua_id, is_a_lua_id: is_lua_id,
|
||||
is_lua_id: is_lua_id, from_lua_id: from_lua_id,
|
||||
bytes_1_to: (start, stop)=> List{byte(@, start, stop)}
|
||||
[as_lua_id "with 1 ->"]: (...)-> (gsub(...))
|
||||
bytes: => List{byte(@, 1, -1)},
|
||||
wrapped_to: (maxlen=80, margin=8)=>
|
||||
lines = {}
|
||||
for line in *@lines!
|
||||
while #line > maxlen
|
||||
chunk = sub(line, 1, maxlen)
|
||||
split = find(chunk, ' ', maxlen-margin, true) or maxlen
|
||||
chunk = sub(line, 1, split)
|
||||
line = sub(line, split+1, -1)
|
||||
lines[#lines+1] = chunk
|
||||
lines[#lines+1] = line
|
||||
return table.concat(lines, "\n")
|
||||
|
||||
line_at: (i)=> (@line_info_at(i))
|
||||
line_number_at: (i)=> select(2, @line_info_at(i))
|
||||
line_position_at: (i)=> select(3, @line_info_at(i))
|
||||
matches: (patt)=> match(@, patt) and true or false
|
||||
matching: (patt)=> (match(@, patt))
|
||||
matching_groups: (patt)=> List{match(@, patt)}
|
||||
[as_lua_id "* 1"]: (n)=> rep(@, n)
|
||||
all_matches_of: (patt)=>
|
||||
result = {}
|
||||
stepper,x,i = gmatch(@, patt)
|
||||
while true
|
||||
tmp = List{stepper(x,i)}
|
||||
break if #tmp == 0
|
||||
i = tmp[1]
|
||||
result[#result+1] = (#tmp == 1) and tmp[1] or tmp
|
||||
return List(result)
|
||||
from_1_to: sub, from: sub,
|
||||
character: (i)=> sub(@, i, i)
|
||||
}
|
||||
for k,v in pairs(string) do Text[k] or= v
|
||||
|
||||
_1_as_text = (x)->
|
||||
if x == true then return "yes"
|
||||
if x == false then return "no"
|
||||
return tostring(x)
|
||||
|
||||
setmetatable Text,
|
||||
__call: (...)=>
|
||||
ret = {...}
|
||||
for i=1,select("#", ...)
|
||||
ret[i] = _1_as_text(ret[i])
|
||||
return table.concat(ret)
|
||||
|
||||
debug.setmetatable "",
|
||||
__type: "Text"
|
||||
__index: (k)=> Text[k] or (type(k) == 'number' and sub(@, k, k) or nil)
|
||||
__add: (x)=> _1_as_text(@).._1_as_text(x)
|
||||
|
||||
return Text
|
Loading…
Reference in New Issue
Block a user