Use runes and save config in localStorage
This commit is contained in:
parent
013089da11
commit
2603c57018
|
@ -1,48 +1,54 @@
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { onDestroy, onMount } from "svelte"
|
import { onDestroy, onMount } from "svelte"
|
||||||
import { writable } from "svelte/store"
|
import { browser } from "$app/environment"
|
||||||
import type { Writable } from "svelte/store"
|
|
||||||
|
|
||||||
const BREAK_INTERVAL_STORE: Writable<number> = writable(20)
|
let BREAK_INTERVAL_STORE = $state(browser ? (localStorage.getItem('breakInterval') || 20) : 20)
|
||||||
const MINI_BREAK_DURATION_STORE: Writable<number> = writable(20)
|
let MINI_BREAK_DURATION_STORE = $state(browser ? (localStorage.getItem('miniBreakDuration') || 20) : 20)
|
||||||
const LONG_BREAK_DURATION_STORE: Writable<number> = writable(5)
|
let LONG_BREAK_DURATION_STORE = $state(browser ? (localStorage.getItem('longBreakDuration') || 5) : 5)
|
||||||
const SOUND_ENABLED: Writable<boolean> = writable(true)
|
let SOUND_ENABLED = $state(browser ? ((localStorage.getItem('soundEnabled') || 'true') === 'true') : true)
|
||||||
const NOTIFICATIONS_ENABLED: Writable<boolean> = writable(true)
|
let NOTIFICATIONS_ENABLED = $state(browser ? ((localStorage.getItem('notificationsEnabled') || 'true') === 'true') : true)
|
||||||
|
|
||||||
let BREAK_INTERVAL: number
|
let BREAK_INTERVAL: number
|
||||||
let MINI_BREAK_DURATION: number
|
let MINI_BREAK_DURATION: number
|
||||||
let LONG_BREAK_DURATION: number
|
let LONG_BREAK_DURATION: number
|
||||||
let SOUND_ENABLED_STATE: boolean
|
|
||||||
let NOTIFICATIONS_ENABLED_STATE: boolean
|
|
||||||
|
|
||||||
let timer: number | null = null
|
let timer: number | null = null
|
||||||
|
|
||||||
const state: Writable<string> = writable("Ready")
|
let timerState = $state("Ready")
|
||||||
const timeLeftDisplay: Writable<string> = writable("")
|
let timeLeftDisplay = $state("")
|
||||||
const timeLeft: Writable<number> = writable($BREAK_INTERVAL_STORE)
|
let timeLeft = $state(BREAK_INTERVAL_STORE)
|
||||||
|
|
||||||
BREAK_INTERVAL_STORE.subscribe((value) => {
|
$effect(() => {
|
||||||
BREAK_INTERVAL = value * 60 * 1000
|
BREAK_INTERVAL = BREAK_INTERVAL_STORE * 60 * 1000
|
||||||
timeLeft.set(BREAK_INTERVAL)
|
timeLeft = BREAK_INTERVAL
|
||||||
timeLeftDisplay.set(formatTime(BREAK_INTERVAL))
|
timeLeftDisplay = formatTime(BREAK_INTERVAL)
|
||||||
|
localStorage.setItem("breakInterval", BREAK_INTERVAL_STORE)
|
||||||
|
})
|
||||||
|
|
||||||
|
$effect(() => {
|
||||||
|
MINI_BREAK_DURATION = MINI_BREAK_DURATION_STORE * 1000
|
||||||
|
localStorage.setItem('miniBreakDuration', MINI_BREAK_DURATION_STORE)
|
||||||
|
})
|
||||||
|
|
||||||
|
$effect(() => {
|
||||||
|
LONG_BREAK_DURATION = LONG_BREAK_DURATION_STORE * 60 * 1000
|
||||||
|
localStorage.setItem('longBreakDuration', LONG_BREAK_DURATION_STORE)
|
||||||
|
})
|
||||||
|
|
||||||
|
$effect(() => {
|
||||||
|
localStorage.setItem('soundEnabled', SOUND_ENABLED)
|
||||||
|
})
|
||||||
|
|
||||||
|
$effect(() => {
|
||||||
|
localStorage.setItem('notificationsEnabled', NOTIFICATIONS_ENABLED)
|
||||||
})
|
})
|
||||||
MINI_BREAK_DURATION_STORE.subscribe(
|
|
||||||
(value) => (MINI_BREAK_DURATION = value * 1000)
|
|
||||||
)
|
|
||||||
LONG_BREAK_DURATION_STORE.subscribe(
|
|
||||||
(value) => (LONG_BREAK_DURATION = value * 60 * 1000)
|
|
||||||
)
|
|
||||||
SOUND_ENABLED.subscribe((value) => (SOUND_ENABLED_STATE = value))
|
|
||||||
NOTIFICATIONS_ENABLED.subscribe(
|
|
||||||
(value) => (NOTIFICATIONS_ENABLED_STATE = value)
|
|
||||||
)
|
|
||||||
|
|
||||||
let miniBreakCount = 0
|
let miniBreakCount = 0
|
||||||
|
|
||||||
let audio: HTMLAudioElement | null = null
|
let audio: HTMLAudioElement | null = null
|
||||||
|
|
||||||
const playSound = () => {
|
const playSound = () => {
|
||||||
if (SOUND_ENABLED_STATE && audio) {
|
if (SOUND_ENABLED && audio) {
|
||||||
audio.pause()
|
audio.pause()
|
||||||
audio.src = "/sounds/Bells.mp3"
|
audio.src = "/sounds/Bells.mp3"
|
||||||
audio.volume = 0.4
|
audio.volume = 0.4
|
||||||
|
@ -51,7 +57,7 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
const showNotification = (title: string, options: NotificationOptions) => {
|
const showNotification = (title: string, options: NotificationOptions) => {
|
||||||
if (NOTIFICATIONS_ENABLED_STATE) {
|
if (NOTIFICATIONS_ENABLED) {
|
||||||
if (Notification.permission === "granted") {
|
if (Notification.permission === "granted") {
|
||||||
new Notification(title, options)
|
new Notification(title, options)
|
||||||
} else if (Notification.permission !== "denied") {
|
} else if (Notification.permission !== "denied") {
|
||||||
|
@ -68,39 +74,42 @@
|
||||||
if (timer) clearInterval(timer)
|
if (timer) clearInterval(timer)
|
||||||
|
|
||||||
timer = setInterval(() => {
|
timer = setInterval(() => {
|
||||||
timeLeft.update((currentTimeLeft) => {
|
const newTimeLeft = timeLeft - 1000
|
||||||
const newTimeLeft = currentTimeLeft - 1000
|
if (newTimeLeft <= 0) {
|
||||||
if (newTimeLeft <= 0) {
|
console.log(timerState)
|
||||||
if ($state !== "Ready") {
|
if (timerState !== "Ready") {
|
||||||
state.set("Ready")
|
timerState = "Ready"
|
||||||
miniBreakCount++
|
miniBreakCount++
|
||||||
timeLeftDisplay.set(formatTime(BREAK_INTERVAL))
|
timeLeftDisplay = formatTime(BREAK_INTERVAL)
|
||||||
showNotification("Ready", { body: "Continue working!" })
|
showNotification("Ready", { body: "Continue working!" })
|
||||||
playSound()
|
playSound()
|
||||||
return BREAK_INTERVAL
|
timeLeft = BREAK_INTERVAL
|
||||||
}
|
return
|
||||||
if (miniBreakCount === 3) {
|
|
||||||
state.set("Long Break")
|
|
||||||
miniBreakCount = 0
|
|
||||||
timeLeftDisplay.set(formatTime(LONG_BREAK_DURATION))
|
|
||||||
showNotification("Long Break", {
|
|
||||||
body: `Take a ${LONG_BREAK_DURATION / (60 * 1000)}-second break now!`,
|
|
||||||
})
|
|
||||||
playSound()
|
|
||||||
return LONG_BREAK_DURATION
|
|
||||||
} else {
|
|
||||||
state.set("Mini Break")
|
|
||||||
timeLeftDisplay.set(formatTime(MINI_BREAK_DURATION))
|
|
||||||
showNotification("Mini Break", {
|
|
||||||
body: `Take a ${MINI_BREAK_DURATION / 1000}-second break now!`,
|
|
||||||
})
|
|
||||||
playSound()
|
|
||||||
return MINI_BREAK_DURATION
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
timeLeftDisplay.set(formatTime(newTimeLeft))
|
if (miniBreakCount === 3) {
|
||||||
return newTimeLeft
|
timerState = "Long Break"
|
||||||
})
|
miniBreakCount = 0
|
||||||
|
timeLeftDisplay = formatTime(LONG_BREAK_DURATION)
|
||||||
|
showNotification("Long Break", {
|
||||||
|
body: `Take a ${LONG_BREAK_DURATION / (60 * 1000)}-second break now!`,
|
||||||
|
})
|
||||||
|
playSound()
|
||||||
|
timeLeft = LONG_BREAK_DURATION
|
||||||
|
return
|
||||||
|
} else {
|
||||||
|
timerState = "Mini Break"
|
||||||
|
timeLeftDisplay = formatTime(MINI_BREAK_DURATION)
|
||||||
|
showNotification("Mini Break", {
|
||||||
|
body: `Take a ${MINI_BREAK_DURATION / 1000}-second break now!`,
|
||||||
|
})
|
||||||
|
playSound()
|
||||||
|
timeLeft = MINI_BREAK_DURATION
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
timeLeftDisplay = formatTime(newTimeLeft)
|
||||||
|
timeLeft = newTimeLeft
|
||||||
|
console.log(timeLeft)
|
||||||
}, 1000)
|
}, 1000)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -115,12 +124,12 @@
|
||||||
})
|
})
|
||||||
|
|
||||||
const skipBreak = () => {
|
const skipBreak = () => {
|
||||||
if ($state === "Ready") {
|
if (timerState === "Ready") {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
state.set("Ready")
|
timerState = "Ready"
|
||||||
timeLeft.set(BREAK_INTERVAL)
|
timeLeft = BREAK_INTERVAL
|
||||||
timeLeftDisplay.set(formatTime(BREAK_INTERVAL))
|
timeLeftDisplay = formatTime(BREAK_INTERVAL)
|
||||||
}
|
}
|
||||||
|
|
||||||
function formatTime(milliseconds: number) {
|
function formatTime(milliseconds: number) {
|
||||||
|
@ -133,59 +142,64 @@
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<main>
|
<main>
|
||||||
|
|
||||||
|
{#if !browser}
|
||||||
|
<div class="loading-spinner"/>
|
||||||
|
{:else}
|
||||||
|
|
||||||
|
|
||||||
<div class="container">
|
<div class="container">
|
||||||
<div class="settings">
|
<div class="settings">
|
||||||
<h1>Prevent CVS</h1>
|
|
||||||
|
|
||||||
<div class="setting">
|
<div class="setting">
|
||||||
<label for="break-interval"
|
<label for="break-interval"
|
||||||
>Break interval: {BREAK_INTERVAL / (60 * 1000)} minutes</label
|
>Break interval: {BREAK_INTERVAL_STORE} minutes</label
|
||||||
>
|
>
|
||||||
<input
|
<input
|
||||||
type="range"
|
type="range"
|
||||||
min="1"
|
min="1"
|
||||||
max="60"
|
max="60"
|
||||||
step="1"
|
step="1"
|
||||||
bind:value={$BREAK_INTERVAL_STORE}
|
bind:value={BREAK_INTERVAL_STORE}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="setting">
|
<div class="setting">
|
||||||
<label for="mini-break-duration"
|
<label for="mini-break-duration"
|
||||||
>Mini break duration: {MINI_BREAK_DURATION / 1000} seconds</label
|
>Mini break duration: {MINI_BREAK_DURATION_STORE} seconds</label
|
||||||
>
|
>
|
||||||
<input
|
<input
|
||||||
type="range"
|
type="range"
|
||||||
min="5"
|
min="5"
|
||||||
max="60"
|
max="60"
|
||||||
step="1"
|
step="1"
|
||||||
bind:value={$MINI_BREAK_DURATION_STORE}
|
bind:value={MINI_BREAK_DURATION_STORE}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="setting">
|
<div class="setting">
|
||||||
<label for="long-break-duration"
|
<label for="long-break-duration"
|
||||||
>Long break duration: {LONG_BREAK_DURATION / (60 * 1000)} minutes</label
|
>Long break duration: {LONG_BREAK_DURATION_STORE} minutes</label
|
||||||
>
|
>
|
||||||
<input
|
<input
|
||||||
type="range"
|
type="range"
|
||||||
min="1"
|
min="1"
|
||||||
max="30"
|
max="30"
|
||||||
step="1"
|
step="1"
|
||||||
bind:value={$LONG_BREAK_DURATION_STORE}
|
bind:value={LONG_BREAK_DURATION_STORE}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="setting">
|
<div class="setting">
|
||||||
<label>
|
<label>
|
||||||
<input type="checkbox" bind:checked={$SOUND_ENABLED} />
|
<input type="checkbox" bind:checked={SOUND_ENABLED} />
|
||||||
Sound
|
Sound
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="setting">
|
<div class="setting">
|
||||||
<label>
|
<label>
|
||||||
<input type="checkbox" bind:checked={$NOTIFICATIONS_ENABLED} />
|
<input type="checkbox" bind:checked={NOTIFICATIONS_ENABLED} />
|
||||||
Notifications
|
Notifications
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
|
@ -193,18 +207,22 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="container">
|
<div class="container">
|
||||||
|
<h1>Prevent CVS</h1>
|
||||||
|
|
||||||
<div class="timer">
|
<div class="timer">
|
||||||
{#if $state !== "Ready"}
|
{#if timerState !== "Ready"}
|
||||||
<div class="timer-content">
|
<div class="timer-content">
|
||||||
<p class="state">{$state}</p>
|
<p class="state">{timerState}</p>
|
||||||
<p>Time left until end of break: {$timeLeftDisplay}</p>
|
<p>Time left until end of break: {timeLeftDisplay}</p>
|
||||||
<button on:click={skipBreak}>Skip break</button>
|
<button on:click={skipBreak}>Skip break</button>
|
||||||
</div>
|
</div>
|
||||||
{:else}
|
{:else}
|
||||||
<p>Time left until break: {$timeLeftDisplay}</p>
|
<p>Time left until break: {timeLeftDisplay}</p>
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{/if}
|
||||||
</main>
|
</main>
|
||||||
|
|
||||||
<style lang="postcss">
|
<style lang="postcss">
|
||||||
|
@ -255,4 +273,26 @@
|
||||||
.state {
|
.state {
|
||||||
@apply text-2xl font-semibold text-red-500;
|
@apply text-2xl font-semibold text-red-500;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loading-spinner {
|
||||||
|
border: 4px solid rgba(0, 0, 0, 0.1);
|
||||||
|
border-left-color: #3498db;
|
||||||
|
border-radius: 50%;
|
||||||
|
width: 100px;
|
||||||
|
height: 100px;
|
||||||
|
animation: spin 1s linear infinite;
|
||||||
|
@apply max-w-xl;
|
||||||
|
@apply mx-auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes spin {
|
||||||
|
0% {
|
||||||
|
transform: rotate(0deg);
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
transform: rotate(360deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
</style>
|
</style>
|
||||||
|
|
Loading…
Reference in New Issue