xref: /freebsd/stand/lua/drawer.lua (revision 60fde7ce5d7bf5d94290720ea53db5701ab406a8)
1--
2-- SPDX-License-Identifier: BSD-2-Clause-FreeBSD
3--
4-- Copyright (c) 2015 Pedro Souza <pedrosouza@freebsd.org>
5-- Copyright (c) 2018 Kyle Evans <kevans@FreeBSD.org>
6-- All rights reserved.
7--
8-- Redistribution and use in source and binary forms, with or without
9-- modification, are permitted provided that the following conditions
10-- are met:
11-- 1. Redistributions of source code must retain the above copyright
12--    notice, this list of conditions and the following disclaimer.
13-- 2. Redistributions in binary form must reproduce the above copyright
14--    notice, this list of conditions and the following disclaimer in the
15--    documentation and/or other materials provided with the distribution.
16--
17-- THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18-- ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19-- IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20-- ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
21-- FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22-- DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23-- OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24-- HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25-- LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26-- OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27-- SUCH DAMAGE.
28--
29-- $FreeBSD$
30--
31
32local color = require("color")
33local config = require("config")
34local core = require("core")
35local screen = require("screen")
36
37local drawer = {}
38
39local fbsd_logo
40local beastie_color
41local beastie
42local fbsd_logo_v
43local orb_color
44local orb
45local none
46
47local function menuEntryName(drawing_menu, entry)
48	local name_handler = drawer.menu_name_handlers[entry.entry_type]
49
50	if name_handler ~= nil then
51		return name_handler(drawing_menu, entry)
52	end
53	if type(entry.name) == "function" then
54		return entry.name()
55	end
56	return entry.name
57end
58
59fbsd_logo = {
60"  ______               ____   _____ _____  ",
61" |  ____|             |  _ \\ / ____|  __ \\ ",
62" | |___ _ __ ___  ___ | |_) | (___ | |  | |",
63" |  ___| '__/ _ \\/ _ \\|  _ < \\___ \\| |  | |",
64" | |   | | |  __/  __/| |_) |____) | |__| |",
65" | |   | | |    |    ||     |      |      |",
66" |_|   |_|  \\___|\\___||____/|_____/|_____/ "
67}
68
69beastie_color = {
70"               \027[31m,        ,",
71"              /(        )`",
72"              \\ \\___   / |",
73"              /- \027[37m_\027[31m  `-/  '",
74"             (\027[37m/\\/ \\\027[31m \\   /\\",
75"             \027[37m/ /   |\027[31m `    \\",
76"             \027[34mO O   \027[37m) \027[31m/    |",
77"             \027[37m`-^--'\027[31m`<     '",
78"            (_.)  _  )   /",
79"             `.___/`    /",
80"               `-----' /",
81"  \027[33m<----.\027[31m     __ / __   \\",
82"  \027[33m<----|====\027[31mO)))\027[33m==\027[31m) \\) /\027[33m====|",
83"  \027[33m<----'\027[31m    `--' `.__,' \\",
84"               |        |",
85"                \\       /       /\\",
86"           \027[36m______\027[31m( (_  / \\______/",
87"         \027[36m,'  ,-----'   |",
88"         `--{__________)\027[37m"
89}
90
91beastie = {
92"               ,        ,",
93"              /(        )`",
94"              \\ \\___   / |",
95"              /- _  `-/  '",
96"             (/\\/ \\ \\   /\\",
97"             / /   | `    \\",
98"             O O   ) /    |",
99"             `-^--'`<     '",
100"            (_.)  _  )   /",
101"             `.___/`    /",
102"               `-----' /",
103"  <----.     __ / __   \\",
104"  <----|====O)))==) \\) /====|",
105"  <----'    `--' `.__,' \\",
106"               |        |",
107"                \\       /       /\\",
108"           ______( (_  / \\______/",
109"         ,'  ,-----'   |",
110"         `--{__________)"
111}
112
113fbsd_logo_v = {
114"  ______",
115" |  ____| __ ___  ___ ",
116" | |__ | '__/ _ \\/ _ \\",
117" |  __|| | |  __/  __/",
118" | |   | | |    |    |",
119" |_|   |_|  \\___|\\___|",
120"  ____   _____ _____",
121" |  _ \\ / ____|  __ \\",
122" | |_) | (___ | |  | |",
123" |  _ < \\___ \\| |  | |",
124" | |_) |____) | |__| |",
125" |     |      |      |",
126" |____/|_____/|_____/"
127}
128
129orb_color = {
130"  \027[31m```                        \027[31;1m`\027[31m",
131" s` `.....---...\027[31;1m....--.```   -/\027[31m",
132" +o   .--`         \027[31;1m/y:`      +.\027[31m",
133"  yo`:.            \027[31;1m:o      `+-\027[31m",
134"   y/               \027[31;1m-/`   -o/\027[31m",
135"  .-                  \027[31;1m::/sy+:.\027[31m",
136"  /                     \027[31;1m`--  /\027[31m",
137" `:                          \027[31;1m:`\027[31m",
138" `:                          \027[31;1m:`\027[31m",
139"  /                          \027[31;1m/\027[31m",
140"  .-                        \027[31;1m-.\027[31m",
141"   --                      \027[31;1m-.\027[31m",
142"    `:`                  \027[31;1m`:`",
143"      \027[31;1m.--             `--.",
144"         .---.....----.\027[37m"
145}
146
147orb = {
148"  ```                        `",
149" s` `.....---.......--.```   -/",
150" +o   .--`         /y:`      +.",
151"  yo`:.            :o      `+-",
152"   y/               -/`   -o/",
153"  .-                  ::/sy+:.",
154"  /                     `--  /",
155" `:                          :`",
156" `:                          :`",
157"  /                          /",
158"  .-                        -.",
159"   --                      -.",
160"    `:`                  `:`",
161"      .--             `--.",
162"         .---.....----."
163}
164
165none = {""}
166
167-- Module exports
168drawer.menu_name_handlers = {
169	-- Menu name handlers should take the menu being drawn and entry being
170	-- drawn as parameters, and return the name of the item.
171	-- This is designed so that everything, including menu separators, may
172	-- have their names derived differently. The default action for entry
173	-- types not specified here is to use entry.name directly.
174	[core.MENU_SEPARATOR] = function(_, entry)
175		if entry.name ~= nil then
176			if type(entry.name) == "function" then
177				return entry.name()
178			end
179			return entry.name
180		end
181		return ""
182	end,
183	[core.MENU_CAROUSEL_ENTRY] = function(_, entry)
184		local carid = entry.carousel_id
185		local caridx = config.getCarouselIndex(carid)
186		local choices = entry.items
187		if type(choices) == "function" then
188			choices = choices()
189		end
190		if #choices < caridx then
191			caridx = 1
192		end
193		return entry.name(caridx, choices[caridx], choices)
194	end,
195}
196
197drawer.brand_position = {x = 2, y = 1}
198drawer.logo_position = {x = 46, y = 4}
199drawer.menu_position = {x = 5, y = 10}
200drawer.frame_size = {w = 42, h = 13}
201drawer.default_shift = {x = 0, y = 0}
202drawer.shift = drawer.default_shift
203
204drawer.branddefs = {
205	-- Indexed by valid values for loader_brand in loader.conf(5). Valid
206	-- keys are: graphic (table depicting graphic)
207	["fbsd"] = {
208		graphic = fbsd_logo,
209	},
210	["none"] = {
211		graphic = none,
212	},
213}
214
215drawer.logodefs = {
216	-- Indexed by valid values for loader_logo in loader.conf(5). Valid keys
217	-- are: requires_color (boolean), graphic (table depicting graphic), and
218	-- shift (table containing x and y).
219	["beastie"] = {
220		requires_color = true,
221		graphic = beastie_color,
222	},
223	["beastiebw"] = {
224		graphic = beastie,
225	},
226	["fbsdbw"] = {
227		graphic = fbsd_logo_v,
228		shift = {x = 5, y = 4},
229	},
230	["orb"] = {
231		requires_color = true,
232		graphic = orb_color,
233		shift = {x = 2, y = 4},
234	},
235	["orbbw"] = {
236		graphic = orb,
237		shift = {x = 2, y = 4},
238	},
239	["tribute"] = {
240		graphic = fbsd_logo,
241	},
242	["tributebw"] = {
243		graphic = fbsd_logo,
244	},
245	["none"] = {
246		graphic = none,
247		shift = {x = 17, y = 0},
248	},
249}
250
251drawer.frame_styles = {
252	-- Indexed by valid values for loader_menu_frame in loader.conf(5).
253	-- All of the keys appearing below must be set for any menu frame style
254	-- added to drawer.frame_styles.
255	["ascii"] = {
256		horizontal	= "-",
257		vertical	= "|",
258		top_left	= "+",
259		bottom_left	= "+",
260		top_right	= "+",
261		bottom_right	= "+",
262	},
263	["single"] = {
264		horizontal	= "\xC4",
265		vertical	= "\xB3",
266		top_left	= "\xDA",
267		bottom_left	= "\xC0",
268		top_right	= "\xBF",
269		bottom_right	= "\xD9",
270	},
271	["double"] = {
272		horizontal	= "\xCD",
273		vertical	= "\xBA",
274		top_left	= "\xC9",
275		bottom_left	= "\xC8",
276		top_right	= "\xBB",
277		bottom_right	= "\xBC",
278	},
279}
280
281function drawer.drawscreen(menu_opts)
282	-- drawlogo() must go first.
283	-- it determines the positions of other elements
284	drawer.drawlogo()
285	drawer.drawbrand()
286	drawer.drawbox()
287	return drawer.drawmenu(menu_opts)
288end
289
290function drawer.drawmenu(menudef)
291	local x = drawer.menu_position.x
292	local y = drawer.menu_position.y
293
294	x = x + drawer.shift.x
295	y = y + drawer.shift.y
296
297	-- print the menu and build the alias table
298	local alias_table = {}
299	local entry_num = 0
300	local menu_entries = menudef.entries
301	local effective_line_num = 0
302	if type(menu_entries) == "function" then
303		menu_entries = menu_entries()
304	end
305	for _, e in ipairs(menu_entries) do
306		-- Allow menu items to be conditionally visible by specifying
307		-- a visible function.
308		if e.visible ~= nil and not e.visible() then
309			goto continue
310		end
311		effective_line_num = effective_line_num + 1
312		if e.entry_type ~= core.MENU_SEPARATOR then
313			entry_num = entry_num + 1
314			screen.setcursor(x, y + effective_line_num)
315
316			printc(entry_num .. ". " .. menuEntryName(menudef, e))
317
318			-- fill the alias table
319			alias_table[tostring(entry_num)] = e
320			if e.alias ~= nil then
321				for _, a in ipairs(e.alias) do
322					alias_table[a] = e
323				end
324			end
325		else
326			screen.setcursor(x, y + effective_line_num)
327			printc(menuEntryName(menudef, e))
328		end
329		::continue::
330	end
331	return alias_table
332end
333
334function drawer.drawbox()
335	local x = drawer.menu_position.x - 3
336	local y = drawer.menu_position.y - 1
337	local w = drawer.frame_size.w
338	local h = drawer.frame_size.h
339
340	local framestyle = loader.getenv("loader_menu_frame") or "double"
341	local framespec = drawer.frame_styles[framestyle]
342	-- If we don't have a framespec for the current frame style, just don't
343	-- draw a box.
344	if framespec == nil then
345		return
346	end
347
348	local hl = framespec.horizontal
349	local vl = framespec.vertical
350
351	local tl = framespec.top_left
352	local bl = framespec.bottom_left
353	local tr = framespec.top_right
354	local br = framespec.bottom_right
355
356	x = x + drawer.shift.x
357	y = y + drawer.shift.y
358
359	screen.setcursor(x, y); printc(tl)
360	screen.setcursor(x, y + h); printc(bl)
361	screen.setcursor(x + w, y); printc(tr)
362	screen.setcursor(x + w, y + h); printc(br)
363
364	screen.setcursor(x + 1, y)
365	for _ = 1, w - 1 do
366		printc(hl)
367	end
368
369	screen.setcursor(x + 1, y + h)
370	for _ = 1, w - 1 do
371		printc(hl)
372	end
373
374	for i = 1, h - 1 do
375		screen.setcursor(x, y + i)
376		printc(vl)
377		screen.setcursor(x + w, y + i)
378		printc(vl)
379	end
380
381	local menu_header = loader.getenv("loader_menu_title") or
382	    "Welcome to FreeBSD"
383	local menu_header_align = loader.getenv("loader_menu_title_align")
384	local menu_header_x
385
386	if menu_header_align ~= nil then
387		menu_header_align = menu_header_align:lower()
388		if menu_header_align == "left" then
389			-- Just inside the left border on top
390			menu_header_x = x + 1
391		elseif menu_header_align == "right" then
392			-- Just inside the right border on top
393			menu_header_x = x + w - #menu_header
394		end
395	end
396	if menu_header_x == nil then
397		menu_header_x = x + (w / 2) - (#menu_header / 2)
398	end
399	screen.setcursor(menu_header_x, y)
400	printc(menu_header)
401end
402
403function drawer.draw(x, y, logo)
404	for i = 1, #logo do
405		screen.setcursor(x, y + i - 1)
406		printc(logo[i])
407	end
408end
409
410function drawer.drawbrand()
411	local x = tonumber(loader.getenv("loader_brand_x")) or
412	    drawer.brand_position.x
413	local y = tonumber(loader.getenv("loader_brand_y")) or
414	    drawer.brand_position.y
415
416	local graphic = drawer.branddefs[loader.getenv("loader_brand")]
417	if graphic == nil then
418		graphic = fbsd_logo
419	end
420
421	x = x + drawer.shift.x
422	y = y + drawer.shift.y
423	drawer.draw(x, y, graphic)
424end
425
426function drawer.drawlogo()
427	local x = tonumber(loader.getenv("loader_logo_x")) or
428	    drawer.logo_position.x
429	local y = tonumber(loader.getenv("loader_logo_y")) or
430	    drawer.logo_position.y
431
432	local logo = loader.getenv("loader_logo")
433	local colored = color.isEnabled()
434
435	-- Lookup
436	local logodef = drawer.logodefs[logo]
437
438	if logodef == nil or logodef.graphic == nil or
439	    (not colored and logodef.requires_color) then
440		-- Choose a sensible default
441		if colored then
442			logodef = drawer.logodefs["orb"]
443		else
444			logodef = drawer.logodefs["orbbw"]
445		end
446	end
447
448	if logodef ~= nil and logodef.graphic == none then
449		drawer.shift = logodef.shift
450	else
451		drawer.shift = drawer.default_shift
452	end
453
454	x = x + drawer.shift.x
455	y = y + drawer.shift.y
456
457	if logodef ~= nil and logodef.shift ~= nil then
458		x = x + logodef.shift.x
459		y = y + logodef.shift.y
460	end
461
462	drawer.draw(x, y, logodef.graphic)
463end
464
465return drawer
466