feat: 新增行业、政策、ESG和数据首页管理页面,并更新API接口。
This commit is contained in:
parent
e7e31213da
commit
5093cf8146
@ -39,6 +39,27 @@ export default {
|
||||
deleteDept: (params = {}) => request.delete('/dept/delete', { params }),
|
||||
// auditlog
|
||||
getAuditLogList: (params = {}) => request.get('/auditlog/list', { params }),
|
||||
// system data
|
||||
getIndustryList: (params = {}) => request.get('/industry/list', { params }),
|
||||
getIndustryById: (params = {}) => request.get('/industry/get', { params }),
|
||||
createIndustry: (data = {}) => request.post('/industry/create', data),
|
||||
updateIndustry: (data = {}) => request.post('/industry/update', data),
|
||||
deleteIndustry: (params = {}) => request.delete('/industry/delete', { params }),
|
||||
getPolicyList: (params = {}) => request.get('/policy/list', { params }),
|
||||
getPolicyById: (params = {}) => request.get('/policy/get', { params }),
|
||||
createPolicy: (data = {}) => request.post('/policy/create', data),
|
||||
updatePolicy: (data = {}) => request.post('/policy/update', data),
|
||||
deletePolicy: (params = {}) => request.delete('/policy/delete', { params }),
|
||||
getESGList: (params = {}) => request.get('/esg/list', { params }),
|
||||
getESGById: (params = {}) => request.get('/esg/get', { params }),
|
||||
createESG: (data = {}) => request.post('/esg/create', data),
|
||||
updateESG: (data = {}) => request.post('/esg/update', data),
|
||||
deleteESG: (params = {}) => request.delete('/esg/delete', { params }),
|
||||
getIndexList: (params = {}) => request.get('/index/list', { params }),
|
||||
getIndexById: (params = {}) => request.get('/index/get', { params }),
|
||||
createIndex: (data = {}) => request.post('/index/create', data),
|
||||
updateIndex: (data = {}) => request.post('/index/update', data),
|
||||
deleteIndex: (params = {}) => request.delete('/index/delete', { params }),
|
||||
// app users (客户端用户管理) - 使用现有的后端接口
|
||||
getAppUserList: (params = {}) => request.get('/app-user-admin/list', { params }),
|
||||
updateAppUserQuota: (data = {}) => request.post('/app-user-admin/quota', data),
|
||||
|
||||
263
web/src/views/data/esg/index.vue
Normal file
263
web/src/views/data/esg/index.vue
Normal file
@ -0,0 +1,263 @@
|
||||
<script setup>
|
||||
import { h, onMounted, ref, resolveDirective, withDirectives } from 'vue'
|
||||
import { NButton, NForm, NFormItem, NInput, NInputNumber, NTag, NPopconfirm } from 'naive-ui'
|
||||
|
||||
import CommonPage from '@/components/page/CommonPage.vue'
|
||||
import QueryBarItem from '@/components/query-bar/QueryBarItem.vue'
|
||||
import CrudModal from '@/components/table/CrudModal.vue'
|
||||
import CrudTable from '@/components/table/CrudTable.vue'
|
||||
import TheIcon from '@/components/icon/TheIcon.vue'
|
||||
|
||||
import { formatDate, renderIcon } from '@/utils'
|
||||
import { useCRUD } from '@/composables'
|
||||
import api from '@/api'
|
||||
|
||||
defineOptions({ name: 'ESG关联' })
|
||||
|
||||
const $table = ref(null)
|
||||
const queryItems = ref({})
|
||||
const vPermission = resolveDirective('permission')
|
||||
|
||||
const {
|
||||
modalVisible,
|
||||
modalTitle,
|
||||
modalLoading,
|
||||
handleSave,
|
||||
modalForm,
|
||||
modalFormRef,
|
||||
handleEdit,
|
||||
handleDelete,
|
||||
handleAdd,
|
||||
} = useCRUD({
|
||||
name: 'ESG',
|
||||
initForm: { code: '', name: '', level: '', number: '', remark: '' },
|
||||
doCreate: api.createESG,
|
||||
doUpdate: api.updateESG,
|
||||
doDelete: api.deleteESG,
|
||||
refresh: () => $table.value?.handleSearch(),
|
||||
})
|
||||
|
||||
onMounted(() => {
|
||||
$table.value?.handleSearch()
|
||||
})
|
||||
|
||||
const columns = [
|
||||
{
|
||||
title: 'ESG代码',
|
||||
key: 'code',
|
||||
width: 120,
|
||||
align: 'center',
|
||||
ellipsis: { tooltip: true },
|
||||
},
|
||||
{
|
||||
title: 'ESG名称',
|
||||
key: 'name',
|
||||
width: 200,
|
||||
align: 'center',
|
||||
ellipsis: { tooltip: true },
|
||||
},
|
||||
{
|
||||
title: 'ESG级别',
|
||||
key: 'level',
|
||||
width: 120,
|
||||
align: 'center',
|
||||
render(row) {
|
||||
const levelMap = {
|
||||
E: { type: 'success', text: '环境(E)' },
|
||||
S: { type: 'info', text: '社会(S)' },
|
||||
G: { type: 'warning', text: '治理(G)' },
|
||||
ESG: { type: 'error', text: '综合ESG' },
|
||||
}
|
||||
const level = levelMap[row.level] || { type: 'default', text: row.level }
|
||||
return h(NTag, { type: level.type }, { default: () => level.text })
|
||||
},
|
||||
},
|
||||
{
|
||||
title: 'ESG编号',
|
||||
key: 'number',
|
||||
width: 120,
|
||||
align: 'center',
|
||||
ellipsis: { tooltip: true },
|
||||
},
|
||||
{
|
||||
title: '备注',
|
||||
key: 'remark',
|
||||
width: 200,
|
||||
align: 'center',
|
||||
ellipsis: { tooltip: true },
|
||||
},
|
||||
{
|
||||
title: '创建时间',
|
||||
key: 'created_at',
|
||||
align: 'center',
|
||||
width: 180,
|
||||
render(row) {
|
||||
return h(
|
||||
NButton,
|
||||
{ size: 'small', type: 'text', ghost: true },
|
||||
{ default: () => formatDate(row.created_at), icon: renderIcon('mdi:update', { size: 16 }) }
|
||||
)
|
||||
},
|
||||
},
|
||||
{
|
||||
title: '操作',
|
||||
key: 'actions',
|
||||
width: 240,
|
||||
align: 'center',
|
||||
fixed: 'right',
|
||||
hideInExcel: true,
|
||||
render(row) {
|
||||
return [
|
||||
withDirectives(
|
||||
h(
|
||||
NButton,
|
||||
{
|
||||
size: 'small',
|
||||
type: 'primary',
|
||||
secondary: true,
|
||||
onClick: () => handleEdit(row),
|
||||
},
|
||||
{ default: () => '编辑', icon: renderIcon('material-symbols:edit-outline', { size: 16 }) }
|
||||
),
|
||||
[[vPermission, 'post/api/v1/esg/update']]
|
||||
),
|
||||
withDirectives(
|
||||
h(
|
||||
NPopconfirm,
|
||||
{
|
||||
onPositiveClick: () => handleDelete({ esg_id: row.id }, false),
|
||||
},
|
||||
{
|
||||
default: () => '确认删除',
|
||||
trigger: () =>
|
||||
h(
|
||||
NButton,
|
||||
{ size: 'small', type: 'error', style: 'margin-left: 12px;' },
|
||||
{ default: () => '删除', icon: renderIcon('material-symbols:delete-outline', { size: 16 }) }
|
||||
),
|
||||
}
|
||||
),
|
||||
[[vPermission, 'delete/api/v1/esg/delete']]
|
||||
),
|
||||
]
|
||||
},
|
||||
},
|
||||
]
|
||||
|
||||
const modalRules = {
|
||||
code: [
|
||||
{
|
||||
required: true,
|
||||
message: '请输入ESG代码',
|
||||
trigger: ['input', 'blur'],
|
||||
},
|
||||
],
|
||||
name: [
|
||||
{
|
||||
required: true,
|
||||
message: '请输入ESG名称',
|
||||
trigger: ['input', 'blur'],
|
||||
},
|
||||
],
|
||||
level: [
|
||||
{
|
||||
required: true,
|
||||
message: '请输入ESG级别',
|
||||
trigger: ['input', 'blur'],
|
||||
},
|
||||
],
|
||||
number: [
|
||||
{
|
||||
required: true,
|
||||
message: '请输入ESG编号',
|
||||
trigger: ['input', 'blur'],
|
||||
},
|
||||
],
|
||||
}
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<CommonPage show-footer title="ESG关联">
|
||||
<template #action>
|
||||
<div>
|
||||
<NButton
|
||||
v-permission="'post/api/v1/esg/create'"
|
||||
class="float-right mb-8"
|
||||
type="primary"
|
||||
@click="handleAdd"
|
||||
>
|
||||
<TheIcon icon="material-symbols:add" :size="18" class="mr-5" />新建ESG
|
||||
</NButton>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<CrudTable
|
||||
ref="$table"
|
||||
v-model:query-items="queryItems"
|
||||
:extra-params="{ ordering: 'id' }"
|
||||
:scroll-x="1200"
|
||||
:columns="columns"
|
||||
:get-data="api.getESGList"
|
||||
>
|
||||
<template #queryBar>
|
||||
<QueryBarItem label="ESG代码" :label-width="80">
|
||||
<NInput
|
||||
v-model:value="queryItems.code"
|
||||
type="text"
|
||||
placeholder="请输入ESG代码"
|
||||
@keydown.enter="$table?.handleSearch()"
|
||||
/>
|
||||
</QueryBarItem>
|
||||
<QueryBarItem label="ESG名称" :label-width="80">
|
||||
<NInput
|
||||
v-model:value="queryItems.name"
|
||||
type="text"
|
||||
placeholder="请输入ESG名称"
|
||||
@keydown.enter="$table?.handleSearch()"
|
||||
/>
|
||||
</QueryBarItem>
|
||||
<QueryBarItem label="ESG级别" :label-width="80">
|
||||
<NInput
|
||||
v-model:value="queryItems.level"
|
||||
type="text"
|
||||
placeholder="请输入ESG级别"
|
||||
@keydown.enter="$table?.handleSearch()"
|
||||
/>
|
||||
</QueryBarItem>
|
||||
</template>
|
||||
</CrudTable>
|
||||
|
||||
<CrudModal
|
||||
v-model:visible="modalVisible"
|
||||
:title="modalTitle"
|
||||
:loading="modalLoading"
|
||||
:show-footer="true"
|
||||
@save="handleSave"
|
||||
>
|
||||
<NForm
|
||||
ref="modalFormRef"
|
||||
label-placement="left"
|
||||
label-align="left"
|
||||
:label-width="80"
|
||||
:model="modalForm"
|
||||
:rules="modalRules"
|
||||
>
|
||||
<NFormItem label="ESG代码" path="code">
|
||||
<NInput v-model:value="modalForm.code" placeholder="请输入ESG代码" />
|
||||
</NFormItem>
|
||||
<NFormItem label="ESG名称" path="name">
|
||||
<NInput v-model:value="modalForm.name" placeholder="请输入ESG名称" />
|
||||
</NFormItem>
|
||||
<NFormItem label="ESG级别" path="level">
|
||||
<NInput v-model:value="modalForm.level" placeholder="请输入ESG级别(如:E, S, G, ESG)" />
|
||||
</NFormItem>
|
||||
<NFormItem label="ESG编号" path="number">
|
||||
<NInput v-model:value="modalForm.number" placeholder="请输入ESG编号" />
|
||||
</NFormItem>
|
||||
<NFormItem label="备注" path="remark">
|
||||
<NInput v-model:value="modalForm.remark" type="textarea" placeholder="请输入备注" />
|
||||
</NFormItem>
|
||||
</NForm>
|
||||
</CrudModal>
|
||||
</CommonPage>
|
||||
</template>
|
||||
238
web/src/views/data/index/index.vue
Normal file
238
web/src/views/data/index/index.vue
Normal file
@ -0,0 +1,238 @@
|
||||
<script setup>
|
||||
import { h, onMounted, ref, resolveDirective, withDirectives } from 'vue'
|
||||
import { NButton, NForm, NFormItem, NInput, NInputNumber, NTag, NPopconfirm } from 'naive-ui'
|
||||
|
||||
import CommonPage from '@/components/page/CommonPage.vue'
|
||||
import QueryBarItem from '@/components/query-bar/QueryBarItem.vue'
|
||||
import CrudModal from '@/components/table/CrudModal.vue'
|
||||
import CrudTable from '@/components/table/CrudTable.vue'
|
||||
import TheIcon from '@/components/icon/TheIcon.vue'
|
||||
|
||||
import { formatDate, renderIcon } from '@/utils'
|
||||
import { useCRUD } from '@/composables'
|
||||
import api from '@/api'
|
||||
|
||||
defineOptions({ name: '行业基准' })
|
||||
|
||||
const $table = ref(null)
|
||||
const queryItems = ref({})
|
||||
const vPermission = resolveDirective('permission')
|
||||
|
||||
const {
|
||||
modalVisible,
|
||||
modalTitle,
|
||||
modalLoading,
|
||||
handleSave,
|
||||
modalForm,
|
||||
modalFormRef,
|
||||
handleEdit,
|
||||
handleDelete,
|
||||
handleAdd,
|
||||
} = useCRUD({
|
||||
name: '指数',
|
||||
initForm: { code: '', name: '', search_num: 0, remark: '' },
|
||||
doCreate: api.createIndex,
|
||||
doUpdate: api.updateIndex,
|
||||
doDelete: api.deleteIndex,
|
||||
refresh: () => $table.value?.handleSearch(),
|
||||
})
|
||||
|
||||
onMounted(() => {
|
||||
$table.value?.handleSearch()
|
||||
})
|
||||
|
||||
const columns = [
|
||||
{
|
||||
title: '指数代码',
|
||||
key: 'code',
|
||||
width: 120,
|
||||
align: 'center',
|
||||
ellipsis: { tooltip: true },
|
||||
},
|
||||
{
|
||||
title: '指数名称',
|
||||
key: 'name',
|
||||
width: 200,
|
||||
align: 'center',
|
||||
ellipsis: { tooltip: true },
|
||||
},
|
||||
{
|
||||
title: '搜索次数',
|
||||
key: 'search_num',
|
||||
width: 120,
|
||||
align: 'center',
|
||||
render(row) {
|
||||
const getSearchType = (num) => {
|
||||
if (num >= 1000) return 'error'
|
||||
if (num >= 500) return 'warning'
|
||||
if (num >= 100) return 'info'
|
||||
return 'default'
|
||||
}
|
||||
return h(NTag, { type: getSearchType(row.search_num) }, { default: () => `${row.search_num}次` })
|
||||
},
|
||||
},
|
||||
{
|
||||
title: '备注',
|
||||
key: 'remark',
|
||||
width: 200,
|
||||
align: 'center',
|
||||
ellipsis: { tooltip: true },
|
||||
},
|
||||
{
|
||||
title: '创建时间',
|
||||
key: 'created_at',
|
||||
align: 'center',
|
||||
width: 180,
|
||||
render(row) {
|
||||
return h(
|
||||
NButton,
|
||||
{ size: 'small', type: 'text', ghost: true },
|
||||
{ default: () => formatDate(row.created_at), icon: renderIcon('mdi:update', { size: 16 }) }
|
||||
)
|
||||
},
|
||||
},
|
||||
{
|
||||
title: '操作',
|
||||
key: 'actions',
|
||||
width: 240,
|
||||
align: 'center',
|
||||
fixed: 'right',
|
||||
hideInExcel: true,
|
||||
render(row) {
|
||||
return [
|
||||
withDirectives(
|
||||
h(
|
||||
NButton,
|
||||
{
|
||||
size: 'small',
|
||||
type: 'primary',
|
||||
secondary: true,
|
||||
onClick: () => handleEdit(row),
|
||||
},
|
||||
{ default: () => '编辑', icon: renderIcon('material-symbols:edit-outline', { size: 16 }) }
|
||||
),
|
||||
[[vPermission, 'post/api/v1/index/update']]
|
||||
),
|
||||
withDirectives(
|
||||
h(
|
||||
NPopconfirm,
|
||||
{
|
||||
onPositiveClick: () => handleDelete({ index_id: row.id }, false),
|
||||
},
|
||||
{
|
||||
default: () => '确认删除',
|
||||
trigger: () =>
|
||||
h(
|
||||
NButton,
|
||||
{ size: 'small', type: 'error', style: 'margin-left: 12px;' },
|
||||
{ default: () => '删除', icon: renderIcon('material-symbols:delete-outline', { size: 16 }) }
|
||||
),
|
||||
}
|
||||
),
|
||||
[[vPermission, 'delete/api/v1/index/delete']]
|
||||
),
|
||||
]
|
||||
},
|
||||
},
|
||||
]
|
||||
|
||||
const modalRules = {
|
||||
code: [
|
||||
{
|
||||
required: true,
|
||||
message: '请输入指数代码',
|
||||
trigger: ['input', 'blur'],
|
||||
},
|
||||
],
|
||||
name: [
|
||||
{
|
||||
required: true,
|
||||
message: '请输入指数名称',
|
||||
trigger: ['input', 'blur'],
|
||||
},
|
||||
],
|
||||
search_num: [
|
||||
{
|
||||
required: true,
|
||||
type: 'number',
|
||||
message: '请输入搜索次数',
|
||||
trigger: ['input', 'blur'],
|
||||
},
|
||||
],
|
||||
}
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<CommonPage show-footer title="行业基准">
|
||||
<template #action>
|
||||
<div>
|
||||
<NButton
|
||||
v-permission="'post/api/v1/index/create'"
|
||||
class="float-right mb-8"
|
||||
type="primary"
|
||||
@click="handleAdd"
|
||||
>
|
||||
<TheIcon icon="material-symbols:add" :size="18" class="mr-5" />新建指数
|
||||
</NButton>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<CrudTable
|
||||
ref="$table"
|
||||
v-model:query-items="queryItems"
|
||||
:extra-params="{ ordering: 'id' }"
|
||||
:scroll-x="1200"
|
||||
:columns="columns"
|
||||
:get-data="api.getIndexList"
|
||||
>
|
||||
<template #queryBar>
|
||||
<QueryBarItem label="指数代码" :label-width="80">
|
||||
<NInput
|
||||
v-model:value="queryItems.code"
|
||||
type="text"
|
||||
placeholder="请输入指数代码"
|
||||
@keydown.enter="$table?.handleSearch()"
|
||||
/>
|
||||
</QueryBarItem>
|
||||
<QueryBarItem label="指数名称" :label-width="80">
|
||||
<NInput
|
||||
v-model:value="queryItems.name"
|
||||
type="text"
|
||||
placeholder="请输入指数名称"
|
||||
@keydown.enter="$table?.handleSearch()"
|
||||
/>
|
||||
</QueryBarItem>
|
||||
</template>
|
||||
</CrudTable>
|
||||
|
||||
<CrudModal
|
||||
v-model:visible="modalVisible"
|
||||
:title="modalTitle"
|
||||
:loading="modalLoading"
|
||||
:show-footer="true"
|
||||
@save="handleSave"
|
||||
>
|
||||
<NForm
|
||||
ref="modalFormRef"
|
||||
label-placement="left"
|
||||
label-align="left"
|
||||
:label-width="80"
|
||||
:model="modalForm"
|
||||
:rules="modalRules"
|
||||
>
|
||||
<NFormItem label="指数代码" path="code">
|
||||
<NInput v-model:value="modalForm.code" placeholder="请输入指数代码" />
|
||||
</NFormItem>
|
||||
<NFormItem label="指数名称" path="name">
|
||||
<NInput v-model:value="modalForm.name" placeholder="请输入指数名称" />
|
||||
</NFormItem>
|
||||
<NFormItem label="搜索次数" path="search_num">
|
||||
<NInputNumber v-model:value="modalForm.search_num" placeholder="请输入搜索次数" :min="0" :precision="0" />
|
||||
</NFormItem>
|
||||
<NFormItem label="备注" path="remark">
|
||||
<NInput v-model:value="modalForm.remark" type="textarea" placeholder="请输入备注" />
|
||||
</NFormItem>
|
||||
</NForm>
|
||||
</CrudModal>
|
||||
</CommonPage>
|
||||
</template>
|
||||
253
web/src/views/data/industry/index.vue
Normal file
253
web/src/views/data/industry/index.vue
Normal file
@ -0,0 +1,253 @@
|
||||
<script setup>
|
||||
import { h, onMounted, ref, resolveDirective, withDirectives } from 'vue'
|
||||
import { NButton, NForm, NFormItem, NInput, NInputNumber, NTag, NPopconfirm } from 'naive-ui'
|
||||
|
||||
import CommonPage from '@/components/page/CommonPage.vue'
|
||||
import QueryBarItem from '@/components/query-bar/QueryBarItem.vue'
|
||||
import CrudModal from '@/components/table/CrudModal.vue'
|
||||
import CrudTable from '@/components/table/CrudTable.vue'
|
||||
import TheIcon from '@/components/icon/TheIcon.vue'
|
||||
|
||||
import { formatDate, renderIcon } from '@/utils'
|
||||
import { useCRUD } from '@/composables'
|
||||
import api from '@/api'
|
||||
|
||||
defineOptions({ name: '行业修正' })
|
||||
|
||||
const $table = ref(null)
|
||||
const queryItems = ref({})
|
||||
const vPermission = resolveDirective('permission')
|
||||
|
||||
const {
|
||||
modalVisible,
|
||||
modalTitle,
|
||||
modalLoading,
|
||||
handleSave,
|
||||
modalForm,
|
||||
modalFormRef,
|
||||
handleEdit,
|
||||
handleDelete,
|
||||
handleAdd,
|
||||
} = useCRUD({
|
||||
name: '行业',
|
||||
initForm: { code: '', name: '', roe: 0, fix_num: 0, remark: '' },
|
||||
doCreate: api.createIndustry,
|
||||
doUpdate: api.updateIndustry,
|
||||
doDelete: api.deleteIndustry,
|
||||
refresh: () => $table.value?.handleSearch(),
|
||||
})
|
||||
|
||||
onMounted(() => {
|
||||
$table.value?.handleSearch()
|
||||
})
|
||||
|
||||
const columns = [
|
||||
{
|
||||
title: '行业代码',
|
||||
key: 'code',
|
||||
width: 120,
|
||||
align: 'center',
|
||||
ellipsis: { tooltip: true },
|
||||
},
|
||||
{
|
||||
title: '行业名称',
|
||||
key: 'name',
|
||||
width: 200,
|
||||
align: 'center',
|
||||
ellipsis: { tooltip: true },
|
||||
},
|
||||
{
|
||||
title: 'ROE',
|
||||
key: 'roe',
|
||||
width: 100,
|
||||
align: 'center',
|
||||
render(row) {
|
||||
return h(NTag, { type: 'info' }, { default: () => `${row.roe}%` })
|
||||
},
|
||||
},
|
||||
{
|
||||
title: '修正数值',
|
||||
key: 'fix_num',
|
||||
width: 120,
|
||||
align: 'center',
|
||||
render(row) {
|
||||
const tagType = row.fix_num > 0 ? 'success' : row.fix_num < 0 ? 'error' : 'default'
|
||||
return h(NTag, { type: tagType }, { default: () => row.fix_num })
|
||||
},
|
||||
},
|
||||
{
|
||||
title: '备注',
|
||||
key: 'remark',
|
||||
width: 200,
|
||||
align: 'center',
|
||||
ellipsis: { tooltip: true },
|
||||
},
|
||||
{
|
||||
title: '创建时间',
|
||||
key: 'created_at',
|
||||
align: 'center',
|
||||
width: 180,
|
||||
render(row) {
|
||||
return h(
|
||||
NButton,
|
||||
{ size: 'small', type: 'text', ghost: true },
|
||||
{ default: () => formatDate(row.created_at), icon: renderIcon('mdi:update', { size: 16 }) }
|
||||
)
|
||||
},
|
||||
},
|
||||
{
|
||||
title: '操作',
|
||||
key: 'actions',
|
||||
width: 240,
|
||||
align: 'center',
|
||||
fixed: 'right',
|
||||
hideInExcel: true,
|
||||
render(row) {
|
||||
return [
|
||||
withDirectives(
|
||||
h(
|
||||
NButton,
|
||||
{
|
||||
size: 'small',
|
||||
type: 'primary',
|
||||
secondary: true,
|
||||
onClick: () => handleEdit(row),
|
||||
},
|
||||
{ default: () => '编辑', icon: renderIcon('material-symbols:edit-outline', { size: 16 }) }
|
||||
),
|
||||
[[vPermission, 'post/api/v1/industry/update']]
|
||||
),
|
||||
withDirectives(
|
||||
h(
|
||||
NPopconfirm,
|
||||
{
|
||||
onPositiveClick: () => handleDelete({ industry_id: row.id }, false),
|
||||
},
|
||||
{
|
||||
default: () => '确认删除',
|
||||
trigger: () =>
|
||||
h(
|
||||
NButton,
|
||||
{ size: 'small', type: 'error', style: 'margin-left: 12px;' },
|
||||
{ default: () => '删除', icon: renderIcon('material-symbols:delete-outline', { size: 16 }) }
|
||||
),
|
||||
}
|
||||
),
|
||||
[[vPermission, 'delete/api/v1/industry/delete']]
|
||||
),
|
||||
]
|
||||
},
|
||||
},
|
||||
]
|
||||
|
||||
const modalRules = {
|
||||
code: [
|
||||
{
|
||||
required: true,
|
||||
message: '请输入行业代码',
|
||||
trigger: ['input', 'blur'],
|
||||
},
|
||||
],
|
||||
name: [
|
||||
{
|
||||
required: true,
|
||||
message: '请输入行业名称',
|
||||
trigger: ['input', 'blur'],
|
||||
},
|
||||
],
|
||||
roe: [
|
||||
{
|
||||
required: true,
|
||||
type: 'number',
|
||||
message: '请输入ROE值',
|
||||
trigger: ['input', 'blur'],
|
||||
},
|
||||
],
|
||||
fix_num: [
|
||||
{
|
||||
required: true,
|
||||
type: 'number',
|
||||
message: '请输入修正数值',
|
||||
trigger: ['input', 'blur'],
|
||||
},
|
||||
],
|
||||
}
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<CommonPage show-footer title="行业修正">
|
||||
<template #action>
|
||||
<div>
|
||||
<NButton
|
||||
v-permission="'post/api/v1/industry/create'"
|
||||
class="float-right mb-8"
|
||||
type="primary"
|
||||
@click="handleAdd"
|
||||
>
|
||||
<TheIcon icon="material-symbols:add" :size="18" class="mr-5" />新建行业
|
||||
</NButton>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<CrudTable
|
||||
ref="$table"
|
||||
v-model:query-items="queryItems"
|
||||
:extra-params="{ ordering: 'id' }"
|
||||
:scroll-x="1200"
|
||||
:columns="columns"
|
||||
:get-data="api.getIndustryList"
|
||||
>
|
||||
<template #queryBar>
|
||||
<QueryBarItem label="行业代码" :label-width="80">
|
||||
<NInput
|
||||
v-model:value="queryItems.code"
|
||||
type="text"
|
||||
placeholder="请输入行业代码"
|
||||
@keydown.enter="$table?.handleSearch()"
|
||||
/>
|
||||
</QueryBarItem>
|
||||
<QueryBarItem label="行业名称" :label-width="80">
|
||||
<NInput
|
||||
v-model:value="queryItems.name"
|
||||
type="text"
|
||||
placeholder="请输入行业名称"
|
||||
@keydown.enter="$table?.handleSearch()"
|
||||
/>
|
||||
</QueryBarItem>
|
||||
</template>
|
||||
</CrudTable>
|
||||
|
||||
<CrudModal
|
||||
v-model:visible="modalVisible"
|
||||
:title="modalTitle"
|
||||
:loading="modalLoading"
|
||||
:show-footer="true"
|
||||
@save="handleSave"
|
||||
>
|
||||
<NForm
|
||||
ref="modalFormRef"
|
||||
label-placement="left"
|
||||
label-align="left"
|
||||
:label-width="80"
|
||||
:model="modalForm"
|
||||
:rules="modalRules"
|
||||
>
|
||||
<NFormItem label="行业代码" path="code">
|
||||
<NInput v-model:value="modalForm.code" placeholder="请输入行业代码" />
|
||||
</NFormItem>
|
||||
<NFormItem label="行业名称" path="name">
|
||||
<NInput v-model:value="modalForm.name" placeholder="请输入行业名称" />
|
||||
</NFormItem>
|
||||
<NFormItem label="ROE" path="roe">
|
||||
<NInputNumber v-model:value="modalForm.roe" placeholder="请输入ROE值" :precision="2" :step="0.01" />
|
||||
</NFormItem>
|
||||
<NFormItem label="修正数值" path="fix_num">
|
||||
<NInputNumber v-model:value="modalForm.fix_num" placeholder="请输入修正数值" :precision="2" :step="0.01" />
|
||||
</NFormItem>
|
||||
<NFormItem label="备注" path="remark">
|
||||
<NInput v-model:value="modalForm.remark" type="textarea" placeholder="请输入备注" />
|
||||
</NFormItem>
|
||||
</NForm>
|
||||
</CrudModal>
|
||||
</CommonPage>
|
||||
</template>
|
||||
264
web/src/views/data/policy/index.vue
Normal file
264
web/src/views/data/policy/index.vue
Normal file
@ -0,0 +1,264 @@
|
||||
<script setup>
|
||||
import { h, onMounted, ref, resolveDirective, withDirectives } from 'vue'
|
||||
import { NButton, NForm, NFormItem, NInput, NInputNumber, NTag, NPopconfirm } from 'naive-ui'
|
||||
|
||||
import CommonPage from '@/components/page/CommonPage.vue'
|
||||
import QueryBarItem from '@/components/query-bar/QueryBarItem.vue'
|
||||
import CrudModal from '@/components/table/CrudModal.vue'
|
||||
import CrudTable from '@/components/table/CrudTable.vue'
|
||||
import TheIcon from '@/components/icon/TheIcon.vue'
|
||||
|
||||
import { formatDate, renderIcon } from '@/utils'
|
||||
import { useCRUD } from '@/composables'
|
||||
import api from '@/api'
|
||||
|
||||
defineOptions({ name: '政策匹配' })
|
||||
|
||||
const $table = ref(null)
|
||||
const queryItems = ref({})
|
||||
const vPermission = resolveDirective('permission')
|
||||
|
||||
const {
|
||||
modalVisible,
|
||||
modalTitle,
|
||||
modalLoading,
|
||||
handleSave,
|
||||
modalForm,
|
||||
modalFormRef,
|
||||
handleEdit,
|
||||
handleDelete,
|
||||
handleAdd,
|
||||
} = useCRUD({
|
||||
name: '政策',
|
||||
initForm: { code: '', name: '', level: '', score: 0 },
|
||||
doCreate: api.createPolicy,
|
||||
doUpdate: api.updatePolicy,
|
||||
doDelete: api.deletePolicy,
|
||||
refresh: () => $table.value?.handleSearch(),
|
||||
})
|
||||
|
||||
onMounted(() => {
|
||||
$table.value?.handleSearch()
|
||||
})
|
||||
|
||||
const columns = [
|
||||
{
|
||||
title: '政策代码',
|
||||
key: 'code',
|
||||
width: 120,
|
||||
align: 'center',
|
||||
ellipsis: { tooltip: true },
|
||||
},
|
||||
{
|
||||
title: '政策名称',
|
||||
key: 'name',
|
||||
width: 200,
|
||||
align: 'center',
|
||||
ellipsis: { tooltip: true },
|
||||
},
|
||||
{
|
||||
title: '政策级别',
|
||||
key: 'level',
|
||||
width: 120,
|
||||
align: 'center',
|
||||
render(row) {
|
||||
const levelMap = {
|
||||
national: { type: 'error', text: '国家级' },
|
||||
provincial: { type: 'warning', text: '省级' },
|
||||
municipal: { type: 'info', text: '市级' },
|
||||
county: { type: 'default', text: '县级' },
|
||||
}
|
||||
const level = levelMap[row.level] || { type: 'default', text: row.level }
|
||||
return h(NTag, { type: level.type }, { default: () => level.text })
|
||||
},
|
||||
},
|
||||
{
|
||||
title: '政策评分',
|
||||
key: 'score',
|
||||
width: 120,
|
||||
align: 'center',
|
||||
render(row) {
|
||||
const getScoreType = (score) => {
|
||||
if (score >= 80) return 'success'
|
||||
if (score >= 60) return 'warning'
|
||||
return 'error'
|
||||
}
|
||||
return h(NTag, { type: getScoreType(row.score) }, { default: () => `${row.score}分` })
|
||||
},
|
||||
},
|
||||
{
|
||||
title: '创建时间',
|
||||
key: 'created_at',
|
||||
align: 'center',
|
||||
width: 180,
|
||||
render(row) {
|
||||
return h(
|
||||
NButton,
|
||||
{ size: 'small', type: 'text', ghost: true },
|
||||
{ default: () => formatDate(row.created_at), icon: renderIcon('mdi:update', { size: 16 }) }
|
||||
)
|
||||
},
|
||||
},
|
||||
{
|
||||
title: '操作',
|
||||
key: 'actions',
|
||||
width: 240,
|
||||
align: 'center',
|
||||
fixed: 'right',
|
||||
hideInExcel: true,
|
||||
render(row) {
|
||||
return [
|
||||
withDirectives(
|
||||
h(
|
||||
NButton,
|
||||
{
|
||||
size: 'small',
|
||||
type: 'primary',
|
||||
secondary: true,
|
||||
onClick: () => handleEdit(row),
|
||||
},
|
||||
{ default: () => '编辑', icon: renderIcon('material-symbols:edit-outline', { size: 16 }) }
|
||||
),
|
||||
[[vPermission, 'post/api/v1/policy/update']]
|
||||
),
|
||||
withDirectives(
|
||||
h(
|
||||
NPopconfirm,
|
||||
{
|
||||
onPositiveClick: () => handleDelete({ policy_id: row.id }, false),
|
||||
},
|
||||
{
|
||||
default: () => '确认删除',
|
||||
trigger: () =>
|
||||
h(
|
||||
NButton,
|
||||
{ size: 'small', type: 'error', style: 'margin-left: 12px;' },
|
||||
{ default: () => '删除', icon: renderIcon('material-symbols:delete-outline', { size: 16 }) }
|
||||
),
|
||||
}
|
||||
),
|
||||
[[vPermission, 'delete/api/v1/policy/delete']]
|
||||
),
|
||||
]
|
||||
},
|
||||
},
|
||||
]
|
||||
|
||||
const modalRules = {
|
||||
code: [
|
||||
{
|
||||
required: true,
|
||||
message: '请输入政策代码',
|
||||
trigger: ['input', 'blur'],
|
||||
},
|
||||
],
|
||||
name: [
|
||||
{
|
||||
required: true,
|
||||
message: '请输入政策名称',
|
||||
trigger: ['input', 'blur'],
|
||||
},
|
||||
],
|
||||
level: [
|
||||
{
|
||||
required: true,
|
||||
message: '请输入政策级别',
|
||||
trigger: ['input', 'blur'],
|
||||
},
|
||||
],
|
||||
score: [
|
||||
{
|
||||
required: true,
|
||||
type: 'number',
|
||||
message: '请输入政策评分',
|
||||
trigger: ['input', 'blur'],
|
||||
},
|
||||
],
|
||||
}
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<CommonPage show-footer title="政策匹配">
|
||||
<template #action>
|
||||
<div>
|
||||
<NButton
|
||||
v-permission="'post/api/v1/policy/create'"
|
||||
class="float-right mb-8"
|
||||
type="primary"
|
||||
@click="handleAdd"
|
||||
>
|
||||
<TheIcon icon="material-symbols:add" :size="18" class="mr-5" />新建政策
|
||||
</NButton>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<CrudTable
|
||||
ref="$table"
|
||||
v-model:query-items="queryItems"
|
||||
:extra-params="{ ordering: 'id' }"
|
||||
:scroll-x="1200"
|
||||
:columns="columns"
|
||||
:get-data="api.getPolicyList"
|
||||
>
|
||||
<template #queryBar>
|
||||
<QueryBarItem label="政策代码" :label-width="80">
|
||||
<NInput
|
||||
v-model:value="queryItems.code"
|
||||
type="text"
|
||||
placeholder="请输入政策代码"
|
||||
@keydown.enter="$table?.handleSearch()"
|
||||
/>
|
||||
</QueryBarItem>
|
||||
<QueryBarItem label="政策名称" :label-width="80">
|
||||
<NInput
|
||||
v-model:value="queryItems.name"
|
||||
type="text"
|
||||
placeholder="请输入政策名称"
|
||||
@keydown.enter="$table?.handleSearch()"
|
||||
/>
|
||||
</QueryBarItem>
|
||||
<QueryBarItem label="政策级别" :label-width="80">
|
||||
<NInput
|
||||
v-model:value="queryItems.level"
|
||||
type="text"
|
||||
placeholder="请输入政策级别"
|
||||
@keydown.enter="$table?.handleSearch()"
|
||||
/>
|
||||
</QueryBarItem>
|
||||
</template>
|
||||
</CrudTable>
|
||||
|
||||
<CrudModal
|
||||
v-model:visible="modalVisible"
|
||||
:title="modalTitle"
|
||||
:loading="modalLoading"
|
||||
:show-footer="true"
|
||||
@save="handleSave"
|
||||
>
|
||||
<NForm
|
||||
ref="modalFormRef"
|
||||
label-placement="left"
|
||||
label-align="left"
|
||||
:label-width="80"
|
||||
:model="modalForm"
|
||||
:rules="modalRules"
|
||||
>
|
||||
<NFormItem label="政策代码" path="code">
|
||||
<NInput v-model:value="modalForm.code" placeholder="请输入政策代码" />
|
||||
</NFormItem>
|
||||
<NFormItem label="政策名称" path="name">
|
||||
<NInput v-model:value="modalForm.name" placeholder="请输入政策名称" />
|
||||
</NFormItem>
|
||||
<NFormItem label="政策级别" path="level">
|
||||
<NInput
|
||||
v-model:value="modalForm.level"
|
||||
placeholder="请输入政策级别(如:national, provincial, municipal, county)"
|
||||
/>
|
||||
</NFormItem>
|
||||
<NFormItem label="政策评分" path="score">
|
||||
<NInputNumber v-model:value="modalForm.score" placeholder="请输入政策评分" :min="0" :max="100" :precision="0" />
|
||||
</NFormItem>
|
||||
</NForm>
|
||||
</CrudModal>
|
||||
</CommonPage>
|
||||
</template>
|
||||
Loading…
x
Reference in New Issue
Block a user