Files
iot-device-management-frontend/apps/web-antd/src/views/mp/draft/modules/news-form.vue
2025-11-25 21:13:46 +08:00

265 lines
7.8 KiB
Vue
Raw 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.

<script lang="ts" setup>
import type { MpDraftApi } from '#/api/mp/draft';
import { computed, ref } from 'vue';
import { confirm } from '@vben/common-ui';
import { IconifyIcon } from '@vben/icons';
import { Button, Col, Input, Layout, Row, Textarea } from 'ant-design-vue';
import { createEmptyNewsItem } from '#/api/mp/draft';
import { Tinymce as RichTextarea } from '#/components/tinymce';
import CoverSelect from './cover-select.vue';
defineOptions({ name: 'NewsForm' });
const props = defineProps<{
isCreating: boolean;
modelValue: MpDraftApi.NewsItem[] | null;
}>();
const emit = defineEmits<{
(e: 'update:modelValue', v: MpDraftApi.NewsItem[]): void;
}>();
const newsList = computed<MpDraftApi.NewsItem[]>({
get() {
return props.modelValue === null
? [createEmptyNewsItem()]
: props.modelValue;
},
set(val) {
emit('update:modelValue', val);
},
});
const activeNewsIndex = ref(0);
const activeNewsItem = computed(() => {
const item = newsList.value[activeNewsIndex.value];
if (!item) {
return createEmptyNewsItem();
}
return item;
});
/** 将图文向下移动 */
function moveDownNews(index: number) {
const current = newsList.value[index];
const next = newsList.value[index + 1];
if (current && next) {
newsList.value[index] = next;
newsList.value[index + 1] = current;
activeNewsIndex.value = index + 1;
}
}
/** 将图文向上移动 */
function moveUpNews(index: number) {
const current = newsList.value[index];
const prev = newsList.value[index - 1];
if (current && prev) {
newsList.value[index] = prev;
newsList.value[index - 1] = current;
activeNewsIndex.value = index - 1;
}
}
/** 删除指定 index 的图文 */
async function removeNews(index: number) {
await confirm('确定删除该图文吗?');
newsList.value.splice(index, 1);
if (activeNewsIndex.value === index) {
activeNewsIndex.value = 0;
}
}
/** 添加一个图文 */
function plusNews() {
newsList.value.push(createEmptyNewsItem());
activeNewsIndex.value = newsList.value.length - 1;
}
</script>
<template>
<Layout>
<Layout.Sider width="40%" theme="light">
<div class="mx-auto mb-[10px] w-[60%] border border-[#eaeaea] p-[10px]">
<!-- TODO @hw头条次条的展示不对微信聊过的呀~ -->
<div v-for="(news, index) in newsList" :key="index">
<div
class="group relative mx-auto mb-[10px] w-full cursor-pointer border-[2px] bg-white"
v-if="index === 0"
:class="
activeNewsIndex === index
? 'border-green-500'
: 'border-transparent'
"
@click="activeNewsIndex = index"
>
<div class="relative w-full bg-[#acadae]">
<img class="h-full w-full" :src="news.thumbUrl" />
<div
class="absolute bottom-0 left-0 inline-block h-[25px] w-[100%] overflow-hidden text-ellipsis whitespace-nowrap bg-black p-[1%] text-center text-[15px] text-white opacity-65"
>
{{ news.title }}
</div>
</div>
<div
class="relative flex justify-center gap-[10px] py-[5px] text-center"
v-if="newsList.length > 1"
>
<Button
type="default"
shape="circle"
size="small"
@click="() => moveDownNews(index)"
>
<IconifyIcon icon="lucide:arrow-down" />
</Button>
<Button
v-if="isCreating"
type="primary"
danger
shape="circle"
size="small"
@click="() => removeNews(index)"
>
<IconifyIcon icon="lucide:trash-2" />
</Button>
</div>
</div>
<div
class="group relative mx-auto mb-[10px] cursor-pointer border-[2px] bg-white"
v-if="index > 0"
:class="
activeNewsIndex === index
? 'border-green-500'
: 'border-transparent'
"
@click="activeNewsIndex = index"
>
<div class="relative">
<div class="bg-[#acadae]">
<img class="block h-full w-full" :src="news.thumbUrl" />
<div
class="absolute bottom-0 left-0 inline-block h-[25px] w-[100%] overflow-hidden text-ellipsis whitespace-nowrap bg-black p-[1%] text-center text-[15px] text-white opacity-65"
>
{{ news.title }}
</div>
</div>
</div>
<div
class="relative flex justify-center gap-[10px] py-[5px] text-center"
>
<Button
v-if="newsList.length > index + 1"
shape="circle"
type="default"
size="small"
@click="() => moveDownNews(index)"
>
<IconifyIcon icon="lucide:arrow-down" />
</Button>
<Button
v-if="index > 0"
type="default"
shape="circle"
size="small"
@click="() => moveUpNews(index)"
>
<IconifyIcon icon="lucide:arrow-up" />
</Button>
<Button
v-if="isCreating"
type="primary"
danger
size="small"
shape="circle"
@click="() => removeNews(index)"
>
<IconifyIcon icon="lucide:trash-2" />
</Button>
</div>
</div>
</div>
<Row
justify="center"
class="mt-[5px] border-t border-[#eaeaea] pt-[5px] text-center"
>
<Button
type="primary"
shape="circle"
@click="plusNews"
v-if="newsList.length < 8 && isCreating"
>
<IconifyIcon icon="lucide:plus" />
</Button>
</Row>
</div>
</Layout.Sider>
<Layout.Content class="bg-white">
<div v-if="newsList.length > 0 && activeNewsItem">
<!-- 标题作者原文地址 -->
<Row :gutter="20">
<Col :span="24">
<Input
v-model:value="activeNewsItem.title"
placeholder="请输入标题(必填)"
/>
</Col>
<Col :span="24" class="mt-[5px]">
<Input
v-model:value="activeNewsItem.author"
placeholder="请输入作者"
/>
</Col>
<Col :span="24" class="mt-[5px]">
<Input
v-model:value="activeNewsItem.contentSourceUrl"
placeholder="请输入原文地址"
/>
</Col>
</Row>
<!-- 封面和摘要 -->
<Row :gutter="20">
<Col :span="12">
<CoverSelect
v-model="activeNewsItem"
:is-first="activeNewsIndex === 0"
/>
</Col>
<Col :span="12">
<p>摘要:</p>
<Textarea
:rows="8"
v-model:value="activeNewsItem.digest"
placeholder="请输入摘要"
class="inline-block w-full align-top"
:maxlength="120"
:show-count="true"
/>
</Col>
</Row>
<!--富文本编辑器组件-->
<Row>
<Col :span="24">
<RichTextarea v-model="activeNewsItem.content" />
</Col>
</Row>
</div>
</Layout.Content>
</Layout>
</template>
<style lang="scss" scoped>
:deep(.ant-row) {
margin-bottom: 20px;
}
:deep(.ant-row:last-child) {
margin-bottom: 0;
}
</style>