Implement basic graphical elements
This commit is contained in:
commit
da6325e080
170
gfx/element.lua
Normal file
170
gfx/element.lua
Normal file
@ -0,0 +1,170 @@
|
||||
local Event = require("event")
|
||||
|
||||
local Element = {
|
||||
x = 0,
|
||||
y = 0,
|
||||
width = 1,
|
||||
height = 1,
|
||||
color = {
|
||||
bg = colors.black,
|
||||
fg = colors.white
|
||||
},
|
||||
render = {
|
||||
window = nil,
|
||||
parent = nil
|
||||
},
|
||||
visible = true,
|
||||
dirty = true,
|
||||
id = "",
|
||||
onClick = nil
|
||||
}
|
||||
|
||||
local function createWindow(element)
|
||||
return window.create(element.render.parent, element:getX(), element:getY(), element:getWidth(), element:getHeight(), element:isVisible())
|
||||
end
|
||||
|
||||
|
||||
function Element:new(o)
|
||||
local obj = o or {}
|
||||
|
||||
setmetatable(obj, self)
|
||||
obj.__index = self
|
||||
|
||||
return obj
|
||||
end
|
||||
|
||||
function Element:draw()
|
||||
self.dirty = false
|
||||
end
|
||||
|
||||
function Element:getX()
|
||||
return self.x
|
||||
end
|
||||
|
||||
function Element:getY()
|
||||
return self.y
|
||||
end
|
||||
|
||||
function Element:setPos(x, y)
|
||||
if (x ~= nil and self.x ~= x) or (y ~= nil and self.y ~= y) then
|
||||
self:setDirty()
|
||||
self.x = x or self.x
|
||||
self.y = y or self.y
|
||||
|
||||
self.render.window = createWindow(self)
|
||||
end
|
||||
end
|
||||
|
||||
function Element:setX(x)
|
||||
if self.x ~= x then
|
||||
self:setDirty()
|
||||
self.x = x
|
||||
self.render.window = createWindow(self)
|
||||
end
|
||||
end
|
||||
|
||||
function Element:setY(y)
|
||||
if self.y ~= y then
|
||||
self:setDirty()
|
||||
self.y = y
|
||||
self.render.window = createWindow(self)
|
||||
end
|
||||
end
|
||||
|
||||
function Element:setParent(parent)
|
||||
if self.parent ~= parent then
|
||||
self:setDirty()
|
||||
self.parent = parent
|
||||
end
|
||||
end
|
||||
|
||||
function Element:setFgColor(color)
|
||||
if color ~= self.color.fg then
|
||||
self:setDirty()
|
||||
self.color.fg = color
|
||||
self.render.window.setTextColor(color)
|
||||
end
|
||||
end
|
||||
|
||||
function Element:setBgColor(color)
|
||||
if color ~= self.color.bg then
|
||||
self:setDirty()
|
||||
self.color.bg = color
|
||||
self.render.window.setBackgroundColor(color)
|
||||
end
|
||||
end
|
||||
|
||||
function Element:setVisible(visible)
|
||||
self.render.window.setVisible(visible)
|
||||
end
|
||||
|
||||
function Element:resize(opts)
|
||||
if (opts.width ~= nil and self.x ~= opts.width) or (opts.height ~= nil and self.y ~= opts.height) then
|
||||
self:setDirty()
|
||||
self.width = opts.width or self.width
|
||||
self.height = opts.height or self.height
|
||||
self.render.window = createWindow(self)
|
||||
end
|
||||
end
|
||||
|
||||
function Element:setWidth(width)
|
||||
if width ~= self.width then
|
||||
self:setDirty()
|
||||
self.width = width
|
||||
self.render.window = createWindow(self)
|
||||
end
|
||||
end
|
||||
|
||||
function Element:setHeight(height)
|
||||
if height ~= self.height then
|
||||
self:setDirty()
|
||||
self.height = height
|
||||
self.render.window = createWindow(self)
|
||||
end
|
||||
end
|
||||
|
||||
function Element:getWidth()
|
||||
return self.width
|
||||
end
|
||||
|
||||
function Element:getHeight()
|
||||
return self.height
|
||||
end
|
||||
|
||||
function Element:getPos()
|
||||
return self.x, self.y
|
||||
end
|
||||
|
||||
function Element:isVisible()
|
||||
return self.visible
|
||||
end
|
||||
|
||||
function Element:setDirty()
|
||||
self.dirty = true
|
||||
end
|
||||
|
||||
function Element:findById(id)
|
||||
if self:getId() == id then
|
||||
return self
|
||||
else
|
||||
return nil
|
||||
end
|
||||
end
|
||||
|
||||
function Element:getId()
|
||||
return self.id
|
||||
end
|
||||
|
||||
function Element:redraw()
|
||||
self.render.window.redraw()
|
||||
end
|
||||
|
||||
function Element:handleEvent(evt)
|
||||
if Event.isClickEvent(evt) and self.onClick ~= nil then
|
||||
local x, y, source = Event.getClickParams(evt)
|
||||
return not not self.onClick(self, x, y, source)
|
||||
end
|
||||
return false
|
||||
end
|
||||
|
||||
return Element
|
27
gfx/event.lua
Normal file
27
gfx/event.lua
Normal file
@ -0,0 +1,27 @@
|
||||
local Event = {}
|
||||
|
||||
function Event.isClickEvent(evt)
|
||||
return evt[1] == "monitor_touch" or evt[1] == "mouse_click"
|
||||
end
|
||||
|
||||
function Event.getClickParams(evt)
|
||||
return evt[3], evt[4], evt[2]
|
||||
end
|
||||
|
||||
function Event.repositionEvent(evt, dX, dY)
|
||||
if Event.isClickEvent(evt) then
|
||||
return evt[1], evt[2], evt[3] + dX, evt[4] + dY
|
||||
else
|
||||
return evt
|
||||
end
|
||||
end
|
||||
|
||||
function Event.containsClick(element, evt, dX, dY)
|
||||
local x, y = Event.getClickParams(evt)
|
||||
x = x + dX
|
||||
y = y + dY
|
||||
local eX, eY = element:getPos()
|
||||
return x >= eX and x < (eX + element:getWidth()) and y >= eY and y < (eY + element:getHeight())
|
||||
end
|
||||
|
||||
return Event
|
108
gfx/list.lua
Normal file
108
gfx/list.lua
Normal file
@ -0,0 +1,108 @@
|
||||
local Element = require("element")
|
||||
local List = Element:new{
|
||||
children = {},
|
||||
vertical = false
|
||||
}
|
||||
|
||||
local function adjustPositions(elements, vertical, from)
|
||||
local newDims = 0
|
||||
local getDim = vertical and function(e) return e:getHeight() end or function(e) return e:getWidth() end
|
||||
for i=1,from-1 do
|
||||
newDims = newDims + getDim(elements[i])
|
||||
end
|
||||
|
||||
local setDim = vertical and function(e, dim) e:setPos(0, dim) end or function(e, dim) e:setPos(dim, 0) end
|
||||
for i=from,#elements do
|
||||
setDim(elements[i])
|
||||
newDims = newDims + getDim(elements[i])
|
||||
end
|
||||
end
|
||||
|
||||
function List:insertChild(child, atIndex)
|
||||
local index = math.min(math.max(1, atIndex or #self.children), #self.children)
|
||||
table.insert(self.children, index, child)
|
||||
adjustPositions(self.children, self:isVertical(), index)
|
||||
end
|
||||
|
||||
function List:removeChild(child)
|
||||
local index
|
||||
local searchType = type(child)
|
||||
if searchType == "string" then
|
||||
for i,v in ipairs(self.children) do
|
||||
if v:getId() == child then
|
||||
index = i
|
||||
break
|
||||
end
|
||||
end
|
||||
return false, nil
|
||||
elseif searchType == "table" then
|
||||
for i,v in ipairs(self.children) do
|
||||
if v == child then
|
||||
index = i
|
||||
end
|
||||
end
|
||||
return false, nil
|
||||
else
|
||||
index = child
|
||||
end
|
||||
|
||||
if index <= 0 or index > #self.children then
|
||||
return false, nil
|
||||
end
|
||||
|
||||
local removed = table.remove(self.children, index)
|
||||
if index <= #self.children then
|
||||
adjustPositions(self.children, self:isVertical(), index)
|
||||
end
|
||||
|
||||
return true, removed
|
||||
end
|
||||
|
||||
function List:isVertical()
|
||||
return self.vertical
|
||||
end
|
||||
|
||||
function List:isHorizontal()
|
||||
return not self:isVertical()
|
||||
end
|
||||
|
||||
function List:draw()
|
||||
Element.draw(self)
|
||||
self.render.window.clear()
|
||||
for _,v in ipairs(self.children) do
|
||||
v:draw()
|
||||
end
|
||||
end
|
||||
|
||||
function List:getHeight()
|
||||
local h = 0
|
||||
for _,v in ipairs(self.children) do
|
||||
h = h + v:getHeight()
|
||||
end
|
||||
return h
|
||||
end
|
||||
|
||||
function List:getWidth()
|
||||
local h = 0
|
||||
for _,v in ipairs(self.children) do
|
||||
h = h + v:getHeight()
|
||||
end
|
||||
return h
|
||||
end
|
||||
|
||||
function List:findById(id)
|
||||
local find = Element.findById(self, id)
|
||||
if find then
|
||||
return find
|
||||
end
|
||||
|
||||
for _,v in ipairs(self.children) do
|
||||
local result = v:findById(id)
|
||||
if result then
|
||||
return result
|
||||
end
|
||||
end
|
||||
return nil
|
||||
end
|
||||
|
||||
return List
|
70
gfx/padding.lua
Normal file
70
gfx/padding.lua
Normal file
@ -0,0 +1,70 @@
|
||||
local Event = require("event")
|
||||
local Element = require("element")
|
||||
local Padding = Element:new{
|
||||
left = 0,
|
||||
right = 0,
|
||||
top = 0,
|
||||
bottom = 0,
|
||||
element = nil
|
||||
}
|
||||
|
||||
function Padding:resize(opts)
|
||||
-- Un-pad dimensions and pass to child element
|
||||
return self.element:resize{
|
||||
width = (opts.width and opts.width - self:getPaddingLeft() - self:getPaddingRight()) or self:getWidth(),
|
||||
height = (opts.height and opts.height - self:getPaddingTop() - self:getPaddingBottom()) or self:getHeight()
|
||||
}
|
||||
end
|
||||
|
||||
function Padding:getPaddingLeft()
|
||||
return self.left
|
||||
end
|
||||
|
||||
function Padding:getPaddingRight()
|
||||
return self.right
|
||||
end
|
||||
|
||||
function Padding:getPaddingTop()
|
||||
return self.top
|
||||
end
|
||||
|
||||
function Padding:getPaddingBottom()
|
||||
return self.bottom
|
||||
end
|
||||
|
||||
function Padding:getInnerWidth()
|
||||
return self.element:getWidth()
|
||||
end
|
||||
|
||||
function Padding:getInnerHeight()
|
||||
return self.element:getHeight()
|
||||
end
|
||||
|
||||
function Padding:getWidth()
|
||||
return self:getInnerWidth() + self:getPaddingLeft() + self:getPaddingRight()
|
||||
end
|
||||
|
||||
function Padding:getHeight()
|
||||
return self:getInnerHeight() + self:getPaddingTop() + self:getPaddingBottom()
|
||||
end
|
||||
|
||||
function Padding:findById(id)
|
||||
return Element.findById(self, id) or self.element:findById(id)
|
||||
end
|
||||
|
||||
function Padding:handleEvent(evt)
|
||||
if Element.handleEvent(self, evt) then
|
||||
return true
|
||||
end
|
||||
|
||||
if Event.isClickEvent(evt) then
|
||||
if Event.containsClick(self.element, evt, -self:getPaddingLeft(), -self:getPaddingTop()) then
|
||||
self.element:handleEvent({Event.repositionEvent(evt, -self:getPaddingLeft(), -self:getPaddingTop())})
|
||||
end
|
||||
else
|
||||
return self.element:handleEvent(evt)
|
||||
end
|
||||
return false
|
||||
end
|
||||
|
||||
return Padding
|
9
gfx/text.lua
Normal file
9
gfx/text.lua
Normal file
@ -0,0 +1,9 @@
|
||||
local Element = require("element")
|
||||
local Text = Element:new{ text = "" }
|
||||
|
||||
function Text:draw()
|
||||
Element.draw(self)
|
||||
self.render.window.write(self.text)
|
||||
end
|
||||
|
||||
return Text
|
Loading…
x
Reference in New Issue
Block a user