add game
This commit is contained in:
parent
5acaa7aeec
commit
005ddc36cf
@ -0,0 +1,4 @@
|
||||
<Button asChild>
|
||||
<Link href="/post/game">Game</Link>
|
||||
</Button>
|
||||
|
@ -0,0 +1,26 @@
|
||||
"use server";
|
||||
|
||||
import { DID } from "@/lib/data/atproto/did";
|
||||
import { getVerifiedHandle } from "@/lib/data/atproto/identity";
|
||||
import { putPost } from "@/lib/data/atproto/game";
|
||||
import { uncached_doesPostExist } from "@/lib/data/db/post";
|
||||
import { DataLayerError } from "@/lib/data/error";
|
||||
import { ensureUser } from "@/lib/data/user";
|
||||
import { redirect } from "next/navigation";
|
||||
|
||||
export async function newPostAction(_prevState: unknown) {
|
||||
"use server";
|
||||
const user = await ensureUser();
|
||||
const [handle] = await Promise.all([
|
||||
getVerifiedHandle(user.did),
|
||||
]);
|
||||
|
||||
const account = `at://did:plc:4hqjfn7m6n5hno3doamuhgef/ai.syui.game.user/${handle}`;
|
||||
try {
|
||||
const { rkey } = await putPost({ account });
|
||||
redirect(`https://at.syu.is/at/${user.did}/ai.syui.game/self`);
|
||||
} catch (error) {
|
||||
if (!(error instanceof DataLayerError)) throw error;
|
||||
return { error: "Failed to create post" };
|
||||
}
|
||||
}
|
@ -0,0 +1,36 @@
|
||||
"use client";
|
||||
|
||||
import { startTransition, useActionState, useId, useState } from "react";
|
||||
import { newPostAction } from "./_action";
|
||||
import { Label } from "@/lib/components/ui/label";
|
||||
import { Input } from "@/lib/components/ui/input";
|
||||
import { Button } from "@/lib/components/ui/button";
|
||||
import { Alert, AlertDescription, AlertTitle } from "@/lib/components/ui/alert";
|
||||
import { Spinner } from "@/lib/components/ui/spinner";
|
||||
import { InputLengthIndicator } from "@/lib/components/input-length-indicator";
|
||||
|
||||
export function NewPostButton() {
|
||||
const [state, action, isPending] = useActionState(newPostAction, null);
|
||||
const [error, setError] = useState<string | null>(null);
|
||||
|
||||
const handleClick = async () => {
|
||||
try {
|
||||
const result = await action({});
|
||||
if (result && 'error' in result) {
|
||||
setError(result.error);
|
||||
}
|
||||
} catch (e) {
|
||||
console.error('Error creating post:', e);
|
||||
setError('予期しないエラーが発生しました');
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<div>
|
||||
<button onClick={handleClick} disabled={isPending}>
|
||||
{isPending ? '投稿中...' : '新規投稿'}
|
||||
</button>
|
||||
{error && <p style={{ color: 'red' }}>{error}</p>}
|
||||
</div>
|
||||
);
|
||||
}
|
@ -0,0 +1,17 @@
|
||||
import { Metadata } from "next";
|
||||
import { NewPostButton } from "./_client";
|
||||
|
||||
export const metadata: Metadata = {
|
||||
title: "New post | Frontpage",
|
||||
robots: "noindex, nofollow",
|
||||
};
|
||||
|
||||
export default function NewPost() {
|
||||
return (
|
||||
<main className="flex flex-col gap-3">
|
||||
<h2 className="text-3xl font-bold tracking-tight text-gray-900 dark:text-gray-100">
|
||||
</h2>
|
||||
<NewPostButton />
|
||||
</main>
|
||||
);
|
||||
}
|
@ -0,0 +1,62 @@
|
||||
import "server-only";
|
||||
import {
|
||||
atprotoPutRecord,
|
||||
atprotoCreateRecord,
|
||||
atprotoDeleteRecord,
|
||||
atprotoGetRecord,
|
||||
} from "./record";
|
||||
import { z } from "zod";
|
||||
import { DataLayerError } from "../error";
|
||||
import { DID, getPdsUrl } from "./did";
|
||||
|
||||
export const PostCollection = "ai.syui.game";
|
||||
|
||||
export const PostRecord = z.object({
|
||||
account: z.string(),
|
||||
createdAt: z.string(),
|
||||
});
|
||||
|
||||
export type Post = z.infer<typeof PostRecord>;
|
||||
|
||||
type PostInput = {
|
||||
account: string;
|
||||
};
|
||||
|
||||
export async function putPost({ account }: PostInput) {
|
||||
const record = { account, createdAt: new Date().toISOString() };
|
||||
PostRecord.parse(record);
|
||||
|
||||
const result = await atprotoPutRecord({
|
||||
rkey: "self",
|
||||
record,
|
||||
collection: PostCollection,
|
||||
});
|
||||
|
||||
return {
|
||||
rkey: "self",
|
||||
};
|
||||
}
|
||||
|
||||
export async function deletePost(rkey: string) {
|
||||
await atprotoDeleteRecord({
|
||||
rkey,
|
||||
collection: PostCollection,
|
||||
});
|
||||
}
|
||||
|
||||
export async function getPost({ rkey, repo }: { rkey: string; repo: DID }) {
|
||||
const service = await getPdsUrl(repo);
|
||||
|
||||
if (!service) {
|
||||
throw new DataLayerError("Failed to get service url");
|
||||
}
|
||||
|
||||
const { value } = await atprotoGetRecord({
|
||||
serviceEndpoint: service,
|
||||
repo,
|
||||
collection: PostCollection,
|
||||
rkey,
|
||||
});
|
||||
|
||||
return PostRecord.parse(value);
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user