| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343 |
- import { useEffect, useRef, useState } from "react";
- import { Button, Space, Typography } from "antd";
- import { LinkOutlined } from "@ant-design/icons";
- import { get } from "../../request";
- import type { ICommentListResponse } from "../../api/Comment";
- import type { IComment } from "./DiscussionItem";
- import type { IAnswerCount } from "./DiscussionDrawer";
- import { type ActionType, ProList } from "@ant-design/pro-components";
- import { renderBadge } from "../channel/ChannelTable";
- import DiscussionCreate from "./DiscussionCreate";
- import User from "../auth/User";
- import type { IArticleListResponse } from "../../api/Article";
- import { useAppSelector } from "../../hooks";
- import { currentUser as _currentUser } from "../../reducers/current-user";
- import { CommentOutlinedIcon, TemplateOutlinedIcon } from "../../assets/icon";
- import type { ISentenceResponse } from "../../api/Corpus";
- import type { TDiscussionType } from "./Discussion";
- import { courseInfo } from "../../reducers/current-course";
- import { courseUser } from "../../reducers/course-user";
- import TimeShow from "../general/TimeShow";
- const { Paragraph } = Typography;
- interface IWidget {
- resId?: string;
- resType?: TResType;
- topicId?: string;
- userId?: string;
- changedAnswerCount?: IAnswerCount;
- type?: TDiscussionType;
- pageSize?: number;
- showStudent?: boolean; //在课程中是否显示学生discussions
- onSelect?: Function;
- onItemCountChange?: Function;
- onReply?: Function;
- onReady?: Function;
- }
- const DiscussionListCardWidget = ({
- resId,
- resType,
- topicId,
- userId,
- showStudent = false,
- onSelect,
- changedAnswerCount,
- type = "discussion",
- pageSize = 10,
- onItemCountChange,
- ___onReply,
- onReady,
- }: IWidget) => {
- const ref = useRef<ActionType | null>(null);
- const [activeKey, setActiveKey] = useState<React.Key | undefined>("active");
- const [activeNumber, setActiveNumber] = useState<number>(0);
- const [closeNumber, setCloseNumber] = useState<number>(0);
- const [count, setCount] = useState<number>(0);
- const [canCreate, setCanCreate] = useState(false);
- const course = useAppSelector(courseInfo);
- const myCourse = useAppSelector(courseUser);
- const user = useAppSelector(_currentUser);
- useEffect(() => {
- ref.current?.reload();
- }, [resId, resType]);
- useEffect(() => {
- console.log("changedAnswerCount", changedAnswerCount);
- ref.current?.reload();
- }, [changedAnswerCount]);
- if (
- typeof resId === "undefined" &&
- typeof topicId === "undefined" &&
- typeof userId === "undefined"
- ) {
- return (
- <Typography.Paragraph>
- 该资源尚未创建,不能发表讨论。
- </Typography.Paragraph>
- );
- }
- return (
- <>
- <ProList<IComment>
- rowKey="id"
- actionRef={ref}
- metas={{
- avatar: {
- render(_dom, entity, _index, _action, _schema) {
- return (
- <>
- <User {...entity.user} showName={false} />
- </>
- );
- },
- },
- title: {
- render(_dom, entity, index, _action, _schema) {
- return (
- <>
- <div>
- {entity.resId !== resId ? <LinkOutlined /> : <></>}
- <Button
- key={index}
- size="small"
- type="link"
- icon={
- entity.newTpl ? <TemplateOutlinedIcon /> : undefined
- }
- onClick={(event) => {
- if (typeof onSelect !== "undefined") {
- onSelect(event, entity);
- }
- }}
- >
- {entity.title}
- </Button>
- </div>
- <div>
- <TimeShow
- type="secondary"
- showIcon={false}
- createdAt={entity.createdAt}
- updatedAt={entity.updatedAt}
- />
- </div>
- </>
- );
- },
- },
- description: {
- dataIndex: "content",
- search: false,
- render(_dom, entity, index, _action, _schema) {
- const content = entity.summary ?? entity.content;
- return (
- <div key={index}>
- <Paragraph
- type="secondary"
- ellipsis={{
- rows: 2,
- expandable: true,
- onEllipsis: (ellipsis) => {
- console.log("Ellipsis changed:", ellipsis);
- },
- }}
- title={content}
- >
- {content}
- </Paragraph>
- </div>
- );
- },
- },
- actions: {
- render: (_text, row, index, _action) => [
- row.childrenCount ? (
- <Space key={index}>
- <CommentOutlinedIcon key={"icon"} />
- <span key={"count"}>{row.childrenCount}</span>
- </Space>
- ) : (
- <></>
- ),
- ],
- },
- }}
- request={async (params = {}, _sorter, _filter) => {
- let url: string = `/v2/discussion?type=${type}&res_type=${resType}&`;
- if (typeof topicId !== "undefined") {
- url += `view=question-by-topic&id=${topicId}`;
- } else if (typeof resId !== "undefined") {
- url += `view=question&id=${resId}`;
- } else if (typeof userId !== "undefined") {
- url += `view=topic-by-user`;
- } else {
- return {
- total: 0,
- succcess: false,
- };
- }
- const offset =
- ((params.current ? params.current : 1) - 1) *
- (params.pageSize ? params.pageSize : pageSize);
- url += `&limit=${params.pageSize}&offset=${offset}`;
- url += params.keyword ? "&search=" + params.keyword : "";
- url += activeKey ? "&status=" + activeKey : "";
- if (myCourse && course) {
- if (myCourse.role !== "student") {
- url += `&course=${course.courseId}`;
- }
- }
- if (showStudent) {
- url += `&show_student=true`;
- }
- console.info("DiscussionListCard api request", url);
- const res = await get<ICommentListResponse>(url);
- console.info("DiscussionListCard api response", res);
- setCount(res.data.active);
- setCanCreate(res.data.can_create);
- const items: IComment[] = res.data.rows.map((item, _id) => {
- return {
- id: item.id,
- resId: item.res_id,
- resType: item.res_type,
- type: item.type,
- user: item.editor,
- title: item.title,
- parent: item.parent,
- tplId: item.tpl_id,
- content: item.content,
- summary: item.summary,
- status: item.status,
- childrenCount: item.children_count,
- createdAt: item.created_at,
- updatedAt: item.updated_at,
- };
- });
- let topicTpl: IComment[] = [];
- if (
- activeKey !== "close" &&
- user?.roles?.includes("basic") === false
- ) {
- //获取channel模版
- let studioName: string | undefined;
- switch (resType) {
- case "sentence":
- const url = `/v2/sentence/${resId}`;
- console.info("api request", url);
- const sentInfo = await get<ISentenceResponse>(url);
- console.info("api response", sentInfo);
- studioName = sentInfo.data.studio.realName;
- break;
- }
- const urlTpl = `/v2/article?view=template&studio_name=${studioName}&subtitle=_template_discussion_topic_&content=true`;
- const resTpl = await get<IArticleListResponse>(urlTpl);
- if (resTpl.ok) {
- console.log("resTpl.data.rows", resTpl.data.rows);
- topicTpl = resTpl.data.rows
- .filter(
- (value) =>
- items.findIndex((old) => old.tplId === value.uid) === -1
- )
- .map((item, _index) => {
- return {
- tplId: item.uid,
- resId: resId,
- resType: resType,
- type: "discussion",
- user: item.editor
- ? item.editor
- : { id: "", userName: "", nickName: "" },
- title: item.title,
- parent: null,
- content: item.content,
- html: item.html,
- summary: item.summary ? item.summary : item._summary,
- status: "active",
- childrenCount: 0,
- newTpl: true,
- createdAt: item.created_at,
- updatedAt: item.updated_at,
- };
- });
- }
- }
- setActiveNumber(res.data.active);
- setCloseNumber(res.data.close);
- if (typeof onReady !== "undefined") {
- onReady();
- }
- return {
- total: res.data.count,
- succcess: true,
- data: [...topicTpl, ...items],
- };
- }}
- bordered
- pagination={{
- showQuickJumper: true,
- showSizeChanger: true,
- pageSize: pageSize,
- }}
- search={false}
- options={{
- search: false,
- }}
- toolbar={{
- menu: {
- activeKey,
- items: [
- {
- key: "active",
- label: (
- <span>
- active
- {renderBadge(activeNumber, activeKey === "active")}
- </span>
- ),
- },
- {
- key: "close",
- label: (
- <span>
- close
- {renderBadge(closeNumber, activeKey === "close")}
- </span>
- ),
- },
- ],
- onChange(key) {
- setActiveKey(key);
- ref.current?.reload();
- },
- },
- }}
- />
- {canCreate && resId && resType ? (
- <DiscussionCreate
- contentType="markdown"
- resId={resId}
- resType={resType}
- type={type}
- onCreated={(_e: IComment) => {
- if (typeof onItemCountChange !== "undefined") {
- onItemCountChange(count + 1);
- }
- ref.current?.reload();
- }}
- />
- ) : undefined}
- </>
- );
- };
- export default DiscussionListCardWidget;
|