wt32powermeter/index.h

380 lines
15 KiB
C
Raw Normal View History

2023-04-24 08:27:35 -04:00
const char MAIN_page[] PROGMEM = R"=====(
<!DOCTYPE html>
<html>
<style>
2023-05-04 14:01:11 -04:00
.row {
width: 100%;
margin: 0 auto;
}
.box{
2023-04-24 08:27:35 -04:00
max-width: 500px;
2023-05-11 10:18:51 -04:00
min-width: 330px;
min-height: 330px;
2023-05-07 01:38:01 -04:00
background: #009879;
2023-04-24 08:27:35 -04:00
padding: 30px;
box-sizing: border-box;
2023-05-08 12:49:10 -04:00
vertical-align: top;
2023-04-24 08:27:35 -04:00
color: #FFF;
margin:20px;
box-shadow: 0px 2px 18px -4px rgba(0,0,0,0.75);
2023-05-04 14:01:11 -04:00
display: inline-block;
2023-04-24 08:27:35 -04:00
}
2023-05-11 10:18:51 -04:00
.innerbox{
padding: 10px;
2023-05-11 16:25:46 -04:00
margin-left: auto;
margin-right: auto;
2023-05-11 10:18:51 -04:00
display: inline-block;
2023-05-11 16:25:46 -04:00
text-align: center;
2023-05-11 16:41:28 -04:00
vertical-align: middle;
2023-05-11 10:18:51 -04:00
}
2023-05-07 01:38:01 -04:00
.redbox{
background: #FFAA79;
}
.whitebox{
background: #FFFFFF;
color: #009879;
border-style: solid;
border-color: #009879;
}
.button{background-color: #009879; border: none; color: white; padding: 5px 5px; text-align: center; text-decoration: none; display: inline-block; margin: 4px 2px; cursor: pointer; border-radius: 8px;}
2023-04-24 08:27:35 -04:00
</style>
2023-05-11 10:18:51 -04:00
<script>
2023-05-12 08:35:49 -04:00
var okaytogo = true;
2023-05-19 16:44:38 -04:00
2023-05-11 10:18:51 -04:00
function vumeter(elem, config){
2023-05-07 01:38:01 -04:00
2023-05-11 10:18:51 -04:00
// Settings
var max = config.max || 100;
var boxCount = config.boxCount || 10;
var boxCountRed = config.boxCountRed || 2;
var boxCountYellow = config.boxCountYellow || 3;
var boxGapFraction = config.boxGapFraction || 0.2;
2023-05-11 16:25:46 -04:00
var jitter = 0;//config.jitter || 0.02;
2023-05-07 01:38:01 -04:00
2023-05-11 10:18:51 -04:00
// Colours
var redOn = 'rgba(255,47,30,0.9)';
var redOff = 'rgba(64,12,8,0.9)';
var yellowOn = 'rgba(255,215,5,0.9)';
var yellowOff = 'rgba(64,53,0,0.9)';
var greenOn = 'rgba(53,255,30,0.9)';
var greenOff = 'rgba(13,64,8,0.9)';
2023-04-24 08:27:35 -04:00
2023-05-11 10:18:51 -04:00
// Derived and starting values
var width = elem.width;
var height = elem.height;
var curVal = 0;
2023-04-24 08:27:35 -04:00
2023-05-11 10:18:51 -04:00
// Gap between boxes and box height
var boxHeight = height / (boxCount + (boxCount+1)*boxGapFraction);
var boxGapY = boxHeight * boxGapFraction;
2023-04-24 08:27:35 -04:00
2023-05-11 10:18:51 -04:00
var boxWidth = width - (boxGapY*2);
var boxGapX = (width - boxWidth) / 2;
// Canvas starting state
var c = elem.getContext('2d');
// Main draw loop
var draw = function(){
var targetVal = parseInt(elem.dataset.val, 10);
2023-05-12 08:35:49 -04:00
2023-05-11 10:18:51 -04:00
// Gradual approach
if (curVal <= targetVal){
curVal += (targetVal - curVal) / 5;
} else {
curVal -= (curVal - targetVal) / 5;
}
2023-05-11 16:25:46 -04:00
curVal = targetVal;
2023-05-11 10:18:51 -04:00
// Apply jitter
if (jitter > 0 && curVal > 0){
var amount = (Math.random()*jitter*max);
if (Math.random() > 0.5){
amount = -amount;
}
curVal += amount;
}
if (curVal < 0) {
curVal = 0;
}
c.save();
c.beginPath();
c.rect(0, 0, width, height);
c.fillStyle = 'rgb(32,32,32)';
c.fill();
c.restore();
drawBoxes(c, curVal);
requestAnimationFrame(draw);
2023-05-20 06:16:41 -04:00
2023-05-11 10:18:51 -04:00
};
// Draw the boxes
function drawBoxes(c, val){
c.save();
c.translate(boxGapX, boxGapY);
for (var i = 0; i < boxCount; i++){
var id = getId(i);
c.beginPath();
if (isOn(id, val)){
c.shadowBlur = 10;
c.shadowColor = getBoxColor(id, val);
}
c.rect(0, 0, boxWidth, boxHeight);
c.fillStyle = getBoxColor(id, val);
c.fill();
c.translate(0, boxHeight + boxGapY);
}
c.restore();
}
// Get the color of a box given it's ID and the current value
function getBoxColor(id, val){
// on colours
if (id > boxCount - boxCountRed){
return isOn(id, val)? redOn : redOff;
}
if (id > boxCount - boxCountRed - boxCountYellow){
return isOn(id, val)? yellowOn : yellowOff;
}
return isOn(id, val)? greenOn : greenOff;
}
function getId(index){
// The ids are flipped, so zero is at the top and
// boxCount-1 is at the bottom. The values work
// the other way around, so align them first to
// make things easier to think about.
return Math.abs(index - (boxCount - 1)) + 1;
}
function isOn(id, val){
// We need to scale the input value (0-max)
// so that it fits into the number of boxes
var maxOn = Math.ceil((val/max) * boxCount);
return (id <= maxOn);
}
2023-05-19 16:44:38 -04:00
draw();
2023-05-11 10:18:51 -04:00
}
2023-04-24 08:27:35 -04:00
setInterval(function() {
2023-05-07 01:38:01 -04:00
// Call a function repetatively
2023-05-12 08:35:49 -04:00
okaytogo = !okaytogo;
2023-05-07 01:38:01 -04:00
getDATA();
2023-05-19 16:44:38 -04:00
}, 500);
2023-04-24 08:27:35 -04:00
2023-05-11 04:50:00 -04:00
function beep() {
var snd = new Audio("data:audio/wav;base64,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qJrx4KOQjahZxWbcZgztj2c49nKmkId44S71j0c8eV9yDK6uPRzx5X18eDvjvQ6yKo9ZSS6l//8elePK/Lf//IInrOF/FvDoADYAGBMGb7FtErm5MXMlmPAJQVgWta7Zx2go+8xJ0UiCb8LHHdftWyLJE0QIAIsI+UbXu67dZMjmgDGCGl1H+vpF4NSDckSIkk7Vd+sxEhBQMRU8j/12UIRhzSaUdQ+rQU5kGeFxm+hb1oh6pWWmv3uvmReDl0UnvtapVaIzo1jZbf/pD6ElLqSX+rUmOQNpJFa/r+sa4e/pBlAABoAAAAA3CUgShLdGIxsY7AUABPRrgCABdDuQ5GC7DqPQCgbbJUAoRSUj+NIEig0YfyWUho1VBBBA//uQZB4ABZx5zfMakeAAAAmwAAAAF5F3P0w9GtAAACfAAAAAwLhMDmAYWMgVEG1U0FIGCBgXBXAtfMH10000EEEEEECUBYln03TTTdNBDZopopYvrTTdNa325mImNg3TTPV9q3pmY0xoO6bv3r00y+IDGid/9aaaZTGMuj9mpu9Mpio1dXrr5HERTZSmqU36A3CumzN/9Robv/Xx4v9ijkSRSNLQhAWumap82WRSBUqXStV/YcS+XVLnSS+WLDroqArFkMEsAS+eWmrUzrO0oEmE40RlMZ5+ODIkAyKAGUwZ3mVKmcamcJnMW26MRPgUw6j+LkhyHGVGYjSUUKNpuJUQoOIAyDvEyG8S5yfK6dhZc0Tx1KI/gviKL6qvvFs1+bWtaz58uUNnryq6kt5RzOCkPWlVqVX2a/EEBUdU1KrXLf40GoiiFXK///qpoiDXrOgqDR38JB0bw7SoL+ZB9o1RCkQjQ2CBYZKd/+VJxZRRZlqSkKiws0WFxUyCwsKiMy7hUVFhIaCrNQsKkTIsLivwKKigsj8XYlwt/WKi2N4d//uQRCSAAjURNIHpMZBGYiaQPSYyAAABLAAAAAAAACWAAAAApUF/Mg+0aohSIRobBAsMlO//Kk4soosy1JSFRYWaLC4qZBYWFRGZdwqKiwkNBVmoWFSJkWFxX4FFRQWR+LsS4W/rFRb/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////VEFHAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAU291bmRib3kuZGUAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAMjAwNGh0dHA6Ly93d3cuc291bmRib3ku
snd.play();
}
2023-05-11 10:18:51 -04:00
function strtoint(x) {
const parsed = parseInt(x);
if (isNaN(parsed)) { return 0; }
return parsed;
}
2023-05-12 07:43:34 -04:00
function formatNum(num, separator, fraction) {
var str = num.toLocaleString('en-US');
str = str.replace(/\./, fraction);
str = str.replace(/,/g, separator);
str = str.substring(0, str.indexOf(fraction)+3);
return str;
}
2023-05-11 16:25:46 -04:00
function convert_power(val){
let ret = "0";
if (val < 0.001){
2023-05-12 07:43:34 -04:00
ret = formatNum(val*1000000,'','.') + " uW";
2023-05-11 16:25:46 -04:00
} else if (val < 1) {
2023-05-12 07:43:34 -04:00
ret = formatNum(val*1000,'','.') + " mW";
2023-05-11 16:25:46 -04:00
} else {
2023-05-12 07:43:34 -04:00
ret = formatNum(val*1.0,'','.') + " W";
2023-05-11 16:25:46 -04:00
}
return ret;
}
function check_dbm(val){
let ret = "0";
2023-05-12 07:43:34 -04:00
if (isNaN(val)) {
2023-05-11 16:25:46 -04:00
ret = "-- dBm";
} else {
ret = val + " dBm";
}
return ret;
}
2023-05-07 01:38:01 -04:00
function getDATA() {
2023-04-24 08:27:35 -04:00
var xhttp = new XMLHttpRequest();
xhttp.onreadystatechange = function() {
if (this.readyState == 4 && this.status == 200) {
2023-05-20 06:16:41 -04:00
//document.getElementById("ref_led_box").style.display = 'none';
//document.getElementById("vswr_led_box").style.display = 'none';
2023-05-11 10:18:51 -04:00
data = this.responseText.split(";");
2023-05-19 16:44:38 -04:00
if (data[15] == "0") {
document.getElementById("FWDWatt").innerHTML = convert_power(data[0]);
document.getElementById("FWDdBm").innerHTML = check_dbm(data[1]);
document.getElementById("FWDVoltage").innerHTML = data[2];
if (data[17] == "true") {
document.getElementById("fwd_led_box").style.display = 'inline-block';
fwd_vu_meter.setAttribute('data-val', strtoint(data[0]*1000000));
} else {
document.getElementById("fwd_led_box").style.display = 'none';
}
} else {
document.getElementById("FWDWatt").innerHTML = "---";
document.getElementById("FWDdBm").innerHTML = "---";
2023-05-20 06:16:41 -04:00
document.getElementById("FWDVoltage").innerHTML = data[2];
if (data[17] == "true") {
fwd_vu_meter.setAttribute('data-val', 0);
} else {
document.getElementById("fwd_led_box").style.display = 'none';
}
2023-05-19 16:44:38 -04:00
}
if (data[16] == "0") {
document.getElementById("REFWatt").innerHTML = convert_power(data[3]);
document.getElementById("REFdBm").innerHTML = check_dbm(data[4]);
document.getElementById("REFVoltage").innerHTML = data[5];
if (data[18] == "true") {
document.getElementById("ref_led_box").style.display = 'inline-block';
ref_vu_meter.setAttribute('data-val', strtoint(data[3]*1000000));
} else {
document.getElementById("ref_led_box").style.display = 'none';
}
} else {
document.getElementById("REFWatt").innerHTML = "---";
document.getElementById("REFdBm").innerHTML = "---";
2023-05-20 06:16:41 -04:00
document.getElementById("REFVoltage").innerHTML = data[5];
if (data[18] == "true") {
ref_vu_meter.setAttribute('data-val', 0);
} else {
document.getElementById("ref_led_box").style.display = 'none';
}
2023-05-19 16:44:38 -04:00
}
2023-05-09 04:44:11 -04:00
document.getElementById("VSWRValue").innerHTML = data[6];
document.getElementById("RLValue").innerHTML = data[7];
document.getElementById("BANDValue").innerHTML = data[8];
2023-05-11 04:50:00 -04:00
document.getElementById("AntennaName").innerHTML = data[10];
2023-05-11 16:25:46 -04:00
document.getElementById("max_led_pwr_fwd").innerHTML = data[12];
document.getElementById("max_led_pwr_ref").innerHTML = data[13];
document.getElementById("max_led_vswr").innerHTML = data[14];
2023-05-11 10:18:51 -04:00
if (data[6] == "-1" || data[6] == "inf") {
document.getElementById("VSWRValue").innerHTML = "--";
2023-05-09 04:44:11 -04:00
document.getElementById("vswr_box").className = "box redbox";
2023-05-07 01:38:01 -04:00
} else {
2023-05-11 16:25:46 -04:00
document.getElementById("VSWRValue").innerHTML = data[6];
2023-05-11 04:50:00 -04:00
if (parseFloat(data[6]) >= parseFloat(data[9]) || data[6] == "inf") {
2023-05-09 04:44:11 -04:00
document.getElementById("vswr_box").className = "box redbox";
2023-05-11 04:50:00 -04:00
if (data[11] == "true") {
beep();
}
2023-05-09 04:44:11 -04:00
} else {
document.getElementById("vswr_box").className = "box";
}
2023-05-07 01:38:01 -04:00
}
2023-05-11 04:50:00 -04:00
if (data[0].startsWith('--') || data[1].startsWith('--')){
document.getElementById("fwd_box").className = "box redbox";
} else {
document.getElementById("fwd_box").className = "box";
}
if (data[3].startsWith('--') || data[4].startsWith('--')){
document.getElementById("ref_box").className = "box redbox";
} else {
document.getElementById("ref_box").className = "box";
}
2023-05-19 16:44:38 -04:00
if (data[19] == "true") {
document.getElementById("vswr_led_box").style.display = 'inline-block';
swr_vu_meter.setAttribute('data-val', strtoint(data[6]-1));
} else {
document.getElementById("vswr_led_box").style.display = 'none';
}
vumeter(fwd_vu_meter, {
"boxCount": 10,
"boxGapFraction": 0.25,
"max": strtoint(data[12]*1000000),
});
vumeter(ref_vu_meter, {
"boxCount": 10,
"boxGapFraction": 0.25,
"max": strtoint(data[13]*1000000),
});
2023-05-20 06:16:41 -04:00
vumeter(swr_vu_meter, {
"boxCount": 10,
"boxGapFraction": 0.25,
"max": 2,
});
2023-04-24 08:27:35 -04:00
}
};
2023-05-07 01:38:01 -04:00
xhttp.open("GET", "readDATA", true);
2023-04-24 08:27:35 -04:00
xhttp.send();
}
2023-05-11 10:18:51 -04:00
2023-04-24 08:27:35 -04:00
</script>
2023-05-11 10:18:51 -04:00
<body>
<div class="row">
<div id="band_box" class="box whitebox">
<h1><span id="AntennaName"></span></h1>
<h1>Band</h1>
<h2><span id="BANDValue">0</span></h2>
</div>
<div id="fwd_box" class="box">
<div class="innerbox">
<h1>FWD Power</h1>
<h2><span id="FWDWatt">0</span> </br><span id="FWDdBm">0</span> </br><span id="FWDVoltage">0</span></h2>
</div>
2023-05-19 16:44:38 -04:00
<div id="fwd_led_box" class="innerbox">
2023-05-11 16:25:46 -04:00
<span id="max_led_pwr_fwd">0</span> W
2023-05-11 10:18:51 -04:00
<section class="main">
<canvas id="fwd_vu_meter" width="60" height="200" data-val="0">No canvas</canvas>
</section>
2023-05-11 16:41:28 -04:00
0 W
2023-05-11 10:18:51 -04:00
</div>
</div>
<div id="ref_box" class="box">
<div class="innerbox">
<h1>REF Power</h1>
<h2><span id="REFWatt">0</span> </br><span id="REFdBm">0</span> </br><span id="REFVoltage">0</span></h2>
</div>
2023-05-19 16:44:38 -04:00
<div id="ref_led_box" class="innerbox">
2023-05-11 16:25:46 -04:00
<span id="max_led_pwr_ref">0</span> W
2023-05-11 10:18:51 -04:00
<section class="main">
<canvas id="ref_vu_meter" width="60" height="200" data-val="0">No canvas</canvas>
</section>
2023-05-11 16:41:28 -04:00
0 W
2023-05-11 10:18:51 -04:00
</div>
</div>
<div id="vswr_box" class="box">
<div class="innerbox">
<h1>VSWR</h1>
<h2><span id="VSWRValue">0</span></h2>
2023-05-12 07:43:34 -04:00
<h2>RL: <span id="RLValue">0</span> dB</h2>
2023-05-11 10:18:51 -04:00
</div>
2023-05-19 16:44:38 -04:00
<div id="vswr_led_box" class="innerbox">
2023-05-11 16:25:46 -04:00
<span id="max_led_vswr">0</span>
2023-05-11 10:18:51 -04:00
<section class="main">
<canvas id="swr_vu_meter" width="60" height="200" data-val="0">No canvas</canvas>
</section>
2023-05-11 16:41:28 -04:00
1
2023-05-11 10:18:51 -04:00
</div>
</div>
<form method='post' action='config'><button class='button' value='config' name='config' type='submit'>Configuration</button></form>
2023-04-24 08:27:35 -04:00
</body>
</html>
)=====";