-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathutil.lua
218 lines (179 loc) · 5.73 KB
/
util.lua
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
local util = {}
local config = require("config")
local metatables = scripts and scripts.metatables
function util.doEvery(tick, func, args)
if (game.tick % tick) == 0 then func(args) end
end
function util.mod(name)
return game.active_mods[name]
end
function util.trim(str)
return str and (str:gsub("^%s*(.-)%s*$", "%1")) or ""
end
function util.fullTrim(str) -- trim and also remove multiple white spaces
return (util.trim(str):gsub("[ \t\r\n]*[\r\n][ \t\r\n]*", "\r\n"):gsub("[ \t]+", " "))
end
local quotepattern = '(['..("%^$().[]*+-?"):gsub("(.)", "%%%1")..'])'
function util.escape(str) -- escape string for use with regex
return str:gsub(quotepattern, "%%%1")
end
function util.offsetPosition(pos, offX, offY)
return { x = pos.x + offX, y = pos.y + offY }
end
function util.getSurroundingArea(pos, radius)
return { left_top = { x = pos.x - radius, y = pos.y - radius },
right_bottom = { x = pos.x + radius, y = pos.y + radius } }
end
function util.calcArea(area)
return math.abs((area.right_bottom.x - area.left_top.x) * (area.right_bottom.y - area.left_top.y))
end
function util.isValid(object)
return type(object) == "table" and object.valid
end
function util.destroyIfValid(object)
if util.isValid(object) then object.destroy() end
end
function util.isValidStack(stack)
return util.isValid(stack) and stack.valid_for_read
end
function util.isValidPlayer(player) -- valid, connected and alive player
return util.isValid(player) and player.connected and player.controller_type ~= defines.controllers.ghost
end
function util.isValidArea(area)
return area and area.left_top and area.right_bottom and util.calcArea(area) > 0
end
function util.fixArea(area) -- fix an area for use in surface.find_entities_filtered, returns key,value pair
if util.isValidArea(area) then
return "area", area
else
return "position", area.left_top
end
end
function util.iterateInvs(player, func) -- iterate over all ItemStacks of this player (in all inventories)
local stack = player.cursor_stack
if util.isValidStack(stack) and func(stack) then return true end
stack = player.blueprint_to_setup
if util.isValidStack(stack) and func(stack) then return true end
for _,invIndex in pairs(defines.inventory) do
local inv = player.get_inventory(invIndex)
if util.isValid(inv) and not inv.is_empty() then
for i = 1, #inv do
stack = inv[i]
if util.isValidStack(stack) and func(stack) then return true end
end
end
end
return false
end
function util.iconHasItem(icon)
return icon and icon.name and util.isValid(game.item_prototypes[icon.name])
end
function util.supportsNotes(entity)
local name = entity.type == "entity-ghost" and entity.ghost_name or entity.name
local type = entity.type == "entity-ghost" and entity.ghost_type or entity.type
if name == "blueprint-note-storage-new" then return false end
return config.offerAttachNote[type] or config.alwaysAttachNote[type] or config.alwaysAttachNote[name]
end
function util.getColorOrDefault(name, settings, note, replacements)
local color = "white" --"black"
if note and note[name.."Color"] then
local saved = note[name.."Color"]
if type(saved) == "table" then return saved end -- if rgb value is saved return it directly
color = config.colors[saved]
elseif settings then
if name == "label" then name = "title" end
color = settings["default-"..name.."-color2"].value
end
return config.colorFromName[(replacements and replacements[color]) or color]
end
function util.shallowCopy(orig)
local copy
if type(orig) == 'table' then
copy = {}
for orig_key, orig_value in pairs(orig) do
copy[orig_key] = orig_value
end
else -- number, string, boolean, etc
copy = orig
end
return copy
end
function util.deepCopy(orig)
local copy
if type(orig) == 'table' then
copy = {}
for orig_key, orig_value in pairs(orig) do
copy[orig_key] = util.deepCopy(orig_value)
end
--setmetatable(copy, util.deepCopy(getmetatable(orig)))
else -- number, string, boolean, etc
copy = orig
end
return copy
end
function util.countTable(tbl)
local count = 0
for __ in pairs(tbl) do count = count + 1 end
return count
end
function util.isEmpty(tbl) -- empty table
return type(tbl) == "table" and next(tbl) == nil
end
function util.isFilled(tbl) -- filled table
return type(tbl) == "table" and next(tbl) ~= nil
end
function util.findInTable(tbl, func)
for key,value in pairs(tbl) do
if func(value, key) then return value, key end
end
end
function util.concat(...)
local result, lastIndex = {}, 0
for _,val in pairs({...}) do
if type(val) == "table" then
for index,value in pairs(val) do
result[lastIndex + index] = value
end
lastIndex = lastIndex + #val
elseif val ~= nil then
result[lastIndex + 1] = val
lastIndex = lastIndex + 1
end
end
return result
end
function util.localize(array, section)
section = section..'.'
local localized = {}
for i,val in ipairs(array) do
localized[i] = {section..val}
end
return localized
end
function util.createRenderingWrapper(id)
return metatables.use({ __id = id }, "rendering")
end
function util.epairs(tbl) -- iterator for tables with entity based indices
local tblId = rawget(tbl, "id")
local tblPos = rawget(tbl, "pos")
local idIterator, posIterator
local currentIterator
if tblPos then
posIterator = pairs(tblPos)
currentIterator = posIterator
end
if tblId then
idIterator = pairs(tblId)
currentIterator = idIterator
end
return function ()
local index, value
if currentIterator then index, value = currentIterator() end
if not index and currentIterator == idIterator then
currentIterator = posIterator
if currentIterator then index, value = currentIterator() end
end
return value
end
end
return util