2022-03-25 11:25:27 +08:00
< %#
Material is a clean HTML5 theme for LuCI. It is based on luci-theme-bootstrap and MUI
luci-theme-material
Copyright 2015 Lutty Yang < lutty @ wcan . in >
luci-theme-neobird
Copyright 2021 2smile < thinktip @ gmail . com >
2023-02-28 22:22:45 +08:00
luci-theme-design
Copyright 2023 2smile < gngppz @ gmail . com >
2022-03-25 11:25:27 +08:00
Have a bug? Please create an issue here on GitHub!
2023-02-28 22:22:45 +08:00
https://github.com/gngpp/luci-theme-design
2022-03-25 11:25:27 +08:00
luci-theme-bootstrap:
Copyright 2008 Steven Barth < steven @ midlink . org >
Copyright 2008 Jo-Philipp Wich < jow @ openwrt . org >
Copyright 2012 David Menting < david @ nut-bolt . nl >
MUI:
https://github.com/muicss/mui
Licensed to the public under the Apache License 2.0
-%>
< %
local ver = require "luci.version"
local sys = require "luci.sys"
local util = require "luci.util"
local http = require "luci.http"
local disp = require "luci.dispatcher"
2023-03-03 00:27:13 +08:00
local uci = require 'luci.model.uci'.cursor()
local fs = require "nixio.fs"
2022-03-25 11:25:27 +08:00
local boardinfo = util.ubus("system", "board")
local boardinfo={}
boardinfo.hostname=sys.hostname()
local request = disp.context.path
local request2 = disp.context.request
local category = request[1]
local cattree = category and disp.node(category)
local leaf = request2[#request2]
local tree = disp.node()
local node = disp.context.dispatched
local categories = disp.node_childs(tree)
local c = tree
local i, r
-- tag all nodes leading to this page
for i, r in ipairs(request) do
if c.nodes and c.nodes[r] then
c = c.nodes[r]
c._menu_selected = true
end
end
-- send as HTML5
http.prepare_content("text/html")
local function nodeurl(prefix, name, query)
local url = controller .. prefix .. name .. "/"
if query then
url = url .. http.build_querystring(query)
end
return pcdata(url)
end
local function subtree(prefix, node, level)
if not level then
level = 1
end
local childs = disp.node_childs(node)
if #childs > 0 then
if level > 2 then
%>
< ul class = "tabs" >
< %
end
local selected_node
local selected_name
local i, v
for i, v in ipairs(childs) do
local nnode = node.nodes[v]
if nnode._menu_selected then
selected_node = nnode
selected_name = v
end
if level > 2 then
%>
< li class = "tabmenu-item-<%=v%><%- if nnode._menu_selected or (node.leaf and v == leaf) then %> active<% end %>" >
< a href = "<%=nodeurl(prefix, v, nnode.query)%>" > < %=striptags(translate(nnode.title))%>< / a >
< / li >
< % end
end
if level > 2 then
%>
< / ul >
< % end
if selected_node then
subtree(prefix .. selected_name .. "/", selected_node, level + 1)
end
end
2023-03-03 00:27:13 +08:00
2022-03-25 11:25:27 +08:00
end
2023-03-03 00:27:13 +08:00
-- Custom settings
local mode = 'normal'
2023-03-06 23:39:07 +08:00
local navbar = 'display'
2023-03-03 00:27:13 +08:00
local navbar_proxy = 'openclash'
if fs.access('/etc/config/design') then
mode = uci:get_first('design', 'global', 'mode')
2023-03-06 23:39:07 +08:00
navbar = uci:get_first('design', 'global', 'navbar')
2023-03-03 00:27:13 +08:00
navbar_proxy = uci:get_first('design', 'global', 'navbar_proxy')
end
2022-03-25 11:25:27 +08:00
-%>
<!DOCTYPE html>
< html lang = "<%=luci.i18n.context.lang%>" >
< head >
< meta charset = "utf-8" >
< meta name = "viewport" content = "width=device-width,initial-scale=1,maximum-scale=1,user-scalable=0,viewport-fit=cover" / >
< link rel = "manifest" href = "<%=media%>/manifest.json" >
< title > < %=striptags( (boardinfo.hostname or "?") .. ( (node and node.title) and ' - ' .. translate(node.title) or '')) %> - LuCI< / title >
< meta name = "format-detection" content = "telephone=no, email=no" / >
< meta name = "x5-fullscreen" content = "true" >
< meta name = "full-screen" content = "yes" >
< meta name = "x5-page-mode" content = "app" >
< meta name = "browsermode" content = "application" >
< meta name = "msapplication-tap-highlight" content = "no" >
< meta name = "apple-mobile-web-app-status-bar-style" content = "default" / >
< meta name = "application-name" content = "<%=striptags( (boardinfo.hostname or " ? " ) ) % > ">
< meta name = "apple-mobile-web-app-title" content = "<%=striptags( (boardinfo.hostname or " ? " ) ) % > ">
2023-02-28 22:22:45 +08:00
< meta name = "msapplication-TileImage" content = "<%=media%>/images/android-chrome-512x512.png" / >
< link rel = "icon" href = "<%=media%>/images/android-chrome-192x192.png" sizes = "192x192" >
< link rel = "apple-touch-icon" sizes = "192x192" href = "<%=media%>/images/apple-touch-icon.png" >
2022-03-25 11:25:27 +08:00
< meta name = "apple-mobile-web-app-capable" content = "yes" >
< meta name = "mobile-web-app-capable" content = "yes" >
< meta name = "apple-touch-fullscreen" content = "yes" / >
< link rel = "stylesheet" href = "<%=media%>/css/style.css?v=<%= ver.luciversion %>" >
2023-03-03 00:27:13 +08:00
< meta name = "theme-color" media = "(prefers-color-scheme: light)" content = "white" >
< meta name = "theme-color" media = "(prefers-color-scheme: dark)" content = "black" >
2022-03-25 11:25:27 +08:00
< link rel = "shortcut icon" href = "<%=media%>/favicon.ico" >
< % if node and node.css then %>
< link rel = "stylesheet" href = "<%=resource%>/<%=node.css%>" >
< % end -%>
< % if css then %>
< style title = "text/css" >
< %-= css %>
< / style >
< % end -%>
2023-03-18 23:35:15 +08:00
< script src = "<%=media%>/js/xhr.js?v=<%= ver.luciversion %>" > < / script >
2023-02-28 22:22:45 +08:00
< style title = "text/css" id = "global-scroll" >
::-webkit-scrollbar {
width: 4px
}
::-webkit-scrollbar-thumb {
background: var(--scrollbarColor) ;
border-radius: 2px;
}
< / style >
2022-03-25 11:25:27 +08:00
< / head >
2023-03-16 23:10:23 +08:00
< body class = "lang_<%=luci.i18n.context.lang%> <%- if node then %><%= striptags( node.title ) %><%- end %> <% if luci.dispatcher.context.authsession then %>logged-in<% end %>"
< % if mode ~= 'normal' then %>
data-theme = "< %= mode %>"
< % end -%>
>
2022-03-25 11:25:27 +08:00
< header >
2023-03-06 23:39:07 +08:00
< % if mode == 'normal' then %>
2023-03-18 23:35:15 +08:00
< script >
function setTheme() {
let color_scheme = window.matchMedia('(prefers-color-scheme: dark)');
document.body.setAttribute('data-theme', color_scheme.matches & & 'dark');
}
setTheme();
window.matchMedia & & window.matchMedia('(prefers-color-scheme: dark)').addEventListener('change', setTheme);
< / script >
2023-03-06 23:39:07 +08:00
< % end -%>
2023-03-03 00:27:13 +08:00
2023-03-06 23:39:07 +08:00
< % if navbar == 'display' then %>
< style >
:root {
--vssrBottom: 50px;
}
< / style >
2022-03-25 11:25:27 +08:00
< div class = "navbar" >
2023-03-06 23:39:07 +08:00
< % else %>
< style >
:root {
--vssrBottom: 0px;
}
< / style >
< div class = "navbar" style = "display: none;" >
< % end -%>
2022-03-25 11:25:27 +08:00
< a href = "/cgi-bin/luci/admin/status/overview" > < img src = "<%=media%>/images/home.png" / > < / a >
2023-03-03 00:27:13 +08:00
< % if navbar_proxy == 'openclash' then %>
< a href = "/cgi-bin/luci/admin/services/<%=navbar_proxy%>" > < img src = "<%=media%>/images/<%=navbar_proxy%>.png" / > < / a >
< % else %>
< a href = "/cgi-bin/luci/admin/services/<%=navbar_proxy%>" > < img src = "<%=media%>/images/ssr.png" / > < / a >
< % end -%>
2022-03-25 11:25:27 +08:00
< a href = "/cgi-bin/luci/admin/network/network" > < img src = "<%=media%>/images/link.png" / > < / a >
< a href = "/cgi-bin/luci/admin/status/realtime" > < img src = "<%=media%>/images/rank.png" / > < / a >
< a href = "/cgi-bin/luci/admin/system/admin" > < img src = "<%=media%>/images/user.png" / > < / a >
< / div >
2023-03-03 00:27:13 +08:00
2022-03-25 11:25:27 +08:00
< div class = "container" >
2023-02-28 22:22:45 +08:00
< span class = "showSide" > < / span >
< a class = "brand" href = "#" > < %=boardinfo.hostname or "OpenWrt"%>< / a >
2022-03-25 11:25:27 +08:00
< div class = "pull-right" >
< %
-- calculate the number of unsaved changes
if tree.nodes[category] and tree.nodes[category].ucidata then
local ucichanges = 0
for i, j in pairs(require("luci.model.uci").cursor():changes()) do
for k, l in pairs(j) do
for m, n in pairs(l) do
ucichanges = ucichanges + 1;
end
end
end
%>
< % if ucichanges > 0 then %>
2023-03-18 23:35:15 +08:00
< a id = "notice_status" class = "label notice" href = "<%=url(category, 'uci/changes')%>?redir=<%=http.urlencode(http.formvalue('redir') or table.concat(disp.context.request, " / " ) ) % > ">< %=translate('Unsaved Changes')%>: < %=ucichanges%>< / a >
2022-03-25 11:25:27 +08:00
< % end %>
< span id = "xhr_poll_status" style = "display:none" onclick = "XHR.running() ? XHR.halt() : XHR.run()" >
< span class = "label success" id = "xhr_poll_status_on" > < span id = "refresh_on" class = "mobile-hide" > < / span > < / span >
< span class = "label" id = "xhr_poll_status_off" style = "display:none" > < span id = "refresh_off" class = "mobile-hide" > < / span > < / span >
< / span >
< % end %>
< / div >
< / div >
< / header >
< div class = "main" >
< div class = "main-left" >
< ul class = "nav" >
< %-
local function submenu(prefix, node)
local childs = disp.node_childs(node)
if #childs > 0 then
%>
< ul class = "slide-menu" >
< %-
for i, r in ipairs(childs) do
local nnode = node.nodes[r]
local href = controller .. "" .. prefix .. r ..
(nnode.query and http.build_querystring(nnode.query) or "")
%>
< li > < a data-title = "<%=pcdata(striptags(nnode.title))%>" href = "<%=pcdata(href)%>" > < %=pcdata(striptags(translate(nnode.title)))%>< / a > < / li >
< %-
end
%>
< / ul >
< %-
end
end
childs = disp.node_childs(cattree)
if #childs > 0 then
for i, r in ipairs(childs) do
local nnode = cattree.nodes[r]
local href = controller .. "/" .. category .. "/" .. r ..
(nnode.query and http.build_querystring(k.query) or "")
local grandchildren = disp.node_childs(nnode)
if #grandchildren > 0 then
%>
< li class = "slide" >
< a class = "menu" data-title = "<%=pcdata(striptags(nnode.title))%>" href = "#" > < %=pcdata(striptags(translate(nnode.title)))%>< / a >
< %- submenu("" .. category .. "/" .. r .. "/", nnode) %>
< / li >
< % else %>
< li class = "slide" >
< a class = "menu2" data-title = "<%=pcdata(striptags(nnode.title))%>" href = "<%=pcdata(href)%>" > < %=pcdata(striptags(translate(nnode.title)))%>< / a >
< / li >
< %
end
end
end
%>
< / ul >
< / div >
< div class = "main-right" >
< div class = "darkMask" > < / div >
< div id = "maincontent" >
< div class = "container" >
< %- if luci.sys.process.info("uid") == 0 and luci.sys.user.getuser("root") and not luci.sys.user.getpasswd("root") then -%>
< div class = "alert-message warning" >
< h4 > < %:No password set!%>< / h4 >
< %:There is no password set on this router. Please configure a root password to protect the web interface and enable SSH.%>< br >
< a href = "<%=pcdata(luci.dispatcher.build_url(" admin / system / admin " ) ) % > ">< %:Go to password configuration...%>< / a >
< / div >
< %- end -%>
< % if category then subtree("/" .. category .. "/", cattree) end %>