This commit is contained in:
2025-03-10 16:48:59 +08:00
commit 396980136a
47 changed files with 3375 additions and 0 deletions

38
cgi-bin/p/address.cgi Executable file
View File

@ -0,0 +1,38 @@
<h3>This camera uses MAC address <b>00:00:23:34:45:66</b> which is a placeholder.</h3>
<p>You need to replace it with the original MAC address from your stock firmware backup or <a href="#" id="generate-mac-address">generate a random valid MAC address</a>.</p>
<form action="fw-network.cgi" method="POST" class="row gy-2 gx-3 align-items-center mb-3">
<input type="hidden" name="action" value="changemac">
<div class="col-auto"><label class="form-label" for="mac_address">New MAC Address</label></div>
<div class="col-auto"><input class="form-control" id="mac_address" name="mac_address"type="text"></div>
<div class="col-auto"><input class="btn btn-danger" type="submit" value="Change MAC address"></div>
</form>
<p class="mb-0">Please note that the new MAC address will most likely give the camera a new IP address assigned by the DHCP server!</p>
<script>
function generateMacAddress(ev) {
ev.preventDefault();
const el = document.querySelector('#mac_address');
if (el.value == "") {
let mac = "";
for (let i = 1; i <= 6; i++) {
let b = ((Math.random() * 255) >>> 0);
if (i === 1) {
b = b | 2;
b = b & ~1;
}
mac += b.toString(16).toUpperCase().padStart(2, '0');
if (i < 6) {
mac += ":";
}
}
el.value = mac;
} else {
alert("There's a value in MAC address field. Please empty the field and try again.");
}
}
window.addEventListener('load', function() {
document.querySelector('#generate-mac-address').addEventListener('click', generateMacAddress);
});
</script>

504
cgi-bin/p/common.cgi Executable file
View File

@ -0,0 +1,504 @@
#!/usr/bin/haserl
<%
IFS_ORIG=$IFS
# tag "text" "classes" "extras"
div() {
tag "div" "$1" "$2" "$3"
}
# tag "tag" "text" "css" "extras"
tag() {
local t="$1"
local n="$2"
local c="$3"
[ -n "$c" ] && c=" class=\"${c}\""
local x="$4"
[ -n "$x" ] && x=" ${x}"
echo "<${t}${c}${x}>${n}</${t}>"
}
# A "tag" "classes" "extras"
A() {
local c="$2"
[ -n "$c" ] && c=" class=\"${c}\""
local x="$3"
[ -n "$x" ] && x=" ${x}"
echo "<${1}${c}${x}>"
}
Z() {
echo "</${1}>"
}
d() {
echo "$1" >&2
}
e() {
echo -e -n "$1"
}
h1() {
tag "h1" "$1" "$2" "$3"
}
h2() {
tag "h2" "$1" "$2" "$3"
}
h3() {
tag "h3" "$1" "$2" "$3"
}
h4() {
tag "h4" "$1" "$2" "$3"
}
h5() {
tag "h5" "$1" "$2" "$3"
}
h6() {
tag "h6" "$1" "$2" "$3"
}
label() {
tag "label" "$1" "$2" "$3"
}
li() {
tag "li" "$1" "$2" "$3"
}
p() {
tag "p" "$1" "$2" "$3"
}
span() {
tag "span" "$1" "$2" "$3"
}
div_() {
A "div" "$1" "$2"
}
_div() {
Z "div"
}
row_() {
echo "<div class\"row ${1}\" ${2}>"
}
_row() {
echo "</div>"
}
row() {
row_ "$2"
echo "$1"
_row
}
span_() {
A "span" "$1" "$2"
}
_span() {
Z "span"
}
# alert "text" "type" "extras"
alert() {
echo "<div class=\"alert alert-${2}\" ${3}>${1}</div>"
}
# button_submit "text" "type" "extras"
button_submit() {
local t="$1"
[ -z "$t" ] && t="Save Changes"
local c="$2"
[ -z "$c" ] && c="primary"
local x="$3"
[ -z "$x" ] && x=" ${x}"
echo "<div class=\"mt-2\"><input type=\"submit\" class=\"btn btn-${c}\"${x} value=\"${t}\"></div>"
}
check_password() {
local p="/cgi-bin/fw-interface.cgi"
[ -z "$SCRIPT_NAME" ] || [ "$SCRIPT_NAME" = "${p}" ] && return
if [ ! -f /etc/shadow- ] || [ -z $(grep root /etc/shadow- | cut -d: -f2) ]; then
redirect_to "${p}" "danger" "You must set your own secure password!"
fi
}
ex() {
echo "<div class=\"${2:-ex}\"><h6># ${1}</h6><pre class=\"small\">"
eval "$1" | sed "s/&/\&amp;/g;s/</\&lt;/g;s/>/\&gt;/g;s/\"/\&quot;/g"
echo "</pre></div>"
}
# field_hidden "name" "value"
field_hidden() {
local n="$1"
local v="$2"
echo "<input type=\"hidden\" name=\"${n}\" id=\"${n}\" value=\"${v}\" class=\"form-hidden\">"
}
# field_integer "name" "label" "value" "min" "max" "hint"
field_integer() {
local n="$1"
local l="$2"
local v="$3"
local x="$4"
local y="$5"
local h="$6"
echo "<p class=\"number\">" \
"<label class=\"form-label\" for=\"${n}\">${l}</label>" \
"<span class=\"input-group\">"
echo "<input type=\"number\" id=\"${n}\" name=\"${n}\" class=\"form-control text-end\" value=\"${v}\" min=\"${x}\" max=\"${y}\" step=\"1\">" \
"</span>"
[ -n "$h" ] && echo "<span class=\"hint text-secondary\">${h}</span>"
echo "</p>"
}
# field_password "name" "label" "hint"
field_password() {
local n="$1"
local l="$2"
local h="$3"
local v=$(t_value "$n")
echo "<p class=\"password\" id=\"${n}_wrap\">" \
"<label for=\"${n}\" class=\"form-label\">${l}</label><span class=\"input-group\">" \
"<input type=\"password\" id=\"${n}\" name=\"${n}\" class=\"form-control\" value=\"${v}\">" \
"<label class=\"input-group-text\">" \
"<input type=\"checkbox\" class=\"form-check-input me-1\" data-for=\"${n}\"> show" \
"</label></span>"
[ -n "$h" ] && echo "<span class=\"hint text-secondary\">${h}</span>"
echo "</p>"
}
# field_range "name" "label" "value" "min" "max" "hint"
field_range() {
local n="$1"
local l="$2"
local v="$3"
local x="$4"
local y="$5"
local h="$6"
echo "<p class=\"range\" id=\"${n}_wrap\">" \
"<label for=\"${n}\" class=\"form-label\">${l}</label>" \
"<span class=\"input-group\">"
echo "<input type=\"hidden\" id=\"${n}\" name=\"${n}\" value=\"${v}\">"
echo "<input type=\"range\" class=\"form-control form-range\" id=\"${n}-range\" value=\"${v}\" min=\"${x}\" max=\"${y}\" step=\"1\">"
echo "<span class=\"input-group-text show-value\" id=\"${n}-show\">${v}</span></span>"
[ -n "$h" ] && echo "<span class=\"hint text-secondary\">${h}</span>"
echo "</p>"
}
# field_switch "name" "label" "value" "hint"
field_switch() {
local n="$1"
local l="$2"
local v="$3"
local h="$4"
[ "$v" = "eval" ] && v=$(t_value "$n")
[ "$v" = "true" ] && v="checked"
echo "<p class=\"boolean\"><span class=\"form-check form-switch\">" \
"<input type=\"hidden\" id=\"${n}-false\" name=\"${n}\" value=\"false\">" \
"<input type=\"checkbox\" id=\"${n}\" name=\"${n}\" value=\"true\" class=\"form-check-input\" ${v}>" \
"<label for=\"${n}\" class=\"form-check-label\">${l}</label></span>"
[ -n "$h" ] && echo "<span class=\"hint text-secondary\">${h}</span>"
echo "</p>"
}
# field_string "name" "label" "value" "enum" "hint"
field_string() {
local n="$1"
local l="$2"
local v="$3"
local e="$4"
local h="$5"
[ "$v" = "eval" ] && v=$(t_value "$n")
if [ -n "$e" ]; then
echo "<p class=\"select\" id=\"${n}_wrap\">" \
"<label for=\"${n}\" class=\"form-label\">${l}</label>" \
"<select class=\"form-select\" id=\"${n}\" name=\"${n}\">"
for e in $e; do
echo -n "<option value=\"${e}\""
[ "$v" = "$e" ] && echo -n " selected"
echo ">${e}</option>"
done
echo "</select>"
else
echo "<p class=\"string\" id=\"${n}_wrap\">" \
"<label for=\"${n}\" class=\"form-label\">${l}</label>" \
"<input type=\"text\" id=\"${n}\" name=\"${n}\" class=\"form-control\" value=\"${v}\">"
fi
[ -n "$h" ] && echo "<span class=\"hint text-secondary\">${h}</span>"
echo "</p>"
}
# field_text "name" "label" "hint"
field_text() {
local n="$1"
local l="$2"
local h="$3"
local v=$(t_value "$n")
echo "<p class=\"string\" id=\"${n}_wrap\">" \
"<label for=\"${n}\" class=\"form-label\">${l}</label>" \
"<input type=\"text\" id=\"${n}\" name=\"${n}\" class=\"form-control\" value=\"${v}\">"
[ -n "$h" ] && echo "<span class=\"hint text-secondary\">${h}</span>"
echo "</p>"
}
# field_textedit "name" "label" "file"
field_textedit() {
local n="$1"
local l="$2"
local v=$(cat "$3")
echo "<p class=\"textarea\" id=\"${n}_wrap\">" \
"<label for=\"${n}\" class=\"form-label\">${l}</label>" \
"<textarea id=\"${n}\" name=\"${n}\" class=\"form-control\">${v}</textarea>"
echo "</p>"
}
get_config() {
echo ${1}/etc/majestic.yaml
}
get_metrics() {
local m=$(pidof majestic)
if [ -z "$m" ]; then
echo 0
else
wget -q -T1 localhost/metrics/night?value=${1} -O -
fi
}
get_schema() {
local m=/tmp/webui/schema.json
if [ ! -e "$m" ]; then
wget -q -T1 localhost/api/v1/config.schema.json -O "$m"
fi
echo "$m"
}
get_night() {
local m=$(pidof majestic)
local v=$(yaml-cli -g .nightMode.$1)
if [ -n "$m" ] && [ -n "$v" ] && [ "$v" != "false" ]; then
echo true
else
echo false
fi
}
log_create() {
echo "${1}:${2}" > "$log_file"
}
log_read() {
[ ! -f "$log_file" ] && return
[ -z "$(cat $log_file)" ] && return
local c
local m
local l
OIFS="$IFS"
IFS=$'\n'
for l in $(cat "$log_file"); do
c="$(echo $l | cut -d':' -f1)"
m="$(echo $l | cut -d':' -f2-)"
echo "<div class=\"alert alert-${c} alert-dismissible fade show\" role=\"alert\">${m}" \
"<button type=\"button\" class=\"btn btn-close\" data-bs-dismiss=\"alert\" aria-label=\"Close\"></button>" \
"</div>"
done
IFS=$OIFS
rm -f "$log_file"
}
set_error_flag() {
echo "danger:${1}" >> "$log_file"
error=1
}
html_title() {
[ -n "$page_title" ] && echo -n "$page_title"
[ -n "$title" ] && echo -n ": $title"
echo -n " - OpenIPC"
}
include() {
[ -f "$1" ] && . "$1"
}
# label "name" "classes" "extras" "units"
label() {
local c="form-label"
[ -n "$2" ] && c="${c} ${2}"
local l="$(t_label "$1")"
[ -z "$l" ] && l="$1" && c="${c} bg-warning"
local x="$3"
[ -n "$x" ] && x=" ${x}"
local u="$4"
[ -n "$u" ] && l="${l}, <span class=\"units text-secondary x-small\">$u</span>"
echo "<label for=\"${1}\" class=\"${c}\"${x}>${l}</label>"
}
# pre "text" "classes" "extras"
pre() {
# replace <, >, &, ", and ' with HTML entities
tag "pre" "$(echo -e "$1" | sed "s/&/\&amp;/g;s/</\&lt;/g;s/>/\&gt;/g;s/\"/\&quot;/g")" "$2" "$3"
}
preview() {
if [ "true" = "$(yaml-cli -g .jpeg.enabled)" ]; then
echo "<video poster=\"/mjpeg\" style=\"background:url(/a/preview.svg); background-size:cover; width:100%\"></video>"
else
echo "<p class=\"alert alert-warning\"><a href=\"mj-settings.cgi?tab=jpeg\">Enable JPEG support</a> to see the preview.</p>"
fi
}
progressbar() {
local c="primary"
[ "$1" -ge "75" ] && c="danger"
echo "<div class=\"progress\" role=\"progressbar\" aria-valuenow=\"${1}\" aria-valuemin=\"0\" aria-valuemax=\"100\">" \
"<div class=\"progress-bar progress-bar-striped progress-bar-animated bg-${c}\" style=\"width:${1}%\"></div>" \
"</div>"
}
# redirect_back "flash class" "flash text"
redirect_back() {
redirect_to "${HTTP_REFERER:-/}" "$1" "$2"
}
# redirect_to "url" "flash class" "flash text"
redirect_to() {
[ -n "$3" ] && log_create "$2" "$3"
echo "HTTP/1.1 303 See Other"
echo "Content-type: text/html; charset=UTF-8"
echo "Cache-Control: no-store"
echo "Pragma: no-cache"
echo "Location: $1"
echo
exit 0
}
report_command() {
echo "<h4># ${1}</h4>"
echo "<pre class=\"small\">${2}</pre>"
}
report_error() {
echo "<h4 class=\"text-danger\">Oops. Something happened.</h4>"
alert "$1" "danger"
}
# report_log "text" "extras"
report_log() {
pre "$1" "small" "$2"
}
generate_signature() {
echo "${soc} (${soc_family} family), $sensor, ${flash_size} MB ${flash_type} flash, ${fw_version}-${fw_variant}, ${network_hostname}, ${network_macaddr}" > $signature_file
}
signature() {
[ ! -f "$signature_file" ] && generate_signature
cat $signature_file
}
t_label() {
eval "echo \$tL_${1}"
}
t_value() {
eval "echo \$${1}"
}
update_caminfo() {
flash_type=$(ipcinfo --flash-type)
mtd_size=$(grep -E "nor|nand" $(ls /sys/class/mtd/mtd*/type) | sed -E "s|type.+|size|g")
flash_size=$(awk '{sum+=$1} END{print sum/1024/1024}' $mtd_size)
sensor_ini=$(ipcinfo --long-sensor)
sensor=$(fw_printenv -n sensor)
[ -z "$sensor" ] && sensor=$(echo $sensor_ini | cut -d_ -f1)
soc_vendor=$(ipcinfo --vendor)
soc_family=$(ipcinfo --family)
soc=$(ipcinfo --chip-name)
if [ -z "$soc" ] || [ "$soc_vendor" = "sigmastar" ]; then
soc=$(fw_printenv -n soc)
fi
soc_temp=$(ipcinfo --temp 2> /dev/null)
if [ -n "$soc_temp" ]; then
soc_has_temp="true"
else
soc_has_temp="false"
fi
# Firmware
fw_version=$(grep "OPENIPC_VERSION" /etc/os-release | cut -d= -f2 | tr -d '"')
fw_variant=$(grep "BUILD_OPTION" /etc/os-release | cut -d= -f2 | tr -d '"')
fw_build=$(grep "GITHUB_VERSION" /etc/os-release | cut -d= -f2 | tr -d '"')
mj_version=$($mj_bin_file -v)
uboot_version=$(fw_printenv -n ver)
# WebUI
ui_password=$(grep root /etc/shadow | cut -d: -f2)
ptz_support=$(fw_printenv -n ptz)
# Network
network_interface=$(ip route | awk '/default/ {print $5}' | head -n1)
network_address=$(ip route | grep ${network_interface:-eth0} | awk '/src/ {print $7}')
network_gateway=$(ip route | awk '/default/ {print $3}')
network_hostname=$(hostname -s)
network_macaddr=$(cat /sys/class/net/${network_interface:-eth0}/address)
# Overlay
overlay_root="/overlay"
# Default timezone is GMT
tz_data=$(cat /etc/TZ)
tz_name=$(cat /etc/timezone)
if [ -z "$tz_data" ] || [ -z "$tz_name" ]; then
tz_data="GMT0"; echo "$tz_data" > /etc/TZ
tz_name="Etc/GMT"; echo "$tz_name" > /etc/timezone
fi
local variables="flash_size flash_type fw_build fw_variant fw_version mj_version network_address
network_gateway network_hostname network_interface network_macaddr overlay_root ptz_support
sensor sensor_ini soc soc_family soc_has_temp soc_vendor tz_data tz_name uboot_version ui_password"
rm -f ${sysinfo_file}
local v
for v in $variables; do
eval "echo ${v}=\'\$${v}\' >> ${sysinfo_file}"
done
generate_signature
}
mj_bin_file=/usr/bin/majestic
log_file=/tmp/webui/logfile.txt
signature_file=/tmp/webui/signature.txt
sysinfo_file=/tmp/webui/sysinfo.txt
[ ! -d /etc/webui ] && mkdir -p /etc/webui
[ ! -d /tmp/webui ] && mkdir -p /tmp/webui
[ ! -f $sysinfo_file ] && update_caminfo
include $sysinfo_file
pagename=$(basename "$SCRIPT_NAME")
pagename="${pagename%%.*}"
include /etc/webui/webui.conf
include /usr/share/libubox/jshn.sh
check_password
%>

16
cgi-bin/p/footer.cgi Executable file
View File

@ -0,0 +1,16 @@
</div>
</main>
<footer class="x-small">
<div class="container pt-3">
<div class="row">
<div class="col col-2">
<p id="uptime" class="text-secondary"></p>
</div>
<div class="col col-10">
<p class="text-end"><a href="https://github.com/openipc/majestic-webui">WebUI</a> by <a href="https://openipc.org/">OpenIPC</a></p>
</div>
</div>
</div>
</footer>
</body>
</html>

131
cgi-bin/p/header.cgi Executable file
View File

@ -0,0 +1,131 @@
#!/usr/bin/haserl
Content-type: text/html; charset=UTF-8
Cache-Control: no-store
Pragma: no-cache
<!DOCTYPE html>
<html lang="en" data-bs-theme="<%= ${webui_theme:=dark} %>">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title><% html_title %></title>
<link rel="stylesheet" href="/a/bootstrap.min.css">
<link rel="stylesheet" href="/a/bootstrap.override.css">
<script src="/a/bootstrap.bundle.min.js"></script>
<script src="/a/main.js"></script>
</head>
<body id="page-<%= $pagename %>" class="<%= $fw_variant %>">
<nav class="navbar navbar-expand-lg bg-body-tertiary">
<div class="container">
<a class="navbar-brand" href="status.cgi"><img alt="Image: OpenIPC logo" height="32" src="/a/logo.svg"><span class="x-small ms-1"><%= $fw_variant %></span></a>
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarNav" aria-controls="navbarNav" aria-expanded="false" aria-label="Toggle navigation">
<span class="navbar-toggler-icon"></span>
</button>
<div class="collapse navbar-collapse justify-content-end" id="navbarNav">
<ul class="navbar-nav">
<li class="nav-item dropdown">
<a aria-expanded="false" class="nav-link dropdown-toggle" data-bs-toggle="dropdown" href="#" id="dropdownInformation" role="button">Information</a>
<ul aria-labelledby="dropdownInformation" class="dropdown-menu">
<li><a class="dropdown-item" href="status.cgi">Status</a></li>
<li><hr class="dropdown-divider"></li>
<li><a class="dropdown-item" href="info-majestic.cgi">Majestic</a></li>
<li><a class="dropdown-item" href="info-kernel.cgi">Kernel</a></li>
<li><a class="dropdown-item" href="info-overlay.cgi">Overlay</a></li>
</ul>
</li>
<li class="nav-item dropdown">
<a aria-expanded="false" class="nav-link dropdown-toggle" data-bs-toggle="dropdown" href="#" id="dropdownMajestic" role="button">Majestic</a>
<ul aria-labelledby="dropdownMajestic" class="dropdown-menu">
<li><a class="dropdown-item" href="mj-settings.cgi">Settings</a></li>
<li><hr class="dropdown-divider"></li>
<li><a class="dropdown-item" href="mj-configuration.cgi">Configuration</a></li>
<li><a class="dropdown-item" href="mj-endpoints.cgi">Endpoints</a></li>
</ul>
</li>
<li class="nav-item dropdown">
<a aria-expanded="false" class="nav-link dropdown-toggle" data-bs-toggle="dropdown" href="#" id="dropdownSettings" role="button">Firmware</a>
<ul aria-labelledby="dropdownSettings" class="dropdown-menu">
<li><a class="dropdown-item" href="fw-network.cgi">Network</a></li>
<li><a class="dropdown-item" href="fw-time.cgi">Time</a></li>
<li><a class="dropdown-item" href="fw-interface.cgi">Interface</a></li>
<li><hr class="dropdown-divider"></li>
<li><a class="dropdown-item" href="fw-update.cgi">Update</a></li>
<li><a class="dropdown-item" href="fw-settings.cgi">Settings</a></li>
</ul>
</li>
<li class="nav-item dropdown">
<a aria-expanded="false" class="nav-link dropdown-toggle" data-bs-toggle="dropdown" href="#" id="dropdownTools" role="button">Tools</a>
<ul aria-labelledby="dropdownTools" class="dropdown-menu">
<li><a class="dropdown-item" href="tool-console.cgi">Console</a></li>
<li><a class="dropdown-item" href="tool-files.cgi">Files</a></li>
<% if [ -e /dev/mmcblk0 ]; then %>
<li><a class="dropdown-item" href="tool-sdcard.cgi">SDcard</a></li>
<% fi %>
</ul>
</li>
<li class="nav-item dropdown">
<a aria-expanded="false" class="nav-link dropdown-toggle" data-bs-toggle="dropdown" href="#" id="dropdownExtensions" role="button">Extensions</a>
<ul aria-labelledby="dropdownExtensions" class="dropdown-menu dropdown-menu-lg-end">
<li><a class="dropdown-item" href="ext-openwall.cgi">OpenWall</a></li>
<li><a class="dropdown-item" href="ext-telegram.cgi">Telegram</a></li>
<li><a class="dropdown-item" href="ext-tunnel.cgi">Tunnel</a></li>
<li><hr class="dropdown-divider"></li>
<li><a class="dropdown-item" href="ext-proxy.cgi">Proxy</a></li>
</ul>
</li>
<li class="nav-item"><a class="nav-link" href="preview.cgi">Preview</a></li>
</ul>
</div>
</div>
</nav>
<main class="pb-4">
<div class="container" style="min-height: 85vh">
<div class="row mt-1 x-small">
<div class="col-lg-2">
<div id="pb-memory" class="progress my-1" role="progressbar" aria-valuenow="0" aria-valuemin="0" aria-valuemax="100"><div class="progress-bar"></div></div>
<div id="pb-overlay" class="progress my-1" role="progressbar" aria-valuenow="0" aria-valuemin="0" aria-valuemax="100"><div class="progress-bar"></div></div>
</div>
<div class="col-md-6 mb-2">
<%= $(signature) %>
</div>
<div class="col-1" id="daynight_value"></div>
<div class="col-md-4 col-lg-3 mb-2 text-end">
<div id="time-now"></div>
<div class="text-secondary" id="soc-temp"></div>
</div>
</div>
<% if [ -z "$network_gateway" ]; then %>
<div class="alert alert-warning">
<p class="mb-0">Internet connection not available, please <a href="fw-network.cgi">check your network settings</a>.</p>
</div>
<% fi %>
<% if [ "$network_macaddr" = "00:00:23:34:45:66" ] && [ -f /etc/shadow- ] && [ -n $(grep root /etc/shadow- | cut -d: -f2) ]; then %>
<div class="alert alert-danger">
<%in p/address.cgi %>
</div>
<% fi %>
<% if [ ! -e $(get_config) ]; then %>
<div class="alert alert-danger">
<p class="mb-0">Majestic configuration not found, please <a href="mj-configuration.cgi">check your Majestic settings</a>.</p>
</div>
<% fi %>
<% if [ "$(cat /etc/TZ)" != "$TZ" ] || [ -e /tmp/system-reboot ]; then %>
<div class="alert alert-danger">
<h3>Warning.</h3>
<p>System settings have been updated, restart to apply pending changes.</p>
<span class="d-flex gap-3">
<a class="btn btn-danger" href="fw-restart.cgi">Restart camera</a>
</span>
</div>
<% fi %>
<h2><%= $page_title %></h2>
<% log_read %>

32
cgi-bin/p/motor.cgi Executable file
View File

@ -0,0 +1,32 @@
<hr class="mb-3"/>
<div class="motor">
<div class="col">
<button class="btn btn-motor" data-dir="ul">↖️</button>
<button class="btn btn-motor" data-dir="uc">⬆️</button>
<button class="btn btn-motor" data-dir="ur">↗️</button>
</div>
<div class="col">
<button class="btn btn-motor" data-dir="lc">⬅️</button>
<button class="btn btn-motor" data-dir="cc">🆗</button>
<button class="btn btn-motor" data-dir="rc">➡️</button>
</div>
<div class="col">
<button class="btn btn-motor" data-dir="dl">↙️</button>
<button class="btn btn-motor" data-dir="dc">⬇️</button>
<button class="btn btn-motor" data-dir="dr">↘️</button>
</div>
</div>
<script>
function control(dir) {
let x = dir.includes("l") ? -1 : dir.includes("r") ? 1 : 0;
let y = dir.includes("d") ? -1 : dir.includes("u") ? 1 : 0;
fetch('/cgi-bin/j/run.cgi?web=' + btoa('motor ' + '<%= $ptz_support %> ' + x + ' ' + y));
}
$$(".motor button").forEach(el => {
el.addEventListener("click", ev => {
control(ev.target.dataset.dir);
});
});
</script>

18
cgi-bin/p/roi.cgi Executable file
View File

@ -0,0 +1,18 @@
<div class="col-10" id="_row">
<h3>Visual editor</h3>
<div class="col">
<iframe id="_iframe" src="/m/img.html" frameborder="0" style="padding: 0px; margin: 0px; border: 1px solid rgb(76, 96, 216);"></iframe>
</div>
<div class="row mb-3 align-items-center">
<div class="col">
<input type="button" class="btn btn-primary" onclick="_clear();" value="Clear all regions">
</div>
</div>
</div>
<script>
function _clear() {
document.getElementById('_iframe').contentWindow.location.reload();
document.getElementById('_motionDetect_roi').value = '';
}
</script>