Files
cdTestPlant3/cdTMP/src/layout/project-layout.vue

704 lines
26 KiB
Vue
Raw Normal View History

2023-06-12 20:47:54 +08:00
<template>
<a-layout class="layout">
<div class="navbar layout-navbar">
2023-06-15 20:13:46 +08:00
<NavBar :title="projectInfo.name" />
2023-06-12 20:47:54 +08:00
</div>
<a-layout class="layout">
<a-layout class="layout layout-demo">
<a-layout-sider class="layout-sider">
2023-07-21 16:08:08 +08:00
<div class="p-2 overflow-auto myhcalc">
<a-input-group class="mb-2 w-full flex items-center h-1/12" size="mini">
2023-06-19 19:51:12 +08:00
<a-input style="height: 32px" v-model="searchKey" allow-clear></a-input>
<a-button @click="handleSearchTreeDataClick">搜索</a-button>
2023-06-12 20:47:54 +08:00
</a-input-group>
2023-08-21 19:57:49 +08:00
<a-button type="primary" @click="toggleExpanded" class="mb-1">
{{ expandedKeys?.length ? "全部收缩" : "全部展开" }}
</a-button>
2024-03-29 19:03:35 +08:00
<a-popconfirm type="warning" @ok="handleCopyNode" content="是否确定根据选中节点进行创建?">
2024-04-02 18:44:26 +08:00
<a-button type="outline" status="warning" class="ml-1">
<template #icon> <icon-plus /></template>点击复制创建轮次
</a-button>
2024-03-29 19:03:35 +08:00
</a-popconfirm>
2023-06-12 20:47:54 +08:00
<a-tree
2023-08-24 17:05:21 +08:00
class="h-10/12 select-none"
2023-06-12 20:47:54 +08:00
:data="treeData"
size="small"
2024-03-29 19:03:35 +08:00
checkable
2023-06-12 20:47:54 +08:00
block-node
animation
@select="pointNode"
:load-more="loadMore"
2023-08-21 19:57:49 +08:00
v-model:expanded-keys="expandedKeys"
2023-08-25 13:28:24 +08:00
v-model:selected-keys="selectedKeys"
2024-03-29 19:03:35 +08:00
v-model:checked-keys="checkedKeys"
2023-06-12 20:47:54 +08:00
showLine
ref="treeRef"
2023-06-16 19:30:36 +08:00
border
2023-06-19 19:51:12 +08:00
:default-selected-keys="[currentNode ? currentNode : route.query.key]"
2023-06-20 18:09:30 +08:00
>
<!-- 在轮次节点可以新增编辑删除复制 -->
<template #extra="nodeData">
<template v-if="nodeData.level === '0'">
<a-tooltip content="点击新增轮次">
<IconPlus
style="
position: absolute;
right: 8px;
font-size: 12px;
top: 8px;
color: #3370ff;
"
@click="() => handleRoundAddClick(nodeData)"
/>
</a-tooltip>
2023-06-28 21:02:31 +08:00
<a-tooltip content="点击删除轮次">
<a-popconfirm
content="确定要删除该轮次吗?"
position="bottom"
@ok="handleRoundDelClick(nodeData)"
>
<IconMinus
style="
position: absolute;
right: 25px;
font-size: 12px;
top: 8px;
color: #3370ff;
"
/>
</a-popconfirm>
</a-tooltip>
2023-06-20 18:09:30 +08:00
<a-tooltip content="点击编辑当前轮次"
><IconEdit
style="
position: absolute;
right: 42px;
font-size: 12px;
top: 8px;
color: #3370ff;
"
@click="() => handleRoundEditClick(nodeData)"
/></a-tooltip>
</template>
</template>
2023-08-09 16:30:05 +08:00
<!-- 节点图标插槽 -->
<template #icon="props">
2023-08-15 17:15:52 +08:00
<template v-if="props.node.level === '1'"> [被测件] </template>
<template v-if="props.node.level === '2'"> [] </template>
<template v-if="props.node.level === '3'"> [] </template>
<template v-if="props.node.level === '4'"> [] </template>
2023-08-09 16:30:05 +08:00
</template>
2023-06-20 18:09:30 +08:00
</a-tree>
2023-06-12 20:47:54 +08:00
</div>
</a-layout-sider>
2023-07-21 16:08:08 +08:00
<a-layout class="layout-content myhcalc">
2023-06-15 20:13:46 +08:00
<a-layout-content class="work-area project-layout">
<PageLayout />
</a-layout-content>
</a-layout>
2023-06-12 20:47:54 +08:00
</a-layout>
</a-layout>
</a-layout>
2023-06-20 18:09:30 +08:00
<ma-form-modal
ref="maFormModalRef"
:title="title"
:column="roundColumn"
:options="roundOption"
width="800px"
2023-06-28 21:02:31 +08:00
:submit="handleRoundSubmit"
2023-06-20 18:09:30 +08:00
></ma-form-modal>
2024-03-29 19:03:35 +08:00
<!-- 如果没有第一轮的SO_dut则弹窗 -->
<ma-form-modal
ref="soDutFormRef"
title="新增第一轮源代码被测件"
:column="soDutColumn"
width="800px"
:submit="handleSoDutSubmit"
:custom-cancel="handleSoDutCancel"
></ma-form-modal>
<Progress
:visible="visible"
:isComplete="isComplete"
:text="ptext"
@clickConfirm="handleModalConfirmClick"
></Progress>
2023-06-12 20:47:54 +08:00
</template>
<script setup>
2023-08-09 16:30:05 +08:00
import { provide, ref, onMounted, h } from "vue"
2023-06-12 20:47:54 +08:00
import NavBar from "@/layout/components/navbar.vue"
import PageLayout from "@/layout/page-layout.vue"
2023-06-20 18:09:30 +08:00
import MaFormModal from "@/components/ma-form-modal/index.vue"
2023-06-12 20:47:54 +08:00
import projectApi from "@/api/project/project"
2023-06-20 18:09:30 +08:00
import roundApi from "@/api/project/round"
2024-03-29 19:03:35 +08:00
import dutApi from "@/api/project/dut"
import copyApi from "@/api/treeOperation/copy"
import { Message, Notification } from "@arco-design/web-vue"
2023-06-12 20:47:54 +08:00
import { useRoute } from "vue-router"
2023-06-15 20:13:46 +08:00
import { useRouter } from "vue-router"
2023-06-19 19:51:12 +08:00
import { useTreeDataStore } from "@/store"
import { storeToRefs } from "pinia"
2023-06-28 21:02:31 +08:00
import dayjs from "dayjs"
2024-03-29 19:03:35 +08:00
import Progress from "@/views/testmanage/projmanage/cpns/progress.vue"
/// 初始化round轮次数据
const treeDataStore = useTreeDataStore()
const route = useRoute()
const router = useRouter()
const treeRef = ref()
const { treeData, currentNode } = storeToRefs(treeDataStore)
const projectInfo = ref({ ...route.query })
const projectId = ref(route.query.id)
//~~~~ 复制粘贴流程
const checkedKeys = ref([])
/// 点击复制按钮
const handleCopyNode = async () => {
2024-04-02 18:44:26 +08:00
// 1.先判断是否选中了节点
if (checkedKeys.value.length < 1) {
Message.error("您未选择节点,请选择后再试...")
return
}
2024-03-29 19:03:35 +08:00
// 打印下checked节点key
visible.value = true
isComplete.value = false
2024-04-02 18:44:26 +08:00
const st = await copyApi
.copyCheckedNode({ checkedNodes: checkedKeys.value, project_id: projectId.value })
.catch((err) => {
isComplete.value = true
visible.value = false
})
2024-03-29 19:03:35 +08:00
isComplete.value = true
Message.success(st.message)
2024-04-02 18:44:26 +08:00
// 处理完后需要更新树结构
treeDataStore.resetTreeData(projectId.value)
// 清除右侧路由的组件显示,不然会因为数据而出错
router.replace({ name: "project" })
2024-03-29 19:03:35 +08:00
}
/// 进度条变量
const visible = ref(false)
const isComplete = ref(false)
const ptext = ref("第二轮数据")
const handleModalConfirmClick = () => {
visible.value = false
}
2023-06-20 18:09:30 +08:00
//~~~~ 缩小后的menu菜单
2023-06-12 20:47:54 +08:00
const drawerVisible = ref(false)
provide("toggleDrawerMenu", () => {
drawerVisible.value = !drawerVisible.value
})
2023-06-20 18:09:30 +08:00
//~~~~ 搜索绑定与搜索按钮点击
2023-06-19 19:51:12 +08:00
const searchKey = ref("")
const handleSearchTreeDataClick = () => {
const loop = (itemdata) => {
const result = []
itemdata.forEach((item) => {
if (item.title.indexOf(searchKey.value) > -1) {
result.push({ ...item })
} else if (item.children) {
const filterdata = loop(item.children)
if (filterdata.length) {
result.push({
...item,
children: filterdata
})
}
}
})
return result
}
// 返回过滤后的treeData
// treeDataStore.originTreeData
if (searchKey.value) {
treeData.value = loop(treeDataStore.treeData)
} else {
treeData.value = treeDataStore.originTreeData
}
}
2023-06-20 18:09:30 +08:00
//~~~~ 树状组件
2024-03-29 19:03:35 +08:00
// ~~~~定义弹出a-form-modal的submit方法~~~
const handleSoDutSubmit = async (data) => {
// 只有一个参数就是被提交的数据然后返回false阻止弹窗消失
const input_data = { ...data, project_id: projectId.value }
const res = await dutApi.createR1SoDut(input_data)
if (res.code == 200) {
treeDataStore.updateDutTreeData(res.data, projectId.value)
Message.success("添加成功...")
setTimeout(() => {
location.reload()
}, 500)
return
}
return false
}
// ~~~~定义弹出a-form-modal的cancel方法-返回false则无法关闭弹窗~~~~
const handleSoDutCancel = () => {
Notification.error("必须添加第一轮源代码信息,无法关闭!")
return false
}
2023-08-15 17:15:52 +08:00
// 初始化树状数据
2024-03-29 19:03:35 +08:00
// so_dut弹窗ref对象
const soDutFormRef = ref()
2023-06-12 20:47:54 +08:00
onMounted(async () => {
2024-04-19 18:53:52 +08:00
await treeDataStore.initTreeData(projectId.value)
2024-03-29 19:03:35 +08:00
// 检查是否存在第一轮的源代码dut
/// 主动后端请求
const res = await dutApi.getSoExists({ id: projectId.value })
/// 如果包含第一轮源代码则不处理不包含则弹窗必须添加dut
if (!res.data.exists) {
// 这里就必须让用户选择添加了
Message.warning("识别到您未添加第一轮源代码的信息,此信息必填")
soDutFormRef.value.open({})
}
2023-06-12 20:47:54 +08:00
})
2023-08-25 13:28:24 +08:00
// v-model绑定选中节点
const selectedKeys = ref([])
const previousKey = ref(null)
2023-08-24 17:05:21 +08:00
// 1.定义展开的tree-key 2.定义全部展开的数据 3.定义展开收缩函数 -> 注意在treeStore里面使用递归处理
const expandedKeys = ref([])
const allExpandedKeys = ref([])
const toggleExpanded = () => {
allExpandedKeys.value = treeDataStore.outExpandNode()
expandedKeys.value = expandedKeys?.value.length ? [] : allExpandedKeys.value
}
2023-08-15 17:15:52 +08:00
/// 点击树状节点-参数1:节点数组参数2:树node对象 - 添加双击处理方式
2023-08-24 17:05:21 +08:00
let timerId = null
let count = 0
2023-06-12 20:47:54 +08:00
const pointNode = (value, data) => {
2023-08-25 13:28:24 +08:00
// 获取处理单击不选中,双击选中的变量
let catch_selected_key = selectedKeys.value
selectedKeys.value = previousKey.value
2023-08-24 17:05:21 +08:00
count++
if (timerId) {
return
2023-06-15 20:13:46 +08:00
}
2023-08-24 17:05:21 +08:00
timerId = setTimeout(async () => {
if (count > 1) {
// 双击触发 value是点击的节点data是节点数据
if (data.node.level == "0") {
projectApi.getDutInfo(projectInfo.value.id, data.node.key, data.node.level).then((res) => {
treeData.value[value].children = res.data
// 添加需要展开数据注意不要一直push判断在已展开节点是否包含点击的节点
if (!expandedKeys.value.includes(value[0])) {
expandedKeys.value.push(value[0])
}
})
}
if (data.node.level == "1") {
projectApi.getDemandInfo(projectInfo.value.id, data.node.key, data.node.level).then((res) => {
data.node.children = res.data
if (!expandedKeys.value.includes(value[0])) {
expandedKeys.value.push(value[0])
}
})
}
if (data.node.level == "2") {
projectApi.getTestInfo(projectInfo.value.id, data.node.key, data.node.level).then((res) => {
data.node.children = res.data
if (!expandedKeys.value.includes(value[0])) {
expandedKeys.value.push(value[0])
}
})
}
if (data.node.level == "3") {
projectApi.getCaseInfo(projectInfo.value.id, data.node.key, data.node.level).then((res) => {
data.node.children = res.data
if (!expandedKeys.value.includes(value[0])) {
expandedKeys.value.push(value[0])
}
})
}
count = 0
clearTimeout(timerId)
timerId = null
} else {
// 单击触发
if (data.node.level === "0") {
router.push({ name: "round", query: { ...projectInfo.value, key: data.node.key } })
}
if (data.node.level === "1") {
router.push({ name: "dut", query: { ...projectInfo.value, key: data.node.key } })
}
if (data.node.level === "2") {
router.push({ name: "designDemand", query: { ...projectInfo.value, key: data.node.key } })
}
if (data.node.level === "3") {
router.push({ name: "testDemand", query: { ...projectInfo.value, key: data.node.key } })
}
if (data.node.level === "4") {
router.push({ name: "case", query: { ...projectInfo.value, key: data.node.key } })
}
2024-03-13 21:18:08 +08:00
// if (data.node.level === "5") {
// router.push({ name: "problem", query: { ...projectInfo.value, key: data.node.key } })
// }
2023-08-24 17:05:21 +08:00
treeDataStore.setCurrentNode(data.node.key)
2023-08-25 13:28:24 +08:00
// 单击设置选择的key
selectedKeys.value = catch_selected_key
// 单击后设置previous选中的key是什么
previousKey.value = catch_selected_key
2023-08-24 17:05:21 +08:00
count = 0
clearTimeout(timerId)
timerId = null
}
}, 250)
2023-06-12 20:47:54 +08:00
}
2023-06-15 20:13:46 +08:00
/// 动态加载函数-参数1:树node对象
2023-06-12 20:47:54 +08:00
const loadMore = (nodeData) => {
2023-06-15 20:13:46 +08:00
if (nodeData.level == "0") {
return new Promise(async (resolve) => {
2024-04-02 18:44:26 +08:00
const res = await projectApi.getDutInfo(projectInfo.value.id, nodeData.key, nodeData.level).catch((err) => {
resolve() // 捕获错误,让动态加载不再一直转圈
})
2023-07-21 16:08:08 +08:00
nodeData.children = res.data
2023-06-15 20:13:46 +08:00
resolve()
2024-04-02 18:44:26 +08:00
})
2023-06-15 20:13:46 +08:00
}
2023-06-16 19:30:36 +08:00
if (nodeData.level == "1") {
2023-06-15 20:13:46 +08:00
return new Promise(async (resolve) => {
2024-04-02 18:44:26 +08:00
const res = await projectApi
.getDemandInfo(projectInfo.value.id, nodeData.key, nodeData.level)
.catch((err) => {
resolve()
})
2023-07-21 16:08:08 +08:00
nodeData.children = res.data
2023-06-15 20:13:46 +08:00
resolve()
})
}
2023-06-16 19:30:36 +08:00
if (nodeData.level == "2") {
2023-06-15 20:13:46 +08:00
return new Promise(async (resolve) => {
2024-04-02 18:44:26 +08:00
const res = await projectApi
.getTestInfo(projectInfo.value.id, nodeData.key, nodeData.level)
.catch((err) => {
resolve()
})
2023-07-21 16:08:08 +08:00
nodeData.children = res.data
2023-06-15 20:13:46 +08:00
resolve()
})
}
2023-06-16 19:30:36 +08:00
if (nodeData.level == "3") {
2023-06-19 19:51:12 +08:00
return new Promise(async (resolve) => {
2024-04-02 18:44:26 +08:00
const res = await projectApi
.getCaseInfo(projectInfo.value.id, nodeData.key, nodeData.level)
.catch((err) => {
resolve()
})
2023-07-21 16:08:08 +08:00
nodeData.children = res.data
2023-06-19 19:51:12 +08:00
resolve()
})
}
2023-06-12 20:47:54 +08:00
}
2023-06-20 18:09:30 +08:00
//~~~~ 表单弹窗组件功能
const maFormModalRef = ref()
const title = ref("")
/// 点击新增轮次按钮
const handleRoundAddClick = (nodeData) => {
2023-07-25 20:03:06 +08:00
let create_number = treeDataStore.getRoundMiddleInfo()
2023-06-28 21:02:31 +08:00
// 这里是文档写错了,调用form是里面组件绑定的数据
2023-06-20 18:09:30 +08:00
maFormModalRef.value.form = {}
2023-07-25 20:03:06 +08:00
// 这里如果删除中间轮次,那么新增信息应该从中间开始
2023-06-28 21:02:31 +08:00
maFormModalRef.value.open({
beginTime: dayjs().format("YYYY-MM-DD"),
grade: "1",
2023-07-25 20:03:06 +08:00
key: `${create_number}`,
name: `${create_number + 1}轮测试`,
ident: `${route.query.ident}-R${create_number + 1}`,
2023-06-28 21:02:31 +08:00
project: projectId.value
})
title.value = "新增轮次"
2023-06-20 18:09:30 +08:00
}
/// 点击编辑轮次按钮
const handleRoundEditClick = async (nodeData) => {
const data = await roundApi.getOneRoundInfo({
projectId: projectId.value,
round: nodeData.key
})
maFormModalRef.value.open(data.data)
title.value = `编辑轮次:${data.data.name}`
}
/// 点击删除轮次按钮
2023-06-28 21:02:31 +08:00
const handleRoundDelClick = async (value) => {
try {
await roundApi.delete(projectId.value, value)
Message.success("删除成功!")
treeDataStore.resetTreeData(projectId.value)
2024-04-02 18:44:26 +08:00
router.replace({ name: "project" })
2024-03-29 19:03:35 +08:00
} catch {}
2023-06-28 21:02:31 +08:00
}
/// Ma-form-Modal的提交按钮
const handleRoundSubmit = async (value) => {
if (title.value.slice(0, 1) === "编") {
try {
await roundApi.update(value.id, value)
Message.success("编辑成功!")
} catch {
Message.error("编辑失败!")
}
}
if (title.value.slice(0, 1) === "新") {
try {
await roundApi.save(projectId.value, value)
Message.success("新增成功!")
treeDataStore.resetTreeData(projectId.value)
} catch {
Message.error("新增失败!")
}
}
2023-06-20 18:09:30 +08:00
}
2023-08-21 19:57:49 +08:00
2023-06-20 18:09:30 +08:00
/// 设置轮次弹窗的列信息
const roundColumn = ref([
{
title: "基本信息",
customClass: ["mb-2", "pb-0"],
formType: "card",
bodyStyle: { paddingBottom: 0 },
formList: [
{
formType: "grid",
cols: [
{
span: 12,
formList: [
{ title: "标识", dataIndex: "ident" },
{
title: "开始时间",
dataIndex: "beginTime",
formType: "date",
placeholder: "请选择时间",
rules: [{ required: true, message: "开始时间必填" }]
},
{
title: "速度等级",
dataIndex: "speedGrade",
placeholder: "请填入速度等级",
rules: [{ required: true, message: "速度等级必填" }]
}
]
},
{
span: 12,
formList: [
{ title: "名称", dataIndex: "name" },
{
title: "结束时间",
dataIndex: "endTime",
formType: "date",
placeholder: "请选择时间",
rules: [{ required: true, message: "结束时间必填" }]
},
{
title: "封装",
dataIndex: "package",
placeholder: "请填入封装",
rules: [{ required: true, message: "封装必填" }]
}
]
}
]
}
]
},
{
title: "质量等级",
dataIndex: "grade",
formType: "radio",
dict: {
data: [
{ label: "军级", value: "1" },
{ label: "商业级", value: "2" },
{ label: "宇航级", value: "3" },
{ label: "工业级", value: "4" }
]
},
placeholder: "请填入质量等级",
rules: [{ required: true, message: "质量等级必填" }]
},
{
formType: "card",
title: "极端工况信息",
customClass: ["mb-2", "pb-0"],
bodyStyle: { paddingBottom: 0 },
formList: [
{
formType: "divider",
title: "最好工况",
orientation: "left",
margin: "20px"
},
{
formType: "grid",
cols: [
{
span: 12,
formList: [{ title: "电压", dataIndex: "best_condition_voltage", placeholder: "请填入电压" }]
},
{
span: 12,
formList: [{ title: "温度", dataIndex: "best_condition_tem", placeholder: "请填入温度" }]
}
]
},
{
formType: "divider",
title: "典型工况",
orientation: "left",
margin: "20px"
},
{
formType: "grid",
cols: [
{
span: 12,
formList: [{ title: "电压", dataIndex: "typical_condition_voltage", placeholder: "请填入电压" }]
},
{
span: 12,
formList: [{ title: "温度", dataIndex: "typical_condition_tem", placeholder: "请填入温度" }]
}
]
},
{
formType: "divider",
title: "最差工况",
orientation: "left",
margin: "20px"
},
{
formType: "grid",
cols: [
{
span: 12,
formList: [{ title: "电压", dataIndex: "low_condition_voltage", placeholder: "请填入电压" }]
},
{
span: 12,
formList: [{ title: "温度", dataIndex: "low_condition_tem", placeholder: "请填入温度" }]
}
]
}
]
}
])
/// 表单的option
const roundOption = ref({
customClass: [""]
})
2024-03-29 19:03:35 +08:00
// 源代码so_dut的弹窗
const soDutColumn = ref([
{
title: "代码版本",
dataIndex: "version",
placeholder: "请输入代码版本",
rules: [{ required: true, message: "代码版本必填" }]
},
{
title: "用户标识",
dataIndex: "userRef",
placeholder: "请输入用户标识"
},
{
title: "单位",
dataIndex: "unit",
placeholder: "请选择单位名称",
formType: "select",
dict: { url: "system/contact/index", props: { label: "name", value: "name" }, translation: true },
rules: [{ required: true, message: "单位必选" }]
},
{
title: "发布时间",
dataIndex: "date",
rules: [{ required: true, message: "时间必填" }],
formType: "date"
},
{
title: "空行",
dataIndex: "black_line",
2024-05-07 17:32:46 +08:00
formType: "input-number",
rules: [{ required: true, message: "空行数必填" }],
2024-03-29 19:03:35 +08:00
},
{
2024-04-19 18:53:52 +08:00
title: "纯注释",
dataIndex: "comment_line",
2024-05-07 17:32:46 +08:00
formType: "input-number",
rules: [{ required: true, message: "纯注释数必填" }],
2024-03-29 19:03:35 +08:00
},
{
2024-04-19 18:53:52 +08:00
title: "混合行",
2024-03-29 19:03:35 +08:00
dataIndex: "mix_line",
2024-05-07 17:32:46 +08:00
formType: "input-number",
rules: [{ required: true, message: "混合行必填" }],
2024-03-29 19:03:35 +08:00
},
{
2024-04-19 18:53:52 +08:00
title: "纯代码",
dataIndex: "code_line",
2024-05-07 17:32:46 +08:00
formType: "input-number",
rules: [{ required: true, message: "纯代码行必填" }],
2024-03-29 19:03:35 +08:00
}
])
2023-06-12 20:47:54 +08:00
</script>
<style lang="less" scoped>
.tree {
height: 100%;
}
2023-06-15 20:13:46 +08:00
.layout {
display: flex;
}
2023-06-12 20:47:54 +08:00
.layout-demo :deep(.arco-layout-sider) {
width: 300px !important;
}
.layout {
width: 100%;
height: 100%;
}
.layout-navbar {
position: fixed;
top: 0;
left: 0;
z-index: 100;
width: 100%;
height: 60px;
}
.layout-sider {
position: fixed;
top: 60px;
left: 0;
z-index: 99;
height: 100%;
transition: all 0.2s cubic-bezier(0.34, 0.69, 0.1, 1);
&::after {
position: absolute;
top: 0;
right: -1px;
display: block;
width: 1px;
height: 100%;
background-color: var(--color-border);
content: "";
}
> :deep(.arco-layout-sider-children) {
overflow-y: hidden;
}
}
2023-06-15 20:13:46 +08:00
.layout-demo {
position: relative;
}
2023-06-12 20:47:54 +08:00
.layout-content {
2023-06-15 20:13:46 +08:00
position: absolute;
top: 60px;
left: 300px;
2023-06-12 20:47:54 +08:00
min-height: 100vh;
2023-06-15 20:13:46 +08:00
width: 100% - 300px;
2023-06-12 20:47:54 +08:00
overflow-y: auto;
background-color: var(--color-fill-2);
transition: padding 0.2s cubic-bezier(0.34, 0.69, 0.1, 1);
2023-06-15 20:13:46 +08:00
position: absolute;
2023-06-12 20:47:54 +08:00
}
2023-07-21 16:08:08 +08:00
.myhcalc {
height: calc(100% - 60px);
}
2023-06-12 20:47:54 +08:00
</style>