CommentList.vue 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328
  1. <template>
  2. <div :style="{ position: 'relative', height: allHeight + 'px' }">
  3. <a-list class="jeecg-comment-list" header="" item-layout="horizontal" :data-source="dataList" :style="{ height: commentHeight + 'px' }">
  4. <template #renderItem="{ item }">
  5. <a-list-item style="padding-left: 10px; flex-direction: column" @click="handleClickItem">
  6. <a-comment>
  7. <template #avatar>
  8. <a-avatar class="tx" :src="getAvatar(item)" :alt="getAvatarText(item)">{{ getAvatarText(item) }}</a-avatar>
  9. </template>
  10. <template #author>
  11. <div class="comment-author">
  12. <span>{{ item.fromUserId_dictText }}</span>
  13. <template v-if="item.toUserId">
  14. <span>回复</span>
  15. <span>{{ item.toUserId_dictText }}</span>
  16. <Tooltip class="comment-last-content" @visibleChange="(v)=>visibleChange(v, item)">
  17. <template #title>
  18. <div v-html="getHtml(item.commentId_dictText)"></div>
  19. </template>
  20. <message-outlined />
  21. </Tooltip>
  22. </template>
  23. </div>
  24. </template>
  25. <template #datetime>
  26. <div>
  27. <Tooltip :title="item.createTime">
  28. <span>{{ getDateDiff(item) }}</span>
  29. </Tooltip>
  30. </div>
  31. </template>
  32. <template #actions>
  33. <span @click="showReply(item)">回复</span>
  34. <Popconfirm title="确定删除吗?" @confirm="deleteComment(item)">
  35. <span>删除</span>
  36. </Popconfirm>
  37. </template>
  38. <template #content>
  39. <div v-html="getHtml(item.commentContent)" style="font-size: 15px">
  40. </div>
  41. <div v-if="item.fileList && item.fileList.length > 0">
  42. <!-- 历史文件 -->
  43. <history-file-list :dataList="item.fileList" isComment></history-file-list>
  44. </div>
  45. </template>
  46. </a-comment>
  47. <div v-if="item.commentStatus" class="inner-comment">
  48. <my-comment inner @cancel="item.commentStatus = false" @comment="(content, fileList) => replyComment(item, content, fileList)" :inputFocus="focusStatus"></my-comment>
  49. </div>
  50. </a-list-item>
  51. </template>
  52. </a-list>
  53. <div style="position: absolute; bottom: 0; left: 0; width: 100%; background: #fff; border-top: 1px solid #eee">
  54. <a-comment style="margin: 0 10px">
  55. <template #avatar>
  56. <a-avatar class="tx" :src="getMyAvatar()" :alt="getMyname()">{{ getMyname() }}</a-avatar>
  57. </template>
  58. <template #content>
  59. <my-comment ref="bottomCommentRef" @comment="sendComment" :inputFocus="focusStatus"></my-comment>
  60. </template>
  61. </a-comment>
  62. </div>
  63. </div>
  64. </template>
  65. <script>
  66. /**
  67. * 评论列表
  68. */
  69. import { defineComponent, ref, onMounted, watch, watchEffect } from 'vue';
  70. import { propTypes } from '/@/utils/propTypes';
  71. import dayjs from 'dayjs';
  72. import 'dayjs/locale/zh.js';
  73. import relativeTime from 'dayjs/plugin/relativeTime';
  74. import customParseFormat from 'dayjs/plugin/customParseFormat';
  75. dayjs.locale('zh');
  76. dayjs.extend(relativeTime);
  77. dayjs.extend(customParseFormat);
  78. import { MessageOutlined } from '@ant-design/icons-vue';
  79. import { Comment, Tooltip } from 'ant-design-vue';
  80. import { useUserStore } from '/@/store/modules/user';
  81. import MyComment from './MyComment.vue';
  82. import { list, saveOne, deleteOne, useCommentWithFile, useEmojiHtml, queryById } from './useComment';
  83. import { useMessage } from '/@/hooks/web/useMessage';
  84. import HistoryFileList from './HistoryFileList.vue';
  85. import { Popconfirm } from 'ant-design-vue';
  86. import { getFileAccessHttpUrl } from '/@/utils/common/compUtils';
  87. export default defineComponent({
  88. name: 'CommentList',
  89. components: {
  90. MessageOutlined,
  91. AComment: Comment,
  92. Tooltip,
  93. MyComment,
  94. Popconfirm,
  95. HistoryFileList,
  96. },
  97. props: {
  98. tableName: propTypes.string.def(''),
  99. dataId: propTypes.string.def(''),
  100. datetime: propTypes.number.def(1)
  101. },
  102. setup(props) {
  103. const { createMessage } = useMessage();
  104. const dataList = ref([]);
  105. const { userInfo } = useUserStore();
  106. /**
  107. * 获取当前用户名称
  108. */
  109. function getMyname() {
  110. if (userInfo.realname) {
  111. return userInfo.realname.substr(0, 2);
  112. }
  113. return '';
  114. }
  115. function getMyAvatar(){
  116. return userInfo.avatar;
  117. }
  118. // 获取头像
  119. function getAvatar(item) {
  120. if (item.fromUserAvatar) {
  121. return getFileAccessHttpUrl(item.fromUserAvatar)
  122. }
  123. return '';
  124. }
  125. // 头像没有获取 用户名前两位
  126. function getAvatarText(item){
  127. if (item.fromUserId_dictText) {
  128. return item.fromUserId_dictText.substr(0, 2);
  129. }
  130. return '未知';
  131. }
  132. function getAuthor(item) {
  133. if (item.toUser) {
  134. return item.fromUserId_dictText + ' 回复 ' + item.fromUserId_dictText;
  135. } else {
  136. return item.fromUserId_dictText;
  137. }
  138. }
  139. function getDateDiff(item) {
  140. if (item.createTime) {
  141. const temp = dayjs(item.createTime, 'YYYY-MM-DD hh:mm:ss');
  142. return temp.fromNow();
  143. }
  144. return '';
  145. }
  146. const commentHeight = ref(300);
  147. const allHeight = ref(300);
  148. onMounted(() => {
  149. commentHeight.value = window.innerHeight - 57 - 46 - 70 - 160;
  150. allHeight.value = window.innerHeight - 57 - 46 - 53 -20;
  151. });
  152. /**
  153. * 加载数据
  154. * @returns {Promise<void>}
  155. */
  156. async function loadData() {
  157. const params = {
  158. tableName: props.tableName,
  159. tableDataId: props.dataId,
  160. column: 'createTime',
  161. order: 'desc',
  162. };
  163. const data = await list(params);
  164. if (!data || !data.records || data.records.length == 0) {
  165. dataList.value = [];
  166. } else {
  167. let array = data.records;
  168. console.log(123, array);
  169. dataList.value = array;
  170. }
  171. }
  172. const { saveCommentAndFiles } = useCommentWithFile(props);
  173. // 回复
  174. async function replyComment(item, content, fileList) {
  175. console.log(content, item);
  176. let obj = {
  177. fromUserId: userInfo.id,
  178. toUserId: item.fromUserId,
  179. commentId: item.id,
  180. commentContent: content
  181. }
  182. await saveCommentAndFiles(obj, fileList)
  183. await loadData();
  184. }
  185. //评论
  186. async function sendComment(content, fileList) {
  187. let obj = {
  188. fromUserId: userInfo.id,
  189. commentContent: content
  190. }
  191. await saveCommentAndFiles(obj, fileList)
  192. await loadData();
  193. focusStatus.value = false;
  194. setTimeout(()=>{
  195. focusStatus.value = true;
  196. },100)
  197. }
  198. //删除
  199. async function deleteComment(item) {
  200. const params = { id: item.id };
  201. await deleteOne(params);
  202. await loadData();
  203. }
  204. /**
  205. * 打开回复时触发
  206. * @type {Ref<UnwrapRef<boolean>>}
  207. */
  208. const focusStatus = ref(false);
  209. function showReply(item) {
  210. let arr = dataList.value;
  211. for (let temp of arr) {
  212. temp.commentStatus = false;
  213. }
  214. item.commentStatus = true;
  215. focusStatus.value = false;
  216. focusStatus.value = true;
  217. }
  218. // 表单改变 -重新加载评论列表
  219. watchEffect(() => {
  220. if(props.datetime){
  221. if (props.tableName && props.dataId) {
  222. loadData();
  223. }
  224. }
  225. });
  226. const { getHtml } = useEmojiHtml();
  227. const bottomCommentRef = ref()
  228. function handleClickItem(){
  229. bottomCommentRef.value.changeActive()
  230. }
  231. /**
  232. * 根据id查询评论信息
  233. */
  234. async function visibleChange(v, item){
  235. if(v==true){
  236. if(!item.commentId_dictText){
  237. const data = await queryById(item.commentId);
  238. if(data.success == true){
  239. item.commentId_dictText = data.result.commentContent
  240. }else{
  241. console.error(data.message)
  242. item.commentId_dictText='该评论已被删除';
  243. }
  244. }
  245. }
  246. }
  247. return {
  248. dataList,
  249. getAvatar,
  250. getAvatarText,
  251. getAuthor,
  252. getDateDiff,
  253. commentHeight,
  254. allHeight,
  255. replyComment,
  256. sendComment,
  257. getMyname,
  258. getMyAvatar,
  259. focusStatus,
  260. showReply,
  261. deleteComment,
  262. getHtml,
  263. handleClickItem,
  264. bottomCommentRef,
  265. visibleChange
  266. };
  267. },
  268. });
  269. </script>
  270. <style lang="less" scoped>
  271. .jeecg-comment-list {
  272. overflow: auto;
  273. /* border-bottom: 1px solid #eee;*/
  274. .inner-comment {
  275. width: 100%;
  276. padding: 0 10px;
  277. }
  278. .ant-comment {
  279. width: 100%;
  280. }
  281. }
  282. .comment-author {
  283. span {
  284. margin: 3px;
  285. }
  286. .comment-last-content {
  287. margin-left: 5px;
  288. &:hover{
  289. color: #1890ff;
  290. }
  291. }
  292. }
  293. .ant-list-items{
  294. .ant-list-item:last-child{
  295. margin-bottom: 46px;
  296. }
  297. }
  298. .tx{
  299. margin-top: 4px;
  300. }
  301. </style>