Compare commits
1 Commits
feature/mo
...
feature/mo
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4ecdd8027b |
@@ -102,9 +102,6 @@ dist/
|
|||||||
false outage)
|
false outage)
|
||||||
- Clickable service URLs
|
- Clickable service URLs
|
||||||
- Canvas-based sparkline rendering with devicePixelRatio scaling
|
- Canvas-based sparkline rendering with devicePixelRatio scaling
|
||||||
- Mobile detection: viewports narrower than 768px show a friendly "not yet
|
|
||||||
available on mobile" message instead of the monitoring UI (no polling or
|
|
||||||
network requests on mobile)
|
|
||||||
- Zero runtime dependencies: all resources bundled into build artifacts
|
- Zero runtime dependencies: all resources bundled into build artifacts
|
||||||
|
|
||||||
## Deployment
|
## Deployment
|
||||||
@@ -128,8 +125,6 @@ properties.
|
|||||||
|
|
||||||
## Limitations
|
## Limitations
|
||||||
|
|
||||||
- **Mobile**: Viewports below 768px wide show a static "not yet available"
|
|
||||||
message. The full monitoring UI requires a desktop-width browser.
|
|
||||||
- **CORS**: Some hosts may block cross-origin HEAD requests. The app uses
|
- **CORS**: Some hosts may block cross-origin HEAD requests. The app uses
|
||||||
`no-cors` mode which allows the request but provides opaque responses. Latency
|
`no-cors` mode which allows the request but provides opaque responses. Latency
|
||||||
is still measurable based on request timing.
|
is still measurable based on request timing.
|
||||||
|
|||||||
48
src/main.js
48
src/main.js
@@ -1126,43 +1126,27 @@ function handleResize(state) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// --- Mobile Detection --------------------------------------------------------
|
|
||||||
|
|
||||||
const MOBILE_BREAKPOINT = 768;
|
|
||||||
|
|
||||||
function isMobileViewport() {
|
|
||||||
return window.innerWidth < MOBILE_BREAKPOINT;
|
|
||||||
}
|
|
||||||
|
|
||||||
function buildMobileUI() {
|
|
||||||
const app = document.getElementById("app");
|
|
||||||
app.innerHTML = `
|
|
||||||
<div class="mx-auto px-[5%] py-8">
|
|
||||||
<header class="mb-8">
|
|
||||||
<h1 class="text-3xl font-bold text-white"><a href="https://git.eeqj.de/sneak/netwatch" target="_blank" rel="noopener" class="underline decoration-dashed decoration-gray-500 underline-offset-4">NetWatch</a> by <a href="https://sneak.berlin" target="_blank" rel="noopener" class="text-blue-400 underline hover:text-blue-300">@sneak</a></h1>
|
|
||||||
<p class="text-gray-400 text-sm mt-2">Real-time network latency monitor</p>
|
|
||||||
</header>
|
|
||||||
<div class="flex items-center justify-center min-h-[60vh]">
|
|
||||||
<div class="bg-gray-800/70 border border-gray-700/50 rounded-lg p-8 max-w-md text-center">
|
|
||||||
<p class="text-4xl mb-4">📡</p>
|
|
||||||
<p class="text-xl font-semibold text-white mb-2">Not yet available on mobile</p>
|
|
||||||
<p class="text-gray-400 text-sm">NetWatch requires a wider viewport to display latency charts and monitoring data. Please visit on a desktop browser.</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<footer class="mt-8 text-center text-gray-600 text-xs">
|
|
||||||
<p><a href="https://git.eeqj.de/sneak/netwatch/commit/${__COMMIT_FULL__}" target="_blank" rel="noopener" class="text-gray-600 hover:text-gray-400">${__COMMIT_HASH__}</a></p>
|
|
||||||
</footer>
|
|
||||||
</div>`;
|
|
||||||
}
|
|
||||||
|
|
||||||
// --- Bootstrap ---------------------------------------------------------------
|
// --- Bootstrap ---------------------------------------------------------------
|
||||||
|
|
||||||
async function init() {
|
async function init() {
|
||||||
log.info("NetWatch starting");
|
log.info("NetWatch starting");
|
||||||
|
|
||||||
if (isMobileViewport()) {
|
// Mobile detection — show a friendly message and bail out early
|
||||||
log.info("Mobile viewport detected — skipping monitoring");
|
if (window.innerWidth < 768) {
|
||||||
buildMobileUI();
|
const app = document.getElementById("app");
|
||||||
|
app.innerHTML = `
|
||||||
|
<div class="mx-auto px-[5%] py-8">
|
||||||
|
<header class="mb-8">
|
||||||
|
<h1 class="text-3xl font-bold text-white"><a href="https://git.eeqj.de/sneak/netwatch" target="_blank" rel="noopener" class="underline decoration-dashed decoration-gray-500 underline-offset-4">NetWatch</a> by <a href="https://sneak.berlin" target="_blank" rel="noopener" class="text-blue-400 underline hover:text-blue-300">@sneak</a></h1>
|
||||||
|
<p class="text-gray-400 mt-2">Real-time network connectivity monitor</p>
|
||||||
|
</header>
|
||||||
|
<div style="display:flex;align-items:center;justify-content:center;min-height:50vh;">
|
||||||
|
<div style="background:#1f2937;border:1px solid #374151;border-radius:12px;padding:2rem 1.5rem;text-align:center;max-width:90vw;">
|
||||||
|
<p style="font-size:1.25rem;color:#e5e7eb;margin:0;">Not yet available on mobile.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>`;
|
||||||
|
log.info("Mobile viewport detected — skipping monitor startup");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -21,96 +21,3 @@ body {
|
|||||||
rgba(255, 255, 255, 0) 100%
|
rgba(255, 255, 255, 0) 100%
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ---- Mobile responsive layout (portrait / narrow viewports) ---- */
|
|
||||||
@media (max-width: 768px) {
|
|
||||||
/* Header: stack title and controls vertically */
|
|
||||||
header .flex.items-center.justify-between {
|
|
||||||
flex-direction: column;
|
|
||||||
align-items: flex-start !important;
|
|
||||||
gap: 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
header .flex.flex-col.items-end {
|
|
||||||
align-items: flex-start !important;
|
|
||||||
flex-direction: row;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
gap: 0.75rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Pause button: smaller on mobile */
|
|
||||||
#pause-btn {
|
|
||||||
padding: 0.5rem 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
#pause-btn svg {
|
|
||||||
width: 1.25rem;
|
|
||||||
height: 1.25rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
#pause-text {
|
|
||||||
font-size: 0.875rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Summary box: wrap into a grid for readability */
|
|
||||||
#summary {
|
|
||||||
display: flex;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
gap: 0.25rem 0.5rem;
|
|
||||||
justify-content: center;
|
|
||||||
line-height: 1.6;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Hide the pipe separators on mobile */
|
|
||||||
#summary .text-gray-600.mx-3 {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Host row: stack vertically */
|
|
||||||
.host-row .flex.items-center.gap-4 {
|
|
||||||
flex-direction: column;
|
|
||||||
align-items: stretch !important;
|
|
||||||
gap: 0.5rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Info section: full width, remove fixed width */
|
|
||||||
.host-row .w-\[420px\] {
|
|
||||||
width: 100% !important;
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: 1fr auto;
|
|
||||||
align-items: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Host name row with dot */
|
|
||||||
.host-row .flex.items-center.gap-2.min-w-\[200px\] {
|
|
||||||
min-width: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Latency value: slightly smaller on mobile */
|
|
||||||
.host-row .latency-value {
|
|
||||||
font-size: 1.875rem;
|
|
||||||
line-height: 2.25rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Sparkline: full width below the info */
|
|
||||||
.host-row .sparkline-container {
|
|
||||||
width: 100%;
|
|
||||||
flex-shrink: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Pin button: inline with the host info */
|
|
||||||
.host-row .pin-btn {
|
|
||||||
position: absolute;
|
|
||||||
right: 0.5rem;
|
|
||||||
top: 0.5rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.host-row {
|
|
||||||
position: relative;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Footer legend: wrap nicely */
|
|
||||||
footer p {
|
|
||||||
line-height: 1.8;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
Reference in New Issue
Block a user