local color = require("color")
local config = require("config")
local core = require("core")
local screen = require("screen")
local drawer = {}
local fbsd_brand
local none
local menu_name_handlers
local branddefs
local logodefs
local brand_position
local logo_position
local menu_position
local frame_size
local default_shift
local shift
if gfx == nil then
gfx = {}
gfx.term_drawrect = loader.term_drawrect
gfx.term_putimage = loader.term_putimage
end
local function menuEntryName(drawing_menu, entry)
local name_handler = menu_name_handlers[entry.entry_type]
if name_handler ~= nil then
return name_handler(drawing_menu, entry)
end
if type(entry.name) == "function" then
return entry.name()
end
return entry.name
end
local function processFile(gfxname)
if gfxname == nil then
return false, "Missing filename"
end
local ret = try_include('gfx-' .. gfxname)
if ret == nil then
return false, "Failed to include gfx-" .. gfxname
end
if type(ret) ~= "table" then
return true
end
for gfxtype, def in pairs(ret) do
if gfxtype == "brand" then
drawer.addBrand(gfxname, def)
elseif gfxtype == "logo" then
drawer.addLogo(gfxname, def)
else
return false, "Unknown graphics type '" .. gfxtype ..
"'"
end
end
return true
end
local function adapt_fb_shim(def)
if not def.ascii then
def.ascii = {
image = def.graphic,
requires_color = def.requires_color,
shift = def.shift,
}
end
if def.image then
assert(not def.fb,
"Unrecognized graphic definition format")
def.fb = {
image = def.image,
width = def.image_rl,
shift = def.shift,
}
end
def.adapted = true
return def
end
local function getBranddef(brand)
if brand == nil then
return nil
end
local branddef = branddefs[brand]
if branddef == nil then
local res, err = processFile(brand)
if not res then
try_include('brand-' .. brand)
if branddefs[brand] == nil then
print(err)
return nil
end
end
branddef = branddefs[brand]
elseif not branddef.adapted then
adapt_fb_shim(branddef)
end
return branddef
end
local function getLogodef(logo)
if logo == nil then
return nil
end
local logodef = logodefs[logo]
if logodef == nil then
local res, err = processFile(logo)
if not res then
try_include('logo-' .. logo)
if logodefs[logo] == nil then
print(err)
return nil
end
end
logodef = logodefs[logo]
elseif not logodef.adapted then
adapt_fb_shim(logodef)
end
return logodef
end
local function draw(x, y, logo)
for i = 1, #logo do
screen.setcursor(x, y + i - 1)
printc(logo[i])
end
end
local function drawmenu(menudef)
local x = menu_position.x
local y = menu_position.y
if string.lower(loader.getenv("loader_menu") or "") == "none" then
return
end
x = x + shift.x
y = y + shift.y
local alias_table = {}
local entry_num = 0
local menu_entries = menudef.entries
local effective_line_num = 0
if type(menu_entries) == "function" then
menu_entries = menu_entries()
end
for _, e in ipairs(menu_entries) do
if e.visible ~= nil and not e.visible() then
goto continue
end
effective_line_num = effective_line_num + 1
if e.entry_type ~= core.MENU_SEPARATOR then
entry_num = entry_num + 1
screen.setcursor(x, y + effective_line_num)
printc(entry_num .. ". " .. menuEntryName(menudef, e))
alias_table[tostring(entry_num)] = e
if e.alias ~= nil then
for _, a in ipairs(e.alias) do
alias_table[a] = e
end
end
else
screen.setcursor(x, y + effective_line_num)
printc(menuEntryName(menudef, e))
end
::continue::
end
return alias_table
end
local function defaultframe()
if core.isSerialConsole() then
return "ascii"
end
return "double"
end
local function gfxenabled()
return (loader.getenv("loader_gfx") or "yes"):lower() ~= "no"
end
local function gfxcapable()
return core.isFramebufferConsole() and gfx.term_putimage
end
local function drawframe()
local x = menu_position.x - 3
local y = menu_position.y - 1
local w = frame_size.w
local h = frame_size.h
local framestyle = loader.getenv("loader_menu_frame") or defaultframe()
local framespec = drawer.frame_styles[framestyle]
if framespec == nil then
return false
end
local hl = framespec.horizontal
local vl = framespec.vertical
local tl = framespec.top_left
local bl = framespec.bottom_left
local tr = framespec.top_right
local br = framespec.bottom_right
x = x + shift.x
y = y + shift.y
if gfxenabled() and gfxcapable() then
gfx.term_drawrect(x, y, x + w, y + h)
return true
end
screen.setcursor(x, y); printc(tl)
screen.setcursor(x, y + h); printc(bl)
screen.setcursor(x + w, y); printc(tr)
screen.setcursor(x + w, y + h); printc(br)
screen.setcursor(x + 1, y)
for _ = 1, w - 1 do
printc(hl)
end
screen.setcursor(x + 1, y + h)
for _ = 1, w - 1 do
printc(hl)
end
for i = 1, h - 1 do
screen.setcursor(x, y + i)
printc(vl)
screen.setcursor(x + w, y + i)
printc(vl)
end
return true
end
local function drawbox()
local x = menu_position.x - 3
local y = menu_position.y - 1
local w = frame_size.w
local menu_header = loader.getenv("loader_menu_title") or
"Welcome to FreeBSD"
local menu_header_align = loader.getenv("loader_menu_title_align")
local menu_header_x
if string.lower(loader.getenv("loader_menu") or "") == "none" then
return
end
x = x + shift.x
y = y + shift.y
if drawframe(x, y, w) == false then
return
end
if menu_header_align ~= nil then
menu_header_align = menu_header_align:lower()
if menu_header_align == "left" then
menu_header_x = x + 1
elseif menu_header_align == "right" then
menu_header_x = x + w - #menu_header
end
end
if menu_header_x == nil then
menu_header_x = x + (w // 2) - (#menu_header // 2)
end
screen.setcursor(menu_header_x - 1, y)
if menu_header ~= "" then
printc(" " .. menu_header .. " ")
end
end
local function drawbrand()
local x = tonumber(loader.getenv("loader_brand_x")) or
brand_position.x
local y = tonumber(loader.getenv("loader_brand_y")) or
brand_position.y
local branddef = getBranddef(loader.getenv("loader_brand"))
if branddef == nil then
branddef = getBranddef(drawer.default_brand)
end
local graphic = branddef.ascii.image
x = x + shift.x
y = y + shift.y
local gfx_requested = branddef.fb and gfxenabled()
if gfx_requested and gfxcapable() then
if branddef.fb.shift then
x = x + (branddef.fb.shift.x or 0)
y = y + (branddef.fb.shift.y or 0)
end
if gfx.term_putimage(branddef.fb.image, x, y, 0, 7, 0) then
return true
end
elseif branddef.ascii.shift then
x = x + (branddef.ascii.shift.x or 0)
y = y + (branddef.ascii.shift.y or 0)
end
draw(x, y, graphic)
end
local function drawlogo()
local x = tonumber(loader.getenv("loader_logo_x")) or
logo_position.x
local y = tonumber(loader.getenv("loader_logo_y")) or
logo_position.y
local logo = loader.getenv("loader_logo")
local colored = color.isEnabled()
local logodef = getLogodef(logo)
if logodef == nil or logodef.ascii == nil or
(not colored and logodef.ascii.requires_color) then
if colored then
logodef = getLogodef(drawer.default_color_logodef)
else
logodef = getLogodef(drawer.default_bw_logodef)
end
if logodef == nil then
logodef = getLogodef(drawer.default_fallback_logodef)
end
end
if logodef and logodef.ascii.image == none then
shift = logodef.shift
else
shift = default_shift
end
x = x + shift.x
y = y + shift.y
local gfx_requested = logodef.fb and gfxenabled()
if gfx_requested and gfxcapable() then
local y1 = logodef.fb.width or 15
if logodef.fb.shift then
x = x + (logodef.fb.shift.x or 0)
y = y + (logodef.fb.shift.y or 0)
end
if gfx.term_putimage(logodef.fb.image, x, y, 0, y + y1, 0) then
return true
end
elseif logodef.ascii.shift then
x = x + (logodef.ascii.shift.x or 0)
y = y + (logodef.ascii.shift.y or 0)
end
draw(x, y, logodef.ascii.image)
end
local function drawitem(func)
local console = loader.getenv("console")
for c in string.gmatch(console, "%w+") do
loader.setenv("console", c)
func()
end
loader.setenv("console", console)
end
fbsd_brand = {
" ______ ____ _____ _____ ",
" | ____| | _ \\ / ____| __ \\ ",
" | |___ _ __ ___ ___ | |_) | (___ | | | |",
" | ___| '__/ _ \\/ _ \\| _ < \\___ \\| | | |",
" | | | | | __/ __/| |_) |____) | |__| |",
" | | | | | | || | | |",
" |_| |_| \\___|\\___||____/|_____/|_____/ "
}
none = {""}
menu_name_handlers = {
[core.MENU_SEPARATOR] = function(_, entry)
if entry.name ~= nil then
if type(entry.name) == "function" then
return entry.name()
end
return entry.name
end
return ""
end,
[core.MENU_CAROUSEL_ENTRY] = function(_, entry)
local carid = entry.carousel_id
local caridx = config.getCarouselIndex(carid)
local choices = entry.items
if type(choices) == "function" then
choices = choices()
end
if #choices < caridx then
caridx = 1
end
return entry.name(caridx, choices[caridx], choices)
end,
}
branddefs = {
["fbsd"] = {
ascii = {
image = fbsd_brand,
},
fb = {
image = "/boot/images/freebsd-brand-rev.png",
},
},
["none"] = {
ascii = { image = none },
},
}
logodefs = {
["tribute"] = {
ascii = {
image = fbsd_brand,
},
},
["tributebw"] = {
ascii = {
image = fbsd_brand,
},
},
["none"] = {
ascii = {
image = none,
},
shift = {x = 17, y = 0},
},
}
brand_position = {x = 2, y = 1}
logo_position = {x = 40, y = 10}
menu_position = {x = 5, y = 10}
frame_size = {w = 39, h = 14}
default_shift = {x = 0, y = 0}
shift = default_shift
drawer.default_brand = 'fbsd'
drawer.default_color_logodef = 'orb'
drawer.default_bw_logodef = 'orbbw'
drawer.default_fallback_logodef = 'none'
function drawer.addBrand(name, def)
branddefs[name] = adapt_fb_shim(def)
end
function drawer.addLogo(name, def)
logodefs[name] = adapt_fb_shim(def)
end
drawer.frame_styles = {
["ascii"] = {
horizontal = "-",
vertical = "|",
top_left = "+",
bottom_left = "+",
top_right = "+",
bottom_right = "+",
},
}
if core.hasUnicode() then
drawer.frame_styles["single"] = {
horizontal = "\xE2\x94\x80",
vertical = "\xE2\x94\x82",
top_left = "\xE2\x94\x8C",
bottom_left = "\xE2\x94\x94",
top_right = "\xE2\x94\x90",
bottom_right = "\xE2\x94\x98",
}
drawer.frame_styles["double"] = {
horizontal = "\xE2\x95\x90",
vertical = "\xE2\x95\x91",
top_left = "\xE2\x95\x94",
bottom_left = "\xE2\x95\x9A",
top_right = "\xE2\x95\x97",
bottom_right = "\xE2\x95\x9D",
}
else
drawer.frame_styles["single"] = {
horizontal = "\xC4",
vertical = "\xB3",
top_left = "\xDA",
bottom_left = "\xC0",
top_right = "\xBF",
bottom_right = "\xD9",
}
drawer.frame_styles["double"] = {
horizontal = "\xCD",
vertical = "\xBA",
top_left = "\xC9",
bottom_left = "\xC8",
top_right = "\xBB",
bottom_right = "\xBC",
}
end
function drawer.drawscreen(menudef)
drawitem(drawlogo)
drawitem(drawbrand)
drawitem(drawbox)
return drawmenu(menudef)
end
return drawer