Save timer variables on database
This commit is contained in:
parent
204f2d89d7
commit
c51ad6a41e
|
@ -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")
|
||||||
|
}
|
||||||
|
|
|
@ -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,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
|
@ -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,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
|
@ -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>
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue