优秀了项目结构
This commit is contained in:
parent
e2e2be9010
commit
fb65612978
|
@ -2,7 +2,7 @@ import SiderNav from "../_ui/SiderNav";
|
||||||
import Search from "../_ui/Search";
|
import Search from "../_ui/Search";
|
||||||
import LinkListBox from "../_ui/LinkListBox";
|
import LinkListBox from "../_ui/LinkListBox";
|
||||||
import { getLinkTypeList } from "../_lib/data/linkType";
|
import { getLinkTypeList } from "../_lib/data/linkType";
|
||||||
import { getLinkList, getLinkListAll } from "../_lib/data/link";
|
import { getLinkList } from "../_lib/data/link";
|
||||||
import Footer from "../_ui/footer";
|
import Footer from "../_ui/footer";
|
||||||
|
|
||||||
export default async function Page() {
|
export default async function Page() {
|
||||||
|
|
|
@ -14,6 +14,7 @@ export type Link = {
|
||||||
logoLink: string;
|
logoLink: string;
|
||||||
isHot?: boolean;
|
isHot?: boolean;
|
||||||
addTime: number;
|
addTime: number;
|
||||||
|
subLinkType?: string[];
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -11,7 +11,7 @@ export type LinkType = {
|
||||||
_id: string;
|
_id: string;
|
||||||
href?: string;
|
href?: string;
|
||||||
priority?: number;
|
priority?: number;
|
||||||
subLinkType?: string[];
|
subCategory?: string[];
|
||||||
location?: string;
|
location?: string;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,15 +2,16 @@
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { LinkType } from "../_lib/data/linkType";
|
import { LinkType } from "../_lib/data/linkType";
|
||||||
import { Link as _Link } from "../_lib/data/link";
|
import { Link as _Link } from "../_lib/data/link";
|
||||||
import { useEffect, useMemo } from "react";
|
import { useEffect, useMemo, useState } from "react";
|
||||||
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
||||||
import { faArrowRight, faClock, faFire, faRightFromBracket, faRightLong, faTimeline, faTimesCircle, faTimesRectangle, faVolumeTimes } from "@fortawesome/free-solid-svg-icons";
|
import { faArrowRight, faClock, faFire, faRightFromBracket, faRightLong, faTimeline, faTimesCircle, faTimesRectangle, faVolumeTimes } from "@fortawesome/free-solid-svg-icons";
|
||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
import { linkTypeAtom } from "../_lib/atom";
|
import { linkTypeAtom } from "../_lib/atom";
|
||||||
import Image from "next/image";
|
import Image from "next/image";
|
||||||
|
import clsx from "clsx";
|
||||||
const PICTURE_PREFIX = 'https://newuitab.oss-cn-hangzhou.aliyuncs.com/ai_upload/downloads/'
|
const PICTURE_PREFIX = 'https://newuitab.oss-cn-hangzhou.aliyuncs.com/ai_upload/downloads/'
|
||||||
export const LinkBlock = ({ val }: { val: _Link }) => {
|
export const LinkBlock = ({ val }: { val: _Link }) => {
|
||||||
return <Link className="flex gap-x-2 bg-white rounded-lg py-4 pl-2 cursor-pointer duration-150 hover:-translate-y-1 shadow-sm"
|
return <Link className="flex gap-x-2 bg-white rounded-lg py-4 px-2 cursor-pointer duration-150 hover:-translate-y-1 shadow-sm"
|
||||||
href={val.articleId ? `/article/${val.articleId}` : val.link || ''} target="_blank">
|
href={val.articleId ? `/article/${val.articleId}` : val.link || ''} target="_blank">
|
||||||
<img src={val.logoLink.startsWith('http') ? val.logoLink : (PICTURE_PREFIX + val.logoLink)} className="w-[40px] h-[40px]"></img>
|
<img src={val.logoLink.startsWith('http') ? val.logoLink : (PICTURE_PREFIX + val.logoLink)} className="w-[40px] h-[40px]"></img>
|
||||||
<div className="flex-1 w-0 flex flex-col justify-between">
|
<div className="flex-1 w-0 flex flex-col justify-between">
|
||||||
|
@ -19,10 +20,25 @@ export const LinkBlock = ({ val }: { val: _Link }) => {
|
||||||
</div>
|
</div>
|
||||||
</Link>
|
</Link>
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function LinkListBox({ linkTypeList, linkList, showHot, showRecent }: { linkTypeList: LinkType[]; linkList: _Link[]; showHot: boolean; showRecent: boolean }) {
|
export default function LinkListBox({ linkTypeList, linkList, showHot, showRecent }: { linkTypeList: LinkType[]; linkList: _Link[]; showHot: boolean; showRecent: boolean }) {
|
||||||
const hotList = useMemo(() => linkList.filter((val, index) => val.isHot && index < 12), [linkList])
|
const hotList = useMemo(() => linkList.filter((val, index) => val.isHot && index < 12), [linkList])
|
||||||
const recentList = useMemo(() => linkList.map(val => val).sort((a, b) => b.addTime - a.addTime).filter((_, idx) => idx < 12), [linkList])
|
const recentList = useMemo(() => linkList.map(val => val).sort((a, b) => b.addTime - a.addTime).filter((_, idx) => idx < 12), [linkList])
|
||||||
const [currentId] = useAtom(linkTypeAtom)
|
const [currentId] = useAtom(linkTypeAtom)
|
||||||
|
const [selectSubType, setSelectSubType] = useState<Map<string, string>>(new Map())
|
||||||
|
useEffect(() => {
|
||||||
|
if (linkTypeList.length === 0) return
|
||||||
|
for (let i = 0; i < linkTypeList.length; i++) {
|
||||||
|
const item = linkTypeList[i];
|
||||||
|
console.log(item);
|
||||||
|
|
||||||
|
if (item.subCategory?.length) {
|
||||||
|
setSelectSubType(pre => {
|
||||||
|
return new Map(pre.set(item._id, item.subCategory?.length ? item.subCategory[0] : ''));
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, [linkTypeList, setSelectSubType])
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
console.log(currentId);
|
console.log(currentId);
|
||||||
|
|
||||||
|
@ -77,7 +93,6 @@ export default function LinkListBox({ linkTypeList, linkList, showHot, showRecen
|
||||||
<span>{item.label}</span>
|
<span>{item.label}</span>
|
||||||
{
|
{
|
||||||
linkList.filter(val => val.type === item._id).length >= 42 &&
|
linkList.filter(val => val.type === item._id).length >= 42 &&
|
||||||
|
|
||||||
<Link
|
<Link
|
||||||
href={"/detail/" + item._id}
|
href={"/detail/" + item._id}
|
||||||
className="absolute cursor-pointer hover:text-[#333] right-1 text-[#666] flex gap-x-1 items-center text-[14px] ">
|
className="absolute cursor-pointer hover:text-[#333] right-1 text-[#666] flex gap-x-1 items-center text-[14px] ">
|
||||||
|
@ -86,12 +101,40 @@ export default function LinkListBox({ linkTypeList, linkList, showHot, showRecen
|
||||||
</Link>
|
</Link>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
|
{
|
||||||
|
item.subCategory && item.subCategory.length > 0 &&
|
||||||
|
<div className=" flex gap-x-3 text-[#666] py-2 text-[16px]">
|
||||||
|
{
|
||||||
|
item.subCategory.map(subItem => (
|
||||||
|
<div key={subItem}
|
||||||
|
onClick={() => {
|
||||||
|
setSelectSubType(pre => {
|
||||||
|
return new Map(pre.set(item._id, subItem))
|
||||||
|
})
|
||||||
|
}}
|
||||||
|
className={clsx(" px-3 py-1 rounded-lg ",
|
||||||
|
selectSubType.get(item._id) === subItem ?
|
||||||
|
"text-white bg-orange-300"
|
||||||
|
: "bg-[#f5f5f5] shadow hover:bg-orange-300/30 cursor-pointer"
|
||||||
|
)}>
|
||||||
|
{subItem}
|
||||||
|
</div>
|
||||||
|
))
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
<div className=" grid grid-cols-3 lg:grid-cols-6 gap-4 ">
|
<div className=" grid grid-cols-3 lg:grid-cols-6 gap-4 ">
|
||||||
{
|
{
|
||||||
linkList.filter(val => val.type === item._id).filter((_, idx) => idx < 42).map(val => (
|
item.subCategory ?
|
||||||
<LinkBlock val={val} key={val._id} />
|
linkList.filter(val => val.type === item._id && val.subLinkType?.includes(selectSubType.get(item._id) || '-1')).filter((_, idx) => idx < 42).map(val => (
|
||||||
|
<LinkBlock val={val} key={val._id} />
|
||||||
|
|
||||||
))
|
))
|
||||||
|
:
|
||||||
|
linkList.filter(val => val.type === item._id).filter((_, idx) => idx < 42).map(val => (
|
||||||
|
<LinkBlock val={val} key={val._id} />
|
||||||
|
|
||||||
|
))
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -1,8 +1,11 @@
|
||||||
"use client";
|
import { Editor } from "@bytemd/react";
|
||||||
import { MdPreview} from 'md-editor-rt';
|
|
||||||
|
|
||||||
export default function MarkdownView({ value = '' }: { value?: string }) {
|
export default function MyMarkdownEdit({ value = '', onChange }: { value?: string; onChange?: (val: string) => void }) {
|
||||||
return (
|
return (
|
||||||
<MdPreview id='markdown-view' value={value} />
|
<Editor
|
||||||
|
value={value}
|
||||||
|
onChange={onChange}
|
||||||
|
// plugins={plugins}
|
||||||
|
/>
|
||||||
)
|
)
|
||||||
}
|
}
|
|
@ -1,5 +1,5 @@
|
||||||
export default function Footer() {
|
export default function Footer() {
|
||||||
return <div className="h-[300px] w-full bg-red-200">
|
return <div className="h-[100px] w-full ">
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
|
@ -8,16 +8,13 @@ import { Button, Card, Col, DatePicker, Form, Input, InputNumber, message, Row,
|
||||||
import { useRouter } from 'next/navigation'
|
import { useRouter } from 'next/navigation'
|
||||||
import 'md-editor-rt/lib/style.css';
|
import 'md-editor-rt/lib/style.css';
|
||||||
import '@ant-design/v5-patch-for-react-19';
|
import '@ant-design/v5-patch-for-react-19';
|
||||||
import { mRequest } from '@/app/_lib/request';
|
|
||||||
import dayjs from 'dayjs';
|
import dayjs from 'dayjs';
|
||||||
import { Editor } from "@bytemd/react";
|
|
||||||
import gfm from '@bytemd/plugin-gfm'
|
import gfm from '@bytemd/plugin-gfm'
|
||||||
import { useEffect, useMemo, useRef, useState } from 'react';
|
import { useEffect, useMemo, useRef, useState } from 'react';
|
||||||
import { valueType } from 'antd/es/statistic/utils';
|
|
||||||
import { getLinkList, updateLink } from '@/app/_lib/data/link';
|
import { getLinkList, updateLink } from '@/app/_lib/data/link';
|
||||||
import { debounce } from '@/app/_lib/utils';
|
import { debounce } from '@/app/_lib/utils';
|
||||||
import { useRequest } from 'ahooks';
|
import { useRequest } from 'ahooks';
|
||||||
import { updateUser } from '@/app/_lib/data/user';
|
import MyMarkdownEdit from '@/app/_ui/MarkdownView';
|
||||||
const plugins = [
|
const plugins = [
|
||||||
gfm(),
|
gfm(),
|
||||||
// Add more plugins here
|
// Add more plugins here
|
||||||
|
@ -27,12 +24,13 @@ const plugins = [
|
||||||
export default function AddOrEdit({ editData }: { editData?: ArticleType | null }) {
|
export default function AddOrEdit({ editData }: { editData?: ArticleType | null }) {
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
const fetchRef = useRef(0)
|
const fetchRef = useRef(0)
|
||||||
const formRef = useRef(null)
|
const [form] = Form.useForm();
|
||||||
const [fetching, setFetching] = useState(false);
|
const [fetching, setFetching] = useState(false);
|
||||||
const [options, setOptions] = useState<any[]>([]);
|
const [options, setOptions] = useState<any[]>([]);
|
||||||
const { data: linkList = [] } = useRequest(() => getLinkList({ filter: { articleId: editData?._id } }).then(res => res.list))
|
const { data: linkList = [] } = useRequest(() => getLinkList({ filter: { articleId: editData?._id || '-1' } }).then(res => res.list))
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (linkList.length === 0) {
|
if (linkList.length !== 0) {
|
||||||
|
form.setFieldValue('linkId', linkList.map(item => ({ key: item._id, value: item._id, label: item.name })))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -78,7 +76,7 @@ export default function AddOrEdit({ editData }: { editData?: ArticleType | null
|
||||||
|
|
||||||
<Card title="新增文章" className='mt-2'>
|
<Card title="新增文章" className='mt-2'>
|
||||||
<Form<ArticleType>
|
<Form<ArticleType>
|
||||||
ref={formRef}
|
form={form}
|
||||||
onFinish={async (res: any) => {
|
onFinish={async (res: any) => {
|
||||||
const linkIdList = res.linkId
|
const linkIdList = res.linkId
|
||||||
const addTime = res.addTime.unix()
|
const addTime = res.addTime.unix()
|
||||||
|
@ -100,8 +98,13 @@ export default function AddOrEdit({ editData }: { editData?: ArticleType | null
|
||||||
addTime: dayjs().unix()
|
addTime: dayjs().unix()
|
||||||
|
|
||||||
}).then(id => {
|
}).then(id => {
|
||||||
console.log(id);
|
linkIdList.map((item: any) => {
|
||||||
|
updateLink(
|
||||||
|
item.value,
|
||||||
|
{
|
||||||
|
articleId: id
|
||||||
|
})
|
||||||
|
})
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -206,10 +209,7 @@ export default function AddOrEdit({ editData }: { editData?: ArticleType | null
|
||||||
message: '请输入内容'
|
message: '请输入内容'
|
||||||
}
|
}
|
||||||
]}>
|
]}>
|
||||||
{/* <MDEditor className='my-markdown ' /> */}
|
<MyMarkdownEdit />
|
||||||
<Editor
|
|
||||||
plugins={plugins} value={''}
|
|
||||||
/>
|
|
||||||
</Form.Item>
|
</Form.Item>
|
||||||
|
|
||||||
<Form.Item className="flex justify-end">
|
<Form.Item className="flex justify-end">
|
||||||
|
|
|
@ -28,6 +28,12 @@ export default function Layout({
|
||||||
href: '/admin/dashboard/article'
|
href: '/admin/dashboard/article'
|
||||||
|
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
label: '营销文章管理',
|
||||||
|
iconElement: <FontAwesomeIcon icon={faMagnet} className=" fa-fw"></FontAwesomeIcon>,
|
||||||
|
_id: 'adArticleMenagement',
|
||||||
|
href: '/admin/dashboard/adArticle'
|
||||||
|
},
|
||||||
{
|
{
|
||||||
label: '搜索管理',
|
label: '搜索管理',
|
||||||
iconElement: <FontAwesomeIcon icon={faSearch}></FontAwesomeIcon>,
|
iconElement: <FontAwesomeIcon icon={faSearch}></FontAwesomeIcon>,
|
||||||
|
|
Loading…
Reference in New Issue