ในโพสต์นี้ เราจะเขียน TODO app แบบง่ายๆ โดยใช้ SvelteKit และ Serverless Redis (Upstash)
SvelteKit เป็นเฟรมเวิร์กสำหรับสร้างเว็บแอปพลิเคชันทุกขนาด พร้อมประสบการณ์การพัฒนาที่สวยงามและการกำหนดเส้นทางตามระบบไฟล์ที่ยืดหยุ่น
สร้างโครงการ
เรียกใช้คำสั่งด้านล่าง:
npm init svelte@next todo-app-with-redis
โครงการพร้อมแล้ว ตอนนี้เรามาติดตั้งการพึ่งพาและเรียกใช้:
npm install
npm run dev
อินเทอร์เฟซผู้ใช้
เราสร้างรูปแบบที่เรียบง่ายและรายการเพื่อเก็บรายการสิ่งที่ต้องทำ ตอนนี้จะว่างเปล่า ต่อไปเราจะผูกกับฐานข้อมูล
// src/routes/index.svelte
<script lang="ts">
import type {Todo} from "../lib/types";
import TodoItem from "../components/Todo.svelte";
export let todos: Todo[] = [
{
id: '1',
text: 'Create database',
status: true
},
{
id: '2',
text: 'Copy database url',
status: false
}
];
const uncheckedTodos = todos.filter((todo) => !todo.status);
const checkedTodos = todos.filter((todo) => todo.status);
</script>
<main class="container">
<form class="new" action="/" method="post">
<input
type="text"
name="text"
autofocus
aria-label="Add todo"
class="input"
placeholder="What needs to be done?"
/>
</form>
{#if uncheckedTodos.length}
<div class="todos">
{#each uncheckedTodos as todo (todo.id)}
<TodoItem {todo}/>
{/each}
</div>
{/if}
{#if checkedTodos.length}
<div class="todos todos-done">
{#each checkedTodos as todo (todo.id)}
<TodoItem {todo}/>
{/each}
</div>
{/if}
</main>
นี่คือองค์ประกอบ TODO ของเรา:
// src/components/Todo.svelte
<script lang="ts">
import type {Todo} from "../lib/types";
export let todo: Todo;
</script>
<div class="todo">
<form action="/?_method=PATCH" method="post">
<input type="hidden" name="todo" value={JSON.stringify(todo)}/>
<button
class="checkbox"
aria-label="Mark todo as {todo.status ? 'not done' : 'done'}"
>
{todo.status ? "✓" : ""}
</button>
</form>
<span class="text">{todo.text}</span>
<form action="/?_method=DELETE" method="post">
<input type="hidden" name="id" value={todo.id}/>
<button class="delete" aria-label="Delete todo">✕</button>
</form>
</div>
ตอนนี้คุณควรเห็น:
เตรียมฐานข้อมูล
เราจะเก็บข้อมูลของเราไว้ใน Upstash Redis ดังนั้นให้สร้างฐานข้อมูล Upstash เราจะใช้ไคลเอนต์ HTTPbased Upstash ซึ่งเป็นมิตรกับสภาพแวดล้อมแบบ Serverless มาติดตั้งกันเถอะ:
npm install @upstash/redis
ตอนนี้ มาสร้างฐานข้อมูล Redis ของเราบน Upstash Console คัดลอก/วาง UPSTASH_REDIS_REST_URL
และ UPSTASH_REDIS_REST_TOKEN
ไปที่ไฟล์ .env
UPSTASH_REDIS_REST_URL=https://global-renewing-gecko-31543.upstash.io
UPSTASH_REDIS_REST_TOKEN=AXs3ACsjfg684jJBFQgN34je7RFJ58wYjg4NjMt=
เราต้องติดตั้ง dotenv เพื่อใช้ตัวแปรสภาพแวดล้อม
npm install dotenv
ตอนนี้สร้าง lib/redis.ts
และสร้างอินสแตนซ์ Redis ใหม่ดังนี้:
// src/lib/redis.ts
import "dotenv/config";
import { Redis } from "@upstash/redis";
export const databaseName =
process.env.NODE_ENV === "development"
? "redis-with-svelte-kit-dev"
: "redis-with-svelte-kit";
const redis = new Redis({
url: process.env.UPSTASH_REDIS_REST_URL,
token: process.env.UPSTASH_REDIS_REST_TOKEN,
});
export default redis;
สร้าง API
ในโฟลเดอร์ js/ts
ไฟล์ถูกใช้เป็นแบ็กเอนด์ API ดังนั้น index.svelte
-> ส่วนหน้า index.ts
-> แบ็กเอนด์
มาสร้าง index.ts
. กันเถอะ และเพิ่มเมธอด API ที่จำเป็น
// src/routes/index.ts
import redis, { databaseName } from "../lib/redis";
import type { RequestHandler } from "@sveltejs/kit";
const redirect = {
status: 303,
headers: {
location: "/",
},
};
export const get: RequestHandler = async () => {
// runs every time the page is loaded
return { body: { todos: [] } };
};
export const post: RequestHandler = async ({ request }) => {
// form: create todo
return redirect;
};
export const patch: RequestHandler = async ({ request }) => {
// form: update todo
return redirect;
};
export const del: RequestHandler = async ({ request }) => {
// form: delete todo
return redirect;
};
POST
สร้างสิ่งที่ต้องทำ
มาดำเนินการสร้างรายการสิ่งที่ต้องทำ:
// src/routes/index.ts
export const post: RequestHandler = async ({ request }) => {
const form = await request.formData();
const text = form.get("text");
const id = Date.now().toString();
const todo = JSON.stringify({ text, status: false });
await redis.hset(databaseName, id, todo);
return redirect;
};
GET
โหลดสิ่งที่ต้องทำ
ตอนนี้ มาแสดงรายการสิ่งที่ต้องทำ:
// src/routes/index.ts
export const get: RequestHandler = async () => {
let todos = [];
const data = await redis.hgetall(databaseName);
if (!data) return { body: { todos } };
// normalize data
todos = Object.keys(data)
.map((key) => ({
id: key,
text: data[key]["text"],
status: data[key]["status"],
}))
// id = timestamp
.sort((a, b) => parseInt(b.id) - parseInt(a.id));
return { body: { todos } };
};
เพื่อให้โพสต์ง่ายขึ้น ฉันไม่ได้เพิ่ม
task deletion
และtaks completion
รหัสที่นี่ แต่คุณสามารถดูได้ใน github repo
ตอนนี้ทุกอย่างพร้อมแล้ว!
รหัสที่มา| แอปสาธิต
ฉันกำลังวางแผนที่จะใช้แอปพลิเคชัน TODO เดียวกันกับ Next.js จากนั้นฉันจะเปรียบเทียบประสบการณ์ของฉันในกรอบเหล่านี้
คอยติดตามและติดตามเราได้ที่ Twitterand Discord