Compare commits

..

No commits in common. "main" and "876842" have entirely different histories.
main ... 876842

20 changed files with 517 additions and 835 deletions

View file

@ -39,7 +39,6 @@ jobs:
submodules: recursive submodules: recursive
- uses: actions/setup-node@v4 - uses: actions/setup-node@v4
with: with:
token: ${{ secrets.GH_TOKEN }}
node-version: lts/* node-version: lts/*
cache: yarn cache: yarn
cache-dependency-path: '**/yarn.lock' cache-dependency-path: '**/yarn.lock'
@ -50,6 +49,9 @@ jobs:
~/node_modules ~/node_modules
~/.cache/ms-playwright ~/.cache/ms-playwright
key: ${{ runner.os }}-pio-playwright-${{ hashFiles('**/yarn.lock') }} key: ${{ runner.os }}-pio-playwright-${{ hashFiles('**/yarn.lock') }}
- uses: actions/setup-python@v5
with:
python-version: '>=3.10'
- name: Get current date - name: Get current date
id: dateAndTime id: dateAndTime
run: echo "dateAndTime=$(date +'%Y-%m-%d-%H:%M')" >> $GITHUB_OUTPUT run: echo "dateAndTime=$(date +'%Y-%m-%d-%H:%M')" >> $GITHUB_OUTPUT
@ -118,7 +120,7 @@ jobs:
output/littlefs.bin output/littlefs.bin
- name: Create release - name: Create release
if: github.event_name == 'push' && github.ref == 'refs/heads/main' if: github.event_name == 'push' && github.ref == 'refs/heads/main'
uses: https://code.forgejo.org/actions/forgejo-release@v2.5.3 uses: https://code.forgejo.org/actions/forgejo-release@v2.5.1
with: with:
url: 'https://git.btclock.dev/' url: 'https://git.btclock.dev/'
repo: '${{ github.repository }}' repo: '${{ github.repository }}'

View file

@ -1,7 +1,6 @@
# BTClock WebUI # BTClock WebUI
[![Latest release](https://git.btclock.dev/btclock/webui/badges/release.svg)](https://git.btclock.dev/btclock/webui/releases/latest) [![BTClock CI](https://github.com/btclock/webui/actions/workflows/workflow.yml/badge.svg)](https://github.com/btclock/webui2/actions/workflows/workflow.yml)
[![BTClock CI](https://git.btclock.dev/btclock/webui/badges/workflows/build.yaml/badge.svg)](https://git.btclock.dev/btclock/webui/actions?workflow=build.yaml&actor=0&status=0)
The web user-interface for the BTClock, based on Svelte-kit. It uses Bootstrap for the lay-out. The web user-interface for the BTClock, based on Svelte-kit. It uses Bootstrap for the lay-out.

Binary file not shown.

Before

Width:  |  Height:  |  Size: 70 KiB

After

Width:  |  Height:  |  Size: 70 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 65 KiB

After

Width:  |  Height:  |  Size: 65 KiB

View file

@ -5,7 +5,6 @@
"scripts": { "scripts": {
"dev": "vite dev", "dev": "vite dev",
"build": "vite build", "build": "vite build",
"build:test": "vite build --config vite.config.test.ts",
"preview": "vite preview", "preview": "vite preview",
"check": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json", "check": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json",
"check:watch": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json --watch", "check:watch": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json --watch",
@ -69,5 +68,6 @@
"es5-ext": ">=0.10.64", "es5-ext": ">=0.10.64",
"ws": ">=8.18.0", "ws": ">=8.18.0",
"micromatch": ">=4.0.8" "micromatch": ">=4.0.8"
} },
"packageManager": "yarn@1.22.21+sha1.1959a18351b811cdeedbd484a8f86c3cc3bbaf72"
} }

View file

@ -1,17 +1,30 @@
diff --git a/node_modules/@sveltejs/kit/src/exports/vite/index.js b/node_modules/@sveltejs/kit/src/exports/vite/index.js diff --git a/node_modules/@sveltejs/kit/src/exports/vite/index.js b/node_modules/@sveltejs/kit/src/exports/vite/index.js
index ddbe746..1d926a4 100644 index 21bc3d4..eef2db3 100644
--- a/node_modules/@sveltejs/kit/src/exports/vite/index.js --- a/node_modules/@sveltejs/kit/src/exports/vite/index.js
+++ b/node_modules/@sveltejs/kit/src/exports/vite/index.js +++ b/node_modules/@sveltejs/kit/src/exports/vite/index.js
@@ -658,9 +658,9 @@ async function kit({ svelte_config }) { @@ -648,9 +648,9 @@ async function kit({ svelte_config }) {
output: { output: {
format: inline ? 'iife' : 'esm', format: inline ? 'iife' : 'esm',
name: `__sveltekit_${version_hash}.app`, name: `__sveltekit_${version_hash}.app`,
- entryFileNames: ssr ? '[name].js' : `${prefix}/[name].[hash].${ext}`, - entryFileNames: ssr ? '[name].js' : `${prefix}/[name].[hash].${ext}`,
- chunkFileNames: ssr ? 'chunks/[name].js' : `${prefix}/chunks/[hash].${ext}`, - chunkFileNames: ssr ? 'chunks/[name].js' : `${prefix}/chunks/[name].[hash].${ext}`,
- assetFileNames: `${prefix}/assets/[name].[hash][extname]`, - assetFileNames: `${prefix}/assets/[name].[hash][extname]`,
+ entryFileNames: ssr ? '[name].js' : `${prefix}/[hash].${ext}`, + entryFileNames: ssr ? '[name].js' : `${prefix}/[hash].${ext}`,
+ chunkFileNames: ssr ? 'chunks/[name].js' : `${prefix}/c[hash].${ext}`, + chunkFileNames: ssr ? 'chunks/[name].js' : `${prefix}/chunks/[hash].${ext}`,
+ assetFileNames: `${prefix}/a[hash][extname]`, + assetFileNames: `${prefix}/assets/[hash][extname]`,
hoistTransitiveImports: false, hoistTransitiveImports: false,
sourcemapIgnoreList, sourcemapIgnoreList,
manualChunks: split ? undefined : () => 'bundle', manualChunks: split ? undefined : () => 'bundle',
@@ -665,9 +665,9 @@ async function kit({ svelte_config }) {
worker: {
rollupOptions: {
output: {
- entryFileNames: `${prefix}/workers/[name]-[hash].js`,
- chunkFileNames: `${prefix}/workers/chunks/[name]-[hash].js`,
- assetFileNames: `${prefix}/workers/assets/[name]-[hash][extname]`,
+ entryFileNames: `${prefix}/workers/[hash].js`,
+ chunkFileNames: `${prefix}/workers/chunks/[hash].js`,
+ assetFileNames: `${prefix}/workers/assets/[hash][extname]`,
hoistTransitiveImports: false
}
}

View file

@ -6,7 +6,7 @@ const config: PlaywrightTestConfig = {
timezoneId: 'Europe/Amsterdam' timezoneId: 'Europe/Amsterdam'
}, },
webServer: { webServer: {
command: 'npm run build:test && npm run preview', command: 'npm run build && npm run preview',
port: 4173 port: 4173
}, },
reporter: process.env.CI ? 'github' : 'list', reporter: process.env.CI ? 'github' : 'list',

View file

@ -20,19 +20,14 @@
const setTextColor = () => { const setTextColor = () => {
$settings.invertedColor = !$settings.invertedColor; $settings.invertedColor = !$settings.invertedColor;
$settings.fgColor = $settings.invertedColor ? 65535 : 0;
$settings.bgColor = $settings.invertedColor ? 0 : 65535;
}; };
const textColorOptions: [string, boolean][] = [ const textColorOptions: [string, boolean][] = [
[$_('colors.black') + ' on ' + $_('colors.white'), false], [$_('colors.black') + ' on ' + $_('colors.white'), false],
[$_('colors.white') + ' on ' + $_('colors.black'), true] [$_('colors.white') + ' on ' + $_('colors.black'), true]
]; ];
const fontPreferenceOptions: [string, string][] = $settings.availableFonts?.map((font) => [
$_(`fonts.${font}`) !== `fonts.${font}`
? $_(`fonts.${font}`)
: font.charAt(0).toUpperCase() + font.slice(1),
font
]);
</script> </script>
<Row> <Row>
@ -50,14 +45,6 @@
on:change={setTextColor} on:change={setTextColor}
/> />
<SettingsSelect
id="fontName"
label={$_('section.settings.fontName')}
bind:value={$settings.fontName}
options={fontPreferenceOptions}
size={$uiSettings.inputSize}
/>
<SettingsInput <SettingsInput
id="timePerScreen" id="timePerScreen"
label={$_('section.settings.timePerScreen')} label={$_('section.settings.timePerScreen')}
@ -115,7 +102,7 @@
max={4095} max={4095}
step={1} step={1}
size={$uiSettings.inputSize} size={$uiSettings.inputSize}
onChange={onFlBrightnessChange} on:change={onFlBrightnessChange}
/> />
<SettingsInput <SettingsInput
@ -190,14 +177,12 @@
size={$uiSettings.inputSize} size={$uiSettings.inputSize}
/> />
{#if $settings.hasLightLevel} <SettingsSwitch
<SettingsSwitch id="flOffWhenDark"
id="flOffWhenDark" bind:checked={$settings.flOffWhenDark}
bind:checked={$settings.flOffWhenDark} label={$_('section.settings.flOffWhenDark')}
label={$_('section.settings.flOffWhenDark')} size={$uiSettings.inputSize}
size={$uiSettings.inputSize} />
/>
{/if}
{/if} {/if}
</Row> </Row>
</ToggleHeader> </ToggleHeader>

View file

@ -13,7 +13,6 @@
export let miningPoolMap: Map<string, string>; export let miningPoolMap: Map<string, string>;
let validBitaxe = false; let validBitaxe = false;
let validLocalPool = false;
const testBitaxe = async () => { const testBitaxe = async () => {
try { try {
const response = await fetch(`http://${$settings.bitaxeHostname}/api/system/info`); const response = await fetch(`http://${$settings.bitaxeHostname}/api/system/info`);
@ -62,49 +61,6 @@
miningPoolMap.get(pool) || pool, miningPoolMap.get(pool) || pool,
pool pool
]); ]);
const testLocalPool = async () => {
try {
const controller = new AbortController();
const timeoutId = setTimeout(() => controller.abort(), 1000);
const response = await fetch(
`http://${$settings.localPoolEndpoint}/api/client/${$settings.miningPoolUser}`,
{ signal: controller.signal }
);
clearTimeout(timeoutId);
if (!response.ok) {
dispatch('showToast', {
color: 'danger',
text: `Failed to connect to local pool! status: ${response.status}`
});
validLocalPool = false;
throw new Error();
}
const poolInfo = await response.json();
dispatch('showToast', {
color: 'success',
text: `Can connect to local public pool, ${poolInfo.workersCount} workers`
});
validLocalPool = true;
} catch (error) {
if (error.name === 'AbortError') {
dispatch('showToast', {
color: 'danger',
text: `Connection to local pool timed out after 1 second`
});
} else {
dispatch('showToast', {
color: 'danger',
text: `Failed to connect to local pool, check the endpoint and make sure you are connected to the same network.`
});
}
console.error('Failed to fetch local pool info:', error);
validLocalPool = false;
}
};
</script> </script>
<Row> <Row>
@ -113,64 +69,6 @@
bind:isOpen bind:isOpen
defaultOpen={false} defaultOpen={false}
> >
<!--- Time based do not disturb settings -->
<SettingsSwitch
id="timeBasedDnd"
label={$_('section.settings.timeBasedDnd')}
bind:checked={$settings.dnd.timeBasedEnabled}
size={$uiSettings.inputSize}
/>
{#if $settings.dnd.timeBasedEnabled}
<Row>
<Col>
<SettingsInput
id="dndStartHour"
type="number"
min="0"
max="23"
label={$_('section.settings.dndStartHour')}
bind:value={$settings.dnd.startHour}
size={$uiSettings.inputSize}
/>
</Col>
<Col>
<SettingsInput
id="dndStartMinute"
type="number"
min="0"
max="59"
label={$_('section.settings.dndStartMinute')}
bind:value={$settings.dnd.startMinute}
size={$uiSettings.inputSize}
/>
</Col>
</Row>
<Row>
<Col>
<SettingsInput
id="dndEndHour"
type="number"
min="0"
max="23"
label={$_('section.settings.dndEndHour')}
bind:value={$settings.dnd.endHour}
size={$uiSettings.inputSize}
/>
</Col>
<Col>
<SettingsInput
id="dndEndMinute"
type="number"
min="0"
max="59"
label={$_('section.settings.dndEndMinute')}
bind:value={$settings.dnd.endMinute}
size={$uiSettings.inputSize}
/>
</Col>
</Row>
{/if}
<!-- BitAxe Settings --> <!-- BitAxe Settings -->
{#if 'bitaxeEnabled' in $settings} {#if 'bitaxeEnabled' in $settings}
<Row class="mb-3"> <Row class="mb-3">
@ -222,21 +120,6 @@
size={$uiSettings.inputSize} size={$uiSettings.inputSize}
selectClass={$uiSettings.selectClass} selectClass={$uiSettings.selectClass}
/> />
{#if $settings.miningPoolName === 'local_public_pool'}
<SettingsInput
id="localPoolEndpoint"
label={$_('section.settings.localPoolEndpoint', { default: 'Local Pool Endpoint' })}
bind:value={$settings.localPoolEndpoint}
placeholder="umbrel.local:2019"
required={true}
valid={validLocalPool}
size={$uiSettings.inputSize}
>
<Button type="button" color="success" on:click={testLocalPool}>
{$_('test', { default: 'Test' })}
</Button>
</SettingsInput>
{/if}
<SettingsInput <SettingsInput
id="miningPoolUser" id="miningPoolUser"
label={$_('section.settings.miningPoolUser')} label={$_('section.settings.miningPoolUser')}

View file

@ -4,7 +4,6 @@
import { Row, Col } from '@sveltestrap/sveltestrap'; import { Row, Col } from '@sveltestrap/sveltestrap';
import ToggleHeader from '../ToggleHeader.svelte'; import ToggleHeader from '../ToggleHeader.svelte';
import { uiSettings } from '$lib/uiSettings'; import { uiSettings } from '$lib/uiSettings';
import { DataSourceType } from '$lib/types/dataSource';
export let settings; export let settings;
export let isOpen = false; export let isOpen = false;
@ -100,7 +99,7 @@
{/each} {/each}
{/if} {/if}
</Row> </Row>
{#if $settings.actCurrencies && $settings.dataSource == DataSourceType.BTCLOCK_SOURCE} {#if $settings.actCurrencies && $settings.useNostr !== true}
<Row> <Row>
<h5>{$_('section.settings.currencies')}</h5> <h5>{$_('section.settings.currencies')}</h5>
<small>{$_('restartRequired')}</small> <small>{$_('restartRequired')}</small>

View file

@ -67,13 +67,7 @@
"dataSource": { "dataSource": {
"nostr": "Nostr-Verlag", "nostr": "Nostr-Verlag",
"custom": "Benutzerdefinierter dataquelle" "custom": "Benutzerdefinierter dataquelle"
}, }
"fontName": "Schriftart",
"timeBasedDnd": "Aktivieren Sie den Zeitplan „Bitte nicht stören“.",
"dndStartHour": "Startstunde",
"dndStartMinute": "Startminute",
"dndEndHour": "Endstunde",
"dndEndMinute": "Schlussminute"
}, },
"control": { "control": {
"systemInfo": "Systeminfo", "systemInfo": "Systeminfo",
@ -101,9 +95,7 @@
"wifiSignalStrength": "WiFi-Signalstärke", "wifiSignalStrength": "WiFi-Signalstärke",
"wsDataConnection": "BTClock-Datenquelle verbindung", "wsDataConnection": "BTClock-Datenquelle verbindung",
"lightSensor": "Lichtsensor", "lightSensor": "Lichtsensor",
"nostrConnection": "Nostr Relay-Verbindung", "nostrConnection": "Nostr Relay-Verbindung"
"doNotDisturb": "Bitte nicht stören",
"timeBasedDnd": "Zeitbasierter Zeitplan"
}, },
"firmwareUpdater": { "firmwareUpdater": {
"fileUploadSuccess": "Datei erfolgreich hochgeladen, Gerät neu gestartet. WebUI in {countdown} Sekunden neu geladen", "fileUploadSuccess": "Datei erfolgreich hochgeladen, Gerät neu gestartet. WebUI in {countdown} Sekunden neu geladen",

View file

@ -84,13 +84,7 @@
}, },
"thirdPartySource": "Use mempool.space/coincap.io", "thirdPartySource": "Use mempool.space/coincap.io",
"ceDisableSSL": "Disable SSL", "ceDisableSSL": "Disable SSL",
"ceEndpoint": "Endpoint hostname", "ceEndpoint": "Endpoint hostname"
"fontName": "Font",
"timeBasedDnd": "Enable Do Not Disturb time schedule",
"dndStartHour": "Start hour",
"dndStartMinute": "Start minute",
"dndEndHour": "End hour",
"dndEndMinute": "End minute"
}, },
"control": { "control": {
"systemInfo": "System info", "systemInfo": "System info",
@ -120,9 +114,7 @@
"wifiSignalStrength": "WiFi Signal strength", "wifiSignalStrength": "WiFi Signal strength",
"wsDataConnection": "BTClock data-source connection", "wsDataConnection": "BTClock data-source connection",
"lightSensor": "Light sensor", "lightSensor": "Light sensor",
"nostrConnection": "Nostr Relay connection", "nostrConnection": "Nostr Relay connection"
"doNotDisturb": "Do not disturb",
"timeBasedDnd": "Time-based schedule"
}, },
"firmwareUpdater": { "firmwareUpdater": {
"fileUploadFailed": "File upload failed. Make sure you have selected the correct file and try again.", "fileUploadFailed": "File upload failed. Make sure you have selected the correct file and try again.",

View file

@ -66,13 +66,7 @@
"dataSource": { "dataSource": {
"nostr": "editorial nostr", "nostr": "editorial nostr",
"custom": "Punto final personalizado" "custom": "Punto final personalizado"
}, }
"fontName": "Fuente",
"timeBasedDnd": "Habilitar el horario de No molestar",
"dndStartHour": "Hora de inicio",
"dndStartMinute": "Minuto de inicio",
"dndEndHour": "Hora final",
"dndEndMinute": "Minuto final"
}, },
"control": { "control": {
"turnOff": "Apagar", "turnOff": "Apagar",
@ -100,9 +94,7 @@
"wifiSignalStrength": "Fuerza de la señal WiFi", "wifiSignalStrength": "Fuerza de la señal WiFi",
"wsDataConnection": "Conexión de fuente de datos BTClock", "wsDataConnection": "Conexión de fuente de datos BTClock",
"lightSensor": "Sensor de luz", "lightSensor": "Sensor de luz",
"nostrConnection": "Conexión de relé Nostr", "nostrConnection": "Conexión de relé Nostr"
"doNotDisturb": "No molestar",
"timeBasedDnd": "Horario basado en el tiempo"
}, },
"firmwareUpdater": { "firmwareUpdater": {
"fileUploadSuccess": "Archivo cargado exitosamente, reiniciando el dispositivo. Recargando WebUI en {countdown} segundos", "fileUploadSuccess": "Archivo cargado exitosamente, reiniciando el dispositivo. Recargando WebUI en {countdown} segundos",

View file

@ -58,13 +58,7 @@
"hideAll": "Alles verbergen", "hideAll": "Alles verbergen",
"flOffWhenDark": "Displaylicht uit als het donker is", "flOffWhenDark": "Displaylicht uit als het donker is",
"luxLightToggleText": "Stel in op 0 om uit te schakelen", "luxLightToggleText": "Stel in op 0 om uit te schakelen",
"verticalDesc": "Verticale schermbeschrijving", "verticalDesc": "Verticale schermbeschrijving"
"fontName": "Lettertype",
"timeBasedDnd": "Schakel het tijdschema Niet storen in",
"dndStartHour": "Begin uur",
"dndStartMinute": "Beginminuut",
"dndEndHour": "Eind uur",
"dndEndMinute": "Einde minuut"
}, },
"control": { "control": {
"systemInfo": "Systeeminformatie", "systemInfo": "Systeeminformatie",
@ -91,9 +85,7 @@
"wifiSignalStrength": "WiFi signaalsterkte", "wifiSignalStrength": "WiFi signaalsterkte",
"wsDataConnection": "BTClock-gegevensbron verbinding", "wsDataConnection": "BTClock-gegevensbron verbinding",
"lightSensor": "Licht sensor", "lightSensor": "Licht sensor",
"nostrConnection": "Nostr Relay-verbinding", "nostrConnection": "Nostr Relay-verbinding"
"doNotDisturb": "Niet storen",
"timeBasedDnd": "Op tijd gebaseerd schema"
}, },
"firmwareUpdater": { "firmwareUpdater": {
"fileUploadSuccess": "Bestand geüpload, apparaat herstart. WebUI opnieuw geladen over {countdown} seconden", "fileUploadSuccess": "Bestand geüpload, apparaat herstart. WebUI opnieuw geladen over {countdown} seconden",

View file

@ -105,6 +105,7 @@
setupObserver(); setupObserver();
const connectEventSource = () => { const connectEventSource = () => {
console.log('Connecting to EventSource');
const evtSource = new EventSource(`${PUBLIC_BASE_URL}/events`); const evtSource = new EventSource(`${PUBLIC_BASE_URL}/events`);
evtSource.addEventListener('status', (e) => { evtSource.addEventListener('status', (e) => {

View file

@ -30,8 +30,7 @@
['public_pool', 'public-pool.io'], ['public_pool', 'public-pool.io'],
['gobrrr_pool', 'Go Brrr pool'], ['gobrrr_pool', 'Go Brrr pool'],
['ckpool', 'CKPool'], ['ckpool', 'CKPool'],
['eu_ckpool', 'EU CKPool'], ['eu_ckpool', 'EU CKPool']
['local_public_pool', 'Public Pool (local)']
]); ]);
const dispatch = createEventDispatcher(); const dispatch = createEventDispatcher();
@ -115,13 +114,9 @@
<CardHeader> <CardHeader>
<div class="float-end"> <div class="float-end">
<small> <small>
<button type="button" on:click={showAll} id="showAllBtn" <button type="button" on:click={showAll}>{$_('section.settings.showAll')}</button>
>{$_('section.settings.showAll')}</button
>
| |
<button type="button" on:click={hideAll} id="hideAllBtn" <button type="button" on:click={hideAll}>{$_('section.settings.hideAll')}</button>
>{$_('section.settings.hideAll')}</button
>
</small> </small>
</div> </div>
<CardTitle>{$_('section.settings.title')}</CardTitle> <CardTitle>{$_('section.settings.title')}</CardTitle>

View file

@ -97,16 +97,6 @@
} }
}; };
const toggleDoNotDisturb = (currentStatus: boolean) => (e: Event) => {
e.preventDefault();
console.log(currentStatus);
if (!currentStatus) {
fetch(`${PUBLIC_BASE_URL}/api/dnd/enable`);
} else {
fetch(`${PUBLIC_BASE_URL}/api/dnd/disable`);
}
};
export let xs = 12; export let xs = 12;
export let sm = xs; export let sm = xs;
export let md = sm; export let md = sm;
@ -168,7 +158,7 @@
<hr /> <hr />
{#if $status.data} {#if $status.data}
<section class={lightMode ? 'lightMode' : 'darkMode'} style="position: relative;"> <section class={lightMode ? 'lightMode' : 'darkMode'} style="position: relative;">
{#if $status.isUpdating === false && ($status.isFake ?? false) === false} {#if $status.isUpdating === false && $status.isFake === false}
<div class="connection-lost-overlay"> <div class="connection-lost-overlay">
<div class="overlay-content"> <div class="overlay-content">
<i class="bi bi-wifi-off"></i> <i class="bi bi-wifi-off"></i>
@ -195,27 +185,7 @@
>{#if $status.timerRunning}&#9205; {$_('timer.running')}{:else}&#9208; {$_( >{#if $status.timerRunning}&#9205; {$_('timer.running')}{:else}&#9208; {$_(
'timer.stopped' 'timer.stopped'
)}{/if}</a )}{/if}</a
><br />
{$_('section.status.doNotDisturb')}:
<a
id="dndStatusText"
href={'#'}
style="cursor: pointer"
tabindex="0"
role="button"
aria-pressed="false"
on:click={toggleDoNotDisturb($status.dnd?.enabled)}
> >
{#if $status.dnd?.active}&#9205; {$_('on')}{:else}&#9208; {$_('off')}{/if}</a
>
<small>
{#if $status.dnd?.timeBasedEnabled}
{$_('section.status.timeBasedDnd')} ( {$settings.dnd
.startHour}:{$settings.dnd.startMinute.toString().padStart(2, '0')} - {$settings
.dnd.endHour}:{$settings.dnd.endMinute.toString().padStart(2, '0')} )
{/if}
</small>
{/if} {/if}
{/if} {/if}
<hr /> <hr />

View file

@ -1,38 +1,3 @@
interface Page {
route: (url: string, handler: (route: Route) => Promise<void>) => Promise<void>;
}
interface Route {
fulfill: (response: {
json?: typeof statusJson | typeof settingsJson | typeof latestReleaseFake;
status?: number;
headers?: Record<string, string>;
body?: ReadableStream;
}) => Promise<void>;
}
export const fetchLatestBlockHeight = async () => {
const response = await fetch('https://ws.btclock.dev/api/lastblock');
const blockHeight = await response.text();
return ['BLOCK/HEIGHT', ...blockHeight.trim().split('')];
};
export const fetchLatestRelease = async () => {
try {
const response = await fetch(
'https://git.btclock.dev/api/v1/repos/btclock/btclock_v3/releases/latest'
);
if (!response.ok) throw new Error('Failed to fetch latest release');
const data = await response.json();
settingsJson.gitTag = data.tag_name;
return data;
} catch (error) {
console.warn('Failed to fetch latest release, using fallback:', error);
settingsJson.gitTag = latestReleaseFake.tag_name;
return latestReleaseFake;
}
};
export const statusJson = { export const statusJson = {
currentScreen: 20, currentScreen: 20,
numScreens: 7, numScreens: 7,
@ -48,7 +13,7 @@ export const statusJson = {
nostr: true nostr: true
}, },
rssi: -66, rssi: -66,
data: ['BLOCK/HEIGHT', '0', '0', '0', '0', '0', '0'], data: ['BLOCK/HEIGHT', '8', '7', '6', '5', '4', '3'],
currency: 'USD', currency: 'USD',
leds: [ leds: [
{ red: 0, green: 0, blue: 0, hex: '#000000' }, { red: 0, green: 0, blue: 0, hex: '#000000' },
@ -57,14 +22,7 @@ export const statusJson = {
{ red: 0, green: 0, blue: 0, hex: '#000000' } { red: 0, green: 0, blue: 0, hex: '#000000' }
], ],
isUpdating: true, isUpdating: true,
isFake: true, isFake: true
dnd: {
enabled: true,
timeBasedEnabled: true,
startTime: '23:00',
endTime: '7:00',
active: true
}
}; };
export const settingsJson = { export const settingsJson = {
@ -90,7 +48,7 @@ export const settingsJson = {
ip: '192.168.20.231', ip: '192.168.20.231',
txPower: 78, txPower: 78,
gitRev: '25d8b92bcbc8938417c140355ea3ba99ff9eb4b7', gitRev: '25d8b92bcbc8938417c140355ea3ba99ff9eb4b7',
gitTag: '3.2.27', gitTag: '3.2.23',
bitaxeEnabled: false, bitaxeEnabled: false,
bitaxeHostname: 'bitaxe1', bitaxeHostname: 'bitaxe1',
miningPoolStats: false, miningPoolStats: false,
@ -150,18 +108,8 @@ export const settingsJson = {
'ckpool', 'ckpool',
'eu_ckpool' 'eu_ckpool'
], ],
dnd: {
enabled: false,
timeBasedEnabled: true,
startHour: 23,
startMinute: 0,
endHour: 7,
endMinute: 0
},
availableFonts: ['antonio', 'oswald'],
invertedColor: false, invertedColor: false,
isLoaded: true, isLoaded: true
isFake: true
}; };
export const latestReleaseFake = { export const latestReleaseFake = {
@ -188,11 +136,7 @@ export const latestReleaseFake = {
} }
}; };
export const initMock = async ({ page }: { page: Page }) => { export const initMock = async ({ page }) => {
// Update status with latest block height
statusJson.data = await fetchLatestBlockHeight();
const latestRelease = await fetchLatestRelease();
await page.route('*/**/api/status', async (route) => { await page.route('*/**/api/status', async (route) => {
await route.fulfill({ json: statusJson }); await route.fulfill({ json: statusJson });
}); });
@ -223,35 +167,20 @@ export const initMock = async ({ page }: { page: Page }) => {
await route.fulfill({ json: settingsJson }); await route.fulfill({ json: settingsJson });
}); });
await page.route('**/events', async (route) => { await page.route('**/events', (route) => {
const newStatus = statusJson; const newStatus = statusJson;
newStatus.data = ['BLOCK/HEIGHT', '8', '0', '0', '8', '1', '5']; newStatus.data = ['BLOCK/HEIGHT', '8', '0', '0', '8', '1', '5'];
newStatus.isUpdating = true; newStatus.isUpdating = true;
// Format the SSE message correctly // Respond with a custom SSE message
const sseMessage = `data: ${JSON.stringify(newStatus)}\n\n`; route.fulfill({
// Create a readable stream for SSE
const stream = new ReadableStream({
start(controller) {
controller.enqueue(new TextEncoder().encode(sseMessage));
// Keep the connection open
// controller.close(); // Don't close if you want to send more events
}
});
await route.fulfill({
status: 200, status: 200,
headers: { contentType: 'text/event-stream',
'Content-Type': 'text/event-stream', json: `${JSON.stringify(newStatus)}\n\n`
'Cache-Control': 'no-cache',
Connection: 'keep-alive'
},
body: stream
}); });
}); });
await page.route('**/api/v1/repos/btclock/btclock_v3/releases/latest', async (route) => { await page.route('**/api/v1/repos/btclock/btclock_v3/releases/latest', async (route) => {
await route.fulfill({ json: latestRelease }); await route.fulfill({ json: latestReleaseFake });
}); });
}; };

View file

@ -1,18 +0,0 @@
import { sveltekit } from '@sveltejs/kit/vite';
import { defineConfig } from 'vite';
export default defineConfig({
plugins: [sveltekit()],
build: {
sourcemap: true,
minify: false,
rollupOptions: {
output: {
manualChunks: undefined // Disable code splitting
}
}
},
test: {
include: ['tests/**/*.{test,spec}.{js,ts}']
}
});

958
yarn.lock

File diff suppressed because it is too large Load diff