Save timer variables on database

This commit is contained in:
Santiago Lo Coco 2024-05-18 16:32:22 +02:00
parent 204f2d89d7
commit c51ad6a41e
6 changed files with 230 additions and 88 deletions

View File

@ -1,22 +1,38 @@
generator client { generator client {
provider = "prisma-client-js" provider = "prisma-client-js"
binaryTargets = ["native", "rhel-openssl-1.0.x"] binaryTargets = ["native", "rhel-openssl-1.0.x"]
} }
datasource db { datasource db {
provider = "postgresql" provider = "postgresql"
url = env("DATABASE_URL") url = env("DATABASE_URL")
} }
model User { model User {
id Int @id @default(autoincrement()) id Int @id @default(autoincrement())
username String @unique username String @unique
passwordHash String passwordHash String
userAuthToken String @unique userAuthToken String @unique
createdAt DateTime @default(now()) createdAt DateTime @default(now())
updatedAt DateTime @updatedAt updatedAt DateTime @updatedAt
Timer Timer?
@@map("user")
} @@map("user")
}
model Timer {
id Int @id @default(autoincrement())
user_id Int @unique
breakInterval Int @default(20)
miniBreakDuration Int @default(20)
longBreakDuration Int @default(5)
soundEnabled Boolean @default(true)
notificationsEnabled Boolean @default(true)
user User @relation(references: [id], fields: [user_id], onDelete: Cascade)
@@index([user_id])
@@map("Timer")
}

View File

@ -10,12 +10,13 @@ export const handle: Handle = async ({ event, resolve }) => {
const user = await db.user.findUnique({ const user = await db.user.findUnique({
where: { userAuthToken: session }, where: { userAuthToken: session },
select: { username: true }, select: { username: true, id: true },
}) })
if (user) { if (user) {
event.locals.user = { event.locals.user = {
name: user.username, name: user.username,
id: user.id,
} }
} }

View File

@ -1,5 +1,6 @@
export type User = { export type User = {
username: string username: string
id: number
} }
export const themes = ["light", "dark"] as const export const themes = ["light", "dark"] as const

13
src/lib/server/timer.ts Normal file
View File

@ -0,0 +1,13 @@
export const updateTimer = async ({ user, breakInterval, miniBreakDuration, longBreakDuration, soundEnabled, notificationsEnabled }) => {
await db.timer.update({
where: { user_id: user.id },
data: {
breakInterval,
miniBreakDuration,
longBreakDuration,
soundEnabled,
notificationsEnabled,
},
})
}

View File

@ -0,0 +1,52 @@
import { redirect, type ServerLoadEvent } from "@sveltejs/kit"
import type { PageServerLoad } from "./$types"
import { db } from "$lib/server/database"
import type { User } from "$lib/index"
export const load: PageServerLoad = async (event) => {
if (!event.locals.user) {
return
}
let timer = await fetchTimer(event.locals.user)
return { timer }
}
async function fetchTimer(user: User) {
let timer = await db.timer.findUnique({ where: { user_id: user.id } })
if (!timer) {
await db.timer.create({
data: {
user_id: user.id,
},
})
}
return await db.timer.findUnique({ where: { user_id: user.id } })
}
export const actions = {
updateTimer: async ({ request, locals }) => {
const user = locals.user
const data = await request.formData()
console.log(data)
const breakInterval = parseInt(data.get("break-interval"))
const miniBreakDuration = parseInt(data.get("mini-break-duration"))
const longBreakDuration = parseInt(data.get("long-break-duration"))
const soundEnabled = data.get("sound-enabled") === "true"
const notificationsEnabled = data.get("notifications-enabled") === "true"
await db.timer.update({
where: { user_id: user.id },
data: {
breakInterval,
miniBreakDuration,
longBreakDuration,
soundEnabled,
notificationsEnabled,
},
})
}
}

View File

@ -1,24 +1,41 @@
<script lang="ts"> <script lang="ts">
import { onDestroy, onMount } from "svelte" import { onDestroy, onMount } from "svelte"
import { browser } from "$app/environment" import { browser } from "$app/environment"
import { enhance } from "$app/forms"
let BREAK_INTERVAL_STORE = $state( let { data } = $props()
browser ? localStorage.getItem("breakInterval") || 20 : 20
) let BREAK_INTERVAL_STORE: any = $state(20)
let MINI_BREAK_DURATION_STORE = $state( let MINI_BREAK_DURATION_STORE: any = $state(20)
browser ? localStorage.getItem("miniBreakDuration") || 20 : 20 let LONG_BREAK_DURATION_STORE: any = $state(5)
) let SOUND_ENABLED: any = $state(true)
let LONG_BREAK_DURATION_STORE = $state( let NOTIFICATIONS_ENABLED: any = $state(true)
browser ? localStorage.getItem("longBreakDuration") || 5 : 5
) $effect(() => {
let SOUND_ENABLED = $state( if (!data.user || !data.timer) {
browser ? (localStorage.getItem("soundEnabled") || "true") === "true" : true BREAK_INTERVAL_STORE = browser
) ? localStorage.getItem("breakInterval") || 20
let NOTIFICATIONS_ENABLED = $state( : 20
browser MINI_BREAK_DURATION_STORE = browser
? (localStorage.getItem("notificationsEnabled") || "true") === "true" ? localStorage.getItem("miniBreakDuration") || 20
: true : 20
) LONG_BREAK_DURATION_STORE = browser
? localStorage.getItem("longBreakDuration") || 5
: 5
SOUND_ENABLED = browser
? (localStorage.getItem("soundEnabled") || "true") === "true"
: true
NOTIFICATIONS_ENABLED = browser
? (localStorage.getItem("notificationsEnabled") || "true") === "true"
: true
} else {
BREAK_INTERVAL_STORE = data.timer.breakInterval
MINI_BREAK_DURATION_STORE = data.timer.miniBreakDuration
LONG_BREAK_DURATION_STORE = data.timer.longBreakDuration
SOUND_ENABLED = data.timer.soundEnabled
NOTIFICATIONS_ENABLED = data.timer.notificationsEnabled
}
})
let BREAK_INTERVAL: number let BREAK_INTERVAL: number
let MINI_BREAK_DURATION: number let MINI_BREAK_DURATION: number
@ -30,29 +47,53 @@
let timeLeftDisplay = $state("") let timeLeftDisplay = $state("")
let timeLeft = $state(0) let timeLeft = $state(0)
const updateTimer = () => {
if (!data.user) {
return
}
const formData = new FormData()
formData.append("user", data.user)
formData.append("break-interval", BREAK_INTERVAL_STORE)
formData.append("mini-break-duration", MINI_BREAK_DURATION_STORE)
formData.append("long-break-duration", LONG_BREAK_DURATION_STORE)
formData.append("sound-enabled", SOUND_ENABLED)
formData.append("notifications-enabled", NOTIFICATIONS_ENABLED)
fetch("?/updateTimer", {
method: "POST",
body: formData,
})
}
$effect(() => { $effect(() => {
BREAK_INTERVAL = BREAK_INTERVAL_STORE * 60 * 1000 BREAK_INTERVAL = BREAK_INTERVAL_STORE * 60 * 1000
timeLeft = BREAK_INTERVAL timeLeft = BREAK_INTERVAL
timeLeftDisplay = formatTime(BREAK_INTERVAL) timeLeftDisplay = formatTime(BREAK_INTERVAL)
localStorage.setItem("breakInterval", BREAK_INTERVAL_STORE) localStorage.setItem("breakInterval", BREAK_INTERVAL_STORE)
updateTimer()
}) })
$effect(() => { $effect(() => {
console.log(MINI_BREAK_DURATION_STORE)
MINI_BREAK_DURATION = MINI_BREAK_DURATION_STORE * 1000 MINI_BREAK_DURATION = MINI_BREAK_DURATION_STORE * 1000
localStorage.setItem("miniBreakDuration", MINI_BREAK_DURATION_STORE) localStorage.setItem("miniBreakDuration", MINI_BREAK_DURATION_STORE)
updateTimer()
}) })
$effect(() => { $effect(() => {
LONG_BREAK_DURATION = LONG_BREAK_DURATION_STORE * 60 * 1000 LONG_BREAK_DURATION = LONG_BREAK_DURATION_STORE * 60 * 1000
localStorage.setItem("longBreakDuration", LONG_BREAK_DURATION_STORE) localStorage.setItem("longBreakDuration", LONG_BREAK_DURATION_STORE)
updateTimer()
}) })
$effect(() => { $effect(() => {
localStorage.setItem("soundEnabled", SOUND_ENABLED) localStorage.setItem("soundEnabled", SOUND_ENABLED)
updateTimer()
}) })
$effect(() => { $effect(() => {
localStorage.setItem("notificationsEnabled", NOTIFICATIONS_ENABLED) localStorage.setItem("notificationsEnabled", NOTIFICATIONS_ENABLED)
updateTimer()
}) })
let miniBreakCount = 0 let miniBreakCount = 0
@ -159,58 +200,76 @@
{:else} {:else}
<div class="container"> <div class="container">
<div class="settings"> <div class="settings">
<div class="setting"> <form method="POST" action="?/updateTimer" use:enhance>
<label for="break-interval" <div class="setting">
>Break interval: {BREAK_INTERVAL_STORE} minutes</label <label for="break-interval"
> >Break interval: {BREAK_INTERVAL_STORE} minutes</label
<input >
type="range" <input
min="1" type="range"
max="60" id="break-interval"
step="1" name="break-interval"
bind:value={BREAK_INTERVAL_STORE} min="1"
/> max="60"
</div> step="1"
bind:value={BREAK_INTERVAL_STORE}
/>
</div>
<div class="setting"> <div class="setting">
<label for="mini-break-duration" <label for="mini-break-duration"
>Mini break duration: {MINI_BREAK_DURATION_STORE} seconds</label >Mini break duration: {MINI_BREAK_DURATION_STORE} seconds</label
> >
<input <input
type="range" id="mini-break-duration"
min="5" name="mini-break-duration"
max="60" type="range"
step="1" min="5"
bind:value={MINI_BREAK_DURATION_STORE} max="60"
/> step="1"
</div> bind:value={MINI_BREAK_DURATION_STORE}
/>
</div>
<div class="setting"> <div class="setting">
<label for="long-break-duration" <label for="long-break-duration"
>Long break duration: {LONG_BREAK_DURATION_STORE} minutes</label >Long break duration: {LONG_BREAK_DURATION_STORE} minutes</label
> >
<input <input
type="range" id="long-break-duration"
min="1" name="long-break-duration"
max="30" type="range"
step="1" min="1"
bind:value={LONG_BREAK_DURATION_STORE} max="30"
/> step="1"
</div> bind:value={LONG_BREAK_DURATION_STORE}
/>
</div>
<div class="setting"> <div class="setting">
<label> <label>
<input type="checkbox" bind:checked={SOUND_ENABLED} /> <input
Sound id="sound-enabled"
</label> name="sound-enabled"
</div> type="checkbox"
bind:checked={SOUND_ENABLED}
/>
Sound
</label>
</div>
<div class="setting"> <div class="setting">
<label> <label>
<input type="checkbox" bind:checked={NOTIFICATIONS_ENABLED} /> <input
Notifications id="notifications-enabled"
</label> name="notifications-enabled"
</div> type="checkbox"
bind:checked={NOTIFICATIONS_ENABLED}
/>
Notifications
</label>
</div>
</form>
</div> </div>
</div> </div>