1
0
mirror of https://github.com/kepler155c/opus synced 2025-01-03 20:30:28 +00:00

canvas use in UI overhaul

This commit is contained in:
kepler155c@gmail.com 2019-01-30 15:11:41 -05:00
parent 02629e266b
commit 3574d26caa
12 changed files with 137 additions and 147 deletions

View File

@ -317,6 +317,7 @@ end
function Manager:setActivePage(page) function Manager:setActivePage(page)
page.parent.currentPage = page page.parent.currentPage = page
page.parent.canvas = page.canvas
end end
function Manager:setPage(pageOrName, ...) function Manager:setPage(pageOrName, ...)
@ -330,7 +331,6 @@ function Manager:setPage(pageOrName, ...)
if page == currentPage then if page == currentPage then
page:draw() page:draw()
else else
local needSync
if currentPage then if currentPage then
if currentPage.focused then if currentPage.focused then
currentPage.focused.focused = false currentPage.focused.focused = false
@ -338,20 +338,16 @@ function Manager:setPage(pageOrName, ...)
end end
currentPage:disable() currentPage:disable()
page.previousPage = currentPage page.previousPage = currentPage
else
needSync = true
end end
self:setActivePage(page) self:setActivePage(page)
page:clear(page.backgroundColor) --page:clear(page.backgroundColor)
page:enable(...) page:enable(...)
page:draw() page:draw()
if page.focused then if page.focused then
page.focused.focused = true page.focused.focused = true
page.focused:focus() page.focused:focus()
end end
if needSync then page:sync()
page:sync() -- first time a page has been set
end
end end
end end
@ -607,12 +603,10 @@ function UI.Window:setTextScale(textScale)
end end
function UI.Window:clear(bg, fg) function UI.Window:clear(bg, fg)
if self.enabled then if self.canvas then
if self.canvas then self.canvas:clear(bg or self.backgroundColor, fg or self.textColor)
self.canvas:clear(bg or self.backgroundColor, fg or self.textColor) else
else self:clearArea(1 + self.offx, 1 + self.offy, self.width, self.height, bg)
self:clearArea(1 + self.offx, 1 + self.offy, self.width, self.height, bg)
end
end end
end end
@ -630,18 +624,17 @@ function UI.Window:clearArea(x, y, width, height, bg)
end end
function UI.Window:write(x, y, text, bg, tc) function UI.Window:write(x, y, text, bg, tc)
if self.enabled then bg = bg or self.backgroundColor
bg = bg or self.backgroundColor tc = tc or self.textColor
tc = tc or self.textColor -- TODO: get rid of offx/y - scroll canvas instead
x = x - self.offx x = x - self.offx
y = y - self.offy y = y - self.offy
if y <= self.height and y > 0 then if y <= self.height and y > 0 then
if self.canvas then if self.canvas then
self.canvas:write(x, y, text, bg, tc) self.canvas:write(x, y, text, bg, tc)
else else
self.parent:write( self.parent:write(
self.x + x - 1, self.y + y - 1, tostring(text), bg, tc) self.x + x - 1, self.y + y - 1, tostring(text), bg, tc)
end
end end
end end
end end
@ -933,12 +926,6 @@ function UI.Device:postInit()
self.isColor = self.device.isColor() self.isColor = self.device.isColor()
self.canvas = Canvas({
x = 1, y = 1, width = self.width, height = self.height,
isColor = self.isColor,
})
self.canvas:clear(self.backgroundColor, self.textColor)
UI.devices[self.device.side or 'terminal'] = self UI.devices[self.device.side or 'terminal'] = self
end end
@ -946,6 +933,7 @@ function UI.Device:resize()
self.device.setTextScale(self.textScale) self.device.setTextScale(self.textScale)
self.width, self.height = self.device.getSize() self.width, self.height = self.device.getSize()
self.lines = { } self.lines = { }
-- TODO: resize all pages added to this device
self.canvas:resize(self.width, self.height) self.canvas:resize(self.width, self.height)
self.canvas:clear(self.backgroundColor, self.textColor) self.canvas:clear(self.backgroundColor, self.textColor)
end end
@ -997,11 +985,13 @@ function UI.Device:addTransition(effect, args)
end end
function UI.Device:runTransitions(transitions, canvas) function UI.Device:runTransitions(transitions, canvas)
--[[
for _,t in ipairs(transitions) do for _,t in ipairs(transitions) do
canvas:punch(t.args) -- punch out the effect areas canvas:punch(t.args) -- punch out the effect areas
end end
canvas:blitClipped(self.device) -- and blit the remainder canvas:blitClipped(self.device) -- and blit the remainder
canvas:reset() canvas:reset()
]]
while true do while true do
for _,k in ipairs(Util.keys(transitions)) do for _,k in ipairs(Util.keys(transitions)) do
@ -1028,10 +1018,9 @@ function UI.Device:sync()
self.device.setCursorBlink(false) self.device.setCursorBlink(false)
end end
self.canvas:render(self.device)
if transitions then if transitions then
self:runTransitions(transitions, self.canvas) self:runTransitions(transitions, self.canvas)
else
self.canvas:render(self.device)
end end
if self:getCursorBlink() then if self:getCursorBlink() then
@ -1126,16 +1115,23 @@ UI.Page.defaults = {
function UI.Page:postInit() function UI.Page:postInit()
self.parent = self.parent or UI.defaultDevice self.parent = self.parent or UI.defaultDevice
self.__target = self self.__target = self
self.canvas = Canvas({
x = 1, y = 1, width = self.parent.width, height = self.parent.height,
isColor = self.parent.isColor,
})
self.canvas:clear(self.backgroundColor, self.textColor)
end end
function UI.Page:setParent() function UI.Page:setParent()
UI.Window.setParent(self) UI.Window.setParent(self)
--[[
if self.z then if self.z then
self.canvas = self:addLayer(self.backgroundColor, self.textColor) self.canvas = self:addLayer(self.backgroundColor, self.textColor)
self.canvas:clear(self.backgroundColor, self.textColor) self.canvas:clear(self.backgroundColor, self.textColor)
else else
self.canvas = self.parent.canvas self.canvas = self.parent.canvas
end end
]]
end end
function UI.Page:enable() function UI.Page:enable()
@ -1154,6 +1150,12 @@ function UI.Page:disable()
UI.Window.disable(self) UI.Window.disable(self)
end end
function UI.Page:sync()
if self.enabled then
self.parent:sync()
end
end
function UI.Page:capture(child) function UI.Page:capture(child)
self.__target = child self.__target = child
end end
@ -2293,9 +2295,9 @@ function UI.Tabs:eventHandler(event)
if event.type == 'tab_change' then if event.type == 'tab_change' then
local tab = self:find(event.tab.tabUid) local tab = self:find(event.tab.tabUid)
if event.current > event.last then if event.current > event.last then
tab:addTransition('slideLeft') self.transitionHint = 'slideLeft'
else else
tab:addTransition('slideRight') self.transitionHint = 'slideRight'
end end
for _,child in pairs(self.children) do for _,child in pairs(self.children) do
@ -2310,6 +2312,30 @@ function UI.Tabs:eventHandler(event)
end end
end end
--[[-- Tab --]]--
UI.Tab = class(UI.Window)
UI.Tab.defaults = {
UIElement = 'Tab',
tabTitle = 'tab',
backgroundColor = colors.cyan,
}
function UI.Tab:setParent()
UI.Window.setParent(self)
self.canvas = self:addLayer()
end
function UI.Tab:enable(...)
self.canvas:setVisible(true)
UI.Window.enable(self, ...)
self:addTransition(self.parent.transitionHint or 'slideLeft')
self:focusFirst()
end
function UI.Tab:disable()
self.canvas:setVisible(false)
UI.Window.disable(self)
end
--[[-- Wizard --]]-- --[[-- Wizard --]]--
UI.Wizard = class(UI.Window) UI.Wizard = class(UI.Window)
UI.Wizard.defaults = { UI.Wizard.defaults = {
@ -3047,7 +3073,7 @@ function UI.Chooser:eventHandler(event)
end end
end end
--[[-- Chooser --]]-- --[[-- Checkbox --]]--
UI.Checkbox = class(UI.Window) UI.Checkbox = class(UI.Window)
UI.Checkbox.defaults = { UI.Checkbox.defaults = {
UIElement = 'Checkbox', UIElement = 'Checkbox',
@ -3243,6 +3269,7 @@ UI.Form.defaults = {
values = { }, values = { },
margin = 2, margin = 2,
event = 'form_complete', event = 'form_complete',
cancelEvent = 'form_cancel',
} }
function UI.Form:postInit() function UI.Form:postInit()
self:createForm() self:createForm()
@ -3319,7 +3346,7 @@ function UI.Form:createForm()
table.insert(self.children, UI.Button { table.insert(self.children, UI.Button {
y = -self.margin, x = -7 - self.margin, y = -self.margin, x = -7 - self.margin,
text = 'Cancel', text = 'Cancel',
event = 'form_cancel', event = self.cancelEvent,
}) })
end end
end end
@ -3346,6 +3373,7 @@ function UI.Form:save()
local s, m = self:validateField(child) local s, m = self:validateField(child)
if not s then if not s then
self:setFocus(child) self:setFocus(child)
Sound.play('entity.villager.no', .5)
self:emit({ type = 'form_invalid', message = m, field = child }) self:emit({ type = 'form_invalid', message = m, field = child })
return false return false
end end
@ -3372,7 +3400,7 @@ function UI.Form:eventHandler(event)
if not self:save() then if not self:save() then
return false return false
end end
self:emit({ type = self.event, UIElement = self }) self:emit({ type = self.event, UIElement = self, values = self.values })
else else
return UI.Window.eventHandler(self, event) return UI.Window.eventHandler(self, event)
end end
@ -3380,49 +3408,38 @@ function UI.Form:eventHandler(event)
end end
--[[-- Dialog --]]-- --[[-- Dialog --]]--
UI.Dialog = class(UI.Page) UI.Dialog = class(UI.SlideOut)
UI.Dialog.defaults = { UI.Dialog.defaults = {
UIElement = 'Dialog', UIElement = 'Dialog',
x = 7,
y = 4,
z = 2,
height = 7, height = 7,
textColor = colors.black, textColor = colors.black,
backgroundColor = colors.white, backgroundColor = colors.white,
okEvent ='dialog_ok',
cancelEvent = 'dialog_cancel',
} }
function UI.Dialog:postInit() function UI.Dialog:postInit()
self.titleBar = UI.TitleBar({ previousPage = true, title = self.title }) self.y = -self.height
self.titleBar = UI.TitleBar({ event = self.cancelEvent, title = self.title })
end end
function UI.Dialog:setParent() function UI.Dialog:show(...)
if not self.width then local canvas = self.parent:getCanvas()
self.width = self.parent.width - 11 self.oldPalette = canvas.palette
end canvas:applyPalette(Canvas.darkPalette)
if self.width > self.parent.width then UI.SlideOut.show(self, ...)
self.width = self.parent.width
end
self.x = math.floor((self.parent.width - self.width) / 2) + 1
self.y = math.floor((self.parent.height - self.height) / 2) + 1
UI.Page.setParent(self)
end end
function UI.Dialog:disable() function UI.Dialog:hide(...)
self.previousPage.canvas.palette = self.oldPalette self.parent:getCanvas().palette = self.oldPalette
UI.Page.disable(self) UI.SlideOut.hide(self, ...)
end self.parent:draw()
function UI.Dialog:enable(...)
self.oldPalette = self.previousPage.canvas.palette
self.previousPage.canvas:applyPalette(Canvas.darkPalette)
self:addTransition('grow')
UI.Page.enable(self, ...)
end end
function UI.Dialog:eventHandler(event) function UI.Dialog:eventHandler(event)
if event.type == 'cancel' then if event.type == 'dialog_cancel' then
UI:setPreviousPage() self:hide()
end end
return UI.Page.eventHandler(self, event) return UI.SlideOut.eventHandler(self, event)
end end
--[[-- Image --]]-- --[[-- Image --]]--

View File

@ -7,24 +7,14 @@ function Transition.slideLeft(args)
local easing = args.easing or 'outQuint' local easing = args.easing or 'outQuint'
local pos = { x = args.ex } local pos = { x = args.ex }
local tween = Tween.new(ticks, pos, { x = args.x }, easing) local tween = Tween.new(ticks, pos, { x = args.x }, easing)
local lastScreen = args.canvas:copy()
args.canvas:move(pos.x, args.canvas.y)
return function(device) return function(device)
local finished = tween:update(1) local finished = tween:update(1)
local x = math.floor(pos.x) args.canvas:move(math.floor(pos.x), args.canvas.y)
lastScreen:dirty() args.canvas:dirty()
lastScreen:blit(device, { args.canvas:render(device)
x = args.ex - x + args.x,
y = args.y,
ex = args.ex,
ey = args.ey },
{ x = args.x, y = args.y })
args.canvas:blit(device, {
x = args.x,
y = args.y,
ex = args.ex - x + args.x,
ey = args.ey },
{ x = x, y = args.y })
return not finished return not finished
end end
end end
@ -32,26 +22,16 @@ end
function Transition.slideRight(args) function Transition.slideRight(args)
local ticks = args.ticks or 6 local ticks = args.ticks or 6
local easing = args.easing or'outQuint' local easing = args.easing or'outQuint'
local pos = { x = args.x } local pos = { x = -args.canvas.width }
local tween = Tween.new(ticks, pos, { x = args.ex }, easing) local tween = Tween.new(ticks, pos, { x = 1 }, easing)
local lastScreen = args.canvas:copy()
args.canvas:move(pos.x, args.canvas.y)
return function(device) return function(device)
local finished = tween:update(1) local finished = tween:update(1)
local x = math.floor(pos.x) args.canvas:move(math.floor(pos.x), args.canvas.y)
lastScreen:dirty() args.canvas:dirty()
lastScreen:blit(device, { args.canvas:render(device)
x = args.x,
y = args.y,
ex = args.ex - x + args.x,
ey = args.ey },
{ x = x, y = args.y })
args.canvas:blit(device, {
x = args.ex - x + args.x,
y = args.y,
ex = args.ex,
ey = args.ey },
{ x = args.x, y = args.y })
return not finished return not finished
end end
end end
@ -62,27 +42,13 @@ function Transition.expandUp(args)
local pos = { y = args.ey + 1 } local pos = { y = args.ey + 1 }
local tween = Tween.new(ticks, pos, { y = args.y }, easing) local tween = Tween.new(ticks, pos, { y = args.y }, easing)
return function(device) args.canvas:move(args.x, pos.y)
local finished = tween:update(1)
args.canvas:blit(device, nil, { x = args.x, y = math.floor(pos.y) })
return not finished
end
end
function Transition.grow(args)
local ticks = args.ticks or 3
local easing = args.easing or 'linear'
local tween = Tween.new(ticks,
{ x = args.width / 2 - 1, y = args.height / 2 - 1, w = 1, h = 1 },
{ x = 1, y = 1, w = args.width, h = args.height }, easing)
return function(device) return function(device)
local finished = tween:update(1) local finished = tween:update(1)
local subj = tween.subject args.canvas:move(args.x, math.floor(pos.y))
local rect = { x = math.floor(subj.x), y = math.floor(subj.y) } args.canvas:dirty()
rect.ex = math.floor(rect.x + subj.w - 1) args.canvas:render(device)
rect.ey = math.floor(rect.y + subj.h - 1)
args.canvas:blit(device, rect, { x = args.x + rect.x - 1, y = args.y + rect.y - 1})
return not finished return not finished
end end
end end

View File

@ -72,6 +72,31 @@ local page = UI.Page {
autospace = true, autospace = true,
}, },
}, },
help = UI.SlideOut {
backgroundColor = colors.cyan,
x = 5, ex = -5, height = 8, y = -8,
titleBar = UI.TitleBar {
title = 'Network Help',
event = 'slide_hide',
},
text = UI.TextArea {
x = 2, y = 2,
backgroundColor = colors.cyan,
value = [[
In order to connect to another computer:
1. The target computer must have a password set (run 'password' from the shell prompt).
2. From this computer, click trust and enter the password for that computer.
This only needs to be done once.
]],
},
accelerators = {
q = 'slide_hide',
}
},
notification = UI.Notification { }, notification = UI.Notification { },
accelerators = { accelerators = {
t = 'telnet', t = 'telnet',
@ -180,27 +205,9 @@ function page:eventHandler(event)
sendCommand(t.id, 'shutdown') sendCommand(t.id, 'shutdown')
end end
end end
if event.type == 'help' then if event.type == 'help' then
UI:setPage(UI.Dialog { self.help:show()
title = 'Network Help',
height = 10,
backgroundColor = colors.white,
text = UI.TextArea {
x = 2, y = 2,
backgroundColor = colors.white,
value = [[
In order to connect to another computer:
1. The target computer must have a password set (run 'password' from the shell prompt).
2. From this computer, click trust and enter the password for that computer.
This only needs to be done once.
]],
},
accelerators = {
q = 'cancel',
}
})
elseif event.type == 'ports' then elseif event.type == 'ports' then
self.ports.grid:update() self.ports.grid:update()

View File

@ -106,6 +106,7 @@ local page = UI.Page {
} }
}, },
editor = UI.SlideOut { editor = UI.SlideOut {
y = -12, height = 12,
backgroundColor = colors.cyan, backgroundColor = colors.cyan,
titleBar = UI.TitleBar { titleBar = UI.TitleBar {
title = 'Edit Application', title = 'Edit Application',

View File

@ -8,7 +8,7 @@ UI:configure('System', ...)
local systemPage = UI.Page { local systemPage = UI.Page {
tabs = UI.Tabs { tabs = UI.Tabs {
settings = UI.Window { settings = UI.Tab {
tabTitle = 'Category', tabTitle = 'Category',
grid = UI.Grid { grid = UI.Grid {
y = 2, y = 2,

View File

@ -1,7 +1,7 @@
local Config = require('config') local Config = require('config')
local UI = require('ui') local UI = require('ui')
local aliasTab = UI.Window { local aliasTab = UI.Tab {
tabTitle = 'Aliases', tabTitle = 'Aliases',
description = 'Shell aliases', description = 'Shell aliases',
alias = UI.TextEntry { alias = UI.TextEntry {

View File

@ -4,7 +4,7 @@ local Util = require('util')
local fs = _G.fs local fs = _G.fs
local os = _G.os local os = _G.os
local labelTab = UI.Window { local labelTab = UI.Tab {
tabTitle = 'Label', tabTitle = 'Label',
description = 'Set the computer label', description = 'Set the computer label',
labelText = UI.Text { labelText = UI.Text {

View File

@ -3,7 +3,7 @@ local UI = require('ui')
local device = _G.device local device = _G.device
local tab = UI.Window { local tab = UI.Tab {
tabTitle = 'Network', tabTitle = 'Network',
description = 'Networking options', description = 'Networking options',
form = UI.Form { form = UI.Form {
@ -40,7 +40,7 @@ function tab:enable()
local config = Config.load('os') local config = Config.load('os')
self.form.modem.value = config.wirelessModem or 'auto' self.form.modem.value = config.wirelessModem or 'auto'
UI.Window.enable(self) UI.Tab.enable(self)
end end
function tab:eventHandler(event) function tab:eventHandler(event)

View File

@ -2,7 +2,7 @@ local Security = require('security')
local SHA1 = require('sha1') local SHA1 = require('sha1')
local UI = require('ui') local UI = require('ui')
local passwordTab = UI.Window { local passwordTab = UI.Tab {
tabTitle = 'Password', tabTitle = 'Password',
description = 'Wireless network password', description = 'Wireless network password',
oldPass = UI.TextEntry { oldPass = UI.TextEntry {

View File

@ -2,7 +2,7 @@ local Config = require('config')
local UI = require('ui') local UI = require('ui')
local Util = require('util') local Util = require('util')
local pathTab = UI.Window { local pathTab = UI.Tab {
tabTitle = 'Path', tabTitle = 'Path',
description = 'Set the shell path', description = 'Set the shell path',
tabClose = true, tabClose = true,

View File

@ -3,7 +3,6 @@ local UI = require('ui')
local settings = _G.settings local settings = _G.settings
if settings then if settings then
local values = { } local values = { }
for _,v in pairs(settings.getNames()) do for _,v in pairs(settings.getNames()) do
local value = settings.get(v) local value = settings.get(v)
@ -16,11 +15,11 @@ if settings then
}) })
end end
local settingsTab = UI.Window { local settingsTab = UI.Tab {
tabTitle = 'Settings', tabTitle = 'Settings',
description = 'Computercraft configurable settings', description = 'Computercraft configurable settings',
grid = UI.Grid { grid = UI.Grid {
y = 1, y = 2,
values = values, values = values,
autospace = true, autospace = true,
sortColumn = 'name', sortColumn = 'name',

View File

@ -9,7 +9,7 @@ if turtle then
local values = { } local values = { }
Config.load('gps', values.home and { values.home } or { }) Config.load('gps', values.home and { values.home } or { })
local gpsTab = UI.Window { local gpsTab = UI.Tab {
tabTitle = 'GPS', tabTitle = 'GPS',
labelText = UI.Text { labelText = UI.Text {
x = 3, y = 2, x = 3, y = 2,