| | |
| | | <template> |
| | | <FormDialog |
| | | v-model="dialogVisible" |
| | | <FormDialog v-model="dialogVisible" |
| | | :title="operationType === 'add' ? '新建工资表' : '编辑工资表'" |
| | | width="90%" |
| | | @close="closeDia" |
| | | > |
| | | @close="closeDia"> |
| | | <template #footer> |
| | | <el-button type="info" @click="saveDraft">保存草稿</el-button> |
| | | <el-button type="primary" @click="submitForm">确认提交</el-button> |
| | | <el-button type="info" |
| | | @click="saveDraft">保存草稿</el-button> |
| | | <el-button type="primary" |
| | | @click="submitForm">确认提交</el-button> |
| | | <el-button @click="closeDia">取消</el-button> |
| | | </template> |
| | | <div class="form-dia-body"> |
| | | <!-- 基础资料 --> |
| | | <el-card class="form-card" shadow="never"> |
| | | <el-card class="form-card" |
| | | shadow="never"> |
| | | <template #header> |
| | | <span class="card-title"><span class="card-title-line">|</span> 基础资料</span> |
| | | <el-icon class="card-collapse"><ArrowUp /></el-icon> |
| | | <el-icon class="card-collapse"> |
| | | <ArrowUp /> |
| | | </el-icon> |
| | | </template> |
| | | <el-form ref="formRef" :model="form" :rules="rules" label-position="top"> |
| | | <el-form ref="formRef" |
| | | :model="form" |
| | | :rules="rules" |
| | | label-position="top"> |
| | | <el-row :gutter="24"> |
| | | <el-col :span="6"> |
| | | <el-form-item label="工资主题" prop="salaryTitle"> |
| | | <el-input |
| | | v-model="form.salaryTitle" |
| | | <el-form-item label="工资主题" |
| | | prop="salaryTitle"> |
| | | <el-input v-model="form.salaryTitle" |
| | | placeholder="请输入" |
| | | clearable |
| | | maxlength="20" |
| | | show-word-limit |
| | | /> |
| | | show-word-limit /> |
| | | </el-form-item> |
| | | </el-col> |
| | | <el-col :span="6"> |
| | | <el-form-item label="选择部门" prop="deptIds"> |
| | | <el-select |
| | | v-model="form.deptIds" |
| | | <el-form-item label="选择部门" |
| | | prop="deptIds"> |
| | | <el-select v-model="form.deptIds" |
| | | placeholder="请选择" |
| | | clearable |
| | | multiple |
| | | collapse-tags-tooltip |
| | | style="width: 100%" |
| | | > |
| | | <el-option |
| | | v-for="item in deptOptions" |
| | | style="width: 100%"> |
| | | <el-option v-for="item in deptOptions" |
| | | :key="item.deptId" |
| | | :label="item.deptName" |
| | | :value="item.deptId" |
| | | /> |
| | | :value="item.deptId" /> |
| | | </el-select> |
| | | </el-form-item> |
| | | </el-col> |
| | | <el-col :span="6"> |
| | | <el-form-item label="选择工资月份" prop="salaryMonth"> |
| | | <el-date-picker |
| | | v-model="form.salaryMonth" |
| | | <el-form-item label="选择工资月份" |
| | | prop="salaryMonth"> |
| | | <el-date-picker v-model="form.salaryMonth" |
| | | type="month" |
| | | value-format="YYYY-MM" |
| | | format="YYYY-MM" |
| | | placeholder="请选择工资月份" |
| | | style="width: 100%" |
| | | clearable |
| | | /> |
| | | clearable /> |
| | | </el-form-item> |
| | | </el-col> |
| | | <el-col :span="6"> |
| | | <el-form-item label="备注" prop="remark"> |
| | | <el-input |
| | | v-model="form.remark" |
| | | <el-form-item label="备注" |
| | | prop="remark"> |
| | | <el-input v-model="form.remark" |
| | | placeholder="请输入" |
| | | clearable |
| | | /> |
| | | clearable /> |
| | | </el-form-item> |
| | | </el-col> |
| | | </el-row> |
| | | <el-row :gutter="24"> |
| | | <el-col :span="6"> |
| | | <el-form-item label="支付银行" prop="payBank"> |
| | | <el-select |
| | | v-model="form.payBank" |
| | | <el-form-item label="支付银行" |
| | | prop="payBank"> |
| | | <el-select v-model="form.payBank" |
| | | placeholder="请选择" |
| | | clearable |
| | | filterable |
| | | style="width: 100%" |
| | | > |
| | | <el-option |
| | | v-for="b in bankOptions" |
| | | style="width: 100%"> |
| | | <el-option v-for="b in bankOptions" |
| | | :key="b" |
| | | :label="b" |
| | | :value="b" |
| | | /> |
| | | :value="b" /> |
| | | </el-select> |
| | | </el-form-item> |
| | | </el-col> |
| | | <el-col :span="6"> |
| | | <el-form-item label="审核人" prop="auditUserId"> |
| | | <el-select |
| | | v-model="form.auditUserId" |
| | | <el-form-item label="审核人" |
| | | prop="auditUserId"> |
| | | <el-select v-model="form.auditUserId" |
| | | placeholder="请选择审核人" |
| | | clearable |
| | | filterable |
| | | style="width: 100%" |
| | | > |
| | | <el-option |
| | | v-for="item in userList" |
| | | style="width: 100%"> |
| | | <el-option v-for="item in userList" |
| | | :key="item.userId" |
| | | :label="item.nickName" |
| | | :value="item.userId" |
| | | /> |
| | | :value="item.userId" /> |
| | | </el-select> |
| | | </el-form-item> |
| | | </el-col> |
| | | </el-row> |
| | | </el-form> |
| | | </el-card> |
| | | |
| | | <!-- 操作按钮 --> |
| | | <div class="toolbar"> |
| | | <el-button type="primary" @click="handleGenerate">生成工资表</el-button> |
| | | <el-button type="primary" |
| | | @click="handleGenerate">生成工资表</el-button> |
| | | <el-button @click="handleClear">清空</el-button> |
| | | <el-button @click="handleBatchDelete">删除</el-button> |
| | | <el-button @click="handleTaxForm">个税表</el-button> |
| | | </div> |
| | | |
| | | <!-- 员工工资详情表格 --> |
| | | <div class="employee-table-wrap"> |
| | | <el-table |
| | | ref="employeeTableRef" |
| | | <el-table ref="employeeTableRef" |
| | | :data="employeeList" |
| | | border |
| | | max-height="400" |
| | | @selection-change="onEmployeeSelectionChange" |
| | | > |
| | | <el-table-column type="selection" width="55" align="center" /> |
| | | <el-table-column label="员工姓名" prop="staffName" minWidth="100" /> |
| | | <el-table-column label="部门" prop="deptName" minWidth="100" /> |
| | | <el-table-column label="基本工资" minWidth="110"> |
| | | @selection-change="onEmployeeSelectionChange"> |
| | | <el-table-column type="selection" |
| | | width="55" |
| | | align="center" /> |
| | | <el-table-column label="员工姓名" |
| | | prop="staffName" |
| | | minWidth="100" /> |
| | | <el-table-column label="部门" |
| | | prop="deptName" |
| | | minWidth="100" /> |
| | | <el-table-column label="基本工资" |
| | | minWidth="110"> |
| | | <template #default="{ row }"> |
| | | <el-input |
| | | v-model.number="row.basicSalary" |
| | | <el-input v-model.number="row.basicSalary" |
| | | type="number" |
| | | placeholder="0" |
| | | size="small" |
| | | @input="row.basicSalary = parseNum(row.basicSalary)" |
| | | /> |
| | | @input="row.basicSalary = parseNum(row.basicSalary)" /> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="计件工资" minWidth="110"> |
| | | <el-table-column label="计件工资" |
| | | minWidth="110"> |
| | | <template #default="{ row }"> |
| | | <el-input |
| | | v-model.number="row.pieceSalary" |
| | | <el-input v-model.number="row.pieceSalary" |
| | | type="number" |
| | | placeholder="0" |
| | | size="small" |
| | | @input="row.pieceSalary = parseNum(row.pieceSalary)" |
| | | /> |
| | | disabled |
| | | @input="row.pieceSalary = parseNum(row.pieceSalary)" /> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="计时工资" minWidth="110"> |
| | | <el-table-column label="计时工资" |
| | | minWidth="110"> |
| | | <template #default="{ row }"> |
| | | <el-input |
| | | v-model.number="row.hourlySalary" |
| | | <el-input v-model.number="row.hourlySalary" |
| | | type="number" |
| | | placeholder="0" |
| | | size="small" |
| | | @input="row.hourlySalary = parseNum(row.hourlySalary)" |
| | | /> |
| | | disabled |
| | | @input="row.hourlySalary = parseNum(row.hourlySalary)" /> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="其他收入" minWidth="110"> |
| | | <el-table-column label="其他收入" |
| | | minWidth="110"> |
| | | <template #default="{ row }"> |
| | | <el-input |
| | | v-model.number="row.otherIncome" |
| | | <el-input v-model.number="row.otherIncome" |
| | | type="number" |
| | | placeholder="0" |
| | | size="small" |
| | | @input="row.otherIncome = parseNum(row.otherIncome)" |
| | | /> |
| | | @input="row.otherIncome = parseNum(row.otherIncome)" /> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="社保个人" minWidth="110"> |
| | | <el-table-column label="社保个人" |
| | | minWidth="110"> |
| | | <template #default="{ row }"> |
| | | <el-input |
| | | v-model.number="row.socialPersonal" |
| | | <el-input v-model.number="row.socialPersonal" |
| | | type="number" |
| | | placeholder="0" |
| | | size="small" |
| | | @input="row.socialPersonal = parseNum(row.socialPersonal)" |
| | | /> |
| | | disabled |
| | | @input="row.socialPersonal = parseNum(row.socialPersonal)" /> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="公积金个人" minWidth="120"> |
| | | <el-table-column label="公积金个人" |
| | | minWidth="120"> |
| | | <template #default="{ row }"> |
| | | <el-input |
| | | v-model.number="row.fundPersonal" |
| | | <el-input v-model.number="row.fundPersonal" |
| | | type="number" |
| | | placeholder="0" |
| | | size="small" |
| | | @input="row.fundPersonal = parseNum(row.fundPersonal)" |
| | | /> |
| | | disabled |
| | | @input="row.fundPersonal = parseNum(row.fundPersonal)" /> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="其他支出" minWidth="110"> |
| | | <el-table-column label="其他支出" |
| | | minWidth="110"> |
| | | <template #default="{ row }"> |
| | | <el-input |
| | | v-model.number="row.otherDeduct" |
| | | <el-input v-model.number="row.otherDeduct" |
| | | type="number" |
| | | placeholder="0" |
| | | size="small" |
| | | @input="row.otherDeduct = parseNum(row.otherDeduct)" |
| | | /> |
| | | disabled |
| | | @input="row.otherDeduct = parseNum(row.otherDeduct)" /> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="工资个税" minWidth="110"> |
| | | <el-table-column label="工资个税" |
| | | minWidth="110"> |
| | | <template #default="{ row }"> |
| | | <el-input |
| | | v-model.number="row.salaryTax" |
| | | <el-input v-model.number="row.salaryTax" |
| | | type="number" |
| | | placeholder="0" |
| | | size="small" |
| | | @input="row.salaryTax = parseNum(row.salaryTax)" |
| | | /> |
| | | disabled |
| | | @input="row.salaryTax = parseNum(row.salaryTax)" /> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="应发工资" minWidth="110"> |
| | | <el-table-column label="应发工资" |
| | | minWidth="110"> |
| | | <template #default="{ row }"> |
| | | <el-input |
| | | v-model.number="row.grossSalary" |
| | | <el-input v-model.number="row.grossSalary" |
| | | type="number" |
| | | placeholder="0" |
| | | size="small" |
| | | @input="row.grossSalary = parseNum(row.grossSalary)" |
| | | /> |
| | | disabled |
| | | @input="row.grossSalary = parseNum(row.grossSalary)" /> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="应扣工资" minWidth="110"> |
| | | <el-table-column label="应扣工资" |
| | | minWidth="110"> |
| | | <template #default="{ row }"> |
| | | <el-input |
| | | v-model.number="row.deductSalary" |
| | | <el-input v-model.number="row.deductSalary" |
| | | type="number" |
| | | placeholder="0" |
| | | size="small" |
| | | @input="row.deductSalary = parseNum(row.deductSalary)" |
| | | /> |
| | | disabled |
| | | @input="row.deductSalary = parseNum(row.deductSalary)" /> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="实发工资" minWidth="110"> |
| | | <el-table-column label="实发工资" |
| | | minWidth="110"> |
| | | <template #default="{ row }"> |
| | | <el-input |
| | | v-model.number="row.netSalary" |
| | | <el-input v-model.number="row.netSalary" |
| | | type="number" |
| | | placeholder="0" |
| | | size="small" |
| | | @input="row.netSalary = parseNum(row.netSalary)" |
| | | /> |
| | | disabled |
| | | @input="row.netSalary = parseNum(row.netSalary)" /> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="备注" minWidth="120"> |
| | | <el-table-column label="备注" |
| | | minWidth="120"> |
| | | <template #default="{ row }"> |
| | | <el-input |
| | | v-model="row.remark" |
| | | <el-input v-model="row.remark" |
| | | placeholder="请输入" |
| | | size="small" |
| | | /> |
| | | size="small" /> |
| | | </template> |
| | | </el-table-column> |
| | | <el-table-column label="操作" width="80" align="center" fixed="right"> |
| | | <el-table-column label="操作" |
| | | width="80" |
| | | align="center" |
| | | fixed="right"> |
| | | <template #default="{ row }"> |
| | | <el-button type="primary" link @click="removeEmployee(row)">删除</el-button> |
| | | <el-button type="primary" |
| | | link |
| | | @click="removeEmployee(row)">删除</el-button> |
| | | </template> |
| | | </el-table-column> |
| | | </el-table> |
| | | <div v-if="!employeeList.length" class="table-empty">暂无数据</div> |
| | | <div v-else class="salary-total"> |
| | | <div v-if="!employeeList.length" |
| | | class="table-empty">暂无数据</div> |
| | | <div v-else |
| | | class="salary-total"> |
| | | <span class="total-label">工资总额:</span> |
| | | <span class="total-value">¥ {{ totalSalary.toFixed(2) }}</span> |
| | | </div> |
| | | </div> |
| | | </div> |
| | | |
| | | |
| | | |
| | | <!-- 新增人员弹窗 --> |
| | | <el-dialog |
| | | v-model="addPersonVisible" |
| | | <el-dialog v-model="addPersonVisible" |
| | | title="新增人员" |
| | | width="400px" |
| | | append-to-body |
| | | @close="addPersonClose" |
| | | > |
| | | @close="addPersonClose"> |
| | | <div class="add-person-tree"> |
| | | <el-tree |
| | | ref="personTreeRef" |
| | | <el-tree ref="personTreeRef" |
| | | :data="deptStaffTree" |
| | | show-checkbox |
| | | node-key="id" |
| | | :props="{ label: 'label', children: 'children' }" |
| | | default-expand-all |
| | | /> |
| | | default-expand-all /> |
| | | </div> |
| | | <template #footer> |
| | | <el-button @click="addPersonVisible = false">取消</el-button> |
| | | <el-button type="primary" @click="confirmAddPerson">确定</el-button> |
| | | <el-button type="primary" |
| | | @click="confirmAddPerson">确定</el-button> |
| | | </template> |
| | | </el-dialog> |
| | | |
| | | <!-- 个税表弹窗 --> |
| | | <el-dialog |
| | | v-model="taxDialogVisible" |
| | | <el-dialog v-model="taxDialogVisible" |
| | | title="个税表" |
| | | width="700px" |
| | | append-to-body |
| | | > |
| | | append-to-body> |
| | | <div class="tax-desc">个人所得税免征额:5000元</div> |
| | | <el-table :data="taxTableData" border style="width: 100%;margin-bottom: 20px;"> |
| | | <el-table-column prop="level" label="级数" width="80" align="center" /> |
| | | <el-table-column |
| | | prop="range" |
| | | <el-table :data="taxTableData" |
| | | border |
| | | style="width: 100%;margin-bottom: 20px;"> |
| | | <el-table-column prop="level" |
| | | label="级数" |
| | | width="80" |
| | | align="center" /> |
| | | <el-table-column prop="range" |
| | | label="全年应纳税所得额/元" |
| | | min-width="220" |
| | | /> |
| | | <el-table-column |
| | | prop="rate" |
| | | min-width="220" /> |
| | | <el-table-column prop="rate" |
| | | label="税率(%)" |
| | | width="100" |
| | | align="center" |
| | | /> |
| | | <el-table-column |
| | | prop="quickDeduction" |
| | | align="center" /> |
| | | <el-table-column prop="quickDeduction" |
| | | label="速算扣除数/元" |
| | | width="160" |
| | | align="center" |
| | | /> |
| | | align="center" /> |
| | | </el-table> |
| | | </el-dialog> |
| | | </FormDialog> |
| | | </template> |
| | | |
| | | <script setup> |
| | | import { ref, reactive, toRefs, computed, getCurrentInstance, nextTick } from "vue"; |
| | | import { |
| | | ref, |
| | | reactive, |
| | | toRefs, |
| | | computed, |
| | | getCurrentInstance, |
| | | nextTick, |
| | | } from "vue"; |
| | | import { ArrowUp } from "@element-plus/icons-vue"; |
| | | import FormDialog from "@/components/Dialog/FormDialog.vue"; |
| | | import { listDept } from "@/api/system/dept.js"; |
| | |
| | | } from "@/api/personnelManagement/staffSalaryMain.js"; |
| | | import { userListNoPageByTenantId } from "@/api/system/user.js"; |
| | | |
| | | |
| | | const emit = defineEmits(["update:modelValue", "close"]); |
| | | const props = defineProps({ |
| | | modelValue: { type: Boolean, default: false }, |
| | |
| | | |
| | | const dialogVisible = computed({ |
| | | get: () => props.modelValue, |
| | | set: (val) => emit("update:modelValue", val), |
| | | set: val => emit("update:modelValue", val), |
| | | }); |
| | | |
| | | const formRef = ref(null); |
| | |
| | | auditUserId: undefined, |
| | | }, |
| | | rules: { |
| | | salaryTitle: [{ required: true, message: "请输入工资主题", trigger: "blur" }], |
| | | salaryTitle: [ |
| | | { required: true, message: "请输入工资主题", trigger: "blur" }, |
| | | ], |
| | | deptIds: [{ required: true, message: "请选择部门", trigger: "change" }], |
| | | salaryMonth: [{ required: true, message: "请选择工资月份", trigger: "change" }], |
| | | auditUserId: [{ required: true, message: "请选择审核人", trigger: "change" }], |
| | | salaryMonth: [ |
| | | { required: true, message: "请选择工资月份", trigger: "change" }, |
| | | ], |
| | | auditUserId: [ |
| | | { required: true, message: "请选择审核人", trigger: "change" }, |
| | | ], |
| | | }, |
| | | }); |
| | | const { form, rules } = toRefs(data); |
| | |
| | | }); |
| | | |
| | | const loadBankOptions = () => { |
| | | return bankList().then((res) => { |
| | | return bankList().then(res => { |
| | | const list = Array.isArray(res?.data) ? res.data : []; |
| | | bankOptions.value = list |
| | | .map((b) => (b?.bankName == null ? "" : String(b.bankName).trim())) |
| | | .filter((v) => v !== ""); |
| | | .map(b => (b?.bankName == null ? "" : String(b.bankName).trim())) |
| | | .filter(v => v !== ""); |
| | | }); |
| | | }; |
| | | |
| | | const loadUserList = () => { |
| | | return userListNoPageByTenantId().then((res) => { |
| | | return userListNoPageByTenantId().then(res => { |
| | | userList.value = res.data || []; |
| | | }); |
| | | }; |
| | |
| | | // 扁平化部门树供下拉使用 |
| | | function flattenDept(tree, list = []) { |
| | | if (!tree?.length) return list; |
| | | tree.forEach((node) => { |
| | | tree.forEach(node => { |
| | | list.push({ deptId: node.deptId, deptName: node.deptName }); |
| | | if (node.children?.length) flattenDept(node.children, list); |
| | | }); |
| | |
| | | } |
| | | |
| | | const loadDeptOptions = () => { |
| | | listDept().then((res) => { |
| | | listDept().then(res => { |
| | | const tree = res.data ?? []; |
| | | deptOptions.value = flattenDept(tree); |
| | | }); |
| | |
| | | const staffList = staffRes.data ?? []; |
| | | const deptMap = new Map(); |
| | | function walk(nodes) { |
| | | nodes.forEach((node) => { |
| | | nodes.forEach(node => { |
| | | deptMap.set(node.deptId, { |
| | | id: "dept_" + node.deptId, |
| | | deptId: node.deptId, |
| | |
| | | }); |
| | | } |
| | | walk(tree); |
| | | staffList.forEach((s) => { |
| | | staffList.forEach(s => { |
| | | const deptId = s.deptId ?? s.dept_id; |
| | | const node = deptMap.get(deptId); |
| | | if (node) { |
| | |
| | | } |
| | | }); |
| | | deptStaffTree.value = Array.from(deptMap.values()).filter( |
| | | (n) => n.children && n.children.length > 0 |
| | | n => n.children && n.children.length > 0 |
| | | ); |
| | | }); |
| | | }; |
| | |
| | | form.value.salaryTitle = row.salaryTitle ?? ""; |
| | | // deptIds 后端是字符串(多个用逗号分隔);当前表单仍是单选 deptId |
| | | form.value.deptIds = row.deptIds |
| | | ? String(row.deptIds).split(",").map((id) => Number(id.trim())).filter(Boolean) |
| | | ? String(row.deptIds) |
| | | .split(",") |
| | | .map(id => Number(id.trim())) |
| | | .filter(Boolean) |
| | | : []; |
| | | form.value.salaryMonth = row.salaryMonth ?? ""; |
| | | form.value.remark = row.remark ?? ""; |
| | |
| | | |
| | | // 如果有员工明细数据,直接反显 |
| | | if (row.staffSalaryDetailList && row.staffSalaryDetailList.length > 0) { |
| | | employeeList.value = row.staffSalaryDetailList.map((e) => ({ |
| | | employeeList.value = row.staffSalaryDetailList.map(e => ({ |
| | | staffOnJobId: e.staffOnJobId ?? e.staffId ?? e.id, |
| | | id: e.staffOnJobId ?? e.staffId ?? e.id, |
| | | staffName: e.staffName ?? "", |
| | |
| | | return; |
| | | } |
| | | const checked = tree.getCheckedNodes(); |
| | | const staffNodes = checked.filter((n) => n.type === "staff"); |
| | | const existIds = new Set(employeeList.value.map((e) => e.staffId || e.id)); |
| | | staffNodes.forEach((node) => { |
| | | const staffNodes = checked.filter(n => n.type === "staff"); |
| | | const existIds = new Set(employeeList.value.map(e => e.staffId || e.id)); |
| | | staffNodes.forEach(node => { |
| | | const id = node.staffId ?? node.id; |
| | | if (existIds.has(id)) return; |
| | | existIds.add(id); |
| | |
| | | addPersonVisible.value = false; |
| | | }; |
| | | |
| | | const removeEmployee = (row) => { |
| | | const removeEmployee = row => { |
| | | employeeList.value = employeeList.value.filter( |
| | | (e) => (e.staffOnJobId || e.id) !== (row.staffOnJobId || row.id) |
| | | e => (e.staffOnJobId || e.id) !== (row.staffOnJobId || row.id) |
| | | ); |
| | | }; |
| | | |
| | | const onEmployeeSelectionChange = (selection) => { |
| | | const onEmployeeSelectionChange = selection => { |
| | | selectedEmployees.value = selection; |
| | | }; |
| | | |
| | |
| | | proxy.$modal.msgWarning("请先勾选要删除的员工"); |
| | | return; |
| | | } |
| | | const ids = new Set(selectedEmployees.value.map((e) => e.staffOnJobId || e.id)); |
| | | const ids = new Set(selectedEmployees.value.map(e => e.staffOnJobId || e.id)); |
| | | employeeList.value = employeeList.value.filter( |
| | | (e) => !ids.has(e.staffOnJobId || e.id) |
| | | e => !ids.has(e.staffOnJobId || e.id) |
| | | ); |
| | | }; |
| | | |
| | |
| | | ids: form.value.deptIds, |
| | | date: form.value.salaryMonth, |
| | | }; |
| | | staffSalaryMainCalculateSalary(payload).then((res) => { |
| | | staffSalaryMainCalculateSalary(payload).then(res => { |
| | | const list = Array.isArray(res?.data) ? res.data : []; |
| | | if (!list.length) { |
| | | proxy.$modal.msgWarning("未计算到工资数据"); |
| | | return; |
| | | } |
| | | employeeList.value = list.map((e) => ({ |
| | | employeeList.value = list.map(e => ({ |
| | | ...e, |
| | | staffOnJobId: e.staffOnJobId ?? e.staffId ?? e.id, |
| | | staffName: e.staffName, |
| | |
| | | }; |
| | | |
| | | const handleClear = () => { |
| | | proxy.$modal.confirm("确定清空当前员工列表吗?").then(() => { |
| | | proxy.$modal |
| | | .confirm("确定清空当前员工列表吗?") |
| | | .then(() => { |
| | | employeeList.value = []; |
| | | }).catch(() => {}); |
| | | }) |
| | | .catch(() => {}); |
| | | }; |
| | | |
| | | const handleTaxForm = () => { |
| | |
| | | }; |
| | | |
| | | const submitForm = () => { |
| | | formRef.value?.validate((valid) => { |
| | | formRef.value?.validate(valid => { |
| | | if (!valid) return; |
| | | saveData(3); // 确认提交,状态为3(待审核) |
| | | }); |
| | | }; |
| | | |
| | | const saveDraft = () => { |
| | | formRef.value?.validate((valid) => { |
| | | formRef.value?.validate(valid => { |
| | | if (!valid) return; |
| | | saveData(1); // 保存草稿,状态为1(草稿) |
| | | }); |
| | | }; |
| | | |
| | | const saveData = (status) => { |
| | | const saveData = status => { |
| | | const payload = { |
| | | id: form.value.id, |
| | | salaryTitle: form.value.salaryTitle, |
| | |
| | | auditUserId: form.value.auditUserId, |
| | | auditUserName: auditUserName.value, |
| | | totalSalary: totalSalary.value, |
| | | staffSalaryDetailList: employeeList.value.map((e) => ({ |
| | | staffSalaryDetailList: employeeList.value.map(e => ({ |
| | | staffOnJobId: e.staffOnJobId ?? e.staffId ?? e.id, |
| | | staffName: e.staffName, |
| | | postName: e.postName ?? "", |