Compare commits
No commits in common. "main" and "876842" have entirely different histories.
20 changed files with 517 additions and 835 deletions
|
@ -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 }}'
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
# BTClock WebUI
|
# BTClock WebUI
|
||||||
|
|
||||||
[](https://git.btclock.dev/btclock/webui/releases/latest)
|
[](https://github.com/btclock/webui2/actions/workflows/workflow.yml)
|
||||||
[](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 |
|
@ -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"
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
||||||
|
}
|
||||||
|
}
|
|
@ -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',
|
||||||
|
|
|
@ -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>
|
||||||
|
|
|
@ -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')}
|
||||||
|
|
|
@ -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>
|
||||||
|
|
|
@ -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",
|
||||||
|
|
|
@ -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.",
|
||||||
|
|
|
@ -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",
|
||||||
|
|
|
@ -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",
|
||||||
|
|
|
@ -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) => {
|
||||||
|
|
|
@ -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>
|
||||||
|
|
|
@ -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}⏵ {$_('timer.running')}{:else}⏸ {$_(
|
>{#if $status.timerRunning}⏵ {$_('timer.running')}{:else}⏸ {$_(
|
||||||
'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}⏵ {$_('on')}{:else}⏸ {$_('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 />
|
||||||
|
|
|
@ -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 });
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
|
@ -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}']
|
|
||||||
}
|
|
||||||
});
|
|
Loading…
Add table
Reference in a new issue