band-plan-web/band-plan.html
2024-05-28 21:56:49 +02:00

397 lines
12 KiB
HTML

<!-- SPDX-License-Identifier: Beerware -->
<!-- Wolfgang Kroener wrote this file. As long as you retain this notice -->
<!-- you can do whatever you want with this stuff. If we meet some day, -->
<!-- and you think this stuff is worth it, you can buy me a beer in return -->
<!doctype html>
<html lang="en">
<head>
<meta charset="utf-8">
<title>band plan</title>
<style>
body>div {
padding-bottom: 1em;
}
div#togglebuttons>button {
margin: 0 0.1em;
}
div>label {
padding-right: 1em;
}
</style>
</head>
<body>
<link href="https://unpkg.com/tabulator-tables/dist/css/tabulator.min.css" rel="stylesheet">
<script src="https://unpkg.com/browser-cjs/require.min.js"></script>
<script src="https://unpkg.com/tabulator-tables/dist/js/tabulator.min.js"></script>
<script src="https://unpkg.com/jspdf/dist/jspdf.umd.min.js"></script>
<script src="https://unpkg.com/jspdf-autotable/dist/jspdf.plugin.autotable.min.js"></script>
<script>
// require in browser-cjs
// eslint-disable-next-line no-undef
const yaml = require("https://unpkg.com/js-yaml/dist/js-yaml.min.js");
// eslint-disable-next-line no-undef
const Measures = require("https://unpkg.com/measures/dist/measures.cjs.js");
// eslint-disable-next-line no-undef
const path = require("https://unpkg.com/path-browserify/index.js");
function create_element_link(element, href) {
element.addEventListener("click", () => {
let lnk = document.createElement("a");
if (typeof href === "function") {
lnk.href = href();
} else {
lnk.href = href;
}
if (document.createEvent) {
let e = document.createEvent("MouseEvents");
e.initMouseEvent("click", true, true, window,
0, 0, 0, 0, 0, false, false, false,
false, 0, null);
lnk.dispatchEvent(e);
} else if (lnk.fireEvent) {
lnk.fireEvent("onclick");
}
});
}
document.addEventListener("DOMContentLoaded", () => {
location.search.substr(1).split("&").forEach((param) => {
let p = param.split("=");
let key = p[0];
["base", "extension", "css"].forEach((id) => {
if (key === id) {
document.getElementById(id + "_input").value = decodeURIComponent(p[1]);
}
});
});
update_inputs();
["base", "extension", "css"].forEach((id) => {
// download buttons
create_element_link(document.getElementById("download_" + id), document.getElementById(id + "_input").value);
// enter on input fields
document.getElementById(id + "_input").addEventListener("keydown", (event) => {
if (event.key === "Enter") {
update_inputs()
}
});
});
create_element_link(document.getElementById("create_link"), function() {
return window.location.href.split("?")[0] + "?"
+ "base=" + document.getElementById("base_input").value
+ "&extension=" + document.getElementById("extension_input").value;
+ "&css=" + document.getElementById("css_input").value;
});
});
// eslint-disable-next-line no-unused-vars
function filter_with_not(headerValue, rowValue, rowData, filterParams){
if (headerValue.startsWith("!")) {
// filter not
const str = headerValue.slice(1);
if ((str.length > 0) && (typeof rowValue !== "undefined")) {
return !rowValue.toString().includes(str);
} else {
return true;
}
} else {
// filter normal
if (typeof rowValue !== "undefined") {
return rowValue.toString().includes(headerValue);
} else {
return false;
}
}
}
function display_error(e) {
let t = document.getElementById("data-table");
t.innerHTML = "";
t.removeAttribute("class");
console.log(e);
document.getElementById("error").innerText = e;
}
function isInRange(number, min, max) {
return Number(number) >= Number(min) && Number(number) <= Number(max);
}
function add_column(row, cols, header) {
header.columns.forEach((col) => {
if (header.columns.includes(col)) {
if (typeof cols[col] === "object") {
for (const [key, value] of Object.entries(cols[col])) {
row[col + "_" + key] = value;
}
} else {
row[col] = cols[col];
}
}
});
}
function fill_comments(header) {
for (const [key, value] of Object.entries(header)) {
if (typeof value.comments !== "undefined") {
let s = "";
value.comments.forEach((comment) => {
for (const [ckey, cvalue] of Object.entries(comment)) {
s += ckey + ": " + cvalue + "<br>";
}
});
document.getElementById("comments_" + key).innerHTML = s;
}
};
}
function update_table(bases, extensions) {
let tdata = [];
let header = {};
// first, fill base_header
bases.forEach((base) => {
if ((typeof base.header !== "undefined") && base.header) {
header["base"] = structuredClone(base);
}
});
// then fill tdata
bases.forEach((base) => {
if ((typeof base.header === "undefined") || (base.header == false)) {
let row = {};
// first, fill ext_header
extensions.forEach((ext) => {
if ((typeof ext.header !== "undefined") && ext.header) {
header["extension"] = structuredClone(ext);
}
});
fill_comments(header);
// then fill tdata
extensions.forEach((ext) => {
if ((typeof ext.header === "undefined") || (ext.header == false)) {
add_column(row, base, header["base"]);
let [ext_start, ext_end] = ext.frequency.split("-");
if (typeof base.frequency === "number") {
// at single frequency
if (isInRange(base.frequency, ext_start, ext_end)) {
add_column(row, ext, header["extension"]);
tdata.push(row);
}
} else {
// frequeny range
let [base_start, base_end] = base.frequency.split("-");
if (isInRange(base_start, ext_start, ext_end) && isInRange(base_end, ext_start, ext_end)) {
// base range is inside of ext range or the same
add_column(row, ext, header["extension"]);
tdata.push(row);
} else {
// base range is split by ext range
if ((base_end > ext_start) && isInRange(ext_start, base_start, base_end)) {
// base range starts below ext range
let r = structuredClone(row);
let start = ext_start;
let end = base_end < ext_end ? base_end : ext_end;
r["frequency"] = start + "-" + end;
add_column(r, ext, header["extension"]);
tdata.push(r);
} else if ((ext_end > base_start) && isInRange(ext_end, base_start, base_end)) {
// ext range starts below base range
let r = structuredClone(row);
let start = base_start;
let end = ext_end;
r["frequency"] = start + "-" + end;
add_column(r, ext, header["extension"]);
tdata.push(r);
}
}
}
}
});
}
});
// require in tabular-tables
// eslint-disable-next-line no-undef
let table = new Tabulator("#data-table", {
layout: "fitDataTable",
initialSort: [
{column: "frequency", dir: "asc"},
{column: "band", dir: "desc"},
],
selectableRows: true,
data: tdata,
autoColumns:"full",
// add filter for all columns
autoColumnsDefinitions: function(definitions){
definitions.forEach((column) => {
if (column.field == "band") {
column.sorter = function(a, b) {
return parseFloat(new Measures().from(a).to("m")) - parseFloat(new Measures().from(b).to("m"));
}
}
if (column.field == "frequency") {
column.sorter = "number";
}
column.headerFilter = "input";
column.headerFilterFunc = filter_with_not;
if (typeof header["base"].titles[column.field] !== "undefined") {
column.title = header["base"].titles[column.field];
}
if (typeof header["extension"].titles[column.field] !== "undefined") {
column.title = header["extension"].titles[column.field];
}
// add css styles for formatting based on cell values
// e. g. cell-bandwidth-2700
// list like cw, narrow will be split to two entries
column.formatter = function(cell) {
let f = String(cell.getField()).toLowerCase().replace(/[\W]+/g, " ").replaceAll(" ", "_");
let v = String(cell.getValue()).toLowerCase().replace(/[^\w,]+/g, " ").replaceAll(" ", "_");
v.split(",").forEach((s) => {
cell.getElement().classList.add("cell-" + f + "-" + s.replace(/^_+|_+$/g, ""));
});
cell.getElement().classList.add("cell-" + f);
return cell.getValue();
}
});
return definitions;
},
// add css styles for formatting based on cell data
// e. g. row-mode-cw, row-bandwidth-200
// list like cw, narrow will be split to two entries
rowFormatter: function(row){
for (const [key, value] of Object.entries(row.getData())) {
let k = String(key).toLowerCase().replace(/[\W]+/g, " ").replaceAll(" ", "_");
let v = String(value).toLowerCase().replace(/[^\w,]+/g, " ").replaceAll(" ", "_");
v.split(",").forEach((s) => {
row.getElement().classList.add("row-" + k + "-" + s.replace(/^_+|_+$/g, ""));
});
}
},
});
document.getElementById("download").addEventListener("click", () => {
let download_range = "all";
if (table.getSelectedRows().length > 0) {
download_range = "selected";
}
table.download(
"pdf",
path.basename(document.getElementById("base_input").value, ".yml") + ".pdf",
{ orientation:"portrait" },
download_range);
});
table.on("tableBuilt", () => {
let div_togglebuttons = document.getElementById("togglebuttons");
div_togglebuttons.innerHTML = "<span>Toggle column:</span>";
table.getColumns().forEach((col) => {
let col_name = col.getField();
let b = document.createElement("button");
b.innerText = col_name;
b.addEventListener("click", () => {
table.toggleColumn(col_name);
});
div_togglebuttons.appendChild(b);
});
});
document.getElementById("error").innerText = "";
}
function update_inputs() {
Promise.all([
fetch(document.getElementById("base_input").value, { mode: "cors" }),
fetch(document.getElementById("extension_input").value, { mode: "cors" }),
]).then((res) => {
res.forEach((r) => {
if (!r.ok) {
throw new Error(r.status);
}
});
Promise.all([res[0].text(), res[1].text()]).then((d) => {
try {
update_table(yaml.load(d[0]),yaml.load(d[1]));
} catch (e) {
display_error(e);
}
}).catch((e) => {
display_error(e);
});
}).catch((e) => {
display_error(e);
});
// add stylesheet, clear before adding
let el = document.getElementById("link_css");
if (el !== null) {
el.remove();
}
let href = document.getElementById("css_input").value;
if (href !== "") {
let link = document.createElement("link");
link.type = "text/css";
link.rel = "stylesheet";
link.href = href;
link.id = "link_css";
document.head.appendChild(link);
}
}
// used in button
// eslint-disable-next-line no-unused-vars
function help() {
alert(
"Base plan, extension plan and css: use your own files for the band plan and css, URLs are possible, CORS needs to be allowed for the files, download standard files for examples\n"
+ "Filter rows with text in column headings, use filter starting with \"!\" as exclusion\n"
+ "Download list: saves .pdf of the current list (selection of rows apply)\n"
+ "Selection of rows with mouse possible\n"
+ "Download plans/css: get the yml and css files\n"
+ "Create link: get link for currently selected plans and css for bookmarking\n"
);
}
</script>
<div>
<label>Base Plan:
<input id="base_input" type="text" value="band-plan-iaru_r1_hf.yml">
</label>
<label>Extension:
<input id="extension_input" type="text" value="band-plan-de.yml">
</label>
<label>CSS:
<input id="css_input" type="text" value="standard.css">
</label>
<button onclick="update_inputs()">Update</button>
</div>
<div id="togglebuttons"></div>
<div>
<button id="download">Download list</button>
<button id="download_base">Download base plan</button>
<button id="download_extension">Download extension plan</button>
<button id="download_css">Download css file</button>
<button id="create_link">Create link</button>
<button onclick="help()">Help</button>
</div>
<div id="data-table"></div>
<div id="error"></div>
<div id="comments_base"></div>
<div id="comments_extension"></div>
<div>
<p>This information is supplied without liability.</p>
<p>Source is at <a href="https://src.dm5wk.de/dm5wk/band-plan-web/">https://src.dm5wk.de/dm5wk/band-plan-web/</a></p>
</div>
</body>
</html>