Files
solo-company-feed/components/CreatePostForm.tsx

176 lines
5.8 KiB
TypeScript
Raw Permalink Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
"use client";
import { FormEvent, useState } from "react";
import { useRouter } from "next/navigation";
import { MarkdownPreview } from "@/components/MarkdownPreview";
import { normalizeImageUrl } from "@/lib/normalize";
const defaultIntro = `## 今日进展
- 今日目标:
- 产品 / 交付:
- 客户 / 收入:
- 增长 / 运营:
- 学习 / 复盘:`;
export function CreatePostForm({
availableTags = [],
publishLimit = 10,
todayCount = 0
}: {
availableTags?: string[];
publishLimit?: number;
todayCount?: number;
}) {
const [title, setTitle] = useState("");
const [cover, setCover] = useState("");
const [tags, setTags] = useState("");
const [markdown, setMarkdown] = useState(defaultIntro);
const [loading, setLoading] = useState(false);
const [preview, setPreview] = useState(false);
const router = useRouter();
const addTag = (tag: string) => {
const current = new Set(
tags
.split(",")
.map((item) => item.trim())
.filter(Boolean)
);
current.add(tag);
setTags(Array.from(current).join(", "));
};
async function handleSubmit(e: FormEvent) {
e.preventDefault();
setLoading(true);
try {
const normalizedCover = normalizeImageUrl(cover);
const res = await fetch("/api/posts", {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({
title,
markdown,
cover: normalizedCover,
tags: Array.from(
new Set(
tags
.split(",")
.map((item) => item.trim())
.filter(Boolean)
)
)
})
});
if (!res.ok) {
const data = await res.json().catch(() => ({}));
alert(data.error ? JSON.stringify(data.error) : "发布失败");
return;
}
const data = await res.json();
router.push(`/p/${data.slug}`);
router.refresh();
} finally {
setLoading(false);
}
}
return (
<form onSubmit={handleSubmit} className="space-y-4 rounded-2xl bg-white/80 p-5 shadow-sm ring-1 ring-slate-100">
<div className="flex flex-wrap items-center justify-between gap-3">
<div>
<h3 className="text-lg font-semibold"></h3>
<p className="text-sm text-slate-500"></p>
</div>
<span className="rounded-full bg-slate-100 px-3 py-1 text-xs text-slate-600 ring-1 ring-slate-200">
{todayCount} / {publishLimit}
</span>
</div>
<label className="block space-y-1">
<span className="text-sm font-medium text-slate-700"></span>
<input
required
value={title}
onChange={(e) => setTitle(e.target.value)}
className="w-full rounded-xl border border-slate-200 bg-white px-3 py-2 text-sm shadow-inner focus:border-brand-500 focus:outline-none"
placeholder="例如:本周交付进度"
/>
</label>
<label className="block space-y-1">
<span className="text-sm font-medium text-slate-700"></span>
<input
value={cover}
onChange={(e) => setCover(e.target.value)}
className="w-full rounded-xl border border-slate-200 bg-white px-3 py-2 text-sm shadow-inner focus:border-brand-500 focus:outline-none"
placeholder="https://img.020417.xyz/xxxx.png"
/>
</label>
<label className="block space-y-1">
<span className="text-sm font-medium text-slate-700"></span>
<input
value={tags}
onChange={(e) => setTags(e.target.value)}
className="w-full rounded-xl border border-slate-200 bg-white px-3 py-2 text-sm shadow-inner focus:border-brand-500 focus:outline-none"
placeholder="产品, 交付, 复盘"
/>
{availableTags.length > 0 ? (
<div className="mt-2 flex flex-wrap gap-2">
{availableTags.map((tag) => (
<button
key={tag}
type="button"
onClick={() => addTag(tag)}
className="rounded-full bg-slate-100 px-2 py-1 text-xs text-slate-600 ring-1 ring-slate-200 hover:text-brand-600"
>
#{tag}
</button>
))}
</div>
) : null}
</label>
<div className="space-y-1">
<div className="flex flex-wrap items-center justify-between gap-2">
<span className="text-sm font-medium text-slate-700"></span>
<div className="flex items-center gap-2">
<button
type="button"
onClick={() => setPreview((prev) => !prev)}
className="rounded-full bg-slate-100 px-4 py-2 text-sm font-medium text-slate-700 ring-1 ring-slate-200 hover:text-brand-600"
>
{preview ? "继续编辑" : "预览"}
</button>
<button
type="submit"
disabled={loading}
className="rounded-full bg-brand-600 px-4 py-2 text-sm font-medium text-white shadow hover:bg-brand-700 disabled:cursor-not-allowed disabled:opacity-60"
>
{loading ? "发布中..." : "发布"}
</button>
</div>
</div>
{!preview ? (
<textarea
required
value={markdown}
onChange={(e) => setMarkdown(e.target.value)}
rows={12}
className="w-full rounded-xl border border-slate-200 bg-white px-3 py-2 text-sm shadow-inner focus:border-brand-500 focus:outline-none"
/>
) : (
<div className="rounded-xl border border-slate-200 bg-white p-4">
<MarkdownPreview markdown={markdown || "(空内容)"} />
</div>
)}
</div>
</form>
);
}