192 changed files with 43814 additions and 10497 deletions
@ -0,0 +1,196 @@
|
||||
import { getAction, deleteAction, putAction, postAction, httpAction } from '@/api/manage' |
||||
|
||||
//首页统计
|
||||
const getBuyAndSaleStatistics = (params)=>getAction("/erp/depotHead/getBuyAndSaleStatistics",params); |
||||
const buyOrSalePrice = (params)=>getAction("/erp/depotItem/buyOrSalePrice",params); |
||||
//租户管理
|
||||
const checkTenant = (params)=>getAction("/erp/tenant/checkIsNameExist",params); |
||||
const editTenant = (params)=>putAction("/erp/tenant/update",params); |
||||
//角色管理
|
||||
const addRole = (params)=>postAction("/erp/role/add",params); |
||||
const editRole = (params)=>putAction("/erp/role/update",params); |
||||
const checkRole = (params)=>getAction("/erp/role/checkIsNameExist",params); |
||||
const roleAllList = (params)=>getAction("/erp/role/allList",params); |
||||
//用户管理
|
||||
const registerUser = (params)=>postAction("/erp/user/registerUser",params); |
||||
const addUser = (params)=>postAction("/erp/user/addUser",params); |
||||
const editUser = (params)=>putAction("/erp/user/updateUser",params); |
||||
const getUserList = (params)=>getAction("/erp/user/getUserList",params); |
||||
const queryPermissionsByUser = (params)=>postAction("/erp/function/findMenuByPNumber",params); |
||||
//机构管理
|
||||
const queryOrganizationTreeList = (params)=>getAction("/erp/organization/getOrganizationTree",params); |
||||
const queryOrganizationById = (params)=>getAction("/erp/organization/findById",params); |
||||
const checkOrganization = (params)=>getAction("/erp/organization/checkIsNameExist",params); |
||||
//经手人管理
|
||||
const addPerson = (params)=>postAction("/erp/person/add",params); |
||||
const editPerson = (params)=>putAction("/erp/person/update",params); |
||||
const checkPerson = (params)=>getAction("/erp/person/checkIsNameExist",params); |
||||
const getPersonByType = (params)=>getAction("/erp/person/getPersonByType",params); |
||||
const getPersonByNumType = (params)=>getAction("/erp/person/getPersonByNumType",params); |
||||
//账户管理
|
||||
const addAccount = (params)=>postAction("/erp/account/add",params); |
||||
const editAccount = (params)=>putAction("/erp/account/update",params); |
||||
const checkAccount = (params)=>getAction("/erp/account/checkIsNameExist",params); |
||||
const getAccount = (params)=>getAction("/erp/account/getAccount",params); |
||||
//收支项目
|
||||
const addInOutItem = (params)=>postAction("/erp/inOutItem/add",params); |
||||
const editInOutItem = (params)=>putAction("/erp/inOutItem/update",params); |
||||
const checkInOutItem = (params)=>getAction("/erp/inOutItem/checkIsNameExist",params); |
||||
const findInOutItemByParam = (params)=>getAction("/erp/inOutItem/findBySelect",params); |
||||
//仓库信息
|
||||
const addDepot = (params)=>postAction("/erp/depot/add",params); |
||||
const editDepot = (params)=>putAction("/erp/depot/update",params); |
||||
const checkDepot = (params)=>getAction("/erp/depot/checkIsNameExist",params); |
||||
//商品属性
|
||||
const editMaterialProperty = (params)=>putAction("/erp/materialProperty/update",params); |
||||
//商品类型
|
||||
const queryMaterialCategoryTreeList = (params)=>getAction("/erp/materialCategory/getMaterialCategoryTree",params); |
||||
const queryMaterialCategoryById = (params)=>getAction("/erp/materialCategory/findById",params); |
||||
const checkMaterialCategory = (params)=>getAction("/erp/materialCategory/checkIsNameExist",params); |
||||
//商品管理
|
||||
const addMaterial = (params)=>postAction("/erp/material/add",params); |
||||
const editMaterial = (params)=>putAction("/erp/material/update",params); |
||||
const checkMaterial = (params)=>getAction("/erp/material/checkIsExist",params); |
||||
const getMaterialBySelect = (params)=>getAction("/erp/material/findBySelect",params); |
||||
const getSerialMaterialBySelect = (params)=>getAction("/erp/material/getMaterialEnableSerialNumberList",params); |
||||
const getMaterialByBarCode = (params)=>getAction("/erp/material/getMaterialByBarCode",params); |
||||
const getMaxBarCode = (params)=>getAction("/erp/material/getMaxBarCode",params); |
||||
const checkMaterialBarCode = (params)=>getAction("/erp/materialsExtend/checkIsBarCodeExist",params); |
||||
//序列号
|
||||
const addSerialNumber = (params)=>postAction("/erp/serialNumber/add",params); |
||||
const editSerialNumber = (params)=>putAction("/erp/serialNumber/update",params); |
||||
const checkSerialNumber = (params)=>getAction("/erp/serialNumber/checkIsNameExist",params); |
||||
const batAddSerialNumber = (params)=>postAction("/erp/serialNumber/batAddSerialNumber",params); |
||||
const getEnableSerialNumberList = (params)=>getAction("/erp/serialNumber/getEnableSerialNumberList",params); |
||||
//多属性
|
||||
const addMaterialAttribute = (params)=>postAction("/erp/materialAttribute/add",params); |
||||
const editMaterialAttribute = (params)=>putAction("/erp/materialAttribute/update",params); |
||||
const checkMaterialAttribute = (params)=>getAction("/erp/materialAttribute/checkIsNameExist",params); |
||||
const getAllMaterialAttribute = (params)=>getAction("/erp/materialAttribute/getAll",params); |
||||
//功能管理
|
||||
const addFunction = (params)=>postAction("/erp/function/add",params); |
||||
const editFunction = (params)=>putAction("/erp/function/update",params); |
||||
const checkFunction = (params)=>getAction("/erp/function/checkIsNameExist",params); |
||||
//系统配置
|
||||
const addSystemConfig = (params)=>postAction("/erp/systemConfig/add",params); |
||||
const editSystemConfig = (params)=>putAction("/erp/systemConfig/update",params); |
||||
const checkSystemConfig = (params)=>getAction("/erp/systemConfig/checkIsNameExist",params); |
||||
const getCurrentSystemConfig = (params)=>getAction("/erp/systemConfig/getCurrentInfo",params); |
||||
const fileSizeLimit = (params)=>getAction("/erp/systemConfig/fileSizeLimit",params); |
||||
//平台参数
|
||||
const addPlatformConfig = (params)=>postAction("/erp/platformConfig/add",params); |
||||
const editPlatformConfig = (params)=>putAction("/erp/platformConfig/update",params); |
||||
const getPlatformConfigByKey = (params)=>getAction("/erp/platformConfig/getPlatformConfigByKey",params); |
||||
//用户|角色|模块关系
|
||||
const addUserBusiness = (params)=>postAction("/erp/userBusiness/add",params); |
||||
const editUserBusiness = (params)=>putAction("/erp/userBusiness/update",params); |
||||
const checkUserBusiness = (params)=>getAction("/erp/userBusiness/checkIsValueExist",params); |
||||
const updateBtnStrByRoleId = (params)=>postAction("/erp/userBusiness/updateBtnStr",params); |
||||
//计量单位
|
||||
const addUnit = (params)=>postAction("/erp/unit/add",params); |
||||
const editUnit = (params)=>putAction("/erp/unit/update",params); |
||||
const checkUnit = (params)=>getAction("/erp/unit/checkIsNameExist",params); |
||||
//供应商|客户|会员
|
||||
const addSupplier = (params)=>postAction("/erp/supplier/add",params); |
||||
const editSupplier = (params)=>putAction("/erp/supplier/update",params); |
||||
const checkSupplier = (params)=>getAction("/erp/supplier/checkIsNameAndTypeExist",params); |
||||
const findBySelectSup = (params)=>postAction("/erp/supplier/findBySelect_sup",params); |
||||
const findBySelectCus = (params)=>postAction("/erp/supplier/findBySelect_cus",params); |
||||
const findBySelectRetail = (params)=>postAction("/erp/supplier/findBySelect_retail",params); |
||||
const findBySelectOrgan = (params)=>postAction("/erp/supplier/findBySelect_organ",params); |
||||
//单据相关
|
||||
const findBillDetailByNumber = (params)=>getAction("/erp/depotHead/getDetailByNumber",params); |
||||
const findStockByDepotAndBarCode = (params)=>getAction("/erp/depotItem/findStockByDepotAndBarCode",params); |
||||
const getBatchNumberList = (params)=>getAction("/erp/depotItem/getBatchNumberList",params); |
||||
const findFinancialDetailByNumber = (params)=>getAction("/erp/accountHead/getDetailByNumber",params); |
||||
|
||||
//租户信息
|
||||
const getAlltenantInfo = (params)=>getAction("/erp/tenant/listAllTenant",params); |
||||
export { |
||||
getBuyAndSaleStatistics, |
||||
buyOrSalePrice, |
||||
checkTenant, |
||||
editTenant, |
||||
addRole, |
||||
editRole, |
||||
checkRole, |
||||
roleAllList, |
||||
registerUser, |
||||
addUser, |
||||
editUser, |
||||
getUserList, |
||||
queryPermissionsByUser, |
||||
queryOrganizationTreeList, |
||||
queryOrganizationById, |
||||
checkOrganization, |
||||
addPerson, |
||||
editPerson, |
||||
checkPerson, |
||||
getPersonByType, |
||||
getPersonByNumType, |
||||
addAccount, |
||||
editAccount, |
||||
checkAccount, |
||||
getAccount, |
||||
addInOutItem, |
||||
editInOutItem, |
||||
checkInOutItem, |
||||
findInOutItemByParam, |
||||
addDepot, |
||||
editDepot, |
||||
checkDepot, |
||||
editMaterialProperty, |
||||
queryMaterialCategoryTreeList, |
||||
queryMaterialCategoryById, |
||||
checkMaterialCategory, |
||||
addMaterial, |
||||
editMaterial, |
||||
checkMaterial, |
||||
getMaterialBySelect, |
||||
getSerialMaterialBySelect, |
||||
getMaterialByBarCode, |
||||
getMaxBarCode, |
||||
checkMaterialBarCode, |
||||
addSerialNumber, |
||||
editSerialNumber, |
||||
checkSerialNumber, |
||||
batAddSerialNumber, |
||||
getEnableSerialNumberList, |
||||
addMaterialAttribute, |
||||
editMaterialAttribute, |
||||
checkMaterialAttribute, |
||||
getAllMaterialAttribute, |
||||
addFunction, |
||||
editFunction, |
||||
checkFunction, |
||||
addSystemConfig, |
||||
editSystemConfig, |
||||
checkSystemConfig, |
||||
getCurrentSystemConfig, |
||||
fileSizeLimit, |
||||
addPlatformConfig, |
||||
editPlatformConfig, |
||||
getPlatformConfigByKey, |
||||
addUserBusiness, |
||||
editUserBusiness, |
||||
checkUserBusiness, |
||||
updateBtnStrByRoleId, |
||||
addUnit, |
||||
editUnit, |
||||
checkUnit, |
||||
addSupplier, |
||||
editSupplier, |
||||
checkSupplier, |
||||
findBySelectSup, |
||||
findBySelectCus, |
||||
findBySelectRetail, |
||||
findBySelectOrgan, |
||||
findBillDetailByNumber, |
||||
findStockByDepotAndBarCode, |
||||
getBatchNumberList, |
||||
findFinancialDetailByNumber, |
||||
getAlltenantInfo |
||||
} |
||||
|
||||
|
||||
|
@ -0,0 +1,273 @@
|
||||
.area-zoom-in-top-enter-active, |
||||
.area-zoom-in-top-leave-active { |
||||
opacity: 1; |
||||
transform: scaleY(1); |
||||
} |
||||
|
||||
.area-zoom-in-top-enter, |
||||
.area-zoom-in-top-leave-active { |
||||
opacity: 0; |
||||
transform: scaleY(0); |
||||
} |
||||
|
||||
.area-select { |
||||
box-sizing: border-box; |
||||
margin: 0; |
||||
padding: 0; |
||||
color: rgba(0, 0, 0, 0.65); |
||||
font-size: 14px; |
||||
font-variant: tabular-nums; |
||||
line-height: 1.5; |
||||
list-style: none; |
||||
font-feature-settings: 'tnum'; |
||||
position: relative; |
||||
outline: 0; |
||||
display: block; |
||||
background-color: #fff; |
||||
border: 1px solid #d9d9d9; |
||||
border-top-width: 1.02px; |
||||
border-radius: 4px; |
||||
outline: none; |
||||
transition: all 0.3s cubic-bezier(0.645, 0.045, 0.355, 1); |
||||
-webkit-user-select: none; |
||||
-ms-user-select: none; |
||||
user-select: none; |
||||
} |
||||
|
||||
.area-select-wrap .area-select { |
||||
display: inline-block; |
||||
} |
||||
|
||||
.area-select * { |
||||
box-sizing: border-box; |
||||
} |
||||
|
||||
.area-select:hover { |
||||
border-color: #40a9ff; |
||||
border-right-width: 1px !important; |
||||
outline: 0; |
||||
} |
||||
|
||||
|
||||
.area-select:active { |
||||
box-shadow: 0 0 0 2px rgba(24, 144, 255, 0.2); |
||||
} |
||||
|
||||
.area-select.small { |
||||
width: 126px; |
||||
} |
||||
|
||||
.area-select.medium { |
||||
width: 160px; |
||||
} |
||||
|
||||
.area-select.large { |
||||
width: 194px; |
||||
} |
||||
|
||||
.area-select.is-disabled { |
||||
background: #eceff5; |
||||
cursor: not-allowed; |
||||
} |
||||
|
||||
.area-select.is-disabled:hover { |
||||
border-color: #e1e2e6; |
||||
} |
||||
|
||||
.area-select.is-disabled .area-selected-trigger { |
||||
cursor: not-allowed; |
||||
} |
||||
|
||||
.area-select .area-selected-trigger { |
||||
position: relative; |
||||
display: block; |
||||
font-size: 14px; |
||||
cursor: pointer; |
||||
margin: 0; |
||||
overflow: hidden; |
||||
white-space: nowrap; |
||||
text-overflow: ellipsis; |
||||
height: 100%; |
||||
padding: 8px 20px 7px 12px; |
||||
} |
||||
|
||||
.area-select .area-select-icon { |
||||
position: absolute; |
||||
top: 50%; |
||||
margin-top: -2px; |
||||
right: 6px; |
||||
content: ""; |
||||
width: 0; |
||||
height: 0; |
||||
border: 6px solid transparent; |
||||
border-top-color: rgba(0, 0, 0, 0.25); |
||||
transition: all .3s linear; |
||||
transform-origin: center; |
||||
} |
||||
|
||||
.area-select .area-select-icon.active { |
||||
margin-top: -8px; |
||||
transform: rotate(180deg); |
||||
} |
||||
|
||||
.area-selectable-list-wrap { |
||||
position: absolute; |
||||
width: 100%; |
||||
max-height: 275px; |
||||
z-index: 15000; |
||||
background-color: #fff; |
||||
box-sizing: border-box; |
||||
overflow-x: auto; |
||||
margin: 2px 0; |
||||
border-radius: 4px; |
||||
outline: none; |
||||
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.15); |
||||
|
||||
transition: opacity 0.15s, transform 0.3s !important; |
||||
transform-origin: center top !important; |
||||
} |
||||
|
||||
.area-selectable-list { |
||||
position: relative; |
||||
margin: 0; |
||||
padding: 6px 0; |
||||
width: 100%; |
||||
font-size: 14px; |
||||
color: #565656; |
||||
list-style: none; |
||||
} |
||||
|
||||
.area-selectable-list .area-select-option { |
||||
position: relative; |
||||
white-space: nowrap; |
||||
overflow: hidden; |
||||
text-overflow: ellipsis; |
||||
cursor: pointer; |
||||
padding: 0 15px 0 10px; |
||||
height: 32px; |
||||
line-height: 32px; |
||||
} |
||||
|
||||
.area-selectable-list .area-select-option.hover { |
||||
background-color: #e6f7ff; |
||||
} |
||||
|
||||
.area-selectable-list .area-select-option.selected { |
||||
color: rgba(0, 0, 0, 0.65); |
||||
font-weight: 600; |
||||
background-color: #efefef; |
||||
} |
||||
|
||||
.cascader-menu-list-wrap { |
||||
position: absolute; |
||||
white-space: nowrap; |
||||
z-index: 15000; |
||||
background-color: #fff; |
||||
box-sizing: border-box; |
||||
overflow: hidden; |
||||
font-size: 0; |
||||
margin: 2px 0; |
||||
border-radius: 4px; |
||||
outline: none; |
||||
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.15); |
||||
|
||||
transition: opacity 0.15s, transform 0.3s !important; |
||||
transform-origin: center top !important; |
||||
} |
||||
|
||||
.cascader-menu-list { |
||||
position: relative; |
||||
margin: 0; |
||||
font-size: 14px; |
||||
color: #565656; |
||||
padding: 6px 0; |
||||
list-style: none; |
||||
display: inline-block; |
||||
height: 204px; |
||||
overflow-x: hidden; |
||||
overflow-y: auto; |
||||
min-width: 160px; |
||||
vertical-align: top; |
||||
background-color: #fff; |
||||
border-right: 1px solid #e4e7ed; |
||||
} |
||||
|
||||
.cascader-menu-list:last-child { |
||||
border-right: none; |
||||
} |
||||
|
||||
.cascader-menu-list .cascader-menu-option { |
||||
position: relative; |
||||
white-space: nowrap; |
||||
overflow: hidden; |
||||
text-overflow: ellipsis; |
||||
cursor: pointer; |
||||
padding: 0 15px 0 10px; |
||||
height: 32px; |
||||
line-height: 32px; |
||||
} |
||||
|
||||
.cascader-menu-list .cascader-menu-option.hover, |
||||
.cascader-menu-list .cascader-menu-option:hover { |
||||
background-color: #e6f7ff; |
||||
} |
||||
|
||||
.cascader-menu-list .cascader-menu-option.selected { |
||||
color: rgba(0, 0, 0, 0.65); |
||||
font-weight: 600; |
||||
background-color: #efefef; |
||||
} |
||||
|
||||
.cascader-menu-list .cascader-menu-option.cascader-menu-extensible:after { |
||||
position: absolute; |
||||
top: 50%; |
||||
margin-top: -4px; |
||||
right: 5px; |
||||
content: ""; |
||||
width: 0; |
||||
height: 0; |
||||
border: 4px solid transparent; |
||||
border-left-color: #a1a4ad; |
||||
} |
||||
|
||||
.cascader-menu-list::-webkit-scrollbar, |
||||
.area-selectable-list-wrap::-webkit-scrollbar { |
||||
width: 8px; |
||||
background: transparent; |
||||
} |
||||
|
||||
.area-selectable-list-wrap::-webkit-scrollbar-button:vertical:decremen, |
||||
.area-selectable-list-wrap::-webkit-scrollbar-button:vertical:end:decrement, |
||||
.area-selectable-list-wrap::-webkit-scrollbar-button:vertical:increment, |
||||
.area-selectable-list-wrap::-webkit-scrollbar-button:vertical:start:increment, |
||||
.cascader-menu-list::-webkit-scrollbar-button:vertical:decremen, |
||||
.cascader-menu-list::-webkit-scrollbar-button:vertical:end:decrement, |
||||
.cascader-menu-list::-webkit-scrollbar-button:vertical:increment, |
||||
.cascader-menu-list::-webkit-scrollbar-button:vertical:start:increment { |
||||
display: none; |
||||
} |
||||
|
||||
.cascader-menu-list::-webkit-scrollbar-thumb:vertical, |
||||
.area-selectable-list-wrap::-webkit-scrollbar-thumb:vertical { |
||||
background-color: #b8b8b8; |
||||
border-radius: 4px; |
||||
} |
||||
|
||||
.cascader-menu-list::-webkit-scrollbar-thumb:vertical:hover, |
||||
.area-selectable-list-wrap::-webkit-scrollbar-thumb:vertical:hover { |
||||
background-color: #777; |
||||
} |
||||
|
||||
.resize-table-th { |
||||
position: relative; |
||||
.table-draggable-handle { |
||||
transform: none !important; |
||||
position: absolute; |
||||
height: 100% !important; |
||||
bottom: 0; |
||||
left: auto !important; |
||||
right: -5px; |
||||
cursor: col-resize; |
||||
touch-action: none; |
||||
} |
||||
} |
@ -0,0 +1,15 @@
|
||||
/** [表格主题样式一] 表格强制列不换行 */ |
||||
.j-table-force-nowrap { |
||||
td, th { |
||||
white-space: nowrap; |
||||
} |
||||
|
||||
.ant-table-selection-column { |
||||
padding: 12px 22px !important; |
||||
} |
||||
|
||||
/** 列自适应,弊端会导致列宽失效 */ |
||||
&.ant-table-wrapper .ant-table-content { |
||||
overflow-x: auto; |
||||
} |
||||
} |
@ -0,0 +1,74 @@
|
||||
/*列表上方操作按钮区域*/ |
||||
.ant-card-body .table-operator { |
||||
margin-bottom: 0px; |
||||
} |
||||
/** Button按钮间距 */ |
||||
.table-operator .ant-btn { |
||||
margin: 0 8px 8px 0; |
||||
} |
||||
.table-operator .ant-btn-group .ant-btn { |
||||
margin: 0; |
||||
} |
||||
.table-operator .ant-btn-group .ant-btn:last-child { |
||||
margin: 0 8px 8px 0; |
||||
} |
||||
/*列表td的padding设置 可以控制列表大小*/ |
||||
.ant-table-tbody .ant-table-row td { |
||||
padding-top: 15px; |
||||
padding-bottom: 15px; |
||||
} |
||||
/*列表页面弹出modal*/ |
||||
.ant-modal-cust-warp { |
||||
height: 100% |
||||
} |
||||
/*弹出modal Y轴滚动条*/ |
||||
.ant-modal-cust-warp .ant-modal-body { |
||||
padding: 24px 24px 12px 24px; |
||||
height: calc(100% - 110px) !important; |
||||
overflow-y: auto |
||||
} |
||||
/*弹出modal 先有content后有body 故滚动条控制在body上*/ |
||||
.ant-modal-cust-warp .ant-modal-content { |
||||
height: 90%; |
||||
overflow-y: hidden |
||||
} |
||||
/*文本框样式*/ |
||||
.ant-modal-cust-warp .ant-form-item { |
||||
margin-bottom: 12px; |
||||
} |
||||
/*商品-列表页面弹出modal*/ |
||||
.ant-modal-material-warp { |
||||
height: 100% |
||||
} |
||||
/*商品-弹出modal Y轴滚动条*/ |
||||
.ant-modal-material-warp .ant-modal-body { |
||||
padding: 12px 24px 12px 24px; |
||||
height: calc(100% - 110px) !important; |
||||
overflow-y: auto |
||||
} |
||||
/*商品-弹出modal 先有content后有body 故滚动条控制在body上*/ |
||||
.ant-modal-material-warp .ant-modal-content { |
||||
height: 90%; |
||||
overflow-y: hidden |
||||
} |
||||
/*商品-文本框样式*/ |
||||
.ant-modal-material-warp .ant-form-item { |
||||
margin-bottom: 12px; |
||||
} |
||||
/*列表中有图片的加这个样式 参考用户管理*/ |
||||
.anty-img-wrap { |
||||
height: 25px; |
||||
position: relative; |
||||
} |
||||
.anty-img-wrap > img { |
||||
max-height: 100%; |
||||
} |
||||
/*列表中范围查询样式*/ |
||||
.query-group-cust{width: calc(50% - 10px)} |
||||
.query-group-split-cust:before{content:"~";width: 20px;display: inline-block;text-align: center} |
||||
/*erp风格子表外框padding设置*/ |
||||
.ant-card-wider-padding.cust-erp-sub-tab>.ant-card-body{padding:5px 12px} |
||||
/* 内嵌子表背景颜色 */ |
||||
.j-inner-table-wrapper /deep/ .ant-table-expanded-row .ant-table-wrapper .ant-table-tbody .ant-table-row { |
||||
background-color: #FFFFFF; |
||||
} |
@ -0,0 +1,29 @@
|
||||
/** |
||||
* 列表查询通用样式,移动端自适应 |
||||
*/ |
||||
.search{ |
||||
margin-bottom: 54px; |
||||
} |
||||
.fold{ |
||||
width: calc(100% - 216px); |
||||
display: inline-block |
||||
} |
||||
.operator{ |
||||
margin-bottom: 18px; |
||||
} |
||||
@media screen and (max-width: 900px) { |
||||
.fold { |
||||
width: 100%; |
||||
} |
||||
} |
||||
.operator button { |
||||
margin-right: 5px; |
||||
} |
||||
i { |
||||
cursor: pointer; |
||||
} |
||||
.trcolor{ |
||||
background-color: rgba(255, 192, 203, 0.31); |
||||
color:red; |
||||
} |
||||
|
@ -0,0 +1,36 @@
|
||||
<script> |
||||
export default { |
||||
name: 'Ellipsis', |
||||
props: { |
||||
prefixCls: { |
||||
type: String, |
||||
default: 'ant-pro-ellipsis' |
||||
}, |
||||
tooltip: { |
||||
type: Boolean, |
||||
default: true, |
||||
}, |
||||
length: { |
||||
type: Number, |
||||
default: 25, |
||||
}, |
||||
lines: { |
||||
type: Number, |
||||
default: 1 |
||||
}, |
||||
fullWidthRecognition: { |
||||
type: Boolean, |
||||
default: false |
||||
} |
||||
}, |
||||
methods: {}, |
||||
render() { |
||||
const { tooltip, length } = this.$props |
||||
let text = '' |
||||
// 处理没有default插槽时的特殊情况 |
||||
if (this.$slots.default) { |
||||
text = this.$slots.default.map(vNode => vNode.text).join('') |
||||
} |
||||
} |
||||
} |
||||
</script> |
@ -0,0 +1,3 @@
|
||||
import Ellipsis from './Ellipsis' |
||||
|
||||
export default Ellipsis |
@ -0,0 +1,135 @@
|
||||
<template> |
||||
<a-popover title="自定义列" trigger="click" placement="leftBottom"> |
||||
<template slot="content"> |
||||
<a-checkbox-group @change="onColSettingsChange" v-model="settingColumns" |
||||
:defaultValue="settingColumns" style="width: 400px;" > |
||||
<a-row> |
||||
<template v-for="(item,index) in defColumns_"> |
||||
<template v-if="!ignoreColumns.includes(item.key) && !ignoreColumns.includes(item.dataIndex)"> |
||||
<a-col :span="12"><a-checkbox :value="item.dataIndex" >{{ item.title }}</a-checkbox></a-col> |
||||
</template> |
||||
</template> |
||||
</a-row> |
||||
</a-checkbox-group> |
||||
</template> |
||||
<a-button type="link" icon="setting">自定义列</a-button> |
||||
</a-popover> |
||||
</template> |
||||
|
||||
<script> |
||||
import Vue from "vue"; |
||||
|
||||
export default { |
||||
name: "ListColumnsSetter", |
||||
|
||||
props: { |
||||
columns:{ |
||||
type:Array, |
||||
required: true |
||||
}, |
||||
defColumns:{ |
||||
type:Array, |
||||
required: true |
||||
}, |
||||
ignoreColumns:{ |
||||
type: Array, |
||||
default: () => ['rowIndex', 'action', 'flag'] |
||||
}, |
||||
listName:{ |
||||
type:String, |
||||
default: '', |
||||
} |
||||
}, |
||||
|
||||
model: { |
||||
prop: 'columns', |
||||
event: 'change' |
||||
}, |
||||
|
||||
data() { |
||||
return { |
||||
defColumns_: [], |
||||
settingColumns:[] |
||||
} |
||||
}, |
||||
|
||||
computed: { |
||||
ls_key() { return this.$route.name + '/' + this.listName + ':colsettings'}, |
||||
}, |
||||
|
||||
mounted() { |
||||
this.defColumns_.push(...this.defColumns); |
||||
this.initColumns(); |
||||
}, |
||||
|
||||
methods: { |
||||
initColumns(){ |
||||
//权限过滤(列权限控制时打开,修改第二个参数为授权码前缀) |
||||
//this.defColumns_ = colAuthFilter(this.defColumns_,'testdemo:'); |
||||
|
||||
console.log(this.ls_key) |
||||
let colSettings = Vue.ls.get(this.ls_key); |
||||
let columns = []; |
||||
if(colSettings===null||colSettings===undefined){ |
||||
let allSettingColumns = []; |
||||
this.defColumns_.forEach(function (item,i,array ) { |
||||
allSettingColumns.push(item.dataIndex); |
||||
}) |
||||
this.settingColumns = allSettingColumns; |
||||
columns = this.defColumns_; |
||||
} else { |
||||
this.settingColumns = colSettings; |
||||
//默认隐藏列 |
||||
// this.settingColumns.forEach((item,index) => { |
||||
// if(item == "model") { |
||||
// this.settingColumns.splice(index,1) |
||||
// } |
||||
// if(item == "color") { |
||||
// this.settingColumns.splice(index,1) |
||||
// } |
||||
// if(item == "categoryName") { |
||||
// this.settingColumns.splice(index,1) |
||||
// } |
||||
// if(item == "materialOther") { |
||||
// this.settingColumns.splice(index,1) |
||||
// } |
||||
// if(item == "commodityDecimal") { |
||||
// this.settingColumns.splice(index,1) |
||||
// } |
||||
// if(item == "lowDecimal") { |
||||
// this.settingColumns.splice(index,1) |
||||
// } |
||||
// }) |
||||
columns = this.defColumns_.filter(item => { |
||||
if(this.ignoreColumns.includes(item.key) || this.ignoreColumns.includes(item.dataIndex)){ |
||||
return true; |
||||
} |
||||
if (colSettings.includes(item.dataIndex)) { |
||||
return true; |
||||
} |
||||
return false; |
||||
}) |
||||
} |
||||
this.$emit('change', columns); |
||||
}, |
||||
|
||||
//列设置更改事件 |
||||
onColSettingsChange (checkedValues) { |
||||
console.log("checkedValues",checkedValues); |
||||
Vue.ls.set(this.ls_key, checkedValues, 7*24*60*60*1000) |
||||
this.settingColumns = checkedValues; |
||||
const columns = this.defColumns_.filter(item => { |
||||
if(this.ignoreColumns.includes(item.key) || this.ignoreColumns.includes(item.dataIndex)){ |
||||
return true |
||||
} |
||||
if (this.settingColumns.includes(item.dataIndex)) { |
||||
return true |
||||
} |
||||
return false |
||||
}) |
||||
this.$emit('change', columns); |
||||
}, |
||||
}, |
||||
|
||||
} |
||||
</script> |
@ -0,0 +1,155 @@
|
||||
<template> |
||||
<div v-if="!reloading" class="j-area-linkage"> |
||||
<area-cascader |
||||
v-if="_type === enums.type[0]" |
||||
:value="innerValue" |
||||
:data="pcaa" |
||||
:level="1" |
||||
:style="{width}" |
||||
v-bind="$attrs" |
||||
v-on="_listeners" |
||||
@change="handleChange" |
||||
/> |
||||
<area-select |
||||
v-else-if="_type === enums.type[1]" |
||||
:value="innerValue" |
||||
:data="pcaa" |
||||
:level="2" |
||||
v-bind="$attrs" |
||||
v-on="_listeners" |
||||
@change="handleChange" |
||||
/> |
||||
<div v-else> |
||||
<span style="color:red;"> Bad type value: {{_type}}</span> |
||||
</div> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
import { pcaa } from 'area-data' |
||||
|
||||
export default { |
||||
name: 'JAreaLinkage', |
||||
props: { |
||||
value: { |
||||
type: String, |
||||
required:false |
||||
}, |
||||
// 组件的类型,可选值: |
||||
// select 下拉样式 |
||||
// cascader 级联样式(默认) |
||||
type: { |
||||
type: String, |
||||
default: 'cascader' |
||||
}, |
||||
width: { |
||||
type: String, |
||||
default: '100%' |
||||
} |
||||
}, |
||||
data() { |
||||
return { |
||||
pcaa, |
||||
innerValue: [], |
||||
usedListeners: ['change'], |
||||
enums: { |
||||
type: ['cascader', 'select'] |
||||
}, |
||||
reloading: false, |
||||
areaData:'' |
||||
} |
||||
}, |
||||
computed: { |
||||
_listeners() { |
||||
let listeners = { ...this.$listeners } |
||||
// 去掉已使用的事件,防止冲突 |
||||
this.usedListeners.forEach(key => { |
||||
delete listeners[key] |
||||
}) |
||||
return listeners |
||||
}, |
||||
_type() { |
||||
if (this.enums.type.includes(this.type)) { |
||||
return this.type |
||||
} else { |
||||
console.error(`JAreaLinkage的type属性只能接收指定的值(${this.enums.type.join('|')})`) |
||||
return this.enums.type[0] |
||||
} |
||||
}, |
||||
}, |
||||
watch: { |
||||
value: { |
||||
immediate: true, |
||||
handler() { |
||||
this.loadDataByValue(this.value) |
||||
} |
||||
}, |
||||
}, |
||||
created() { |
||||
this.initAreaData(); |
||||
}, |
||||
methods: { |
||||
/** 通过 value 反推 options */ |
||||
loadDataByValue(value) { |
||||
if(!value || value.length==0){ |
||||
this.innerValue = [] |
||||
this.reloading = true; |
||||
setTimeout(()=>{ |
||||
this.reloading = false |
||||
},100) |
||||
}else{ |
||||
this.initAreaData(); |
||||
let arr = this.areaData.getRealCode(value); |
||||
this.innerValue = arr |
||||
} |
||||
}, |
||||
/** 通过地区code获取子级 */ |
||||
loadDataByCode(value) { |
||||
let options = [] |
||||
let data = pcaa[value] |
||||
if (data) { |
||||
for (let key in data) { |
||||
if (data.hasOwnProperty(key)) { |
||||
options.push({ value: key, label: data[key], }) |
||||
} |
||||
} |
||||
return options |
||||
} else { |
||||
return [] |
||||
} |
||||
}, |
||||
/** 判断是否有子节点 */ |
||||
hasChildren(options) { |
||||
options.forEach(option => { |
||||
let data = this.loadDataByCode(option.value) |
||||
option.isLeaf = data.length === 0 |
||||
}) |
||||
}, |
||||
handleChange(values) { |
||||
let value = values[values.length - 1] |
||||
this.$emit('change', value) |
||||
}, |
||||
initAreaData(){ |
||||
if(!this.areaData){ |
||||
this.areaData = new Area(); |
||||
} |
||||
}, |
||||
|
||||
}, |
||||
model: { prop: 'value', event: 'change' }, |
||||
} |
||||
</script> |
||||
|
||||
<style lang="less" scoped> |
||||
.j-area-linkage { |
||||
height:40px; |
||||
/deep/ .area-cascader-wrap .area-select { |
||||
width: 100%; |
||||
} |
||||
|
||||
/deep/ .area-select .area-selected-trigger { |
||||
line-height: 1.15; |
||||
} |
||||
} |
||||
|
||||
</style> |
@ -0,0 +1,238 @@
|
||||
<template> |
||||
<a-tree-select |
||||
allowClear |
||||
labelInValue |
||||
style="width: 100%" |
||||
:disabled="disabled" |
||||
:dropdownStyle="{ maxHeight: '400px', overflow: 'auto' }" |
||||
:placeholder="placeholder" |
||||
:loadData="asyncLoadTreeData" |
||||
:value="treeValue" |
||||
:treeData="treeData" |
||||
:multiple="multiple" |
||||
@change="onChange"> |
||||
</a-tree-select> |
||||
</template> |
||||
<script> |
||||
|
||||
import { getAction } from '@/api/manage' |
||||
|
||||
export default { |
||||
name: 'JCategorySelect', |
||||
props: { |
||||
value:{ |
||||
type: String, |
||||
required: false |
||||
}, |
||||
placeholder:{ |
||||
type: String, |
||||
default: '请选择', |
||||
required: false |
||||
}, |
||||
disabled:{ |
||||
type:Boolean, |
||||
default:false, |
||||
required:false |
||||
}, |
||||
condition:{ |
||||
type:String, |
||||
default:'', |
||||
required:false |
||||
}, |
||||
// 是否支持多选 |
||||
multiple: { |
||||
type: Boolean, |
||||
default: false, |
||||
}, |
||||
loadTriggleChange:{ |
||||
type: Boolean, |
||||
default: false, |
||||
required:false |
||||
}, |
||||
pid:{ |
||||
type:String, |
||||
default:'', |
||||
required:false |
||||
}, |
||||
pcode:{ |
||||
type:String, |
||||
default:'', |
||||
required:false |
||||
}, |
||||
back:{ |
||||
type:String, |
||||
default:'', |
||||
required:false |
||||
} |
||||
}, |
||||
data () { |
||||
return { |
||||
treeValue:"", |
||||
treeData:[], |
||||
url:"/sys/category/loadTreeData", |
||||
view:'/sys/category/loadDictItem/', |
||||
tableName:"", |
||||
text:"", |
||||
code:"", |
||||
|
||||
} |
||||
}, |
||||
watch: { |
||||
value () { |
||||
this.loadItemByCode() |
||||
}, |
||||
pcode(){ |
||||
this.loadRoot(); |
||||
} |
||||
}, |
||||
created(){ |
||||
this.validateProp().then(()=>{ |
||||
this.loadRoot() |
||||
this.loadItemByCode() |
||||
}) |
||||
}, |
||||
methods: { |
||||
/**加载一级节点 */ |
||||
loadRoot(){ |
||||
let param = { |
||||
pid:this.pid, |
||||
pcode:!this.pcode?'0':this.pcode, |
||||
condition:this.condition |
||||
} |
||||
getAction(this.url,param).then(res=>{ |
||||
if(res.success && res.result){ |
||||
for(let i of res.result){ |
||||
i.value = i.key |
||||
if(i.leaf==false){ |
||||
i.isLeaf=false |
||||
}else if(i.leaf==true){ |
||||
i.isLeaf=true |
||||
} |
||||
} |
||||
this.treeData = [...res.result] |
||||
}else{ |
||||
console.log("树一级节点查询结果-else",res) |
||||
} |
||||
}) |
||||
}, |
||||
|
||||
/** 数据回显*/ |
||||
loadItemByCode(){ |
||||
if(!this.value || this.value=="0"){ |
||||
this.treeValue = [] |
||||
}else{ |
||||
getAction(this.view,{ids:this.value}).then(res=>{ |
||||
if(res.success){ |
||||
let values = this.value.split(',') |
||||
this.treeValue = res.result.map((item, index) => ({ |
||||
key: values[index], |
||||
value: values[index], |
||||
label: item |
||||
})) |
||||
this.onLoadTriggleChange(res.result[0]); |
||||
} |
||||
}) |
||||
} |
||||
}, |
||||
onLoadTriggleChange(text){ |
||||
//只有单选才会触发 |
||||
if(!this.multiple && this.loadTriggleChange){ |
||||
this.backValue(this.value,text) |
||||
} |
||||
}, |
||||
backValue(value,label){ |
||||
let obj = {} |
||||
if(this.back){ |
||||
obj[this.back] = label |
||||
} |
||||
this.$emit('change', value, obj) |
||||
}, |
||||
asyncLoadTreeData (treeNode) { |
||||
return new Promise((resolve) => { |
||||
if (treeNode.$vnode.children) { |
||||
resolve() |
||||
return |
||||
} |
||||
let pid = treeNode.$vnode.key |
||||
let param = { |
||||
pid:pid, |
||||
condition:this.condition |
||||
} |
||||
getAction(this.url,param).then(res=>{ |
||||
if(res.success){ |
||||
for(let i of res.result){ |
||||
i.value = i.key |
||||
if(i.leaf==false){ |
||||
i.isLeaf=false |
||||
}else if(i.leaf==true){ |
||||
i.isLeaf=true |
||||
} |
||||
} |
||||
this.addChildren(pid,res.result,this.treeData) |
||||
this.treeData = [...this.treeData] |
||||
} |
||||
resolve() |
||||
}) |
||||
}) |
||||
}, |
||||
addChildren(pid,children,treeArray){ |
||||
if(treeArray && treeArray.length>0){ |
||||
for(let item of treeArray){ |
||||
if(item.key == pid){ |
||||
if(!children || children.length==0){ |
||||
item.isLeaf=true |
||||
}else{ |
||||
item.children = children |
||||
} |
||||
break |
||||
}else{ |
||||
this.addChildren(pid,children,item.children) |
||||
} |
||||
} |
||||
} |
||||
}, |
||||
|
||||
onChange(value){ |
||||
if(!value){ |
||||
this.$emit('change', ''); |
||||
this.treeValue = '' |
||||
} else if (value instanceof Array) { |
||||
//this.$emit('change', value.map(item => item.value).join(',')) |
||||
//this.treeValue = value |
||||
} else { |
||||
this.backValue(value.value,value.label) |
||||
this.treeValue = value |
||||
} |
||||
}, |
||||
getCurrTreeData(){ |
||||
return this.treeData |
||||
}, |
||||
validateProp(){ |
||||
let mycondition = this.condition |
||||
return new Promise((resolve,reject)=>{ |
||||
if(!mycondition){ |
||||
resolve(); |
||||
}else{ |
||||
try { |
||||
let test=JSON.parse(mycondition); |
||||
if(typeof test == 'object' && test){ |
||||
resolve() |
||||
}else{ |
||||
this.$message.error("组件JTreeSelect-condition传值有误,需要一个json字符串!") |
||||
reject() |
||||
} |
||||
} catch(e) { |
||||
this.$message.error("组件JTreeSelect-condition传值有误,需要一个json字符串!") |
||||
reject() |
||||
} |
||||
} |
||||
}) |
||||
} |
||||
}, |
||||
//2.2新增 在组件内定义 指定父组件调用时候的传值属性和事件类型 这个牛逼 |
||||
model: { |
||||
prop: 'value', |
||||
event: 'change' |
||||
} |
||||
} |
||||
</script> |
@ -0,0 +1,43 @@
|
||||
<template> |
||||
<a-checkbox-group :options="options" :value="checkboxArray" v-bind="$attrs" @change="onChange" /> |
||||
</template> |
||||
|
||||
<script> |
||||
export default { |
||||
name: 'JCheckbox', |
||||
props: { |
||||
value:{ |
||||
type: String, |
||||
required: false |
||||
}, |
||||
/*label value*/ |
||||
options:{ |
||||
type: Array, |
||||
required: true |
||||
} |
||||
}, |
||||
data(){ |
||||
return { |
||||
checkboxArray:!this.value?[]:this.value.split(",") |
||||
} |
||||
}, |
||||
watch:{ |
||||
value (val) { |
||||
if(!val){ |
||||
this.checkboxArray = [] |
||||
}else{ |
||||
this.checkboxArray = this.value.split(",") |
||||
} |
||||
} |
||||
}, |
||||
methods:{ |
||||
onChange (checkedValues) { |
||||
this.$emit('change', checkedValues.join(",")); |
||||
}, |
||||
}, |
||||
model: { |
||||
prop: 'value', |
||||
event: 'change' |
||||
} |
||||
} |
||||
</script> |
@ -0,0 +1,429 @@
|
||||
<template> |
||||
<div v-bind="fullScreenParentProps"> |
||||
<a-icon v-if="fullScreen" class="full-screen-icon" :type="iconType" @click="()=>fullCoder=!fullCoder"/> |
||||
|
||||
<div class="code-editor-cust full-screen-child"> |
||||
<textarea ref="textarea"></textarea> |
||||
<span @click="nullTipClick" class="null-tip" :class="{'null-tip-hidden':hasCode}" :style="nullTipStyle">{{ placeholderShow }}</span> |
||||
<template v-if="languageChange"> |
||||
<a-select v-model="mode" size="small" class="code-mode-select" @change="changeMode" placeholder="请选择主题"> |
||||
<a-select-option |
||||
v-for="mode in modes" |
||||
:key="mode.value" |
||||
:value="mode.value"> |
||||
{{ mode.label }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</template> |
||||
|
||||
</div> |
||||
</div> |
||||
</template> |
||||
|
||||
<script type="text/ecmascript-6"> |
||||
// 引入全局实例 |
||||
import _CodeMirror from 'codemirror' |
||||
|
||||
// 核心样式 |
||||
import 'codemirror/lib/codemirror.css' |
||||
// 引入主题后还需要在 options 中指定主题才会生效 darcula gruvbox-dark hopscotch monokai |
||||
import 'codemirror/theme/panda-syntax.css' |
||||
//提示css |
||||
import "codemirror/addon/hint/show-hint.css"; |
||||
|
||||
// 需要引入具体的语法高亮库才会有对应的语法高亮效果 |
||||
// codemirror 官方其实支持通过 /addon/mode/loadmode.js 和 /mode/meta.js 来实现动态加载对应语法高亮库 |
||||
// 但 vue 貌似没有无法在实例初始化后再动态加载对应 JS ,所以此处才把对应的 JS 提前引入 |
||||
import 'codemirror/mode/javascript/javascript.js' |
||||
import 'codemirror/mode/css/css.js' |
||||
import 'codemirror/mode/xml/xml.js' |
||||
import 'codemirror/mode/clike/clike.js' |
||||
import 'codemirror/mode/markdown/markdown.js' |
||||
import 'codemirror/mode/python/python.js' |
||||
import 'codemirror/mode/r/r.js' |
||||
import 'codemirror/mode/shell/shell.js' |
||||
import 'codemirror/mode/sql/sql.js' |
||||
import 'codemirror/mode/swift/swift.js' |
||||
import 'codemirror/mode/vue/vue.js' |
||||
|
||||
// 尝试获取全局实例 |
||||
const CodeMirror = window.CodeMirror || _CodeMirror |
||||
|
||||
export default { |
||||
name: 'JCodeEditor', |
||||
props: { |
||||
// 外部传入的内容,用于实现双向绑定 |
||||
value: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
// 外部传入的语法类型 |
||||
language: { |
||||
type: String, |
||||
default: null |
||||
}, |
||||
languageChange:{ |
||||
type: Boolean, |
||||
default:false, |
||||
required:false |
||||
}, |
||||
placeholder: { |
||||
type: String, |
||||
default: null |
||||
}, |
||||
// 显示行号 |
||||
lineNumbers: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
// 是否显示全屏按钮 |
||||
fullScreen: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
// 全屏以后的z-index |
||||
zIndex: { |
||||
type: [Number, String], |
||||
default: 999 |
||||
} |
||||
}, |
||||
data () { |
||||
return { |
||||
// 内部真实的内容 |
||||
code: '', |
||||
iconType: 'fullscreen', |
||||
hasCode:false, |
||||
// 默认的语法类型 |
||||
mode: 'javascript', |
||||
// 编辑器实例 |
||||
coder: null, |
||||
// 默认配置 |
||||
options: { |
||||
// 缩进格式 |
||||
tabSize: 2, |
||||
// 主题,对应主题库 JS 需要提前引入 |
||||
theme: 'panda-syntax', |
||||
line: true, |
||||
// extraKeys: {'Ctrl': 'autocomplete'},//自定义快捷键 |
||||
hintOptions: { |
||||
tables: { |
||||
users: ['name', 'score', 'birthDate'], |
||||
countries: ['name', 'population', 'size'] |
||||
} |
||||
}, |
||||
}, |
||||
// 支持切换的语法高亮类型,对应 JS 已经提前引入 |
||||
// 使用的是 MIME-TYPE ,不过作为前缀的 text/ 在后面指定时写死了 |
||||
modes: [{ |
||||
value: 'css', |
||||
label: 'CSS' |
||||
}, { |
||||
value: 'javascript', |
||||
label: 'Javascript' |
||||
}, { |
||||
value: 'html', |
||||
label: 'XML/HTML' |
||||
}, { |
||||
value: 'x-java', |
||||
label: 'Java' |
||||
}, { |
||||
value: 'x-objectivec', |
||||
label: 'Objective-C' |
||||
}, { |
||||
value: 'x-python', |
||||
label: 'Python' |
||||
}, { |
||||
value: 'x-rsrc', |
||||
label: 'R' |
||||
}, { |
||||
value: 'x-sh', |
||||
label: 'Shell' |
||||
}, { |
||||
value: 'x-sql', |
||||
label: 'SQL' |
||||
}, { |
||||
value: 'x-swift', |
||||
label: 'Swift' |
||||
}, { |
||||
value: 'x-vue', |
||||
label: 'Vue' |
||||
}, { |
||||
value: 'markdown', |
||||
label: 'Markdown' |
||||
}], |
||||
// code 编辑器 是否全屏 |
||||
fullCoder: false |
||||
} |
||||
}, |
||||
watch: { |
||||
fullCoder:{ |
||||
handler(value) { |
||||
if(value){ |
||||
this.iconType="fullscreen-exit" |
||||
}else{ |
||||
this.iconType="fullscreen" |
||||
} |
||||
} |
||||
}, |
||||
// value: { |
||||
// immediate: false, |
||||
// handler(value) { |
||||
// this._getCoder().then(() => { |
||||
// this.coder.setValue(value) |
||||
// }) |
||||
// } |
||||
// }, |
||||
language: { |
||||
immediate: true, |
||||
handler(language) { |
||||
this._getCoder().then(() => { |
||||
// 尝试从父容器获取语法类型 |
||||
if (language) { |
||||
// 获取具体的语法类型对象 |
||||
let modeObj = this._getLanguage(language) |
||||
|
||||
// 判断父容器传入的语法是否被支持 |
||||
if (modeObj) { |
||||
this.mode = modeObj.label |
||||
this.coder.setOption('mode', `text/${modeObj.value}`) |
||||
} |
||||
} |
||||
}) |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
placeholderShow() { |
||||
if (this.placeholder == null) { |
||||
return `请在此输入${this.language}代码` |
||||
} else { |
||||
return this.placeholder |
||||
} |
||||
}, |
||||
nullTipStyle(){ |
||||
if (this.lineNumbers) { |
||||
return { left: '36px' } |
||||
} else { |
||||
return { left: '12px' } |
||||
} |
||||
}, |
||||
// coder 配置 |
||||
coderOptions() { |
||||
return { |
||||
tabSize: this.options.tabSize, |
||||
theme: this.options.theme, |
||||
lineNumbers: this.lineNumbers, |
||||
line: true, |
||||
hintOptions: this.options.hintOptions |
||||
} |
||||
}, |
||||
fullScreenParentProps(){ |
||||
let props = { |
||||
class: ['full-screen-parent', this.fullCoder ? 'full-screen' : ''], |
||||
style: {} |
||||
} |
||||
if (this.fullCoder) { |
||||
props.style['z-index'] = this.zIndex |
||||
} |
||||
return props |
||||
} |
||||
}, |
||||
mounted () { |
||||
// 初始化 |
||||
this._initialize() |
||||
}, |
||||
methods: { |
||||
// 初始化 |
||||
_initialize () { |
||||
// 初始化编辑器实例,传入需要被实例化的文本域对象和默认配置 |
||||
this.coder = CodeMirror.fromTextArea(this.$refs.textarea, this.coderOptions) |
||||
// 编辑器赋值 |
||||
if(this.value||this.code){ |
||||
this.hasCode=true |
||||
this.coder.setValue(this.value || this.code) |
||||
}else{ |
||||
this.coder.setValue('') |
||||
this.hasCode=false |
||||
} |
||||
// 支持双向绑定 |
||||
this.coder.on('change', (coder) => { |
||||
this.code = coder.getValue() |
||||
if(this.code){ |
||||
this.hasCode=true |
||||
}else{ |
||||
this.hasCode=false |
||||
} |
||||
if (this.$emit) { |
||||
this.$emit('input', this.code) |
||||
} |
||||
}) |
||||
this.coder.on('focus', () => { |
||||
this.hasCode=true |
||||
}) |
||||
this.coder.on('blur', () => { |
||||
if(this.code){ |
||||
this.hasCode=true |
||||
}else{ |
||||
this.hasCode=false |
||||
} |
||||
}) |
||||
|
||||
/* this.coder.on('cursorActivity',()=>{ |
||||
this.coder.showHint() |
||||
})*/ |
||||
|
||||
}, |
||||
getCodeContent(){ |
||||
return this.code |
||||
}, |
||||
setCodeContent(val){ |
||||
setTimeout(()=>{ |
||||
if(!val){ |
||||
this.coder.setValue('') |
||||
}else{ |
||||
this.coder.setValue(val) |
||||
} |
||||
},300) |
||||
}, |
||||
// 获取当前语法类型 |
||||
_getLanguage (language) { |
||||
// 在支持的语法类型列表中寻找传入的语法类型 |
||||
return this.modes.find((mode) => { |
||||
// 所有的值都忽略大小写,方便比较 |
||||
let currentLanguage = language.toLowerCase() |
||||
let currentLabel = mode.label.toLowerCase() |
||||
let currentValue = mode.value.toLowerCase() |
||||
|
||||
// 由于真实值可能不规范,例如 java 的真实值是 x-java ,所以讲 value 和 label 同时和传入语法进行比较 |
||||
return currentLabel === currentLanguage || currentValue === currentLanguage |
||||
}) |
||||
}, |
||||
_getCoder() { |
||||
let _this = this |
||||
return new Promise((resolve) => { |
||||
(function get() { |
||||
if (_this.coder) { |
||||
resolve(_this.coder) |
||||
} else { |
||||
setTimeout(get, 10) |
||||
} |
||||
})() |
||||
}) |
||||
}, |
||||
// 更改模式 |
||||
changeMode (val) { |
||||
// 修改编辑器的语法配置 |
||||
this.coder.setOption('mode', `text/${val}`) |
||||
|
||||
// 获取修改后的语法 |
||||
let label = this._getLanguage(val).label.toLowerCase() |
||||
|
||||
// 允许父容器通过以下函数监听当前的语法值 |
||||
this.$emit('language-change', label) |
||||
}, |
||||
nullTipClick(){ |
||||
this.coder.focus() |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="less"> |
||||
.code-editor-cust{ |
||||
flex-grow:1; |
||||
display:flex; |
||||
position:relative; |
||||
height:100%; |
||||
.CodeMirror{ |
||||
flex-grow:1; |
||||
z-index:1; |
||||
.CodeMirror-code{ |
||||
line-height:19px; |
||||
} |
||||
|
||||
} |
||||
.code-mode-select{ |
||||
position:absolute; |
||||
z-index:2; |
||||
right:10px; |
||||
top:10px; |
||||
max-width:130px; |
||||
} |
||||
.CodeMirror{ |
||||
height: auto; |
||||
min-height:100%; |
||||
} |
||||
.null-tip{ |
||||
position: absolute; |
||||
top: 4px; |
||||
left: 36px; |
||||
z-index: 10; |
||||
color: #ffffffc9; |
||||
line-height: initial; |
||||
} |
||||
.null-tip-hidden{ |
||||
display: none; |
||||
} |
||||
} |
||||
|
||||
/* 全屏样式 */ |
||||
.full-screen-parent { |
||||
position: relative; |
||||
|
||||
.full-screen-icon { |
||||
opacity: 0; |
||||
color: black; |
||||
width: 20px; |
||||
height: 20px; |
||||
line-height: 24px; |
||||
background-color: white; |
||||
position: absolute; |
||||
top: 2px; |
||||
right: 2px; |
||||
z-index: 9; |
||||
cursor: pointer; |
||||
transition: opacity 0.3s; |
||||
} |
||||
|
||||
&:hover { |
||||
.full-screen-icon { |
||||
opacity: 1; |
||||
|
||||
&:hover { |
||||
background-color: rgba(255, 255, 255, 0.88); |
||||
} |
||||
} |
||||
} |
||||
|
||||
&.full-screen { |
||||
position: fixed; |
||||
top: 10px; |
||||
left: 10px; |
||||
width: calc(100% - 20px); |
||||
height: calc(100% - 20px); |
||||
padding: 10px; |
||||
background-color: #f5f5f5; |
||||
|
||||
.full-screen-icon { |
||||
top: 12px; |
||||
right: 12px; |
||||
} |
||||
.full-screen-child { |
||||
height: 100%; |
||||
max-height: 100%; |
||||
min-height: 100%; |
||||
} |
||||
} |
||||
|
||||
.full-screen-child { |
||||
min-height: 120px; |
||||
max-height: 320px; |
||||
overflow:hidden; |
||||
} |
||||
|
||||
} |
||||
|
||||
.CodeMirror-cursor{ |
||||
height:18.4px !important; |
||||
} |
||||
</style> |
@ -0,0 +1,65 @@
|
||||
<template> |
||||
<div class="components-input-demo-presuffix"> |
||||
<a-input @click="openModal" placeholder="corn表达式" v-model="cron" @change="handleOK"> |
||||
<a-icon slot="prefix" type="schedule" title="corn控件"/> |
||||
<a-icon v-if="cron" slot="suffix" type="close-circle" @click="handleEmpty" title="清空"/> |
||||
</a-input> |
||||
<JCronModal ref="innerVueCron" :data="cron" @ok="handleOK"></JCronModal> |
||||
</div> |
||||
</template> |
||||
<script> |
||||
import JCronModal from "./modal/JCronModal"; |
||||
export default { |
||||
name: 'JCron', |
||||
components: { |
||||
JCronModal |
||||
}, |
||||
props: { |
||||
value: { |
||||
required: false, |
||||
type: String, |
||||
} |
||||
}, |
||||
data(){ |
||||
return { |
||||
cron: this.value, |
||||
} |
||||
}, |
||||
watch:{ |
||||
value(val){ |
||||
this.cron = val |
||||
} |
||||
}, |
||||
methods:{ |
||||
openModal(){ |
||||
this.$refs.innerVueCron.show(); |
||||
}, |
||||
handleOK(val){ |
||||
this.cron = val; |
||||
this.$emit("change", this.cron); |
||||
//this.$emit("change", Object.assign({}, this.cron)); |
||||
}, |
||||
handleEmpty(){ |
||||
this.handleOK('') |
||||
} |
||||
}, |
||||
model: { |
||||
prop: 'value', |
||||
event: 'change' |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
.components-input-demo-presuffix .anticon-close-circle { |
||||
cursor: pointer; |
||||
color: #ccc; |
||||
transition: color 0.3s; |
||||
font-size: 12px; |
||||
} |
||||
.components-input-demo-presuffix .anticon-close-circle:hover { |
||||
color: #f5222d; |
||||
} |
||||
.components-input-demo-presuffix .anticon-close-circle:active { |
||||
color: #666; |
||||
} |
||||
</style> |
@ -0,0 +1,85 @@
|
||||
<template> |
||||
<a-date-picker |
||||
:disabled="disabled || readOnly" |
||||
:placeholder="placeholder" |
||||
@change="handleDateChange" |
||||
:value="momVal" |
||||
:showTime="showTime" |
||||
:format="dateFormat" |
||||
:getCalendarContainer="getCalendarContainer" |
||||
/> |
||||
</template> |
||||
<script> |
||||
import moment from 'moment' |
||||
export default { |
||||
name: 'JDate', |
||||
props: { |
||||
placeholder:{ |
||||
type: String, |
||||
default: '', |
||||
required: false |
||||
}, |
||||
value:{ |
||||
type: String, |
||||
required: false |
||||
}, |
||||
dateFormat:{ |
||||
type: String, |
||||
default: 'YYYY-MM-DD HH:mm:ss', |
||||
required: false |
||||
}, |
||||
//此属性可以被废弃了 |
||||
triggerChange:{ |
||||
type: Boolean, |
||||
required: false, |
||||
default: false |
||||
}, |
||||
readOnly:{ |
||||
type: Boolean, |
||||
required: false, |
||||
default: false |
||||
}, |
||||
disabled:{ |
||||
type: Boolean, |
||||
required: false, |
||||
default: false |
||||
}, |
||||
showTime:{ |
||||
type: Boolean, |
||||
required: false, |
||||
default: false |
||||
}, |
||||
getCalendarContainer: { |
||||
type: Function, |
||||
default: (node) => node.parentNode |
||||
} |
||||
}, |
||||
data () { |
||||
let dateStr = this.value; |
||||
return { |
||||
decorator:"", |
||||
momVal:!dateStr?null:moment(dateStr,this.dateFormat) |
||||
} |
||||
}, |
||||
watch: { |
||||
value (val) { |
||||
if(!val){ |
||||
this.momVal = null |
||||
}else{ |
||||
this.momVal = moment(val,this.dateFormat) |
||||
} |
||||
} |
||||
}, |
||||
methods: { |
||||
moment, |
||||
handleDateChange(mom,dateStr){ |
||||
this.$emit('change', dateStr); |
||||
} |
||||
}, |
||||
//2.2新增 在组件内定义 指定父组件调用时候的传值属性和事件类型 这个牛逼 |
||||
model: { |
||||
prop: 'value', |
||||
event: 'change' |
||||
} |
||||
} |
||||
</script> |
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,29 @@
|
||||
<template> |
||||
<a-tooltip placement="topLeft"> |
||||
<template slot="title"> |
||||
<span>{{value}}</span> |
||||
</template> |
||||
{{ value | ellipsis(length) }} |
||||
</a-tooltip> |
||||
</template> |
||||
|
||||
<script> |
||||
export default { |
||||
name: 'JEllipsis', |
||||
props: { |
||||
value: { |
||||
type: String, |
||||
required: false, |
||||
}, |
||||
length: { |
||||
type: Number, |
||||
required: false, |
||||
default: 25, |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,61 @@
|
||||
<template> |
||||
<div :class="disabled?'jeecg-form-container-disabled':''"> |
||||
<fieldset :disabled="disabled"> |
||||
<slot name="detail"></slot> |
||||
</fieldset> |
||||
<slot name="edit"></slot> |
||||
<fieldset disabled> |
||||
<slot></slot> |
||||
</fieldset> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
/** |
||||
* 使用方法 |
||||
* 在form下直接写这个组件就行了, |
||||
*<a-form layout="inline" :form="form" > |
||||
* <j-form-container :disabled="true"> |
||||
* <!-- 表单内容省略..... --> |
||||
* </j-form-container> |
||||
*</a-form> |
||||
*/ |
||||
export default { |
||||
name: 'JFormContainer', |
||||
props:{ |
||||
disabled:{ |
||||
type:Boolean, |
||||
default:false, |
||||
required:false |
||||
} |
||||
}, |
||||
mounted(){ |
||||
console.log("我是表单禁用专用组件,但是我并不支持表单中iframe的内容禁用") |
||||
} |
||||
} |
||||
</script> |
||||
<style> |
||||
.jeecg-form-container-disabled{ |
||||
cursor: not-allowed; |
||||
} |
||||
.jeecg-form-container-disabled fieldset[disabled] { |
||||
-ms-pointer-events: none; |
||||
pointer-events: none; |
||||
} |
||||
.jeecg-form-container-disabled .ant-select{ |
||||
-ms-pointer-events: none; |
||||
pointer-events: none; |
||||
} |
||||
|
||||
.jeecg-form-container-disabled .ant-upload-select{display:none} |
||||
.jeecg-form-container-disabled .ant-upload-list{cursor:grabbing} |
||||
.jeecg-form-container-disabled fieldset[disabled] .ant-upload-list{ |
||||
-ms-pointer-events: auto !important; |
||||
pointer-events: auto !important; |
||||
} |
||||
|
||||
.jeecg-form-container-disabled .ant-upload-list-item-actions .anticon-delete, |
||||
.jeecg-form-container-disabled .ant-upload-list-item .anticon-close{ |
||||
display: none; |
||||
} |
||||
</style> |
@ -0,0 +1,202 @@
|
||||
<template> |
||||
<div class="gc-canvas" @click="reloadPic"> |
||||
<canvas id="gc-canvas" :width="contentWidth" :height="contentHeight"></canvas> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
import { getAction } from '@/api/manage' |
||||
|
||||
export default { |
||||
name: 'JGraphicCode', |
||||
props: { |
||||
length:{ |
||||
type: Number, |
||||
default: 4 |
||||
}, |
||||
fontSizeMin: { |
||||
type: Number, |
||||
default: 20 |
||||
}, |
||||
fontSizeMax: { |
||||
type: Number, |
||||
default: 45 |
||||
}, |
||||
backgroundColorMin: { |
||||
type: Number, |
||||
default: 180 |
||||
}, |
||||
backgroundColorMax: { |
||||
type: Number, |
||||
default: 240 |
||||
}, |
||||
colorMin: { |
||||
type: Number, |
||||
default: 50 |
||||
}, |
||||
colorMax: { |
||||
type: Number, |
||||
default: 160 |
||||
}, |
||||
lineColorMin: { |
||||
type: Number, |
||||
default: 40 |
||||
}, |
||||
lineColorMax: { |
||||
type: Number, |
||||
default: 180 |
||||
}, |
||||
dotColorMin: { |
||||
type: Number, |
||||
default: 0 |
||||
}, |
||||
dotColorMax: { |
||||
type: Number, |
||||
default: 255 |
||||
}, |
||||
contentWidth: { |
||||
type: Number, |
||||
default:136 |
||||
}, |
||||
contentHeight: { |
||||
type: Number, |
||||
default: 38 |
||||
}, |
||||
remote:{ |
||||
type:Boolean, |
||||
default:false, |
||||
required:false |
||||
} |
||||
}, |
||||
methods: { |
||||
// 生成一个随机数 |
||||
randomNum (min, max) { |
||||
return Math.floor(Math.random() * (max - min) + min) |
||||
}, |
||||
// 生成一个随机的颜色 |
||||
randomColor (min, max) { |
||||
let r = this.randomNum(min, max) |
||||
let g = this.randomNum(min, max) |
||||
let b = this.randomNum(min, max) |
||||
return 'rgb(' + r + ',' + g + ',' + b + ')' |
||||
}, |
||||
drawPic () { |
||||
this.randomCode().then(()=>{ |
||||
let canvas = document.getElementById('gc-canvas') |
||||
let ctx = canvas.getContext('2d') |
||||
ctx.textBaseline = 'bottom' |
||||
// 绘制背景 |
||||
ctx.fillStyle = this.randomColor(this.backgroundColorMin, this.backgroundColorMax) |
||||
ctx.fillRect(0, 0, this.contentWidth, this.contentHeight) |
||||
// 绘制文字 |
||||
for (let i = 0; i < this.code.length; i++) { |
||||
this.drawText(ctx, this.code[i], i) |
||||
} |
||||
this.drawLine(ctx) |
||||
this.drawDot(ctx) |
||||
this.$emit("success",this.code) |
||||
}) |
||||
}, |
||||
drawText (ctx, txt, i) { |
||||
ctx.fillStyle = this.randomColor(this.colorMin, this.colorMax) |
||||
let fontSize = this.randomNum(this.fontSizeMin, this.fontSizeMax) |
||||
ctx.font = fontSize + 'px SimHei' |
||||
let padding = 10; |
||||
let offset = (this.contentWidth-40)/(this.code.length-1) |
||||
let x=padding; |
||||
if(i>0){ |
||||
x = padding+(i*offset) |
||||
} |
||||
//let x = (i + 1) * (this.contentWidth / (this.code.length + 1)) |
||||
let y = this.randomNum(this.fontSizeMax, this.contentHeight - 5) |
||||
if(fontSize>40){ |
||||
y=40 |
||||
} |
||||
var deg = this.randomNum(-10,10) |
||||
// 修改坐标原点和旋转角度 |
||||
ctx.translate(x, y) |
||||
ctx.rotate(deg * Math.PI / 180) |
||||
ctx.fillText(txt, 0, 0) |
||||
// 恢复坐标原点和旋转角度 |
||||
ctx.rotate(-deg * Math.PI / 180) |
||||
ctx.translate(-x, -y) |
||||
}, |
||||
drawLine (ctx) { |
||||
// 绘制干扰线 |
||||
for (let i = 0; i <1; i++) { |
||||
ctx.strokeStyle = this.randomColor(this.lineColorMin, this.lineColorMax) |
||||
ctx.beginPath() |
||||
ctx.moveTo(this.randomNum(0, this.contentWidth), this.randomNum(0, this.contentHeight)) |
||||
ctx.lineTo(this.randomNum(0, this.contentWidth), this.randomNum(0, this.contentHeight)) |
||||
ctx.stroke() |
||||
} |
||||
}, |
||||
drawDot (ctx) { |
||||
// 绘制干扰点 |
||||
for (let i = 0; i < 100; i++) { |
||||
ctx.fillStyle = this.randomColor(0, 255) |
||||
ctx.beginPath() |
||||
ctx.arc(this.randomNum(0, this.contentWidth), this.randomNum(0, this.contentHeight), 1, 0, 2 * Math.PI) |
||||
ctx.fill() |
||||
} |
||||
}, |
||||
reloadPic(){ |
||||
this.drawPic() |
||||
}, |
||||
randomCode(){ |
||||
return new Promise((resolve)=>{ |
||||
if(this.remote==true){ |
||||
getAction("/sys/getCheckCode").then(res=>{ |
||||
if(res.success){ |
||||
this.checkKey = res.result.key |
||||
this.code = window.atob(res.result.code) |
||||
resolve(); |
||||
}else{ |
||||
this.$message.error("生成验证码错误,请联系系统管理员") |
||||
this.code = 'BUG' |
||||
resolve(); |
||||
} |
||||
}).catch(()=>{ |
||||
console.log("生成验证码连接服务器异常") |
||||
this.code = 'BUG' |
||||
resolve(); |
||||
}) |
||||
}else{ |
||||
this.randomLocalCode(); |
||||
resolve(); |
||||
} |
||||
}) |
||||
}, |
||||
randomLocalCode(){ |
||||
let random = '' |
||||
//去掉了I l i o O |
||||
let str = "QWERTYUPLKJHGFDSAZXCVBNMqwertyupkjhgfdsazxcvbnm1234567890" |
||||
for(let i = 0; i < this.length; i++) { |
||||
let index = Math.floor(Math.random()*57); |
||||
random += str[index]; |
||||
} |
||||
this.code = random |
||||
}, |
||||
getLoginParam(){ |
||||
return { |
||||
checkCode:this.code, |
||||
checkKey:this.checkKey |
||||
} |
||||
} |
||||
}, |
||||
mounted () { |
||||
this.drawPic() |
||||
}, |
||||
data(){ |
||||
return { |
||||
code:"", |
||||
checkKey:"" |
||||
} |
||||
} |
||||
|
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,226 @@
|
||||
<template> |
||||
<a-upload |
||||
name="file" |
||||
listType="picture-card" |
||||
:multiple="isMultiple" |
||||
:action="uploadAction" |
||||
:headers="headers" |
||||
:data="{biz:bizPath}" |
||||
:fileList="fileList" |
||||
:beforeUpload="beforeUpload" |
||||
:disabled="disabled" |
||||
:isMultiple="isMultiple" |
||||
:showUploadList="isMultiple" |
||||
@change="handleChange" |
||||
@preview="handlePreview"> |
||||
<img v-if="!isMultiple && picUrl" :src="getAvatarView()" style="height:104px;max-width:300px"/> |
||||
<div v-else > |
||||
<a-icon :type="uploadLoading ? 'loading' : 'plus'" /> |
||||
<div class="ant-upload-text">{{ text }}</div> |
||||
</div> |
||||
<a-modal :visible="previewVisible" :width="1000" :footer="null" @cancel="handleCancel()"> |
||||
<img alt="example" style="width: 100%" :src="previewImage"/> |
||||
</a-modal> |
||||
</a-upload> |
||||
</template> |
||||
|
||||
<script> |
||||
import Vue from 'vue' |
||||
import { getFileAccessHttpUrl } from '@/api/manage' |
||||
import { fileSizeLimit } from '@/api/api' |
||||
|
||||
const uidGenerator=()=>{ |
||||
return '-'+parseInt(Math.random()*10000+1,10); |
||||
} |
||||
const getFileName=(path)=>{ |
||||
if(path.lastIndexOf("\\")>=0){ |
||||
let reg=new RegExp("\\\\","g"); |
||||
path = path.replace(reg,"/"); |
||||
} |
||||
return path.substring(path.lastIndexOf("/")+1); |
||||
} |
||||
export default { |
||||
name: 'JImageUpload', |
||||
data(){ |
||||
return { |
||||
uploadAction:window._CONFIG['domianURL']+"/systemConfig/upload", |
||||
uploadLoading:false, |
||||
picUrl:false, |
||||
headers:{}, |
||||
fileList: [], |
||||
previewImage:"", |
||||
previewVisible: false, |
||||
sizeLimit: 0, |
||||
uploadGoOn:true, |
||||
} |
||||
}, |
||||
props:{ |
||||
text:{ |
||||
type:String, |
||||
required:false, |
||||
default:"上传" |
||||
}, |
||||
/*这个属性用于控制文件上传的业务路径*/ |
||||
bizPath:{ |
||||
type:String, |
||||
required:false, |
||||
default:"temp" |
||||
}, |
||||
value:{ |
||||
type:[String,Array], |
||||
required:false |
||||
}, |
||||
disabled:{ |
||||
type:Boolean, |
||||
required:false, |
||||
default: false |
||||
}, |
||||
isMultiple:{ |
||||
type:Boolean, |
||||
required:false, |
||||
default: false |
||||
} |
||||
}, |
||||
watch:{ |
||||
value(val){ |
||||
if (val instanceof Array) { |
||||
this.initFileList(val.join(',')) |
||||
} else { |
||||
this.initFileList(val) |
||||
} |
||||
} |
||||
}, |
||||
created(){ |
||||
this.initFileSizeLimit() |
||||
this.headers = {"X-Access-Token":""} |
||||
}, |
||||
methods:{ |
||||
initFileSizeLimit() { |
||||
fileSizeLimit().then((res)=>{ |
||||
if(res.code === 200) { |
||||
this.sizeLimit = res.data |
||||
} |
||||
}) |
||||
}, |
||||
initFileList(paths){ |
||||
if(!paths || paths.length==0){ |
||||
this.fileList = []; |
||||
this.picUrl = false; |
||||
return; |
||||
} |
||||
this.picUrl = true; |
||||
let fileList = []; |
||||
let arr = paths.split(",") |
||||
for(var a=0;a<arr.length;a++){ |
||||
let url = getFileAccessHttpUrl(arr[a]); |
||||
fileList.push({ |
||||
uid: uidGenerator(), |
||||
name: getFileName(arr[a]), |
||||
status: 'done', |
||||
url: url, |
||||
response:{ |
||||
code:"history", |
||||
data:arr[a] |
||||
} |
||||
}) |
||||
} |
||||
this.fileList = fileList |
||||
}, |
||||
beforeUpload: function(file){ |
||||
this.uploadGoOn=true |
||||
let fileType = file.type; |
||||
let fileSize = file.size; |
||||
if(fileType.indexOf('image')<0){ |
||||
this.$message.warning('请上传图片'); |
||||
this.uploadGoOn=false |
||||
return false; |
||||
} |
||||
//验证文件大小 |
||||
if(fileSize>this.sizeLimit) { |
||||
let parseSizeLimit = (this.sizeLimit/1024/1024).toFixed(2) |
||||
this.$message.warning('抱歉,图片大小不能超过' + parseSizeLimit + 'M'); |
||||
this.uploadGoOn=false |
||||
return false; |
||||
} |
||||
return true |
||||
}, |
||||
handleChange(info) { |
||||
console.log(info,"--文件列表改变--") |
||||
if(!info.file.status && this.uploadGoOn === false){ |
||||
info.fileList.pop(); |
||||
} |
||||
this.picUrl = false; |
||||
let fileList = info.fileList |
||||
if(info.file.status==='done'){ |
||||
if(info.file.response.code === 200){ |
||||
this.picUrl = true; |
||||
fileList = fileList.map((file) => { |
||||
if (file.response) { |
||||
file.url = file.response.data; |
||||
} |
||||
return file; |
||||
}); |
||||
} |
||||
//this.$message.success(`${info.file.name} 上传成功!`); |
||||
}else if (info.file.status === 'error') { |
||||
this.$message.error(`${info.file.name} 上传失败.`); |
||||
}else if(info.file.status === 'removed'){ |
||||
this.handleDelete(info.file) |
||||
} |
||||
this.fileList = fileList |
||||
if(info.file.status==='done' || info.file.status === 'removed'){ |
||||
this.handlePathChange() |
||||
} |
||||
}, |
||||
// 预览 |
||||
handlePreview (file) { |
||||
this.previewImage = file.url || file.thumbUrl |
||||
this.previewVisible = true |
||||
}, |
||||
getAvatarView(){ |
||||
if(this.fileList.length>0){ |
||||
let url = this.fileList[0].url |
||||
return url |
||||
} |
||||
}, |
||||
handlePathChange(){ |
||||
let uploadFiles = this.fileList |
||||
let path = '' |
||||
if(!uploadFiles || uploadFiles.length==0){ |
||||
path = '' |
||||
} |
||||
let arr = []; |
||||
if(!this.isMultiple){ |
||||
arr.push(uploadFiles[uploadFiles.length-1].response.data) |
||||
}else{ |
||||
for(var a=0;a<uploadFiles.length;a++){ |
||||
arr.push(uploadFiles[a].response.data) |
||||
} |
||||
} |
||||
if(arr.length>0){ |
||||
path = arr.join(",") |
||||
} |
||||
this.$emit('change', path); |
||||
}, |
||||
handleDelete(file){ |
||||
//如有需要新增 删除逻辑 |
||||
console.log(file) |
||||
}, |
||||
handleCancel() { |
||||
this.close(); |
||||
this.previewVisible = false; |
||||
}, |
||||
close () { |
||||
|
||||
}, |
||||
}, |
||||
model: { |
||||
prop: 'value', |
||||
event: 'change' |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,123 @@
|
||||
<template> |
||||
<a-modal |
||||
title="导入EXCEL" |
||||
:width="600" |
||||
:visible="visible" |
||||
:confirmLoading="uploading" |
||||
@cancel="handleClose"> |
||||
|
||||
<a-upload |
||||
name="file" |
||||
:multiple="true" |
||||
accept=".xls,.xlsx" |
||||
:fileList="fileList" |
||||
:remove="handleRemove" |
||||
:beforeUpload="beforeUpload"> |
||||
<a-button> |
||||
<a-icon type="upload" /> |
||||
选择导入文件 |
||||
</a-button> |
||||
</a-upload> |
||||
|
||||
<template slot="footer"> |
||||
<a-button @click="handleClose">关闭</a-button> |
||||
<a-button |
||||
type="primary" |
||||
@click="handleImport" |
||||
:disabled="fileList.length === 0" |
||||
:loading="uploading"> |
||||
{{ uploading ? '上传中...' : '开始上传' }} |
||||
</a-button> |
||||
</template> |
||||
|
||||
</a-modal> |
||||
</template> |
||||
|
||||
<script> |
||||
import { postAction } from '@/api/manage' |
||||
export default { |
||||
name: 'JImportModal', |
||||
props:{ |
||||
url:{ |
||||
type: String, |
||||
default: '', |
||||
required: false |
||||
}, |
||||
biz:{ |
||||
type: String, |
||||
default: '', |
||||
required: false |
||||
} |
||||
}, |
||||
data(){ |
||||
return { |
||||
visible:false, |
||||
uploading:false, |
||||
fileList:[], |
||||
uploadAction:'', |
||||
foreignKeys:'' |
||||
} |
||||
}, |
||||
watch: { |
||||
url (val) { |
||||
if(val){ |
||||
this.uploadAction = window._CONFIG['domianURL']+val |
||||
} |
||||
} |
||||
}, |
||||
created () { |
||||
this.uploadAction = window._CONFIG['domianURL']+this.url |
||||
}, |
||||
|
||||
methods:{ |
||||
handleClose(){ |
||||
this.visible=false |
||||
}, |
||||
show(arg){ |
||||
this.fileList = [] |
||||
this.uploading = false |
||||
this.visible = true |
||||
this.foreignKeys = arg; |
||||
}, |
||||
handleRemove(file) { |
||||
const index = this.fileList.indexOf(file); |
||||
const newFileList = this.fileList.slice(); |
||||
newFileList.splice(index, 1); |
||||
this.fileList = newFileList |
||||
}, |
||||
beforeUpload(file) { |
||||
this.fileList = [...this.fileList, file] |
||||
return false; |
||||
}, |
||||
handleImport() { |
||||
const { fileList } = this; |
||||
const formData = new FormData(); |
||||
if(this.biz){ |
||||
formData.append('isSingleTableImport',this.biz); |
||||
} |
||||
if(this.foreignKeys && this.foreignKeys.length>0){ |
||||
formData.append('foreignKeys',this.foreignKeys); |
||||
} |
||||
fileList.forEach((file) => { |
||||
formData.append('files[]', file); |
||||
}); |
||||
this.uploading = true |
||||
postAction(this.uploadAction, formData).then((res) => { |
||||
this.uploading = false |
||||
if(res.success){ |
||||
this.$message.success(res.message) |
||||
this.visible=false |
||||
this.$emit('ok') |
||||
}else{ |
||||
this.$message.warning(res.message) |
||||
} |
||||
}) |
||||
} |
||||
|
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,100 @@
|
||||
<template> |
||||
<a-input :placeholder="placeholder" :value="inputVal" @input="backValue"></a-input> |
||||
</template> |
||||
|
||||
<script> |
||||
const JINPUT_QUERY_LIKE = 'like'; |
||||
const JINPUT_QUERY_NE = 'ne'; |
||||
const JINPUT_QUERY_GE = 'ge'; //大于等于 |
||||
const JINPUT_QUERY_LE = 'le'; //小于等于 |
||||
|
||||
export default { |
||||
name: 'JInput', |
||||
props:{ |
||||
value:{ |
||||
type:String, |
||||
required:false |
||||
}, |
||||
type:{ |
||||
type:String, |
||||
required:false, |
||||
default:JINPUT_QUERY_LIKE |
||||
}, |
||||
placeholder:{ |
||||
type:String, |
||||
required:false, |
||||
default:'' |
||||
} |
||||
}, |
||||
watch:{ |
||||
value:{ |
||||
immediate:true, |
||||
handler:function(){ |
||||
this.initVal(); |
||||
} |
||||
}, |
||||
// update-begin author:sunjianlei date:20200225 for:当 type 变化的时候重新计算值 ------ |
||||
type() { |
||||
this.backValue({ target: { value: this.inputVal } }) |
||||
}, |
||||
// update-end author:sunjianlei date:20200225 for:当 type 变化的时候重新计算值 ------ |
||||
}, |
||||
model: { |
||||
prop: 'value', |
||||
event: 'change' |
||||
}, |
||||
data(){ |
||||
return { |
||||
inputVal:'' |
||||
} |
||||
}, |
||||
methods:{ |
||||
initVal(){ |
||||
if(!this.value){ |
||||
this.inputVal = '' |
||||
}else{ |
||||
let text = this.value |
||||
switch (this.type) { |
||||
case JINPUT_QUERY_LIKE: |
||||
text = text.substring(1,text.length-1); |
||||
break; |
||||
case JINPUT_QUERY_NE: |
||||
text = text.substring(1); |
||||
break; |
||||
case JINPUT_QUERY_GE: |
||||
text = text.substring(2); |
||||
break; |
||||
case JINPUT_QUERY_LE: |
||||
text = text.substring(2); |
||||
break; |
||||
default: |
||||
} |
||||
this.inputVal = text |
||||
} |
||||
}, |
||||
backValue(e){ |
||||
let text = e.target.value |
||||
switch (this.type) { |
||||
case JINPUT_QUERY_LIKE: |
||||
text = "*"+text+"*"; |
||||
break; |
||||
case JINPUT_QUERY_NE: |
||||
text = "!"+text; |
||||
break; |
||||
case JINPUT_QUERY_GE: |
||||
text = ">="+text; |
||||
break; |
||||
case JINPUT_QUERY_LE: |
||||
text = "<="+text; |
||||
break; |
||||
default: |
||||
} |
||||
this.$emit("change",text) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,237 @@
|
||||
<template> |
||||
<a-modal |
||||
ref="modal" |
||||
:class="getClass(modalClass)" |
||||
:style="getStyle(modalStyle)" |
||||
:visible="visible" |
||||
v-bind="_attrs" |
||||
v-on="$listeners" |
||||
@ok="handleOk" |
||||
@cancel="handleCancel" |
||||
> |
||||
|
||||
<slot></slot> |
||||
|
||||
<template v-if="!isNoTitle" slot="title"> |
||||
<a-row class="j-modal-title-row" type="flex"> |
||||
<a-col class="left"> |
||||
<slot name="title">{{ title }}</slot> |
||||
</a-col> |
||||
<a-col class="right"> |
||||
<a-button v-if="switchFullscreen" @click="toggleFullscreen" class="ant-modal-close ant-modal-close-x" |
||||
ghost type="link" :icon="fullscreenButtonIcon"/> |
||||
</a-col> |
||||
</a-row> |
||||
</template> |
||||
|
||||
<!-- 处理 scopedSlots --> |
||||
<template v-for="slotName of scopedSlotsKeys" :slot="slotName"> |
||||
<slot :name="slotName"></slot> |
||||
</template> |
||||
|
||||
<!-- 处理 slots --> |
||||
<template v-for="slotName of slotsKeys" v-slot:[slotName]> |
||||
<slot :name="slotName"></slot> |
||||
</template> |
||||
|
||||
</a-modal> |
||||
</template> |
||||
|
||||
<script> |
||||
|
||||
import { getClass, getStyle } from '@/utils/props-util' |
||||
import { triggerWindowResizeEvent } from "@/utils/util" |
||||
import Vue from 'vue' |
||||
|
||||
export default { |
||||
name: 'JModal', |
||||
props: { |
||||
title: String, |
||||
// 可使用 .sync 修饰符 |
||||
visible: Boolean, |
||||
// 前缀代号 |
||||
prefixNo: String, |
||||
// 是否全屏弹窗,当全屏时无论如何都会禁止 body 滚动。可使用 .sync 修饰符 |
||||
fullscreen: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
// 是否允许展示新手引导(允许后右上角会出现一个按钮) |
||||
switchHelp: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
// 是否允许切换全屏(允许后右上角会出现一个按钮) |
||||
switchFullscreen: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
// 点击确定按钮的时候是否关闭弹窗 |
||||
okClose: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
}, |
||||
data() { |
||||
return { |
||||
// 内部使用的 slots ,不再处理 |
||||
usedSlots: ['title'], |
||||
// 实际控制是否全屏的参数 |
||||
innerFullscreen: this.fullscreen, |
||||
} |
||||
}, |
||||
computed: { |
||||
// 一些未处理的参数或特殊处理的参数绑定到 a-modal 上 |
||||
_attrs() { |
||||
let attrs = { ...this.$attrs } |
||||
// 如果全屏就将宽度设为 100% |
||||
if (this.innerFullscreen) { |
||||
attrs['width'] = '100%' |
||||
} |
||||
return attrs |
||||
}, |
||||
modalClass() { |
||||
return { |
||||
'j-modal-box': true, |
||||
'fullscreen': this.innerFullscreen, |
||||
'no-title': this.isNoTitle, |
||||
'no-footer': this.isNoFooter, |
||||
} |
||||
}, |
||||
modalStyle() { |
||||
let style = {} |
||||
// 如果全屏就将top设为 0 |
||||
if (this.innerFullscreen) { |
||||
style['top'] = '0' |
||||
} |
||||
return style |
||||
}, |
||||
isNoTitle() { |
||||
return !this.title && !this.allSlotsKeys.includes('title') |
||||
}, |
||||
isNoFooter() { |
||||
return this._attrs['footer'] === null |
||||
}, |
||||
slotsKeys() { |
||||
return Object.keys(this.$slots).filter(key => !this.usedSlots.includes(key)) |
||||
}, |
||||
scopedSlotsKeys() { |
||||
return Object.keys(this.$scopedSlots).filter(key => !this.usedSlots.includes(key)) |
||||
}, |
||||
allSlotsKeys() { |
||||
return this.slotsKeys.concat(this.scopedSlotsKeys) |
||||
}, |
||||
// 切换全屏的按钮图标 |
||||
fullscreenButtonIcon() { |
||||
return this.innerFullscreen ? 'fullscreen-exit' : 'fullscreen' |
||||
}, |
||||
}, |
||||
watch: { |
||||
visible() { |
||||
if (this.visible) { |
||||
this.innerFullscreen = this.fullscreen |
||||
} |
||||
}, |
||||
innerFullscreen(val) { |
||||
this.$emit('update:fullscreen', val) |
||||
}, |
||||
}, |
||||
methods: { |
||||
|
||||
getClass(clazz) { |
||||
return { ...getClass(this), ...clazz } |
||||
}, |
||||
getStyle(style) { |
||||
return { ...getStyle(this), ...style } |
||||
}, |
||||
|
||||
close() { |
||||
this.$emit('update:visible', false) |
||||
}, |
||||
|
||||
handleOk() { |
||||
if (this.okClose) { |
||||
this.close() |
||||
} |
||||
}, |
||||
handleCancel() { |
||||
this.close() |
||||
}, |
||||
|
||||
|
||||
/** 切换全屏 */ |
||||
toggleFullscreen() { |
||||
this.innerFullscreen = !this.innerFullscreen |
||||
triggerWindowResizeEvent() |
||||
}, |
||||
|
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="less"> |
||||
.j-modal-box { |
||||
|
||||
&.fullscreen { |
||||
top: 0; |
||||
left: 0; |
||||
padding: 0; |
||||
|
||||
height: 100vh; |
||||
|
||||
& .ant-modal-content { |
||||
height: 100vh; |
||||
border-radius: 0; |
||||
|
||||
& .ant-modal-body { |
||||
/* title 和 footer 各占 55px */ |
||||
height: calc(100% - 55px - 55px); |
||||
overflow: auto; |
||||
} |
||||
} |
||||
|
||||
&.no-title, &.no-footer { |
||||
.ant-modal-body { |
||||
height: calc(100% - 55px); |
||||
} |
||||
} |
||||
|
||||
&.no-title.no-footer { |
||||
.ant-modal-body { |
||||
height: 100%; |
||||
} |
||||
} |
||||
|
||||
} |
||||
|
||||
.j-modal-title-row { |
||||
.left { |
||||
width: calc(100% - 56px - 56px); |
||||
} |
||||
|
||||
.right { |
||||
width: 56px; |
||||
position: inherit; |
||||
|
||||
.ant-modal-close { |
||||
right: 56px; |
||||
color: rgba(0, 0, 0, 0.45); |
||||
|
||||
&:hover { |
||||
color: rgba(0, 0, 0, 0.75); |
||||
} |
||||
|
||||
} |
||||
} |
||||
} |
||||
|
||||
|
||||
} |
||||
|
||||
@media (max-width: 767px) { |
||||
.j-modal-box.fullscreen { |
||||
margin: 0; |
||||
max-width: 100vw; |
||||
} |
||||
} |
||||
</style> |
@ -0,0 +1,66 @@
|
||||
<template> |
||||
<a-select :value="arrayValue" @change="onChange" mode="multiple" :placeholder="placeholder"> |
||||
<a-select-option |
||||
v-for="(item,index) in options" |
||||
:key="index" |
||||
:value="item.value"> |
||||
{{ item.text || item.label }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</template> |
||||
|
||||
<script> |
||||
//option {label:,value:} |
||||
export default { |
||||
name: 'JSelectMultiple', |
||||
props: { |
||||
placeholder:{ |
||||
type: String, |
||||
default:'', |
||||
required: false |
||||
}, |
||||
value:{ |
||||
type: String, |
||||
required: false |
||||
}, |
||||
readOnly:{ |
||||
type: Boolean, |
||||
required: false, |
||||
default: false |
||||
}, |
||||
options:{ |
||||
type: Array, |
||||
required: true |
||||
}, |
||||
triggerChange:{ |
||||
type: Boolean, |
||||
required: false, |
||||
default: false |
||||
} |
||||
}, |
||||
data(){ |
||||
return { |
||||
arrayValue:!this.value?[]:this.value.split(",") |
||||
} |
||||
}, |
||||
watch:{ |
||||
value (val) { |
||||
if(!val){ |
||||
this.arrayValue = [] |
||||
}else{ |
||||
this.arrayValue = this.value.split(",") |
||||
} |
||||
} |
||||
}, |
||||
methods:{ |
||||
onChange (selectedValue) { |
||||
if(this.triggerChange){ |
||||
this.$emit('change', selectedValue.join(",")); |
||||
}else{ |
||||
this.$emit('input', selectedValue.join(",")); |
||||
} |
||||
}, |
||||
}, |
||||
|
||||
} |
||||
</script> |
@ -0,0 +1,116 @@
|
||||
<template> |
||||
<div class="drag" ref="dragDiv"> |
||||
<div class="drag_bg"></div> |
||||
<div class="drag_text">{{confirmWords}}</div> |
||||
<div ref="moveDiv" @mousedown="mousedownFn($event)" :class="{'handler_ok_bg':confirmSuccess}" class="handler handler_bg" style="border: 0.5px solid #fff;height: 34px;position: absolute;top: 0px;left: 0px;"></div> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
export default { |
||||
name:"JSlider", |
||||
data(){ |
||||
return { |
||||
beginClientX:0, /*距离屏幕左端距离*/ |
||||
mouseMoveStata:false, /*触发拖动状态 判断*/ |
||||
maxwidth:'', /*拖动最大宽度,依据滑块宽度算出来的*/ |
||||
confirmWords:'拖动滑块验证', /*滑块文字*/ |
||||
confirmSuccess:false /*验证成功判断*/ |
||||
} |
||||
}, |
||||
methods: { |
||||
isSuccess(){ |
||||
return this.confirmSuccess |
||||
}, |
||||
mousedownFn:function (e) { |
||||
if(!this.confirmSuccess){ |
||||
e.preventDefault && e.preventDefault(); //阻止文字选中等 浏览器默认事件 |
||||
this.mouseMoveStata = true; |
||||
this.beginClientX = e.clientX; |
||||
} |
||||
}, //mousedoen 事件 |
||||
successFunction(){ |
||||
this.confirmSuccess = true |
||||
this.confirmWords = '验证通过'; |
||||
if(window.addEventListener){ |
||||
document.getElementsByTagName('html')[0].removeEventListener('mousemove',this.mouseMoveFn); |
||||
document.getElementsByTagName('html')[0].removeEventListener('mouseup',this.moseUpFn); |
||||
}else { |
||||
document.getElementsByTagName('html')[0].removeEventListener('mouseup',()=>{}); |
||||
} |
||||
document.getElementsByClassName('drag_text')[0].style.color = '#fff' |
||||
document.getElementsByClassName('handler')[0].style.left = this.maxwidth + 'px'; |
||||
document.getElementsByClassName('drag_bg')[0].style.width = this.maxwidth + 'px'; |
||||
|
||||
this.$emit("onSuccess",true) |
||||
}, //验证成功函数 |
||||
mouseMoveFn(e){ |
||||
if(this.mouseMoveStata){ |
||||
let width = e.clientX - this.beginClientX; |
||||
if(width>0 && width<=this.maxwidth){ |
||||
document.getElementsByClassName('handler')[0].style.left = width + 'px'; |
||||
document.getElementsByClassName('drag_bg')[0].style.width = width + 'px'; |
||||
}else if(width>this.maxwidth){ |
||||
this.successFunction(); |
||||
} |
||||
} |
||||
}, //mousemove事件 |
||||
moseUpFn(e){ |
||||
this.mouseMoveStata = false; |
||||
var width = e.clientX - this.beginClientX; |
||||
if(width<this.maxwidth){ |
||||
// ---- update-begin- author:sunjianlei --- date:20191009 --- for: 修复获取不到 handler 的时候报错 ---- |
||||
let handler = document.getElementsByClassName('handler')[0] |
||||
if (handler) { |
||||
handler.style.left = 0 + 'px' |
||||
document.getElementsByClassName('drag_bg')[0].style.width = 0 + 'px' |
||||
} |
||||
// ---- update-end- author:sunjianlei --- date:20191009 --- for: 修复获取不到 handler 的时候报错 ---- |
||||
} |
||||
} //mouseup事件 |
||||
}, |
||||
mounted(){ |
||||
this.maxwidth = this.$refs.dragDiv.clientWidth - this.$refs.moveDiv.clientWidth; |
||||
document.getElementsByTagName('html')[0].addEventListener('mousemove',this.mouseMoveFn); |
||||
document.getElementsByTagName('html')[0].addEventListener('mouseup',this.moseUpFn) |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
.drag{ |
||||
position: relative; |
||||
background-color: #e8e8e8; |
||||
width: 100%; |
||||
height: 34px; |
||||
line-height: 34px; |
||||
text-align: center; |
||||
} |
||||
.handler{ |
||||
width: 40px; |
||||
height: 32px; |
||||
border: 1px solid #ccc; |
||||
cursor: move; |
||||
} |
||||
.handler_bg{ |
||||
background: #fff url("") no-repeat center; |
||||
} |
||||
.handler_ok_bg{ |
||||
background: #fff url("") no-repeat center; |
||||
} |
||||
.drag_bg{ |
||||
background-color: #7ac23c; |
||||
height: 34px; |
||||
width: 0px; |
||||
} |
||||
.drag_text{ |
||||
position: absolute; |
||||
top: 0px; |
||||
width: 100%;text-align: center; |
||||
-moz-user-select: none; |
||||
-webkit-user-select: none; |
||||
user-select: none; |
||||
-o-user-select:none; |
||||
-ms-user-select:none; |
||||
} |
||||
</style> |
@ -0,0 +1,638 @@
|
||||
<template> |
||||
<div class="j-super-query-box"> |
||||
|
||||
<slot name="button" :isActive="superQueryFlag" :isMobile="izMobile" :open="handleOpen" :reset="handleReset"> |
||||
<a-tooltip v-if="superQueryFlag" v-bind="tooltipProps" :mouseLeaveDelay="0.2"> |
||||
<!-- begin 不知道为什么不加上这段代码就无法生效 --> |
||||
<span v-show="false">{{tooltipProps}}</span> |
||||
<!-- end 不知道为什么不加上这段代码就无法生效 --> |
||||
<template slot="title"> |
||||
<span>已有高级查询条件生效</span> |
||||
<a-divider type="vertical"/> |
||||
<a @click="handleReset">清空</a> |
||||
</template> |
||||
<a-button-group> |
||||
<a-button type="primary" @click="handleOpen"> |
||||
<a-icon type="appstore" theme="twoTone" spin/> |
||||
<span>高级查询</span> |
||||
</a-button> |
||||
<a-button v-if="izMobile" type="primary" icon="delete" @click="handleReset"/> |
||||
</a-button-group> |
||||
</a-tooltip> |
||||
<a-button v-else type="primary" icon="filter" @click="handleOpen">高级查询</a-button> |
||||
</slot> |
||||
|
||||
<j-modal |
||||
title="高级查询构造器" |
||||
:width="1000" |
||||
:visible="visible" |
||||
@cancel="handleCancel" |
||||
:mask="false" |
||||
:fullscreen="izMobile" |
||||
class="j-super-query-modal" |
||||
style="top:5%;max-height: 95%;" |
||||
> |
||||
|
||||
<template slot="footer"> |
||||
<div style="float: left"> |
||||
<a-button :loading="loading" @click="handleReset">重置</a-button> |
||||
<a-button :loading="loading" @click="handleSave">保存查询条件</a-button> |
||||
</div> |
||||
<a-button :loading="loading" @click="handleCancel">关闭</a-button> |
||||
<a-button :loading="loading" type="primary" @click="handleOk">查询</a-button> |
||||
</template> |
||||
|
||||
<a-spin :spinning="loading"> |
||||
<a-row> |
||||
<a-col :sm="24" :md="24-5"> |
||||
|
||||
<a-empty v-if="queryParamsModel.length === 0" style="margin-bottom: 12px;"> |
||||
<div slot="description"> |
||||
<span>没有任何查询条件</span> |
||||
<a-divider type="vertical"/> |
||||
<a @click="handleAdd">点击新增</a> |
||||
</div> |
||||
</a-empty> |
||||
|
||||
<a-form v-else layout="inline"> |
||||
|
||||
<a-row style="margin-bottom: 12px;"> |
||||
<a-col :md="12" :xs="24"> |
||||
<a-form-item label="过滤条件匹配" :labelCol="{md: 6,xs:24}" :wrapperCol="{md: 18,xs:24}" style="width: 100%;"> |
||||
<a-select v-model="matchType" :getPopupContainer="node=>node.parentNode" style="width: 100%;"> |
||||
<a-select-option value="and">AND(所有条件都要求匹配)</a-select-option> |
||||
<a-select-option value="or">OR(条件中的任意一个匹配)</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
|
||||
<a-row type="flex" style="margin-bottom:10px" :gutter="16" v-for="(item, index) in queryParamsModel" :key="index"> |
||||
|
||||
<a-col :md="8" :xs="24" style="margin-bottom: 12px;"> |
||||
<a-tree-select |
||||
showSearch |
||||
v-model="item.field" |
||||
:treeData="fieldTreeData" |
||||
:dropdownStyle="{ maxHeight: '400px', overflow: 'auto' }" |
||||
placeholder="选择查询字段" |
||||
allowClear |
||||
treeDefaultExpandAll |
||||
:getPopupContainer="node=>node.parentNode" |
||||
style="width: 100%" |
||||
@select="(val,option)=>handleSelected(option,item)" |
||||
> |
||||
</a-tree-select> |
||||
</a-col> |
||||
|
||||
<a-col :md="4" :xs="24" style="margin-bottom: 12px;"> |
||||
<a-select placeholder="匹配规则" :value="item.rule" :getPopupContainer="node=>node.parentNode" @change="handleRuleChange(item,$event)"> |
||||
<a-select-option value="eq">等于</a-select-option> |
||||
<a-select-option value="like">包含</a-select-option> |
||||
<a-select-option value="right_like">以..开始</a-select-option> |
||||
<a-select-option value="left_like">以..结尾</a-select-option> |
||||
<a-select-option value="in">在...中</a-select-option> |
||||
<a-select-option value="ne">不等于</a-select-option> |
||||
<a-select-option value="gt">大于</a-select-option> |
||||
<a-select-option value="ge">大于等于</a-select-option> |
||||
<a-select-option value="lt">小于</a-select-option> |
||||
<a-select-option value="le">小于等于</a-select-option> |
||||
</a-select> |
||||
</a-col> |
||||
|
||||
<a-col :md="8" :xs="24" style="margin-bottom: 12px;"> |
||||
<template v-if="item.dictCode"> |
||||
<template v-if="item.type === 'table-dict'"> |
||||
<j-popup |
||||
v-model="item.val" |
||||
:code="item.dictTable" |
||||
:field="item.dictCode" |
||||
:orgFields="item.dictCode" |
||||
:destFields="item.dictCode" |
||||
></j-popup> |
||||
</template> |
||||
<template v-else> |
||||
<j-multi-select-tag v-show="allowMultiple(item)" v-model="item.val" :dictCode="item.dictCode" placeholder="请选择"/> |
||||
<j-dict-select-tag v-show="!allowMultiple(item)" v-model="item.val" :dictCode="item.dictCode" placeholder="请选择"/> |
||||
</template> |
||||
</template> |
||||
<j-popup v-else-if="item.type === 'popup'" :value="item.val" v-bind="item.popup" group-id="superQuery" @input="(e,v)=>handleChangeJPopup(item,e,v)"/> |
||||
<j-select-multi-user |
||||
v-else-if="item.type === 'select-user' || item.type === 'sel_user'" |
||||
v-model="item.val" |
||||
:buttons="false" |
||||
:multiple="false" |
||||
placeholder="请选择用户" |
||||
:returnKeys="['id', item.customReturnField || 'username']" |
||||
/> |
||||
<j-select-depart |
||||
v-else-if="item.type === 'select-depart' || item.type === 'sel_depart'" |
||||
v-model="item.val" |
||||
:multi="false" |
||||
placeholder="请选择部门" |
||||
:customReturnField="item.customReturnField || 'id'" |
||||
/> |
||||
<a-select |
||||
v-else-if="item.options instanceof Array" |
||||
v-model="item.val" |
||||
:options="item.options" |
||||
allowClear |
||||
placeholder="请选择" |
||||
:mode="allowMultiple(item)?'multiple':''" |
||||
/> |
||||
<j-area-linkage v-model="item.val" v-else-if="item.type==='area-linkage' || item.type==='pca'" style="width: 100%"/> |
||||
<j-date v-else-if=" item.type=='date' " v-model="item.val" placeholder="请选择日期" style="width: 100%"></j-date> |
||||
<j-date v-else-if=" item.type=='datetime' " v-model="item.val" placeholder="请选择时间" :show-time="true" date-format="YYYY-MM-DD HH:mm:ss" style="width: 100%"></j-date> |
||||
<a-time-picker v-else-if="item.type==='time'" :value="item.val ? moment(item.val,'HH:mm:ss') : null" format="HH:mm:ss" style="width: 100%" @change="(time,value)=>item.val=value"/> |
||||
<a-input-number v-else-if=" item.type=='int'||item.type=='number' " style="width: 100%" placeholder="请输入数值" v-model="item.val"/> |
||||
<a-input v-else v-model="item.val" placeholder="请输入值"/> |
||||
</a-col> |
||||
|
||||
<a-col :md="4" :xs="0" style="margin-bottom: 12px;"> |
||||
<a-button @click="handleAdd" icon="plus"></a-button> |
||||
<a-button @click="handleDel( index )" icon="minus"></a-button> |
||||
</a-col> |
||||
|
||||
<a-col :md="0" :xs="24" style="margin-bottom: 12px;text-align: right;"> |
||||
<a-button @click="handleAdd" icon="plus"></a-button> |
||||
<a-button @click="handleDel( index )" icon="minus"></a-button> |
||||
</a-col> |
||||
|
||||
</a-row> |
||||
|
||||
</a-form> |
||||
</a-col> |
||||
<a-col :sm="24" :md="5"> |
||||
<!-- 查询记录 --> |
||||
|
||||
<a-card class="j-super-query-history-card" :bordered="true"> |
||||
<div slot="title"> |
||||
保存的查询 |
||||
</div> |
||||
|
||||
<a-empty v-if="saveTreeData.length === 0" class="j-super-query-history-empty" description="没有保存任何查询"/> |
||||
<a-tree |
||||
v-else |
||||
class="j-super-query-history-tree" |
||||
showIcon |
||||
:treeData="saveTreeData" |
||||
:selectedKeys="[]" |
||||
@select="handleTreeSelect" |
||||
> |
||||
</a-tree> |
||||
</a-card> |
||||
|
||||
|
||||
</a-col> |
||||
</a-row> |
||||
|
||||
|
||||
</a-spin> |
||||
|
||||
<a-modal title="请输入保存的名称" :visible="prompt.visible" @cancel="prompt.visible=false" @ok="handlePromptOk"> |
||||
<a-input v-model="prompt.value"></a-input> |
||||
</a-modal> |
||||
|
||||
</j-modal> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
import moment from 'moment' |
||||
import * as utils from '@/utils/util' |
||||
import JDate from '@/components/jeecg/JDate.vue' |
||||
import JSelectDepart from '@/components/jeecgbiz/JSelectDepart' |
||||
import JSelectMultiUser from '@/components/jeecgbiz/JSelectMultiUser' |
||||
import JAreaLinkage from '@comp/jeecg/JAreaLinkage' |
||||
|
||||
export default { |
||||
name: 'JSuperQuery', |
||||
// mixins: [mixinDevice], |
||||
components: { JAreaLinkage, JDate, JSelectDepart, JSelectMultiUser }, |
||||
props: { |
||||
/* |
||||
fieldList: [{ |
||||
value:'', |
||||
text:'', |
||||
type:'', |
||||
dictCode:'' // 只要 dictCode 有值,无论 type 是什么,都显示为字典下拉框 |
||||
}] |
||||
type:date datetime int number string |
||||
* */ |
||||
fieldList: { |
||||
type: Array, |
||||
required: true |
||||
}, |
||||
/* |
||||
* 这个回调函数接收一个数组参数 即查询条件 |
||||
* */ |
||||
callback: { |
||||
type: String, |
||||
required: false, |
||||
default: 'handleSuperQuery' |
||||
}, |
||||
|
||||
// 当前是否在加载中 |
||||
loading: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
|
||||
// 保存查询条件的唯一 code,通过该 code 区分 |
||||
// 默认为 null,代表以当前路由全路径为区分Code |
||||
saveCode: { |
||||
type: String, |
||||
default: null |
||||
} |
||||
|
||||
}, |
||||
data() { |
||||
return { |
||||
moment, |
||||
fieldTreeData: [], |
||||
|
||||
prompt: { |
||||
visible: false, |
||||
value: '' |
||||
}, |
||||
|
||||
visible: false, |
||||
queryParamsModel: [], |
||||
treeIcon: <a-icon type="file-text"/>, |
||||
// 保存查询条件的treeData |
||||
saveTreeData: [], |
||||
// 保存查询条件的前缀名 |
||||
saveCodeBefore: 'JSuperQuerySaved_', |
||||
// 查询类型,过滤条件匹配(and、or) |
||||
matchType: 'and', |
||||
superQueryFlag: false, |
||||
} |
||||
}, |
||||
computed: { |
||||
izMobile() { |
||||
return this.device === 'mobile' |
||||
}, |
||||
tooltipProps() { |
||||
return this.izMobile ? { visible: false } : {} |
||||
}, |
||||
fullSaveCode() { |
||||
let saveCode = this.saveCode |
||||
if (saveCode == null || saveCode === '') { |
||||
saveCode = this.$route.fullPath |
||||
} |
||||
return this.saveCodeBefore + saveCode |
||||
}, |
||||
}, |
||||
watch: { |
||||
// 当 saveCode 变化时,重新查询已保存的条件 |
||||
fullSaveCode: { |
||||
immediate: true, |
||||
handler() { |
||||
let list = this.$ls.get(this.fullSaveCode) |
||||
if (list instanceof Array) { |
||||
this.saveTreeData = list.map(i => this.renderSaveTreeData(i)) |
||||
} |
||||
} |
||||
}, |
||||
fieldList: { |
||||
deep: true, |
||||
immediate: true, |
||||
handler(val) { |
||||
let mainData = [], subData = [] |
||||
val.forEach(item => { |
||||
let data = { ...item } |
||||
data.label = data.label || data.text |
||||
let hasChildren = (data.children instanceof Array) |
||||
data.disabled = hasChildren |
||||
data.selectable = !hasChildren |
||||
if (hasChildren) { |
||||
data.children = data.children.map(item2 => { |
||||
let child = { ...item2 } |
||||
child.label = child.label || child.text |
||||
child.label = data.label + '-' + child.label |
||||
child.value = data.value + ',' + child.value |
||||
child.val = '' |
||||
return child |
||||
}) |
||||
data.val = '' |
||||
subData.push(data) |
||||
} else { |
||||
mainData.push(data) |
||||
} |
||||
}) |
||||
this.fieldTreeData = mainData.concat(subData) |
||||
} |
||||
} |
||||
}, |
||||
|
||||
methods: { |
||||
show() { |
||||
if (!this.queryParamsModel || this.queryParamsModel.length === 0) { |
||||
this.resetLine() |
||||
} |
||||
this.visible = true |
||||
}, |
||||
handleOk() { |
||||
if (!this.isNullArray(this.queryParamsModel)) { |
||||
let event = { |
||||
matchType: this.matchType, |
||||
params: this.removeEmptyObject(this.queryParamsModel) |
||||
} |
||||
// 移动端模式下关闭弹窗 |
||||
if (this.izMobile) { |
||||
this.visible = false |
||||
} |
||||
this.emitCallback(event) |
||||
} else { |
||||
this.$message.warn("不能查询空条件") |
||||
} |
||||
}, |
||||
emitCallback(event = {}) { |
||||
let { params = [], matchType = this.matchType } = event |
||||
this.superQueryFlag = (params && params.length > 0) |
||||
for (let param of params) { |
||||
if (Array.isArray(param.val)) { |
||||
param.val = param.val.join(',') |
||||
} |
||||
} |
||||
console.debug('---高级查询参数--->', { params, matchType }) |
||||
this.$emit(this.callback, params, matchType) |
||||
}, |
||||
handleCancel() { |
||||
this.close() |
||||
}, |
||||
close() { |
||||
this.$emit('close') |
||||
this.visible = false |
||||
}, |
||||
handleAdd() { |
||||
this.addNewLine() |
||||
}, |
||||
addNewLine() { |
||||
this.queryParamsModel.push({ rule: 'eq' }) |
||||
}, |
||||
resetLine() { |
||||
this.superQueryFlag = false |
||||
this.queryParamsModel = [] |
||||
this.addNewLine() |
||||
}, |
||||
handleDel(index) { |
||||
this.queryParamsModel.splice(index, 1) |
||||
}, |
||||
handleSelected(node, item) { |
||||
let { type, options, dictCode, dictTable, customReturnField, popup } = node.dataRef |
||||
item['type'] = type |
||||
item['options'] = options |
||||
item['dictCode'] = dictCode |
||||
item['dictTable'] = dictTable |
||||
item['customReturnField'] = customReturnField |
||||
if (popup) { |
||||
item['popup'] = popup |
||||
} |
||||
this.$set(item, 'val', undefined) |
||||
}, |
||||
handleOpen() { |
||||
this.show() |
||||
}, |
||||
handleReset() { |
||||
this.resetLine() |
||||
this.emitCallback() |
||||
}, |
||||
handleSave() { |
||||
let queryParams = this.removeEmptyObject(this.queryParamsModel) |
||||
if (this.isNullArray(queryParams)) { |
||||
this.$message.warning('空条件不能保存') |
||||
} else { |
||||
this.prompt.value = '' |
||||
this.prompt.visible = true |
||||
} |
||||
}, |
||||
handlePromptOk() { |
||||
let { value } = this.prompt |
||||
if(!value){ |
||||
this.$message.warning('保存名称不能为空') |
||||
return |
||||
} |
||||
// 取出查询条件 |
||||
let records = this.removeEmptyObject(this.queryParamsModel) |
||||
// 判断有没有重名的 |
||||
let filterList = this.saveTreeData.filter(i => i.originTitle === value) |
||||
if (filterList.length > 0) { |
||||
this.$confirm({ |
||||
content: `${value} 已存在,是否覆盖?`, |
||||
onOk: () => { |
||||
this.prompt.visible = false |
||||
filterList[0].records = records |
||||
this.saveToLocalStore() |
||||
this.$message.success('保存成功') |
||||
} |
||||
}) |
||||
} else { |
||||
// 没有重名的,直接添加 |
||||
this.prompt.visible = false |
||||
// 添加到树列表中 |
||||
this.saveTreeData.push(this.renderSaveTreeData({ |
||||
title: value, |
||||
matchType: this.matchType, |
||||
records: records |
||||
})) |
||||
// 保存到 LocalStore |
||||
this.saveToLocalStore() |
||||
this.$message.success('保存成功') |
||||
} |
||||
}, |
||||
handleTreeSelect(idx, event) { |
||||
if (event.selectedNodes[0]) { |
||||
let { matchType, records } = event.selectedNodes[0].data.props |
||||
// 将保存的matchType取出,兼容旧数据,如果没有保存就还是使用原来的 |
||||
this.matchType = matchType || this.matchType |
||||
this.queryParamsModel = utils.cloneObject(records) |
||||
} |
||||
}, |
||||
handleRemoveSaveTreeItem(event, vNode) { |
||||
// 阻止事件冒泡 |
||||
event.stopPropagation() |
||||
|
||||
this.$confirm({ |
||||
content: '是否删除当前查询?', |
||||
onOk: () => { |
||||
let { eventKey } = vNode |
||||
this.saveTreeData.splice(Number.parseInt(eventKey.substring(2)), 1) |
||||
this.saveToLocalStore() |
||||
}, |
||||
}) |
||||
}, |
||||
|
||||
// 将查询保存到 LocalStore 里 |
||||
saveToLocalStore() { |
||||
let saveValue = this.saveTreeData.map(({ originTitle, matchType, records }) => ({ title: originTitle, matchType, records })) |
||||
this.$ls.set(this.fullSaveCode, saveValue) |
||||
}, |
||||
|
||||
isNullArray(array) { |
||||
//判断是不是空数组对象 |
||||
if (!array || array.length === 0) { |
||||
return true |
||||
} |
||||
if (array.length === 1) { |
||||
let obj = array[0] |
||||
if (!obj.field || (obj.val == null || obj.val === '') || !obj.rule) { |
||||
return true |
||||
} |
||||
} |
||||
return false |
||||
}, |
||||
// 去掉数组中的空对象 |
||||
removeEmptyObject(arr) { |
||||
let array = utils.cloneObject(arr) |
||||
for (let i = 0; i < array.length; i++) { |
||||
let item = array[i] |
||||
if (item == null || Object.keys(item).length <= 0) { |
||||
array.splice(i--, 1) |
||||
} else { |
||||
if (Array.isArray(item.options)) { |
||||
// 如果有字典属性,就不需要保存 options 了 |
||||
if (item.dictCode) { |
||||
// 去掉特殊属性 |
||||
delete item.options |
||||
} |
||||
} |
||||
} |
||||
} |
||||
return array |
||||
}, |
||||
|
||||
/** 渲染保存查询条件的 title(加个删除按钮) */ |
||||
renderSaveTreeData(item) { |
||||
item.icon = this.treeIcon |
||||
item.originTitle = item['title'] |
||||
item.title = (fn, vNode) => { |
||||
let { originTitle } = vNode.dataRef |
||||
return ( |
||||
<div class="j-history-tree-title"> |
||||
<span>{originTitle}</span> |
||||
|
||||
<div class="j-history-tree-title-closer" onClick={e => this.handleRemoveSaveTreeItem(e, vNode)}> |
||||
<a-icon type="close-circle"/> |
||||
</div> |
||||
</div> |
||||
) |
||||
} |
||||
return item |
||||
}, |
||||
|
||||
/** 判断是否允许多选 */ |
||||
allowMultiple(item) { |
||||
return item.rule === 'in' |
||||
}, |
||||
|
||||
handleRuleChange(item, newValue) { |
||||
let oldValue = item.rule |
||||
this.$set(item, 'rule', newValue) |
||||
// 上一个规则是否是 in,且type是字典或下拉 |
||||
if (oldValue === 'in') { |
||||
if (item.dictCode || item.options instanceof Array) { |
||||
let value = item.val |
||||
if (typeof item.val === 'string') { |
||||
value = item.val.split(',')[0] |
||||
} else if (Array.isArray(item.val)) { |
||||
value = item.val[0] |
||||
} |
||||
this.$set(item, 'val', value) |
||||
} |
||||
} |
||||
}, |
||||
|
||||
handleChangeJPopup(item, e, values) { |
||||
item.val = values[item.popup['destFields']] |
||||
}, |
||||
|
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="less" scoped> |
||||
|
||||
.j-super-query-box { |
||||
display: inline-block; |
||||
} |
||||
|
||||
.j-super-query-modal { |
||||
|
||||
.j-super-query-history-card { |
||||
/deep/ .ant-card-body, |
||||
/deep/ .ant-card-head-title { |
||||
padding: 0; |
||||
} |
||||
|
||||
/deep/ .ant-card-head { |
||||
padding: 4px 8px; |
||||
min-height: initial; |
||||
} |
||||
} |
||||
|
||||
.j-super-query-history-empty { |
||||
/deep/ .ant-empty-image { |
||||
height: 80px; |
||||
line-height: 80px; |
||||
margin-bottom: 0; |
||||
} |
||||
|
||||
/deep/ img { |
||||
width: 80px; |
||||
height: 65px; |
||||
} |
||||
|
||||
/deep/ .ant-empty-description { |
||||
color: #afafaf; |
||||
margin: 8px 0; |
||||
} |
||||
} |
||||
|
||||
.j-super-query-history-tree { |
||||
|
||||
.j-history-tree-title { |
||||
width: calc(100% - 24px); |
||||
position: relative; |
||||
display: inline-block; |
||||
|
||||
&-closer { |
||||
color: #999999; |
||||
position: absolute; |
||||
top: 0; |
||||
right: 0; |
||||
width: 24px; |
||||
height: 24px; |
||||
text-align: center; |
||||
opacity: 0; |
||||
transition: opacity 0.3s, color 0.3s; |
||||
|
||||
&:hover { |
||||
color: #666666; |
||||
} |
||||
|
||||
&:active { |
||||
color: #333333; |
||||
} |
||||
} |
||||
|
||||
&:hover { |
||||
.j-history-tree-title-closer { |
||||
opacity: 1; |
||||
} |
||||
} |
||||
|
||||
} |
||||
|
||||
/deep/ .ant-tree-switcher { |
||||
display: none; |
||||
} |
||||
|
||||
/deep/ .ant-tree-node-content-wrapper { |
||||
width: 100%; |
||||
} |
||||
} |
||||
|
||||
} |
||||
|
||||
</style> |
@ -0,0 +1,57 @@
|
||||
<template> |
||||
<a-switch v-model="checkStatus" :disabled="disabled" @change="handleChange"/> |
||||
</template> |
||||
<script> |
||||
|
||||
export default { |
||||
name: 'JSwitch', |
||||
props: { |
||||
value:{ |
||||
type: String, |
||||
required: false |
||||
}, |
||||
disabled:{ |
||||
type: Boolean, |
||||
required: false, |
||||
default: false |
||||
}, |
||||
options:{ |
||||
type:Array, |
||||
required:false, |
||||
default:()=>['Y','N'] |
||||
} |
||||
}, |
||||
data () { |
||||
return { |
||||
checkStatus: false |
||||
} |
||||
}, |
||||
watch: { |
||||
value:{ |
||||
immediate: true, |
||||
handler(val){ |
||||
if(!val){ |
||||
this.checkStatus = false |
||||
this.$emit('change', this.options[1]); |
||||
}else{ |
||||
if(this.options[0]==val){ |
||||
this.checkStatus = true |
||||
}else{ |
||||
this.checkStatus = false |
||||
} |
||||
} |
||||
} |
||||
} |
||||
}, |
||||
methods: { |
||||
handleChange(checked){ |
||||
let flag = checked===false?this.options[1]:this.options[0]; |
||||
this.$emit('change', flag); |
||||
} |
||||
}, |
||||
model: { |
||||
prop: 'value', |
||||
event: 'change' |
||||
} |
||||
} |
||||
</script> |
@ -0,0 +1,199 @@
|
||||
<template> |
||||
<a-tree-select |
||||
allowClear |
||||
labelInValue |
||||
style="width: 100%" |
||||
:disabled="disabled" |
||||
:dropdownStyle="{ maxHeight: '400px', overflow: 'auto' }" |
||||
:placeholder="placeholder" |
||||
:loadData="asyncLoadTreeData" |
||||
:value="treeValue" |
||||
:treeData="treeData" |
||||
@change="onChange" |
||||
@search="onSearch"> |
||||
</a-tree-select> |
||||
</template> |
||||
|
||||
<script> |
||||
import { getAction } from '@/api/manage' |
||||
|
||||
export default { |
||||
name: 'JTreeDict', |
||||
data(){ |
||||
return { |
||||
treeData:[], |
||||
treeValue: null, |
||||
url_root:"/sys/category/loadTreeRoot", |
||||
url_children:"/sys/category/loadTreeChildren", |
||||
url_view:'/sys/category/loadOne', |
||||
} |
||||
}, |
||||
props:{ |
||||
value:{ |
||||
type: String, |
||||
required: false |
||||
}, |
||||
placeholder:{ |
||||
type: String, |
||||
default: '请选择', |
||||
required: false |
||||
}, |
||||
parentCode:{ |
||||
type: String, |
||||
default: '', |
||||
required: false |
||||
}, |
||||
field:{ |
||||
type: String, |
||||
default: 'id', |
||||
required: false |
||||
}, |
||||
root:{ |
||||
type:Object, |
||||
required:false, |
||||
default:()=>{ |
||||
return { |
||||
pid:'0' |
||||
} |
||||
} |
||||
}, |
||||
async:{ |
||||
type:Boolean, |
||||
default:false, |
||||
required:false |
||||
}, |
||||
disabled:{ |
||||
type:Boolean, |
||||
default:false, |
||||
required:false |
||||
} |
||||
}, |
||||
watch:{ |
||||
root:{ |
||||
handler(val){ |
||||
console.log("root-change",val) |
||||
}, |
||||
deep:true |
||||
}, |
||||
parentCode:{ |
||||
handler(){ |
||||
this.loadRoot() |
||||
} |
||||
}, |
||||
value:{ |
||||
handler(){ |
||||
this.loadViewInfo() |
||||
} |
||||
} |
||||
}, |
||||
created(){ |
||||
this.loadRoot() |
||||
this.loadViewInfo() |
||||
}, |
||||
model: { |
||||
prop: 'value', |
||||
event: 'change' |
||||
}, |
||||
methods:{ |
||||
loadViewInfo(){ |
||||
if(!this.value || this.value=="0"){ |
||||
this.treeValue = null |
||||
}else{ |
||||
let param = { |
||||
field:this.field, |
||||
val:this.value |
||||
} |
||||
getAction(this.url_view,param).then(res=>{ |
||||
if(res.success){ |
||||
this.treeValue = { |
||||
value:this.value, |
||||
label:res.result.name |
||||
} |
||||
} |
||||
}) |
||||
} |
||||
}, |
||||
loadRoot(){ |
||||
let param = { |
||||
async:this.async, |
||||
pcode:this.parentCode |
||||
} |
||||
getAction(this.url_root,param).then(res=>{ |
||||
if(res.success){ |
||||
this.handleTreeNodeValue(res.result) |
||||
console.log("aaaa",res.result) |
||||
this.treeData = [...res.result] |
||||
}else{ |
||||
this.$message.error(res.message) |
||||
} |
||||
}) |
||||
}, |
||||
asyncLoadTreeData (treeNode) { |
||||
return new Promise((resolve) => { |
||||
if(!this.async){ |
||||
resolve() |
||||
return |
||||
} |
||||
if (treeNode.$vnode.children) { |
||||
resolve() |
||||
return |
||||
} |
||||
let pid = treeNode.$vnode.key |
||||
let param = { |
||||
pid:pid |
||||
} |
||||
getAction(this.url_children,param).then(res=>{ |
||||
if(res.success){ |
||||
this.handleTreeNodeValue(res.result) |
||||
this.addChildren(pid,res.result,this.treeData) |
||||
this.treeData = [...this.treeData] |
||||
} |
||||
resolve() |
||||
}) |
||||
}) |
||||
}, |
||||
addChildren(pid,children,treeArray){ |
||||
if(treeArray && treeArray.length>0){ |
||||
for(let item of treeArray){ |
||||
if(item.key == pid){ |
||||
if(!children || children.length==0){ |
||||
item.leaf = true |
||||
}else{ |
||||
item.children = children |
||||
} |
||||
break |
||||
}else{ |
||||
this.addChildren(pid,children,item.children) |
||||
} |
||||
} |
||||
} |
||||
}, |
||||
handleTreeNodeValue(result){ |
||||
let storeField = this.field=='code'?'code':'key' |
||||
for(let i of result){ |
||||
i.value = i[storeField] |
||||
i.isLeaf = (!i.leaf)?false:true |
||||
if(i.children && i.children.length>0){ |
||||
this.handleTreeNodeValue(i.children) |
||||
} |
||||
} |
||||
}, |
||||
onChange(value){ |
||||
console.log(value) |
||||
if(!value){ |
||||
this.$emit('change', ''); |
||||
}else{ |
||||
this.$emit('change', value.value); |
||||
} |
||||
this.treeValue = value |
||||
}, |
||||
onSearch(value){ |
||||
console.log(value) |
||||
}, |
||||
getCurrTreeData(){ |
||||
return this.treeData |
||||
} |
||||
} |
||||
|
||||
} |
||||
</script> |
@ -0,0 +1,260 @@
|
||||
<template> |
||||
<a-tree-select |
||||
allowClear |
||||
labelInValue |
||||
:getPopupContainer="(node) => node.parentNode" |
||||
style="width: 100%" |
||||
:disabled="disabled" |
||||
:dropdownStyle="{ maxHeight: '400px', overflow: 'auto' }" |
||||
:placeholder="placeholder" |
||||
:loadData="asyncLoadTreeData" |
||||
:value="treeValue" |
||||
:treeData="treeData" |
||||
:multiple="multiple" |
||||
@change="onChange" |
||||
@search="onSearch"> |
||||
</a-tree-select> |
||||
</template> |
||||
<script> |
||||
|
||||
/* |
||||
* 异步树加载组件 通过传入表名 显示字段 存储字段 加载一个树控件 |
||||
* <j-tree-select dict="aa_tree_test,aad,id" pid-field="pid" ></j-tree-select> |
||||
* */ |
||||
import { getAction } from '@/api/manage' |
||||
|
||||
export default { |
||||
name: 'JTreeSelect', |
||||
props: { |
||||
value:{ |
||||
type: String, |
||||
required: false |
||||
}, |
||||
placeholder:{ |
||||
type: String, |
||||
default: '请选择', |
||||
required: false |
||||
}, |
||||
dict:{ |
||||
type: String, |
||||
default: '', |
||||
required: false |
||||
}, |
||||
pidField:{ |
||||
type: String, |
||||
default: 'pid', |
||||
required: false |
||||
}, |
||||
pidValue:{ |
||||
type: String, |
||||
default: '', |
||||
required: false |
||||
}, |
||||
disabled:{ |
||||
type:Boolean, |
||||
default:false, |
||||
required:false |
||||
}, |
||||
hasChildField:{ |
||||
type: String, |
||||
default: '', |
||||
required: false |
||||
}, |
||||
condition:{ |
||||
type:String, |
||||
default:'', |
||||
required:false |
||||
}, |
||||
// 是否支持多选 |
||||
multiple: { |
||||
type: Boolean, |
||||
default: false, |
||||
}, |
||||
loadTriggleChange:{ |
||||
type: Boolean, |
||||
default: false, |
||||
required:false |
||||
} |
||||
}, |
||||
data () { |
||||
return { |
||||
treeValue: null, |
||||
treeData:[], |
||||
url:"/sys/dict/loadTreeData", |
||||
view:'/sys/dict/loadDictItem/', |
||||
tableName:"", |
||||
text:"", |
||||
code:"", |
||||
|
||||
} |
||||
}, |
||||
watch: { |
||||
value () { |
||||
this.loadItemByCode() |
||||
}, |
||||
dict(){ |
||||
this.initDictInfo() |
||||
this.loadRoot(); |
||||
} |
||||
}, |
||||
created(){ |
||||
this.validateProp().then(()=>{ |
||||
this.initDictInfo() |
||||
this.loadRoot() |
||||
this.loadItemByCode() |
||||
}) |
||||
}, |
||||
methods: { |
||||
loadItemByCode(){ |
||||
if(!this.value || this.value=="0"){ |
||||
this.treeValue = null |
||||
}else{ |
||||
getAction(`${this.view}${this.dict}`,{key:this.value}).then(res=>{ |
||||
if(res.success){ |
||||
let values = this.value.split(',') |
||||
this.treeValue = res.result.map((item, index) => ({ |
||||
key: values[index], |
||||
value: values[index], |
||||
label: item |
||||
})) |
||||
this.onLoadTriggleChange(res.result[0]); |
||||
} |
||||
}) |
||||
} |
||||
}, |
||||
onLoadTriggleChange(text){ |
||||
//只有单选才会触发 |
||||
if(!this.multiple && this.loadTriggleChange){ |
||||
this.$emit('change', this.value,text) |
||||
} |
||||
}, |
||||
initDictInfo(){ |
||||
let arr = this.dict.split(",") |
||||
this.tableName = arr[0] |
||||
this.text = arr[1] |
||||
this.code = arr[2] |
||||
}, |
||||
asyncLoadTreeData (treeNode) { |
||||
return new Promise((resolve) => { |
||||
if (treeNode.$vnode.children) { |
||||
resolve() |
||||
return |
||||
} |
||||
let pid = treeNode.$vnode.key |
||||
let param = { |
||||
pid:pid, |
||||
tableName:this.tableName, |
||||
text:this.text, |
||||
code:this.code, |
||||
pidField:this.pidField, |
||||
hasChildField:this.hasChildField, |
||||
condition:this.condition |
||||
} |
||||
getAction(this.url,param).then(res=>{ |
||||
if(res.success){ |
||||
for(let i of res.result){ |
||||
i.value = i.key |
||||
if(i.leaf==false){ |
||||
i.isLeaf=false |
||||
}else if(i.leaf==true){ |
||||
i.isLeaf=true |
||||
} |
||||
} |
||||
this.addChildren(pid,res.result,this.treeData) |
||||
this.treeData = [...this.treeData] |
||||
} |
||||
resolve() |
||||
}) |
||||
}) |
||||
}, |
||||
addChildren(pid,children,treeArray){ |
||||
if(treeArray && treeArray.length>0){ |
||||
for(let item of treeArray){ |
||||
if(item.key == pid){ |
||||
if(!children || children.length==0){ |
||||
item.isLeaf=true |
||||
}else{ |
||||
item.children = children |
||||
} |
||||
break |
||||
}else{ |
||||
this.addChildren(pid,children,item.children) |
||||
} |
||||
} |
||||
} |
||||
}, |
||||
loadRoot(){ |
||||
let param = { |
||||
pid:this.pidValue, |
||||
tableName:this.tableName, |
||||
text:this.text, |
||||
code:this.code, |
||||
pidField:this.pidField, |
||||
hasChildField:this.hasChildField, |
||||
condition:this.condition |
||||
} |
||||
getAction(this.url,param).then(res=>{ |
||||
if(res.success && res.result){ |
||||
for(let i of res.result){ |
||||
i.value = i.key |
||||
if(i.leaf==false){ |
||||
i.isLeaf=false |
||||
}else if(i.leaf==true){ |
||||
i.isLeaf=true |
||||
} |
||||
} |
||||
this.treeData = [...res.result] |
||||
}else{ |
||||
console.log("数根节点查询结果-else",res) |
||||
} |
||||
}) |
||||
}, |
||||
onChange(value){ |
||||
if(!value){ |
||||
this.$emit('change', ''); |
||||
this.treeValue = null |
||||
} else if (value instanceof Array) { |
||||
this.$emit('change', value.map(item => item.value).join(',')) |
||||
this.treeValue = value |
||||
} else { |
||||
this.$emit('change', value.value,value.label) |
||||
this.treeValue = value |
||||
} |
||||
|
||||
}, |
||||
onSearch(value){ |
||||
console.log(value) |
||||
}, |
||||
getCurrTreeData(){ |
||||
return this.treeData |
||||
}, |
||||
validateProp(){ |
||||
let mycondition = this.condition |
||||
return new Promise((resolve,reject)=>{ |
||||
if(!mycondition){ |
||||
resolve(); |
||||
}else{ |
||||
try { |
||||
let test=JSON.parse(mycondition); |
||||
console.log("aaaaasdsdd",typeof test) |
||||
if(typeof test == 'object' && test){ |
||||
resolve() |
||||
}else{ |
||||
this.$message.error("组件JTreeSelect-condition传值有误,需要一个json字符串!") |
||||
reject() |
||||
} |
||||
} catch(e) { |
||||
this.$message.error("组件JTreeSelect-condition传值有误,需要一个json字符串!") |
||||
reject() |
||||
} |
||||
} |
||||
}) |
||||
} |
||||
}, |
||||
//2.2新增 在组件内定义 指定父组件调用时候的传值属性和事件类型 这个牛逼 |
||||
model: { |
||||
prop: 'value', |
||||
event: 'change' |
||||
} |
||||
} |
||||
</script> |
@ -0,0 +1,180 @@
|
||||
<template> |
||||
<a-table |
||||
:rowKey="rowKey" |
||||
:columns="columns" |
||||
:dataSource="dataSource" |
||||
:expandedRowKeys="expandedRowKeys" |
||||
v-bind="tableAttrs" |
||||
v-on="$listeners" |
||||
@expand="handleExpand" |
||||
@expandedRowsChange="expandedRowKeys=$event"> |
||||
|
||||
<template v-for="(slotItem) of slots" :slot="slotItem" slot-scope="text, record, index"> |
||||
<slot :name="slotItem" v-bind="{text,record,index}"></slot> |
||||
</template> |
||||
|
||||
</a-table> |
||||
</template> |
||||
|
||||
<script> |
||||
import { getAction } from '@/api/manage' |
||||
|
||||
export default { |
||||
name: 'JTreeTable', |
||||
props: { |
||||
rowKey: { |
||||
type: String, |
||||
default: 'id' |
||||
}, |
||||
// 根据什么查询,如果传递 id 就根据 id 查询 |
||||
queryKey: { |
||||
type: String, |
||||
default: 'parentId' |
||||
}, |
||||
queryParams: { |
||||
type: Object, |
||||
default: () => ({}) |
||||
}, |
||||
// 查询顶级时的值,如果顶级为0,则传0 |
||||
topValue: { |
||||
type: String, |
||||
default: null |
||||
}, |
||||
columns: { |
||||
type: Array, |
||||
required: true |
||||
}, |
||||
url: { |
||||
type: String, |
||||
required: true |
||||
}, |
||||
childrenUrl: { |
||||
type: String, |
||||
default: null |
||||
}, |
||||
tableProps: { |
||||
type: Object, |
||||
default: () => ({}) |
||||
}, |
||||
/** 是否在创建组件的时候就查询数据 */ |
||||
immediateRequest: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
condition:{ |
||||
type:String, |
||||
default:'', |
||||
required:false |
||||
} |
||||
}, |
||||
data() { |
||||
return { |
||||
dataSource: [], |
||||
expandedRowKeys: [] |
||||
} |
||||
}, |
||||
computed: { |
||||
getChildrenUrl() { |
||||
if (this.childrenUrl) { |
||||
return this.childrenUrl |
||||
} else { |
||||
return this.url |
||||
} |
||||
}, |
||||
slots() { |
||||
let slots = [] |
||||
for (let column of this.columns) { |
||||
if (column.scopedSlots && column.scopedSlots.customRender) { |
||||
slots.push(column.scopedSlots.customRender) |
||||
} |
||||
} |
||||
return slots |
||||
}, |
||||
tableAttrs() { |
||||
return Object.assign(this.$attrs, this.tableProps) |
||||
} |
||||
}, |
||||
watch: { |
||||
queryParams: { |
||||
deep: true, |
||||
handler() { |
||||
this.loadData() |
||||
} |
||||
} |
||||
}, |
||||
created() { |
||||
if (this.immediateRequest) this.loadData() |
||||
}, |
||||
methods: { |
||||
|
||||
/** 加载数据*/ |
||||
loadData(id = this.topValue, first = true, url = this.url) { |
||||
this.$emit('requestBefore', { first }) |
||||
|
||||
if (first) { |
||||
this.expandedRowKeys = [] |
||||
} |
||||
|
||||
let params = Object.assign({}, this.queryParams || {}) |
||||
params[this.queryKey] = id |
||||
if(this.condition && this.condition.length>0){ |
||||
params['condition'] = this.condition |
||||
} |
||||
|
||||
return getAction(url, params).then(res => { |
||||
let list = [] |
||||
if (res.result instanceof Array) { |
||||
list = res.result |
||||
} else if (res.result.records instanceof Array) { |
||||
list = res.result.records |
||||
} else { |
||||
throw '返回数据类型不识别' |
||||
} |
||||
let dataSource = list.map(item => { |
||||
// 判断是否标记了带有子级 |
||||
if (item.hasChildren === true) { |
||||
// 查找第一个带有dataIndex的值的列 |
||||
let firstColumn |
||||
for (let column of this.columns) { |
||||
firstColumn = column.dataIndex |
||||
if (firstColumn) break |
||||
} |
||||
// 定义默认展开时显示的loading子级,实际子级数据只在展开时加载 |
||||
let loadChild = { id: `${item.id}_loadChild`, [firstColumn]: 'loading...', isLoading: true } |
||||
item.children = [loadChild] |
||||
} |
||||
return item |
||||
}) |
||||
if (first) { |
||||
this.dataSource = dataSource |
||||
} |
||||
this.$emit('requestSuccess', { first, dataSource, res }) |
||||
return Promise.resolve(dataSource) |
||||
}).finally(() => this.$emit('requestFinally', { first })) |
||||
}, |
||||
|
||||
/** 点击展开图标时触发 */ |
||||
handleExpand(expanded, record) { |
||||
// 判断是否是展开状态 |
||||
if (expanded) { |
||||
// 判断子级的首个项的标记是否是“正在加载中”,如果是就加载数据 |
||||
if (record.children[0].isLoading === true) { |
||||
this.loadData(record.id, false, this.getChildrenUrl).then(dataSource => { |
||||
// 处理好的数据可直接赋值给children |
||||
if (dataSource.length === 0) { |
||||
record.children = null |
||||
} else { |
||||
record.children = dataSource |
||||
} |
||||
}) |
||||
} |
||||
} |
||||
} |
||||
|
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,452 @@
|
||||
<template> |
||||
<div :id="containerId" style="position: relative"> |
||||
|
||||
<!-- ---------------------------- begin 图片左右换位置 ------------------------------------- --> |
||||
<div class="movety-container" :style="{top:top+'px',left:left+'px',display:moveDisplay}" style="padding:0 8px;position: absolute;z-index: 91;height: 32px;width: 104px;text-align: center;"> |
||||
<div :id="containerId+'-mover'" :class="showMoverTask?'uploadty-mover-mask':'movety-opt'" style="margin-top: 12px"> |
||||
<a @click="moveLast" style="margin: 0 5px;"><a-icon type="arrow-left" style="color: #fff;font-size: 16px"/></a> |
||||
<a @click="moveNext" style="margin: 0 5px;"><a-icon type="arrow-right" style="color: #fff;font-size: 16px"/></a> |
||||
</div> |
||||
</div> |
||||
<!-- ---------------------------- end 图片左右换位置 ------------------------------------- --> |
||||
|
||||
<a-upload |
||||
name="file" |
||||
:multiple="true" |
||||
:action="uploadAction" |
||||
:headers="headers" |
||||
:data="{'biz':bizPath}" |
||||
:fileList="fileList" |
||||
:beforeUpload="beforeUpload" |
||||
@change="handleChange" |
||||
:disabled="disabled" |
||||
:returnUrl="returnUrl" |
||||
:listType="complistType" |
||||
@preview="handlePreview" |
||||
:class="{'uploadty-disabled':disabled}"> |
||||
<template> |
||||
<div v-if="isImageComp"> |
||||
<a-icon type="plus" /> |
||||
<div class="ant-upload-text">{{ text }}</div> |
||||
</div> |
||||
<a-button v-else-if="buttonVisible"> |
||||
<a-icon type="upload" />{{ text }} |
||||
</a-button> |
||||
</template> |
||||
</a-upload> |
||||
<a-modal :visible="previewVisible" :width="1000" :footer="null" @cancel="handleCancel"> |
||||
<img alt="example" style="width: 100%" :src="previewImage" /> |
||||
</a-modal> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
|
||||
import Vue from 'vue' |
||||
import { getFileAccessHttpUrl } from '@/api/manage'; |
||||
import { fileSizeLimit } from '@/api/api' |
||||
|
||||
const FILE_TYPE_ALL = "all" |
||||
const FILE_TYPE_IMG = "image" |
||||
const FILE_TYPE_TXT = "file" |
||||
const uidGenerator=()=>{ |
||||
return '-'+parseInt(Math.random()*10000+1,10); |
||||
} |
||||
const getFileName=(path)=>{ |
||||
if(path.lastIndexOf("\\")>=0){ |
||||
let reg=new RegExp("\\\\","g"); |
||||
path = path.replace(reg,"/"); |
||||
} |
||||
return path.substring(path.lastIndexOf("/")+1); |
||||
} |
||||
export default { |
||||
name: 'JUpload', |
||||
data(){ |
||||
return { |
||||
uploadAction:window._CONFIG['domianURL']+"/systemConfig/upload", |
||||
headers:{}, |
||||
fileList: [], |
||||
newFileList: [], |
||||
uploadGoOn:true, |
||||
previewVisible: false, |
||||
//---------------------------- begin 图片左右换位置 ------------------------------------- |
||||
previewImage: '', |
||||
containerId:'', |
||||
top:'', |
||||
left:'', |
||||
moveDisplay:'none', |
||||
showMoverTask:false, |
||||
moverHold:false, |
||||
currentImg:'', |
||||
//---------------------------- end 图片左右换位置 ------------------------------------- |
||||
sizeLimit: 0 |
||||
} |
||||
}, |
||||
props:{ |
||||
text:{ |
||||
type:String, |
||||
required:false, |
||||
default:"点击上传" |
||||
}, |
||||
fileType:{ |
||||
type:String, |
||||
required:false, |
||||
default:FILE_TYPE_ALL |
||||
}, |
||||
/*这个属性用于控制文件上传的业务路径*/ |
||||
bizPath:{ |
||||
type:String, |
||||
required:false, |
||||
default:"temp" |
||||
}, |
||||
value:{ |
||||
type:[String,Array], |
||||
required:false |
||||
}, |
||||
// update-begin- --- author:wangshuai ------ date:20190929 ---- for:Jupload组件增加是否能够点击 |
||||
disabled:{ |
||||
type:Boolean, |
||||
required:false, |
||||
default: false |
||||
}, |
||||
// update-end- --- author:wangshuai ------ date:20190929 ---- for:Jupload组件增加是否能够点击 |
||||
//此属性被废弃了 |
||||
triggerChange:{ |
||||
type: Boolean, |
||||
required: false, |
||||
default: false |
||||
}, |
||||
/** |
||||
* update -- author:lvdandan -- date:20190219 -- for:Jupload组件增加是否返回url, |
||||
* true:仅返回url |
||||
* false:返回fileName filePath fileSize |
||||
*/ |
||||
returnUrl:{ |
||||
type:Boolean, |
||||
required:false, |
||||
default: true |
||||
}, |
||||
number:{ |
||||
type:Number, |
||||
required:false, |
||||
default: 0 |
||||
}, |
||||
buttonVisible:{ |
||||
type:Boolean, |
||||
required:false, |
||||
default: true |
||||
}, |
||||
}, |
||||
watch:{ |
||||
value:{ |
||||
immediate: true, |
||||
handler() { |
||||
let val = this.value |
||||
if (val instanceof Array) { |
||||
if(this.returnUrl){ |
||||
this.initFileList(val.join(',')) |
||||
}else{ |
||||
this.initFileListArr(val); |
||||
} |
||||
} else { |
||||
this.initFileList(val) |
||||
} |
||||
} |
||||
} |
||||
}, |
||||
computed:{ |
||||
isImageComp(){ |
||||
return this.fileType === FILE_TYPE_IMG |
||||
}, |
||||
complistType(){ |
||||
return this.fileType === FILE_TYPE_IMG?'picture-card':'text' |
||||
} |
||||
}, |
||||
created(){ |
||||
this.initFileSizeLimit() |
||||
const token = Vue.ls.get(""); |
||||
//---------------------------- begin 图片左右换位置 ------------------------------------- |
||||
this.headers = {"X-Access-Token":token}; |
||||
this.containerId = 'container-ty-'+new Date().getTime(); |
||||
//---------------------------- end 图片左右换位置 ------------------------------------- |
||||
}, |
||||
|
||||
methods:{ |
||||
initFileSizeLimit() { |
||||
fileSizeLimit().then((res)=>{ |
||||
if(res.code === 200) { |
||||
this.sizeLimit = res.data |
||||
} |
||||
}) |
||||
}, |
||||
initFileListArr(val){ |
||||
if(!val || val.length==0){ |
||||
this.fileList = []; |
||||
return; |
||||
} |
||||
let fileList = []; |
||||
for(var a=0;a<val.length;a++){ |
||||
let url = getFileAccessHttpUrl(val[a].filePath); |
||||
fileList.push({ |
||||
uid:uidGenerator(), |
||||
name:val[a].fileName, |
||||
status: 'done', |
||||
url: url, |
||||
response:{ |
||||
code:"history", |
||||
data:val[a].filePath |
||||
} |
||||
}) |
||||
} |
||||
this.fileList = fileList |
||||
}, |
||||
initFileList(paths){ |
||||
if(!paths || paths.length==0){ |
||||
//return []; |
||||
// update-begin- --- author:os_chengtgen ------ date:20190729 ---- for:issues:326,Jupload组件初始化bug |
||||
this.fileList = []; |
||||
return; |
||||
// update-end- --- author:os_chengtgen ------ date:20190729 ---- for:issues:326,Jupload组件初始化bug |
||||
} |
||||
let fileList = []; |
||||
let arr = paths.split(",") |
||||
for(var a=0;a<arr.length;a++){ |
||||
let url = getFileAccessHttpUrl(arr[a]); |
||||
fileList.push({ |
||||
uid:uidGenerator(), |
||||
name:getFileName(arr[a]), |
||||
status: 'done', |
||||
url: url, |
||||
response:{ |
||||
code:"history", |
||||
data:arr[a] |
||||
} |
||||
}) |
||||
} |
||||
this.fileList = fileList |
||||
}, |
||||
handlePathChange(){ |
||||
let uploadFiles = this.fileList |
||||
let path = '' |
||||
if(!uploadFiles || uploadFiles.length==0){ |
||||
path = '' |
||||
} |
||||
let arr = []; |
||||
|
||||
for(var a=0;a<uploadFiles.length;a++){ |
||||
arr.push(uploadFiles[a].response.data) |
||||
} |
||||
if(arr.length>0){ |
||||
path = arr.join(",") |
||||
} |
||||
this.$emit('change', path); |
||||
}, |
||||
beforeUpload(file){ |
||||
this.uploadGoOn=true |
||||
let fileType = file.type; |
||||
let fileSize = file.size; |
||||
if(this.fileType===FILE_TYPE_IMG){ |
||||
if(fileType.indexOf('image')<0){ |
||||
this.$message.warning('请上传图片'); |
||||
this.uploadGoOn=false |
||||
return false; |
||||
} |
||||
} |
||||
//验证文件大小 |
||||
if(fileSize>this.sizeLimit) { |
||||
let parseSizeLimit = (this.sizeLimit/1024/1024).toFixed(2) |
||||
this.$message.warning('抱歉,文件大小不能超过' + parseSizeLimit + 'M'); |
||||
this.uploadGoOn=false |
||||
return false; |
||||
} |
||||
return true |
||||
}, |
||||
handleChange(info) { |
||||
console.log("--文件列表改变--") |
||||
if(!info.file.status && this.uploadGoOn === false){ |
||||
info.fileList.pop(); |
||||
} |
||||
let fileList = info.fileList |
||||
if(info.file.status==='done'){ |
||||
if(this.number>0){ |
||||
fileList = fileList.slice(-this.number); |
||||
} |
||||
if(info.file.response.code === 200){ |
||||
fileList = fileList.map((file) => { |
||||
if (file.response) { |
||||
let reUrl = file.response.data; |
||||
file.url = getFileAccessHttpUrl(reUrl); |
||||
} |
||||
return file; |
||||
}); |
||||
} |
||||
//this.$message.success(`${info.file.name} 上传成功!`); |
||||
}else if (info.file.status === 'error') { |
||||
this.$message.error(`${info.file.name} 上传失败.`); |
||||
}else if(info.file.status === 'removed'){ |
||||
this.handleDelete(info.file) |
||||
} |
||||
this.fileList = fileList |
||||
if(info.file.status==='done' || info.file.status === 'removed'){ |
||||
//returnUrl为true时仅返回文件路径 |
||||
if(this.returnUrl){ |
||||
this.handlePathChange() |
||||
}else{ |
||||
//returnUrl为false时返回文件名称、文件路径及文件大小 |
||||
this.newFileList = []; |
||||
for(var a=0;a<fileList.length;a++){ |
||||
var fileJson = { |
||||
fileName:fileList[a].name, |
||||
filePath:fileList[a].response.data, |
||||
fileSize:fileList[a].size |
||||
}; |
||||
this.newFileList.push(fileJson); |
||||
} |
||||
this.$emit('change', this.newFileList); |
||||
} |
||||
} |
||||
}, |
||||
handleDelete(file){ |
||||
//如有需要新增 删除逻辑 |
||||
console.log(file) |
||||
}, |
||||
handlePreview(file){ |
||||
let postfix = file.name.substring(file.name.indexOf('.')) |
||||
if(postfix === '.gif' || postfix === '.jpg' || postfix === '.jpeg' || postfix === '.png' || |
||||
postfix === '.GIF' || postfix === '.JPG' || postfix === '.JPEG' || postfix === '.PNG') { |
||||
this.previewImage = file.url || file.thumbUrl; |
||||
this.previewVisible = true; |
||||
}else{ |
||||
location.href=file.url |
||||
} |
||||
}, |
||||
handleCancel(){ |
||||
this.previewVisible = false; |
||||
}, |
||||
//---------------------------- begin 图片左右换位置 ------------------------------------- |
||||
moveLast(){ |
||||
//console.log(ev) |
||||
//console.log(this.fileList) |
||||
//console.log(this.currentImg) |
||||
let index = this.getIndexByUrl(); |
||||
if(index==0){ |
||||
this.$message.warn('未知的操作') |
||||
}else{ |
||||
let curr = this.fileList[index].url; |
||||
let last = this.fileList[index-1].url; |
||||
let arr =[] |
||||
for(let i=0;i<this.fileList.length;i++){ |
||||
if(i==index-1){ |
||||
arr.push(curr) |
||||
}else if(i==index){ |
||||
arr.push(last) |
||||
}else{ |
||||
arr.push(this.fileList[i].url) |
||||
} |
||||
} |
||||
this.currentImg = last |
||||
this.$emit('change',arr.join(',')) |
||||
} |
||||
}, |
||||
moveNext(){ |
||||
let index = this.getIndexByUrl(); |
||||
if(index==this.fileList.length-1){ |
||||
this.$message.warn('已到最后~') |
||||
}else{ |
||||
let curr = this.fileList[index].url; |
||||
let next = this.fileList[index+1].url; |
||||
let arr =[] |
||||
for(let i=0;i<this.fileList.length;i++){ |
||||
if(i==index+1){ |
||||
arr.push(curr) |
||||
}else if(i==index){ |
||||
arr.push(next) |
||||
}else{ |
||||
arr.push(this.fileList[i].url) |
||||
} |
||||
} |
||||
this.currentImg = next |
||||
this.$emit('change',arr.join(',')) |
||||
} |
||||
}, |
||||
getIndexByUrl(){ |
||||
for(let i=0;i<this.fileList.length;i++){ |
||||
if(this.fileList[i].url === this.currentImg || encodeURI(this.fileList[i].url) === this.currentImg){ |
||||
return i; |
||||
} |
||||
} |
||||
return -1; |
||||
} |
||||
}, |
||||
mounted(){ |
||||
const moverObj = document.getElementById(this.containerId+'-mover'); |
||||
moverObj.addEventListener('mouseover',()=>{ |
||||
this.moverHold = true |
||||
this.moveDisplay = 'block'; |
||||
}); |
||||
moverObj.addEventListener('mouseout',()=>{ |
||||
this.moverHold = false |
||||
this.moveDisplay = 'none'; |
||||
}); |
||||
let picList = document.getElementById(this.containerId)?document.getElementById(this.containerId).getElementsByClassName('ant-upload-list-picture-card'):[]; |
||||
if(picList && picList.length>0){ |
||||
picList[0].addEventListener('mouseover',(ev)=>{ |
||||
ev = ev || window.event; |
||||
let target = ev.target || ev.srcElement; |
||||
if('ant-upload-list-item-info' == target.className){ |
||||
this.showMoverTask=false |
||||
let item = target.parentElement |
||||
this.left = item.offsetLeft |
||||
this.top=item.offsetTop+item.offsetHeight-50; |
||||
this.moveDisplay = 'block'; |
||||
this.currentImg = target.getElementsByTagName('img')[0].src |
||||
} |
||||
|
||||
}); |
||||
|
||||
picList[0].addEventListener('mouseout',(ev)=>{ |
||||
ev = ev || window.event; |
||||
let target = ev.target || ev.srcElement; |
||||
//console.log('移除',target) |
||||
if('ant-upload-list-item-info' == target.className){ |
||||
this.showMoverTask=true |
||||
setTimeout(()=>{ |
||||
if(this.moverHold === false) |
||||
this.moveDisplay = 'none'; |
||||
},100) |
||||
} |
||||
if('ant-upload-list-item ant-upload-list-item-done' == target.className || 'ant-upload-list ant-upload-list-picture-card'== target.className){ |
||||
this.moveDisplay = 'none'; |
||||
} |
||||
}) |
||||
//---------------------------- end 图片左右换位置 ------------------------------------- |
||||
} |
||||
}, |
||||
model: { |
||||
prop: 'value', |
||||
event: 'change' |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="less"> |
||||
.uploadty-disabled{ |
||||
.ant-upload-list-item { |
||||
.anticon-close{ |
||||
display: none; |
||||
} |
||||
.anticon-delete{ |
||||
display: none; |
||||
} |
||||
} |
||||
} |
||||
//---------------------------- begin 图片左右换位置 ------------------------------------- |
||||
.uploadty-mover-mask{ |
||||
background-color: rgba(0, 0, 0, 0.5); |
||||
opacity: .8; |
||||
color: #fff; |
||||
height: 28px; |
||||
line-height: 28px; |
||||
} |
||||
//---------------------------- end 图片左右换位置 ------------------------------------- |
||||
</style> |
@ -0,0 +1,509 @@
|
||||
# JDate 日期组件 使用文档 |
||||
|
||||
###### 说明: antd-vue日期组件需要用moment中转一下,用起来不是很方便,特二次封装,使用时只需要传字符串即可 |
||||
## 参数配置 |
||||
| 参数 | 类型 | 必填 |说明| |
||||
|--------------|---------|----|---------| |
||||
| placeholder |string | | placeholder | |
||||
| readOnly | boolean | | true/false 默认false | |
||||
| value | string | | 绑定v-model或是v-decorator后不需要设置 | |
||||
| showTime | boolean | | 是否展示时间true/false 默认false | |
||||
| dateFormat | string | |日期格式 默认'YYYY-MM-DD' 若showTime设置为true则需要将其设置成对应的时间格式(如:YYYY-MM-DD HH:mm:ss) | |
||||
| triggerChange | string | |触发组件值改变的事件是否是change,当使用v-decorator时且没有设置decorator的option.trigger为input需要设置该值为true | |
||||
使用示例 |
||||
---- |
||||
1.组件带有v-model的使用方法 |
||||
```vue |
||||
<j-date v-model="dateStr"></j-date> |
||||
``` |
||||
|
||||
2.组件带有v-decorator的使用方法 |
||||
a).设置trigger-change属性为true |
||||
```vue |
||||
<j-date :trigger-change="true" v-decorator="['dateStr',{}]"></j-date> |
||||
``` |
||||
|
||||
b).设置decorator的option.trigger为input |
||||
```vue |
||||
<j-date v-decorator="['dateStr',{trigger:'input'}]"></j-date> |
||||
``` |
||||
|
||||
3.其他使用 |
||||
添加style |
||||
```vue |
||||
<j-date v-model="dateStr" style="width:100%"></j-date> |
||||
``` |
||||
添加placeholder |
||||
```vue |
||||
<j-date v-model="dateStr" placeholder="请输入dateStr"></j-date> |
||||
``` |
||||
添加readOnly |
||||
```vue |
||||
<j-date v-model="dateStr" :read-only="true"></j-date> |
||||
``` |
||||
|
||||
备注: |
||||
script内需引入jdate |
||||
```vue |
||||
<script> |
||||
import JDate from '@/components/jeecg/JDate' |
||||
export default { |
||||
name: "demo", |
||||
components: { |
||||
JDate |
||||
} |
||||
//... |
||||
} |
||||
</script> |
||||
``` |
||||
|
||||
|
||||
--- |
||||
|
||||
|
||||
# JSuperQuery 高级查询 使用文档 |
||||
## 参数配置 |
||||
| 参数 | 类型 | 必填 | 说明 | |
||||
|--------------|---------|----|----------------------| |
||||
| fieldList | array |✔| 需要查询的列集合示例如下,type类型有:date/datetime/string/int/number | |
||||
| callback | array | | 回调函数名称(非必须)默认handleSuperQuery | |
||||
|
||||
fieldList结构示例: |
||||
```vue |
||||
const superQueryFieldList=[{ |
||||
type:"date", |
||||
value:"birthday", |
||||
text:"生日" |
||||
},{ |
||||
type:"string", |
||||
value:"name", |
||||
text:"用户名" |
||||
},{ |
||||
type:"int", |
||||
value:"age", |
||||
text:"年龄" |
||||
}] |
||||
``` |
||||
页面代码概述: |
||||
---- |
||||
1.import之后再components之内声明 |
||||
```vue |
||||
import JSuperQuery from '@/components/jeecg/JSuperQuery.vue'; |
||||
export default { |
||||
name: "JeecgDemoList", |
||||
components: { |
||||
JSuperQuery |
||||
}, |
||||
|
||||
``` |
||||
2.页面引用 |
||||
```vue |
||||
<!-- 高级查询区域 --> |
||||
<j-super-query :fieldList="fieldList" ref="superQueryModal" @handleSuperQuery="handleSuperQuery"></j-super-query> |
||||
``` |
||||
3.list页面data中需要定义三个属性: |
||||
```vue |
||||
fieldList:superQueryFieldList, |
||||
superQueryFlag:false, |
||||
superQueryParams:"" |
||||
``` |
||||
4.list页面声明回调事件handleSuperQuery(与组件的callback对应即可) |
||||
```vue |
||||
//高级查询方法 |
||||
handleSuperQuery(arg) { |
||||
if(!arg){ |
||||
this.superQueryParams='' |
||||
this.superQueryFlag = false |
||||
}else{ |
||||
this.superQueryFlag = true |
||||
this.superQueryParams=JSON.stringify(arg) |
||||
} |
||||
this.loadData() |
||||
}, |
||||
``` |
||||
5.改造list页面方法 |
||||
```vue |
||||
// 获取查询条件 |
||||
getQueryParams() { |
||||
let sqp = {} |
||||
if(this.superQueryParams){ |
||||
sqp['superQueryParams']=encodeURI(this.superQueryParams) |
||||
} |
||||
var param = Object.assign(sqp, this.queryParam, this.isorter); |
||||
param.field = this.getQueryField(); |
||||
param.pageNo = this.ipagination.current; |
||||
param.pageSize = this.ipagination.pageSize; |
||||
return filterObj(param); |
||||
}, |
||||
``` |
||||
6.打开弹框调用show方法: |
||||
```vue |
||||
this.$refs.superQueryModal.show(); |
||||
``` |
||||
|
||||
# JEllipsis 字符串超长截取省略号显示 |
||||
|
||||
###### 说明: 遇到超长文本展示,通过此标签可以截取省略号显示,鼠标放置会提示全文本 |
||||
## 参数配置 |
||||
| 参数 | 类型 | 必填 | 说明 | |
||||
|--------|---------|----|----------------| |
||||
| value |string | 必填 | 字符串文本| |
||||
| length | number | 非必填 | 默认25 | |
||||
使用示例 |
||||
---- |
||||
1.组件带有v-model的使用方法 |
||||
```vue |
||||
<j-ellipsis :value="text"/> |
||||
|
||||
|
||||
# Modal弹框实现最大化功能 |
||||
|
||||
1.定义modal的宽度: |
||||
```vue |
||||
<a-modal |
||||
:width="modalWidth" |
||||
|
||||
|
||||
/> |
||||
``` |
||||
2.自定义modal的title,居右显示切换图标 |
||||
```vue |
||||
<template slot="title"> |
||||
<div style="width: 100%;"> |
||||
<span>{{ title }}</span> |
||||
<span style="display:inline-block;width:calc(100% - 51px);padding-right:10px;text-align: right"> |
||||
<a-button @click="toggleScreen" icon="appstore" style="height:20px;width:20px;border:0px"></a-button> |
||||
</span> |
||||
</div> |
||||
</template> |
||||
``` |
||||
3.定义toggleScreen事件,用于切换modal宽度 |
||||
```vue |
||||
toggleScreen(){ |
||||
if(this.modaltoggleFlag){ |
||||
this.modalWidth = window.innerWidth; |
||||
}else{ |
||||
this.modalWidth = 800; |
||||
} |
||||
this.modaltoggleFlag = !this.modaltoggleFlag; |
||||
}, |
||||
``` |
||||
4.data中声明上述用到的属性 |
||||
```vue |
||||
data () { |
||||
return { |
||||
modalWidth:800, |
||||
modaltoggleFlag:true, |
||||
``` |
||||
|
||||
# <a-select/> 下拉选项滚动错位的解决方法 |
||||
|
||||
## 问题描述 |
||||
|
||||
当使用了 `a-modal` 或其他带有滚动条的组件时,使用`a-select`组件并打开下拉框时滚动滚动条,就会导致错位的问题产生。 |
||||
|
||||
## 解决方法 |
||||
|
||||
大多数情况下,在 `a-select` 上添加一个 `getPopupContainer` 属性,值为`node => node.parentNode`即可解决。 |
||||
但是如果遇到 `a-select` 标签层级过深的情况,可能仍然会显示异常,只需要多加几个`.parentNode` (例:node => node.parentNode.parentNode.parentNode)多尝试几次直到解决问题即可。 |
||||
|
||||
### 代码示例 |
||||
|
||||
```html |
||||
<a-select |
||||
placeholder="请选择展示模板" |
||||
:options="dicts.displayTemplate" |
||||
:getPopupContainer="node => node.parentNode" |
||||
/> |
||||
``` |
||||
|
||||
# JAsyncTreeList 异步数列表组件使用说明 |
||||
|
||||
## 引入组件 |
||||
|
||||
```js |
||||
import JTreeTable from '@/components/jeecg/JTreeTable' |
||||
export default { |
||||
components: { JTreeTable } |
||||
} |
||||
``` |
||||
|
||||
## 所需参数 |
||||
|
||||
| 参数 | 类型 | 必填 | 说明 | |
||||
|-------------|--------|--------|--------------------------------------------------------------| |
||||
| rowKey | String | 非必填 | 表格行 key 的取值,默认为"id" | |
||||
| columns | Array | 必填 | 表格列的配置描述,具体见Antd官方文档 | |
||||
| url | String | 必填 | 数据查询url | |
||||
| childrenUrl | String | 非必填 | 查询子级时的url,若不填则使用url参数查询子级 | |
||||
| queryKey | String | 非必填 | 根据某个字段查询,如果传递 id 就根据 id 查询,默认为parentId | |
||||
| queryParams | Object | 非必填 | 查询参数,当查询参数改变的时候会自动重新查询,默认为{} | |
||||
| topValue | String | 非必填 | 查询顶级时的值,如果顶级为0,则传0,默认为null | |
||||
| tableProps | Object | 非必填 | 自定义给内部table绑定的props | |
||||
|
||||
## 代码示例 |
||||
|
||||
```html |
||||
<template> |
||||
<a-card :bordered="false"> |
||||
<j-tree-table :url="url" :columns="columns" :tableProps="tableProps"/> |
||||
</a-card> |
||||
</template> |
||||
|
||||
<script> |
||||
import JTreeTable from '@/components/jeecg/JTreeTable' |
||||
|
||||
export default { |
||||
name: 'AsyncTreeTable', |
||||
components: { JTreeTable }, |
||||
data() { |
||||
return { |
||||
url: '/api/asynTreeList', |
||||
columns: [ |
||||
{ title: '菜单名称', dataIndex: 'name' }, |
||||
{ title: '组件', dataIndex: 'component' }, |
||||
{ title: '排序', dataIndex: 'orderNum' } |
||||
], |
||||
selectedRowKeys: [] |
||||
} |
||||
}, |
||||
computed: { |
||||
tableProps() { |
||||
let _this = this |
||||
return { |
||||
// 列表项是否可选择 |
||||
// 配置项见:https://vue.ant.design/components/table-cn/#rowSelection |
||||
rowSelection: { |
||||
selectedRowKeys: _this.selectedRowKeys, |
||||
onChange: (selectedRowKeys) => _this.selectedRowKeys = selectedRowKeys |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
``` |
||||
|
||||
# JCheckbox 使用文档 |
||||
|
||||
###### 说明: antd-vue checkbox组件处理的是数组,用起来不是很方便,特二次封装,使用时只需处理字符串即可 |
||||
## 参数配置 |
||||
| 参数 | 类型 | 必填 |说明| |
||||
|--------------|---------|----|---------| |
||||
| options |array |✔| checkbox需要配置的项,是个数组,数组中每个对象包含两个属性:label(用于显示)和value(用于存储) | |
||||
|
||||
使用示例 |
||||
---- |
||||
```vue |
||||
<template> |
||||
<a-form :form="form"> |
||||
<a-form-item label="v-model式用法"> |
||||
<j-checkbox v-model="sport" :options="sportOptions"></j-checkbox><span>{{ sport }}</span> |
||||
</a-form-item> |
||||
|
||||
<a-form-item label="v-decorator式用法"> |
||||
<j-checkbox v-decorator="['sport']" :options="sportOptions"></j-checkbox><span>{{ getFormFieldValue('sport') }}</span> |
||||
</a-form-item> |
||||
</a-form> |
||||
</template> |
||||
|
||||
<script> |
||||
import JCheckbox from '@/components/jeecg/JCheckbox' |
||||
export default { |
||||
components: {JCheckbox}, |
||||
data() { |
||||
return { |
||||
form: this.$form.createForm(this), |
||||
sport:'', |
||||
sportOptions:[ |
||||
{ |
||||
label:"足球", |
||||
value:"1" |
||||
},{ |
||||
label:"篮球", |
||||
value:"2" |
||||
},{ |
||||
label:"乒乓球", |
||||
value:"3" |
||||
}] |
||||
} |
||||
}, |
||||
methods: { |
||||
getFormFieldValue(field){ |
||||
return this.form.getFieldValue(field) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
``` |
||||
|
||||
# JCodeEditor 使用文档 |
||||
|
||||
###### 说明: 一个简易版的代码编辑器,支持语法高亮 |
||||
## 参数配置 |
||||
| 参数 | 类型 | 必填 |说明| |
||||
|--------------|---------|----|---------| |
||||
| language |string | | 表示当前编写代码的类型 javascript/html/css/sql | |
||||
| placeholder |string | | placeholder | |
||||
| lineNumbers |Boolean | | 是否显示行号 | |
||||
| fullScreen |Boolean | | 是否显示全屏按钮 | |
||||
| zIndex |string | | 全屏以后的z-index | |
||||
|
||||
使用示例 |
||||
---- |
||||
```vue |
||||
<template> |
||||
<div> |
||||
<j-code-editor |
||||
language="javascript" |
||||
v-model="editorValue" |
||||
:fullScreen="true" |
||||
style="min-height: 100px"/> |
||||
{{ editorValue }} |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
import JCodeEditor from '@/components/jeecg/JCodeEditor' |
||||
export default { |
||||
components: {JCodeEditor}, |
||||
data() { |
||||
return { |
||||
form: this.$form.createForm(this), |
||||
editorValue:'', |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
``` |
||||
|
||||
# JFormContainer 使用文档 |
||||
|
||||
###### 说明: 暂用于表单禁用 |
||||
|
||||
使用示例 |
||||
---- |
||||
```vue |
||||
<!-- 在form下直接写这个组件,设置disabled为true就能将此form中的控件禁用 --> |
||||
<a-form layout="inline" :form="form" > |
||||
<j-form-container disabled> |
||||
<!-- 表单内容省略..... --> |
||||
</j-form-container> |
||||
</a-form> |
||||
``` |
||||
|
||||
# JImportModal 使用文档 |
||||
|
||||
###### 说明: 用于列表页面导入excel功能 |
||||
|
||||
使用示例 |
||||
---- |
||||
```vue |
||||
|
||||
<template> |
||||
<!-- 此处省略部分代码...... --> |
||||
<a-button @click="handleImportXls" type="primary" icon="upload">导入</a-button> |
||||
<!-- 此处省略部分代码...... --> |
||||
<j-import-modal ref="importModal" :url="getImportUrl()" @ok="importOk"></j-import-modal> |
||||
<!-- 此处省略部分代码...... --> |
||||
</template> |
||||
|
||||
<script> |
||||
import JCodeEditor from '@/components/jeecg/JCodeEditor' |
||||
export default { |
||||
components: {JCodeEditor}, |
||||
data() { |
||||
return { |
||||
//省略代码...... |
||||
} |
||||
}, |
||||
methods:{ |
||||
//省略部分代码...... |
||||
handleImportXls(){ |
||||
this.$refs.importModal.show() |
||||
}, |
||||
getImportUrl(){ |
||||
return '你自己处理上传业务的后台地址' |
||||
}, |
||||
importOk(){ |
||||
this.loadData(1) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
``` |
||||
|
||||
# JSlider 滑块验证码 |
||||
|
||||
使用示例 |
||||
---- |
||||
```vue |
||||
<template> |
||||
<div style="width: 300px"> |
||||
<j-slider @onSuccess="sliderSuccess"></j-slider> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
import JSlider from '@/components/jeecg/JSlider' |
||||
export default { |
||||
components: {JSlider}, |
||||
data() { |
||||
return { |
||||
form: this.$form.createForm(this), |
||||
editorValue:'', |
||||
} |
||||
}, |
||||
methods:{ |
||||
sliderSuccess(){ |
||||
console.log("验证完成") |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
``` |
||||
|
||||
|
||||
# JTreeSelect 树形下拉组件 |
||||
异步加载的树形下拉组件 |
||||
|
||||
## 参数配置 |
||||
| 参数 | 类型 | 必填 |说明| |
||||
|--------------|---------|----|---------| |
||||
| placeholder |string | | placeholder | |
||||
| dict |string | ✔| 表名,显示字段名,存储字段名拼接的字符串 | |
||||
| pidField |string | ✔| 父ID的字段名 | |
||||
| pidValue |string | | 根节点父ID的值 默认'0' 不可以设置为空,如果想使用此组件,而数据库根节点父ID为空,请修改之 | |
||||
| multiple |boolean | |是否支持多选 | |
||||
|
||||
使用示例 |
||||
---- |
||||
```vue |
||||
<template> |
||||
<a-form> |
||||
<a-form-item label="树形下拉测试" style="width: 300px"> |
||||
<j-tree-select |
||||
v-model="departId" |
||||
placeholder="请选择部门" |
||||
dict="sys_depart,depart_name,id" |
||||
pidField="parent_id"> |
||||
</j-tree-select> |
||||
{{ departId }} |
||||
</a-form-item> |
||||
</a-form > |
||||
</template> |
||||
|
||||
<script> |
||||
import JTreeSelect from '@/components/jeecg/JTreeSelect' |
||||
export default { |
||||
components: {JTreeSelect}, |
||||
data() { |
||||
return { |
||||
departId:"" |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
``` |
||||
|
||||
|
@ -0,0 +1,9 @@
|
||||
import JModal from './JModal' |
||||
import JFormContainer from './JFormContainer.vue' |
||||
|
||||
export default { |
||||
install(Vue) { |
||||
Vue.component('JFormContainer', JFormContainer) |
||||
Vue.component(JModal.name, JModal) |
||||
} |
||||
} |
@ -0,0 +1,113 @@
|
||||
<template> |
||||
<div> |
||||
<a-modal |
||||
title="文件上传" |
||||
:width="width" |
||||
:visible="visible" |
||||
@ok="ok" |
||||
cancelText="取消" |
||||
@cancel="close"> |
||||
<!--style="top: 20px;"--> |
||||
<j-upload :file-type="fileType" :value="filePath" @change="handleChange" :disabled="disabled"></j-upload> |
||||
</a-modal> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
import JUpload from '@/components/jeecg/JUpload' |
||||
import { getFileAccessHttpUrl } from '@/api/manage'; |
||||
|
||||
const getFileName=(path)=>{ |
||||
if(path.lastIndexOf("\\")>=0){ |
||||
let reg=new RegExp("\\\\","g"); |
||||
path = path.replace(reg,"/"); |
||||
} |
||||
return path.substring(path.lastIndexOf("/")+1); |
||||
} |
||||
|
||||
export default { |
||||
name: 'JFilePop', |
||||
components: { JUpload }, |
||||
props:{ |
||||
title:{ |
||||
type:String, |
||||
default:'', |
||||
required:false |
||||
}, |
||||
position:{ |
||||
type:String, |
||||
default:'right', |
||||
required:false |
||||
}, |
||||
height:{ |
||||
type:Number, |
||||
default:200, |
||||
required:false |
||||
}, |
||||
width:{ |
||||
type:Number, |
||||
default:520, |
||||
required:false |
||||
}, |
||||
|
||||
popContainer:{ |
||||
type:String, |
||||
default:'', |
||||
required:false |
||||
}, |
||||
disabled:{ |
||||
type:Boolean, |
||||
default:false, |
||||
required:false |
||||
} |
||||
}, |
||||
data(){ |
||||
return { |
||||
visible:false, |
||||
filePath:'', |
||||
id:'', |
||||
fileType:'file' |
||||
|
||||
} |
||||
}, |
||||
methods:{ |
||||
handleChange(value){ |
||||
this.filePath = value; |
||||
}, |
||||
show(id,value,flag){ |
||||
this.id = id; |
||||
this.filePath = value; |
||||
this.visible=true |
||||
if(flag === 'img'){ |
||||
this.fileType = 'image' |
||||
}else{ |
||||
this.fileType = 'file' |
||||
} |
||||
|
||||
}, |
||||
ok(){ |
||||
if(!this.filePath){ |
||||
this.$message.error("未上传任何文件") |
||||
return false; |
||||
} |
||||
let arr = this.filePath.split(",") |
||||
let obj = { |
||||
name:getFileName(arr[0]), |
||||
url:getFileAccessHttpUrl(arr[0]), |
||||
path:this.filePath, |
||||
status: 'done', |
||||
id:this.id |
||||
} |
||||
this.$emit('ok',obj) |
||||
this.visible=false |
||||
}, |
||||
close(){ |
||||
this.visible=false |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,98 @@
|
||||
<template> |
||||
<a-popover trigger="contextmenu" v-model="visible" :placement="position"> |
||||
<!--"(node) => node.parentNode.parentNode"--> |
||||
<div slot="title"> |
||||
<span>{{ title }}</span> |
||||
<span style="float: right" title="关闭"> |
||||
<a-icon type="close" @click="visible=false"/> |
||||
</span> |
||||
</div> |
||||
<a-input :value="inputContent" @change="handleInputChange"> |
||||
<a-icon slot="suffix" type="fullscreen" @click.stop="pop" /> |
||||
</a-input> |
||||
<div slot="content"> |
||||
<textarea :value="inputContent" @input="handleInputChange" :style="{ height: height + 'px', width: width + 'px' }"></textarea> |
||||
</div> |
||||
</a-popover> |
||||
</template> |
||||
|
||||
<script> |
||||
export default { |
||||
name: 'JInputPop', |
||||
props:{ |
||||
title:{ |
||||
type:String, |
||||
default:'', |
||||
required:false |
||||
}, |
||||
position:{ |
||||
type:String, |
||||
default:'right', |
||||
required:false |
||||
}, |
||||
height:{ |
||||
type:Number, |
||||
default:200, |
||||
required:false |
||||
}, |
||||
width:{ |
||||
type:Number, |
||||
default:150, |
||||
required:false |
||||
}, |
||||
value:{ |
||||
type:String, |
||||
required:false |
||||
}, |
||||
popContainer:{ |
||||
type:String, |
||||
default:'', |
||||
required:false |
||||
} |
||||
|
||||
}, |
||||
data(){ |
||||
return { |
||||
visible:false, |
||||
inputContent:'' |
||||
|
||||
} |
||||
}, |
||||
|
||||
watch:{ |
||||
value:{ |
||||
immediate:true, |
||||
handler:function(){ |
||||
if(this.value && this.value.length>0){ |
||||
this.inputContent = this.value; |
||||
} |
||||
} |
||||
}, |
||||
}, |
||||
model: { |
||||
prop: 'value', |
||||
event: 'change' |
||||
}, |
||||
methods:{ |
||||
handleInputChange(event){ |
||||
this.inputContent = event.target.value |
||||
this.$emit('change',this.inputContent) |
||||
}, |
||||
pop(){ |
||||
this.visible=true |
||||
}, |
||||
getPopupContainer(node){ |
||||
if(!this.popContainer){ |
||||
return node.parentNode |
||||
}else{ |
||||
return document.getElementById(this.popContainer) |
||||
} |
||||
|
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,928 @@
|
||||
<template> |
||||
<a-modal |
||||
title="corn表达式" |
||||
:width="modalWidth" |
||||
:visible="visible" |
||||
:confirmLoading="confirmLoading" |
||||
@ok="handleSubmit" |
||||
@cancel="close" |
||||
cancelText="关闭"> |
||||
<div class="card-container"> |
||||
<a-tabs type="card"> |
||||
<a-tab-pane key="1" type="card"> |
||||
<span slot="tab"><a-icon type="schedule" /> 秒</span> |
||||
<a-radio-group v-model="result.second.cronEvery"> |
||||
<a-row> |
||||
<a-radio value="1">每一秒钟</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="2">每隔 |
||||
<a-input-number size="small" v-model="result.second.incrementIncrement" :min="1" :max="59"></a-input-number> |
||||
秒执行 从 |
||||
<a-input-number size="small" v-model="result.second.incrementStart" :min="0" :max="59"></a-input-number> |
||||
秒开始 |
||||
</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="3">具体秒数(可多选)</a-radio> |
||||
<a-select style="width:354px;" size="small" mode="multiple" v-model="result.second.specificSpecific"> |
||||
<a-select-option v-for="(val,index) in 60" :key="index" :value="index">{{ index }}</a-select-option> |
||||
</a-select> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="4">周期从 |
||||
<a-input-number size="small" v-model="result.second.rangeStart" :min="1" :max="59"></a-input-number> |
||||
到 |
||||
<a-input-number size="small" v-model="result.second.rangeEnd" :min="0" :max="59"></a-input-number> |
||||
秒 |
||||
</a-radio> |
||||
</a-row> |
||||
</a-radio-group> |
||||
</a-tab-pane> |
||||
<a-tab-pane key="2"> |
||||
<span slot="tab"><a-icon type="schedule" />分</span> |
||||
<div class="tabBody"> |
||||
<a-radio-group v-model="result.minute.cronEvery"> |
||||
<a-row> |
||||
<a-radio value="1">每一分钟</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="2">每隔 |
||||
<a-input-number size="small" v-model="result.minute.incrementIncrement" :min="1" :max="60"></a-input-number> |
||||
分执行 从 |
||||
<a-input-number size="small" v-model="result.minute.incrementStart" :min="0" :max="59"></a-input-number> |
||||
分开始 |
||||
</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="3">具体分钟数(可多选)</a-radio> |
||||
<a-select style="width:340px;" size="small" mode="multiple" v-model="result.minute.specificSpecific"> |
||||
<a-select-option v-for="(val,index) in Array(60)" :key="index" :value="index"> {{ index }}</a-select-option> |
||||
</a-select> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="4">周期从 |
||||
<a-input-number size="small" v-model="result.minute.rangeStart" :min="1" :max="60"></a-input-number> |
||||
到 |
||||
<a-input-number size="small" v-model="result.minute.rangeEnd" :min="0" :max="59"></a-input-number> |
||||
分 |
||||
</a-radio> |
||||
</a-row> |
||||
</a-radio-group> |
||||
</div> |
||||
</a-tab-pane> |
||||
<a-tab-pane key="3"> |
||||
<span slot="tab"><a-icon type="schedule" /> 时</span> |
||||
<div class="tabBody"> |
||||
<a-radio-group v-model="result.hour.cronEvery"> |
||||
<a-row> |
||||
<a-radio value="1">每一小时</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="2">每隔 |
||||
<a-input-number size="small" v-model="result.hour.incrementIncrement" :min="0" :max="23"></a-input-number> |
||||
小时执行 从 |
||||
<a-input-number size="small" v-model="result.hour.incrementStart" :min="0" :max="23"></a-input-number> |
||||
小时开始 |
||||
</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio class="long" value="3">具体小时数(可多选)</a-radio> |
||||
<a-select style="width:340px;" size="small" mode="multiple" v-model="result.hour.specificSpecific"> |
||||
<a-select-option v-for="(val,index) in Array(24)" :key="index" >{{ index }}</a-select-option> |
||||
</a-select> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="4">周期从 |
||||
<a-input-number size="small" v-model="result.hour.rangeStart" :min="0" :max="23"></a-input-number> |
||||
到 |
||||
<a-input-number size="small" v-model="result.hour.rangeEnd" :min="0" :max="23"></a-input-number> |
||||
小时 |
||||
</a-radio> |
||||
</a-row> |
||||
</a-radio-group> |
||||
</div> |
||||
</a-tab-pane> |
||||
<a-tab-pane key="4"> |
||||
<span slot="tab"><a-icon type="schedule" /> 天</span> |
||||
<div class="tabBody"> |
||||
<a-radio-group v-model="result.day.cronEvery"> |
||||
<a-row> |
||||
<a-radio value="1">每一天</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="2">每隔 |
||||
<a-input-number size="small" v-model="result.week.incrementIncrement" :min="1" :max="7"></a-input-number> |
||||
周执行 从 |
||||
<a-select size="small" v-model="result.week.incrementStart"> |
||||
<a-select-option v-for="(val,index) in Array(7)" :key="index" :value="index+1">{{ weekDays[index] }}</a-select-option> |
||||
</a-select> |
||||
开始 |
||||
</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="3">每隔 |
||||
<a-input-number size="small" v-model="result.day.incrementIncrement" :min="1" :max="31"></a-input-number> |
||||
天执行 从 |
||||
<a-input-number size="small" v-model="result.day.incrementStart" :min="1" :max="31"></a-input-number> |
||||
天开始 |
||||
</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio class="long" value="4">具体星期几(可多选)</a-radio> |
||||
<a-select style="width:340px;" size="small" mode="multiple" v-model="result.week.specificSpecific"> |
||||
<a-select-option v-for="(val,index) in Array(7)" :key="index" :value="index+1">{{ weekDays[index] }}</a-select-option> |
||||
</a-select> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio class="long" value="5">具体天数(可多选)</a-radio> |
||||
<a-select style="width:354px;" size="small" mode="multiple" v-model="result.day.specificSpecific"> |
||||
<a-select-option v-for="(val,index) in Array(31)" :key="index" :value="index+1">{{ index+1 }}</a-select-option> |
||||
</a-select> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="6">在这个月的最后一天</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="7">在这个月的最后一个工作日</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="8">在这个月的最后一个 |
||||
<a-select size="small" v-model="result.day.cronLastSpecificDomDay"> |
||||
<a-select-option v-for="(val,index) in Array(7)" :key="index" :value="index+1">{{ weekDays[index] }}</a-select-option> |
||||
</a-select> |
||||
</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="9"> |
||||
在本月底前 |
||||
<a-input-number size="small" v-model="result.day.cronDaysBeforeEomMinus" :min="1" :max="31"></a-input-number> |
||||
天 |
||||
</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="10">最近的工作日(周一至周五)至本月 |
||||
<a-input-number size="small" v-model="result.day.cronDaysNearestWeekday" :min="1" :max="31"></a-input-number> |
||||
日 |
||||
</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="11">在这个月的第 |
||||
<a-input-number size="small" v-model="result.week.cronNthDayNth" :min="1" :max="5"></a-input-number> |
||||
个 |
||||
<a-select size="small" v-model="result.week.cronNthDayDay"> |
||||
<a-select-option v-for="(val,index) in Array(7)" :key="index" :value="index+1">{{ weekDays[index] }}</a-select-option> |
||||
</a-select> |
||||
|
||||
</a-radio> |
||||
</a-row> |
||||
</a-radio-group> |
||||
</div> |
||||
</a-tab-pane> |
||||
<a-tab-pane key="5"> |
||||
<span slot="tab"><a-icon type="schedule" /> 月</span> |
||||
<div class="tabBody"> |
||||
<a-radio-group v-model="result.month.cronEvery"> |
||||
<a-row> |
||||
<a-radio value="1">每一月</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="2">每隔 |
||||
<a-input-number size="small" v-model="result.month.incrementIncrement" :min="0" :max="12"></a-input-number> |
||||
月执行 从 |
||||
<a-input-number size="small" v-model="result.month.incrementStart" :min="0" :max="12"></a-input-number> |
||||
月开始 |
||||
</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio class="long" value="3">具体月数(可多选)</a-radio> |
||||
<a-select style="width:354px;" size="small" filterable mode="multiple" v-model="result.month.specificSpecific"> |
||||
<a-select-option v-for="(val,index) in Array(12)" :key="index" :value="index+1">{{ index+1 }}</a-select-option> |
||||
</a-select> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="4">从 |
||||
<a-input-number size="small" v-model="result.month.rangeStart" :min="1" :max="12"></a-input-number> |
||||
到 |
||||
<a-input-number size="small" v-model="result.month.rangeEnd" :min="1" :max="12"></a-input-number> |
||||
月之间的每个月 |
||||
</a-radio> |
||||
</a-row> |
||||
</a-radio-group> |
||||
</div> |
||||
</a-tab-pane> |
||||
<a-tab-pane key="6"> |
||||
<span slot="tab"><a-icon type="schedule" /> 年</span> |
||||
<div class="tabBody"> |
||||
<a-radio-group v-model="result.year.cronEvery"> |
||||
<a-row> |
||||
<a-radio value="1">每一年</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="2">每隔 |
||||
<a-input-number size="small" v-model="result.year.incrementIncrement" :min="1" :max="99"></a-input-number> |
||||
年执行 从 |
||||
<a-input-number size="small" v-model="result.year.incrementStart" :min="2019" :max="2119"></a-input-number> |
||||
年开始 |
||||
</a-radio> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio class="long" value="3">具体年份(可多选)</a-radio> |
||||
<a-select style="width:354px;" size="small" filterable mode="multiple" v-model="result.year.specificSpecific"> |
||||
<a-select-option v-for="(val,index) in Array(100)" :key="index" :value="2019+index">{{ 2019+index }}</a-select-option> |
||||
</a-select> |
||||
</a-row> |
||||
<a-row> |
||||
<a-radio value="4">从 |
||||
<a-input-number size="small" v-model="result.year.rangeStart" :min="2019" :max="2119"></a-input-number> |
||||
到 |
||||
<a-input-number size="small" v-model="result.year.rangeEnd" :min="2019" :max="2119"></a-input-number> |
||||
年之间的每一年 |
||||
</a-radio> |
||||
</a-row> |
||||
</a-radio-group> |
||||
</div> |
||||
</a-tab-pane> |
||||
</a-tabs> |
||||
<div class="bottom"> |
||||
<span class="value">{{this.cron }}</span> |
||||
</div> |
||||
</div> |
||||
</a-modal> |
||||
</template> |
||||
<script> |
||||
export default { |
||||
name:'VueCron', |
||||
props:['data'], |
||||
data(){ |
||||
return { |
||||
visible: false, |
||||
confirmLoading:false, |
||||
size:'large', |
||||
weekDays:['天','一','二','三','四','五','六'].map(val=>'星期'+val), |
||||
result: { |
||||
second:{}, |
||||
minute:{}, |
||||
hour:{}, |
||||
day:{}, |
||||
week:{}, |
||||
month:{}, |
||||
year:{} |
||||
}, |
||||
defaultValue: { |
||||
second:{ |
||||
cronEvery:'', |
||||
incrementStart:3, |
||||
incrementIncrement:5, |
||||
rangeStart:1, |
||||
rangeEnd:0, |
||||
specificSpecific:[], |
||||
}, |
||||
minute:{ |
||||
cronEvery:'', |
||||
incrementStart:3, |
||||
incrementIncrement:5, |
||||
rangeStart:1, |
||||
rangeEnd:'0', |
||||
specificSpecific:[], |
||||
}, |
||||
hour:{ |
||||
cronEvery:'', |
||||
incrementStart:3, |
||||
incrementIncrement:5, |
||||
rangeStart:'0', |
||||
rangeEnd:'0', |
||||
specificSpecific:[], |
||||
}, |
||||
day:{ |
||||
cronEvery:'', |
||||
incrementStart:1, |
||||
incrementIncrement:'1', |
||||
rangeStart:'', |
||||
rangeEnd:'', |
||||
specificSpecific:[], |
||||
cronLastSpecificDomDay:1, |
||||
cronDaysBeforeEomMinus:1, |
||||
cronDaysNearestWeekday:1, |
||||
}, |
||||
week:{ |
||||
cronEvery:'', |
||||
incrementStart:1, |
||||
incrementIncrement:1, |
||||
specificSpecific:[], |
||||
cronNthDayDay:1, |
||||
cronNthDayNth:1, |
||||
}, |
||||
month:{ |
||||
cronEvery:'', |
||||
incrementStart:3, |
||||
incrementIncrement:5, |
||||
rangeStart:1, |
||||
rangeEnd:1, |
||||
specificSpecific:[], |
||||
}, |
||||
year:{ |
||||
cronEvery:'', |
||||
incrementStart:2017, |
||||
incrementIncrement:1, |
||||
rangeStart:2019, |
||||
rangeEnd: 2019, |
||||
specificSpecific:[], |
||||
}, |
||||
label:'' |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
modalWidth(){ |
||||
return 608; |
||||
}, |
||||
secondsText() { |
||||
let seconds = ''; |
||||
let cronEvery=this.result.second.cronEvery||''; |
||||
switch (cronEvery.toString()){ |
||||
case '1': |
||||
seconds = '*'; |
||||
break; |
||||
case '2': |
||||
seconds = this.result.second.incrementStart+'/'+this.result.second.incrementIncrement; |
||||
break; |
||||
case '3': |
||||
this.result.second.specificSpecific.map(val=> {seconds += val+','}); |
||||
seconds = seconds.slice(0, -1); |
||||
break; |
||||
case '4': |
||||
seconds = this.result.second.rangeStart+'-'+this.result.second.rangeEnd; |
||||
break; |
||||
} |
||||
return seconds; |
||||
}, |
||||
minutesText() { |
||||
let minutes = ''; |
||||
let cronEvery=this.result.minute.cronEvery||''; |
||||
switch (cronEvery.toString()){ |
||||
case '1': |
||||
minutes = '*'; |
||||
break; |
||||
case '2': |
||||
minutes = this.result.minute.incrementStart+'/'+this.result.minute.incrementIncrement; |
||||
break; |
||||
case '3': |
||||
this.result.minute.specificSpecific.map(val=> { |
||||
minutes += val+',' |
||||
}); |
||||
minutes = minutes.slice(0, -1); |
||||
break; |
||||
case '4': |
||||
minutes = this.result.minute.rangeStart+'-'+this.result.minute.rangeEnd; |
||||
break; |
||||
} |
||||
return minutes; |
||||
}, |
||||
hoursText() { |
||||
let hours = ''; |
||||
let cronEvery=this.result.hour.cronEvery||''; |
||||
switch (cronEvery.toString()){ |
||||
case '1': |
||||
hours = '*'; |
||||
break; |
||||
case '2': |
||||
hours = this.result.hour.incrementStart+'/'+this.result.hour.incrementIncrement; |
||||
break; |
||||
case '3': |
||||
this.result.hour.specificSpecific.map(val=> { |
||||
hours += val+',' |
||||
}); |
||||
hours = hours.slice(0, -1); |
||||
break; |
||||
case '4': |
||||
hours = this.result.hour.rangeStart+'-'+this.result.hour.rangeEnd; |
||||
break; |
||||
} |
||||
return hours; |
||||
}, |
||||
daysText() { |
||||
let days=''; |
||||
let cronEvery=this.result.day.cronEvery||''; |
||||
switch (cronEvery.toString()){ |
||||
case '1': |
||||
break; |
||||
case '2': |
||||
case '4': |
||||
case '11': |
||||
days = '?'; |
||||
break; |
||||
case '3': |
||||
days = this.result.day.incrementStart+'/'+this.result.day.incrementIncrement; |
||||
break; |
||||
case '5': |
||||
this.result.day.specificSpecific.map(val=> { |
||||
days += val+',' |
||||
}); |
||||
days = days.slice(0, -1); |
||||
break; |
||||
case '6': |
||||
days = "L"; |
||||
break; |
||||
case '7': |
||||
days = "LW"; |
||||
break; |
||||
case '8': |
||||
days = this.result.day.cronLastSpecificDomDay + 'L'; |
||||
break; |
||||
case '9': |
||||
days = 'L-' + this.result.day.cronDaysBeforeEomMinus; |
||||
break; |
||||
case '10': |
||||
days = this.result.day.cronDaysNearestWeekday+"W"; |
||||
break |
||||
} |
||||
return days; |
||||
}, |
||||
weeksText() { |
||||
let weeks = ''; |
||||
let cronEvery=this.result.day.cronEvery||''; |
||||
switch (cronEvery.toString()){ |
||||
case '1': |
||||
case '3': |
||||
case '5': |
||||
weeks = '?'; |
||||
break; |
||||
case '2': |
||||
weeks = this.result.week.incrementStart+'/'+this.result.week.incrementIncrement; |
||||
break; |
||||
case '4': |
||||
this.result.week.specificSpecific.map(val=> { |
||||
weeks += val+',' |
||||
}); |
||||
weeks = weeks.slice(0, -1); |
||||
break; |
||||
case '6': |
||||
case '7': |
||||
case '8': |
||||
case '9': |
||||
case '10': |
||||
weeks = "?"; |
||||
break; |
||||
case '11': |
||||
weeks = this.result.week.cronNthDayDay+"#"+this.result.week.cronNthDayNth; |
||||
break; |
||||
} |
||||
return weeks; |
||||
}, |
||||
monthsText() { |
||||
let months = ''; |
||||
let cronEvery=this.result.month.cronEvery||''; |
||||
switch (cronEvery.toString()){ |
||||
case '1': |
||||
months = '*'; |
||||
break; |
||||
case '2': |
||||
months = this.result.month.incrementStart+'/'+this.result.month.incrementIncrement; |
||||
break; |
||||
case '3': |
||||
this.result.month.specificSpecific.map(val=> { |
||||
months += val+',' |
||||
}); |
||||
months = months.slice(0, -1); |
||||
break; |
||||
case '4': |
||||
months = this.result.month.rangeStart+'-'+this.result.month.rangeEnd; |
||||
break; |
||||
} |
||||
return months; |
||||
}, |
||||
yearsText() { |
||||
let years = ''; |
||||
let cronEvery=this.result.year.cronEvery||''; |
||||
switch (cronEvery.toString()){ |
||||
case '1': |
||||
years = '*'; |
||||
break; |
||||
case '2': |
||||
years = this.result.year.incrementStart+'/'+this.result.year.incrementIncrement; |
||||
break; |
||||
case '3': |
||||
this.result.year.specificSpecific.map(val=> { |
||||
years += val+',' |
||||
}); |
||||
years = years.slice(0, -1); |
||||
break; |
||||
case '4': |
||||
years = this.result.year.rangeStart+'-'+this.result.year.rangeEnd; |
||||
break; |
||||
} |
||||
return years; |
||||
}, |
||||
cron(){ |
||||
return `${this.secondsText||'*'} ${this.minutesText||'*'} ${this.hoursText||'*'} ${this.daysText||'*'} ${this.monthsText||'*'} ${this.weeksText||'?'} ${this.yearsText||'*'}` |
||||
}, |
||||
}, |
||||
watch:{ |
||||
visible:{ |
||||
handler() { |
||||
// if(this.data){ |
||||
// //this. result = Object.keys(this.data.value).length>0?this.deepCopy(this.data.value):this.deepCopy(this.defaultValue); |
||||
// //this.result = Object.keys(this.data.value).length>0?clone(this.data.value):clone(this.defaultValue); |
||||
// //this.result = Object.keys(this.data.value).length>0?clone(JSON.parse(this.data.value)):clone(this.defaultValue); |
||||
// this.result = Object.keys(this.data.value).length>0?JSON.parse(this.data.value):JSON.parse(JSON.stringify(this.defaultValue)); |
||||
// }else{ |
||||
// //this.result = this.deepCopy(this.defaultValue); |
||||
// //this.result = clone(this.defaultValue); |
||||
// this.result = JSON.parse(JSON.stringify(this.defaultValue)); |
||||
// } |
||||
let label = this.data; |
||||
if(label){ |
||||
this.secondsReverseExp(label) |
||||
this.minutesReverseExp(label); |
||||
this.hoursReverseExp(label); |
||||
this.daysReverseExp(label); |
||||
this.daysReverseExp(label); |
||||
this.monthsReverseExp(label); |
||||
this.yearReverseExp(label); |
||||
JSON.parse(JSON.stringify(label)); |
||||
}else { |
||||
this.result = JSON.parse(JSON.stringify(this.defaultValue)); |
||||
} |
||||
} |
||||
} |
||||
}, |
||||
methods: { |
||||
show(){ |
||||
this.visible = true; |
||||
// console.log('secondsReverseExp',this.secondsReverseExp(this.data)); |
||||
// console.log('minutesReverseExp',this.minutesReverseExp(this.data)); |
||||
// console.log('hoursReverseExp',this.hoursReverseExp(this.data)); |
||||
// console.log('daysReverseExp',this.daysReverseExp(this.data)); |
||||
// console.log('monthsReverseExp',this.monthsReverseExp(this.data)); |
||||
// console.log('yearReverseExp',this.yearReverseExp(this.data)); |
||||
|
||||
}, |
||||
handleSubmit(){ |
||||
this.$emit('ok',this.cron); |
||||
this.close(); |
||||
this.visible = false; |
||||
}, |
||||
close(){ |
||||
this.visible = false; |
||||
}, |
||||
secondsReverseExp(seconds) { |
||||
let val = seconds.split(" ")[0]; |
||||
//alert(val); |
||||
let second = { |
||||
cronEvery:'', |
||||
incrementStart:3, |
||||
incrementIncrement:5, |
||||
rangeStart:1, |
||||
rangeEnd:0, |
||||
specificSpecific:[] |
||||
}; |
||||
switch (true) { |
||||
case val.includes('*'): |
||||
second.cronEvery = '1'; |
||||
break; |
||||
case val.includes('/'): |
||||
second.cronEvery = '2'; |
||||
second.incrementStart = val.split('/')[0]; |
||||
second.incrementIncrement = val.split('/')[1]; |
||||
break; |
||||
case val.includes(','): |
||||
second.cronEvery = '3'; |
||||
second.specificSpecific = val.split(',').map(Number).sort(); |
||||
break; |
||||
case val.includes('-'): |
||||
second.cronEvery = '4'; |
||||
second.rangeStart = val.split('-')[0]; |
||||
second.rangeEnd = val.split('-')[1]; |
||||
break; |
||||
default: |
||||
second.cronEvery = '1'; |
||||
} |
||||
this.result.second = second; |
||||
}, |
||||
minutesReverseExp(minutes) { |
||||
let val = minutes.split(" ")[1]; |
||||
let minute = { |
||||
cronEvery:'', |
||||
incrementStart:3, |
||||
incrementIncrement:5, |
||||
rangeStart:1, |
||||
rangeEnd:0, |
||||
specificSpecific:[], |
||||
} |
||||
switch (true) { |
||||
case val.includes('*'): |
||||
minute.cronEvery = '1'; |
||||
break; |
||||
case val.includes('/'): |
||||
minute.cronEvery = '2'; |
||||
minute.incrementStart = val.split('/')[0]; |
||||
minute.incrementIncrement = val.split('/')[1]; |
||||
break; |
||||
case val.includes(','): |
||||
minute.cronEvery = '3'; |
||||
minute.specificSpecific = val.split(',').map(Number).sort(); |
||||
break; |
||||
case val.includes('-'): |
||||
minute.cronEvery = '4'; |
||||
minute.rangeStart = val.split('-')[0]; |
||||
minute.rangeEnd = val.split('-')[1]; |
||||
break; |
||||
default: |
||||
minute.cronEvery = '1'; |
||||
} |
||||
this.result.minute = minute; |
||||
}, |
||||
hoursReverseExp(hours) { |
||||
let val = hours.split(" ")[2]; |
||||
let hour ={ |
||||
cronEvery:'', |
||||
incrementStart:3, |
||||
incrementIncrement:5, |
||||
rangeStart:1, |
||||
rangeEnd:'0', |
||||
specificSpecific:[], |
||||
}; |
||||
switch (true) { |
||||
case val.includes('*'): |
||||
hour.cronEvery = '1'; |
||||
break; |
||||
case val.includes('/'): |
||||
hour.cronEvery = '2'; |
||||
hour.incrementStart = val.split('/')[0]; |
||||
hour.incrementIncrement = val.split('/')[1]; |
||||
break; |
||||
case val.includes(','): |
||||
hour.cronEvery = '3'; |
||||
hour.specificSpecific = val.split(',').map(Number).sort(); |
||||
break; |
||||
case val.includes('-'): |
||||
hour.cronEvery = '4'; |
||||
hour.rangeStart = val.split('-')[0]; |
||||
hour.rangeEnd = val.split('-')[1]; |
||||
break; |
||||
default: |
||||
hour.cronEvery = '1'; |
||||
} |
||||
this.result.hour = hour; |
||||
}, |
||||
daysReverseExp(cron) { |
||||
let days = cron.split(" ")[3]; |
||||
let weeks = cron.split(" ")[5]; |
||||
let day ={ |
||||
cronEvery:'', |
||||
incrementStart:1, |
||||
incrementIncrement:1, |
||||
rangeStart:1, |
||||
rangeEnd:1, |
||||
specificSpecific:[], |
||||
cronLastSpecificDomDay:1, |
||||
cronDaysBeforeEomMinus:1, |
||||
cronDaysNearestWeekday:1, |
||||
}; |
||||
let week = { |
||||
cronEvery:'', |
||||
incrementStart:1, |
||||
incrementIncrement:1, |
||||
specificSpecific:[], |
||||
cronNthDayDay:1, |
||||
cronNthDayNth:'1', |
||||
}; |
||||
if (!days.includes('?')) { |
||||
switch (true) { |
||||
case days.includes('*'): |
||||
day.cronEvery = '1'; |
||||
break; |
||||
case days.includes('?'): |
||||
// 2、4、11 |
||||
break; |
||||
case days.includes('/'): |
||||
day.cronEvery = '3'; |
||||
day.incrementStart = days.split('/')[0]; |
||||
day.incrementIncrement = days.split('/')[1]; |
||||
break; |
||||
case days.includes(','): |
||||
day.cronEvery = '5'; |
||||
day.specificSpecific = days.split(',').map(Number).sort(); |
||||
// day.specificSpecific.forEach(function (value, index) { |
||||
// day.specificSpecific[index] = value -1; |
||||
// }); |
||||
break; |
||||
case days.includes('LW'): |
||||
day.cronEvery = '7'; |
||||
break; |
||||
case days.includes('L-'): |
||||
day.cronEvery = '9'; |
||||
day.cronDaysBeforeEomMinus = days.split('L-')[1]; |
||||
break; |
||||
case days.includes('L'): |
||||
|
||||
//alert(days); |
||||
if(days.len == 1){ |
||||
day.cronEvery = '6'; |
||||
day.cronLastSpecificDomDay = '1'; |
||||
} |
||||
else |
||||
{ |
||||
day.cronEvery = '8'; |
||||
day.cronLastSpecificDomDay = Number(days.split('L')[0]); |
||||
} |
||||
break; |
||||
case days.includes('W'): |
||||
day.cronEvery = '10'; |
||||
day.cronDaysNearestWeekday = days.split('W')[0]; |
||||
break; |
||||
default: |
||||
day.cronEvery = '1'; |
||||
} |
||||
}else { |
||||
switch (true){ |
||||
case weeks.includes('/'): |
||||
day.cronEvery = '2'; |
||||
week.incrementStart = weeks.split("/")[0]; |
||||
week.incrementIncrement = weeks.split("/")[1]; |
||||
break; |
||||
case weeks.includes(','): |
||||
day.cronEvery = '4'; |
||||
week.specificSpecific = weeks.split(',').map(Number).sort(); |
||||
break; |
||||
case '#': |
||||
day.cronEvery = '11'; |
||||
week.cronNthDayDay = weeks.split("#")[0]; |
||||
week.cronNthDayNth = weeks.split("#")[1]; |
||||
break; |
||||
default: |
||||
day.cronEvery = '1'; |
||||
week.cronEvery = '1'; |
||||
} |
||||
} |
||||
this.result.day = day; |
||||
this.result.week = week; |
||||
}, |
||||
monthsReverseExp(cron) { |
||||
let months = cron.split(" ")[4]; |
||||
let month = { |
||||
cronEvery:'', |
||||
incrementStart:3, |
||||
incrementIncrement:5, |
||||
rangeStart:1, |
||||
rangeEnd:1, |
||||
specificSpecific:[], |
||||
}; |
||||
switch (true){ |
||||
case months.includes('*'): |
||||
month.cronEvery = '1'; |
||||
break; |
||||
case months.includes('/'): |
||||
month.cronEvery = '2'; |
||||
month.incrementStart = months.split('/')[0]; |
||||
month.incrementIncrement = months.split('/')[1]; |
||||
break; |
||||
case months.includes(','): |
||||
month.cronEvery = '3'; |
||||
month.specificSpecific = months.split(',').map(Number).sort(); |
||||
break; |
||||
case months.includes('-'): |
||||
month.cronEvery = '4'; |
||||
month.rangeStart = months.split('-')[0]; |
||||
month.rangeEnd = months.split('-')[1]; |
||||
break; |
||||
default: |
||||
month.cronEvery = '1'; |
||||
} |
||||
this.result.month = month; |
||||
}, |
||||
yearReverseExp(cron) { |
||||
let years = cron.split(" ")[6]; |
||||
let year = { |
||||
cronEvery:'', |
||||
incrementStart:3, |
||||
incrementIncrement:5, |
||||
rangeStart:2019, |
||||
rangeEnd:2019, |
||||
specificSpecific:[], |
||||
}; |
||||
switch (true){ |
||||
case years.includes('*'): |
||||
year.cronEvery = '1'; |
||||
break; |
||||
case years.includes('/'): |
||||
year.cronEvery = '2'; |
||||
year.incrementStart = years.split('/')[0]; |
||||
year.incrementIncrement = years.split('/')[1]; |
||||
break; |
||||
case years.includes(','): |
||||
year.cronEvery = '3'; |
||||
year.specificSpecific = years.split(',').map(Number).sort(); |
||||
break; |
||||
case years.includes('-'): |
||||
year.cronEvery = '4'; |
||||
year.rangeStart = years.split('-')[0]; |
||||
year.rangeEnd = years.split('-')[1]; |
||||
break; |
||||
default: |
||||
year.cronEvery = '1'; |
||||
} |
||||
this.result.year = year; |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="less"> |
||||
.card-container { |
||||
background: #fff; |
||||
overflow: hidden; |
||||
padding: 12px; |
||||
position: relative; |
||||
width: 100%; |
||||
.ant-tabs{ |
||||
border:1px solid #e6ebf5; |
||||
padding: 0; |
||||
.ant-tabs-bar { |
||||
margin: 0; |
||||
outline: none; |
||||
border-bottom: none; |
||||
.ant-tabs-nav-container{ |
||||
margin: 0; |
||||
.ant-tabs-tab { |
||||
padding: 0 24px!important; |
||||
background-color: #f5f7fa!important; |
||||
margin-right: 0px!important; |
||||
border-radius: 0; |
||||
line-height: 38px; |
||||
border: 1px solid transparent!important; |
||||
border-bottom: 1px solid #e6ebf5!important; |
||||
} |
||||
.ant-tabs-tab-active.ant-tabs-tab{ |
||||
color: #409eff; |
||||
background-color: #fff!important; |
||||
border-right:1px solid #e6ebf5!important; |
||||
border-left:1px solid #e6ebf5!important; |
||||
border-bottom:1px solid #fff!important; |
||||
font-weight: normal; |
||||
transition:none!important; |
||||
} |
||||
} |
||||
} |
||||
.ant-tabs-tabpane{ |
||||
padding: 15px; |
||||
.ant-row{ |
||||
margin: 10px 0; |
||||
} |
||||
.ant-select,.ant-input-number{ |
||||
width: 100px; |
||||
} |
||||
} |
||||
} |
||||
} |
||||
</style> |
||||
<style lang="less" scoped> |
||||
.container-widthEn{ |
||||
width: 755px; |
||||
} |
||||
.container-widthCn{ |
||||
width: 608px; |
||||
} |
||||
.language{ |
||||
text-align: center; |
||||
position: absolute; |
||||
right: 13px; |
||||
top: 13px; |
||||
border: 1px solid transparent; |
||||
height: 40px; |
||||
line-height: 38px; |
||||
font-size: 16px; |
||||
color: #409eff; |
||||
z-index: 1; |
||||
background: #f5f7fa; |
||||
outline: none; |
||||
width: 47px; |
||||
border-bottom: 1px solid #e6ebf5; |
||||
border-radius: 0; |
||||
} |
||||
.card-container{ |
||||
.bottom{ |
||||
display: flex; |
||||
justify-content: center; |
||||
padding: 10px 0 0 0; |
||||
.cronButton{ |
||||
margin: 0 10px; |
||||
line-height: 40px; |
||||
} |
||||
} |
||||
} |
||||
.tabBody{ |
||||
.a-row{ |
||||
margin: 10px 0; |
||||
.long{ |
||||
.a-select{ |
||||
width:354px; |
||||
} |
||||
} |
||||
.a-input-number{ |
||||
width: 110px; |
||||
} |
||||
} |
||||
} |
||||
</style> |
@ -0,0 +1,334 @@
|
||||
<template> |
||||
<a-modal |
||||
centered |
||||
:title="name + '选择'" |
||||
:width="width" |
||||
:visible="visible" |
||||
@ok="handleOk" |
||||
@cancel="close" |
||||
cancelText="关闭"> |
||||
|
||||
<a-row :gutter="18"> |
||||
<a-col :span="16"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<a-form layout="inline"> |
||||
<a-row :gutter="24"> |
||||
|
||||
<a-col :span="14"> |
||||
<a-form-item :label="(queryParamText||name)"> |
||||
<a-input v-model="queryParam[queryParamCode||valueKey]" :placeholder="'请输入' + (queryParamText||name)" @pressEnter="searchQuery"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :span="8"> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-button type="primary" @click="searchQuery" icon="search">查询</a-button> |
||||
<a-button type="primary" @click="searchReset" icon="reload" style="margin-left: 8px">重置</a-button> |
||||
</span> |
||||
</a-col> |
||||
|
||||
</a-row> |
||||
</a-form> |
||||
</div> |
||||
|
||||
<a-table |
||||
size="small" |
||||
bordered |
||||
:rowKey="rowKey" |
||||
:columns="innerColumns" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:loading="loading" |
||||
:scroll="{ y: 240 }" |
||||
:rowSelection="{selectedRowKeys, onChange: onSelectChange, type: multiple ? 'checkbox':'radio'}" |
||||
:customRow="customRowFn" |
||||
@change="handleTableChange"> |
||||
</a-table> |
||||
|
||||
</a-col> |
||||
<a-col :span="8"> |
||||
<a-card :title="'已选' + name" :bordered="false" :head-style="{padding:0}" :body-style="{padding:0}"> |
||||
|
||||
<a-table size="small" :rowKey="rowKey" bordered v-bind="selectedTable"> |
||||
<span slot="action" slot-scope="text, record, index"> |
||||
<a @click="handleDeleteSelected(record, index)">删除</a> |
||||
</span> |
||||
</a-table> |
||||
|
||||
</a-card> |
||||
</a-col> |
||||
</a-row> |
||||
</a-modal> |
||||
</template> |
||||
|
||||
<script> |
||||
import { getAction } from '@/api/manage' |
||||
import Ellipsis from '@/components/Ellipsis' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
import { cloneObject, pushIfNotExist } from '@/utils/util' |
||||
|
||||
export default { |
||||
name: 'JSelectBizComponentModal', |
||||
mixins: [JeecgListMixin], |
||||
components: { Ellipsis }, |
||||
props: { |
||||
value: { |
||||
type: Array, |
||||
default: () => [] |
||||
}, |
||||
visible: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
valueKey: { |
||||
type: String, |
||||
required: true |
||||
}, |
||||
multiple: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
width: { |
||||
type: Number, |
||||
default: 900 |
||||
}, |
||||
|
||||
name: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
listUrl: { |
||||
type: String, |
||||
required: true, |
||||
default: '' |
||||
}, |
||||
// 根据 value 获取显示文本的地址,例如存的是 username,可以通过该地址获取到 realname |
||||
valueUrl: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
displayKey: { |
||||
type: String, |
||||
default: null |
||||
}, |
||||
columns: { |
||||
type: Array, |
||||
required: true, |
||||
default: () => [] |
||||
}, |
||||
// 查询条件Code |
||||
queryParamCode: { |
||||
type: String, |
||||
default: null |
||||
}, |
||||
// 查询条件文字 |
||||
queryParamText: { |
||||
type: String, |
||||
default: null |
||||
}, |
||||
rowKey: { |
||||
type: String, |
||||
default: 'id' |
||||
}, |
||||
// 过长裁剪长度,设置为 -1 代表不裁剪 |
||||
ellipsisLength: { |
||||
type: Number, |
||||
default: 12 |
||||
}, |
||||
}, |
||||
data() { |
||||
return { |
||||
innerValue: [], |
||||
// 已选择列表 |
||||
selectedTable: { |
||||
pagination: false, |
||||
scroll: { y: 240 }, |
||||
columns: [ |
||||
{ |
||||
...this.columns[0], |
||||
width: this.columns[0].widthRight || this.columns[0].width, |
||||
}, |
||||
{ title: '操作', dataIndex: 'action', align: 'center', width: 60, scopedSlots: { customRender: 'action' }, } |
||||
], |
||||
dataSource: [], |
||||
}, |
||||
renderEllipsis: (value) => (<ellipsis length={this.ellipsisLength}>{value}</ellipsis>), |
||||
url: { list: this.listUrl }, |
||||
/* 分页参数 */ |
||||
ipagination: { |
||||
current: 1, |
||||
pageSize: 5, |
||||
pageSizeOptions: ['5', '10', '20', '30'], |
||||
showTotal: (total, range) => { |
||||
return range[0] + '-' + range[1] + ' 共' + total + '条' |
||||
}, |
||||
showQuickJumper: true, |
||||
showSizeChanger: true, |
||||
total: 0 |
||||
}, |
||||
options: [], |
||||
dataSourceMap: {}, |
||||
} |
||||
}, |
||||
computed: { |
||||
// 表头 |
||||
innerColumns() { |
||||
let columns = cloneObject(this.columns) |
||||
columns.forEach(column => { |
||||
// 给所有的列加上过长裁剪 |
||||
if (this.ellipsisLength !== -1) { |
||||
column.customRender = (text) => this.renderEllipsis(text) |
||||
} |
||||
}) |
||||
return columns |
||||
}, |
||||
}, |
||||
watch: { |
||||
value: { |
||||
deep: true, |
||||
immediate: true, |
||||
handler(val) { |
||||
this.innerValue = cloneObject(val) |
||||
this.selectedRowKeys = [] |
||||
this.valueWatchHandler(val) |
||||
this.queryOptionsByValue(val) |
||||
} |
||||
}, |
||||
dataSource: { |
||||
deep: true, |
||||
handler(val) { |
||||
this.emitOptions(val) |
||||
this.valueWatchHandler(this.innerValue) |
||||
} |
||||
}, |
||||
selectedRowKeys: { |
||||
immediate: true, |
||||
deep: true, |
||||
handler(val) { |
||||
this.selectedTable.dataSource = val.map(key => { |
||||
for (let data of this.dataSource) { |
||||
if (data[this.rowKey] === key) { |
||||
pushIfNotExist(this.innerValue, data[this.valueKey]) |
||||
return data |
||||
} |
||||
} |
||||
for (let data of this.selectedTable.dataSource) { |
||||
if (data[this.rowKey] === key) { |
||||
pushIfNotExist(this.innerValue, data[this.valueKey]) |
||||
return data |
||||
} |
||||
} |
||||
console.warn('未找到选择的行信息,key:' + key) |
||||
return {} |
||||
}) |
||||
}, |
||||
} |
||||
}, |
||||
|
||||
methods: { |
||||
|
||||
/** 关闭弹窗 */ |
||||
close() { |
||||
this.$emit('update:visible', false) |
||||
}, |
||||
|
||||
valueWatchHandler(val) { |
||||
val.forEach(item => { |
||||
this.dataSource.concat(this.selectedTable.dataSource).forEach(data => { |
||||
if (data[this.valueKey] === item) { |
||||
pushIfNotExist(this.selectedRowKeys, data[this.rowKey]) |
||||
} |
||||
}) |
||||
}) |
||||
}, |
||||
|
||||
queryOptionsByValue(value) { |
||||
if (!value || value.length === 0) { |
||||
return |
||||
} |
||||
// 判断options是否存在value,如果已存在数据就不再请求后台了 |
||||
let notExist = false |
||||
for (let val of value) { |
||||
let find = false |
||||
for (let option of this.options) { |
||||
if (val === option.value) { |
||||
find = true |
||||
break |
||||
} |
||||
} |
||||
if (!find) { |
||||
notExist = true |
||||
break |
||||
} |
||||
} |
||||
if (!notExist) return |
||||
getAction(this.valueUrl || this.listUrl, { |
||||
// 这里最后加一个 , 的原因是因为无论如何都要使用 in 查询,防止后台进行了模糊匹配,导致查询结果不准确 |
||||
[this.valueKey]: value.join(',') + ',', |
||||
pageNo: 1, |
||||
pageSize: value.length |
||||
}).then((res) => { |
||||
if (res.success) { |
||||
let dataSource = res.result |
||||
if (!(dataSource instanceof Array)) { |
||||
dataSource = res.result.records |
||||
} |
||||
this.emitOptions(dataSource, (data) => { |
||||
pushIfNotExist(this.innerValue, data[this.valueKey]) |
||||
pushIfNotExist(this.selectedRowKeys, data[this.rowKey]) |
||||
pushIfNotExist(this.selectedTable.dataSource, data, this.rowKey) |
||||
}) |
||||
} |
||||
}) |
||||
}, |
||||
|
||||
emitOptions(dataSource, callback) { |
||||
dataSource.forEach(data => { |
||||
let key = data[this.valueKey] |
||||
this.dataSourceMap[key] = data |
||||
pushIfNotExist(this.options, { label: data[this.displayKey || this.valueKey], value: key }, 'value') |
||||
typeof callback === 'function' ? callback(data) : '' |
||||
}) |
||||
this.$emit('options', this.options, this.dataSourceMap) |
||||
}, |
||||
|
||||
/** 完成选择 */ |
||||
handleOk() { |
||||
let value = this.selectedTable.dataSource.map(data => data[this.valueKey]) |
||||
this.$emit('input', value) |
||||
this.close() |
||||
}, |
||||
|
||||
/** 删除已选择的 */ |
||||
handleDeleteSelected(record, index) { |
||||
this.selectedRowKeys.splice(this.selectedRowKeys.indexOf(record[this.rowKey]), 1) |
||||
this.selectedTable.dataSource.splice(index, 1) |
||||
}, |
||||
|
||||
customRowFn(record) { |
||||
return { |
||||
on: { |
||||
click: () => { |
||||
let key = record[this.rowKey] |
||||
if (!this.multiple) { |
||||
this.selectedRowKeys = [key] |
||||
this.selectedTable.dataSource = [record] |
||||
} else { |
||||
let index = this.selectedRowKeys.indexOf(key) |
||||
if (index === -1) { |
||||
this.selectedRowKeys.push(key) |
||||
this.selectedTable.dataSource.push(record) |
||||
} else { |
||||
this.handleDeleteSelected(record, index) |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
||||
}, |
||||
|
||||
} |
||||
} |
||||
</script> |
||||
<style lang="less" scoped> |
||||
</style> |
@ -0,0 +1,36 @@
|
||||
# JSelectBizComponent |
||||
|
||||
Jeecg 选择组件的公共可复用组件 |
||||
|
||||
## 引用方式 |
||||
|
||||
```js |
||||
import JSelectBizComponent from '@/src/components/jeecgbiz/JSelectBizComponent' |
||||
|
||||
export default { |
||||
components: { JSelectBizComponent } |
||||
} |
||||
``` |
||||
|
||||
## 参数 |
||||
|
||||
### 配置参数 |
||||
|
||||
| 参数名 | 类型 | 必填 | 默认值 | 备注 | |
||||
|-----------------------|---------|------|--------------|--------------------------------------------------------------------------------------| |
||||
| rowKey | String | | "id" | 唯一标识的字段名 | |
||||
| value(v-model) | String | | "" | 默认选择的数据,多个用半角逗号分割 | |
||||
| name | String | | "" | 显示名字,例如选择用户就填写"用户" | |
||||
| listUrl | String | 是 | | 数据请求地址,必须是封装了分页的地址 | |
||||
| valueUrl | String | | "" | 获取显示文本的地址,例如存的是 username,可以通过该地址获取到 realname | |
||||
| displayKey | String | | null | 显示在标签上的字段 key ,不传则直接显示数据 | |
||||
| returnKeys | Array | | ['id', 'id'] | v-model 绑定的 keys,是个数组,默认使用第二项,当配置了 `returnId=true` 就返回第一项 | |
||||
| returnId | Boolean | | false | 返回ID,设为true后将返回配置的 `returnKeys` 中的第一项 | |
||||
| selectButtonText | String | | "选择" | 选择按钮的文字 | |
||||
| queryParamText | String | | null | 查询条件显示文字,不传则使用 `name` | |
||||
| columns | Array | 是 | | 列配置项,与antd的table的配置完全一致。列的第一项会被配置成右侧已选择的列表上 | |
||||
| columns[0].widthRight | String | | null | 仅列的第一项可以应用此配置,表示右侧已选择列表的宽度,建议 `70%`,不传则应用`width` | |
||||
| placeholder | String | | "请选择" | 占位符 | |
||||
| disabled | Boolean | | false | 是否禁用 | |
||||
| multiple | Boolean | | false | 是否可多选 | |
||||
| buttons | Boolean | | true | 是否显示"选择"按钮,如果不显示,可以直接点击文本框打开选择界面 | |
@ -0,0 +1,165 @@
|
||||
<template> |
||||
<a-row class="j-select-biz-component-box" type="flex" :gutter="8"> |
||||
<a-col class="left" :class="{'full': !buttons}"> |
||||
<slot name="left"> |
||||
<a-select |
||||
mode="multiple" |
||||
:placeholder="placeholder" |
||||
v-model="selectValue" |
||||
:options="selectOptions" |
||||
allowClear |
||||
:disabled="disabled" |
||||
:open="selectOpen" |
||||
style="width: 100%;" |
||||
@dropdownVisibleChange="handleDropdownVisibleChange" |
||||
@click.native="visible=(buttons?visible:true)" |
||||
/> |
||||
</slot> |
||||
</a-col> |
||||
|
||||
<a-col v-if="buttons" class="right"> |
||||
<a-button type="primary" icon="search" :disabled="disabled" @click="visible=true">{{selectButtonText}}</a-button> |
||||
</a-col> |
||||
|
||||
<j-select-biz-component-modal |
||||
v-model="selectValue" |
||||
:visible.sync="visible" |
||||
v-bind="modalProps" |
||||
@options="handleOptions" |
||||
/> |
||||
</a-row> |
||||
</template> |
||||
|
||||
<script> |
||||
import JSelectBizComponentModal from './JSelectBizComponentModal' |
||||
|
||||
export default { |
||||
name: 'JSelectBizComponent', |
||||
components: { JSelectBizComponentModal }, |
||||
props: { |
||||
value: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
/** 是否返回 id,默认 false,返回 code */ |
||||
returnId: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
placeholder: { |
||||
type: String, |
||||
default: '请选择' |
||||
}, |
||||
disabled: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
// 是否支持多选,默认 true |
||||
multiple: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
// 是否显示按钮,默认 true |
||||
buttons: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
// 显示的 Key |
||||
displayKey: { |
||||
type: String, |
||||
default: null |
||||
}, |
||||
// 返回的 key |
||||
returnKeys: { |
||||
type: Array, |
||||
default: () => ['id', 'id'] |
||||
}, |
||||
// 选择按钮文字 |
||||
selectButtonText: { |
||||
type: String, |
||||
default: '选择' |
||||
}, |
||||
|
||||
}, |
||||
data() { |
||||
return { |
||||
selectValue: [], |
||||
selectOptions: [], |
||||
dataSourceMap: {}, |
||||
visible: false, |
||||
selectOpen: false, |
||||
} |
||||
}, |
||||
computed: { |
||||
valueKey() { |
||||
return this.returnId ? this.returnKeys[0] : this.returnKeys[1] |
||||
}, |
||||
modalProps() { |
||||
return Object.assign({ |
||||
valueKey: this.valueKey, |
||||
multiple: this.multiple, |
||||
returnKeys: this.returnKeys, |
||||
displayKey: this.displayKey || this.valueKey |
||||
}, this.$attrs) |
||||
}, |
||||
}, |
||||
watch: { |
||||
value: { |
||||
immediate: true, |
||||
handler(val) { |
||||
if (val) { |
||||
this.selectValue = val.split(',') |
||||
} else { |
||||
this.selectValue = [] |
||||
} |
||||
} |
||||
}, |
||||
selectValue: { |
||||
deep: true, |
||||
handler(val) { |
||||
let rows = val.map(key => this.dataSourceMap[key]) |
||||
this.$emit('select', rows) |
||||
let data = val.join(',') |
||||
this.$emit('input', data) |
||||
this.$emit('change', data) |
||||
} |
||||
} |
||||
}, |
||||
methods: { |
||||
handleOptions(options, dataSourceMap) { |
||||
this.selectOptions = options |
||||
this.dataSourceMap = dataSourceMap |
||||
}, |
||||
handleDropdownVisibleChange() { |
||||
// 解决antdv自己的bug —— open 设置为 false 了,点击后还是添加了 open 样式,导致点击事件失效 |
||||
this.selectOpen = true |
||||
this.$nextTick(() => { |
||||
this.selectOpen = false |
||||
}) |
||||
}, |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="less" scoped> |
||||
.j-select-biz-component-box { |
||||
|
||||
@width: 82px; |
||||
|
||||
.left { |
||||
width: calc(100% - @width - 8px); |
||||
} |
||||
|
||||
.right { |
||||
width: @width; |
||||
} |
||||
|
||||
.full { |
||||
width: 100%; |
||||
} |
||||
|
||||
/deep/ .ant-select-search__field { |
||||
display: none !important; |
||||
} |
||||
} |
||||
</style> |
@ -0,0 +1,122 @@
|
||||
<template> |
||||
<div class="components-input-demo-presuffix"> |
||||
<!----> |
||||
<a-input @click="openModal" placeholder="请点击选择部门" v-model="departNames" readOnly :disabled="disabled"> |
||||
<a-icon slot="prefix" type="cluster" title="部门选择控件"/> |
||||
<a-icon v-if="departIds" slot="suffix" type="close-circle" @click="handleEmpty" title="清空"/> |
||||
</a-input> |
||||
|
||||
<j-select-depart-modal |
||||
ref="innerDepartSelectModal" |
||||
:modal-width="modalWidth" |
||||
:multi="multi" |
||||
:rootOpened="rootOpened" |
||||
:depart-id="departIds" |
||||
@ok="handleOK" |
||||
@initComp="initComp"/> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
import JSelectDepartModal from './modal/JSelectDepartModal' |
||||
export default { |
||||
name: 'JSelectDepart', |
||||
components:{ |
||||
JSelectDepartModal |
||||
}, |
||||
props:{ |
||||
modalWidth:{ |
||||
type:Number, |
||||
default:500, |
||||
required:false |
||||
}, |
||||
multi:{ |
||||
type:Boolean, |
||||
default:false, |
||||
required:false |
||||
}, |
||||
rootOpened:{ |
||||
type:Boolean, |
||||
default:true, |
||||
required:false |
||||
}, |
||||
value:{ |
||||
type:String, |
||||
required:false |
||||
}, |
||||
disabled:{ |
||||
type: Boolean, |
||||
required: false, |
||||
default: false |
||||
}, |
||||
// 自定义返回字段,默认返回 id |
||||
customReturnField: { |
||||
type: String, |
||||
default: 'id' |
||||
} |
||||
}, |
||||
data(){ |
||||
return { |
||||
visible:false, |
||||
confirmLoading:false, |
||||
departNames:"", |
||||
departIds:'' |
||||
} |
||||
}, |
||||
mounted(){ |
||||
this.departIds = this.value |
||||
}, |
||||
watch:{ |
||||
value(val){ |
||||
if (this.customReturnField === 'id') { |
||||
this.departIds = val |
||||
} |
||||
} |
||||
}, |
||||
methods:{ |
||||
initComp(departNames){ |
||||
this.departNames = departNames |
||||
}, |
||||
openModal(){ |
||||
this.$refs.innerDepartSelectModal.show() |
||||
}, |
||||
handleOK(rows, idstr) { |
||||
let value = '' |
||||
if (!rows && rows.length <= 0) { |
||||
this.departNames = '' |
||||
this.departIds = '' |
||||
} else { |
||||
value = rows.map(row => row[this.customReturnField]).join(',') |
||||
this.departNames = rows.map(row => row['departName']).join(',') |
||||
this.departIds = idstr |
||||
} |
||||
this.$emit("change", value) |
||||
}, |
||||
getDepartNames(){ |
||||
return this.departNames |
||||
}, |
||||
handleEmpty(){ |
||||
this.handleOK('') |
||||
} |
||||
}, |
||||
model: { |
||||
prop: 'value', |
||||
event: 'change' |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
.components-input-demo-presuffix .anticon-close-circle { |
||||
cursor: pointer; |
||||
color: #ccc; |
||||
transition: color 0.3s; |
||||
font-size: 12px; |
||||
} |
||||
.components-input-demo-presuffix .anticon-close-circle:hover { |
||||
color: #f5222d; |
||||
} |
||||
.components-input-demo-presuffix .anticon-close-circle:active { |
||||
color: #666; |
||||
} |
||||
</style> |
@ -0,0 +1,327 @@
|
||||
<template> |
||||
<div> |
||||
<!-- <a-input-search v-if="kind === 'material'" v-model="names" placeholder="请选择" @search="onSearch"></a-input-search> --> |
||||
<a-select |
||||
:value="value" |
||||
v-if="kind === 'material'" |
||||
placeholder="编码 名称" |
||||
show-search |
||||
style="width: 100px" |
||||
dropdownClassName="drop-down-Style" |
||||
option-label-prop="value" |
||||
:dropdownMatchSelectWidth="false" |
||||
:default-active-first-option="false" |
||||
:show-arrow="false" |
||||
:filter-option="false" |
||||
:not-found-content="fetching ? undefined : null" |
||||
@search="fetchUser" |
||||
@change="handleChange" |
||||
@focus="loadData()" |
||||
> |
||||
<div slot="notFoundContent"> |
||||
<a-spin v-if="fetching" size="small" /> |
||||
</div> |
||||
<div slot="dropdownRender" slot-scope="menu"> |
||||
<v-nodes :vnodes="menu" /> |
||||
<a-divider style="margin: 4px 0" /> |
||||
<div |
||||
@mousedown="(e) => e.preventDefault()" |
||||
style="display: flex; justify-content: space-between; margin: 0 5px" |
||||
> |
||||
<a-pagination |
||||
:default-current="6" |
||||
:current.sync="queryParam.current" |
||||
:pageSize.sync="queryParam.pageSize" |
||||
@change="changeCurrent" |
||||
:total="queryParam.total" |
||||
style="margin-bottom: 4px" |
||||
/> |
||||
<a-button type="primary" @click="confirmSelected">确定</a-button> |
||||
</div> |
||||
</div> |
||||
<a-select-opt-group v-if="materailList.length"> |
||||
<span slot="label"> |
||||
<div class="select-option"> |
||||
<a-checkbox style="margin-right:5px" :indeterminate="indeterminate" :checked="checkAll" @change="onCheckAllChange"> |
||||
</a-checkbox> |
||||
<span class="mr10" style="width: 100px; font-size: 16px">物料编码</span> |
||||
<span class="mr10" style="width: 400px; font-size: 16px;">名称</span> |
||||
<span class="mr10" style="width: 100px; font-size: 16px">类别</span> |
||||
<span class="mr10" style="width: 100px; font-size: 16px">单位</span> |
||||
<span class="mr10" style="width: 100px; font-size: 16px">多属性</span> |
||||
<span class="mr10" style="width: 100px; font-size: 16px">库存</span> |
||||
<span class="mr10" style="width: 100px; font-size: 16px">扩展信息</span> |
||||
</div> |
||||
</span> |
||||
|
||||
<a-select-option |
||||
style="width: 400px" |
||||
:value="item.mBarCode" |
||||
v-for="(item, index) in materailList" |
||||
:key="item.id" |
||||
> |
||||
<div class="select-option"> |
||||
<div style="margin-left: -8px;"> |
||||
<a-checkbox |
||||
style="margin-right:5px" |
||||
:id="item.mBarCode" |
||||
:key="index" |
||||
:checked="checkedList.indexOf(item.mBarCode) !== -1" |
||||
@change="(e) => onchange(e, item)" |
||||
@click.stop.native="() => {}" |
||||
> |
||||
</a-checkbox> |
||||
</div> |
||||
<span class="mr10" style="width: 100px">{{ item.mBarCode }}</span> |
||||
<span class="mr10" style="width: 400px;width-space:nowrap;text-overflow:ellipsis;overflow: hidden;white-space:nowrap;">{{ item.name }}</span> |
||||
<span class="mr10" style="width: 100px">{{ item.categoryName }}</span> |
||||
<span class="mr10" style="width: 100px">{{ item.unit }}</span> |
||||
<span class="mr10" style="width: 100px">{{ item.sku }}</span> |
||||
<span class="mr10" style="width: 100px">{{ item.stock }}</span> |
||||
<span class="mr10" style="width: 100px">{{ item.expand }}</span> |
||||
</div> |
||||
</a-select-option> |
||||
</a-select-opt-group> |
||||
</a-select> |
||||
<a-input-search |
||||
v-if="kind === 'batch' || kind === 'sn'" |
||||
v-model="names" |
||||
placeholder="请选择" |
||||
readOnly |
||||
@search="onSearch" |
||||
></a-input-search> |
||||
<j-select-material-modal |
||||
v-if="kind === 'material'" |
||||
ref="selectModal" |
||||
:modal-width="modalWidth" |
||||
:rows="rows" |
||||
:multi="multi" |
||||
:bar-code="value" |
||||
@ok="selectOK" |
||||
@initComp="initComp" |
||||
/> |
||||
<j-select-batch-modal |
||||
v-if="kind === 'batch'" |
||||
ref="selectModal" |
||||
:modal-width="modalWidth" |
||||
:rows="rows" |
||||
:multi="multi" |
||||
:bar-code="value" |
||||
@ok="selectOK" |
||||
@initComp="initComp" |
||||
/> |
||||
<j-select-sn-modal |
||||
v-if="kind === 'sn'" |
||||
ref="selectModal" |
||||
:modal-width="modalWidth" |
||||
:rows="rows" |
||||
:multi="multi" |
||||
:bar-code="value" |
||||
@ok="selectOK" |
||||
@initComp="initComp" |
||||
/> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
import JSelectMaterialModal from './modal/JSelectMaterialModal' |
||||
import JSelectBatchModal from './modal/JSelectBatchModal' |
||||
import JSelectSnModal from './modal/JSelectSnModal' |
||||
import { filterObj, getMpListShort } from '@/utils/util' |
||||
import { getMaterialByBarCode, getMaterialBySelect } from '@/api/api' |
||||
import Vue from 'vue' |
||||
|
||||
export default { |
||||
name: 'JSelectList', |
||||
components: { |
||||
JSelectMaterialModal, |
||||
JSelectBatchModal, |
||||
JSelectSnModal, |
||||
VNodes: { |
||||
functional: true, |
||||
render: (h, ctx) => ctx.props.vnodes, |
||||
}, |
||||
}, |
||||
props: { |
||||
modalWidth: { |
||||
type: Number, |
||||
default: 1300, |
||||
required: false, |
||||
}, |
||||
value: { |
||||
type: String, |
||||
required: false, |
||||
}, |
||||
disabled: { |
||||
type: Boolean, |
||||
required: false, |
||||
default: false, |
||||
}, |
||||
rows: { |
||||
type: String, |
||||
required: false, |
||||
}, |
||||
kind: { |
||||
type: String, |
||||
required: false, |
||||
}, |
||||
multi: { |
||||
type: Boolean, |
||||
default: true, |
||||
required: false, |
||||
}, |
||||
}, |
||||
data() { |
||||
return { |
||||
ids: '', |
||||
names: '', |
||||
materailList: [], |
||||
fetching: false, |
||||
queryParam: { |
||||
q: '', |
||||
current: 1, |
||||
pageSize: 10, |
||||
total: 0, |
||||
}, |
||||
checkedList: [], |
||||
checkAll: false, |
||||
indeterminate: false, |
||||
} |
||||
}, |
||||
mounted() { |
||||
this.ids = this.value |
||||
// this.loadData() |
||||
}, |
||||
watch: { |
||||
value(val) { |
||||
this.ids = val |
||||
}, |
||||
}, |
||||
model: { |
||||
prop: 'value', |
||||
event: 'change', |
||||
}, |
||||
methods: { |
||||
onCheckAllChange(e) { |
||||
Object.assign(this, { |
||||
checkedList: e.target.checked ? this.materailList.map((item) => item.mBarCode) : [], |
||||
indeterminate: false, |
||||
checkAll: e.target.checked, |
||||
}) |
||||
}, |
||||
onchange(event, item) { |
||||
let index = this.checkedList.indexOf(item.mBarCode) |
||||
if (index !== -1) { |
||||
this.checkedList.splice(index, 1) |
||||
} else { |
||||
this.checkedList.push(item.mBarCode) |
||||
} |
||||
this.indeterminate = !!this.checkedList.length && this.checkedList.length < this.materailList.length |
||||
this.checkAll = this.checkedList.length === this.materailList.length |
||||
}, |
||||
confirmSelected() { |
||||
let ids = this.checkedList.join(',') |
||||
this.$emit('change', ids) |
||||
this.open = false |
||||
this.loadData() |
||||
}, |
||||
changeCurrent(current) { |
||||
this.queryParam.current = current |
||||
this.loadData() |
||||
}, |
||||
fetchUser(value) { |
||||
clearTimeout(this.timeout) |
||||
this.timeout = setTimeout(() => { |
||||
this.queryParam.q = value |
||||
this.loadData(1) |
||||
}, 800) |
||||
}, |
||||
getQueryParams() { |
||||
let param = {} |
||||
param.mpList = getMpListShort(Vue.ls.get('materialPropertyList')) //扩展属性 |
||||
param.page = this.queryParam.current |
||||
param.rows = this.queryParam.pageSize |
||||
param.q = this.queryParam.q |
||||
return filterObj(param) |
||||
}, |
||||
async loadData(arg) { |
||||
if (arg === 1) { |
||||
this.queryParam.current = 1 |
||||
} |
||||
this.fetching = true |
||||
let params = this.getQueryParams() //查询条件 |
||||
await getMaterialBySelect(params) |
||||
.then((res) => { |
||||
if (res) { |
||||
this.checkedList = [] |
||||
this.materailList = res.rows |
||||
this.queryParam.total = res.total || 0 |
||||
} |
||||
}) |
||||
.finally(() => { |
||||
this.fetching = false |
||||
}) |
||||
}, |
||||
initComp(name) { |
||||
this.names = name |
||||
}, |
||||
onSearch() { |
||||
if (this.kind === 'material') { |
||||
let param = { |
||||
barCode: this.names, |
||||
mpList: getMpListShort(Vue.ls.get('materialPropertyList')), //扩展属性 |
||||
prefixNo: this.prefixNo, |
||||
} |
||||
getMaterialByBarCode(param).then((res) => { |
||||
if (res && res.code === 200) { |
||||
let mList = res.data |
||||
if (mList && mList.length === 1) { |
||||
//如果条码可以查到商品,则直接加载,不用弹窗再选择 |
||||
this.$emit('change', this.names) |
||||
} else { |
||||
this.$refs.selectModal.showModal(this.names) |
||||
} |
||||
} |
||||
}) |
||||
} else { |
||||
this.$refs.selectModal.showModal() |
||||
} |
||||
}, |
||||
selectOK(rows, idstr) { |
||||
console.log('选中id', idstr) |
||||
if (!rows) { |
||||
this.ids = '' |
||||
} else { |
||||
this.names = idstr |
||||
this.ids = idstr |
||||
} |
||||
this.$emit('change', this.ids) |
||||
}, |
||||
handleChange(value) { |
||||
this.names = value |
||||
this.$emit('change', value) |
||||
}, |
||||
focus() { |
||||
if (this.names) { |
||||
this.queryParam.q = '' |
||||
this.loadData() |
||||
} |
||||
}, |
||||
}, |
||||
} |
||||
</script> |
||||
|
||||
<style lang="less" scoped> |
||||
/deep/ .drop-down-Style { |
||||
width: auto; |
||||
} |
||||
.select-option { |
||||
display: flex; |
||||
justify-content: space-between; |
||||
padding-top: 5px; |
||||
width: 1000px; |
||||
} |
||||
.mr10 { |
||||
margin-right: 10px; |
||||
} |
||||
</style> |
@ -0,0 +1,87 @@
|
||||
<template> |
||||
<div> |
||||
<a-input-search |
||||
v-model="materialNames" |
||||
placeholder="请选择商品" |
||||
readOnly |
||||
@search="onSearchMaterial"> |
||||
</a-input-search> |
||||
<j-select-material-modal ref="selectModal" :modal-width="modalWidth" :multi="multi" :bar-code="value" @ok="selectOK" @initComp="initComp"/> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
import JSelectMaterialModal from './modal/JSelectMaterialModal' |
||||
|
||||
export default { |
||||
name: 'JSelectMaterial', |
||||
components: {JSelectMaterialModal}, |
||||
props: { |
||||
modalWidth: { |
||||
type: Number, |
||||
default: 1300, |
||||
required: false |
||||
}, |
||||
value: { |
||||
type: String, |
||||
required: false |
||||
}, |
||||
disabled: { |
||||
type: Boolean, |
||||
required: false, |
||||
default: false |
||||
}, |
||||
multi: { |
||||
type: Boolean, |
||||
default: true, |
||||
required: false |
||||
} |
||||
}, |
||||
data() { |
||||
return { |
||||
materialIds: "", |
||||
materialNames: "", |
||||
} |
||||
}, |
||||
mounted() { |
||||
this.materialIds = this.value |
||||
}, |
||||
watch: { |
||||
value(val) { |
||||
this.materialIds = val |
||||
} |
||||
}, |
||||
model: { |
||||
prop: 'value', |
||||
event: 'change' |
||||
}, |
||||
methods: { |
||||
initComp(barCode) { |
||||
this.materialNames = barCode |
||||
}, |
||||
onSearchMaterial() { |
||||
this.$refs.selectModal.showModal() |
||||
}, |
||||
selectOK(rows, idstr) { |
||||
console.log("选中商品", rows) |
||||
console.log("选中商品id", idstr) |
||||
if (!rows) { |
||||
this.materialNames = '' |
||||
this.materialIds = '' |
||||
} else { |
||||
let temp = '' |
||||
for (let item of rows) { |
||||
temp += ',' + item.mBarCode |
||||
} |
||||
this.materialNames = temp.substring(1) |
||||
this.materialIds = idstr |
||||
} |
||||
this.$emit("change", this.materialIds) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,47 @@
|
||||
<template> |
||||
<!-- 定义在这里的参数都是不可在外部覆盖的,防止出现问题 --> |
||||
<j-select-biz-component |
||||
:value="value" |
||||
:ellipsisLength="25" |
||||
:listUrl="url.list" |
||||
:columns="columns" |
||||
v-on="$listeners" |
||||
v-bind="attrs" |
||||
/> |
||||
</template> |
||||
|
||||
<script> |
||||
import JSelectBizComponent from './JSelectBizComponent' |
||||
|
||||
export default { |
||||
name: 'JSelectMultiUser', |
||||
components: { JSelectBizComponent }, |
||||
props: ['value'], |
||||
data() { |
||||
return { |
||||
url: { list: '/sys/user/list' }, |
||||
columns: [ |
||||
{ title: '姓名', align: 'center', width: '25%', widthRight: '70%', dataIndex: 'realname' }, |
||||
{ title: '账号', align: 'center', width: '25%', dataIndex: 'username' }, |
||||
{ title: '电话', align: 'center', width: '20%', dataIndex: 'phone' }, |
||||
{ title: '出生日期', align: 'center', width: '20%', dataIndex: 'birthday' } |
||||
], |
||||
// 定义在这里的参数都是可以在外部传递覆盖的,可以更灵活的定制化使用的组件 |
||||
default: { |
||||
name: '用户', |
||||
width: 1200, |
||||
displayKey: 'realname', |
||||
returnKeys: ['id', 'username'], |
||||
queryParamText: '账号', |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
attrs() { |
||||
return Object.assign(this.default, this.$attrs) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="less" scoped></style> |
@ -0,0 +1,37 @@
|
||||
<template> |
||||
<j-select-biz-component :width="1000" v-bind="configs" v-on="$listeners"/> |
||||
</template> |
||||
|
||||
<script> |
||||
import JSelectBizComponent from './JSelectBizComponent' |
||||
|
||||
export default { |
||||
name: 'JSelectPosition', |
||||
components: { JSelectBizComponent }, |
||||
props: ['value'], |
||||
data() { |
||||
return { |
||||
settings: { |
||||
name: '职务', |
||||
displayKey: 'name', |
||||
returnKeys: ['id', 'code'], |
||||
listUrl: '/sys/position/list', |
||||
queryParamCode: 'name', |
||||
queryParamText: '职务名称', |
||||
columns: [ |
||||
{ title: '职务名称', dataIndex: 'name', align: 'center', width: '30%', widthRight: '70%' }, |
||||
{ title: '职务编码', dataIndex: 'code', align: 'center', width: '35%' }, |
||||
{ title: '职级', dataIndex: 'rank_dictText', align: 'center', width: '25%' } |
||||
] |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
configs() { |
||||
return Object.assign({ value: this.value }, this.settings, this.$attrs) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="less" scoped></style> |
@ -0,0 +1,38 @@
|
||||
<template> |
||||
<j-select-biz-component |
||||
:value="value" |
||||
|
||||
name="角色" |
||||
displayKey="roleName" |
||||
|
||||
:returnKeys="returnKeys" |
||||
:listUrl="url.list" |
||||
:columns="columns" |
||||
queryParamText="角色编码" |
||||
|
||||
v-on="$listeners" |
||||
v-bind="$attrs" |
||||
/> |
||||
</template> |
||||
|
||||
<script> |
||||
import JSelectBizComponent from './JSelectBizComponent' |
||||
|
||||
export default { |
||||
name: 'JSelectMultiUser', |
||||
components: { JSelectBizComponent }, |
||||
props: ['value'], |
||||
data() { |
||||
return { |
||||
returnKeys: ['id', 'roleCode'], |
||||
url: { list: '/sys/role/list' }, |
||||
columns: [ |
||||
{ title: '角色名称', dataIndex: 'roleName', align: 'center', width: 120 }, |
||||
{ title: '角色编码', dataIndex: 'roleCode', align: 'center', width: 120 } |
||||
] |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="less" scoped></style> |
@ -0,0 +1,87 @@
|
||||
<template> |
||||
<div> |
||||
<a-input-search |
||||
v-model="materialNames" |
||||
placeholder="请选择序列号商品" |
||||
readOnly |
||||
@search="onSearchMaterial"> |
||||
</a-input-search> |
||||
<j-select-serial-material-modal ref="selectModal" :modal-width="modalWidth" :multi="multi" :bar-code="value" @ok="selectOK" @initComp="initComp"/> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
import JSelectSerialMaterialModal from './modal/JSelectSerialMaterialModal' |
||||
|
||||
export default { |
||||
name: 'JSelectSerialMaterial', |
||||
components: {JSelectSerialMaterialModal}, |
||||
props: { |
||||
modalWidth: { |
||||
type: Number, |
||||
default: 1100, |
||||
required: false |
||||
}, |
||||
value: { |
||||
type: String, |
||||
required: false |
||||
}, |
||||
disabled: { |
||||
type: Boolean, |
||||
required: false, |
||||
default: false |
||||
}, |
||||
multi: { |
||||
type: Boolean, |
||||
default: true, |
||||
required: false |
||||
} |
||||
}, |
||||
data() { |
||||
return { |
||||
materialIds: "", |
||||
materialNames: "", |
||||
} |
||||
}, |
||||
mounted() { |
||||
this.materialIds = this.value |
||||
}, |
||||
watch: { |
||||
value(val) { |
||||
this.materialIds = val |
||||
} |
||||
}, |
||||
model: { |
||||
prop: 'value', |
||||
event: 'change' |
||||
}, |
||||
methods: { |
||||
initComp(barCode) { |
||||
this.materialNames = barCode |
||||
}, |
||||
onSearchMaterial() { |
||||
this.$refs.selectModal.showModal() |
||||
}, |
||||
selectOK(rows, idstr) { |
||||
console.log("选中序列号商品", rows) |
||||
console.log("选中序列号商品id", idstr) |
||||
if (!rows) { |
||||
this.materialNames = '' |
||||
this.materialIds = '' |
||||
} else { |
||||
let temp = '' |
||||
for (let item of rows) { |
||||
temp += ',' + item.mBarCode |
||||
} |
||||
this.materialNames = temp.substring(1) |
||||
this.materialIds = idstr |
||||
} |
||||
this.$emit("change", this.materialIds) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,89 @@
|
||||
<template> |
||||
<div> |
||||
<a-input-search |
||||
v-model="userNames" |
||||
placeholder="请先选择用户" |
||||
readOnly |
||||
unselectable="on" |
||||
@search="onSearchDepUser"> |
||||
<a-button slot="enterButton" :disabled="disabled">选择用户</a-button> |
||||
</a-input-search> |
||||
<j-select-user-by-dep-modal ref="selectModal" :modal-width="modalWidth" :multi="multi" @ok="selectOK" :user-ids="value" @initComp="initComp"/> |
||||
</div> |
||||
</template> |
||||
|
||||
<script> |
||||
import JSelectUserByDepModal from './modal/JSelectUserByDepModal' |
||||
|
||||
export default { |
||||
name: 'JSelectUserByDep', |
||||
components: {JSelectUserByDepModal}, |
||||
props: { |
||||
modalWidth: { |
||||
type: Number, |
||||
default: 1250, |
||||
required: false |
||||
}, |
||||
value: { |
||||
type: String, |
||||
required: false |
||||
}, |
||||
disabled: { |
||||
type: Boolean, |
||||
required: false, |
||||
default: false |
||||
}, |
||||
multi: { |
||||
type: Boolean, |
||||
default: true, |
||||
required: false |
||||
}, |
||||
}, |
||||
data() { |
||||
return { |
||||
userIds: "", |
||||
userNames: "" |
||||
} |
||||
}, |
||||
mounted() { |
||||
this.userIds = this.value |
||||
}, |
||||
watch: { |
||||
value(val) { |
||||
this.userIds = val |
||||
} |
||||
}, |
||||
model: { |
||||
prop: 'value', |
||||
event: 'change' |
||||
}, |
||||
methods: { |
||||
initComp(userNames) { |
||||
this.userNames = userNames |
||||
}, |
||||
onSearchDepUser() { |
||||
this.$refs.selectModal.showModal() |
||||
}, |
||||
selectOK(rows, idstr) { |
||||
console.log("当前选中用户", rows) |
||||
console.log("当前选中用户ID", idstr) |
||||
if (!rows) { |
||||
this.userNames = '' |
||||
this.userIds = '' |
||||
} else { |
||||
let temp = '' |
||||
for (let item of rows) { |
||||
temp += ',' + item.realname |
||||
} |
||||
this.userNames = temp.substring(1) |
||||
this.userIds = idstr |
||||
} |
||||
this.$emit("change", this.userIds) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,137 @@
|
||||
# JSelectDepart 部门选择组件 |
||||
选择部门组件,存储部门ID,显示部门名称 |
||||
|
||||
## 参数配置 |
||||
| 参数 | 类型 | 必填 |说明| |
||||
|--------------|---------|----|---------| |
||||
| modalWidth |Number | | 弹框宽度 默认500 | |
||||
| multi |Boolean | | 是否多选 默认false | |
||||
| rootOpened |Boolean | | 是否展开根节点 默认true | |
||||
| disabled |Boolean | | 是否禁用 默认false| |
||||
|
||||
使用示例 |
||||
---- |
||||
```vue |
||||
<template> |
||||
<a-form :form="form"> |
||||
<a-form-item label="部门选择v-decorator" style="width: 300px"> |
||||
<j-select-depart v-decorator="['bumen']"/> |
||||
{{ getFormFieldValue('bumen') }} |
||||
</a-form-item> |
||||
|
||||
<a-form-item label="部门选择v-model" style="width: 300px"> |
||||
<j-select-depart v-model="bumen"/> |
||||
{{ bumen }} |
||||
</a-form-item> |
||||
|
||||
<a-form-item label="部门多选v-model" style="width: 300px"> |
||||
<j-select-depart v-model="bumens" :multi="true"/> |
||||
{{ bumens }} |
||||
</a-form-item> |
||||
|
||||
</a-form > |
||||
</template> |
||||
|
||||
<script> |
||||
import JSelectDepart from '@/components/jeecgbiz/JSelectDepart' |
||||
export default { |
||||
components: {JSelectDepart}, |
||||
data() { |
||||
return { |
||||
form: this.$form.createForm(this), |
||||
bumen:"", |
||||
bumens:"" |
||||
} |
||||
}, |
||||
methods:{ |
||||
getFormFieldValue(field){ |
||||
return this.form.getFieldValue(field) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
``` |
||||
# JSelectMultiUser 用户多选组件 |
||||
|
||||
使用示例 |
||||
---- |
||||
```vue |
||||
<template> |
||||
<a-form :form="form"> |
||||
<a-form-item label="用户选择v-decorator" style="width: 500px"> |
||||
<j-select-multi-user v-decorator="['users']"/> |
||||
{{ getFormFieldValue('users') }} |
||||
</a-form-item> |
||||
|
||||
<a-form-item label="用户选择v-model" style="width: 500px"> |
||||
<j-select-multi-user v-model="users" ></j-select-multi-user> |
||||
{{ users }} |
||||
</a-form-item> |
||||
|
||||
</a-form > |
||||
</template> |
||||
|
||||
<script> |
||||
import JSelectMultiUser from '@/components/jeecgbiz/JSelectMultiUser' |
||||
export default { |
||||
components: {JSelectMultiUser}, |
||||
data() { |
||||
return { |
||||
form: this.$form.createForm(this), |
||||
users:"", |
||||
} |
||||
}, |
||||
methods:{ |
||||
getFormFieldValue(field){ |
||||
return this.form.getFieldValue(field) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
``` |
||||
|
||||
# JSelectUserByDep 根据部门选择用户 |
||||
|
||||
## 参数配置 |
||||
| 参数 | 类型 | 必填 |说明| |
||||
|--------------|---------|----|---------| |
||||
| modalWidth |Number | | 弹框宽度 默认1250 | |
||||
| disabled |Boolean | | 是否禁用 | |
||||
|
||||
使用示例 |
||||
---- |
||||
```vue |
||||
<template> |
||||
<a-form :form="form"> |
||||
<a-form-item label="用户选择v-decorator" style="width: 500px"> |
||||
<j-select-user-by-dep v-decorator="['users']"/> |
||||
{{ getFormFieldValue('users') }} |
||||
</a-form-item> |
||||
|
||||
<a-form-item label="用户选择v-model" style="width: 500px"> |
||||
<j-select-user-by-dep v-model="users" ></j-select-user-by-dep> |
||||
{{ users }} |
||||
</a-form-item> |
||||
|
||||
</a-form > |
||||
</template> |
||||
|
||||
<script> |
||||
import JSelectUserByDep from '@/components/jeecgbiz/JSelectUserByDep' |
||||
export default { |
||||
components: {JSelectUserByDep}, |
||||
data() { |
||||
return { |
||||
form: this.$form.createForm(this), |
||||
users:"", |
||||
} |
||||
}, |
||||
methods:{ |
||||
getFormFieldValue(field){ |
||||
return this.form.getFieldValue(field) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
``` |
||||
|
@ -0,0 +1,248 @@
|
||||
<template> |
||||
<a-modal |
||||
:width="modalWidth" |
||||
:visible="visible" |
||||
:title="title" |
||||
@ok="handleSubmit" |
||||
@cancel="close" |
||||
cancelText="关闭" |
||||
style="top:5%;height: 100%;overflow-y: hidden" |
||||
wrapClassName="ant-modal-cust-warp" |
||||
> |
||||
<a-row :gutter="10" style="padding: 10px; margin: -10px"> |
||||
<a-col :md="24" :sm="24"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="onSearch"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="6" :sm="8"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="批号"> |
||||
<a-input placeholder="请输入批号" v-model="queryParam.name"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-button type="primary" @click="onSearch">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset(1)">重置</a-button> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
<a-table |
||||
ref="table" |
||||
:scroll="scrollTrigger" |
||||
size="middle" |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:dataSource="dataSource" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange,type: getType}" |
||||
:loading="loading" |
||||
:customRow="rowAction"> |
||||
</a-table> |
||||
</div> |
||||
</a-col> |
||||
</a-row> |
||||
</a-modal> |
||||
</template> |
||||
|
||||
<script> |
||||
import { getAction } from '@/api/manage' |
||||
import {getBatchNumberList} from '@/api/api' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
|
||||
export default { |
||||
name: 'JSelectBatchModal', |
||||
mixins:[JeecgListMixin], |
||||
components: {}, |
||||
props: ['modalWidth', 'rows', 'multi', 'barCode'], |
||||
data() { |
||||
return { |
||||
queryParam: { |
||||
name: "", |
||||
depotId: '', |
||||
barCode: '' |
||||
}, |
||||
labelCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 5 }, |
||||
}, |
||||
wrapperCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 16 }, |
||||
}, |
||||
categoryTree:[], |
||||
columns: [ |
||||
{dataIndex: 'batchNumber', title: '批号', width: 100, align: 'left'}, |
||||
{dataIndex: 'barCode', title: '物料编码', width: 100}, |
||||
{dataIndex: 'name', title: '名称', width: 100}, |
||||
{dataIndex: 'standard', title: '规格', width: 80}, |
||||
{dataIndex: 'model', title: '型号', width: 80}, |
||||
{dataIndex: 'expirationDateStr', title: '有效期至', width: 80}, |
||||
{dataIndex: 'totalNum', title: '库存', width: 80} |
||||
], |
||||
scrollTrigger: {}, |
||||
dataSource: [], |
||||
selectedRowKeys: [], |
||||
selectRows: [], |
||||
selectIds: [], |
||||
title: '选择批号', |
||||
isorter: { |
||||
column: 'createTime', |
||||
order: 'desc' |
||||
}, |
||||
departTree: [], |
||||
depotList: [], |
||||
visible: false, |
||||
form: this.$form.createForm(this), |
||||
loading: false, |
||||
expandedKeys: [], |
||||
} |
||||
}, |
||||
computed: { |
||||
// 计算属性的 getter |
||||
getType: function () { |
||||
return this.multi == true ? 'checkbox' : 'radio'; |
||||
} |
||||
}, |
||||
watch: { |
||||
barCode: { |
||||
immediate: true, |
||||
handler() { |
||||
this.initBarCode() |
||||
} |
||||
}, |
||||
}, |
||||
created() { |
||||
this.loadData() |
||||
}, |
||||
methods: { |
||||
initBarCode() { |
||||
if (this.barCode) { |
||||
this.$emit('initComp', this.barCode) |
||||
} else { |
||||
// JSelectUserByDep组件bug issues/I16634 |
||||
this.$emit('initComp', '') |
||||
} |
||||
}, |
||||
async loadData(arg) { |
||||
if(this.rows) { |
||||
if(JSON.parse(this.rows).depotId && JSON.parse(this.rows).barCode ){ |
||||
this.queryParam.depotId = JSON.parse(this.rows).depotId-0 |
||||
this.queryParam.barCode = JSON.parse(this.rows).barCode |
||||
} |
||||
} |
||||
if (arg === 1) { |
||||
this.ipagination.current = 1; |
||||
} |
||||
this.loading = true |
||||
let params = this.getQueryParams()//查询条件 |
||||
await getBatchNumberList(params).then((res) => { |
||||
if (res && res.code === 200) { |
||||
this.dataSource = res.data.rows |
||||
this.ipagination.total = res.data.total |
||||
} |
||||
}).finally(() => { |
||||
this.loading = false |
||||
}) |
||||
}, |
||||
showModal() { |
||||
this.visible = true; |
||||
this.loadData(); |
||||
this.form.resetFields(); |
||||
}, |
||||
getQueryParams() { |
||||
let param = Object.assign({}, this.queryParam, this.isorter); |
||||
return param; |
||||
}, |
||||
getQueryField() { |
||||
let str = 'id,'; |
||||
for (let a = 0; a < this.columns.length; a++) { |
||||
str += ',' + this.columns[a].dataIndex; |
||||
} |
||||
return str; |
||||
}, |
||||
searchReset(num) { |
||||
let that = this; |
||||
if (num !== 0) { |
||||
if(this.rows) { |
||||
this.queryParam.name='' |
||||
if(JSON.parse(this.rows).depotId && JSON.parse(this.rows).barCode ){ |
||||
this.queryParam.depotId = JSON.parse(this.rows).depotId-0 |
||||
this.queryParam.barCode = JSON.parse(this.rows).barCode |
||||
} |
||||
} |
||||
that.loadData(1); |
||||
} |
||||
that.selectedRowKeys = []; |
||||
that.selectIds = []; |
||||
}, |
||||
close() { |
||||
this.searchReset(0); |
||||
this.visible = false; |
||||
}, |
||||
handleSubmit() { |
||||
let that = this; |
||||
this.getSelectRows(); |
||||
that.$emit('ok', that.selectRows, that.selectIds); |
||||
that.searchReset(0) |
||||
that.close(); |
||||
}, |
||||
//获取选择信息 |
||||
getSelectRows(rowId) { |
||||
let dataSource = this.dataSource; |
||||
let ids = ""; |
||||
this.selectRows = []; |
||||
for (let i = 0, len = dataSource.length; i < len; i++) { |
||||
if (this.selectedRowKeys.includes(dataSource[i].id)) { |
||||
this.selectRows.push(dataSource[i]); |
||||
ids = ids + "," + dataSource[i].batchNumber |
||||
} |
||||
} |
||||
this.selectIds = ids.substring(1); |
||||
}, |
||||
onSelectChange(selectedRowKeys, selectionRows) { |
||||
this.selectedRowKeys = selectedRowKeys; |
||||
this.selectionRows = selectionRows; |
||||
}, |
||||
onSearch() { |
||||
this.loadData(1); |
||||
}, |
||||
modalFormOk() { |
||||
this.loadData(); |
||||
}, |
||||
rowAction(record, index) { |
||||
return { |
||||
on: { |
||||
click: () => { |
||||
let arr = [] |
||||
arr.push(record.id) |
||||
this.selectedRowKeys = arr |
||||
}, |
||||
dblclick: () => { |
||||
let arr = [] |
||||
arr.push(record.id) |
||||
this.selectedRowKeys = arr |
||||
this.handleSubmit() |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
.ant-table-tbody .ant-table-row td { |
||||
padding-top: 10px; |
||||
padding-bottom: 10px; |
||||
} |
||||
|
||||
#components-layout-demo-custom-trigger .trigger { |
||||
font-size: 18px; |
||||
line-height: 64px; |
||||
padding: 0 24px; |
||||
cursor: pointer; |
||||
transition: color .3s; |
||||
} |
||||
</style> |
@ -0,0 +1,241 @@
|
||||
<template> |
||||
<a-modal |
||||
title="选择部门" |
||||
:width="modalWidth" |
||||
:visible="visible" |
||||
:confirmLoading="confirmLoading" |
||||
@ok="handleSubmit" |
||||
@cancel="handleCancel" |
||||
cancelText="关闭"> |
||||
<a-spin tip="Loading..." :spinning="false"> |
||||
<a-input-search style="margin-bottom: 1px" placeholder="请输入部门名称按回车进行搜索" @search="onSearch" /> |
||||
<a-tree |
||||
checkable |
||||
:treeData="treeData" |
||||
:checkStrictly="true" |
||||
@check="onCheck" |
||||
@select="onSelect" |
||||
@expand="onExpand" |
||||
:autoExpandParent="autoExpandParent" |
||||
:expandedKeys="expandedKeys" |
||||
:checkedKeys="checkedKeys"> |
||||
|
||||
<template slot="title" slot-scope="{title}"> |
||||
<span v-if="title.indexOf(searchValue) > -1"> |
||||
{{title.substr(0, title.indexOf(searchValue))}} |
||||
<span style="color: #f50">{{searchValue}}</span> |
||||
{{title.substr(title.indexOf(searchValue) + searchValue.length)}} |
||||
</span> |
||||
<span v-else>{{title}}</span> |
||||
</template> |
||||
</a-tree> |
||||
|
||||
</a-spin> |
||||
</a-modal> |
||||
</template> |
||||
|
||||
<script> |
||||
import { queryDepartTreeList } from '@/api/api' |
||||
export default { |
||||
name: 'JSelectDepartModal', |
||||
props:['modalWidth','multi','rootOpened','departId'], |
||||
data(){ |
||||
return { |
||||
visible:false, |
||||
confirmLoading:false, |
||||
treeData:[], |
||||
autoExpandParent:true, |
||||
expandedKeys:[], |
||||
dataList:[], |
||||
checkedKeys:[], |
||||
checkedRows:[], |
||||
searchValue:"" |
||||
} |
||||
}, |
||||
created(){ |
||||
this.loadDepart(); |
||||
}, |
||||
watch:{ |
||||
departId(){ |
||||
this.initDepartComponent() |
||||
}, |
||||
visible: { |
||||
handler() { |
||||
if (this.departId) { |
||||
this.checkedKeys = this.departId.split(","); |
||||
// console.log('this.departId', this.departId) |
||||
} else { |
||||
this.checkedKeys = []; |
||||
} |
||||
} |
||||
} |
||||
}, |
||||
methods:{ |
||||
show(){ |
||||
this.visible=true |
||||
this.checkedRows=[] |
||||
this.checkedKeys=[] |
||||
}, |
||||
loadDepart(){ |
||||
queryDepartTreeList().then(res=>{ |
||||
if(res.success){ |
||||
let arr = [...res.result] |
||||
this.reWriterWithSlot(arr) |
||||
this.treeData = arr |
||||
this.initDepartComponent() |
||||
if(this.rootOpened){ |
||||
this.initExpandedKeys(res.result) |
||||
} |
||||
} |
||||
}) |
||||
}, |
||||
initDepartComponent(){ |
||||
let names = '' |
||||
if(this.departId){ |
||||
let currDepartId = this.departId |
||||
for(let item of this.dataList){ |
||||
if(currDepartId.indexOf(item.key)>=0){ |
||||
names+=","+item.title |
||||
} |
||||
} |
||||
if(names){ |
||||
names = names.substring(1) |
||||
} |
||||
} |
||||
this.$emit("initComp",names) |
||||
}, |
||||
reWriterWithSlot(arr){ |
||||
for(let item of arr){ |
||||
if(item.children && item.children.length>0){ |
||||
this.reWriterWithSlot(item.children) |
||||
let temp = Object.assign({},item) |
||||
temp.children = {} |
||||
this.dataList.push(temp) |
||||
}else{ |
||||
this.dataList.push(item) |
||||
item.scopedSlots={ title: 'title' } |
||||
} |
||||
} |
||||
}, |
||||
initExpandedKeys(arr){ |
||||
if(arr && arr.length>0){ |
||||
let keys = [] |
||||
for(let item of arr){ |
||||
if(item.children && item.children.length>0){ |
||||
keys.push(item.id) |
||||
} |
||||
} |
||||
this.expandedKeys=[...keys] |
||||
}else{ |
||||
this.expandedKeys=[] |
||||
} |
||||
}, |
||||
onCheck (checkedKeys,info) { |
||||
if(!this.multi){ |
||||
let arr = checkedKeys.checked.filter(item => this.checkedKeys.indexOf(item) < 0) |
||||
this.checkedKeys = [...arr] |
||||
this.checkedRows = (this.checkedKeys.length === 0) ? [] : [info.node.dataRef] |
||||
}else{ |
||||
this.checkedKeys = checkedKeys.checked |
||||
this.checkedRows = this.getCheckedRows(this.checkedKeys) |
||||
} |
||||
}, |
||||
onSelect(selectedKeys,info) { |
||||
let keys = [] |
||||
keys.push(selectedKeys[0]) |
||||
if(!this.checkedKeys || this.checkedKeys.length===0 || !this.multi){ |
||||
this.checkedKeys = [...keys] |
||||
this.checkedRows=[info.node.dataRef] |
||||
}else{ |
||||
let currKey = info.node.dataRef.key |
||||
if(this.checkedKeys.indexOf(currKey)>=0){ |
||||
this.checkedKeys = this.checkedKeys.filter(item=> item !==currKey) |
||||
}else{ |
||||
this.checkedKeys.push(...keys) |
||||
} |
||||
} |
||||
this.checkedRows = this.getCheckedRows(this.checkedKeys) |
||||
}, |
||||
onExpand (expandedKeys) { |
||||
this.expandedKeys = expandedKeys |
||||
this.autoExpandParent = false |
||||
}, |
||||
handleSubmit(){ |
||||
if(!this.checkedKeys || this.checkedKeys.length==0){ |
||||
this.$emit("ok",'') |
||||
}else{ |
||||
this.$emit("ok",this.checkedRows,this.checkedKeys.join(",")) |
||||
} |
||||
this.handleClear() |
||||
}, |
||||
handleCancel(){ |
||||
this.handleClear() |
||||
}, |
||||
handleClear(){ |
||||
this.visible=false |
||||
this.checkedKeys=[] |
||||
}, |
||||
getParentKey(currKey,treeData){ |
||||
let parentKey |
||||
for (let i = 0; i < treeData.length; i++) { |
||||
const node = treeData[i] |
||||
if (node.children) { |
||||
if (node.children.some(item => item.key === currKey)) { |
||||
parentKey = node.key |
||||
} else if (this.getParentKey(currKey, node.children)) { |
||||
parentKey = this.getParentKey(currKey, node.children) |
||||
} |
||||
} |
||||
} |
||||
return parentKey |
||||
}, |
||||
onSearch(value){ |
||||
const expandedKeys = this.dataList.map((item) => { |
||||
if (item.title.indexOf(value) > -1) { |
||||
return this.getParentKey(item.key,this.treeData) |
||||
} |
||||
return null |
||||
}).filter((item, i, self) => item && self.indexOf(item) === i) |
||||
|
||||
Object.assign(this, { |
||||
expandedKeys, |
||||
searchValue: value, |
||||
autoExpandParent: true, |
||||
}) |
||||
|
||||
|
||||
}, |
||||
// 根据 checkedKeys 获取 rows |
||||
getCheckedRows(checkedKeys) { |
||||
const forChildren = (list, key) => { |
||||
for (let item of list) { |
||||
if (item.id === key) { |
||||
return item |
||||
} |
||||
if (item.children instanceof Array) { |
||||
let value = forChildren(item.children, key) |
||||
if (value != null) { |
||||
return value |
||||
} |
||||
} |
||||
} |
||||
return null |
||||
} |
||||
|
||||
let rows = [] |
||||
for (let key of checkedKeys) { |
||||
let row = forChildren(this.treeData, key) |
||||
if (row != null) { |
||||
rows.push(row) |
||||
} |
||||
} |
||||
return rows |
||||
} |
||||
} |
||||
} |
||||
|
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,370 @@
|
||||
<template> |
||||
<a-modal |
||||
:width="modalWidth" |
||||
:visible="visible" |
||||
:title="title" |
||||
@ok="handleSubmit" |
||||
@cancel="close" |
||||
cancelText="关闭" |
||||
style="top:5%;height: 100%;overflow-y: hidden" |
||||
wrapClassName="ant-modal-cust-warp" |
||||
> |
||||
<a-row :gutter="10" style="padding: 10px; margin: -10px"> |
||||
<a-col :md="24" :sm="24"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="onSearch"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="6" :sm="8"> |
||||
<a-form-item label="商品" :labelCol="{span: 5}" :wrapperCol="{span: 18, offset: 1}"> |
||||
<a-input ref="material" placeholder="物料编码、名称、规格、型号" v-model="queryParam.q"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="8"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="类别"> |
||||
<a-tree-select style="width:100%" :dropdownStyle="{maxHeight:'200px',overflow:'auto'}" allow-clear |
||||
:treeData="categoryTree" v-model="queryParam.categoryId" placeholder="请选择类别"> |
||||
</a-tree-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="8"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="仓库"> |
||||
<a-select placeholder="选择仓库" v-model="queryParam.depotId" @change="onDepotChange" |
||||
:dropdownMatchSelectWidth="false" showSearch optionFilterProp="children" allow-clear> |
||||
<a-select-option v-for="(item,index) in depotList" :key="index" :value="item.id"> |
||||
{{ item.depotName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-button type="primary" @click="loadData(1)">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset(1)">重置</a-button> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
<a-table |
||||
ref="table" |
||||
:scroll="scrollTrigger" |
||||
size="middle" |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange,type: getType}" |
||||
:loading="loading" |
||||
:customRow="rowAction" |
||||
@change="handleTableChange"> |
||||
<template slot="customRenderEnableSerialNumber" slot-scope="enableSerialNumber"> |
||||
<a-tag v-if="enableSerialNumber==1" color="green">有</a-tag> |
||||
<a-tag v-if="enableSerialNumber==0" color="orange">无</a-tag> |
||||
</template> |
||||
<template slot="customRenderEnableBatchNumber" slot-scope="enableBatchNumber"> |
||||
<a-tag v-if="enableBatchNumber==1" color="green">有</a-tag> |
||||
<a-tag v-if="enableBatchNumber==0" color="orange">无</a-tag> |
||||
</template> |
||||
</a-table> |
||||
</div> |
||||
</a-col> |
||||
</a-row> |
||||
</a-modal> |
||||
</template> |
||||
|
||||
<script> |
||||
import { httpAction, getAction } from '@/api/manage' |
||||
import {filterObj, getMpListShort} from '@/utils/util' |
||||
import {getMaterialBySelect, queryMaterialCategoryTreeList} from '@/api/api' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
import Vue from 'vue' |
||||
|
||||
export default { |
||||
name: 'JSelectMaterialModal', |
||||
mixins:[JeecgListMixin], |
||||
components: {}, |
||||
props: ['modalWidth', 'rows', 'multi', 'barCode'], |
||||
data() { |
||||
return { |
||||
queryParam: { |
||||
q: '', |
||||
depotId: '' |
||||
}, |
||||
labelCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 5 }, |
||||
}, |
||||
wrapperCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 16 }, |
||||
}, |
||||
categoryTree:[], |
||||
columns: [ |
||||
{dataIndex: 'mBarCode', title: '物料编码', width: 100, align: 'left'}, |
||||
{dataIndex: 'name', title: '名称', width: 120, ellipsis:true}, |
||||
{dataIndex: 'categoryName', title: '类别', width: 80}, |
||||
{dataIndex: 'standard', title: '规格', width: 80}, |
||||
{dataIndex: 'model', title: '型号', width: 80}, |
||||
{dataIndex: 'color', title: '颜色', width: 80}, |
||||
{dataIndex: 'unit', title: '单位', width: 70, ellipsis:true}, |
||||
{dataIndex: 'sku', title: '多属性', width: 80}, |
||||
{dataIndex: 'stock', title: '库存', width: 60}, |
||||
{dataIndex: 'expand', title: '扩展信息', width: 80, ellipsis:true}, |
||||
{dataIndex: 'enableSerialNumber', title: '序列号', width: 60, align: "center", |
||||
scopedSlots: { customRender: 'customRenderEnableSerialNumber' } |
||||
}, |
||||
{dataIndex: 'enableBatchNumber', title: '批号', width: 50, align: "center", |
||||
scopedSlots: { customRender: 'customRenderEnableBatchNumber' } |
||||
} |
||||
], |
||||
scrollTrigger: {}, |
||||
dataSource: [], |
||||
selectedRowKeys: [], |
||||
selectMaterialRows: [], |
||||
selectMaterialIds: [], |
||||
title: '选择商品', |
||||
ipagination: { |
||||
current: 1, |
||||
pageSize: 2, |
||||
pageSizeOptions: ['10', '20', '30'], |
||||
showTotal: (total, range) => { |
||||
return range[0] + '-' + range[1] + ' 共' + total + '条' |
||||
}, |
||||
showQuickJumper: true, |
||||
showSizeChanger: true, |
||||
total: 0 |
||||
}, |
||||
isorter: { |
||||
column: 'createTime', |
||||
order: 'desc' |
||||
}, |
||||
departTree: [], |
||||
depotList: [], |
||||
visible: false, |
||||
form: this.$form.createForm(this), |
||||
loading: false, |
||||
expandedKeys: [], |
||||
} |
||||
}, |
||||
computed: { |
||||
// 计算属性的 getter |
||||
getType: function () { |
||||
return this.multi == true ? 'checkbox' : 'radio'; |
||||
} |
||||
}, |
||||
watch: { |
||||
barCode: { |
||||
immediate: true, |
||||
handler() { |
||||
this.initBarCode() |
||||
} |
||||
}, |
||||
}, |
||||
created() { |
||||
// 该方法触发屏幕自适应 |
||||
this.resetScreenSize() |
||||
// this.loadTreeData() |
||||
// this.getDepotList() |
||||
}, |
||||
methods: { |
||||
initBarCode() { |
||||
if (this.barCode) { |
||||
this.$emit('initComp', this.barCode) |
||||
} else { |
||||
// JSelectUserByDep组件bug issues/I16634 |
||||
this.$emit('initComp', '') |
||||
} |
||||
}, |
||||
async loadData(arg) { |
||||
if (arg === 1) { |
||||
this.ipagination.current = 1; |
||||
} |
||||
this.loading = true |
||||
let params = this.getQueryParams()//查询条件 |
||||
if(this.visible) { |
||||
await getMaterialBySelect(params).then((res) => { |
||||
if (res) { |
||||
this.dataSource = res.rows |
||||
this.ipagination.total = res.total |
||||
if(res.total ===1) { |
||||
this.title = '选择商品【再次回车可以直接选中】' |
||||
this.$nextTick(() =>{ |
||||
this.$refs.material.focus() |
||||
}); |
||||
} else { |
||||
this.title = '选择商品' |
||||
} |
||||
} |
||||
}).finally(() => { |
||||
this.loading = false |
||||
}) |
||||
} |
||||
|
||||
}, |
||||
loadTreeData(){ |
||||
let that = this; |
||||
let params = {}; |
||||
params.id=''; |
||||
queryMaterialCategoryTreeList(params).then((res)=>{ |
||||
if(res){ |
||||
that.categoryTree = []; |
||||
for (let i = 0; i < res.length; i++) { |
||||
let temp = res[i]; |
||||
that.categoryTree.push(temp); |
||||
} |
||||
} |
||||
}) |
||||
}, |
||||
// 触发屏幕自适应 |
||||
resetScreenSize() { |
||||
let screenWidth = document.body.clientWidth; |
||||
if (screenWidth < 500) { |
||||
this.scrollTrigger = {x: 800}; |
||||
} else { |
||||
this.scrollTrigger = {}; |
||||
} |
||||
}, |
||||
showModal(barCode) { |
||||
this.visible = true; |
||||
this.title = '选择商品' |
||||
this.queryParam.q = barCode |
||||
this.$nextTick(() => this.$refs.material.focus()); |
||||
this.initDepotSelect() |
||||
this.loadData(); |
||||
this.loadTreeData() |
||||
this.getDepotList() |
||||
this.form.resetFields(); |
||||
}, |
||||
getQueryParams() { |
||||
let param = Object.assign({}, this.queryParam, this.isorter); |
||||
param.mpList = getMpListShort(Vue.ls.get('materialPropertyList')) //扩展属性 |
||||
param.page = this.ipagination.current; |
||||
param.rows = this.ipagination.pageSize; |
||||
return filterObj(param); |
||||
}, |
||||
getQueryField() { |
||||
let str = 'id,'; |
||||
for (let a = 0; a < this.columns.length; a++) { |
||||
str += ',' + this.columns[a].dataIndex; |
||||
} |
||||
return str; |
||||
}, |
||||
searchReset(num) { |
||||
let that = this; |
||||
if (num !== 0) { |
||||
that.queryParam = {}; |
||||
that.loadData(1); |
||||
} |
||||
that.selectedRowKeys = []; |
||||
that.selectMaterialIds = []; |
||||
}, |
||||
close() { |
||||
this.searchReset(0); |
||||
this.visible = false; |
||||
}, |
||||
handleTableChange(pagination, filters, sorter) { |
||||
if (Object.keys(sorter).length > 0) { |
||||
this.isorter.column = sorter.field; |
||||
this.isorter.order = 'ascend' === sorter.order ? 'asc' : 'desc'; |
||||
} |
||||
this.ipagination = pagination; |
||||
this.loadData(); |
||||
}, |
||||
handleSubmit() { |
||||
let that = this; |
||||
this.getSelectMaterialRows(); |
||||
that.$emit('ok', that.selectMaterialRows, that.selectMaterialIds); |
||||
that.searchReset(0) |
||||
that.close(); |
||||
}, |
||||
//获取选择信息 |
||||
getSelectMaterialRows(rowId) { |
||||
let dataSource = this.dataSource; |
||||
let materialIds = ""; |
||||
this.selectMaterialRows = []; |
||||
for (let i = 0, len = dataSource.length; i < len; i++) { |
||||
if (this.selectedRowKeys.includes(dataSource[i].id)) { |
||||
this.selectMaterialRows.push(dataSource[i]); |
||||
materialIds = materialIds + "," + dataSource[i].mBarCode |
||||
} |
||||
} |
||||
this.selectMaterialIds = materialIds.substring(1); |
||||
}, |
||||
getDepotList() { |
||||
let that = this; |
||||
getAction('/depot/findDepotByCurrentUser').then((res) => { |
||||
if(res.code === 200){ |
||||
that.depotList = res.data |
||||
} |
||||
}) |
||||
}, |
||||
initDepotSelect() { |
||||
if(this.rows) { |
||||
if(JSON.parse(this.rows).depotId){ |
||||
this.queryParam.depotId = JSON.parse(this.rows).depotId-0 |
||||
} |
||||
} |
||||
}, |
||||
onDepotChange(value) { |
||||
this.queryParam.depotId = value |
||||
}, |
||||
onSelectChange(selectedRowKeys, selectionRows) { |
||||
this.selectedRowKeys = selectedRowKeys; |
||||
this.selectionRows = selectionRows; |
||||
}, |
||||
onSearch() { |
||||
if(this.dataSource && this.dataSource.length===1) { |
||||
if(this.queryParam.q === this.dataSource[0].mBarCode|| |
||||
this.queryParam.q === this.dataSource[0].name|| |
||||
this.queryParam.q === this.dataSource[0].standard|| |
||||
this.queryParam.q === this.dataSource[0].model) { |
||||
let arr = [] |
||||
arr.push(this.dataSource[0].id) |
||||
this.selectedRowKeys = arr |
||||
this.handleSubmit() |
||||
} else { |
||||
this.loadData(1) |
||||
} |
||||
} else { |
||||
this.loadData(1) |
||||
} |
||||
}, |
||||
modalFormOk() { |
||||
this.loadData() |
||||
}, |
||||
rowAction(record, index) { |
||||
return { |
||||
on: { |
||||
click: () => { |
||||
let arr = [] |
||||
arr.push(record.id) |
||||
this.selectedRowKeys = arr |
||||
}, |
||||
dblclick: () => { |
||||
let arr = [] |
||||
arr.push(record.id) |
||||
this.selectedRowKeys = arr |
||||
this.handleSubmit() |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
.ant-table-tbody .ant-table-row td { |
||||
padding-top: 10px; |
||||
padding-bottom: 10px; |
||||
} |
||||
|
||||
#components-layout-demo-custom-trigger .trigger { |
||||
font-size: 18px; |
||||
line-height: 64px; |
||||
padding: 0 24px; |
||||
cursor: pointer; |
||||
transition: color .3s; |
||||
} |
||||
</style> |
@ -0,0 +1,262 @@
|
||||
<template> |
||||
<a-modal |
||||
:width="modalWidth" |
||||
:visible="visible" |
||||
:title="title" |
||||
@ok="handleSubmit" |
||||
@cancel="close" |
||||
cancelText="关闭" |
||||
style="margin-top: -70px" |
||||
wrapClassName="ant-modal-cust-warp" |
||||
> |
||||
<a-row :gutter="10" style="padding: 10px; margin: -10px"> |
||||
<a-col :md="24" :sm="24"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="onSearch"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="12" :sm="12"> |
||||
<a-form-item label="商品信息" :labelCol="{span: 5}" :wrapperCol="{span: 18, offset: 1}"> |
||||
<a-input placeholder="请输入条码、名称、规格、型号" v-model="queryParam.q"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="12" :sm="12"> |
||||
<a-button type="primary" @click="onSearch">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset(1)">重置</a-button> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
<a-table |
||||
ref="table" |
||||
:scroll="scrollTrigger" |
||||
size="middle" |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange,type: getType}" |
||||
:loading="loading" |
||||
:customRow="rowAction" |
||||
@change="handleTableChange"> |
||||
</a-table> |
||||
</div> |
||||
</a-col> |
||||
</a-row> |
||||
</a-modal> |
||||
</template> |
||||
|
||||
<script> |
||||
import {filterObj} from '@/utils/util' |
||||
import {getSerialMaterialBySelect} from '@/api/api' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
|
||||
export default { |
||||
name: 'JSelectSerialMaterialModal', |
||||
mixins:[JeecgListMixin], |
||||
components: {}, |
||||
props: ['modalWidth', 'multi', 'barCode'], |
||||
data() { |
||||
return { |
||||
queryParam: { |
||||
q: '' |
||||
}, |
||||
labelCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 5 }, |
||||
}, |
||||
wrapperCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 16 }, |
||||
}, |
||||
columns: [ |
||||
{dataIndex: 'mBarCode', title: '物料编码', width: 100, align: 'left'}, |
||||
{dataIndex: 'name', title: '名称', width: 100}, |
||||
{dataIndex: 'standard', title: '规格', width: 80}, |
||||
{dataIndex: 'model', title: '型号', width: 80} |
||||
], |
||||
scrollTrigger: {}, |
||||
dataSource: [], |
||||
selectedRowKeys: [], |
||||
selectMaterialRows: [], |
||||
selectMaterialIds: [], |
||||
title: '选择序列号商品', |
||||
ipagination: { |
||||
current: 1, |
||||
pageSize: 5, |
||||
pageSizeOptions: ['5', '10', '20', '30'], |
||||
showTotal: (total, range) => { |
||||
return range[0] + '-' + range[1] + ' 共' + total + '条' |
||||
}, |
||||
showQuickJumper: true, |
||||
showSizeChanger: true, |
||||
total: 0 |
||||
}, |
||||
isorter: { |
||||
column: 'createTime', |
||||
order: 'desc' |
||||
}, |
||||
visible: false, |
||||
form: this.$form.createForm(this), |
||||
loading: false, |
||||
expandedKeys: [], |
||||
} |
||||
}, |
||||
computed: { |
||||
// 计算属性的 getter |
||||
getType: function () { |
||||
return this.multi == true ? 'checkbox' : 'radio'; |
||||
} |
||||
}, |
||||
watch: { |
||||
barCode: { |
||||
immediate: true, |
||||
handler() { |
||||
this.initBarCode() |
||||
} |
||||
}, |
||||
}, |
||||
created() { |
||||
// 该方法触发屏幕自适应 |
||||
this.resetScreenSize() |
||||
this.loadData() |
||||
}, |
||||
methods: { |
||||
initBarCode() { |
||||
if (this.barCode) { |
||||
this.$emit('initComp', this.barCode) |
||||
} else { |
||||
// JSelectUserByDep组件bug issues/I16634 |
||||
this.$emit('initComp', '') |
||||
} |
||||
}, |
||||
async loadData(arg) { |
||||
if (arg === 1) { |
||||
this.ipagination.current = 1; |
||||
} |
||||
this.loading = true |
||||
let params = this.getQueryParams()//查询条件 |
||||
await getSerialMaterialBySelect(params).then((res) => { |
||||
if (res) { |
||||
this.dataSource = res.rows |
||||
this.ipagination.total = res.total |
||||
} |
||||
}).finally(() => { |
||||
this.loading = false |
||||
}) |
||||
}, |
||||
// 触发屏幕自适应 |
||||
resetScreenSize() { |
||||
let screenWidth = document.body.clientWidth; |
||||
if (screenWidth < 500) { |
||||
this.scrollTrigger = {x: 800}; |
||||
} else { |
||||
this.scrollTrigger = {}; |
||||
} |
||||
}, |
||||
showModal() { |
||||
this.visible = true; |
||||
this.loadData(); |
||||
this.form.resetFields(); |
||||
}, |
||||
getQueryParams() { |
||||
let param = Object.assign({}, this.queryParam, this.isorter); |
||||
param.page = this.ipagination.current; |
||||
param.rows = this.ipagination.pageSize; |
||||
return filterObj(param); |
||||
}, |
||||
getQueryField() { |
||||
let str = 'id,'; |
||||
for (let a = 0; a < this.columns.length; a++) { |
||||
str += ',' + this.columns[a].dataIndex; |
||||
} |
||||
return str; |
||||
}, |
||||
searchReset(num) { |
||||
let that = this; |
||||
if (num !== 0) { |
||||
that.queryParam = {}; |
||||
that.loadData(1); |
||||
} |
||||
that.selectedRowKeys = []; |
||||
that.selectMaterialIds = []; |
||||
}, |
||||
close() { |
||||
this.searchReset(0); |
||||
this.visible = false; |
||||
}, |
||||
handleTableChange(pagination, filters, sorter) { |
||||
if (Object.keys(sorter).length > 0) { |
||||
this.isorter.column = sorter.field; |
||||
this.isorter.order = 'ascend' === sorter.order ? 'asc' : 'desc'; |
||||
} |
||||
this.ipagination = pagination; |
||||
this.loadData(); |
||||
}, |
||||
handleSubmit() { |
||||
let that = this; |
||||
this.getSelectMaterialRows(); |
||||
that.$emit('ok', that.selectMaterialRows, that.selectMaterialIds); |
||||
that.searchReset(0) |
||||
that.close(); |
||||
}, |
||||
//获取选择信息 |
||||
getSelectMaterialRows(rowId) { |
||||
let dataSource = this.dataSource; |
||||
let materialIds = ""; |
||||
this.selectMaterialRows = []; |
||||
for (let i = 0, len = dataSource.length; i < len; i++) { |
||||
if (this.selectedRowKeys.includes(dataSource[i].id)) { |
||||
this.selectMaterialRows.push(dataSource[i]); |
||||
materialIds = materialIds + "," + dataSource[i].mBarCode |
||||
} |
||||
} |
||||
this.selectMaterialIds = materialIds.substring(1); |
||||
}, |
||||
onSelectChange(selectedRowKeys, selectionRows) { |
||||
this.selectedRowKeys = selectedRowKeys; |
||||
this.selectionRows = selectionRows; |
||||
}, |
||||
onSearch() { |
||||
this.loadData(1); |
||||
}, |
||||
modalFormOk() { |
||||
this.loadData(); |
||||
}, |
||||
rowAction(record, index) { |
||||
return { |
||||
on: { |
||||
click: () => { |
||||
let arr = [] |
||||
arr.push(record.id) |
||||
this.selectedRowKeys = arr |
||||
}, |
||||
dblclick: () => { |
||||
let arr = [] |
||||
arr.push(record.id) |
||||
this.selectedRowKeys = arr |
||||
this.handleSubmit() |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
.ant-table-tbody .ant-table-row td { |
||||
padding-top: 10px; |
||||
padding-bottom: 10px; |
||||
} |
||||
|
||||
#components-layout-demo-custom-trigger .trigger { |
||||
font-size: 18px; |
||||
line-height: 64px; |
||||
padding: 0 24px; |
||||
cursor: pointer; |
||||
transition: color .3s; |
||||
} |
||||
</style> |
@ -0,0 +1,265 @@
|
||||
<template> |
||||
<a-modal |
||||
:width="modalWidth" |
||||
:visible="visible" |
||||
:title="title" |
||||
@ok="handleSubmit" |
||||
@cancel="close" |
||||
cancelText="关闭" |
||||
style="top:5%;height: 100%;overflow-y: hidden" |
||||
wrapClassName="ant-modal-cust-warp" |
||||
> |
||||
<a-row :gutter="10" style="padding: 10px; margin: -10px"> |
||||
<a-col :md="24" :sm="24"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="onSearch"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="6" :sm="8"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="序列号"> |
||||
<a-input placeholder="请输入序列号" v-model="queryParam.name"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-button type="primary" @click="onSearch">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset(1)">重置</a-button> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
<a-table |
||||
ref="table" |
||||
:scroll="scrollTrigger" |
||||
size="middle" |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange,type: getType}" |
||||
:loading="loading" |
||||
:customRow="rowAction" |
||||
@change="handleTableChange"> |
||||
</a-table> |
||||
</div> |
||||
</a-col> |
||||
</a-row> |
||||
</a-modal> |
||||
</template> |
||||
|
||||
<script> |
||||
import { getAction } from '@/api/manage' |
||||
import {getEnableSerialNumberList} from '@/api/api' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
|
||||
export default { |
||||
name: 'JSelectSnModal', |
||||
mixins:[JeecgListMixin], |
||||
components: {}, |
||||
props: ['modalWidth', 'rows', 'multi', 'barCode'], |
||||
data() { |
||||
return { |
||||
queryParam: { |
||||
name: "", |
||||
depotId: '', |
||||
barCode: '' |
||||
}, |
||||
labelCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 5 }, |
||||
}, |
||||
wrapperCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 16 }, |
||||
}, |
||||
categoryTree:[], |
||||
columns: [ |
||||
{dataIndex: 'serialNumber', title: '序列号', width: 100, align: 'left'} |
||||
], |
||||
scrollTrigger: {}, |
||||
dataSource: [], |
||||
selectedRowKeys: [], |
||||
selectRows: [], |
||||
selectIds: [], |
||||
title: '选择序列号', |
||||
ipagination: { |
||||
current: 1, |
||||
pageSize: 10, |
||||
pageSizeOptions: ['10', '20', '30', '100', '200'], |
||||
showTotal: (total, range) => { |
||||
return range[0] + '-' + range[1] + ' 共' + total + '条' |
||||
}, |
||||
showQuickJumper: true, |
||||
showSizeChanger: true, |
||||
total: 0 |
||||
}, |
||||
isorter: { |
||||
column: 'createTime', |
||||
order: 'desc' |
||||
}, |
||||
departTree: [], |
||||
depotList: [], |
||||
visible: false, |
||||
form: this.$form.createForm(this), |
||||
loading: false, |
||||
expandedKeys: [], |
||||
} |
||||
}, |
||||
computed: { |
||||
// 计算属性的 getter |
||||
getType: function () { |
||||
return this.multi == true ? 'checkbox' : 'radio'; |
||||
} |
||||
}, |
||||
watch: { |
||||
barCode: { |
||||
immediate: true, |
||||
handler() { |
||||
this.initBarCode() |
||||
} |
||||
}, |
||||
}, |
||||
created() { |
||||
this.loadData() |
||||
}, |
||||
methods: { |
||||
initBarCode() { |
||||
if (this.barCode) { |
||||
this.$emit('initComp', this.barCode) |
||||
} else { |
||||
// JSelectUserByDep组件bug issues/I16634 |
||||
this.$emit('initComp', '') |
||||
} |
||||
}, |
||||
async loadData(arg) { |
||||
if(this.rows) { |
||||
if(JSON.parse(this.rows).depotId && JSON.parse(this.rows).barCode ){ |
||||
this.queryParam.depotId = JSON.parse(this.rows).depotId-0 |
||||
this.queryParam.barCode = JSON.parse(this.rows).barCode |
||||
} |
||||
} |
||||
if (arg === 1) { |
||||
this.ipagination.current = 1; |
||||
} |
||||
this.loading = true |
||||
let params = this.getQueryParams()//查询条件 |
||||
await getEnableSerialNumberList(params).then((res) => { |
||||
if (res && res.code === 200) { |
||||
this.dataSource = res.data.rows |
||||
this.ipagination.total = res.data.total |
||||
} |
||||
}).finally(() => { |
||||
this.loading = false |
||||
}) |
||||
}, |
||||
showModal() { |
||||
this.visible = true; |
||||
this.loadData(); |
||||
this.form.resetFields(); |
||||
}, |
||||
getQueryParams() { |
||||
let param = Object.assign({}, this.queryParam, this.isorter); |
||||
param.page = this.ipagination.current; |
||||
param.rows = this.ipagination.pageSize; |
||||
return param; |
||||
}, |
||||
getQueryField() { |
||||
let str = 'id,'; |
||||
for (let a = 0; a < this.columns.length; a++) { |
||||
str += ',' + this.columns[a].dataIndex; |
||||
} |
||||
return str; |
||||
}, |
||||
searchReset(num) { |
||||
let that = this; |
||||
if (num !== 0) { |
||||
if(this.rows) { |
||||
this.queryParam.name='' |
||||
if(JSON.parse(this.rows).depotId && JSON.parse(this.rows).barCode ){ |
||||
this.queryParam.depotId = JSON.parse(this.rows).depotId-0 |
||||
this.queryParam.barCode = JSON.parse(this.rows).barCode |
||||
} |
||||
} |
||||
that.loadData(1); |
||||
} |
||||
that.selectedRowKeys = []; |
||||
that.selectIds = []; |
||||
}, |
||||
close() { |
||||
this.searchReset(0); |
||||
this.visible = false; |
||||
}, |
||||
handleTableChange(pagination, filters, sorter) { |
||||
if (Object.keys(sorter).length > 0) { |
||||
this.isorter.column = sorter.field; |
||||
this.isorter.order = 'ascend' === sorter.order ? 'asc' : 'desc'; |
||||
} |
||||
this.ipagination = pagination; |
||||
this.loadData(); |
||||
}, |
||||
handleSubmit() { |
||||
let that = this; |
||||
this.getSelectRows(); |
||||
that.$emit('ok', that.selectRows, that.selectIds); |
||||
that.searchReset(0) |
||||
that.close(); |
||||
}, |
||||
//获取选择信息 |
||||
getSelectRows(rowId) { |
||||
let dataSource = this.dataSource; |
||||
let ids = ""; |
||||
this.selectRows = []; |
||||
for (let i = 0, len = dataSource.length; i < len; i++) { |
||||
if (this.selectedRowKeys.includes(dataSource[i].id)) { |
||||
this.selectRows.push(dataSource[i]); |
||||
ids = ids + "," + dataSource[i].serialNumber |
||||
} |
||||
} |
||||
this.selectIds = ids.substring(1); |
||||
}, |
||||
onSelectChange(selectedRowKeys, selectionRows) { |
||||
this.selectedRowKeys = selectedRowKeys; |
||||
this.selectionRows = selectionRows; |
||||
}, |
||||
onSearch() { |
||||
this.loadData(1); |
||||
}, |
||||
modalFormOk() { |
||||
this.loadData(); |
||||
}, |
||||
rowAction(record, index) { |
||||
return { |
||||
on: { |
||||
click: () => { |
||||
let arr = [] |
||||
arr.push(record.id) |
||||
this.selectedRowKeys = arr |
||||
}, |
||||
dblclick: () => { |
||||
let arr = [] |
||||
arr.push(record.id) |
||||
this.selectedRowKeys = arr |
||||
this.handleSubmit() |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
.ant-table-tbody .ant-table-row td { |
||||
padding-top: 10px; |
||||
padding-bottom: 10px; |
||||
} |
||||
|
||||
#components-layout-demo-custom-trigger .trigger { |
||||
font-size: 18px; |
||||
line-height: 64px; |
||||
padding: 0 24px; |
||||
cursor: pointer; |
||||
transition: color .3s; |
||||
} |
||||
</style> |
@ -0,0 +1,329 @@
|
||||
<template> |
||||
<a-modal |
||||
:width="modalWidth" |
||||
:visible="visible" |
||||
:title="title" |
||||
@ok="handleSubmit" |
||||
@cancel="close" |
||||
cancelText="关闭" |
||||
style="margin-top: -70px" |
||||
wrapClassName="ant-modal-cust-warp" |
||||
> |
||||
<a-row :gutter="10" style="background-color: #ececec; padding: 10px; margin: -10px"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-card :bordered="false"> |
||||
<!--组织机构--> |
||||
<a-directory-tree |
||||
selectable |
||||
:selectedKeys="selectedDepIds" |
||||
:checkStrictly="true" |
||||
:dropdownStyle="{maxHeight:'200px',overflow:'auto'}" |
||||
:treeData="departTree" |
||||
:expandAction="false" |
||||
:expandedKeys.sync="expandedKeys" |
||||
@select="onDepSelect" |
||||
/> |
||||
</a-card> |
||||
</a-col> |
||||
<a-col :md="18" :sm="24"> |
||||
<a-card :bordered="false"> |
||||
用户账号: |
||||
<a-input-search |
||||
:style="{width:'150px',marginBottom:'15px'}" |
||||
placeholder="请输入账号" |
||||
v-model="queryParam.username" |
||||
@search="onSearch" |
||||
></a-input-search> |
||||
<a-button @click="searchReset(1)" style="margin-left: 20px" icon="redo">重置</a-button> |
||||
<!--用户列表--> |
||||
<a-table |
||||
ref="table" |
||||
:scroll="scrollTrigger" |
||||
size="middle" |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange,type: getType}" |
||||
:loading="loading" |
||||
@change="handleTableChange"> |
||||
</a-table> |
||||
</a-card> |
||||
</a-col> |
||||
</a-row> |
||||
</a-modal> |
||||
</template> |
||||
|
||||
<script> |
||||
import {filterObj} from '@/utils/util' |
||||
import {queryDepartTreeList, getUserList, queryUserByDepId} from '@/api/api' |
||||
|
||||
export default { |
||||
name: 'JSelectUserByDepModal', |
||||
components: {}, |
||||
props: ['modalWidth', 'multi', 'userIds'], |
||||
data() { |
||||
return { |
||||
queryParam: { |
||||
username: "", |
||||
}, |
||||
columns: [ |
||||
{ |
||||
title: '用户账号', |
||||
align: 'center', |
||||
dataIndex: 'username' |
||||
}, |
||||
{ |
||||
title: '用户姓名', |
||||
align: 'center', |
||||
dataIndex: 'realname' |
||||
}, |
||||
{ |
||||
title: '性别', |
||||
align: 'center', |
||||
dataIndex: 'sex', |
||||
customRender: function (text) { |
||||
if (text === 1) { |
||||
return '男' |
||||
} else if (text === 2) { |
||||
return '女' |
||||
} else { |
||||
return text |
||||
} |
||||
} |
||||
}, |
||||
{ |
||||
title: '手机', |
||||
align: 'center', |
||||
dataIndex: 'phone' |
||||
}, |
||||
{ |
||||
title: '部门', |
||||
align: 'center', |
||||
dataIndex: 'orgCode' |
||||
} |
||||
], |
||||
scrollTrigger: {}, |
||||
dataSource: [], |
||||
selectedRowKeys: [], |
||||
selectUserRows: [], |
||||
selectUserIds: [], |
||||
title: '根据部门选择用户', |
||||
ipagination: { |
||||
current: 1, |
||||
pageSize: 10, |
||||
pageSizeOptions: ['10', '20', '30'], |
||||
showTotal: (total, range) => { |
||||
return range[0] + '-' + range[1] + ' 共' + total + '条' |
||||
}, |
||||
showQuickJumper: true, |
||||
showSizeChanger: true, |
||||
total: 0 |
||||
}, |
||||
isorter: { |
||||
column: 'createTime', |
||||
order: 'desc' |
||||
}, |
||||
selectedDepIds: [], |
||||
departTree: [], |
||||
visible: false, |
||||
form: this.$form.createForm(this), |
||||
loading: false, |
||||
expandedKeys: [], |
||||
} |
||||
}, |
||||
computed: { |
||||
// 计算属性的 getter |
||||
getType: function () { |
||||
return this.multi == true ? 'checkbox' : 'radio'; |
||||
} |
||||
}, |
||||
watch: { |
||||
userIds: { |
||||
immediate: true, |
||||
handler() { |
||||
this.initUserNames() |
||||
} |
||||
}, |
||||
}, |
||||
created() { |
||||
// 该方法触发屏幕自适应 |
||||
this.resetScreenSize(); |
||||
this.loadData() |
||||
}, |
||||
methods: { |
||||
initUserNames() { |
||||
if (this.userIds) { |
||||
// 这里最后加一个 , 的原因是因为无论如何都要使用 in 查询,防止后台进行了模糊匹配,导致查询结果不准确 |
||||
let values = this.userIds.split(',') + ',' |
||||
getUserList({ |
||||
username: values, |
||||
pageNo: 1, |
||||
pageSize: values.length |
||||
}).then((res) => { |
||||
if (res.success) { |
||||
let selectedRowKeys = [] |
||||
let realNames = [] |
||||
res.result.records.forEach(user => { |
||||
realNames.push(user['realname']) |
||||
selectedRowKeys.push(user['id']) |
||||
}) |
||||
this.selectedRowKeys = selectedRowKeys |
||||
this.$emit('initComp', realNames.join(',')) |
||||
} |
||||
}) |
||||
} else { |
||||
// JSelectUserByDep组件bug issues/I16634 |
||||
this.$emit('initComp', '') |
||||
} |
||||
}, |
||||
async loadData(arg) { |
||||
if (arg === 1) { |
||||
this.ipagination.current = 1; |
||||
} |
||||
if (this.selectedDepIds && this.selectedDepIds.length > 0) { |
||||
await this.initQueryUserByDepId(this.selectedDepIds) |
||||
} else { |
||||
this.loading = true |
||||
let params = this.getQueryParams()//查询条件 |
||||
await getUserList(params).then((res) => { |
||||
if (res.success) { |
||||
this.dataSource = res.result.records |
||||
this.ipagination.total = res.result.total |
||||
} |
||||
}).finally(() => { |
||||
this.loading = false |
||||
}) |
||||
} |
||||
}, |
||||
// 触发屏幕自适应 |
||||
resetScreenSize() { |
||||
let screenWidth = document.body.clientWidth; |
||||
if (screenWidth < 500) { |
||||
this.scrollTrigger = {x: 800}; |
||||
} else { |
||||
this.scrollTrigger = {}; |
||||
} |
||||
}, |
||||
showModal() { |
||||
this.visible = true; |
||||
this.queryDepartTree(); |
||||
this.initUserNames() |
||||
this.loadData(); |
||||
this.form.resetFields(); |
||||
}, |
||||
getQueryParams() { |
||||
let param = Object.assign({}, this.queryParam, this.isorter); |
||||
param.field = this.getQueryField(); |
||||
param.pageNo = this.ipagination.current; |
||||
param.pageSize = this.ipagination.pageSize; |
||||
return filterObj(param); |
||||
}, |
||||
getQueryField() { |
||||
let str = 'id,'; |
||||
for (let a = 0; a < this.columns.length; a++) { |
||||
str += ',' + this.columns[a].dataIndex; |
||||
} |
||||
return str; |
||||
}, |
||||
searchReset(num) { |
||||
let that = this; |
||||
if (num !== 0) { |
||||
that.queryParam = {}; |
||||
that.loadData(1); |
||||
} |
||||
that.selectedRowKeys = []; |
||||
that.selectUserIds = []; |
||||
that.selectedDepIds = []; |
||||
}, |
||||
close() { |
||||
this.searchReset(0); |
||||
this.visible = false; |
||||
}, |
||||
handleTableChange(pagination, filters, sorter) { |
||||
//TODO 筛选 |
||||
if (Object.keys(sorter).length > 0) { |
||||
this.isorter.column = sorter.field; |
||||
this.isorter.order = 'ascend' === sorter.order ? 'asc' : 'desc'; |
||||
} |
||||
this.ipagination = pagination; |
||||
this.loadData(); |
||||
}, |
||||
handleSubmit() { |
||||
let that = this; |
||||
this.getSelectUserRows(); |
||||
that.$emit('ok', that.selectUserRows, that.selectUserIds); |
||||
that.searchReset(0) |
||||
that.close(); |
||||
}, |
||||
//获取选择用户信息 |
||||
getSelectUserRows(rowId) { |
||||
let dataSource = this.dataSource; |
||||
let userIds = ""; |
||||
this.selectUserRows = []; |
||||
for (let i = 0, len = dataSource.length; i < len; i++) { |
||||
if (this.selectedRowKeys.includes(dataSource[i].id)) { |
||||
this.selectUserRows.push(dataSource[i]); |
||||
userIds = userIds + "," + dataSource[i].username |
||||
} |
||||
} |
||||
this.selectUserIds = userIds.substring(1); |
||||
}, |
||||
// 点击树节点,筛选出对应的用户 |
||||
onDepSelect(selectedDepIds) { |
||||
if (selectedDepIds[0] != null) { |
||||
this.initQueryUserByDepId(selectedDepIds); // 调用方法根据选选择的id查询用户信息 |
||||
if (this.selectedDepIds[0] !== selectedDepIds[0]) { |
||||
this.selectedDepIds = [selectedDepIds[0]]; |
||||
} |
||||
} |
||||
}, |
||||
onSelectChange(selectedRowKeys, selectionRows) { |
||||
this.selectedRowKeys = selectedRowKeys; |
||||
this.selectionRows = selectionRows; |
||||
}, |
||||
onSearch() { |
||||
this.loadData(1); |
||||
}, |
||||
// 根据选择的id来查询用户信息 |
||||
initQueryUserByDepId(selectedDepIds) { |
||||
this.loading = true |
||||
return queryUserByDepId({id: selectedDepIds.toString()}).then((res) => { |
||||
if (res.success) { |
||||
this.dataSource = res.result; |
||||
this.ipagination.total = res.result.length; |
||||
} |
||||
}).finally(() => { |
||||
this.loading = false |
||||
}) |
||||
}, |
||||
queryDepartTree() { |
||||
queryDepartTreeList().then((res) => { |
||||
if (res.success) { |
||||
this.departTree = res.result; |
||||
// 默认展开父节点 |
||||
this.expandedKeys = this.departTree.map(item => item.id) |
||||
} |
||||
}) |
||||
}, |
||||
modalFormOk() { |
||||
this.loadData(); |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
.ant-table-tbody .ant-table-row td { |
||||
padding-top: 10px; |
||||
padding-bottom: 10px; |
||||
} |
||||
|
||||
#components-layout-demo-custom-trigger .trigger { |
||||
font-size: 18px; |
||||
line-height: 64px; |
||||
padding: 0 24px; |
||||
cursor: pointer; |
||||
transition: color .3s; |
||||
} |
||||
</style> |
@ -0,0 +1,122 @@
|
||||
<template> |
||||
<a-modal |
||||
title="用户列表" |
||||
:width="1000" |
||||
:visible="visible" |
||||
:confirmLoading="confirmLoading" |
||||
@ok="handleSubmit" |
||||
@cancel="handleCancel"> |
||||
|
||||
<a-table |
||||
ref="table" |
||||
bordered |
||||
size="middle" |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:loading="loading" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange}"></a-table> |
||||
</a-modal> |
||||
</template> |
||||
|
||||
<script> |
||||
import {getUserList} from '@/api/api' |
||||
import {JeecgListMixin} from '@/mixins/JeecgListMixin' |
||||
|
||||
export default { |
||||
name: "SelectUserListModal", |
||||
mixins: [JeecgListMixin], |
||||
data() { |
||||
return { |
||||
title: "操作", |
||||
visible: false, |
||||
model: {}, |
||||
confirmLoading: false, |
||||
url: { |
||||
add: "/act/model/create", |
||||
list: "/sys/user/list" |
||||
}, |
||||
columns: [ |
||||
{ |
||||
title: '用户账号', |
||||
align: "center", |
||||
dataIndex: 'username', |
||||
fixed: 'left', |
||||
width: 200 |
||||
}, |
||||
{ |
||||
title: '用户姓名', |
||||
align: "center", |
||||
dataIndex: 'realname', |
||||
}, |
||||
{ |
||||
title: '性别', |
||||
align: "center", |
||||
dataIndex: 'sex_dictText' |
||||
}, |
||||
{ |
||||
title: '手机号码', |
||||
align: "center", |
||||
dataIndex: 'phone' |
||||
}, |
||||
{ |
||||
title: '邮箱', |
||||
align: "center", |
||||
dataIndex: 'email' |
||||
}, |
||||
{ |
||||
title: '状态', |
||||
align: "center", |
||||
dataIndex: 'status_dictText' |
||||
} |
||||
] |
||||
} |
||||
}, |
||||
created() { |
||||
//Step.2 加载用户数据 |
||||
getUserList().then((res) => { |
||||
if (res.success) { |
||||
this.dataSource = res.result.records; |
||||
this.ipagination.total = res.result.total; |
||||
} |
||||
}) |
||||
}, |
||||
methods: { |
||||
open() { |
||||
this.visible = true; |
||||
|
||||
//Step.1 清空选中用户 |
||||
this.selectedRowKeys = [] |
||||
this.selectedRows = [] |
||||
}, |
||||
close() { |
||||
this.$emit('close'); |
||||
this.visible = false; |
||||
}, |
||||
handleChange(info) { |
||||
let file = info.file; |
||||
if (file.response.success) { |
||||
this.$message.success(file.response.message); |
||||
this.$emit('ok'); |
||||
this.close() |
||||
} else { |
||||
this.$message.warn(file.response.message); |
||||
this.close() |
||||
} |
||||
|
||||
}, |
||||
handleCancel() { |
||||
this.close() |
||||
}, |
||||
handleSubmit() { |
||||
this.$emit('ok', this.selectionRows); |
||||
this.close() |
||||
}, |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style> |
||||
|
||||
</style> |
@ -0,0 +1,36 @@
|
||||
<template> |
||||
<component |
||||
:is="comp" |
||||
:formData="formData" |
||||
ref="compModel" |
||||
v-if="comp"> |
||||
</component> |
||||
</template> |
||||
<script> |
||||
export default { |
||||
name: 'DynamicNotice', |
||||
data () { |
||||
return { |
||||
compName: this.path |
||||
} |
||||
}, |
||||
computed: { |
||||
comp: function () { |
||||
if(!this.path){ |
||||
return null; |
||||
} |
||||
return () => import(`@/views/${this.path}.vue`) |
||||
} |
||||
}, |
||||
props: ['path','formData'], |
||||
methods: { |
||||
detail () { |
||||
setTimeout(() => { |
||||
if(this.path){ |
||||
this.$refs.compModel.view(this.formData); |
||||
} |
||||
}, 200) |
||||
}, |
||||
} |
||||
} |
||||
</script> |
@ -0,0 +1,139 @@
|
||||
<template> |
||||
<j-modal |
||||
:title="title" |
||||
:width="modelStyle.width" |
||||
:visible="visible" |
||||
:bodyStyle ="bodyStyle" |
||||
:switchFullscreen="switchFullscreen" |
||||
@cancel="handleCancel" |
||||
> |
||||
<template slot="footer"> |
||||
<a-button key="back" @click="handleCancel">关闭</a-button> |
||||
<a-button v-if="record.openType==='url'" type="primary" @click="toHandle">去处理</a-button> |
||||
</template> |
||||
<a-card class="daily-article" :loading="loading"> |
||||
<a-card-meta |
||||
:title="record.titile" |
||||
:description="'标题:'+record.msgTitle"> |
||||
</a-card-meta> |
||||
<a-divider /> |
||||
<span v-html="record.msgContent" class="article-content"></span> |
||||
</a-card> |
||||
</j-modal> |
||||
</template> |
||||
|
||||
<script> |
||||
export default { |
||||
name: "SysAnnouncementModal", |
||||
components: { |
||||
}, |
||||
data () { |
||||
return { |
||||
title:"通知消息", |
||||
record: {}, |
||||
labelCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 5 }, |
||||
}, |
||||
wrapperCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 16 }, |
||||
}, |
||||
visible: false, |
||||
switchFullscreen: true, |
||||
loading: false, |
||||
bodyStyle:{ |
||||
padding: "0", |
||||
height:(window.innerHeight*0.6)+"px", |
||||
"overflow-y":"auto", |
||||
|
||||
}, |
||||
modelStyle:{ |
||||
width: '60%', |
||||
style: { top: '20px' }, |
||||
fullScreen: false |
||||
} |
||||
} |
||||
}, |
||||
created () { |
||||
}, |
||||
methods: { |
||||
detail (record) { |
||||
this.visible = true; |
||||
this.record = record; |
||||
}, |
||||
handleCancel () { |
||||
this.visible = false; |
||||
}, |
||||
/** 切换全屏显示 */ |
||||
handleClickToggleFullScreen() { |
||||
let mode = !this.modelStyle.fullScreen |
||||
if (mode) { |
||||
this.modelStyle.width = '100%' |
||||
this.modelStyle.style.top = '20px' |
||||
} else { |
||||
this.modelStyle.width = '60%' |
||||
this.modelStyle.style.top = '50px' |
||||
} |
||||
this.modelStyle.fullScreen = mode |
||||
}, |
||||
toHandle(){ |
||||
if(this.record.openType==='url'){ |
||||
this.visible = false; |
||||
//链接跳转 |
||||
this.$router.push({path: this.record.openPage}) |
||||
} |
||||
}, |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="less"> |
||||
.announcementCustomModal{ |
||||
.ant-modal-header { |
||||
border: none; |
||||
display: inline-block; |
||||
position: absolute; |
||||
z-index: 1; |
||||
right: 56px; |
||||
padding: 0; |
||||
.ant-modal-title{ |
||||
.custom-btn{ |
||||
width: 56px; |
||||
height: 56px; |
||||
border: none; |
||||
box-shadow: none; |
||||
} |
||||
} |
||||
} |
||||
.daily-article{ |
||||
border-bottom: 0; |
||||
} |
||||
} |
||||
</style> |
||||
<style scoped lang="less"> |
||||
.daily-article { |
||||
.article-button { |
||||
font-size: 1.2rem !important; |
||||
} |
||||
.ant-card-body { |
||||
padding: 18px !important; |
||||
} |
||||
.ant-card-head { |
||||
padding: 0 1rem; |
||||
} |
||||
.ant-card-meta { |
||||
margin-bottom: 1rem; |
||||
} |
||||
.article-content { |
||||
p { |
||||
word-wrap: break-word; |
||||
word-break: break-all; |
||||
text-overflow: initial; |
||||
white-space: normal; |
||||
font-size: .9rem !important; |
||||
margin-bottom: .8rem; |
||||
} |
||||
} |
||||
} |
||||
</style> |
@ -0,0 +1,173 @@
|
||||
import JEditableTable from '@/components/jeecg/JEditableTable' |
||||
import { VALIDATE_NO_PASSED, getRefPromise, validateFormAndTables } from '@/utils/JEditableTableUtil' |
||||
import { httpAction, getAction } from '@/api/manage' |
||||
|
||||
export const JEditableTableMixin = { |
||||
components: { |
||||
JEditableTable |
||||
}, |
||||
data() { |
||||
return { |
||||
title: '操作', |
||||
visible: false, |
||||
form: this.$form.createForm(this), |
||||
confirmLoading: false, |
||||
model: {}, |
||||
labelCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 6 } |
||||
}, |
||||
wrapperCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 18 } |
||||
} |
||||
} |
||||
}, |
||||
methods: { |
||||
|
||||
/** 获取所有的editableTable实例 */ |
||||
getAllTable() { |
||||
if (!(this.refKeys instanceof Array)) { |
||||
throw this.throwNotArray('refKeys') |
||||
} |
||||
let values = this.refKeys.map(key => getRefPromise(this, key)) |
||||
return Promise.all(values) |
||||
}, |
||||
|
||||
/** 遍历所有的JEditableTable实例 */ |
||||
eachAllTable(callback) { |
||||
// 开始遍历
|
||||
this.getAllTable().then(tables => { |
||||
tables.forEach((item, index) => { |
||||
if (typeof callback === 'function') { |
||||
callback(item, index) |
||||
} |
||||
}) |
||||
}) |
||||
}, |
||||
|
||||
/** 当点击新增按钮时调用此方法 */ |
||||
add() { |
||||
// addBefore 没有地方用到
|
||||
if (typeof this.addBefore === 'function') this.addBefore()
|
||||
// 默认新增空数据
|
||||
let rowNum = this.addDefaultRowNum |
||||
if (typeof rowNum !== 'number') { |
||||
rowNum = 1 |
||||
console.warn('由于你没有在 data 中定义 addDefaultRowNum 或 addDefaultRowNum 不是数字,所以默认添加一条空数据,如果不想默认添加空数据,请将定义 addDefaultRowNum 为 0') |
||||
} |
||||
this.eachAllTable((item) => { |
||||
item.add(rowNum) |
||||
}) |
||||
if (typeof this.addAfter === 'function') this.addAfter(this.model) |
||||
this.edit({}) |
||||
}, |
||||
/** 当点击了编辑(修改)按钮时调用此方法 */ |
||||
edit(record) { |
||||
if (typeof this.editBefore === 'function') this.editBefore(record) |
||||
this.visible = true |
||||
this.activeKey = this.refKeys[0] |
||||
this.form.resetFields() |
||||
this.model = Object.assign({}, record) |
||||
if (typeof this.editAfter === 'function') this.editAfter(this.model) |
||||
}, |
||||
/** 关闭弹窗,并将所有JEditableTable实例回归到初始状态 */ |
||||
close() { |
||||
this.visible = false |
||||
this.eachAllTable((item) => { |
||||
item.initialize() |
||||
}) |
||||
this.$emit('close') |
||||
}, |
||||
|
||||
/** 查询某个tab的数据 */ |
||||
requestSubTableData(url, params, tab, success) { |
||||
tab.loading = true |
||||
getAction(url, params).then(res => { |
||||
if(res && res.code === 200){ |
||||
tab.dataSource = res.data.rows |
||||
typeof success === 'function' ? success(res) : '' |
||||
} |
||||
}).finally(() => { |
||||
tab.loading = false |
||||
}) |
||||
}, |
||||
/** 发起请求,自动判断是执行新增还是修改操作 */ |
||||
request(formData) { |
||||
let url = this.url.add, method = 'post' |
||||
if (this.model.id) { |
||||
url = this.url.edit |
||||
method = 'put' |
||||
} |
||||
this.confirmLoading = true |
||||
httpAction(url, formData, method).then((res) => { |
||||
if(res.code === 200){ |
||||
this.$emit('ok') |
||||
this.confirmLoading = false |
||||
this.close() |
||||
} else { |
||||
this.$message.warning(res.msg); |
||||
this.confirmLoading = false |
||||
} |
||||
}).finally(() => { |
||||
}) |
||||
}, |
||||
|
||||
/* --- handle 事件 --- */ |
||||
|
||||
/** ATab 选项卡切换事件 */ |
||||
handleChangeTabs(key) { |
||||
// 自动重置scrollTop状态,防止出现白屏
|
||||
getRefPromise(this, key).then(editableTable => { |
||||
editableTable.resetScrollTop() |
||||
}) |
||||
}, |
||||
/** 关闭按钮点击事件 */ |
||||
handleCancel() { |
||||
this.close() |
||||
}, |
||||
/** 确定按钮点击事件 */ |
||||
handleOk() { |
||||
/** 触发表单验证 */ |
||||
this.getAllTable().then(tables => { |
||||
let inputValues = tables[0].inputValues
|
||||
let ids = []
|
||||
inputValues.forEach((item) => { |
||||
if(!item.barCode && !item.operNumber) { |
||||
ids.push(item.id) |
||||
} |
||||
}) |
||||
tables[0].removeRows(ids) |
||||
/** 一次性验证主表和所有的次表 */ |
||||
return validateFormAndTables(this.form, tables) |
||||
}).then(allValues => { |
||||
if (typeof this.classifyIntoFormData !== 'function') { |
||||
throw this.throwNotFunction('classifyIntoFormData') |
||||
} |
||||
let formData = this.classifyIntoFormData(allValues) |
||||
// 发起请求
|
||||
return this.request(formData) |
||||
}).catch(e => { |
||||
if (e.error === VALIDATE_NO_PASSED) { |
||||
// 如果有未通过表单验证的子表,就自动跳转到它所在的tab
|
||||
this.activeKey = e.index == null ? this.activeKey : this.refKeys[e.index] |
||||
} else { |
||||
console.error(e) |
||||
} |
||||
}) |
||||
}, |
||||
|
||||
/* --- throw --- */ |
||||
|
||||
/** not a function */ |
||||
throwNotFunction(name) { |
||||
return `${name} 未定义或不是一个函数` |
||||
}, |
||||
|
||||
/** not a array */ |
||||
throwNotArray(name) { |
||||
return `${name} 未定义或不是一个数组` |
||||
} |
||||
|
||||
} |
||||
} |
@ -0,0 +1,438 @@
|
||||
/** |
||||
* 新增修改完成调用 modalFormOk方法 编辑弹框组件ref定义为modalForm |
||||
* 高级查询按钮调用 superQuery方法 高级查询组件ref定义为superQueryModal |
||||
* data中url定义 list为查询列表 delete为删除单条记录 deleteBatch为批量删除 |
||||
*/ |
||||
import { filterObj } from '@/utils/util'; |
||||
import { deleteAction, getAction, postAction, downFile, getFileAccessHttpUrl } from '@/api/manage' |
||||
import Vue from 'vue' |
||||
import {mixinDevice} from '@/utils/mixin.js' |
||||
|
||||
export const JeecgListMixin = { |
||||
mixins: [mixinDevice], |
||||
data(){ |
||||
return { |
||||
//token header
|
||||
tokenHeader: {'X-Access-Token': ""}, |
||||
/*卡片样式 */ |
||||
cardStyle: '', |
||||
/* 查询条件-请不要在queryParam中声明非字符串值的属性 */ |
||||
queryParam: {}, |
||||
/* 数据源 */ |
||||
dataSource:[], |
||||
/* 分页参数 */ |
||||
ipagination:{ |
||||
current: 1, |
||||
pageSize: 10, |
||||
pageSizeOptions: ['10', '20', '30', '50', '100'], |
||||
showTotal: (total, range) => { |
||||
return range[0] + "-" + range[1] + " 共" + total + "条" |
||||
}, |
||||
showQuickJumper: true, |
||||
showSizeChanger: true, |
||||
total: 0 |
||||
}, |
||||
/* 控制table高度 */ |
||||
scroll: { |
||||
x:1300 |
||||
}, |
||||
/* 排序参数 */ |
||||
isorter:{ |
||||
column: 'createTime', |
||||
order: 'desc', |
||||
}, |
||||
/* 筛选参数 */ |
||||
filters: {}, |
||||
/* table加载状态 */ |
||||
loading:false, |
||||
/* table选中keys*/ |
||||
selectedRowKeys: [], |
||||
/* table选中records*/ |
||||
selectionRows: [], |
||||
/* 查询折叠 */ |
||||
toggleSearchStatus:false, |
||||
/* 高级查询条件生效状态 */ |
||||
superQueryFlag:false, |
||||
/* 高级查询条件 */ |
||||
superQueryParams: '', |
||||
/** 高级查询拼接方式 */ |
||||
superQueryMatchType: 'and', |
||||
/** 是否加载时就执行 */ |
||||
disableMixinCreated: false, |
||||
/* 按钮权限 */ |
||||
btnEnableList: '' |
||||
} |
||||
}, |
||||
created() { |
||||
if(this.isDesktop()) { |
||||
this.cardStyle = 'height:' + (document.documentElement.clientHeight-125) + 'px' |
||||
} |
||||
if(!this.disableMixinCreated){ |
||||
//console.log(' -- mixin created -- ')
|
||||
this.loadData(); |
||||
//初始化字典配置 在自己页面定义
|
||||
this.initDictConfig(); |
||||
//初始化按钮权限
|
||||
this.initActiveBtnStr(); |
||||
} |
||||
}, |
||||
mounted () { |
||||
this.initScroll() |
||||
}, |
||||
methods:{ |
||||
loadData(arg) { |
||||
if(!this.url.list){ |
||||
this.$message.error("请设置url.list属性!") |
||||
return |
||||
} |
||||
//加载数据 若传入参数1则加载第一页的内容
|
||||
if (arg === 1) { |
||||
this.ipagination.current = 1; |
||||
} |
||||
var params = this.getQueryParams();//查询条件
|
||||
this.loading = true; |
||||
getAction(this.url.list, params).then((res) => { |
||||
if (res.code===200) { |
||||
this.dataSource = res.data.rows; |
||||
this.ipagination.total = Number(res.data.total); |
||||
this.tableAddTotalRow(this.columns, this.dataSource) |
||||
} |
||||
if(res.code===510){ |
||||
this.$message.warning(res.data) |
||||
} |
||||
this.loading = false; |
||||
}) |
||||
}, |
||||
initDictConfig(){ |
||||
//console.log("--这是一个假的方法!")
|
||||
}, |
||||
handleSuperQuery(params, matchType) { |
||||
//高级查询方法
|
||||
if(!params){ |
||||
this.superQueryParams='' |
||||
this.superQueryFlag = false |
||||
}else{ |
||||
this.superQueryFlag = true |
||||
this.superQueryParams=JSON.stringify(params) |
||||
this.superQueryMatchType = matchType |
||||
} |
||||
this.loadData(1) |
||||
}, |
||||
getQueryParams() { |
||||
//获取查询条件
|
||||
let sqp = {} |
||||
if(this.superQueryParams){ |
||||
sqp['superQueryParams']=encodeURI(this.superQueryParams) |
||||
sqp['superQueryMatchType'] = this.superQueryMatchType |
||||
} |
||||
let searchObj = {} |
||||
searchObj.search = JSON.stringify(this.queryParam); |
||||
var param = Object.assign(sqp, searchObj, this.isorter ,this.filters); |
||||
param.field = this.getQueryField(); |
||||
param.currentPage = this.ipagination.current; |
||||
param.pageSize = this.ipagination.pageSize; |
||||
return filterObj(param); |
||||
}, |
||||
getQueryField() { |
||||
var str = "id,"; |
||||
this.columns.forEach(function (value) { |
||||
str += "," + value.dataIndex; |
||||
}); |
||||
return str; |
||||
}, |
||||
|
||||
onSelectChange(selectedRowKeys, selectionRows) { |
||||
this.selectedRowKeys = selectedRowKeys; |
||||
this.selectionRows = selectionRows; |
||||
}, |
||||
onClearSelected() { |
||||
this.selectedRowKeys = []; |
||||
this.selectionRows = []; |
||||
}, |
||||
searchQuery() { |
||||
this.loadData(1); |
||||
}, |
||||
superQuery() { |
||||
this.$refs.superQueryModal.show(); |
||||
}, |
||||
searchReset() { |
||||
this.queryParam = {} |
||||
this.loadData(1); |
||||
}, |
||||
batchSetStatus: function (status) { |
||||
if(!this.url.batchSetStatusUrl){ |
||||
this.$message.error("请设置url.batchSetStatusUrl属性!") |
||||
return |
||||
} |
||||
if (this.selectedRowKeys.length <= 0) { |
||||
this.$message.warning('请选择一条记录!'); |
||||
return; |
||||
} else { |
||||
var ids = ""; |
||||
for (var a = 0; a < this.selectedRowKeys.length; a++) { |
||||
ids += this.selectedRowKeys[a] + ","; |
||||
} |
||||
var that = this; |
||||
this.$confirm({ |
||||
title: "确认操作", |
||||
content: "是否操作选中数据?", |
||||
onOk: function () { |
||||
that.loading = true; |
||||
postAction(that.url.batchSetStatusUrl, {status: status, ids: ids}).then((res) => { |
||||
if(res.code === 200){ |
||||
that.loadData(); |
||||
that.onClearSelected(); |
||||
} else { |
||||
that.$message.warning(res.msg); |
||||
} |
||||
}).finally(() => { |
||||
that.loading = false; |
||||
}); |
||||
} |
||||
}); |
||||
} |
||||
}, |
||||
batchDel: function () { |
||||
if(!this.url.deleteBatch){ |
||||
this.$message.error("请设置url.deleteBatch属性!") |
||||
return |
||||
} |
||||
if (this.selectedRowKeys.length <= 0) { |
||||
this.$message.warning('请选择一条记录!'); |
||||
return; |
||||
} else { |
||||
var ids = ""; |
||||
for (var a = 0; a < this.selectedRowKeys.length; a++) { |
||||
ids += this.selectedRowKeys[a] + ","; |
||||
} |
||||
var that = this; |
||||
this.$confirm({ |
||||
title: "确认删除", |
||||
content: "是否删除选中数据?", |
||||
onOk: function () { |
||||
that.loading = true; |
||||
deleteAction(that.url.deleteBatch, {ids: ids}).then((res) => { |
||||
if(res.code === 200){ |
||||
that.loadData(); |
||||
that.onClearSelected(); |
||||
} else { |
||||
that.$message.warning(res.msg); |
||||
} |
||||
}).finally(() => { |
||||
that.loading = false; |
||||
}); |
||||
} |
||||
}); |
||||
} |
||||
}, |
||||
handleDelete: function (id) { |
||||
if(!this.url.delete){ |
||||
this.$message.error("请设置url.delete属性!") |
||||
return |
||||
} |
||||
var that = this; |
||||
deleteAction(that.url.delete, {id: id}).then((res) => { |
||||
if(res.code === 200){ |
||||
that.loadData(); |
||||
} else { |
||||
that.$message.warning(res.msg); |
||||
} |
||||
}); |
||||
}, |
||||
handleEdit: function (record) { |
||||
this.$refs.modalForm.edit(record); |
||||
this.$refs.modalForm.title = "编辑"; |
||||
this.$refs.modalForm.disableSubmit = false; |
||||
}, |
||||
handleAdd: function () { |
||||
this.$refs.modalForm.add(); |
||||
this.$refs.modalForm.title = "新增"; |
||||
this.$refs.modalForm.disableSubmit = false; |
||||
}, |
||||
handleTableChange(pagination, filters, sorter) { |
||||
//分页、排序、筛选变化时触发
|
||||
if (Object.keys(sorter).length > 0) { |
||||
this.isorter.column = sorter.field; |
||||
this.isorter.order = "ascend" == sorter.order ? "asc" : "desc" |
||||
} |
||||
if(pagination && pagination.current) { |
||||
this.ipagination = pagination; |
||||
} |
||||
this.loadData(); |
||||
}, |
||||
handleToggleSearch(){ |
||||
this.toggleSearchStatus = !this.toggleSearchStatus; |
||||
}, |
||||
// 给popup查询使用(查询区域不支持回填多个字段,限制只返回一个字段)
|
||||
getPopupField(fields){ |
||||
return fields.split(',')[0] |
||||
}, |
||||
modalFormOk() { |
||||
// 新增/修改 成功时,重载列表
|
||||
this.loadData(); |
||||
}, |
||||
handleDetail:function(record, type){ |
||||
this.$refs.modalDetail.show(record, type); |
||||
this.$refs.modalDetail.title=type+"-详情"; |
||||
}, |
||||
/* 导出 */ |
||||
handleExportXls2(){ |
||||
let paramsStr = encodeURI(JSON.stringify(this.getQueryParams())); |
||||
let url = `${window._CONFIG['domianURL']}/${this.url.exportXlsUrl}?paramsStr=${paramsStr}`; |
||||
window.location.href = url; |
||||
}, |
||||
handleExportXls(fileName){ |
||||
if(!fileName || typeof fileName != "string"){ |
||||
fileName = "导出文件" |
||||
} |
||||
let param = {...this.queryParam}; |
||||
if(this.selectedRowKeys && this.selectedRowKeys.length>0){ |
||||
param['selections'] = this.selectedRowKeys.join(",") |
||||
} |
||||
console.log("导出参数",param) |
||||
downFile(this.url.exportXlsUrl,param).then((data)=>{ |
||||
if (!data) { |
||||
this.$message.warning("文件下载失败") |
||||
return |
||||
} |
||||
if (typeof window.navigator.msSaveBlob !== 'undefined') { |
||||
window.navigator.msSaveBlob(new Blob([data],{type: 'application/vnd.ms-excel'}), fileName+'.xls') |
||||
}else{ |
||||
let url = window.URL.createObjectURL(new Blob([data],{type: 'application/vnd.ms-excel'})) |
||||
let link = document.createElement('a') |
||||
link.style.display = 'none' |
||||
link.href = url |
||||
link.setAttribute('download', fileName+'.xls') |
||||
document.body.appendChild(link) |
||||
link.click() |
||||
document.body.removeChild(link); //下载完成移除元素
|
||||
window.URL.revokeObjectURL(url); //释放掉blob对象
|
||||
} |
||||
}) |
||||
}, |
||||
/* 导入 */ |
||||
handleImportExcel(info){ |
||||
console.log(info,'info') |
||||
if (info.file.status !== 'uploading') { |
||||
console.log(info.file, info.fileList); |
||||
} |
||||
if (info.file.status === 'done') { |
||||
if (info.file.response) { |
||||
// this.$message.success(`${info.file.name} 文件上传成功`);
|
||||
if (info.file.response.code === 200) { |
||||
this.$message.success(info.file.response.data || `${info.file.name} 文件上传成功`) |
||||
} else { |
||||
this.$message.warning(info.file.response.data) |
||||
} |
||||
this.loadData() |
||||
} else { |
||||
this.$message.error(`${info.file.name} ${info.file.response.data}.`); |
||||
} |
||||
} else if (info.file.status === 'error') { |
||||
this.$message.error(`文件上传失败: ${info.file.msg} `); |
||||
} |
||||
}, |
||||
/* 图片预览 */ |
||||
getImgView(text){ |
||||
if(text && text.indexOf(",")>0){ |
||||
text = text.substring(0,text.indexOf(",")) |
||||
} |
||||
return getFileAccessHttpUrl(text) |
||||
}, |
||||
/* 文件下载 */ |
||||
uploadFile(text){ |
||||
if(!text){ |
||||
this.$message.warning("未知的文件") |
||||
return; |
||||
} |
||||
if(text.indexOf(",")>0){ |
||||
text = text.substring(0,text.indexOf(",")) |
||||
} |
||||
let url = getFileAccessHttpUrl(text) |
||||
window.open(url); |
||||
}, |
||||
/* 按钮权限 */ |
||||
initActiveBtnStr() { |
||||
let funId = Vue.ls.get('funId'); //功能id
|
||||
let btnStrList = Vue.ls.get('winBtnStrList'); //按钮功能列表 JSON字符串
|
||||
this.btnEnableList = [1,2,3,4,5,,6,7]; //按钮列表
|
||||
if (funId && btnStrList) { |
||||
for (let i = 0; i < btnStrList.length; i++) { |
||||
if (btnStrList[i].funId == funId) { |
||||
if (btnStrList[i].btnStr) { |
||||
this.btnEnableList = btnStrList[i].btnStr; |
||||
} |
||||
} |
||||
} |
||||
} |
||||
}, |
||||
/* 初始化表格横向或纵向滚动 */ |
||||
initScroll() { |
||||
if (this.isMobile()) { |
||||
this.scroll.y = '' |
||||
} else { |
||||
let basicLength = 274 |
||||
let searchWrapperDomLen=0, operatorDomLen=0 |
||||
//搜索区域
|
||||
let searchWrapperDom = document.getElementsByClassName('table-page-search-wrapper') |
||||
//操作按钮区域
|
||||
let operatorDom = document.getElementsByClassName('table-operator') |
||||
if(searchWrapperDom && searchWrapperDom[0]) { |
||||
searchWrapperDomLen = searchWrapperDom[0].offsetHeight |
||||
} |
||||
if(operatorDom && operatorDom[0]) { |
||||
operatorDomLen = operatorDom[0].offsetHeight+10 |
||||
} |
||||
this.scroll.y = document.documentElement.clientHeight-searchWrapperDomLen-operatorDomLen-basicLength |
||||
} |
||||
}, |
||||
/** 表格增加合计行 */ |
||||
tableAddTotalRow(columns, dataSource) { |
||||
if(dataSource.length>0 && this.ipagination.pageSize%10===1) { |
||||
//分页条数为11、21、31等的时候增加合计行
|
||||
let numKey = 'rowIndex' |
||||
let totalRow = { [numKey]: '合计' } |
||||
//需要合计的列
|
||||
let parseCols = 'initialStock,currentStock,currentStockPrice,initialAmount,thisMonthAmount,currentAmount,inSum,inSumPrice,inOutSumPrice,' + |
||||
'outSum,outSumPrice,outInSumPrice,operNumber,allPrice,numSum,priceSum,prevSum,thisSum,thisAllPrice,billMoney,changeAmount,' + |
||||
'allPrice,currentNumber,lowSafeStock,highSafeStock,lowCritical,highCritical,initialPrice,intoPrice,intoStock,outPrice,outStock' |
||||
columns.forEach(column => { |
||||
let { key, dataIndex } = column |
||||
if (![key, dataIndex].includes(numKey)) { |
||||
let total = 0 |
||||
dataSource.forEach(data => { |
||||
if(parseCols.indexOf(dataIndex)>-1) { |
||||
if(data[dataIndex]) { |
||||
total += Number.parseFloat(data[dataIndex]) |
||||
} else { |
||||
total += 0 |
||||
} |
||||
} else { |
||||
total = '-' |
||||
} |
||||
}) |
||||
if (total !== '-') { |
||||
total = total.toFixed(2) |
||||
} |
||||
totalRow[dataIndex] = total |
||||
} |
||||
}) |
||||
dataSource.push(totalRow) |
||||
//总数要增加合计的行数,每页都有一行合计,所以总数要加上
|
||||
let size = Math.ceil(this.ipagination.total/(this.ipagination.pageSize-1)) |
||||
this.ipagination.total = this.ipagination.total + size |
||||
} |
||||
}, |
||||
paginationChange(page, pageSize) { |
||||
this.ipagination.current = page |
||||
this.ipagination.pageSize = pageSize |
||||
this.loadData(this.ipagination.current); |
||||
}, |
||||
paginationShowSizeChange(current, size) { |
||||
this.ipagination.current = current |
||||
this.ipagination.pageSize = size |
||||
this.loadData(this.ipagination.current); |
||||
} |
||||
} |
||||
|
||||
} |
@ -0,0 +1,82 @@
|
||||
//mixins/tableDragResize.js
|
||||
import Vue from 'vue' |
||||
import VueDraggableResizable from 'vue-draggable-resizable' |
||||
Vue.component('vue-draggable-resizable', VueDraggableResizable) |
||||
/** |
||||
* @param { 表格columns } tbCols |
||||
*/ |
||||
function initDrag(tbCols) { |
||||
const draggingMap = {} |
||||
tbCols.forEach((col) => { |
||||
const key = col.dataIndex || col.key//这儿要求表格数据中要有这两个属性
|
||||
draggingMap[key] = col.width || 0 |
||||
}) |
||||
const draggingState = Vue.observable(draggingMap) |
||||
return (h, props, children) => { |
||||
let thDom = null |
||||
const { key, ...restProps } = props |
||||
let col |
||||
if (key === 'selection-column') { |
||||
//表格加了复选框,不加这个判断col会是undefided
|
||||
col = {} |
||||
} else { |
||||
col = tbCols.find((item) => { |
||||
const k = item.dataIndex || item.key |
||||
return k === key |
||||
}) |
||||
} |
||||
if (!col.width) {//这儿要求表格数据中要有宽width属性,若是没有是不会执行下面的拖拽的
|
||||
return <th {...restProps}>{children}</th> |
||||
} |
||||
|
||||
const onDrag = (x) => { |
||||
draggingState[key] = 0 |
||||
col.width = Math.max(x, 1) |
||||
} |
||||
const onDragstop = () => { |
||||
draggingState[key] = thDom.getBoundingClientRect().width |
||||
} |
||||
|
||||
return ( |
||||
<th |
||||
{...restProps} |
||||
v-ant-ref={(r) => { |
||||
thDom = r |
||||
}} |
||||
width={draggingState[key]} |
||||
class="resize-table-th" |
||||
> |
||||
{children} |
||||
<vue-draggable-resizable |
||||
key={col.dataIndex || col.key} |
||||
class="table-draggable-handle" |
||||
w={10} |
||||
x={col.width || draggingState[key]} |
||||
z={1} |
||||
axis="x" |
||||
draggable={true} |
||||
resizable={false} |
||||
onDragging={onDrag} |
||||
onDragstop={onDragstop} |
||||
></vue-draggable-resizable> |
||||
</th> |
||||
) |
||||
} |
||||
} |
||||
export default { |
||||
methods: { |
||||
/** |
||||
* https://github.com/mauricius/vue-draggable-resizable
|
||||
* 表格列可拖拽 |
||||
* 表格上使用::components="drag(columns)" |
||||
* tips:columns中需包含dataIndex或者key和width(Number) |
||||
*/ |
||||
drag(columns) { |
||||
return { |
||||
header: { |
||||
cell: initDrag(columns), |
||||
}, |
||||
} |
||||
}, |
||||
}, |
||||
} |
@ -0,0 +1,176 @@
|
||||
const FormTypes = { |
||||
normal: 'normal', |
||||
input: 'input', |
||||
inputNumber: 'inputNumber', |
||||
checkbox: 'checkbox', |
||||
select: 'select', |
||||
date: 'date', |
||||
datetime: 'datetime', |
||||
upload: 'upload', |
||||
file: 'file', |
||||
image: 'image', |
||||
popup:'popup', |
||||
popupJsh:'popupJsh', |
||||
list_multi:"list_multi", |
||||
sel_search:"sel_search", |
||||
radio:'radio', |
||||
checkbox_meta:"checkbox_meta", |
||||
input_pop:'input_pop', |
||||
slot: 'slot', |
||||
hidden: 'hidden' |
||||
} |
||||
const VALIDATE_NO_PASSED = Symbol() |
||||
export { FormTypes, VALIDATE_NO_PASSED } |
||||
|
||||
/** |
||||
* 获取指定的 $refs 对象 |
||||
* 有时候可能会遇到组件未挂载到页面中的情况,导致无法获取 $refs 中的某个对象 |
||||
* 这个方法可以等待挂载完成之后再返回 $refs 的对象,避免报错 |
||||
* @author sunjianlei |
||||
**/ |
||||
export function getRefPromise(vm, name) { |
||||
return new Promise((resolve) => { |
||||
(function next() { |
||||
let ref = vm.$refs[name] |
||||
if (ref) { |
||||
resolve(ref) |
||||
} else { |
||||
setTimeout(() => { |
||||
next() |
||||
}, 10) |
||||
} |
||||
})() |
||||
}) |
||||
} |
||||
|
||||
/** |
||||
* 一次性验证主表单和所有的次表单 |
||||
* @param form 主表单 form 对象 |
||||
* @param cases 接收一个数组,每项都是一个JEditableTable实例 |
||||
* @returns {Promise<any>} |
||||
* @author sunjianlei |
||||
*/ |
||||
export function validateFormAndTables(form, cases) { |
||||
|
||||
if (!(form && typeof form.validateFields === 'function')) { |
||||
throw `form 参数需要的是一个form对象,而传入的却是${typeof form}` |
||||
} |
||||
|
||||
let options = {} |
||||
return new Promise((resolve, reject) => { |
||||
// 验证主表表单
|
||||
form.validateFields((err, values) => { |
||||
err ? reject({ error: VALIDATE_NO_PASSED }) : resolve(values) |
||||
}) |
||||
}).then(values => { |
||||
Object.assign(options, { formValue: values }) |
||||
// 验证所有子表的表单
|
||||
return validateTables(cases) |
||||
}).then(all => { |
||||
Object.assign(options, { tablesValue: all }) |
||||
return Promise.resolve(options) |
||||
}).catch(error => { |
||||
return Promise.reject(error) |
||||
}) |
||||
|
||||
} |
||||
|
||||
/** |
||||
* 验证并获取一个或多个表格的所有值 |
||||
* @param cases 接收一个数组,每项都是一个JEditableTable实例 |
||||
* @param deleteTempId 是否删除临时ID,如果设为true,行编辑就不返回新增行的ID,ID需要后台生成 |
||||
* @author sunjianlei |
||||
*/ |
||||
export function validateTables(cases, deleteTempId) { |
||||
if (!(cases instanceof Array)) { |
||||
throw `'validateTables'函数的'cases'参数需要的是一个数组,而传入的却是${typeof cases}` |
||||
} |
||||
return new Promise((resolve, reject) => { |
||||
let tables = [] |
||||
let index = 0; |
||||
if(!cases || cases.length==0){ |
||||
resolve() |
||||
} |
||||
(function next() { |
||||
let vm = cases[index] |
||||
vm.getAll(true, deleteTempId).then(all => { |
||||
tables[index] = all |
||||
// 判断校验是否全部完成,完成返回成功,否则继续进行下一步校验
|
||||
if (++index === cases.length) { |
||||
resolve(tables) |
||||
} else ( |
||||
next() |
||||
) |
||||
}, error => { |
||||
// 出现未验证通过的表单,不再进行下一步校验,直接返回失败并跳转到该表格
|
||||
if (error === VALIDATE_NO_PASSED) { |
||||
reject({ error: VALIDATE_NO_PASSED, index }) |
||||
} |
||||
reject(error) |
||||
}) |
||||
})() |
||||
}) |
||||
} |
||||
|
||||
/** |
||||
* 一次性验证主表单和所有的次表单-只校验单号 |
||||
* @param form 主表单 form 对象 |
||||
* @param cases 接收一个数组,每项都是一个JEditableTable实例 |
||||
* @returns {Promise<any>} |
||||
* @author sunjianlei |
||||
*/ |
||||
export function getListData(form, cases) { |
||||
let options = {} |
||||
return new Promise((resolve, reject) => { |
||||
// 验证主表表单
|
||||
form.validateFields(['number'],(err, values) => { |
||||
err ? reject({ error: VALIDATE_NO_PASSED }) : resolve(values) |
||||
}) |
||||
}).then(values => { |
||||
Object.assign(options, { formValue: values }) |
||||
// 验证所有子表的表单
|
||||
return getListTables(cases) |
||||
}).then(all => { |
||||
Object.assign(options, { tablesValue: all }) |
||||
return Promise.resolve(options) |
||||
}).catch(error => { |
||||
return Promise.reject(error) |
||||
}) |
||||
} |
||||
|
||||
/** |
||||
* 不验证直接获取一个或多个表格的所有值 |
||||
* @param cases 接收一个数组,每项都是一个JEditableTable实例 |
||||
* @param deleteTempId 是否删除临时ID,如果设为true,行编辑就不返回新增行的ID,ID需要后台生成 |
||||
* @author sunjianlei |
||||
*/ |
||||
export function getListTables(cases, deleteTempId) { |
||||
if (!(cases instanceof Array)) { |
||||
throw `'validateTables'函数的'cases'参数需要的是一个数组,而传入的却是${typeof cases}` |
||||
} |
||||
return new Promise((resolve, reject) => { |
||||
let tables = [] |
||||
let index = 0; |
||||
if(!cases || cases.length==0){ |
||||
resolve() |
||||
} |
||||
(function next() { |
||||
let vm = cases[index] |
||||
vm.getAll(false, deleteTempId).then(all => { |
||||
tables[index] = all |
||||
// 判断校验是否全部完成,完成返回成功,否则继续进行下一步校验
|
||||
if (++index === cases.length) { |
||||
resolve(tables) |
||||
} else ( |
||||
next() |
||||
) |
||||
}, error => { |
||||
// 出现未验证通过的表单,不再进行下一步校验,直接返回失败并跳转到该表格
|
||||
if (error === VALIDATE_NO_PASSED) { |
||||
reject({ error: VALIDATE_NO_PASSED, index }) |
||||
} |
||||
reject(error) |
||||
}) |
||||
})() |
||||
}) |
||||
} |
@ -0,0 +1,40 @@
|
||||
// import Vue from 'vue'
|
||||
import { mapState } from "vuex"; |
||||
|
||||
// const mixinsComputed = Vue.config.optionMergeStrategies.computed
|
||||
// const mixinsMethods = Vue.config.optionMergeStrategies.methods
|
||||
|
||||
const mixin = { |
||||
computed: { |
||||
...mapState({ |
||||
layoutMode: state => state.app.layout, |
||||
navTheme: state => state.app.theme, |
||||
primaryColor: state => state.app.color, |
||||
colorWeak: state => state.app.weak, |
||||
multipage: state => state.app.multipage,//多页签设置
|
||||
fixedHeader: state => state.app.fixedHeader, |
||||
fixSiderbar: state => state.app.fixSiderbar, |
||||
contentWidth: state => state.app.contentWidth, |
||||
autoHideHeader: state => state.app.autoHideHeader, |
||||
sidebarOpened: state => state.app.sidebar.opened |
||||
}) |
||||
} |
||||
} |
||||
|
||||
const mixinDevice = { |
||||
computed: { |
||||
...mapState({ |
||||
device: state => state.app.device, |
||||
}) |
||||
}, |
||||
methods: { |
||||
isMobile () { |
||||
return this.device === 'mobile' |
||||
}, |
||||
isDesktop () { |
||||
return this.device === 'desktop' |
||||
} |
||||
} |
||||
} |
||||
|
||||
export { mixin, mixinDevice } |
@ -0,0 +1,115 @@
|
||||
/** |
||||
* 该文件截取自 "ant-design-vue/es/_util/props-util.js" 文件,并对其做出特殊修改 |
||||
*/ |
||||
function classNames() { |
||||
let classes = [] |
||||
|
||||
for (let i = 0; i < arguments.length; i++) { |
||||
let arg = arguments[i] |
||||
if (!arg) continue |
||||
|
||||
let argType = typeof arg |
||||
|
||||
if (argType === 'string' || argType === 'number') { |
||||
classes.push(arg) |
||||
} else if (Array.isArray(arg) && arg.length) { |
||||
let inner = classNames.apply(null, arg) |
||||
if (inner) { |
||||
classes.push(inner) |
||||
} |
||||
} else if (argType === 'object') { |
||||
for (let key in arg) { |
||||
if (arg.hasOwnProperty(key) && arg[key]) { |
||||
classes.push(key) |
||||
} |
||||
} |
||||
} |
||||
} |
||||
return classes.join(' ') |
||||
} |
||||
|
||||
const camelizeRE = /-(\w)/g |
||||
|
||||
function camelize(str) { |
||||
return str.replace(camelizeRE, (_, c) => (c ? c.toUpperCase() : '')) |
||||
} |
||||
|
||||
|
||||
function objectCamelize(obj) { |
||||
let res = {} |
||||
Object.keys(obj).forEach(k => (res[camelize(k)] = obj[k])) |
||||
return res |
||||
} |
||||
|
||||
function parseStyleText(cssText = '', camel) { |
||||
const res = {} |
||||
const listDelimiter = /;(?![^(]*\))/g |
||||
const propertyDelimiter = /:(.+)/ |
||||
cssText.split(listDelimiter).forEach(function (item) { |
||||
if (item) { |
||||
const tmp = item.split(propertyDelimiter) |
||||
if (tmp.length > 1) { |
||||
const k = camel ? camelize(tmp[0].trim()) : tmp[0].trim() |
||||
res[k] = tmp[1].trim() |
||||
} |
||||
} |
||||
}) |
||||
return res |
||||
} |
||||
|
||||
export function getClass(ele) { |
||||
let data = {} |
||||
if (ele.data) { |
||||
data = ele.data |
||||
} else if (ele.$vnode && ele.$vnode.data) { |
||||
data = ele.$vnode.data |
||||
} |
||||
const tempCls = data.class || {} |
||||
const staticClass = data.staticClass |
||||
let cls = {} |
||||
staticClass && |
||||
staticClass.split(' ').forEach(c => { |
||||
cls[c.trim()] = true |
||||
}) |
||||
if (typeof tempCls === 'string') { |
||||
tempCls.split(' ').forEach(c => { |
||||
cls[c.trim()] = true |
||||
}) |
||||
} else if (Array.isArray(tempCls)) { |
||||
classNames(tempCls) |
||||
.split(' ') |
||||
.forEach(c => { |
||||
cls[c.trim()] = true |
||||
}) |
||||
} else { |
||||
cls = { ...cls, ...tempCls } |
||||
} |
||||
return cls |
||||
} |
||||
|
||||
export function getStyle(ele, camel) { |
||||
|
||||
getClass(ele) |
||||
|
||||
let data = {} |
||||
if (ele.data) { |
||||
data = ele.data |
||||
} else if (ele.$vnode && ele.$vnode.data) { |
||||
data = ele.$vnode.data |
||||
} |
||||
|
||||
// update-begin-author:sunjianlei date:20200303 for: style 和 staticStyle 可以共存
|
||||
let style = data.style || {} |
||||
let staticStyle = data.staticStyle |
||||
staticStyle = staticStyle ? objectCamelize(data.staticStyle) : {} |
||||
// update-end-author:sunjianlei date:20200303 for: style 和 staticStyle 可以共存
|
||||
|
||||
if (typeof style === 'string') { |
||||
style = parseStyleText(style, camel) |
||||
} else if (camel && style) { |
||||
// 驼峰化
|
||||
style = objectCamelize(style) |
||||
} |
||||
return { ...staticStyle, ...style } |
||||
} |
||||
|
@ -0,0 +1,669 @@
|
||||
import api from '@/api/api' |
||||
import isURL from '@/utils/validate' |
||||
import XLSX from 'xlsx' |
||||
|
||||
export function timeFix() { |
||||
const time = new Date() |
||||
const hour = time.getHours() |
||||
return hour < 9 ? '早上好' : (hour <= 11 ? '上午好' : (hour <= 13 ? '中午好' : (hour < 20 ? '下午好' : '晚上好'))) |
||||
} |
||||
|
||||
export function welcome() { |
||||
const arr = ['休息一会儿吧', '准备吃什么呢?', '要不要打一把 DOTA', '我猜你可能累了'] |
||||
let index = Math.floor((Math.random()*arr.length)) |
||||
return arr[index] |
||||
} |
||||
|
||||
/** |
||||
* 触发 window.resize |
||||
*/ |
||||
export function triggerWindowResizeEvent() { |
||||
let event = document.createEvent('HTMLEvents') |
||||
event.initEvent('resize', true, true) |
||||
event.eventType = 'message' |
||||
window.dispatchEvent(event) |
||||
} |
||||
|
||||
/** |
||||
* 过滤对象中为空的属性 |
||||
* @param obj |
||||
* @returns {*} |
||||
*/ |
||||
export function filterObj(obj) { |
||||
if (!(typeof obj == 'object')) { |
||||
return; |
||||
} |
||||
|
||||
for ( let key in obj) { |
||||
if (obj.hasOwnProperty(key) |
||||
&& (obj[key] == null || obj[key] == undefined || obj[key] === '')) { |
||||
delete obj[key]; |
||||
} |
||||
} |
||||
return obj; |
||||
} |
||||
|
||||
/** |
||||
* 时间格式化 |
||||
* @param value |
||||
* @param fmt |
||||
* @returns {*} |
||||
*/ |
||||
export function formatDate(value, fmt) { |
||||
let regPos = /^\d+(\.\d+)?$/; |
||||
if(regPos.test(value)){ |
||||
//如果是数字
|
||||
let getDate = new Date(value); |
||||
let o = { |
||||
'M+': getDate.getMonth() + 1, |
||||
'd+': getDate.getDate(), |
||||
'h+': getDate.getHours(), |
||||
'm+': getDate.getMinutes(), |
||||
's+': getDate.getSeconds(), |
||||
'q+': Math.floor((getDate.getMonth() + 3) / 3), |
||||
'S': getDate.getMilliseconds() |
||||
}; |
||||
if (/(y+)/.test(fmt)) { |
||||
fmt = fmt.replace(RegExp.$1, (getDate.getFullYear() + '').substr(4 - RegExp.$1.length)) |
||||
} |
||||
for (let k in o) { |
||||
if (new RegExp('(' + k + ')').test(fmt)) { |
||||
fmt = fmt.replace(RegExp.$1, (RegExp.$1.length === 1) ? (o[k]) : (('00' + o[k]).substr(('' + o[k]).length))) |
||||
} |
||||
} |
||||
return fmt; |
||||
}else{ |
||||
//TODO
|
||||
value = value.trim(); |
||||
return value.substr(0,fmt.length); |
||||
} |
||||
} |
||||
|
||||
// 生成首页路由
|
||||
|
||||
|
||||
// 生成嵌套路由(子路由)
|
||||
|
||||
function generateChildRouters (data) { |
||||
const routers = []; |
||||
for (let item of data) { |
||||
let componentPath = ""; |
||||
item.route = "1"; |
||||
if(item.component.indexOf("layouts")>=0){ |
||||
componentPath = () => import('@/components'+item.component); |
||||
} else { |
||||
componentPath = () => import('@/views'+item.component); |
||||
} |
||||
// eslint-disable-next-line
|
||||
let URL = (item.url|| '').replace(/{{([^}}]+)?}}/g, (s1, s2) => eval(s2)) // URL支持{{ window.xxx }}占位符变量
|
||||
if (isURL(URL)) { |
||||
item.url = URL; |
||||
} |
||||
let componentName ='' |
||||
if(item.component) { |
||||
let index = item.component.lastIndexOf("\/"); |
||||
componentName = item.component.substring(index + 1, item.component.length); |
||||
} |
||||
let menu = { |
||||
path: item.url, |
||||
name: item.text, |
||||
component: componentPath, |
||||
meta: { |
||||
id: item.id, |
||||
title: item.text, |
||||
icon: item.icon, |
||||
url: item.url, |
||||
componentName:componentName, |
||||
internalOrExternal:true, |
||||
keepAlive: true |
||||
// permissionList:""
|
||||
} |
||||
} |
||||
if (item.children && item.children.length > 0) { |
||||
menu.children = [...generateChildRouters( item.children)]; |
||||
} |
||||
//--update-begin----author:scott---date:20190320------for:根据后台菜单配置,判断是否路由菜单字段,动态选择是否生成路由(为了支持参数URL菜单)------
|
||||
//判断是否生成路由
|
||||
if(item.route && item.route === '0'){ |
||||
//console.log(' 不生成路由 item.route: '+item.route);
|
||||
//console.log(' 不生成路由 item.path: '+item.path);
|
||||
}else{ |
||||
routers.push(menu); |
||||
} |
||||
//--update-end----author:scott---date:20190320------for:根据后台菜单配置,判断是否路由菜单字段,动态选择是否生成路由(为了支持参数URL菜单)------
|
||||
} |
||||
return routers |
||||
} |
||||
|
||||
/** |
||||
* 深度克隆对象、数组 |
||||
* @param obj 被克隆的对象 |
||||
* @return 克隆后的对象 |
||||
*/ |
||||
export function cloneObject(obj) { |
||||
return JSON.parse(JSON.stringify(obj)) |
||||
} |
||||
|
||||
/** |
||||
* 随机生成数字 |
||||
* |
||||
* 示例:生成长度为 12 的随机数:randomNumber(12) |
||||
* 示例:生成 3~23 之间的随机数:randomNumber(3, 23) |
||||
* |
||||
* @param1 最小值 | 长度 |
||||
* @param2 最大值 |
||||
* @return int 生成后的数字 |
||||
*/ |
||||
export function randomNumber() { |
||||
// 生成 最小值 到 最大值 区间的随机数
|
||||
const random = (min, max) => { |
||||
return Math.floor(Math.random() * (max - min + 1) + min) |
||||
} |
||||
if (arguments.length === 1) { |
||||
let [length] = arguments |
||||
// 生成指定长度的随机数字,首位一定不是 0
|
||||
let nums = [...Array(length).keys()].map((i) => (i > 0 ? random(0, 9) : random(1, 9))) |
||||
return parseInt(nums.join('')) |
||||
} else if (arguments.length >= 2) { |
||||
let [min, max] = arguments |
||||
return random(min, max) |
||||
} else { |
||||
return Number.NaN |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 随机生成字符串 |
||||
* @param length 字符串的长度 |
||||
* @param chats 可选字符串区间(只会生成传入的字符串中的字符) |
||||
* @return string 生成的字符串 |
||||
*/ |
||||
export function randomString(length, chats) { |
||||
if (!length) length = 1 |
||||
if (!chats) chats = '0123456789qwertyuioplkjhgfdsazxcvbnm' |
||||
let str = '' |
||||
for (let i = 0; i < length; i++) { |
||||
let num = randomNumber(0, chats.length - 1) |
||||
str += chats[num] |
||||
} |
||||
return str |
||||
} |
||||
|
||||
/** |
||||
* 随机生成uuid |
||||
* @return string 生成的uuid |
||||
*/ |
||||
export function randomUUID() { |
||||
let chats = '0123456789abcdef' |
||||
return randomString(32, chats) |
||||
} |
||||
|
||||
/** |
||||
* 下划线转驼峰 |
||||
* @param string |
||||
* @returns {*} |
||||
*/ |
||||
export function underLine2CamelCase(string){ |
||||
return string.replace( /_([a-z])/g, function( all, letter ) { |
||||
return letter.toUpperCase(); |
||||
}); |
||||
} |
||||
|
||||
/** |
||||
* 判断是否显示办理按钮 |
||||
* @param bpmStatus |
||||
* @returns {*} |
||||
*/ |
||||
export function showDealBtn(bpmStatus){ |
||||
if(bpmStatus!="1"&&bpmStatus!="3"&&bpmStatus!="4"){ |
||||
return true; |
||||
} |
||||
return false; |
||||
} |
||||
|
||||
/** |
||||
* 增强CSS,可以在页面上输出全局css |
||||
* @param css 要增强的css |
||||
* @param id style标签的id,可以用来清除旧样式 |
||||
*/ |
||||
export function cssExpand(css, id) { |
||||
let style = document.createElement('style') |
||||
style.type = "text/css" |
||||
style.innerHTML = `@charset "UTF-8"; ${css}` |
||||
// 清除旧样式
|
||||
if (id) { |
||||
let $style = document.getElementById(id) |
||||
if ($style != null) $style.outerHTML = '' |
||||
style.id = id |
||||
} |
||||
// 应用新样式
|
||||
document.head.appendChild(style) |
||||
} |
||||
|
||||
|
||||
/** 用于js增强事件,运行JS代码,可以传参 */ |
||||
// options 所需参数:
|
||||
// 参数名 类型 说明
|
||||
// vm VueComponent vue实例
|
||||
// event Object event对象
|
||||
// jsCode String 待执行的js代码
|
||||
// errorMessage String 执行出错后的提示(控制台)
|
||||
export function jsExpand(options = {}) { |
||||
|
||||
// 绑定到window上的keyName
|
||||
let windowKeyName = 'J_CLICK_EVENT_OPTIONS' |
||||
if (typeof window[windowKeyName] != 'object') { |
||||
window[windowKeyName] = {} |
||||
} |
||||
|
||||
// 随机生成JS增强的执行id,防止冲突
|
||||
let id = randomString(16, 'qwertyuioplkjhgfdsazxcvbnm'.toUpperCase()) |
||||
// 封装按钮点击事件
|
||||
let code = ` |
||||
(function (o_${id}) { |
||||
try { |
||||
(function (globalEvent, vm) { |
||||
${options.jsCode} |
||||
})(o_${id}.event, o_${id}.vm) |
||||
} catch (e) { |
||||
o_${id}.error(e) |
||||
} |
||||
o_${id}.done() |
||||
})(window['${windowKeyName}']['EVENT_${id}']) |
||||
` |
||||
// 创建script标签
|
||||
const script = document.createElement('script') |
||||
// 将需要传递的参数挂载到window对象上
|
||||
window[windowKeyName]['EVENT_' + id] = { |
||||
vm: options.vm, |
||||
event: options.event, |
||||
// 当执行完成时,无论如何都会调用的回调事件
|
||||
done() { |
||||
// 执行完后删除新增的 script 标签不会撤销执行结果(已产生的结果不会被撤销)
|
||||
script.outerHTML = '' |
||||
delete window[windowKeyName]['EVENT_' + id] |
||||
}, |
||||
// 当js运行出错的时候调用的事件
|
||||
error(e) { |
||||
console.group(`${options.errorMessage || '用户自定义JS增强代码运行出错'}(${new Date()})`) |
||||
console.error(e) |
||||
console.groupEnd() |
||||
} |
||||
} |
||||
// 将事件挂载到document中
|
||||
script.innerHTML = code |
||||
document.body.appendChild(script) |
||||
} |
||||
|
||||
|
||||
/** |
||||
* 重复值验证工具方法 |
||||
* |
||||
* 使用示例: |
||||
* { validator: (rule, value, callback) => validateDuplicateValue('sys_fill_rule', 'rule_code', value, this.model.id, callback) } |
||||
* |
||||
* @param tableName 被验证的表名 |
||||
* @param fieldName 被验证的字段名 |
||||
* @param fieldVal 被验证的值 |
||||
* @param dataId 数据ID,可空 |
||||
* @param callback |
||||
*/ |
||||
export function validateDuplicateValue(tableName, fieldName, fieldVal, dataId, callback) { |
||||
if (fieldVal) { |
||||
let params = { tableName, fieldName, fieldVal, dataId } |
||||
api.duplicateCheck(params).then(res => { |
||||
res['success'] ? callback() : callback(res['message']) |
||||
}).catch(err => { |
||||
callback(err.message || err) |
||||
}) |
||||
} else { |
||||
callback() |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 根据编码校验规则code,校验传入的值是否合法 |
||||
* |
||||
* 使用示例: |
||||
* { validator: (rule, value, callback) => validateCheckRule('common', value, callback) } |
||||
* |
||||
* @param ruleCode 编码校验规则 code |
||||
* @param value 被验证的值 |
||||
* @param callback |
||||
*/ |
||||
export function validateCheckRule(ruleCode, value, callback) { |
||||
if (ruleCode && value) { |
||||
value = encodeURIComponent(value) |
||||
api.checkRuleByCode({ ruleCode, value }).then(res => { |
||||
res['success'] ? callback() : callback(res['message']) |
||||
}).catch(err => { |
||||
callback(err.message || err) |
||||
}) |
||||
} else { |
||||
callback() |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 如果值不存在就 push 进数组,反之不处理 |
||||
* @param array 要操作的数据 |
||||
* @param value 要添加的值 |
||||
* @param key 可空,如果比较的是对象,可能存在地址不一样但值实际上是一样的情况,可以传此字段判断对象中唯一的字段,例如 id。不传则直接比较实际值 |
||||
* @returns {boolean} 成功 push 返回 true,不处理返回 false |
||||
*/ |
||||
export function pushIfNotExist(array, value, key) { |
||||
for (let item of array) { |
||||
if (key && (item[key] === value[key])) { |
||||
return false |
||||
} else if (item === value) { |
||||
return false |
||||
} |
||||
} |
||||
array.push(value) |
||||
return true |
||||
} |
||||
|
||||
/** |
||||
* 可用于判断是否成功 |
||||
* @type {symbol} |
||||
*/ |
||||
export const succeedSymbol = Symbol() |
||||
/** |
||||
* 可用于判断是否失败 |
||||
* @type {symbol} |
||||
*/ |
||||
export const failedSymbol = Symbol() |
||||
|
||||
/** |
||||
* 使 promise 无论如何都会 resolve,除非传入的参数不是一个Promise对象或返回Promise对象的方法 |
||||
* 一般用在 Promise.all 中 |
||||
* |
||||
* @param promise 可传Promise对象或返回Promise对象的方法 |
||||
* @returns {Promise<any>} |
||||
*/ |
||||
export function alwaysResolve(promise) { |
||||
return new Promise((resolve, reject) => { |
||||
let p = promise |
||||
if (typeof promise === 'function') { |
||||
p = promise() |
||||
} |
||||
if (p instanceof Promise) { |
||||
p.then(data => { |
||||
resolve({ type: succeedSymbol, data }) |
||||
}).catch(error => { |
||||
resolve({ type: failedSymbol, error }) |
||||
}) |
||||
} else { |
||||
reject('alwaysResolve: 传入的参数不是一个Promise对象或返回Promise对象的方法') |
||||
} |
||||
}) |
||||
} |
||||
|
||||
/** |
||||
* 简单实现防抖方法 |
||||
* |
||||
* 防抖(debounce)函数在第一次触发给定的函数时,不立即执行函数,而是给出一个期限值(delay),比如100ms。 |
||||
* 如果100ms内再次执行函数,就重新开始计时,直到计时结束后再真正执行函数。 |
||||
* 这样做的好处是如果短时间内大量触发同一事件,只会执行一次函数。 |
||||
* |
||||
* @param fn 要防抖的函数 |
||||
* @param delay 防抖的毫秒数 |
||||
* @returns {Function} |
||||
*/ |
||||
export function simpleDebounce(fn, delay = 100) { |
||||
let timer = null |
||||
return function () { |
||||
let args = arguments |
||||
if (timer) { |
||||
clearTimeout(timer) |
||||
} |
||||
timer = setTimeout(() => { |
||||
fn.apply(null, args) |
||||
}, delay) |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 不用正则的方式替换所有值 |
||||
* @param text 被替换的字符串 |
||||
* @param checker 替换前的内容 |
||||
* @param replacer 替换后的内容 |
||||
* @returns {String} 替换后的字符串 |
||||
*/ |
||||
export function replaceAll(text, checker, replacer) { |
||||
let lastText = text |
||||
text = text.replace(checker, replacer) |
||||
if (lastText !== text) { |
||||
return replaceAll(text, checker, replacer) |
||||
} |
||||
return text |
||||
} |
||||
|
||||
/** |
||||
* 转换商品扩展字段的格式 |
||||
* @param thisRows |
||||
* @param checker |
||||
* @param replacer |
||||
* @returns {string} |
||||
*/ |
||||
export function getMpListShort(thisRows, checker, replacer) { |
||||
let mPropertyListShort = '' |
||||
let nativeNameStr = '' |
||||
for (let i = 0; i < thisRows.length; i++) { |
||||
if (thisRows[i].enabled) { |
||||
nativeNameStr += thisRows[i].nativeName + ","; |
||||
} |
||||
} |
||||
if (nativeNameStr) { |
||||
mPropertyListShort = nativeNameStr.substring(0, nativeNameStr.length - 1); |
||||
} |
||||
return mPropertyListShort |
||||
} |
||||
|
||||
/** |
||||
* js获取当前月份, 格式“yyyy-MM” |
||||
*/ |
||||
export function getNowFormatMonth() { |
||||
var date = new Date(); |
||||
var seperator1 = "-"; |
||||
var month = date.getMonth() + 1; |
||||
if (month >= 1 && month <= 9) { |
||||
month = "0" + month; |
||||
} |
||||
var currentdate = date.getFullYear() + seperator1 + month; |
||||
return currentdate; |
||||
} |
||||
|
||||
/** |
||||
* js获取当前日期, 格式“yyyy-MM-dd” |
||||
*/ |
||||
export function getFormatDate() { |
||||
var date = new Date(); |
||||
var seperator1 = "-"; |
||||
var year = date.getFullYear(); |
||||
var month = date.getMonth() + 1; |
||||
var strDate = date.getDate(); |
||||
if (month >= 1 && month <= 9) { |
||||
month = "0" + month; |
||||
} |
||||
if (strDate >= 0 && strDate <= 9) { |
||||
strDate = "0" + strDate; |
||||
} |
||||
var currentdate = year + seperator1 + month + seperator1 + strDate; |
||||
return currentdate; |
||||
} |
||||
|
||||
/** |
||||
* js获取当前时间, 格式“yyyy-MM-dd HH:MM:SS” |
||||
*/ |
||||
export function getNowFormatDateTime() { |
||||
var date = new Date(); |
||||
var seperator1 = "-"; |
||||
var seperator2 = ":"; |
||||
var month = date.getMonth() + 1; |
||||
var strDate = date.getDate(); |
||||
var strHours = date.getHours(); |
||||
var strMinutes = date.getMinutes(); |
||||
var strSeconds = date.getSeconds(); |
||||
if (month >= 1 && month <= 9) { |
||||
month = "0" + month; |
||||
} |
||||
if (strDate >= 0 && strDate <= 9) { |
||||
strDate = "0" + strDate; |
||||
} |
||||
if (strHours >= 0 && strHours <= 9) { |
||||
strHours = "0" + strHours; |
||||
} |
||||
if (strMinutes >= 0 && strMinutes <= 9) { |
||||
strMinutes = "0" + strMinutes; |
||||
} |
||||
if (strSeconds >= 0 && strSeconds <= 9) { |
||||
strSeconds = "0" + strSeconds; |
||||
} |
||||
var currentdate = date.getFullYear() + seperator1 + month + seperator1 + strDate |
||||
+ " " + strHours + seperator2 + strMinutes |
||||
+ seperator2 + strSeconds; |
||||
return currentdate; |
||||
} |
||||
|
||||
/** |
||||
* js获取当前时间, 格式“MMddHHMMSS” |
||||
*/ |
||||
export function getNowFormatStr() { |
||||
var date = new Date(); |
||||
var month = date.getMonth() + 1; |
||||
var strDate = date.getDate(); |
||||
var strHours = date.getHours(); |
||||
var strMinutes = date.getMinutes(); |
||||
var strSeconds = date.getSeconds(); |
||||
if (month >= 1 && month <= 9) { |
||||
month = "0" + month; |
||||
} |
||||
if (strDate >= 0 && strDate <= 9) { |
||||
strDate = "0" + strDate; |
||||
} |
||||
if (strHours >= 0 && strHours <= 9) { |
||||
strHours = "0" + strHours; |
||||
} |
||||
if (strMinutes >= 0 && strMinutes <= 9) { |
||||
strMinutes = "0" + strMinutes; |
||||
} |
||||
if (strSeconds >= 0 && strSeconds <= 9) { |
||||
strSeconds = "0" + strSeconds; |
||||
} |
||||
var currentdate = month + strDate + strHours + strMinutes + strSeconds; |
||||
console.log(currentdate,'currentdate') |
||||
return currentdate; |
||||
} |
||||
|
||||
/** |
||||
* JS中根据指定值删除数组中的元素 |
||||
* @param arrylist |
||||
* @param val |
||||
*/ |
||||
export function removeByVal(arrylist, val) { |
||||
for(var i = 0; i < arrylist .length; i++) { |
||||
if(arrylist [i] == val) { |
||||
arrylist .splice(i, 1); |
||||
break; |
||||
} |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 将数组单个金额中的数值转为负数 |
||||
* @param arr |
||||
* @returns {Array} |
||||
*/ |
||||
export function changeListFmtMinus(arr) { |
||||
var newArr = new Array(); |
||||
for(var i=0; i<arr.length; i++) { |
||||
if(arr[i] < 0){ |
||||
newArr.push((arr[i]-0).toString()); |
||||
} |
||||
else { |
||||
newArr.push((0 - arr[i]).toString()); |
||||
} |
||||
} |
||||
return newArr; |
||||
} |
||||
|
||||
/** |
||||
通用的打开下载对话框方法,没有测试过具体兼容性 |
||||
@param url 下载地址,也可以是一个blob对象,必选 |
||||
@param saveName 保存文件名,可选 |
||||
*/ |
||||
export function openDownloadDialog (url, saveName) { |
||||
if (typeof url === 'object' && url instanceof Blob) { |
||||
url = URL.createObjectURL(url) // 创建blob地址
|
||||
} |
||||
let aLink = document.createElement('a') |
||||
aLink.href = url |
||||
saveName = saveName + '_' + getNowFormatStr() + '.xlsx' |
||||
aLink.download = saveName || '' // HTML5新增的属性,指定保存文件名,可以不要后缀,注意,file:///模式下不会生效
|
||||
let event |
||||
if (window.MouseEvent) event = new MouseEvent('click') |
||||
else { |
||||
event = document.createEvent('MouseEvents') |
||||
event.initMouseEvent('click', true, false, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null) |
||||
} |
||||
aLink.dispatchEvent(event) |
||||
} |
||||
|
||||
/** |
||||
* 将一个sheet转成最终的excel文件的blob对象,然后利用URL.createObjectURL下载 |
||||
* @param sheet |
||||
* @param sheetName |
||||
* @returns {Blob} |
||||
*/ |
||||
export function sheet2blob (aoa, sheetName) { |
||||
let sheet = XLSX.utils.aoa_to_sheet(aoa) |
||||
sheetName = sheetName || 'sheet1' |
||||
let workbook = { |
||||
SheetNames: [sheetName], |
||||
Sheets: {} |
||||
} |
||||
workbook.Sheets[sheetName] = sheet |
||||
// 生成excel的配置项
|
||||
let wopts = { |
||||
bookType: 'xlsx', // 要生成的文件类型
|
||||
bookSST: false, // 是否生成Shared String Table,官方解释是,如果开启生成速度会下降,但在低版本IOS设备上有更好的兼容性
|
||||
type: 'binary' |
||||
} |
||||
let wbout = XLSX.write(workbook, wopts) |
||||
let blob = new Blob([s2ab(wbout)], { type: 'application/octet-stream' }) |
||||
// 字符串转ArrayBuffer
|
||||
function s2ab (s) { |
||||
let buf = new ArrayBuffer(s.length) |
||||
let view = new Uint8Array(buf) |
||||
for (let i = 0; i != s.length; ++i) view[i] = s.charCodeAt(i) & 0xFF |
||||
return buf |
||||
} |
||||
return blob |
||||
} |
||||
|
||||
|
||||
/** |
||||
* 回车后自动跳到下一个input |
||||
*/ |
||||
export function autoJumpNextInput(domInfo) { |
||||
let domIndex = 0 |
||||
let inputs = document.getElementById(domInfo).getElementsByTagName('input') |
||||
inputs[domIndex].focus() |
||||
document.getElementById(domInfo).addEventListener('keydown',function(e){ |
||||
if(e.keyCode === 13){ |
||||
domIndex++ |
||||
if(domIndex === inputs.length) { |
||||
domIndex = 0 |
||||
} |
||||
inputs[domIndex].focus() |
||||
} |
||||
}) |
||||
for(let i=0; i<inputs.length; i++){ |
||||
//这个index就是做个介质,来获取当前的i是第几个
|
||||
inputs[i].index = i; |
||||
inputs[i].onclick = function () { |
||||
domIndex = this.index |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,216 @@
|
||||
<template> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="24"> |
||||
<a-card :style="cardStyle" :bordered="false"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="searchQuery"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据编号" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入单据编号" v-model="queryParam.number"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="商品信息" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入物料编码、名称、规格、型号" v-model="queryParam.materialParam"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据日期" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-range-picker |
||||
style="width:100%" |
||||
v-model="queryParam.createTimeRange" |
||||
format="YYYY-MM-DD" |
||||
:placeholder="['开始时间', '结束时间']" |
||||
@change="onDateChange" |
||||
@ok="onDateOk" |
||||
/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<template v-if="toggleSearchStatus"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="仓库名称" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="请选择仓库" showSearch optionFilterProp="children" v-model="queryParam.depotId"> |
||||
<a-select-option v-for="(depot,index) in depotList" :key="index" :value="depot.id"> |
||||
{{ depot.depotName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="操作员" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择操作员" showSearch optionFilterProp="children" v-model="queryParam.creator"> |
||||
<a-select-option v-for="(item,index) in userList" :key="index" :value="item.id"> |
||||
{{ item.userName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
</template> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-button type="primary" @click="searchQuery">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset">重置</a-button> |
||||
<a @click="handleToggleSearch" style="margin-left: 8px"> |
||||
{{ toggleSearchStatus ? '收起' : '展开' }} |
||||
<a-icon :type="toggleSearchStatus ? 'up' : 'down'"/> |
||||
</a> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
</div> |
||||
<!-- 操作按钮区域 --> |
||||
<div class="table-operator" style="margin-top: 5px"> |
||||
<a-button v-if="btnEnableList.indexOf(1)>-1" @click="myHandleAdd" type="primary" icon="plus">新增</a-button> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" v-if="btnEnableList.indexOf(1)>-1" @click="batchDel"><a-icon type="delete"/>删除</a-menu-item> |
||||
<a-menu-item key="2" v-if="btnEnableList.indexOf(2)>-1" @click="batchSetStatus(1)"><a-icon type="check"/>审核</a-menu-item> |
||||
<a-menu-item key="3" v-if="btnEnableList.indexOf(7)>-1" @click="batchSetStatus(0)"><a-icon type="stop"/>反审核</a-menu-item> |
||||
</a-menu> |
||||
<a-button> |
||||
批量操作 <a-icon type="down" /> |
||||
</a-button> |
||||
</a-dropdown> |
||||
<a-tooltip placement="left" title="用于两个仓库之间的商品调拨,调拨单会影响库存。" slot="action"> |
||||
<a-icon v-if="btnEnableList.indexOf(1)>-1" type="question-circle" style="font-size:20px;float:right;" /> |
||||
</a-tooltip> |
||||
</div> |
||||
<!-- table区域-begin --> |
||||
<a-row> |
||||
<a-col :md="2.5" :sm="24"> |
||||
<list-columns-setter v-model="columns" :def-columns="columns" style="float: right;"/> |
||||
</a-col> |
||||
</a-row> |
||||
<div> |
||||
<a-table |
||||
ref="table" |
||||
size="middle" |
||||
bordered |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:components="drag(columns)" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:scroll="scroll" |
||||
:loading="loading" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange}" |
||||
@change="handleTableChange"> |
||||
<span slot="action" slot-scope="text, record"> |
||||
<a @click="myHandleDetail(record, '调拨出库')">查看</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleEdit(record)">编辑</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleCopyAdd(record)">复制</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a-popconfirm v-if="btnEnableList.indexOf(1)>-1" title="确定删除吗?" @confirm="() => myHandleDelete(record)"> |
||||
<a>删除</a> |
||||
</a-popconfirm> |
||||
</span> |
||||
<template slot="customRenderStatus" slot-scope="status"> |
||||
<a-tag v-if="status == '0'" color="red">未审核</a-tag> |
||||
<a-tag v-if="status == '1'" color="green">已审核</a-tag> |
||||
</template> |
||||
</a-table> |
||||
</div> |
||||
<!-- table区域-end --> |
||||
<!-- 表单区域 --> |
||||
<allocation-out-modal ref="modalForm" @ok="modalFormOk"></allocation-out-modal> |
||||
<bill-detail ref="modalDetail"></bill-detail> |
||||
</a-card> |
||||
</a-col> |
||||
</a-row> |
||||
</template> |
||||
<!--power by jishenghua--> |
||||
<script> |
||||
import AllocationOutModal from './modules/AllocationOutModal' |
||||
import BillDetail from './dialog/BillDetail' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
import { BillListMixin } from './mixins/BillListMixin' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
import ListColumnsSetter from '@/components/ListColumnsSetter' |
||||
import tableDragResizeMixin from '@/mixins/tableDragResizeMixin' |
||||
export default { |
||||
name: "AllocationOutList", |
||||
mixins:[JeecgListMixin,BillListMixin,tableDragResizeMixin], |
||||
components: { |
||||
AllocationOutModal, |
||||
BillDetail, |
||||
JDate, |
||||
ListColumnsSetter |
||||
}, |
||||
data () { |
||||
return { |
||||
// 查询条件 |
||||
queryParam: { |
||||
number: "", |
||||
materialParam: "", |
||||
type: "出库", |
||||
subType: "调拨", |
||||
roleType: Vue.ls.get('roleType'), |
||||
depotId: "", |
||||
creator: "" |
||||
}, |
||||
labelCol: { |
||||
span: 5 |
||||
}, |
||||
wrapperCol: { |
||||
span: 18, |
||||
offset: 1 |
||||
}, |
||||
// 表头 |
||||
columns: [ |
||||
{ title: '单据编号', dataIndex: 'number',width:160, align: 'center', |
||||
customRender:function (text,record,index) { |
||||
if(record.linkNumber) { |
||||
return text + "[转]"; |
||||
} else { |
||||
return text; |
||||
} |
||||
} |
||||
}, |
||||
{ title: '商品信息', dataIndex: 'materialsList',width:220, align: 'center', ellipsis:true, |
||||
customRender:function (text,record,index) { |
||||
if(text) { |
||||
return text.replace(",",","); |
||||
} |
||||
} |
||||
}, |
||||
{ title: '单据日期', dataIndex: 'operTimeStr',width:145, align: 'center',}, |
||||
{ title: '操作员', dataIndex: 'userName',width:80, align: 'center', ellipsis:true}, |
||||
{ title: '金额合计', dataIndex: 'totalPrice',width:80, align: 'center',}, |
||||
{ title: '状态', dataIndex: 'status', width: 80, align: "center", |
||||
scopedSlots: { customRender: 'customRenderStatus' } |
||||
}, |
||||
{ |
||||
title: '操作', |
||||
dataIndex: 'action', |
||||
align:"center", width: 150, |
||||
scopedSlots: { customRender: 'action' }, |
||||
} |
||||
], |
||||
url: { |
||||
list: "/erp/depotHead/list", |
||||
delete: "/erp/depotHead/delete", |
||||
deleteBatch: "/erp/depotHead/deleteBatch", |
||||
batchSetStatusUrl: "/erp/depotHead/batchSetStatus" |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
}, |
||||
created() { |
||||
this.getDepotData() |
||||
this.initUser() |
||||
}, |
||||
methods: { |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
@import '~@assets/less/common.less' |
||||
</style> |
@ -0,0 +1,216 @@
|
||||
<template> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="24"> |
||||
<a-card :style="cardStyle" :bordered="false"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="searchQuery"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据编号" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入单据编号" v-model="queryParam.number"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="商品信息" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入物料编码、名称、规格、型号" v-model="queryParam.materialParam"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据日期" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-range-picker |
||||
style="width:100%" |
||||
v-model="queryParam.createTimeRange" |
||||
format="YYYY-MM-DD" |
||||
:placeholder="['开始时间', '结束时间']" |
||||
@change="onDateChange" |
||||
@ok="onDateOk" |
||||
/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<template v-if="toggleSearchStatus"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="仓库名称" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="请选择仓库" showSearch optionFilterProp="children" v-model="queryParam.depotId"> |
||||
<a-select-option v-for="(depot,index) in depotList" :key="index" :value="depot.id"> |
||||
{{ depot.depotName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="操作员" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择操作员" showSearch optionFilterProp="children" v-model="queryParam.creator"> |
||||
<a-select-option v-for="(item,index) in userList" :key="index" :value="item.id"> |
||||
{{ item.userName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
</template> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-button type="primary" @click="searchQuery">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset">重置</a-button> |
||||
<a @click="handleToggleSearch" style="margin-left: 8px"> |
||||
{{ toggleSearchStatus ? '收起' : '展开' }} |
||||
<a-icon :type="toggleSearchStatus ? 'up' : 'down'"/> |
||||
</a> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
</div> |
||||
<!-- 操作按钮区域 --> |
||||
<div class="table-operator" style="margin-top: 5px"> |
||||
<a-button v-if="btnEnableList.indexOf(1)>-1" @click="myHandleAdd" type="primary" icon="plus">新增</a-button> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" v-if="btnEnableList.indexOf(1)>-1" @click="batchDel"><a-icon type="delete"/>删除</a-menu-item> |
||||
<a-menu-item key="2" v-if="btnEnableList.indexOf(2)>-1" @click="batchSetStatus(1)"><a-icon type="check"/>审核</a-menu-item> |
||||
<a-menu-item key="3" v-if="btnEnableList.indexOf(7)>-1" @click="batchSetStatus(0)"><a-icon type="stop"/>反审核</a-menu-item> |
||||
</a-menu> |
||||
<a-button> |
||||
批量操作 <a-icon type="down" /> |
||||
</a-button> |
||||
</a-dropdown> |
||||
<a-tooltip placement="left" title="用于将多种商品合并成一个商品,被合并的商品库存减少,合并后的商品库存增加。" slot="action"> |
||||
<a-icon v-if="btnEnableList.indexOf(1)>-1" type="question-circle" style="font-size:20px;float:right;" /> |
||||
</a-tooltip> |
||||
</div> |
||||
<!-- table区域-begin --> |
||||
<a-row> |
||||
<a-col :md="2.5" :sm="24"> |
||||
<list-columns-setter v-model="columns" :def-columns="columns" style="float: right;"/> |
||||
</a-col> |
||||
</a-row> |
||||
<div> |
||||
<a-table |
||||
ref="table" |
||||
size="middle" |
||||
bordered |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:components="drag(columns)" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:scroll="scroll" |
||||
:loading="loading" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange}" |
||||
@change="handleTableChange"> |
||||
<span slot="action" slot-scope="text, record"> |
||||
<a @click="myHandleDetail(record, '组装单')">查看</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleEdit(record)">编辑</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleCopyAdd(record)">复制</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a-popconfirm v-if="btnEnableList.indexOf(1)>-1" title="确定删除吗?" @confirm="() => myHandleDelete(record)"> |
||||
<a>删除</a> |
||||
</a-popconfirm> |
||||
</span> |
||||
<template slot="customRenderStatus" slot-scope="status"> |
||||
<a-tag v-if="status == '0'" color="red">未审核</a-tag> |
||||
<a-tag v-if="status == '1'" color="green">已审核</a-tag> |
||||
</template> |
||||
</a-table> |
||||
</div> |
||||
<!-- table区域-end --> |
||||
<!-- 表单区域 --> |
||||
<assemble-modal ref="modalForm" @ok="modalFormOk"></assemble-modal> |
||||
<bill-detail ref="modalDetail"></bill-detail> |
||||
</a-card> |
||||
</a-col> |
||||
</a-row> |
||||
</template> |
||||
<!--power by ji sheng hua--> |
||||
<script> |
||||
import AssembleModal from './modules/AssembleModal' |
||||
import BillDetail from './dialog/BillDetail' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
import { BillListMixin } from './mixins/BillListMixin' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
import ListColumnsSetter from '@/components/ListColumnsSetter' |
||||
import tableDragResizeMixin from '@/mixins/tableDragResizeMixin' |
||||
export default { |
||||
name: "AssembleList", |
||||
mixins:[JeecgListMixin,BillListMixin,tableDragResizeMixin], |
||||
components: { |
||||
AssembleModal, |
||||
BillDetail, |
||||
JDate, |
||||
ListColumnsSetter |
||||
}, |
||||
data () { |
||||
return { |
||||
// 查询条件 |
||||
queryParam: { |
||||
number: "", |
||||
materialParam: "", |
||||
type: "其它", |
||||
subType: "组装单", |
||||
roleType: Vue.ls.get('roleType'), |
||||
depotId: "", |
||||
creator: "" |
||||
}, |
||||
labelCol: { |
||||
span: 5 |
||||
}, |
||||
wrapperCol: { |
||||
span: 18, |
||||
offset: 1 |
||||
}, |
||||
// 表头 |
||||
columns: [ |
||||
{ title: '单据编号', dataIndex: 'number',width:160, align:'center', |
||||
customRender:function (text,record,index) { |
||||
if(record.linkNumber) { |
||||
return text + "[转]"; |
||||
} else { |
||||
return text; |
||||
} |
||||
} |
||||
}, |
||||
{ title: '商品信息', dataIndex: 'materialsList',width:220, align:'center', ellipsis:true, |
||||
customRender:function (text,record,index) { |
||||
if(text) { |
||||
return text.replace(",",","); |
||||
} |
||||
} |
||||
}, |
||||
{ title: '单据日期', dataIndex: 'operTimeStr',width:145, align:'center',}, |
||||
{ title: '操作员', dataIndex: 'userName',width:80, align:'center', ellipsis:true}, |
||||
{ title: '金额合计', dataIndex: 'totalPrice',width:80, align:'center',}, |
||||
{ title: '状态', dataIndex: 'status', width: 80, align: "center", |
||||
scopedSlots: { customRender: 'customRenderStatus' } |
||||
}, |
||||
{ |
||||
title: '操作', |
||||
dataIndex: 'action', |
||||
align:"center", width: 150, |
||||
scopedSlots: { customRender: 'action' }, |
||||
} |
||||
], |
||||
url: { |
||||
list: "/erp/depotHead/list", |
||||
delete: "/erp/depotHead/delete", |
||||
deleteBatch: "/erp/depotHead/deleteBatch", |
||||
batchSetStatusUrl: "/erp/depotHead/batchSetStatus" |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
}, |
||||
created() { |
||||
this.getDepotData() |
||||
this.initUser() |
||||
}, |
||||
methods: { |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
@import '~@assets/less/common.less' |
||||
</style> |
@ -0,0 +1,216 @@
|
||||
<template> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="24"> |
||||
<a-card :style="cardStyle" :bordered="false"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="searchQuery"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据编号" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入单据编号" v-model="queryParam.number"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="商品信息" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入物料编码、名称、规格、型号" v-model="queryParam.materialParam"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据日期" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-range-picker |
||||
style="width:100%" |
||||
v-model="queryParam.createTimeRange" |
||||
format="YYYY-MM-DD" |
||||
:placeholder="['开始时间', '结束时间']" |
||||
@change="onDateChange" |
||||
@ok="onDateOk" |
||||
/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<template v-if="toggleSearchStatus"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="仓库名称" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="请选择仓库" showSearch optionFilterProp="children" v-model="queryParam.depotId"> |
||||
<a-select-option v-for="(depot,index) in depotList" :key="index" :value="depot.id"> |
||||
{{ depot.depotName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="操作员" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择操作员" showSearch optionFilterProp="children" v-model="queryParam.creator"> |
||||
<a-select-option v-for="(item,index) in userList" :key="index" :value="item.id"> |
||||
{{ item.userName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
</template> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-button type="primary" @click="searchQuery">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset">重置</a-button> |
||||
<a @click="handleToggleSearch" style="margin-left: 8px"> |
||||
{{ toggleSearchStatus ? '收起' : '展开' }} |
||||
<a-icon :type="toggleSearchStatus ? 'up' : 'down'"/> |
||||
</a> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
</div> |
||||
<!-- 操作按钮区域 --> |
||||
<div class="table-operator" style="margin-top: 5px"> |
||||
<a-button v-if="btnEnableList.indexOf(1)>-1" @click="myHandleAdd" type="primary" icon="plus">新增</a-button> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" v-if="btnEnableList.indexOf(1)>-1" @click="batchDel"><a-icon type="delete"/>删除</a-menu-item> |
||||
<a-menu-item key="2" v-if="btnEnableList.indexOf(2)>-1" @click="batchSetStatus(1)"><a-icon type="check"/>审核</a-menu-item> |
||||
<a-menu-item key="3" v-if="btnEnableList.indexOf(7)>-1" @click="batchSetStatus(0)"><a-icon type="stop"/>反审核</a-menu-item> |
||||
</a-menu> |
||||
<a-button> |
||||
批量操作 <a-icon type="down" /> |
||||
</a-button> |
||||
</a-dropdown> |
||||
<a-tooltip placement="left" title="用于将一个商品拆分成多种商品,被拆分的商品库存增加,拆分后的商品库存减少。" slot="action"> |
||||
<a-icon v-if="btnEnableList.indexOf(1)>-1" type="question-circle" style="font-size:20px;float:right;" /> |
||||
</a-tooltip> |
||||
</div> |
||||
<!-- table区域-begin --> |
||||
<a-row> |
||||
<a-col :md="2.5" :sm="24"> |
||||
<list-columns-setter v-model="columns" :def-columns="columns" style="float: right;"/> |
||||
</a-col> |
||||
</a-row> |
||||
<div> |
||||
<a-table |
||||
ref="table" |
||||
size="middle" |
||||
bordered |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:components="drag(columns)" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:scroll="scroll" |
||||
:loading="loading" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange}" |
||||
@change="handleTableChange"> |
||||
<span slot="action" slot-scope="text, record"> |
||||
<a @click="myHandleDetail(record, '拆卸单')">查看</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleEdit(record)">编辑</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleCopyAdd(record)">复制</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a-popconfirm v-if="btnEnableList.indexOf(1)>-1" title="确定删除吗?" @confirm="() => myHandleDelete(record)"> |
||||
<a>删除</a> |
||||
</a-popconfirm> |
||||
</span> |
||||
<template slot="customRenderStatus" slot-scope="status"> |
||||
<a-tag v-if="status == '0'" color="red">未审核</a-tag> |
||||
<a-tag v-if="status == '1'" color="green">已审核</a-tag> |
||||
</template> |
||||
</a-table> |
||||
</div> |
||||
<!-- table区域-end --> |
||||
<!-- 表单区域 --> |
||||
<disassemble-modal ref="modalForm" @ok="modalFormOk"></disassemble-modal> |
||||
<bill-detail ref="modalDetail"></bill-detail> |
||||
</a-card> |
||||
</a-col> |
||||
</a-row> |
||||
</template> |
||||
<!--power by jisheng hua--> |
||||
<script> |
||||
import DisassembleModal from './modules/DisassembleModal' |
||||
import BillDetail from './dialog/BillDetail' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
import { BillListMixin } from './mixins/BillListMixin' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
import ListColumnsSetter from '@/components/ListColumnsSetter' |
||||
import tableDragResizeMixin from '@/mixins/tableDragResizeMixin' |
||||
export default { |
||||
name: "DisassembleList", |
||||
mixins:[JeecgListMixin,BillListMixin,tableDragResizeMixin], |
||||
components: { |
||||
DisassembleModal, |
||||
BillDetail, |
||||
JDate, |
||||
ListColumnsSetter |
||||
}, |
||||
data () { |
||||
return { |
||||
// 查询条件 |
||||
queryParam: { |
||||
number: "", |
||||
materialParam: "", |
||||
type: "其它", |
||||
subType: "拆卸单", |
||||
roleType: Vue.ls.get('roleType'), |
||||
depotId: "", |
||||
creator: "" |
||||
}, |
||||
labelCol: { |
||||
span: 5 |
||||
}, |
||||
wrapperCol: { |
||||
span: 18, |
||||
offset: 1 |
||||
}, |
||||
// 表头 |
||||
columns: [ |
||||
{ title: '单据编号', dataIndex: 'number',width:160, |
||||
customRender:function (text,record,index) { |
||||
if(record.linkNumber) { |
||||
return text + "[转]"; |
||||
} else { |
||||
return text; |
||||
} |
||||
} |
||||
}, |
||||
{ title: '商品信息', dataIndex: 'materialsList',width:220, ellipsis:true, |
||||
customRender:function (text,record,index) { |
||||
if(text) { |
||||
return text.replace(",",","); |
||||
} |
||||
} |
||||
}, |
||||
{ title: '单据日期', dataIndex: 'operTimeStr',width:145}, |
||||
{ title: '操作员', dataIndex: 'userName',width:80, ellipsis:true}, |
||||
{ title: '金额合计', dataIndex: 'totalPrice',width:80}, |
||||
{ title: '状态', dataIndex: 'status', width: 80, align: "center", |
||||
scopedSlots: { customRender: 'customRenderStatus' } |
||||
}, |
||||
{ |
||||
title: '操作', |
||||
dataIndex: 'action', |
||||
align:"center", width: 150, |
||||
scopedSlots: { customRender: 'action' }, |
||||
} |
||||
], |
||||
url: { |
||||
list: "/erp/depotHead/list", |
||||
delete: "/erp/depotHead/delete", |
||||
deleteBatch: "/erp/depotHead/deleteBatch", |
||||
batchSetStatusUrl: "/erp/depotHead/batchSetStatus" |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
}, |
||||
created() { |
||||
this.getDepotData() |
||||
this.initUser() |
||||
}, |
||||
methods: { |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
@import '~@assets/less/common.less' |
||||
</style> |
@ -0,0 +1,224 @@
|
||||
<template> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="24"> |
||||
<a-card :style="cardStyle" :bordered="false"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="searchQuery"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据编号" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入单据编号" v-model="queryParam.number"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="商品信息" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入物料编码、名称、规格、型号" v-model="queryParam.materialParam"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据日期" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-range-picker |
||||
style="width:100%" |
||||
v-model="queryParam.createTimeRange" |
||||
format="YYYY-MM-DD" |
||||
:placeholder="['开始时间', '结束时间']" |
||||
@change="onDateChange" |
||||
@ok="onDateOk" |
||||
/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<template v-if="toggleSearchStatus"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="仓库名称" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="请选择仓库" showSearch optionFilterProp="children" v-model="queryParam.depotId"> |
||||
<a-select-option v-for="(depot,index) in depotList" :key="index" :value="depot.id"> |
||||
{{ depot.depotName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="操作员" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择操作员" showSearch optionFilterProp="children" v-model="queryParam.creator"> |
||||
<a-select-option v-for="(item,index) in userList" :key="index" :value="item.id"> |
||||
{{ item.userName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="关联单据" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入关联单据" v-model="queryParam.linkNumber"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
</template> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-button type="primary" @click="searchQuery">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset">重置</a-button> |
||||
<a @click="handleToggleSearch" style="margin-left: 8px"> |
||||
{{ toggleSearchStatus ? '收起' : '展开' }} |
||||
<a-icon :type="toggleSearchStatus ? 'up' : 'down'"/> |
||||
</a> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
</div> |
||||
<!-- 操作按钮区域 --> |
||||
<div class="table-operator" style="margin-top: 5px"> |
||||
<a-button v-if="btnEnableList.indexOf(1)>-1" @click="myHandleAdd" type="primary" icon="plus">新增</a-button> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" v-if="btnEnableList.indexOf(1)>-1" @click="batchDel"><a-icon type="delete"/>删除</a-menu-item> |
||||
<a-menu-item key="2" v-if="btnEnableList.indexOf(2)>-1" @click="batchSetStatus(1)"><a-icon type="check"/>审核</a-menu-item> |
||||
<a-menu-item key="3" v-if="btnEnableList.indexOf(7)>-1" @click="batchSetStatus(0)"><a-icon type="stop"/>反审核</a-menu-item> |
||||
</a-menu> |
||||
<a-button> |
||||
批量操作 <a-icon type="down" /> |
||||
</a-button> |
||||
</a-dropdown> |
||||
<a-tooltip placement="left" title="可以进行库存初始化,生产管理模块的成品入库。" slot="action"> |
||||
<a-icon v-if="btnEnableList.indexOf(1)>-1" type="question-circle" style="font-size:20px;float:right;" /> |
||||
</a-tooltip> |
||||
</div> |
||||
<!-- table区域-begin --> |
||||
<a-row> |
||||
<a-col :md="2.5" :sm="24"> |
||||
<list-columns-setter v-model="columns" :def-columns="columns" style="float: right;"/> |
||||
</a-col> |
||||
</a-row> |
||||
<div> |
||||
<a-table |
||||
ref="table" |
||||
size="middle" |
||||
bordered |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:components="drag(columns)" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:scroll="scroll" |
||||
:loading="loading" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange}" |
||||
@change="handleTableChange"> |
||||
<span slot="action" slot-scope="text, record"> |
||||
<a @click="myHandleDetail(record, '其它入库')">查看</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleEdit(record)">编辑</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleCopyAdd(record)">复制</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a-popconfirm v-if="btnEnableList.indexOf(1)>-1" title="确定删除吗?" @confirm="() => myHandleDelete(record)"> |
||||
<a>删除</a> |
||||
</a-popconfirm> |
||||
</span> |
||||
<template slot="customRenderStatus" slot-scope="status"> |
||||
<a-tag v-if="status == '0'" color="red">未审核</a-tag> |
||||
<a-tag v-if="status == '1'" color="green">已审核</a-tag> |
||||
</template> |
||||
</a-table> |
||||
</div> |
||||
<!-- table区域-end --> |
||||
<!-- 表单区域 --> |
||||
<inventory-review-modal ref="modalForm" @ok="modalFormOk"></inventory-review-modal> |
||||
<bill-detail ref="modalDetail"></bill-detail> |
||||
</a-card> |
||||
</a-col> |
||||
</a-row> |
||||
</template> |
||||
<!--power by ji shenghua--> |
||||
<script> |
||||
import InventoryReviewModal from './modules/InventoryReviewModal.vue' |
||||
import BillDetail from './dialog/BillDetail' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
import { BillListMixin } from './mixins/BillListMixin' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
import ListColumnsSetter from '@/components/ListColumnsSetter' |
||||
import tableDragResizeMixin from '@/mixins/tableDragResizeMixin' |
||||
export default { |
||||
name: "OtherInList", |
||||
mixins:[JeecgListMixin,BillListMixin,tableDragResizeMixin], |
||||
components: { |
||||
InventoryReviewModal, |
||||
BillDetail, |
||||
JDate, |
||||
ListColumnsSetter |
||||
}, |
||||
data () { |
||||
return { |
||||
// 查询条件 |
||||
queryParam: { |
||||
number: "", |
||||
materialParam: "", |
||||
type: "其它", |
||||
subType: "盘点复盘", |
||||
roleType: Vue.ls.get('roleType'), |
||||
organId: "", |
||||
depotId: "", |
||||
creator: "", |
||||
linkNumber: "" |
||||
}, |
||||
labelCol: { |
||||
span: 5 |
||||
}, |
||||
wrapperCol: { |
||||
span: 18, |
||||
offset: 1 |
||||
}, |
||||
// 表头 |
||||
columns: [ |
||||
{ title: '单据编号', dataIndex: 'number',width:160, align:'center', |
||||
customRender:function (text,record,index) { |
||||
if(record.linkNumber) { |
||||
return text + "[转]"; |
||||
} else { |
||||
return text; |
||||
} |
||||
} |
||||
}, |
||||
{ title: '商品信息', dataIndex: 'materialsList',width:220, align:'center', ellipsis:true, |
||||
customRender:function (text,record,index) { |
||||
if(text) { |
||||
return text.replace(",",","); |
||||
} |
||||
} |
||||
}, |
||||
{ title: '单据日期', dataIndex: 'operTimeStr',width:145, align:'center',}, |
||||
{ title: '操作员', dataIndex: 'userName',width:80, align:'center', ellipsis:true}, |
||||
{ title: '金额合计', dataIndex: 'totalPrice',width:80, align:'center',}, |
||||
{ title: '状态', dataIndex: 'status', width: 80, align: "center", |
||||
scopedSlots: { customRender: 'customRenderStatus' } |
||||
}, |
||||
{ |
||||
title: '操作', |
||||
dataIndex: 'action', |
||||
align:"center", width: 150, |
||||
scopedSlots: { customRender: 'action' }, |
||||
} |
||||
], |
||||
url: { |
||||
list: "/erp/depotHead/list", |
||||
delete: "/erp/depotHead/delete", |
||||
deleteBatch: "/erp/depotHead/deleteBatch", |
||||
batchSetStatusUrl: "/erp/depotHead/batchSetStatus" |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
}, |
||||
created () { |
||||
this.initSupplier() |
||||
this.getDepotData() |
||||
this.initUser() |
||||
}, |
||||
methods: { |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
@import '~@assets/less/common.less' |
||||
</style> |
@ -0,0 +1,234 @@
|
||||
<template> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="24"> |
||||
<a-card :style="cardStyle" :bordered="false"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="searchQuery"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据编号" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入单据编号" v-model="queryParam.number"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="商品信息" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入物料编码、名称、规格、型号" v-model="queryParam.materialParam"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据日期" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-range-picker |
||||
style="width:100%" |
||||
v-model="queryParam.createTimeRange" |
||||
format="YYYY-MM-DD" |
||||
:placeholder="['开始时间', '结束时间']" |
||||
@change="onDateChange" |
||||
@ok="onDateOk" |
||||
/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<template v-if="toggleSearchStatus"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="供应商" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择供应商" showSearch optionFilterProp="children" v-model="queryParam.organId"> |
||||
<a-select-option v-for="(item,index) in supList" :key="index" :value="item.id"> |
||||
{{ item.supplier }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="仓库名称" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="请选择仓库" showSearch optionFilterProp="children" v-model="queryParam.depotId"> |
||||
<a-select-option v-for="(depot,index) in depotList" :key="index" :value="depot.id"> |
||||
{{ depot.depotName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="操作员" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择操作员" showSearch optionFilterProp="children" v-model="queryParam.creator"> |
||||
<a-select-option v-for="(item,index) in userList" :key="index" :value="item.id"> |
||||
{{ item.userName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="关联单据" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入关联单据" v-model="queryParam.linkNumber"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
</template> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-button type="primary" @click="searchQuery">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset">重置</a-button> |
||||
<a @click="handleToggleSearch" style="margin-left: 8px"> |
||||
{{ toggleSearchStatus ? '收起' : '展开' }} |
||||
<a-icon :type="toggleSearchStatus ? 'up' : 'down'"/> |
||||
</a> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
</div> |
||||
<!-- 操作按钮区域 --> |
||||
<div class="table-operator" style="margin-top: 5px"> |
||||
<a-button v-if="btnEnableList.indexOf(1)>-1" @click="myHandleAdd" type="primary" icon="plus">新增</a-button> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" v-if="btnEnableList.indexOf(1)>-1" @click="batchDel"><a-icon type="delete"/>删除</a-menu-item> |
||||
<a-menu-item key="2" v-if="btnEnableList.indexOf(2)>-1" @click="batchSetStatus(1)"><a-icon type="check"/>审核</a-menu-item> |
||||
<a-menu-item key="3" v-if="btnEnableList.indexOf(7)>-1" @click="batchSetStatus(0)"><a-icon type="stop"/>反审核</a-menu-item> |
||||
</a-menu> |
||||
<a-button> |
||||
批量操作 <a-icon type="down" /> |
||||
</a-button> |
||||
</a-dropdown> |
||||
<a-tooltip placement="left" title="可以进行库存初始化,生产管理模块的成品入库。" slot="action"> |
||||
<a-icon v-if="btnEnableList.indexOf(1)>-1" type="question-circle" style="font-size:20px;float:right;" /> |
||||
</a-tooltip> |
||||
</div> |
||||
<!-- table区域-begin --> |
||||
<a-row> |
||||
<a-col :md="2.5" :sm="24"> |
||||
<list-columns-setter v-model="columns" :def-columns="columns" style="float: right;"/> |
||||
</a-col> |
||||
</a-row> |
||||
<div> |
||||
<a-table |
||||
ref="table" |
||||
size="middle" |
||||
bordered |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:components="drag(columns)" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:scroll="scroll" |
||||
:loading="loading" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange}" |
||||
@change="handleTableChange"> |
||||
<span slot="action" slot-scope="text, record"> |
||||
<a @click="myHandleDetail(record, '其它入库')">查看</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleEdit(record)">编辑</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleCopyAdd(record)">复制</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a-popconfirm v-if="btnEnableList.indexOf(1)>-1" title="确定删除吗?" @confirm="() => myHandleDelete(record)"> |
||||
<a>删除</a> |
||||
</a-popconfirm> |
||||
</span> |
||||
<template slot="customRenderStatus" slot-scope="status"> |
||||
<a-tag v-if="status == '0'" color="red">未审核</a-tag> |
||||
<a-tag v-if="status == '1'" color="green">已审核</a-tag> |
||||
</template> |
||||
</a-table> |
||||
</div> |
||||
<!-- table区域-end --> |
||||
<!-- 表单区域 --> |
||||
<other-in-modal ref="modalForm" @ok="modalFormOk"></other-in-modal> |
||||
<bill-detail ref="modalDetail"></bill-detail> |
||||
</a-card> |
||||
</a-col> |
||||
</a-row> |
||||
</template> |
||||
<!--power by ji shenghua--> |
||||
<script> |
||||
import OtherInModal from './modules/OtherInModal' |
||||
import BillDetail from './dialog/BillDetail' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
import { BillListMixin } from './mixins/BillListMixin' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
import ListColumnsSetter from '@/components/ListColumnsSetter' |
||||
import tableDragResizeMixin from '@/mixins/tableDragResizeMixin' |
||||
export default { |
||||
name: "OtherInList", |
||||
mixins:[JeecgListMixin,BillListMixin,tableDragResizeMixin], |
||||
components: { |
||||
OtherInModal, |
||||
BillDetail, |
||||
JDate, |
||||
ListColumnsSetter |
||||
}, |
||||
data () { |
||||
return { |
||||
// 查询条件 |
||||
queryParam: { |
||||
number: "", |
||||
materialParam: "", |
||||
type: "入库", |
||||
subType: "其它", |
||||
roleType: Vue.ls.get('roleType'), |
||||
organId: "", |
||||
depotId: "", |
||||
creator: "", |
||||
linkNumber: "" |
||||
}, |
||||
labelCol: { |
||||
span: 5 |
||||
}, |
||||
wrapperCol: { |
||||
span: 18, |
||||
offset: 1 |
||||
}, |
||||
// 表头 |
||||
columns: [ |
||||
{ title: '供应商', dataIndex: 'organName',width:120, align:'center', ellipsis:true}, |
||||
{ title: '单据编号', dataIndex: 'number',width:160, align:'center', |
||||
customRender:function (text,record,index) { |
||||
if(record.linkNumber) { |
||||
return text + "[转]"; |
||||
} else { |
||||
return text; |
||||
} |
||||
} |
||||
}, |
||||
{ title: '商品信息', dataIndex: 'materialsList',width:220, align:'center', ellipsis:true, |
||||
customRender:function (text,record,index) { |
||||
if(text) { |
||||
return text.replace(",",","); |
||||
} |
||||
} |
||||
}, |
||||
{ title: '单据日期', dataIndex: 'operTimeStr',width:145, align:'center',}, |
||||
{ title: '操作员', dataIndex: 'userName',width:80, align:'center', ellipsis:true}, |
||||
{ title: '金额合计', dataIndex: 'totalPrice',width:80, align:'center',}, |
||||
{ title: '状态', dataIndex: 'status', width: 80, align: "center", |
||||
scopedSlots: { customRender: 'customRenderStatus' } |
||||
}, |
||||
{ |
||||
title: '操作', |
||||
dataIndex: 'action', |
||||
align:"center", width: 150, |
||||
scopedSlots: { customRender: 'action' }, |
||||
} |
||||
], |
||||
url: { |
||||
list: "/erp/depotHead/list", |
||||
delete: "/erp/depotHead/delete", |
||||
deleteBatch: "/erp/depotHead/deleteBatch", |
||||
batchSetStatusUrl: "/erp/depotHead/batchSetStatus" |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
}, |
||||
created () { |
||||
this.initSupplier() |
||||
this.getDepotData() |
||||
this.initUser() |
||||
}, |
||||
methods: { |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
@import '~@assets/less/common.less' |
||||
</style> |
@ -0,0 +1,234 @@
|
||||
<template> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="24"> |
||||
<a-card :style="cardStyle" :bordered="false"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="searchQuery"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据编号" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入单据编号" v-model="queryParam.number"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="商品信息" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入物料编码、名称、规格、型号" v-model="queryParam.materialParam"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据日期" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-range-picker |
||||
style="width:100%" |
||||
v-model="queryParam.createTimeRange" |
||||
format="YYYY-MM-DD" |
||||
:placeholder="['开始时间', '结束时间']" |
||||
@change="onDateChange" |
||||
@ok="onDateOk" |
||||
/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<template v-if="toggleSearchStatus"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="客户" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择客户" showSearch optionFilterProp="children" v-model="queryParam.organId"> |
||||
<a-select-option v-for="(item,index) in cusList" :key="index" :value="item.id"> |
||||
{{ item.supplier }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="仓库名称" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="请选择仓库" showSearch optionFilterProp="children" v-model="queryParam.depotId"> |
||||
<a-select-option v-for="(depot,index) in depotList" :key="index" :value="depot.id"> |
||||
{{ depot.depotName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="操作员" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择操作员" showSearch optionFilterProp="children" v-model="queryParam.creator"> |
||||
<a-select-option v-for="(item,index) in userList" :key="index" :value="item.id"> |
||||
{{ item.userName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="关联单据" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入关联单据" v-model="queryParam.linkNumber"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
</template> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-button type="primary" @click="searchQuery">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset">重置</a-button> |
||||
<a @click="handleToggleSearch" style="margin-left: 8px"> |
||||
{{ toggleSearchStatus ? '收起' : '展开' }} |
||||
<a-icon :type="toggleSearchStatus ? 'up' : 'down'"/> |
||||
</a> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
</div> |
||||
<!-- 操作按钮区域 --> |
||||
<div class="table-operator" style="margin-top: 5px"> |
||||
<a-button v-if="btnEnableList.indexOf(1)>-1" @click="myHandleAdd" type="primary" icon="plus">新增</a-button> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" v-if="btnEnableList.indexOf(1)>-1" @click="batchDel"><a-icon type="delete"/>删除</a-menu-item> |
||||
<a-menu-item key="2" v-if="btnEnableList.indexOf(2)>-1" @click="batchSetStatus(1)"><a-icon type="check"/>审核</a-menu-item> |
||||
<a-menu-item key="3" v-if="btnEnableList.indexOf(7)>-1" @click="batchSetStatus(0)"><a-icon type="stop"/>反审核</a-menu-item> |
||||
</a-menu> |
||||
<a-button> |
||||
批量操作 <a-icon type="down" /> |
||||
</a-button> |
||||
</a-dropdown> |
||||
<a-tooltip placement="left" title="可以进行库存初始化,生产管理模块的领料出库。" slot="action"> |
||||
<a-icon v-if="btnEnableList.indexOf(1)>-1" type="question-circle" style="font-size:20px;float:right;" /> |
||||
</a-tooltip> |
||||
</div> |
||||
<!-- table区域-begin --> |
||||
<a-row> |
||||
<a-col :md="2.5" :sm="24"> |
||||
<list-columns-setter v-model="columns" :def-columns="columns" style="float: right;"/> |
||||
</a-col> |
||||
</a-row> |
||||
<div> |
||||
<a-table |
||||
ref="table" |
||||
size="middle" |
||||
bordered |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:components="drag(columns)" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:scroll="scroll" |
||||
:loading="loading" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange}" |
||||
@change="handleTableChange"> |
||||
<span slot="action" slot-scope="text, record"> |
||||
<a @click="myHandleDetail(record, '其它出库')">查看</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleEdit(record)">编辑</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleCopyAdd(record)">复制</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a-popconfirm v-if="btnEnableList.indexOf(1)>-1" title="确定删除吗?" @confirm="() => myHandleDelete(record)"> |
||||
<a>删除</a> |
||||
</a-popconfirm> |
||||
</span> |
||||
<template slot="customRenderStatus" slot-scope="status"> |
||||
<a-tag v-if="status == '0'" color="red">未审核</a-tag> |
||||
<a-tag v-if="status == '1'" color="green">已审核</a-tag> |
||||
</template> |
||||
</a-table> |
||||
</div> |
||||
<!-- table区域-end --> |
||||
<!-- 表单区域 --> |
||||
<other-out-modal ref="modalForm" @ok="modalFormOk"></other-out-modal> |
||||
<bill-detail ref="modalDetail"></bill-detail> |
||||
</a-card> |
||||
</a-col> |
||||
</a-row> |
||||
</template> |
||||
<!--power by j i s h e n g h u a--> |
||||
<script> |
||||
import OtherOutModal from './modules/OtherOutModal' |
||||
import BillDetail from './dialog/BillDetail' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
import { BillListMixin } from './mixins/BillListMixin' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
import ListColumnsSetter from '@/components/ListColumnsSetter' |
||||
import tableDragResizeMixin from '@/mixins/tableDragResizeMixin' |
||||
export default { |
||||
name: "OtherOutList", |
||||
mixins:[JeecgListMixin,BillListMixin,tableDragResizeMixin], |
||||
components: { |
||||
OtherOutModal, |
||||
BillDetail, |
||||
JDate, |
||||
ListColumnsSetter |
||||
}, |
||||
data () { |
||||
return { |
||||
// 查询条件 |
||||
queryParam: { |
||||
number: "", |
||||
materialParam: "", |
||||
type: "出库", |
||||
subType: "其它", |
||||
roleType: Vue.ls.get('roleType'), |
||||
organId: "", |
||||
depotId: "", |
||||
creator: "", |
||||
linkNumber: "" |
||||
}, |
||||
labelCol: { |
||||
span: 5 |
||||
}, |
||||
wrapperCol: { |
||||
span: 18, |
||||
offset: 1 |
||||
}, |
||||
// 表头 |
||||
columns: [ |
||||
{ title: '客户', dataIndex: 'organName',width:120, align:'center', ellipsis:true}, |
||||
{ title: '单据编号', dataIndex: 'number',width:160, align:'center', |
||||
customRender:function (text,record,index) { |
||||
if(record.linkNumber) { |
||||
return text + "[转]"; |
||||
} else { |
||||
return text; |
||||
} |
||||
} |
||||
}, |
||||
{ title: '商品信息', dataIndex: 'materialsList',width:220, align:'center', ellipsis:true, |
||||
customRender:function (text,record,index) { |
||||
if(text) { |
||||
return text.replace(",",","); |
||||
} |
||||
} |
||||
}, |
||||
{ title: '单据日期', dataIndex: 'operTimeStr',width:145, align:'center',}, |
||||
{ title: '操作员', dataIndex: 'userName',width:80, align:'center', ellipsis:true}, |
||||
{ title: '金额合计', dataIndex: 'totalPrice',width:80, align:'center',}, |
||||
{ title: '状态', dataIndex: 'status', width: 80, align:'center', align: "center", |
||||
scopedSlots: { customRender: 'customRenderStatus' } |
||||
}, |
||||
{ |
||||
title: '操作', |
||||
dataIndex: 'action', |
||||
align:"center", width: 150, |
||||
scopedSlots: { customRender: 'action' }, |
||||
} |
||||
], |
||||
url: { |
||||
list: "/erp/depotHead/list", |
||||
delete: "/erp/depotHead/delete", |
||||
deleteBatch: "/erp/depotHead/deleteBatch", |
||||
batchSetStatusUrl: "/erp/depotHead/batchSetStatus" |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
}, |
||||
created() { |
||||
this.initCustomer() |
||||
this.getDepotData() |
||||
this.initUser() |
||||
}, |
||||
methods: { |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
@import '~@assets/less/common.less' |
||||
</style> |
@ -0,0 +1,246 @@
|
||||
<template> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="24"> |
||||
<a-card :style="cardStyle" :bordered="false"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="searchQuery"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据编号" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入单据编号" v-model="queryParam.number"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="商品信息" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入物料编码、名称、规格、型号" v-model="queryParam.materialParam"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据日期" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-range-picker |
||||
style="width:100%" |
||||
v-model="queryParam.createTimeRange" |
||||
format="YYYY-MM-DD" |
||||
:placeholder="['开始时间', '结束时间']" |
||||
@change="onDateChange" |
||||
@ok="onDateOk" |
||||
/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<template v-if="toggleSearchStatus"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="供应商" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择供应商" showSearch optionFilterProp="children" v-model="queryParam.organId"> |
||||
<a-select-option v-for="(item,index) in supList" :key="index" :value="item.id"> |
||||
{{ item.supplier }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="仓库名称" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="请选择仓库" showSearch optionFilterProp="children" v-model="queryParam.depotId"> |
||||
<a-select-option v-for="(depot,index) in depotList" :key="index" :value="depot.id"> |
||||
{{ depot.depotName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="操作员" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择操作员" showSearch optionFilterProp="children" v-model="queryParam.creator"> |
||||
<a-select-option v-for="(item,index) in userList" :key="index" :value="item.id"> |
||||
{{ item.userName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="关联单据" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入关联单据" v-model="queryParam.linkNumber"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
</template> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-button type="primary" @click="searchQuery">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset">重置</a-button> |
||||
<a @click="handleToggleSearch" style="margin-left: 8px"> |
||||
{{ toggleSearchStatus ? '收起' : '展开' }} |
||||
<a-icon :type="toggleSearchStatus ? 'up' : 'down'"/> |
||||
</a> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
</div> |
||||
<!-- 操作按钮区域 --> |
||||
<div class="table-operator" style="margin-top: 5px"> |
||||
<a-button v-if="btnEnableList.indexOf(1)>-1" @click="myHandleAdd" type="primary" icon="plus">新增</a-button> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" v-if="btnEnableList.indexOf(1)>-1" @click="batchDel"><a-icon type="delete"/>删除</a-menu-item> |
||||
<a-menu-item key="2" v-if="btnEnableList.indexOf(2)>-1" @click="batchSetStatus(1)"><a-icon type="check"/>审核</a-menu-item> |
||||
<a-menu-item key="3" v-if="btnEnableList.indexOf(7)>-1" @click="batchSetStatus(0)"><a-icon type="stop"/>反审核</a-menu-item> |
||||
</a-menu> |
||||
<a-button> |
||||
批量操作 <a-icon type="down" /> |
||||
</a-button> |
||||
</a-dropdown> |
||||
<a-tooltip placement="left" title="用于采购入库单据的退货。采购退货单可以由采购出库单转过来,也可以单独创建。" slot="action"> |
||||
<a-icon v-if="btnEnableList.indexOf(1)>-1" type="question-circle" style="font-size:20px;float:right;" /> |
||||
</a-tooltip> |
||||
</div> |
||||
<!-- table区域-begin --> |
||||
<a-row> |
||||
<a-col :md="2.5" :sm="24"> |
||||
<list-columns-setter v-model="columns" :def-columns="columns" style="float: right;"/> |
||||
</a-col> |
||||
</a-row> |
||||
<div> |
||||
<a-table |
||||
ref="table" |
||||
size="middle" |
||||
bordered |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:components="drag(columns)" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:scroll="scroll" |
||||
:loading="loading" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange}" |
||||
@change="handleTableChange"> |
||||
<span slot="action" slot-scope="text, record"> |
||||
<a @click="myHandleDetail(record, '采购退货出库')">查看</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleEdit(record)">编辑</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleCopyAdd(record)">复制</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a-popconfirm v-if="btnEnableList.indexOf(1)>-1" title="确定删除吗?" @confirm="() => myHandleDelete(record)"> |
||||
<a>删除</a> |
||||
</a-popconfirm> |
||||
</span> |
||||
<template slot="customRenderStatus" slot-scope="status"> |
||||
<a-tag v-if="status == '0'" color="red">未审核</a-tag> |
||||
<a-tag v-if="status == '1'" color="green">已审核</a-tag> |
||||
</template> |
||||
</a-table> |
||||
</div> |
||||
<!-- table区域-end --> |
||||
<!-- 表单区域 --> |
||||
<purchase-back-modal ref="modalForm" @ok="modalFormOk"></purchase-back-modal> |
||||
<bill-detail ref="modalDetail"></bill-detail> |
||||
</a-card> |
||||
</a-col> |
||||
</a-row> |
||||
</template> |
||||
<!-- by jisheng hua--> |
||||
<script> |
||||
import PurchaseBackModal from './modules/PurchaseBackModal' |
||||
import BillDetail from './dialog/BillDetail' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
import { BillListMixin } from './mixins/BillListMixin' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
import ListColumnsSetter from '@/components/ListColumnsSetter' |
||||
import tableDragResizeMixin from '@/mixins/tableDragResizeMixin' |
||||
export default { |
||||
name: "PurchaseBackList", |
||||
mixins:[JeecgListMixin,BillListMixin,tableDragResizeMixin], |
||||
components: { |
||||
PurchaseBackModal, |
||||
BillDetail, |
||||
JDate, |
||||
ListColumnsSetter |
||||
}, |
||||
data () { |
||||
return { |
||||
// 查询条件 |
||||
queryParam: { |
||||
number: "", |
||||
materialParam: "", |
||||
type: "出库", |
||||
subType: "采购退货", |
||||
roleType: Vue.ls.get('roleType'), |
||||
organId: "", |
||||
depotId: "", |
||||
creator: "", |
||||
linkNumber: "" |
||||
}, |
||||
labelCol: { |
||||
span: 5 |
||||
}, |
||||
wrapperCol: { |
||||
span: 18, |
||||
offset: 1 |
||||
}, |
||||
// 表头 |
||||
columns: [ |
||||
{ title: '供应商', dataIndex: 'organName',width:120, ellipsis:true,align:'center',}, |
||||
{ title: '单据编号', dataIndex: 'number',width:160,align:'center', |
||||
customRender:function (text,record,index) { |
||||
if(record.linkNumber) { |
||||
return text + "[转]"; |
||||
} else { |
||||
return text; |
||||
} |
||||
} |
||||
}, |
||||
{ title: '商品信息', dataIndex: 'materialsList',width:220,align:'center', ellipsis:true, |
||||
customRender:function (text,record,index) { |
||||
if(text) { |
||||
return text.replace(",",","); |
||||
} |
||||
} |
||||
}, |
||||
{ title: '单据日期', dataIndex: 'operTimeStr',width:145,align:'center',}, |
||||
{ title: '操作员', dataIndex: 'userName',width:80,align:'center', ellipsis:true}, |
||||
{ title: '金额合计', dataIndex: 'totalPrice',width:80,align:'center',}, |
||||
{ title: '含税合计', dataIndex: 'totalTaxLastMoney',width:80,align:'center', |
||||
customRender:function (text,record,index) { |
||||
return (record.discountMoney + record.discountLastMoney).toFixed(2); |
||||
} |
||||
}, |
||||
{ title: '待退金额', dataIndex: 'needBackMoney',width:80,align:'center', |
||||
customRender:function (text,record,index) { |
||||
let needBackMoney = record.discountLastMoney + record.otherMoney |
||||
return needBackMoney? needBackMoney.toFixed(2):'' |
||||
} |
||||
}, |
||||
{ title: '退款', dataIndex: 'changeAmount',width:50,align:'center',}, |
||||
{ title: '状态', dataIndex: 'status', width: 80, align: "center", |
||||
scopedSlots: { customRender: 'customRenderStatus' } |
||||
}, |
||||
{ |
||||
title: '操作', |
||||
dataIndex: 'action', |
||||
align:"center", width: 160, |
||||
scopedSlots: { customRender: 'action' }, |
||||
} |
||||
], |
||||
url: { |
||||
list: "/erp/depotHead/list", |
||||
delete: "/erp/depotHead/delete", |
||||
deleteBatch: "/erp/depotHead/deleteBatch", |
||||
batchSetStatusUrl: "/erp/depotHead/batchSetStatus" |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
}, |
||||
created () { |
||||
this.initSupplier() |
||||
this.getDepotData() |
||||
this.initUser() |
||||
}, |
||||
methods: { |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
@import '~@assets/less/common.less' |
||||
</style> |
@ -0,0 +1,255 @@
|
||||
<template> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="24"> |
||||
<a-card :style="cardStyle" :bordered="false"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="searchQuery"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据编号" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入单据编号" v-model="queryParam.number"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="商品信息" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入物料编码、名称、规格、型号" v-model="queryParam.materialParam"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据日期" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-range-picker |
||||
style="width:100%" |
||||
v-model="queryParam.createTimeRange" |
||||
format="YYYY-MM-DD" |
||||
:placeholder="['开始时间', '结束时间']" |
||||
@change="onDateChange" |
||||
@ok="onDateOk" |
||||
/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<template v-if="toggleSearchStatus"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="供应商" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择供应商" showSearch optionFilterProp="children" v-model="queryParam.organId"> |
||||
<a-select-option v-for="(item,index) in supList" :key="index" :value="item.id"> |
||||
{{ item.supplier }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="仓库名称" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="请选择仓库" showSearch optionFilterProp="children" v-model="queryParam.depotId"> |
||||
<a-select-option v-for="(depot,index) in depotList" :key="index" :value="depot.id"> |
||||
{{ depot.depotName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="操作员" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择操作员" showSearch optionFilterProp="children" v-model="queryParam.creator"> |
||||
<a-select-option v-for="(item,index) in userList" :key="index" :value="item.id"> |
||||
{{ item.userName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="关联订单" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入关联订单" v-model="queryParam.linkNumber"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
</template> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-button type="primary" @click="searchQuery">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset">重置</a-button> |
||||
<a @click="handleToggleSearch" style="margin-left: 8px"> |
||||
{{ toggleSearchStatus ? '收起' : '展开' }} |
||||
<a-icon :type="toggleSearchStatus ? 'up' : 'down'"/> |
||||
</a> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
</div> |
||||
<!-- 操作按钮区域 --> |
||||
<div class="table-operator" style="margin-top: 5px"> |
||||
<a-button v-if="btnEnableList.indexOf(1)>-1" @click="myHandleAdd" type="primary" icon="plus">新增</a-button> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" v-if="btnEnableList.indexOf(1)>-1" @click="batchDel"><a-icon type="delete"/>删除</a-menu-item> |
||||
<a-menu-item key="2" v-if="btnEnableList.indexOf(2)>-1" @click="batchSetStatus(1)"><a-icon type="check"/>审核</a-menu-item> |
||||
<a-menu-item key="3" v-if="btnEnableList.indexOf(7)>-1" @click="batchSetStatus(0)"><a-icon type="stop"/>反审核</a-menu-item> |
||||
</a-menu> |
||||
<a-button> |
||||
批量操作 <a-icon type="down" /> |
||||
</a-button> |
||||
</a-dropdown> |
||||
<a-tooltip placement="left" title="采购入库单可以由采购订单转过来,也可以单独创建。 |
||||
采购入库单据中的仓库列表只显示当前用户有权限的仓库。采购入库单可以使用多账户付款。 |
||||
勾选单据之后可以进行批量操作(删除、审核、反审核)" slot="action"> |
||||
<a-icon v-if="btnEnableList.indexOf(1)>-1" type="question-circle" style="font-size:20px;float:right;" /> |
||||
</a-tooltip> |
||||
</div> |
||||
<!-- table区域-begin --> |
||||
<!-- table区域-begin --> |
||||
<a-row> |
||||
<a-col :md="2.5" :sm="24"> |
||||
<list-columns-setter v-model="columns" :def-columns="columns" style="float: right;"/> |
||||
</a-col> |
||||
</a-row> |
||||
<div> |
||||
<a-table |
||||
ref="table" |
||||
size="middle" |
||||
bordered |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:components="drag(columns)" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:scroll="scroll" |
||||
:loading="loading" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange}" |
||||
@change="handleTableChange"> |
||||
<span slot="action" slot-scope="text, record"> |
||||
<a @click="myHandleDetail(record, '采购入库')">查看</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleEdit(record)">编辑</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleCopyAdd(record)">复制</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a-popconfirm v-if="btnEnableList.indexOf(1)>-1" title="确定删除吗?" @confirm="() => myHandleDelete(record)"> |
||||
<a>删除</a> |
||||
</a-popconfirm> |
||||
</span> |
||||
<template slot="customRenderStatus" slot-scope="status"> |
||||
<a-tag v-if="status == '0'" color="red">未审核</a-tag> |
||||
<a-tag v-if="status == '1'" color="green">已审核</a-tag> |
||||
</template> |
||||
</a-table> |
||||
</div> |
||||
<!-- table区域-end --> |
||||
<!-- 表单区域 --> |
||||
<purchase-in-modal ref="modalForm" @ok="modalFormOk"></purchase-in-modal> |
||||
<bill-detail ref="modalDetail"></bill-detail> |
||||
</a-card> |
||||
</a-col> |
||||
</a-row> |
||||
</template> |
||||
<!-- by ji sheng hua--> |
||||
<script> |
||||
import PurchaseInModal from './modules/PurchaseInModal' |
||||
import BillDetail from './dialog/BillDetail' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
import { BillListMixin } from './mixins/BillListMixin' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
import ListColumnsSetter from '@/components/ListColumnsSetter' |
||||
import tableDragResizeMixin from '@/mixins/tableDragResizeMixin' |
||||
export default { |
||||
name: "PurchaseInList", |
||||
mixins:[JeecgListMixin,BillListMixin,tableDragResizeMixin], |
||||
components: { |
||||
PurchaseInModal, |
||||
BillDetail, |
||||
JDate, |
||||
ListColumnsSetter |
||||
}, |
||||
data () { |
||||
return { |
||||
// 查询条件 |
||||
queryParam: { |
||||
number: "", |
||||
materialParam: "", |
||||
type: "入库", |
||||
subType: "采购", |
||||
roleType: Vue.ls.get('roleType'), |
||||
organId: "", |
||||
depotId: "", |
||||
creator: "", |
||||
linkNumber: "" |
||||
}, |
||||
labelCol: { |
||||
span: 5 |
||||
}, |
||||
wrapperCol: { |
||||
span: 18, |
||||
offset: 1 |
||||
}, |
||||
// 表头 |
||||
columns: [ |
||||
{ title: '供应商', dataIndex: 'organName',width:120, ellipsis:true, align:'center',}, |
||||
{ title: '单据编号', dataIndex: 'number',width:160,align:'center', |
||||
customRender:function (text,record,index) { |
||||
if(record.linkNumber) { |
||||
return text + "[订]"; |
||||
} else { |
||||
return text; |
||||
} |
||||
} |
||||
}, |
||||
{ title: '商品信息', dataIndex: 'materialsList',width:220, ellipsis:true,align:'center', |
||||
customRender:function (text,record,index) { |
||||
if(text) { |
||||
return text.replace(",",","); |
||||
} |
||||
} |
||||
}, |
||||
{ title: '单据日期', dataIndex: 'operTimeStr',width:145,align:'center',}, |
||||
{ title: '操作员', dataIndex: 'userName',width:80, ellipsis:true,align:'center',}, |
||||
{ title: '金额合计', dataIndex: 'totalPrice',width:80,align:'center',}, |
||||
{ title: '含税合计', dataIndex: 'totalTaxLastMoney',width:80,align:'center', |
||||
customRender:function (text,record,index) { |
||||
return (record.discountMoney + record.discountLastMoney).toFixed(2); |
||||
} |
||||
}, |
||||
{ title: '待付金额', dataIndex: 'needInMoney',width:80,align:'center', |
||||
customRender:function (text,record,index) { |
||||
let needInMoney = record.discountLastMoney + record.otherMoney |
||||
return needInMoney? needInMoney.toFixed(2):'' |
||||
} |
||||
}, |
||||
{ title: '付款', dataIndex: 'changeAmount',width:60,align:'center',}, |
||||
{ title: '欠款', dataIndex: 'debt',width:60,align:'center', |
||||
customRender:function (text,record,index) { |
||||
let debt = record.discountLastMoney + record.otherMoney - record.changeAmount |
||||
return debt? debt.toFixed(2):'' |
||||
} |
||||
}, |
||||
{ title: '状态', dataIndex: 'status', width: 80, align: "center",align:'center', |
||||
scopedSlots: { customRender: 'customRenderStatus' } |
||||
}, |
||||
{ |
||||
title: '操作', |
||||
dataIndex: 'action', |
||||
align:"center", width: 180, |
||||
scopedSlots: { customRender: 'action' }, |
||||
} |
||||
], |
||||
url: { |
||||
list: "/erp/depotHead/list", |
||||
delete: "/erp/depotHead/delete", |
||||
deleteBatch: "/erp/depotHead/deleteBatch", |
||||
batchSetStatusUrl: "/erp/depotHead/batchSetStatus" |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
}, |
||||
created () { |
||||
this.initSupplier() |
||||
this.getDepotData() |
||||
this.initUser() |
||||
}, |
||||
methods: { |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
@import '~@assets/less/common.less' |
||||
</style> |
@ -0,0 +1,230 @@
|
||||
<template> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="24"> |
||||
<a-card :style="cardStyle" :bordered="false"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="searchQuery"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据编号" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入单据编号" v-model="queryParam.number"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="商品信息" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入物料编码、名称、规格、型号" v-model="queryParam.materialParam"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据日期" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-range-picker |
||||
style="width:100%" |
||||
v-model="queryParam.createTimeRange" |
||||
format="YYYY-MM-DD" |
||||
:placeholder="['开始时间', '结束时间']" |
||||
@change="onDateChange" |
||||
@ok="onDateOk" |
||||
/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<template v-if="toggleSearchStatus"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="供应商" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择供应商" showSearch optionFilterProp="children" v-model="queryParam.organId"> |
||||
<a-select-option v-for="(item,index) in supList" :key="index" :value="item.id"> |
||||
{{ item.supplier }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="操作员" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择操作员" showSearch optionFilterProp="children" v-model="queryParam.creator"> |
||||
<a-select-option v-for="(item,index) in userList" :key="index" :value="item.id"> |
||||
{{ item.userName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
</template> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-button type="primary" @click="searchQuery">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset">重置</a-button> |
||||
<a @click="handleToggleSearch" style="margin-left: 8px"> |
||||
{{ toggleSearchStatus ? '收起' : '展开' }} |
||||
<a-icon :type="toggleSearchStatus ? 'up' : 'down'"/> |
||||
</a> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
</div> |
||||
<!-- 操作按钮区域 --> |
||||
<div class="table-operator" style="margin-top: 5px"> |
||||
<a-button v-if="btnEnableList.indexOf(1)>-1" @click="myHandleAdd" type="primary" icon="plus">新增</a-button> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" v-if="btnEnableList.indexOf(1)>-1" @click="batchDel"><a-icon type="delete"/>删除</a-menu-item> |
||||
<a-menu-item key="2" v-if="btnEnableList.indexOf(2)>-1" @click="batchSetStatus(1)"><a-icon type="check"/>审核</a-menu-item> |
||||
<a-menu-item key="3" v-if="btnEnableList.indexOf(7)>-1" @click="batchSetStatus(0)"><a-icon type="stop"/>反审核</a-menu-item> |
||||
</a-menu> |
||||
<a-button> |
||||
批量操作 <a-icon type="down" /> |
||||
</a-button> |
||||
</a-dropdown> |
||||
<a-tooltip placement="left" title="采购订单不涉及付款金额,采购订单可以转采购入库单,但需要先对采购订单进行审核。 |
||||
勾选单据之后可以进行批量操作(删除、审核、反审核)" slot="action"> |
||||
<a-icon v-if="btnEnableList.indexOf(1)>-1" type="question-circle" style="font-size:20px;float:right;" /> |
||||
</a-tooltip> |
||||
</div> |
||||
<!-- table区域-begin --> |
||||
<a-row> |
||||
<a-col :md="2.5" :sm="24"> |
||||
<list-columns-setter v-model="columns" :def-columns="columns" style="float: right;"/> |
||||
</a-col> |
||||
</a-row> |
||||
<div> |
||||
<a-table |
||||
ref="table" |
||||
size="middle" |
||||
bordered |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:components="drag(columns)" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:scroll="scroll" |
||||
:loading="loading" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange}" |
||||
@change="handleTableChange"> |
||||
<span slot="action" slot-scope="text, record"> |
||||
<a @click="myHandleDetail(record, '采购订单')">查看</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleEdit(record)">编辑</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleCopyAdd(record)">复制</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a-popconfirm v-if="btnEnableList.indexOf(1)>-1" title="确定删除吗?" @confirm="() => myHandleDelete(record)"> |
||||
<a>删除</a> |
||||
</a-popconfirm> |
||||
</span> |
||||
<template slot="customRenderStatus" slot-scope="status"> |
||||
<a-tag v-if="status == '0'" color="red">未审核</a-tag> |
||||
<a-tag v-if="status == '1'" color="green">已审核</a-tag> |
||||
<a-tag v-if="status == '2'" color="cyan">完成采购</a-tag> |
||||
<a-tag v-if="status == '3'" color="blue">部分采购</a-tag> |
||||
</template> |
||||
</a-table> |
||||
</div> |
||||
<!-- table区域-end --> |
||||
<!-- 表单区域 --> |
||||
<purchase-order-modal ref="modalForm" @ok="modalFormOk"></purchase-order-modal> |
||||
<bill-detail ref="modalDetail"></bill-detail> |
||||
</a-card> |
||||
</a-col> |
||||
</a-row> |
||||
</template> |
||||
<!-- by ji sheng hua--> |
||||
<script> |
||||
import PurchaseOrderModal from './modules/PurchaseOrderModal' |
||||
import BillDetail from './dialog/BillDetail' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
import { BillListMixin } from './mixins/BillListMixin' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
import ListColumnsSetter from '@/components/ListColumnsSetter' |
||||
import tableDragResizeMixin from '@/mixins/tableDragResizeMixin' |
||||
export default { |
||||
name: "PurchaseOrderList", |
||||
mixins:[JeecgListMixin,BillListMixin,tableDragResizeMixin], |
||||
components: { |
||||
PurchaseOrderModal, |
||||
BillDetail, |
||||
JDate, |
||||
ListColumnsSetter |
||||
}, |
||||
data () { |
||||
return { |
||||
// 查询条件 |
||||
queryParam: { |
||||
number: "", |
||||
materialParam: "", |
||||
type: "其它", |
||||
subType: "采购订单", |
||||
roleType: Vue.ls.get('roleType'), |
||||
organId: "", |
||||
depotId: "", |
||||
creator: "" |
||||
}, |
||||
labelCol: { |
||||
span: 5 |
||||
}, |
||||
wrapperCol: { |
||||
span: 18, |
||||
offset: 1 |
||||
}, |
||||
// 表头 |
||||
columns: [ |
||||
{ title: '供应商', dataIndex: 'organName',width:120, align:'center', ellipsis:true}, |
||||
{ title: '单据编号', dataIndex: 'number',width:160, align:'center', |
||||
customRender:function (text,record,index) { |
||||
if(record.linkNumber) { |
||||
return text + "[转]"; |
||||
} else { |
||||
return text; |
||||
} |
||||
} |
||||
}, |
||||
{ title: '商品信息', dataIndex: 'materialsList',width:220, align:'center', ellipsis:true, |
||||
customRender:function (text,record,index) { |
||||
if(text) { |
||||
return text.replace(",",","); |
||||
} |
||||
} |
||||
}, |
||||
{ title: '单据日期', dataIndex: 'operTimeStr',width:145,align:'center',}, |
||||
{ title: '操作员', dataIndex: 'userName',width:80, ellipsis:true,align:'center',}, |
||||
{ title: '金额合计', dataIndex: 'totalPrice',width:80,align:'center',}, |
||||
{ title: '含税合计', dataIndex: 'totalTaxLastMoney',width:80,align:'center', |
||||
customRender:function (text,record,index) { |
||||
if(record.discountLastMoney) { |
||||
return (record.discountMoney + record.discountLastMoney).toFixed(2); |
||||
} else { |
||||
return record.totalPrice; |
||||
} |
||||
} |
||||
}, |
||||
{ title: '状态', dataIndex: 'status', width: 80, align: "center",align:'center', |
||||
scopedSlots: { customRender: 'customRenderStatus' } |
||||
}, |
||||
{ |
||||
title: '操作', |
||||
dataIndex: 'action', |
||||
align:"center", width: 150, |
||||
scopedSlots: { customRender: 'action' }, |
||||
} |
||||
], |
||||
url: { |
||||
list: "/erp/depotHead/list", |
||||
delete: "/erp/depotHead/delete", |
||||
deleteBatch: "/erp/depotHead/deleteBatch", |
||||
batchSetStatusUrl: "/erp/depotHead/batchSetStatus" |
||||
} |
||||
} |
||||
}, |
||||
created() { |
||||
this.initSupplier() |
||||
this.initUser() |
||||
}, |
||||
computed: { |
||||
}, |
||||
methods: { |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
@import '~@assets/less/common.less' |
||||
</style> |
@ -0,0 +1,244 @@
|
||||
<template> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="24"> |
||||
<a-card :style="cardStyle" :bordered="false"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="searchQuery"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据编号" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入单据编号" v-model="queryParam.number"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="商品信息" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入物料编码、名称、规格、型号" v-model="queryParam.materialParam"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据日期" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-range-picker |
||||
style="width:100%" |
||||
v-model="queryParam.createTimeRange" |
||||
format="YYYY-MM-DD" |
||||
:placeholder="['开始时间', '结束时间']" |
||||
@change="onDateChange" |
||||
@ok="onDateOk" |
||||
/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<template v-if="toggleSearchStatus"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="会员卡号" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择会员卡号" showSearch optionFilterProp="children" v-model="queryParam.organId"> |
||||
<a-select-option v-for="(item,index) in retailList" :key="index" :value="item.id"> |
||||
{{ item.supplier }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="仓库名称" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="请选择仓库" showSearch optionFilterProp="children" v-model="queryParam.depotId"> |
||||
<a-select-option v-for="(depot,index) in depotList" :key="index" :value="depot.id"> |
||||
{{ depot.depotName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="操作员" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择操作员" showSearch optionFilterProp="children" v-model="queryParam.creator"> |
||||
<a-select-option v-for="(item,index) in userList" :key="index" :value="item.id"> |
||||
{{ item.userName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="关联单据" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入关联单据" v-model="queryParam.linkNumber"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
</template> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-button type="primary" @click="searchQuery">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset">重置</a-button> |
||||
<a @click="handleToggleSearch" style="margin-left: 8px"> |
||||
{{ toggleSearchStatus ? '收起' : '展开' }} |
||||
<a-icon :type="toggleSearchStatus ? 'up' : 'down'"/> |
||||
</a> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
</div> |
||||
<!-- 操作按钮区域 --> |
||||
<div class="table-operator" style="margin-top: 5px"> |
||||
<a-button v-if="btnEnableList.indexOf(1)>-1" @click="myHandleAdd" type="primary" icon="plus">新增</a-button> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" v-if="btnEnableList.indexOf(1)>-1" @click="batchDel"><a-icon type="delete"/>删除</a-menu-item> |
||||
<a-menu-item key="2" v-if="btnEnableList.indexOf(2)>-1" @click="batchSetStatus(1)"><a-icon type="check"/>审核</a-menu-item> |
||||
<a-menu-item key="3" v-if="btnEnableList.indexOf(7)>-1" @click="batchSetStatus(0)"><a-icon type="stop"/>反审核</a-menu-item> |
||||
</a-menu> |
||||
<a-button> |
||||
批量操作 <a-icon type="down" /> |
||||
</a-button> |
||||
</a-dropdown> |
||||
<a-tooltip placement="left" title="用于零售出库单据的退货。零售退货单可以由零售出库单转过来,也可以单独创建。" slot="action"> |
||||
<a-icon v-if="btnEnableList.indexOf(1)>-1" type="question-circle" style="font-size:20px;float:right;" /> |
||||
</a-tooltip> |
||||
</div> |
||||
<!-- table区域-begin --> |
||||
<a-row> |
||||
<a-col :md="2.5" :sm="24"> |
||||
<list-columns-setter v-model="columns" :def-columns="columns" style="float: right;"/> |
||||
</a-col> |
||||
</a-row> |
||||
<div> |
||||
<a-table |
||||
ref="table" |
||||
size="middle" |
||||
bordered |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:components="drag(columns)" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:scroll="scroll" |
||||
:loading="loading" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange}" |
||||
@change="handleTableChange"> |
||||
<span slot="action" slot-scope="text, record"> |
||||
<a @click="myHandleDetail(record, '零售退货入库')">查看</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleEdit(record)">编辑</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleCopyAdd(record)">复制</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a-popconfirm v-if="btnEnableList.indexOf(1)>-1" title="确定删除吗?" @confirm="() => myHandleDelete(record)"> |
||||
<a>删除</a> |
||||
</a-popconfirm> |
||||
</span> |
||||
<template slot="customRenderStatus" slot-scope="status"> |
||||
<a-tag v-if="status == '0'" color="red">未审核</a-tag> |
||||
<a-tag v-if="status == '1'" color="green">已审核</a-tag> |
||||
</template> |
||||
</a-table> |
||||
</div> |
||||
<!-- table区域-end --> |
||||
<!-- 表单区域 --> |
||||
<retail-back-modal ref="modalForm" @ok="modalFormOk"></retail-back-modal> |
||||
<bill-detail ref="modalDetail"></bill-detail> |
||||
</a-card> |
||||
</a-col> |
||||
</a-row> |
||||
</template> |
||||
<!-- by jisheng h u a--> |
||||
<script> |
||||
import RetailBackModal from './modules/RetailBackModal' |
||||
import BillDetail from './dialog/BillDetail' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
import { BillListMixin } from './mixins/BillListMixin' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
import ListColumnsSetter from '@/components/ListColumnsSetter' |
||||
import tableDragResizeMixin from '@/mixins/tableDragResizeMixin' |
||||
export default { |
||||
name: "RetailBackList", |
||||
mixins:[JeecgListMixin,BillListMixin,tableDragResizeMixin], |
||||
components: { |
||||
RetailBackModal, |
||||
BillDetail, |
||||
JDate, |
||||
ListColumnsSetter |
||||
}, |
||||
data () { |
||||
return { |
||||
// 查询条件 |
||||
queryParam: { |
||||
number: "", |
||||
materialParam: "", |
||||
type: "入库", |
||||
subType: "零售退货", |
||||
roleType: Vue.ls.get('roleType'), |
||||
organId: "", |
||||
depotId: "", |
||||
creator: "", |
||||
linkNumber: "" |
||||
}, |
||||
labelCol: { |
||||
span: 5 |
||||
}, |
||||
wrapperCol: { |
||||
span: 18, |
||||
offset: 1 |
||||
}, |
||||
// 表头 |
||||
columns: [ |
||||
{ title: '会员', dataIndex: 'organName',width:120, ellipsis:true, align:'center'}, |
||||
{ title: '单据编号', dataIndex: 'number',width:160, align:'center', |
||||
customRender:function (text,record,index) { |
||||
if(record.linkNumber) { |
||||
return text + "[转]"; |
||||
} else { |
||||
return text; |
||||
} |
||||
} |
||||
}, |
||||
{ title: '商品信息', dataIndex: 'materialsList',width:220, ellipsis:true, |
||||
customRender:function (text,record,index) { |
||||
if(text) { |
||||
return text.replace(",",","); |
||||
} |
||||
}, align:'center' |
||||
}, |
||||
{ title: '单据日期', dataIndex: 'operTimeStr',width:145, align:'center'}, |
||||
{ title: '操作员', dataIndex: 'userName',width:80, ellipsis:true, align:'center'}, |
||||
{ title: '金额合计', dataIndex: 'totalPrice',width:80, align:'center'}, |
||||
{ title: '付款金额', dataIndex: 'getAmount',width:80, align:'center', |
||||
customRender:function (text,record,index) { |
||||
if(record.backAmount) { |
||||
return record.changeAmount + record.backAmount |
||||
} else { |
||||
return record.changeAmount |
||||
} |
||||
} |
||||
}, |
||||
{ title: '找零', dataIndex: 'backAmount',width:50, align:'center'}, |
||||
{ title: '状态', dataIndex: 'status', width: 80, align: "center", |
||||
scopedSlots: { customRender: 'customRenderStatus' } |
||||
}, |
||||
{ |
||||
title: '操作', |
||||
dataIndex: 'action', |
||||
align:"center", width: 150, |
||||
scopedSlots: { customRender: 'action' }, |
||||
} |
||||
], |
||||
url: { |
||||
list: "/erp/depotHead/list", |
||||
delete: "/erp/depotHead/delete", |
||||
deleteBatch: "/erp/depotHead/deleteBatch", |
||||
batchSetStatusUrl: "/erp/depotHead/batchSetStatus" |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
}, |
||||
created () { |
||||
this.initRetail() |
||||
this.getDepotData() |
||||
this.initUser() |
||||
}, |
||||
methods: { |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
@import '~@assets/less/common.less' |
||||
</style> |
@ -0,0 +1,238 @@
|
||||
<template> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="24"> |
||||
<a-card :style="cardStyle" :bordered="false"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="searchQuery"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据编号" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入单据编号" v-model="queryParam.number"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="商品信息" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入物料编码、名称、规格、型号" v-model="queryParam.materialParam"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据日期" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-range-picker |
||||
style="width:100%" |
||||
v-model="queryParam.createTimeRange" |
||||
format="YYYY-MM-DD" |
||||
:placeholder="['开始时间', '结束时间']" |
||||
@change="onDateChange" |
||||
@ok="onDateOk" |
||||
/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<template v-if="toggleSearchStatus"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="会员卡号" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择会员卡号" showSearch optionFilterProp="children" v-model="queryParam.organId"> |
||||
<a-select-option v-for="(item,index) in retailList" :key="index" :value="item.id"> |
||||
{{ item.supplier }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="仓库名称" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="请选择仓库" showSearch optionFilterProp="children" v-model="queryParam.depotId"> |
||||
<a-select-option v-for="(depot,index) in depotList" :key="index" :value="depot.id"> |
||||
{{ depot.depotName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="操作员" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择操作员" showSearch optionFilterProp="children" v-model="queryParam.creator"> |
||||
<a-select-option v-for="(item,index) in userList" :key="index" :value="item.id"> |
||||
{{ item.userName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
</template> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-button type="primary" @click="searchQuery">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset">重置</a-button> |
||||
<a @click="handleToggleSearch" style="margin-left: 8px"> |
||||
{{ toggleSearchStatus ? '收起' : '展开' }} |
||||
<a-icon :type="toggleSearchStatus ? 'up' : 'down'"/> |
||||
</a> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
</div> |
||||
<!-- 操作按钮区域 --> |
||||
<div class="table-operator" style="margin-top: 5px"> |
||||
<a-button v-if="btnEnableList.indexOf(1)>-1" @click="myHandleAdd" type="primary" icon="plus">新增</a-button> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" v-if="btnEnableList.indexOf(1)>-1" @click="batchDel"><a-icon type="delete"/>删除</a-menu-item> |
||||
<a-menu-item key="2" v-if="btnEnableList.indexOf(2)>-1" @click="batchSetStatus(1)"><a-icon type="check"/>审核</a-menu-item> |
||||
<a-menu-item key="3" v-if="btnEnableList.indexOf(7)>-1" @click="batchSetStatus(0)"><a-icon type="stop"/>反审核</a-menu-item> |
||||
</a-menu> |
||||
<a-button> |
||||
批量操作 <a-icon type="down" /> |
||||
</a-button> |
||||
</a-dropdown> |
||||
<a-tooltip placement="left" title="用于非会员和会员的单据录入,主要是用于散户使用,不能欠款。" slot="action"> |
||||
<a-icon v-if="btnEnableList.indexOf(1)>-1" type="question-circle" style="font-size:20px;float:right;" /> |
||||
</a-tooltip> |
||||
</div> |
||||
<!-- table区域-begin --> |
||||
<a-row> |
||||
<a-col :md="2.5" :sm="24"> |
||||
<list-columns-setter v-model="columns" :def-columns="columns" style="float: right;"/> |
||||
</a-col> |
||||
</a-row> |
||||
<div> |
||||
<a-table |
||||
ref="table" |
||||
size="middle" |
||||
bordered |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:components="drag(columns)" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:scroll="scroll" |
||||
:loading="loading" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange}" |
||||
@change="handleTableChange"> |
||||
<span slot="action" slot-scope="text, record"> |
||||
<a @click="myHandleDetail(record, '零售出库')">查看</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleEdit(record)">编辑</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleCopyAdd(record)">复制</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a-popconfirm v-if="btnEnableList.indexOf(1)>-1" title="确定删除吗?" @confirm="() => myHandleDelete(record)"> |
||||
<a>删除</a> |
||||
</a-popconfirm> |
||||
</span> |
||||
<template slot="customRenderStatus" slot-scope="status"> |
||||
<a-tag v-if="status == '0'" color="red">未审核</a-tag> |
||||
<a-tag v-if="status == '1'" color="green">已审核</a-tag> |
||||
</template> |
||||
</a-table> |
||||
</div> |
||||
<!-- table区域-end --> |
||||
<!-- 表单区域 --> |
||||
<retail-out-modal ref="modalForm" @ok="modalFormOk"></retail-out-modal> |
||||
<bill-detail ref="modalDetail"></bill-detail> |
||||
</a-card> |
||||
</a-col> |
||||
</a-row> |
||||
</template> |
||||
<!-- create ji sheng hua--> |
||||
<script> |
||||
import RetailOutModal from './modules/RetailOutModal' |
||||
import BillDetail from './dialog/BillDetail' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
import { BillListMixin } from './mixins/BillListMixin' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
import ListColumnsSetter from '@/components/ListColumnsSetter' |
||||
import tableDragResizeMixin from '@/mixins/tableDragResizeMixin' |
||||
export default { |
||||
name: "RetailOutList", |
||||
mixins:[JeecgListMixin,BillListMixin,tableDragResizeMixin], |
||||
components: { |
||||
RetailOutModal, |
||||
BillDetail, |
||||
JDate, |
||||
ListColumnsSetter |
||||
}, |
||||
data () { |
||||
return { |
||||
// 查询条件 |
||||
queryParam: { |
||||
number: "", |
||||
materialParam: "", |
||||
type: "出库", |
||||
subType: "零售", |
||||
roleType: Vue.ls.get('roleType'), |
||||
organId: "", |
||||
depotId: "", |
||||
creator: "" |
||||
}, |
||||
labelCol: { |
||||
span: 5 |
||||
}, |
||||
wrapperCol: { |
||||
span: 18, |
||||
offset: 1 |
||||
}, |
||||
// 表头 |
||||
columns: [ |
||||
{ title: '会员', dataIndex: 'organName',width:120, ellipsis:true, align:'center'}, |
||||
{ title: '单据编号', dataIndex: 'number',width:160, align:'center', |
||||
customRender:function (text,record,index) { |
||||
if(record.linkNumber) { |
||||
return text + "[转]"; |
||||
} else { |
||||
return text; |
||||
} |
||||
} |
||||
}, |
||||
{ title: '商品信息', dataIndex: 'materialsList',width:220, ellipsis:true, align:'center', |
||||
customRender:function (text,record,index) { |
||||
if(text) { |
||||
return text.replace(",",","); |
||||
} |
||||
} |
||||
}, |
||||
{ title: '单据日期', dataIndex: 'operTimeStr',width:145, align:'center'}, |
||||
{ title: '操作员', dataIndex: 'userName',width:80, ellipsis:true, align:'center'}, |
||||
{ title: '金额合计', dataIndex: 'totalPrice',width:80, align:'center'}, |
||||
{ title: '收款金额', dataIndex: 'getAmount',width:80, align:'center', |
||||
customRender:function (text,record,index) { |
||||
if(record.backAmount) { |
||||
return record.changeAmount + record.backAmount |
||||
} else { |
||||
return record.changeAmount |
||||
} |
||||
} |
||||
}, |
||||
{ title: '找零', dataIndex: 'backAmount',width:50, align:'center'}, |
||||
{ title: '状态', dataIndex: 'status', width: 80, align: "center", |
||||
scopedSlots: { customRender: 'customRenderStatus' }, align:'center' |
||||
}, |
||||
{ |
||||
title: '操作', |
||||
dataIndex: 'action', |
||||
align:"center", width: 150, |
||||
scopedSlots: { customRender: 'action' }, |
||||
} |
||||
], |
||||
url: { |
||||
list: "/erp/depotHead/list", |
||||
delete: "/erp/depotHead/delete", |
||||
deleteBatch: "/erp/depotHead/deleteBatch", |
||||
batchSetStatusUrl: "/erp/depotHead/batchSetStatus" |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
}, |
||||
created () { |
||||
this.initRetail() |
||||
this.getDepotData() |
||||
this.initUser() |
||||
}, |
||||
methods: { |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
@import '~@assets/less/common.less' |
||||
</style> |
@ -0,0 +1,246 @@
|
||||
<!-- create ji sheng hua--> |
||||
<template> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="24"> |
||||
<a-card :style="cardStyle" :bordered="false"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="searchQuery"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据编号" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入单据编号" v-model="queryParam.number"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="商品信息" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入条码、名称、规格、型号" v-model="queryParam.materialParam"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据日期" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-range-picker |
||||
style="width:100%" |
||||
v-model="queryParam.createTimeRange" |
||||
format="YYYY-MM-DD" |
||||
:placeholder="['开始时间', '结束时间']" |
||||
@change="onDateChange" |
||||
@ok="onDateOk" |
||||
/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<template v-if="toggleSearchStatus"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="客户" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择客户" showSearch optionFilterProp="children" v-model="queryParam.organId"> |
||||
<a-select-option v-for="(item,index) in cusList" :key="index" :value="item.id"> |
||||
{{ item.supplier }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="仓库名称" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="请选择仓库" showSearch optionFilterProp="children" v-model="queryParam.depotId"> |
||||
<a-select-option v-for="(depot,index) in depotList" :key="index" :value="depot.id"> |
||||
{{ depot.depotName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="操作员" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择操作员" showSearch optionFilterProp="children" v-model="queryParam.creator"> |
||||
<a-select-option v-for="(item,index) in userList" :key="index" :value="item.id"> |
||||
{{ item.userName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="关联单据" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入关联单据" v-model="queryParam.linkNumber"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
</template> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-button type="primary" @click="searchQuery">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset">重置</a-button> |
||||
<a @click="handleToggleSearch" style="margin-left: 8px"> |
||||
{{ toggleSearchStatus ? '收起' : '展开' }} |
||||
<a-icon :type="toggleSearchStatus ? 'up' : 'down'"/> |
||||
</a> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
</div> |
||||
<!-- 操作按钮区域 --> |
||||
<div class="table-operator" style="margin-top: 5px"> |
||||
<a-button v-if="btnEnableList.indexOf(1)>-1" @click="myHandleAdd" type="primary" icon="plus">新增</a-button> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" v-if="btnEnableList.indexOf(1)>-1" @click="batchDel"><a-icon type="delete"/>删除</a-menu-item> |
||||
<a-menu-item key="2" v-if="btnEnableList.indexOf(2)>-1" @click="batchSetStatus(1)"><a-icon type="check"/>审核</a-menu-item> |
||||
<a-menu-item key="3" v-if="btnEnableList.indexOf(7)>-1" @click="batchSetStatus(0)"><a-icon type="stop"/>反审核</a-menu-item> |
||||
</a-menu> |
||||
<a-button> |
||||
批量操作 <a-icon type="down" /> |
||||
</a-button> |
||||
</a-dropdown> |
||||
<a-tooltip placement="left" title="用于销售出库单据的退货。销售退货单可以由销售出库单转过来,也可以单独创建。" slot="action"> |
||||
<a-icon v-if="btnEnableList.indexOf(1)>-1" type="question-circle" style="font-size:20px;float:right;" /> |
||||
</a-tooltip> |
||||
</div> |
||||
<!-- table区域-begin --> |
||||
<a-row> |
||||
<a-col :md="2.5" :sm="24"> |
||||
<list-columns-setter v-model="columns" :def-columns="columns" style="float: right;"/> |
||||
</a-col> |
||||
</a-row> |
||||
<div> |
||||
<a-table |
||||
ref="table" |
||||
size="middle" |
||||
bordered |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:components="drag(columns)" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:scroll="scroll" |
||||
:loading="loading" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange}" |
||||
@change="handleTableChange"> |
||||
<span slot="action" slot-scope="text, record"> |
||||
<a @click="myHandleDetail(record, '销售退货入库')">查看</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleEdit(record)">编辑</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleCopyAdd(record)">复制</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a-popconfirm v-if="btnEnableList.indexOf(1)>-1" title="确定删除吗?" @confirm="() => myHandleDelete(record)"> |
||||
<a>删除</a> |
||||
</a-popconfirm> |
||||
</span> |
||||
<template slot="customRenderStatus" slot-scope="status"> |
||||
<a-tag v-if="status == '0'" color="red">未审核</a-tag> |
||||
<a-tag v-if="status == '1'" color="green">已审核</a-tag> |
||||
</template> |
||||
</a-table> |
||||
</div> |
||||
<!-- table区域-end --> |
||||
<!-- 表单区域 --> |
||||
<sale-back-modal ref="modalForm" @ok="modalFormOk"></sale-back-modal> |
||||
<bill-detail ref="modalDetail"></bill-detail> |
||||
</a-card> |
||||
</a-col> |
||||
</a-row> |
||||
</template> |
||||
<script> |
||||
import SaleBackModal from './modules/SaleBackModal' |
||||
import BillDetail from './dialog/BillDetail' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
import { BillListMixin } from './mixins/BillListMixin' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
import ListColumnsSetter from '@/components/ListColumnsSetter' |
||||
import tableDragResizeMixin from '@/mixins/tableDragResizeMixin' |
||||
export default { |
||||
name: "SaleBackList", |
||||
mixins:[JeecgListMixin,BillListMixin,tableDragResizeMixin], |
||||
components: { |
||||
SaleBackModal, |
||||
BillDetail, |
||||
JDate, |
||||
ListColumnsSetter |
||||
}, |
||||
data () { |
||||
return { |
||||
// 查询条件 |
||||
queryParam: { |
||||
number: "", |
||||
materialParam: "", |
||||
type: "入库", |
||||
subType: "销售退货", |
||||
roleType: Vue.ls.get('roleType'), |
||||
organId: "", |
||||
depotId: "", |
||||
creator: "", |
||||
linkNumber: "" |
||||
}, |
||||
labelCol: { |
||||
span: 5 |
||||
}, |
||||
wrapperCol: { |
||||
span: 18, |
||||
offset: 1 |
||||
}, |
||||
// 表头 |
||||
columns: [ |
||||
{ title: '客户', dataIndex: 'organName',width:120, align:"center", ellipsis:true}, |
||||
{ title: '单据编号', dataIndex: 'number',width:160, align:"center", |
||||
customRender:function (text,record,index) { |
||||
if(record.linkNumber) { |
||||
return text + "[转]"; |
||||
} else { |
||||
return text; |
||||
} |
||||
} |
||||
}, |
||||
{ title: '商品信息', dataIndex: 'materialsList',width:220, align:"center", ellipsis:true, |
||||
customRender:function (text,record,index) { |
||||
if(text) { |
||||
return text.replace(",",","); |
||||
} |
||||
} |
||||
}, |
||||
{ title: '单据日期', dataIndex: 'operTimeStr',width:145, align:"center", }, |
||||
{ title: '操作员', dataIndex: 'userName',width:80, align:"center", ellipsis:true}, |
||||
{ title: '金额合计', dataIndex: 'totalPrice',width:80, align:"center", }, |
||||
{ title: '含税合计', dataIndex: 'totalTaxLastMoney',width:80, align:"center", |
||||
customRender:function (text,record,index) { |
||||
return (record.discountMoney + record.discountLastMoney).toFixed(2); |
||||
} |
||||
}, |
||||
{ title: '待退金额', dataIndex: 'needBackMoney',width:80, align:"center", |
||||
customRender:function (text,record,index) { |
||||
let needBackMoney = record.discountLastMoney + record.otherMoney |
||||
return needBackMoney? needBackMoney.toFixed(2):'' |
||||
} |
||||
}, |
||||
{ title: '退款', dataIndex: 'changeAmount',width:50, align:"center", }, |
||||
{ title: '状态', dataIndex: 'status', width: 80, align: "center", |
||||
scopedSlots: { customRender: 'customRenderStatus' } |
||||
}, |
||||
{ |
||||
title: '操作', |
||||
dataIndex: 'action', |
||||
align:"center", width: 160, |
||||
scopedSlots: { customRender: 'action' }, |
||||
} |
||||
], |
||||
url: { |
||||
list: "/erp/depotHead/list", |
||||
delete: "/erp/depotHead/delete", |
||||
deleteBatch: "/erp/depotHead/deleteBatch", |
||||
batchSetStatusUrl: "/erp/depotHead/batchSetStatus" |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
}, |
||||
created () { |
||||
this.initCustomer() |
||||
this.getDepotData() |
||||
this.initUser() |
||||
}, |
||||
methods: { |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
@import '~@assets/less/common.less' |
||||
</style> |
@ -0,0 +1,245 @@
|
||||
<!-- create jishenghua--> |
||||
<template> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="24"> |
||||
<a-card :style="cardStyle" :bordered="false"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="searchQuery"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据编号" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入单据编号" v-model="queryParam.number"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="商品信息" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入物料编码、名称、规格、型号" v-model="queryParam.materialParam"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据日期" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-range-picker |
||||
style="width:100%" |
||||
v-model="queryParam.createTimeRange" |
||||
format="YYYY-MM-DD" |
||||
:placeholder="['开始时间', '结束时间']" |
||||
@change="onDateChange" |
||||
@ok="onDateOk" |
||||
/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<template v-if="toggleSearchStatus"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="客户" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择客户" showSearch optionFilterProp="children" v-model="queryParam.organId"> |
||||
<a-select-option v-for="(item,index) in cusList" :key="index" :value="item.id"> |
||||
{{ item.supplier }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="操作员" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择操作员" showSearch optionFilterProp="children" v-model="queryParam.creator"> |
||||
<a-select-option v-for="(item,index) in userList" :key="index" :value="item.id"> |
||||
{{ item.userName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
</template> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-button type="primary" @click="searchQuery">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset">重置</a-button> |
||||
<a @click="handleToggleSearch" style="margin-left: 8px"> |
||||
{{ toggleSearchStatus ? '收起' : '展开' }} |
||||
<a-icon :type="toggleSearchStatus ? 'up' : 'down'"/> |
||||
</a> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
</div> |
||||
<!-- 操作按钮区域 --> |
||||
<div class="table-operator" style="margin-top: 5px"> |
||||
<a-button v-if="btnEnableList.indexOf(1)>-1" @click="myHandleAdd" type="primary" icon="plus">新增</a-button> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" v-if="btnEnableList.indexOf(1)>-1" @click="batchDel"><a-icon type="delete"/>删除</a-menu-item> |
||||
<a-menu-item key="2" v-if="btnEnableList.indexOf(2)>-1" @click="batchSetStatus(1)"><a-icon type="check"/>审核</a-menu-item> |
||||
<a-menu-item key="3" v-if="btnEnableList.indexOf(7)>-1" @click="batchSetStatus(0)"><a-icon type="stop"/>反审核</a-menu-item> |
||||
</a-menu> |
||||
<a-button> |
||||
批量操作 <a-icon type="down" /> |
||||
</a-button> |
||||
</a-dropdown> |
||||
<a-tooltip placement="left" title="销售订单不涉及收款金额,销售订单可以转销售出库单,但需要先对销售订单进行审核。 |
||||
勾选单据之后可以进行批量操作(删除、审核、反审核)" slot="action"> |
||||
<a-icon v-if="btnEnableList.indexOf(1)>-1" type="question-circle" style="font-size:20px;float:right;" /> |
||||
</a-tooltip> |
||||
</div> |
||||
<!-- table区域-begin --> |
||||
<a-row> |
||||
<a-col :md="2.5" :sm="24"> |
||||
<list-columns-setter v-model="columns" :def-columns="columns" style="float: right;"/> |
||||
</a-col> |
||||
</a-row> |
||||
<div> |
||||
<a-table |
||||
ref="table" |
||||
size="middle" |
||||
bordered |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:components="drag(columns)" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:scroll="scroll" |
||||
:loading="loading" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange}" |
||||
@change="handleTableChange"> |
||||
<span slot="action" slot-scope="text, record"> |
||||
<a @click="myHandleDetail(record, '销售订单')">查看</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleEdit(record)">编辑</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleCopyAdd(record)">复制</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a-popconfirm v-if="btnEnableList.indexOf(1)>-1" title="确定删除吗?" @confirm="() => myHandleDelete(record)"> |
||||
<a>删除</a> |
||||
</a-popconfirm> |
||||
</span> |
||||
<template slot="customRenderStatus" slot-scope="status"> |
||||
<a-tag v-if="status == '0'" color="red">未审核</a-tag> |
||||
<a-tag v-if="status == '1'" color="green">已审核</a-tag> |
||||
<a-tag v-if="status == '2'" color="cyan">完成销售</a-tag> |
||||
<a-tag v-if="status == '3'" color="blue">部分销售</a-tag> |
||||
</template> |
||||
</a-table> |
||||
</div> |
||||
<!-- table区域-end --> |
||||
<!-- 表单区域 --> |
||||
<sale-order-modal ref="modalForm" @ok="modalFormOk"></sale-order-modal> |
||||
<bill-detail ref="modalDetail"></bill-detail> |
||||
</a-card> |
||||
</a-col> |
||||
</a-row> |
||||
</template> |
||||
<script> |
||||
import SaleOrderModal from './modules/SaleOrderModal' |
||||
import BillDetail from './dialog/BillDetail' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
import { BillListMixin } from './mixins/BillListMixin' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
import ListColumnsSetter from '@/components/ListColumnsSetter' |
||||
import tableDragResizeMixin from '@/mixins/tableDragResizeMixin' |
||||
export default { |
||||
name: "SaleOrderList", |
||||
mixins:[JeecgListMixin,BillListMixin,tableDragResizeMixin], |
||||
components: { |
||||
SaleOrderModal, |
||||
BillDetail, |
||||
JDate, |
||||
ListColumnsSetter |
||||
}, |
||||
data () { |
||||
return { |
||||
// 查询条件 |
||||
queryParam: { |
||||
number: "", |
||||
materialParam: "", |
||||
type: "其它", |
||||
subType: "销售订单", |
||||
roleType: Vue.ls.get('roleType'), |
||||
organId: "", |
||||
depotId: "", |
||||
creator: "" |
||||
}, |
||||
labelCol: { |
||||
span: 5 |
||||
}, |
||||
wrapperCol: { |
||||
span: 18, |
||||
offset: 1 |
||||
}, |
||||
// 表头 |
||||
columns: [ |
||||
{ title: '客户', dataIndex: 'organName',width:120, align:'center', ellipsis:true}, |
||||
{ title: '单据编号', dataIndex: 'number',width:160, align:'center', |
||||
customRender:function (text,record,index) { |
||||
if(record.linkNumber) { |
||||
return text + "[转]"; |
||||
} else { |
||||
return text; |
||||
} |
||||
} |
||||
}, |
||||
{ title: '商品信息', dataIndex: 'materialsList',width:220, align:'center', ellipsis:true, |
||||
customRender:function (text,record,index) { |
||||
if(text) { |
||||
return text.replace(",",","); |
||||
} |
||||
} |
||||
}, |
||||
{ title: '单据日期', dataIndex: 'operTimeStr',width:145, align:'center',}, |
||||
{ title: '操作员', dataIndex: 'userName',width:80, align:'center', ellipsis:true}, |
||||
{ title: '金额合计', dataIndex: 'totalPrice',width:80, align:'center',}, |
||||
{ title: '含税合计', dataIndex: 'totalTaxLastMoney',width:80, align:'center', |
||||
customRender:function (text,record,index) { |
||||
if(record.discountLastMoney) { |
||||
return (record.discountMoney + record.discountLastMoney).toFixed(2); |
||||
} else { |
||||
return record.totalPrice; |
||||
} |
||||
} |
||||
}, |
||||
{ title: '状态', dataIndex: 'status', width: 70, align: "center", |
||||
scopedSlots: { customRender: 'customRenderStatus' } |
||||
}, |
||||
{ |
||||
title: '操作', |
||||
dataIndex: 'action', |
||||
align:"center", width: 150, |
||||
scopedSlots: { customRender: 'action' }, |
||||
} |
||||
], |
||||
url: { |
||||
list: "/erp/depotHead/list", |
||||
delete: "/erp/depotHead/delete", |
||||
deleteBatch: "/erp/depotHead/deleteBatch", |
||||
batchSetStatusUrl: "/erp/depotHead/batchSetStatus" |
||||
} |
||||
} |
||||
}, |
||||
created() { |
||||
this.initCustomer() |
||||
this.initUser() |
||||
}, |
||||
computed: { |
||||
}, |
||||
methods: { |
||||
myHandleEdit(record) { |
||||
if(record.status === '0') { |
||||
this.$refs.modalForm.action = "edit"; |
||||
this.handleEdit(record); |
||||
} else { |
||||
this.$message.warning("抱歉,只有未审核的单据才能编辑!") |
||||
} |
||||
}, |
||||
myHandleDelete(record) { |
||||
if(record.status === '0') { |
||||
this.handleDelete(record.id) |
||||
} else { |
||||
this.$message.warning("抱歉,只有未审核的单据才能删除!") |
||||
} |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
@import '~@assets/less/common.less' |
||||
</style> |
@ -0,0 +1,254 @@
|
||||
<!-- create j i s h e n g h u a --> |
||||
<template> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="24"> |
||||
<a-card :style="cardStyle" :bordered="false"> |
||||
<!-- 查询区域 --> |
||||
<div class="table-page-search-wrapper"> |
||||
<!-- 搜索区域 --> |
||||
<a-form layout="inline" @keyup.enter.native="searchQuery"> |
||||
<a-row :gutter="24"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据编号" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入单据编号" v-model="queryParam.number"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="商品信息" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入物料编码、名称、规格、型号" v-model="queryParam.materialParam"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="单据日期" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-range-picker |
||||
style="width:100%" |
||||
v-model="queryParam.createTimeRange" |
||||
format="YYYY-MM-DD" |
||||
:placeholder="['开始时间', '结束时间']" |
||||
@change="onDateChange" |
||||
@ok="onDateOk" |
||||
/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<template v-if="toggleSearchStatus"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="客户" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择客户" showSearch optionFilterProp="children" v-model="queryParam.organId"> |
||||
<a-select-option v-for="(item,index) in cusList" :key="index" :value="item.id"> |
||||
{{ item.supplier }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="仓库名称" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="请选择仓库" showSearch optionFilterProp="children" v-model="queryParam.depotId"> |
||||
<a-select-option v-for="(depot,index) in depotList" :key="index" :value="depot.id"> |
||||
{{ depot.depotName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="操作员" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-select placeholder="选择操作员" showSearch optionFilterProp="children" v-model="queryParam.creator"> |
||||
<a-select-option v-for="(item,index) in userList" :key="index" :value="item.id"> |
||||
{{ item.userName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-form-item label="关联订单" :labelCol="labelCol" :wrapperCol="wrapperCol"> |
||||
<a-input placeholder="请输入关联订单" v-model="queryParam.linkNumber"></a-input> |
||||
</a-form-item> |
||||
</a-col> |
||||
</template> |
||||
<span style="float: left;overflow: hidden;" class="table-page-search-submitButtons"> |
||||
<a-col :md="6" :sm="24"> |
||||
<a-button type="primary" @click="searchQuery">查询</a-button> |
||||
<a-button style="margin-left: 8px" @click="searchReset">重置</a-button> |
||||
<a @click="handleToggleSearch" style="margin-left: 8px"> |
||||
{{ toggleSearchStatus ? '收起' : '展开' }} |
||||
<a-icon :type="toggleSearchStatus ? 'up' : 'down'"/> |
||||
</a> |
||||
</a-col> |
||||
</span> |
||||
</a-row> |
||||
</a-form> |
||||
</div> |
||||
<!-- 操作按钮区域 --> |
||||
<div class="table-operator" style="margin-top: 5px"> |
||||
<a-button v-if="btnEnableList.indexOf(1)>-1" @click="myHandleAdd" type="primary" icon="plus">新增</a-button> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" v-if="btnEnableList.indexOf(1)>-1" @click="batchDel"><a-icon type="delete"/>删除</a-menu-item> |
||||
<a-menu-item key="2" v-if="btnEnableList.indexOf(2)>-1" @click="batchSetStatus(1)"><a-icon type="check"/>审核</a-menu-item> |
||||
<a-menu-item key="3" v-if="btnEnableList.indexOf(7)>-1" @click="batchSetStatus(0)"><a-icon type="stop"/>反审核</a-menu-item> |
||||
</a-menu> |
||||
<a-button> |
||||
批量操作 <a-icon type="down" /> |
||||
</a-button> |
||||
</a-dropdown> |
||||
<a-tooltip placement="left" title="销售出库单可以由销售订单转过来,也可以单独创建。 |
||||
销售出库单据中的仓库列表只显示当前用户有权限的仓库。销售出库单可以使用多账户收款。 |
||||
勾选单据之后可以进行批量操作(删除、审核、反审核)" slot="action"> |
||||
<a-icon v-if="btnEnableList.indexOf(1)>-1" type="question-circle" style="font-size:20px;float:right;" /> |
||||
</a-tooltip> |
||||
</div> |
||||
<!-- table区域-begin --> |
||||
<a-row> |
||||
<a-col :md="2.5" :sm="24"> |
||||
<list-columns-setter v-model="columns" :def-columns="columns" style="float: right;"/> |
||||
</a-col> |
||||
</a-row> |
||||
<div> |
||||
<a-table |
||||
ref="table" |
||||
size="middle" |
||||
bordered |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:components="drag(columns)" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:scroll="scroll" |
||||
:loading="loading" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange}" |
||||
@change="handleTableChange"> |
||||
<span slot="action" slot-scope="text, record"> |
||||
<a @click="myHandleDetail(record, '销售出库')">查看</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleEdit(record)">编辑</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a v-if="btnEnableList.indexOf(1)>-1" @click="myHandleCopyAdd(record)">复制</a> |
||||
<a-divider v-if="btnEnableList.indexOf(1)>-1" type="vertical" /> |
||||
<a-popconfirm v-if="btnEnableList.indexOf(1)>-1" title="确定删除吗?" @confirm="() => myHandleDelete(record)"> |
||||
<a>删除</a> |
||||
</a-popconfirm> |
||||
</span> |
||||
<template slot="customRenderStatus" slot-scope="status"> |
||||
<a-tag v-if="status == '0'" color="red">未审核</a-tag> |
||||
<a-tag v-if="status == '1'" color="green">已审核</a-tag> |
||||
</template> |
||||
</a-table> |
||||
</div> |
||||
<!-- table区域-end --> |
||||
<!-- 表单区域 --> |
||||
<sale-out-modal ref="modalForm" @ok="modalFormOk"></sale-out-modal> |
||||
<bill-detail ref="modalDetail"></bill-detail> |
||||
</a-card> |
||||
</a-col> |
||||
</a-row> |
||||
</template> |
||||
<script> |
||||
import SaleOutModal from './modules/SaleOutModal' |
||||
import BillDetail from './dialog/BillDetail' |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
import { BillListMixin } from './mixins/BillListMixin' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
import ListColumnsSetter from '@/components/ListColumnsSetter' |
||||
import tableDragResizeMixin from '@/mixins/tableDragResizeMixin' |
||||
export default { |
||||
name: "SaleOutList", |
||||
mixins:[JeecgListMixin,BillListMixin,tableDragResizeMixin], |
||||
components: { |
||||
SaleOutModal, |
||||
BillDetail, |
||||
JDate, |
||||
ListColumnsSetter |
||||
}, |
||||
data () { |
||||
return { |
||||
// 查询条件 |
||||
queryParam: { |
||||
number: "", |
||||
materialParam: "", |
||||
type: "出库", |
||||
subType: "销售", |
||||
roleType: Vue.ls.get('roleType'), |
||||
organId: "", |
||||
depotId: "", |
||||
creator: "", |
||||
linkNumber: "" |
||||
}, |
||||
labelCol: { |
||||
span: 5 |
||||
}, |
||||
wrapperCol: { |
||||
span: 18, |
||||
offset: 1 |
||||
}, |
||||
// 表头 |
||||
columns: [ |
||||
{ title: '客户', dataIndex: 'organName',width:120, align:'center', ellipsis:true}, |
||||
{ title: '单据编号', dataIndex: 'number',width:160, align:'center', |
||||
customRender:function (text,record,index) { |
||||
if(record.linkNumber) { |
||||
return text + "[订]"; |
||||
} else { |
||||
return text; |
||||
} |
||||
} |
||||
}, |
||||
{ title: '商品信息', dataIndex: 'materialsList',width:220, align:'center', ellipsis:true, |
||||
customRender:function (text,record,index) { |
||||
if(text) { |
||||
return text.replace(",",","); |
||||
} |
||||
} |
||||
}, |
||||
{ title: '单据日期', dataIndex: 'operTimeStr',width:145, align:'center',}, |
||||
{ title: '操作员', dataIndex: 'userName',width:80, align:'center', ellipsis:true}, |
||||
{ title: '金额合计', dataIndex: 'totalPrice',width:80, align:'center',}, |
||||
{ title: '含税合计', dataIndex: 'totalTaxLastMoney',width:80, align:'center', |
||||
customRender:function (text,record,index) { |
||||
return (record.discountMoney + record.discountLastMoney).toFixed(2); |
||||
} |
||||
}, |
||||
{ title: '待收金额', dataIndex: 'needOutMoney',width:80, align:'center', |
||||
customRender:function (text,record,index) { |
||||
let needOutMoney = record.discountLastMoney + record.otherMoney |
||||
return needOutMoney? needOutMoney.toFixed(2):'' |
||||
} |
||||
}, |
||||
{ title: '收款', dataIndex: 'changeAmount',width:60, align:'center',}, |
||||
{ title: '欠款', dataIndex: 'debt',width:60, align:'center', |
||||
customRender:function (text,record,index) { |
||||
let debt = record.discountLastMoney + record.otherMoney - record.changeAmount |
||||
return debt? debt.toFixed(2):'' |
||||
} |
||||
}, |
||||
{ title: '状态', dataIndex: 'status', width: 80, align: "center", |
||||
scopedSlots: { customRender: 'customRenderStatus' } |
||||
}, |
||||
{ |
||||
title: '操作', |
||||
dataIndex: 'action', |
||||
align:"center", width: 180, |
||||
scopedSlots: { customRender: 'action' }, |
||||
} |
||||
], |
||||
url: { |
||||
list: "/erp/depotHead/list", |
||||
delete: "/erp/depotHead/delete", |
||||
deleteBatch: "/erp/depotHead/deleteBatch", |
||||
batchSetStatusUrl: "/erp/depotHead/batchSetStatus" |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
}, |
||||
created() { |
||||
this.initCustomer() |
||||
this.getDepotData() |
||||
this.initUser() |
||||
}, |
||||
methods: { |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
@import '~@assets/less/common.less' |
||||
</style> |
@ -0,0 +1,112 @@
|
||||
<template> |
||||
<a-modal |
||||
:title="title" |
||||
:width="500" |
||||
:visible="visible" |
||||
:confirmLoading="confirmLoading" |
||||
@ok="handleOk" |
||||
@cancel="handleCancel" |
||||
cancelText="关闭" |
||||
wrapClassName="ant-modal-cust-warp" |
||||
style="top:30%;height: 35%;overflow-y: hidden"> |
||||
<template slot="footer"> |
||||
<a-button key="back" v-if="isReadOnly" @click="handleCancel"> |
||||
关闭 |
||||
</a-button> |
||||
</template> |
||||
<a-spin :spinning="confirmLoading"> |
||||
<a-form :form="form" id="batchSetDepot"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="仓库名称"> |
||||
<a-select placeholder="请选择仓库" v-decorator="[ 'depotId', validatorRules.depotId ]" showSearch optionFilterProp="children"> |
||||
<a-select-option v-for="(depot,index) in depotList" :key="index" :value="depot.id"> |
||||
{{ depot.depotName }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-form> |
||||
</a-spin> |
||||
</a-modal> |
||||
</template> |
||||
<script> |
||||
import pick from 'lodash.pick' |
||||
import { getAction } from '@/api/manage' |
||||
export default { |
||||
name: "BatchSetDepot", |
||||
data () { |
||||
return { |
||||
title:"操作", |
||||
visible: false, |
||||
model: {}, |
||||
depotList: [], |
||||
isReadOnly: false, |
||||
labelCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 5 }, |
||||
}, |
||||
wrapperCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 16 }, |
||||
}, |
||||
confirmLoading: false, |
||||
form: this.$form.createForm(this), |
||||
validatorRules:{ |
||||
depotId:{ |
||||
rules: [ |
||||
{ required: true, message: '请选择仓库!' } |
||||
] |
||||
} |
||||
}, |
||||
} |
||||
}, |
||||
created () { |
||||
}, |
||||
methods: { |
||||
getDepotData() { |
||||
getAction('/depot/findDepotByCurrentUser').then((res)=>{ |
||||
if(res.code === 200){ |
||||
this.depotList = res.data; |
||||
}else{ |
||||
this.$message.info(res.data); |
||||
} |
||||
}) |
||||
}, |
||||
add () { |
||||
this.edit({}); |
||||
this.getDepotData() |
||||
}, |
||||
edit (record) { |
||||
this.form.resetFields(); |
||||
this.model = Object.assign({}, record); |
||||
this.visible = true; |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue(pick(this.model, 'depotId')) |
||||
}); |
||||
}, |
||||
close () { |
||||
this.$emit('close'); |
||||
this.visible = false; |
||||
}, |
||||
handleOk () { |
||||
const that = this; |
||||
// 触发表单验证 |
||||
this.form.validateFields((err, values) => { |
||||
if (!err) { |
||||
that.confirmLoading = true; |
||||
let formData = Object.assign(this.model, values); |
||||
let depotId = formData.depotId |
||||
that.$emit('ok', depotId); |
||||
that.confirmLoading = false; |
||||
that.close(); |
||||
} |
||||
}) |
||||
}, |
||||
handleCancel () { |
||||
this.close() |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,71 @@
|
||||
<template> |
||||
<a-modal |
||||
:title="title" |
||||
:width="width" |
||||
:visible="visible" |
||||
@cancel="handleCancel" |
||||
cancelText="关闭" |
||||
wrapClassName="ant-modal-cust-warp" |
||||
style="top:5%;height: 100%;overflow-y: hidden"> |
||||
<template slot="footer"> |
||||
<a-button key="back" @click="handleCancel">取消</a-button> |
||||
</template> |
||||
<a-form :form="form"> |
||||
<template> |
||||
<iframe :src="billPrintUrl" width="100%" :height="height" frameborder="0" scrolling="no"></iframe> |
||||
</template> |
||||
<template> |
||||
<a-row> |
||||
<a-col> |
||||
<a-form-item> |
||||
<a-input v-decorator="['id']" hidden/> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
</template> |
||||
</a-form> |
||||
</a-modal> |
||||
</template> |
||||
|
||||
<script> |
||||
import pick from 'lodash.pick' |
||||
export default { |
||||
name: 'BillPrintIframe', |
||||
data () { |
||||
return { |
||||
title: "三联打印预览", |
||||
width: '1550px', |
||||
visible: false, |
||||
billPrintUrl: '', |
||||
height: "", |
||||
model: {}, |
||||
form: this.$form.createForm(this), |
||||
loading: false |
||||
} |
||||
}, |
||||
created () { |
||||
}, |
||||
methods: { |
||||
show(record, billPrintUrl, billPrintHeight) { |
||||
this.height = billPrintHeight |
||||
this.billPrintUrl = billPrintUrl |
||||
this.visible = true; |
||||
this.model = Object.assign({}, record); |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue(pick(this.model,'id')) |
||||
}); |
||||
}, |
||||
handleCancel() { |
||||
this.close() |
||||
}, |
||||
close() { |
||||
this.$emit('close'); |
||||
this.visible = false; |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,159 @@
|
||||
<template> |
||||
<a-modal |
||||
:title="title" |
||||
:width="1250" |
||||
:visible="visible" |
||||
@ok="handleOk" |
||||
@cancel="handleCancel" |
||||
cancelText="关闭" |
||||
wrapClassName="ant-modal-cust-warp" |
||||
style="top:5%;height: 100%;overflow-y: hidden"> |
||||
<!-- table区域-begin --> |
||||
<a-table |
||||
bordered |
||||
ref="table" |
||||
size="middle" |
||||
rowKey="id" |
||||
:columns="columns" |
||||
:dataSource="dataSource" |
||||
:pagination="ipagination" |
||||
:loading="loading" |
||||
:rowSelection="{selectedRowKeys: selectedRowKeys, onChange: onSelectChange, type: getType}" |
||||
:customRow="rowAction" |
||||
@change="handleTableChange"> |
||||
<template slot="customRenderStatus" slot-scope="status"> |
||||
<a-tag v-if="status === '0'" color="red">未审核</a-tag> |
||||
<a-tag v-if="status === '1'" color="green">已审核</a-tag> |
||||
<a-tag v-if="status === '2' && queryParam.subType === '采购订单'" color="cyan">完成采购</a-tag> |
||||
<a-tag v-if="status === '2' && queryParam.subType === '销售订单'" color="cyan">完成销售</a-tag> |
||||
<a-tag v-if="status === '3' && queryParam.subType === '采购订单'" color="blue">部分采购</a-tag> |
||||
<a-tag v-if="status === '3' && queryParam.subType === '销售订单'" color="blue">部分销售</a-tag> |
||||
</template> |
||||
</a-table> |
||||
<!-- table区域-end --> |
||||
</a-modal> |
||||
</template> |
||||
|
||||
<script> |
||||
import { JeecgListMixin } from '@/mixins/JeecgListMixin' |
||||
export default { |
||||
name: 'LinkBillList', |
||||
mixins:[JeecgListMixin], |
||||
data () { |
||||
return { |
||||
title: "操作", |
||||
visible: false, |
||||
disableMixinCreated: true, |
||||
selectedRowKeys: [], |
||||
selectionRows: [], |
||||
selectBillRows: [], |
||||
selectBillIds: '', |
||||
queryParam: { |
||||
number: "", |
||||
searchMaterial: "", |
||||
type: "", |
||||
subType: "", |
||||
status: "" |
||||
}, |
||||
// 表头 |
||||
columns: [ |
||||
{ |
||||
title: '#', |
||||
dataIndex: '', |
||||
key:'rowIndex', |
||||
width:40, |
||||
align:"center", |
||||
customRender:function (t,r,index) { |
||||
return parseInt(index)+1; |
||||
} |
||||
}, |
||||
{ title: '', dataIndex: 'organName',width:120}, |
||||
{ title: '单据编号', dataIndex: 'number',width:150}, |
||||
{ title: '商品信息', dataIndex: 'materialsList',width:280, ellipsis:true, |
||||
customRender:function (text,record,index) { |
||||
if(text) { |
||||
return text.replace(",",","); |
||||
} |
||||
} |
||||
}, |
||||
{ title: '单据日期', dataIndex: 'operTimeStr',width:145}, |
||||
{ title: '操作员', dataIndex: 'userName',width:70}, |
||||
{ title: '金额合计', dataIndex: 'totalPrice',width:70}, |
||||
{ title: '状态', dataIndex: 'status', width: 70, align: "center", |
||||
scopedSlots: { customRender: 'customRenderStatus' } |
||||
} |
||||
], |
||||
url: { |
||||
list: "/depotHead/list" |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
getType: function () { |
||||
return 'radio'; |
||||
} |
||||
}, |
||||
created() { |
||||
}, |
||||
methods: { |
||||
show(type, subType, organType, status) { |
||||
this.queryParam.type = type |
||||
this.queryParam.subType = subType |
||||
this.queryParam.status = status |
||||
this.columns[1].title = organType |
||||
this.model = Object.assign({}, {}); |
||||
this.visible = true; |
||||
this.loadData(1) |
||||
}, |
||||
close () { |
||||
this.$emit('close'); |
||||
this.visible = false; |
||||
}, |
||||
handleCancel () { |
||||
this.close() |
||||
}, |
||||
onSelectChange(selectedRowKeys, selectionRows) { |
||||
this.selectedRowKeys = selectedRowKeys; |
||||
this.selectionRows = selectionRows; |
||||
}, |
||||
handleOk () { |
||||
this.getSelectBillRows(); |
||||
this.$emit('ok', this.selectBillRows); |
||||
this.close(); |
||||
}, |
||||
getSelectBillRows() { |
||||
let dataSource = this.dataSource; |
||||
let billIds = ""; |
||||
this.selectBillRows = []; |
||||
for (let i = 0, len = dataSource.length; i < len; i++) { |
||||
if (this.selectedRowKeys.includes(dataSource[i].id)) { |
||||
this.selectBillRows.push(dataSource[i]); |
||||
billIds = billIds + "," + dataSource[i].id |
||||
} |
||||
} |
||||
this.selectBillIds = billIds.substring(1); |
||||
}, |
||||
rowAction(record, index) { |
||||
return { |
||||
on: { |
||||
click: () => { |
||||
let arr = [] |
||||
arr.push(record.id) |
||||
this.selectedRowKeys = arr |
||||
}, |
||||
dblclick: () => { |
||||
let arr = [] |
||||
arr.push(record.id) |
||||
this.selectedRowKeys = arr |
||||
this.handleOk() |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,188 @@
|
||||
<template> |
||||
<a-modal |
||||
:title="title" |
||||
:width="650" |
||||
:visible="visible" |
||||
:confirmLoading="confirmLoading" |
||||
@ok="handleOk" |
||||
@cancel="handleCancel" |
||||
cancelText="关闭" |
||||
wrapClassName="ant-modal-cust-warp" |
||||
style="top:20%;height: 60%;overflow-y: hidden"> |
||||
<a-spin :spinning="confirmLoading"> |
||||
<a-form :form="form"> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="12" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="结算账户1"> |
||||
<a-select style="width:185px;" placeholder="请选择结算账户" v-decorator="[ 'oneAccountId' ]" :dropdownMatchSelectWidth="false" allowClear> |
||||
<a-select-option v-for="(item,index) in accountList" :key="index" :value="item.id"> |
||||
{{ item.name }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="12" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="结算金额"> |
||||
<a-input-number placeholder="请输入金额" v-decorator.trim="[ 'oneAccountPrice' ]" /> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="12" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="结算账户2"> |
||||
<a-select style="width:185px;" placeholder="请选择结算账户" v-decorator="[ 'twoAccountId' ]" :dropdownMatchSelectWidth="false" allowClear> |
||||
<a-select-option v-for="(item,index) in accountList" :key="index" :value="item.id"> |
||||
{{ item.name }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="12" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="结算金额"> |
||||
<a-input-number placeholder="请输入金额" v-decorator.trim="[ 'twoAccountPrice' ]" /> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="12" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="结算账户3"> |
||||
<a-select style="width:185px;" placeholder="请选择结算账户" v-decorator="[ 'threeAccountId' ]" :dropdownMatchSelectWidth="false" allowClear> |
||||
<a-select-option v-for="(item,index) in accountList" :key="index" :value="item.id"> |
||||
{{ item.name }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="12" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="结算金额"> |
||||
<a-input-number placeholder="请输入金额" v-decorator.trim="[ 'threeAccountPrice' ]" /> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
</a-form> |
||||
</a-spin> |
||||
</a-modal> |
||||
</template> |
||||
<script> |
||||
import pick from 'lodash.pick' |
||||
import {getAccount} from '@/api/api' |
||||
export default { |
||||
name: 'ManyAccountModal', |
||||
data () { |
||||
return { |
||||
title:"操作", |
||||
visible: false, |
||||
model: {}, |
||||
accountList: [], |
||||
accountIdList: [], |
||||
accountMoneyList: [], |
||||
labelCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 8 }, |
||||
}, |
||||
wrapperCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 16 }, |
||||
}, |
||||
confirmLoading: false, |
||||
form: this.$form.createForm(this) |
||||
} |
||||
}, |
||||
created () { |
||||
}, |
||||
methods: { |
||||
edit (idStr, moneyStr) { |
||||
this.initAccount() |
||||
this.form.resetFields(); |
||||
this.model = Object.assign({}, {}); |
||||
let idList = [], moneyList = [] |
||||
if(idStr && idStr.indexOf(',')>-1) { |
||||
idList = idStr.split(",") |
||||
moneyList = moneyStr.split(",") |
||||
} else { |
||||
idList = idStr |
||||
moneyList = moneyStr |
||||
} |
||||
if(idList[0]) {this.model.oneAccountId = idList[0]-0} |
||||
if(idList[1]) {this.model.twoAccountId = idList[1]-0} |
||||
if(idList[2]) {this.model.threeAccountId = idList[2]-0} |
||||
if(moneyList[0]) {this.model.oneAccountPrice = Math.abs(moneyList[0])} |
||||
if(moneyList[1]) {this.model.twoAccountPrice = Math.abs(moneyList[1])} |
||||
if(moneyList[2]) {this.model.threeAccountPrice = Math.abs(moneyList[2])} |
||||
this.visible = true; |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue(pick(this.model,'oneAccountId','oneAccountPrice', |
||||
'twoAccountId','twoAccountPrice','threeAccountId','threeAccountPrice')) |
||||
}); |
||||
}, |
||||
close () { |
||||
this.$emit('close'); |
||||
this.visible = false; |
||||
}, |
||||
handleOk () { |
||||
const that = this; |
||||
// 触发表单验证 |
||||
this.form.validateFields((err, values) => { |
||||
if (!err) { |
||||
let allPrice = 0 |
||||
that.confirmLoading = true; |
||||
that.accountIdList = [] |
||||
that.accountMoneyList = [] |
||||
let formData = Object.assign(this.model, values); |
||||
if(formData.oneAccountId!==undefined) { |
||||
that.accountIdList.push(formData.oneAccountId) |
||||
} |
||||
if(formData.twoAccountId!==undefined) { |
||||
that.accountIdList.push(formData.twoAccountId) |
||||
} |
||||
if(formData.threeAccountId!==undefined) { |
||||
that.accountIdList.push(formData.threeAccountId) |
||||
} |
||||
if(formData.oneAccountPrice!==undefined) { |
||||
that.accountMoneyList.push(formData.oneAccountPrice) |
||||
allPrice = allPrice + formData.oneAccountPrice |
||||
} |
||||
if(formData.twoAccountPrice!==undefined) { |
||||
that.accountMoneyList.push(formData.twoAccountPrice) |
||||
allPrice = allPrice + formData.twoAccountPrice |
||||
} |
||||
if(formData.threeAccountPrice!==undefined) { |
||||
that.accountMoneyList.push(formData.threeAccountPrice) |
||||
allPrice = allPrice + formData.threeAccountPrice |
||||
} |
||||
if(that.accountIdList.length<2 || that.accountMoneyList.length<2) { |
||||
this.$message.warning('抱歉,多账户结算必须选择两个以上账户和金额!'); |
||||
that.confirmLoading = false; |
||||
return; |
||||
} |
||||
if((formData.oneAccountId && !formData.oneAccountPrice)|| |
||||
(formData.twoAccountId && !formData.twoAccountPrice)|| |
||||
(formData.threeAccountId && !formData.threeAccountPrice)) { |
||||
this.$message.warning('抱歉,请填写结算金额!'); |
||||
that.confirmLoading = false; |
||||
return; |
||||
} |
||||
that.$emit('ok', that.accountIdList, that.accountMoneyList, allPrice); |
||||
that.confirmLoading = false; |
||||
that.close(); |
||||
} |
||||
}) |
||||
}, |
||||
handleCancel () { |
||||
this.close() |
||||
}, |
||||
initAccount(){ |
||||
let that = this; |
||||
getAccount({}).then((res)=>{ |
||||
if(res && res.code === 200) { |
||||
that.accountList = res.data.accountList |
||||
} |
||||
}) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,133 @@
|
||||
import Vue from 'vue' |
||||
import {getAction } from '@/api/manage' |
||||
import {findBySelectSup, findBySelectCus, findBySelectRetail, getUserList } from '@/api/api' |
||||
|
||||
export const BillListMixin = { |
||||
data () { |
||||
return { |
||||
supList: [], |
||||
cusList: [], |
||||
retailList: [], |
||||
userList: [] |
||||
} |
||||
}, |
||||
computed: { |
||||
importExcelUrl: function(){ |
||||
return `${window._CONFIG['domianURL']}/${this.url.importExcelUrl}`; |
||||
}, |
||||
|
||||
isBatchDelEnabled: function () { |
||||
for (let i = 0; i < this.selectedRowKeys.length; i++) { |
||||
if (!this.selectionRows[i].actionsEnabled.delete) { |
||||
return false; |
||||
} |
||||
} |
||||
return true; |
||||
} |
||||
}, |
||||
created() { |
||||
this.removeStatusColumn() |
||||
}, |
||||
methods: { |
||||
myHandleAdd() { |
||||
this.$refs.modalForm.action = "add"; |
||||
this.handleAdd(); |
||||
}, |
||||
myHandleCopyAdd(record) { |
||||
this.$refs.modalForm.action = "copyAdd"; |
||||
this.$refs.modalForm.edit(record); |
||||
this.$refs.modalForm.title = "复制新增"; |
||||
this.$refs.modalForm.disableSubmit = false; |
||||
}, |
||||
myHandleEdit(record) { |
||||
if(record.status === '0') { |
||||
this.$refs.modalForm.action = "edit"; |
||||
this.handleEdit(record); |
||||
} else { |
||||
this.$message.warning("抱歉,只有未审核的单据才能编辑!") |
||||
} |
||||
}, |
||||
myHandleDelete(record) { |
||||
if(record.status === '0') { |
||||
this.handleDelete(record.id) |
||||
} else { |
||||
this.$message.warning("抱歉,只有未审核的单据才能删除!") |
||||
} |
||||
}, |
||||
myHandleDetail(record, type) { |
||||
this.handleDetail(record, type); |
||||
}, |
||||
handleApprove(record) { |
||||
this.$refs.modalForm.action = "approve"; |
||||
this.$refs.modalForm.edit(record); |
||||
this.$refs.modalForm.title = "审核"; |
||||
}, |
||||
searchReset() { |
||||
this.queryParam = { |
||||
type: this.queryParam.type, |
||||
subType: this.queryParam.subType |
||||
} |
||||
this.loadData(1); |
||||
}, |
||||
onDateChange: function (value, dateString) { |
||||
this.queryParam.beginTime=dateString[0]; |
||||
this.queryParam.endTime=dateString[1]; |
||||
}, |
||||
onDateOk(value) { |
||||
console.log(value); |
||||
}, |
||||
removeStatusColumn() { |
||||
//没有审核反审核权限的时候直接移除状态列
|
||||
if(this.btnEnableList.indexOf(2)===-1 && this.btnEnableList.indexOf(7)===-1) { |
||||
let statusIndex = 0 |
||||
for(let i=0; i<this.columns.length; i++){ |
||||
if(this.columns[i].dataIndex === 'status') { |
||||
statusIndex = i |
||||
} |
||||
} |
||||
//移除状态列
|
||||
this.columns.splice(statusIndex,1) |
||||
} |
||||
}, |
||||
initSupplier() { |
||||
let that = this; |
||||
findBySelectSup({}).then((res)=>{ |
||||
if(res) { |
||||
that.supList = res; |
||||
} |
||||
}); |
||||
}, |
||||
initCustomer() { |
||||
let that = this; |
||||
findBySelectCus({}).then((res)=>{ |
||||
if(res) { |
||||
that.cusList = res; |
||||
} |
||||
}); |
||||
}, |
||||
initRetail() { |
||||
let that = this; |
||||
findBySelectRetail({}).then((res)=>{ |
||||
if(res) { |
||||
that.retailList = res; |
||||
} |
||||
}); |
||||
}, |
||||
getDepotData() { |
||||
getAction('/depot/findDepotByCurrentUser').then((res)=>{ |
||||
if(res.code === 200){ |
||||
this.depotList = res.data; |
||||
}else{ |
||||
this.$message.info(res.data); |
||||
} |
||||
}) |
||||
}, |
||||
initUser() { |
||||
getUserList({}).then((res)=>{ |
||||
if(res) { |
||||
this.userList = res; |
||||
} |
||||
}); |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,748 @@
|
||||
import { FormTypes, getListData } from '@/utils/JEditableTableUtil' |
||||
import {findBySelectSup,findBySelectCus,findBySelectRetail,getMaterialByBarCode,findStockByDepotAndBarCode,getAccount, |
||||
getPersonByNumType, getBatchNumberList} from '@/api/api' |
||||
import { getAction,putAction } from '@/api/manage' |
||||
import { getMpListShort, getNowFormatDateTime } from "@/utils/util" |
||||
import Vue from 'vue' |
||||
|
||||
export const BillModalMixin = { |
||||
data() { |
||||
return { |
||||
action: '', |
||||
manyAccountBtnStatus: false, |
||||
supList: [], |
||||
cusList: [], |
||||
retailList: [], |
||||
personList: { |
||||
options: [], |
||||
value: '' |
||||
}, |
||||
depotList: [], |
||||
accountList: [], |
||||
accountIdList: [], |
||||
accountMoneyList: [], |
||||
billUnitPirce: '', |
||||
scanBarCode: '', |
||||
scanStatus: true, |
||||
isTenant: false, |
||||
spans: { |
||||
labelCol1: {span: 2}, |
||||
wrapperCol1: {span: 22}, |
||||
//1_5: 分为1.5列(相当于占了2/3)
|
||||
labelCol1_5: { span: 3 }, |
||||
wrapperCol1_5: { span: 21 }, |
||||
labelCol2: {span: 4}, |
||||
wrapperCol2: {span: 20}, |
||||
labelCol3: {span: 6}, |
||||
wrapperCol3: {span: 18}, |
||||
labelCol6: {span: 12}, |
||||
wrapperCol6: {span: 12} |
||||
}, |
||||
}; |
||||
}, |
||||
created () { |
||||
// this.isTenant = userInfo.id === userInfo.tenantId? true:false
|
||||
this.isTenant = true; |
||||
var that = this |
||||
document.onkeydown = function (e) { |
||||
var key = window.event.keyCode |
||||
if (key === 13) { |
||||
that.handleOk() // 触发事件
|
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
readOnly: function() { |
||||
return this.action !== "add" && this.action !== "edit"; |
||||
} |
||||
}, |
||||
methods: { |
||||
addInit(amountNum) { |
||||
getAction('/sequence/buildNumber').then((res) => { |
||||
if (res && res.code === 200) { |
||||
this.form.setFieldsValue({'number':amountNum + res.data.defaultNumber}) |
||||
} |
||||
}) |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue({'operTime':getNowFormatDateTime(), 'discount': 0, |
||||
'discountMoney': 0, 'discountLastMoney': 0, 'otherMoney': 0, 'changeAmount': 0, 'debt': 0}) |
||||
}) |
||||
this.$nextTick(() => { |
||||
getAccount({}).then((res)=>{ |
||||
if(res && res.code === 200) { |
||||
for (const item of res.data.accountList) { |
||||
if(item.isDefault){ |
||||
this.form.setFieldsValue({'accountId': Number(item.id)}) |
||||
} |
||||
} |
||||
} |
||||
}) |
||||
}) |
||||
this.accountIdList = [] |
||||
this.accountMoneyList = [] |
||||
this.manyAccountBtnStatus = false |
||||
}, |
||||
copyAddInit(amountNum) { |
||||
getAction('/sequence/buildNumber').then((res) => { |
||||
if (res && res.code === 200) { |
||||
this.form.setFieldsValue({'number':amountNum + res.data.defaultNumber}) |
||||
} |
||||
}) |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue({'operTime':getNowFormatDateTime()}) |
||||
}) |
||||
}, |
||||
/** 查询某个tab的数据 */ |
||||
requestSubTableData(url, params, tab, success) { |
||||
tab.loading = true |
||||
getAction(url, params).then(res => { |
||||
if(res && res.code === 200){ |
||||
tab.dataSource = res.data.rows |
||||
for(let i=0; i<tab.dataSource.length; i++){ |
||||
let info = tab.dataSource[i] |
||||
this.changeColumnShow(info) |
||||
} |
||||
typeof success === 'function' ? success(res) : '' |
||||
} |
||||
}).finally(() => { |
||||
tab.loading = false |
||||
}) |
||||
}, |
||||
//改变字段的状态,1-显示 0-隐藏
|
||||
changeFormTypes(columns, key, type) { |
||||
for(let i=0; i<columns.length; i++){ |
||||
if(columns[i].key === key) { |
||||
if(type){ |
||||
if(key === 'snList' || key === 'batchNumber') { |
||||
if(this.prefixNo === 'LSCK' || this.prefixNo === 'CGTH' || this.prefixNo === 'XSCK' || this.prefixNo === 'QTCK') { |
||||
columns[i].type = FormTypes.popupJsh //显示
|
||||
} else { |
||||
columns[i].type = FormTypes.input //显示
|
||||
} |
||||
} else if(key === 'expirationDate') { |
||||
if(this.prefixNo === 'LSTH' || this.prefixNo === 'CGRK' || this.prefixNo === 'XSTH' || this.prefixNo === 'QTRK') { |
||||
columns[i].type = FormTypes.date //显示
|
||||
} else { |
||||
columns[i].type = FormTypes.normal //显示
|
||||
} |
||||
} else { |
||||
columns[i].type = FormTypes.normal //显示
|
||||
} |
||||
} else { |
||||
columns[i].type = FormTypes.hidden //隐藏
|
||||
} |
||||
} |
||||
} |
||||
}, |
||||
initSupplier() { |
||||
let that = this; |
||||
findBySelectSup({}).then((res)=>{ |
||||
if(res) { |
||||
that.supList = res; |
||||
// this.form.setFieldsValue({'organId': Number(that.supList[0].id)})
|
||||
} |
||||
}); |
||||
}, |
||||
initCustomer() { |
||||
let that = this; |
||||
findBySelectCus({}).then((res)=>{ |
||||
if(res) { |
||||
that.cusList = res; |
||||
} |
||||
}); |
||||
}, |
||||
initRetail() { |
||||
let that = this; |
||||
findBySelectRetail({}).then((res)=>{ |
||||
if(res) { |
||||
that.retailList = res; |
||||
} |
||||
}); |
||||
}, |
||||
initSalesman() { |
||||
let that = this; |
||||
getPersonByNumType({type:1}).then((res)=>{ |
||||
if(res) { |
||||
that.personList.options = res; |
||||
} |
||||
}); |
||||
}, |
||||
initDepot() { |
||||
let that = this; |
||||
getAction('/depot/findDepotByCurrentUser').then((res) => { |
||||
if(res.code === 200){ |
||||
let arr = res.data |
||||
for(let item of that.materialTable.columns){ |
||||
if(item.key == 'depotId' || item.key == 'anotherDepotId') { |
||||
item.options = [] |
||||
for(let i=0; i<arr.length; i++) { |
||||
let depotInfo = {}; |
||||
depotInfo.value = arr[i].id + '' //注意-此处value必须为字符串格式
|
||||
depotInfo.text = arr[i].depotName |
||||
depotInfo.title = arr[i].depotName |
||||
item.options.push(depotInfo) |
||||
} |
||||
} |
||||
} |
||||
} |
||||
}) |
||||
}, |
||||
initAccount(){ |
||||
let that = this; |
||||
getAccount({}).then((res)=>{ |
||||
if(res && res.code === 200) { |
||||
let list = res.data.accountList |
||||
list.splice(0,0,{id: 0, name: '多账户'}) |
||||
that.accountList = list |
||||
} |
||||
}) |
||||
}, |
||||
handleManyAccount(){ |
||||
this.selectAccount(0) |
||||
}, |
||||
selectAccount(value){ |
||||
if(value === 0) { //多账户
|
||||
this.$refs.manyAccountModalForm.edit(this.accountIdList, this.accountMoneyList) |
||||
this.$refs.manyAccountModalForm.title = "多账户结算" |
||||
this.manyAccountBtnStatus = true |
||||
} else { |
||||
this.accountIdList = [] |
||||
this.accountMoneyList = [] |
||||
this.manyAccountBtnStatus = false |
||||
} |
||||
}, |
||||
manyAccountModalFormOk(idList, moneyList, allPrice) { |
||||
this.accountIdList = idList |
||||
this.accountMoneyList = moneyList |
||||
let discountLastMoney = this.form.getFieldValue('discountLastMoney')-0 |
||||
let otherMoney = this.form.getFieldValue('otherMoney')-0 |
||||
let debt = (discountLastMoney + otherMoney - allPrice).toFixed(2) |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue({'changeAmount':allPrice, 'debt':debt}) |
||||
}); |
||||
}, |
||||
addSupplier() { |
||||
this.$refs.vendorModalForm.add(); |
||||
this.$refs.vendorModalForm.title = "新增供应商"; |
||||
this.$refs.vendorModalForm.disableSubmit = false; |
||||
}, |
||||
addCustomer() { |
||||
this.$refs.customerModalForm.add(); |
||||
this.$refs.customerModalForm.title = "新增客户(提醒:如果找不到新添加的客户,请到用户管理检查是否分配了该客户权限)"; |
||||
this.$refs.customerModalForm.disableSubmit = false; |
||||
}, |
||||
addMember() { |
||||
this.$refs.memberModalForm.add(); |
||||
this.$refs.memberModalForm.title = "新增会员"; |
||||
this.$refs.memberModalForm.disableSubmit = false; |
||||
}, |
||||
handleBatchSetDepot() { |
||||
this.$refs.batchSetDepotModalForm.add(); |
||||
this.$refs.batchSetDepotModalForm.title = "批量设置仓库"; |
||||
this.$refs.batchSetDepotModalForm.disableSubmit = false; |
||||
}, |
||||
addDepot() { |
||||
this.$refs.depotModalForm.add(); |
||||
this.$refs.depotModalForm.title = "新增仓库"; |
||||
this.$refs.depotModalForm.disableSubmit = false; |
||||
}, |
||||
addAccount() { |
||||
this.$refs.accountModalForm.add(); |
||||
this.$refs.accountModalForm.title = "新增结算账户"; |
||||
this.$refs.accountModalForm.disableSubmit = false; |
||||
}, |
||||
vendorModalFormOk() { |
||||
this.initSupplier() |
||||
}, |
||||
customerModalFormOk() { |
||||
this.initCustomer() |
||||
}, |
||||
memberModalFormOk() { |
||||
this.initRetail() |
||||
}, |
||||
batchSetDepotModalFormOk(depotId) { |
||||
this.getAllTable().then(tables => { |
||||
return getListData(this.form, tables) |
||||
}).then(allValues => { |
||||
//获取单据明细列表信息
|
||||
let detailArr = allValues.tablesValue[0].values |
||||
let barCodes = '' |
||||
for(let detail of detailArr){ |
||||
barCodes += detail.barCode + ',' |
||||
} |
||||
if(barCodes) { |
||||
barCodes = barCodes.substring(0, barCodes.length-1) |
||||
} |
||||
let param = { |
||||
barCode: barCodes, |
||||
depotId: depotId, |
||||
mpList: getMpListShort(Vue.ls.get('materialPropertyList')), //扩展属性
|
||||
prefixNo: this.prefixNo |
||||
} |
||||
getMaterialByBarCode(param).then((res) => { |
||||
if (res && res.code === 200) { |
||||
let mList = res.data |
||||
//构造新的列表数组,用于存放单据明细信息
|
||||
let newDetailArr = [] |
||||
if(mList && mList.length) { |
||||
for (let i = 0; i < detailArr.length; i++) { |
||||
let item = detailArr[i] |
||||
item.depotId = depotId |
||||
item.stock = mList[i] ? (mList[i].stock ? mList[i].stock : 0) : 0 |
||||
newDetailArr.push(item) |
||||
} |
||||
} else { |
||||
for (let i = 0; i < detailArr.length; i++) { |
||||
let item = detailArr[i] |
||||
item.depotId = depotId |
||||
newDetailArr.push(item) |
||||
} |
||||
} |
||||
this.materialTable.dataSource = newDetailArr |
||||
} |
||||
}) |
||||
}) |
||||
}, |
||||
depotModalFormOk() { |
||||
this.initDepot() |
||||
}, |
||||
accountModalFormOk() { |
||||
this.initAccount() |
||||
}, |
||||
onAdded(event) { |
||||
const { row, target } = event |
||||
getAction('/depot/findDepotByCurrentUser').then((res) => { |
||||
if (res.code === 200) { |
||||
let arr = res.data |
||||
for (let i = 0; i < arr.length; i++) { |
||||
if(arr[i].isDefault){ |
||||
target.setValues([{rowKey: row.id, values: {depotId: arr[i].id+''}}]) |
||||
} |
||||
} |
||||
} |
||||
}) |
||||
}, |
||||
//单元值改变一个字符就触发一次
|
||||
onValueChange(event) { |
||||
let that = this |
||||
const { type, row, column, value, target } = event |
||||
let param,snList,batchNumber,operNumber,unitPrice,allPrice,taxRate,taxMoney,taxLastMoney |
||||
switch(column.key) { |
||||
case "depotId": |
||||
if(row.barCode){ |
||||
that.getStockByDepotBarCode(row, target) |
||||
} |
||||
break; |
||||
case "barCode": |
||||
param = { |
||||
barCode: value, |
||||
mpList: getMpListShort(Vue.ls.get('materialPropertyList')), //扩展属性
|
||||
prefixNo: this.prefixNo |
||||
} |
||||
getMaterialByBarCode(param).then((res) => { |
||||
if (res && res.code === 200) { |
||||
let mList = res.data |
||||
if (value.indexOf(',') > -1) { |
||||
//多个物料编码
|
||||
this.$refs.materialDataTable.getValues((error, values) => { |
||||
values.pop() //移除最后一行数据
|
||||
let mArr = values |
||||
for (let i = 0; i < mList.length; i++) { |
||||
let mInfo = mList[i] |
||||
this.changeColumnShow(mInfo) |
||||
let mObj = this.parseInfoToObj(mInfo) |
||||
mObj.depotId = mInfo.depotId |
||||
mObj.stock = mInfo.stock |
||||
mObj.barCode = mInfo.mBarCode |
||||
mArr.push(mObj) |
||||
} |
||||
let taxLastMoneyTotal = 0 |
||||
for (let j = 0; j < mArr.length; j++) { |
||||
taxLastMoneyTotal += mArr[j].taxLastMoney-0 |
||||
//组合和拆分单据给商品类型进行重新赋值
|
||||
if(j===0) { |
||||
mArr[0].mType = '组合件' |
||||
} else { |
||||
mArr[j].mType = '普通子件' |
||||
} |
||||
} |
||||
this.materialTable.dataSource = mArr |
||||
target.statisticsColumns.taxLastMoney = taxLastMoneyTotal |
||||
that.autoChangePrice(target) |
||||
}) |
||||
} else { |
||||
//单个物料编码
|
||||
findStockByDepotAndBarCode({ depotId: row.depotId, barCode: row.barCode }).then((res) => { |
||||
if (res && res.code === 200) { |
||||
let mArr = [] |
||||
let mInfo = mList[0] |
||||
this.changeColumnShow(mInfo) |
||||
let mInfoEx = this.parseInfoToObj(mInfo) |
||||
mInfoEx.stock = res.data.stock |
||||
mInfoEx.barCode = row.barCode |
||||
let mObj = { |
||||
rowKey: row.id, |
||||
values: mInfoEx |
||||
} |
||||
mArr.push(mObj) |
||||
target.setValues(mArr); |
||||
target.recalcAllStatisticsColumns() |
||||
that.autoChangePrice(target) |
||||
target.autoSelectBySpecialKey('operNumber') |
||||
} |
||||
}) |
||||
} |
||||
} |
||||
}); |
||||
break; |
||||
case "snList": |
||||
snList = value |
||||
if(snList) { |
||||
let snArr = snList.split(',') |
||||
operNumber = snArr.length |
||||
taxRate = row.taxRate-0 //税率
|
||||
unitPrice = row.unitPrice-0 //单价
|
||||
allPrice = (unitPrice*operNumber).toFixed(2)-0 |
||||
taxMoney =((taxRate*0.01)*allPrice).toFixed(2)-0 |
||||
taxLastMoney = (allPrice + taxMoney).toFixed(2)-0 |
||||
target.setValues([{rowKey: row.id, values: {operNumber: operNumber, allPrice: allPrice, taxMoney: taxMoney, taxLastMoney: taxLastMoney}}]) |
||||
target.recalcAllStatisticsColumns() |
||||
that.autoChangePrice(target) |
||||
} |
||||
break; |
||||
case "batchNumber": |
||||
batchNumber = value |
||||
getBatchNumberList({name:'', depotId: row.depotId, barCode: row.barCode, batchNumber: batchNumber}).then((res) => { |
||||
if (res && res.code === 200) { |
||||
if(res.data && res.data.rows) { |
||||
let info = res.data.rows[0] |
||||
operNumber = info.totalNum |
||||
taxRate = row.taxRate-0 //税率
|
||||
unitPrice = row.unitPrice-0 //单价
|
||||
allPrice = (unitPrice*operNumber).toFixed(2)-0 |
||||
taxMoney =((taxRate*0.01)*allPrice).toFixed(2)-0 |
||||
taxLastMoney = (allPrice + taxMoney).toFixed(2)-0 |
||||
target.setValues([{rowKey: row.id, values: {expirationDate: info.expirationDateStr, operNumber: operNumber, |
||||
allPrice: allPrice, taxMoney: taxMoney, taxLastMoney: taxLastMoney}}]) |
||||
target.recalcAllStatisticsColumns() |
||||
that.autoChangePrice(target) |
||||
} |
||||
} |
||||
}) |
||||
break; |
||||
case "operNumber": |
||||
operNumber = value-0 |
||||
taxRate = row.taxRate-0 //税率
|
||||
unitPrice = row.unitPrice-0 //单价
|
||||
allPrice = (unitPrice*operNumber).toFixed(2)-0 |
||||
taxMoney =((taxRate*0.01)*allPrice).toFixed(2)-0 |
||||
taxLastMoney = (allPrice + taxMoney).toFixed(2)-0 |
||||
target.setValues([{rowKey: row.id, values: {allPrice: allPrice, taxMoney: taxMoney, taxLastMoney: taxLastMoney}}]) |
||||
target.recalcAllStatisticsColumns() |
||||
that.autoChangePrice(target) |
||||
break; |
||||
case "unitPrice": |
||||
operNumber = row.operNumber-0 //数量
|
||||
unitPrice = value-0 //单价
|
||||
taxRate = row.taxRate-0 //税率
|
||||
allPrice = (unitPrice*operNumber).toFixed(2)-0 |
||||
taxMoney =((taxRate*0.01)*allPrice).toFixed(2)-0 |
||||
taxLastMoney = (allPrice + taxMoney).toFixed(2)-0 |
||||
target.setValues([{rowKey: row.id, values: {allPrice: allPrice, taxMoney: taxMoney, taxLastMoney: taxLastMoney}}]) |
||||
target.recalcAllStatisticsColumns() |
||||
that.autoChangePrice(target) |
||||
break; |
||||
case "allPrice": |
||||
operNumber = row.operNumber-0 //数量
|
||||
taxRate = row.taxRate-0 //税率
|
||||
allPrice = value-0 |
||||
unitPrice = (allPrice/operNumber).toFixed(6)-0 //单价
|
||||
taxMoney =((taxRate*0.01)*allPrice).toFixed(2)-0 |
||||
taxLastMoney = (allPrice + taxMoney).toFixed(2)-0 |
||||
target.setValues([{rowKey: row.id, values: {unitPrice: unitPrice, taxMoney: taxMoney, taxLastMoney: taxLastMoney}}]) |
||||
target.recalcAllStatisticsColumns() |
||||
that.autoChangePrice(target) |
||||
break; |
||||
case "taxRate": |
||||
operNumber = row.operNumber-0 //数量
|
||||
allPrice = row.allPrice-0 |
||||
unitPrice = row.unitPrice-0 |
||||
taxRate = value-0 //税率
|
||||
taxMoney =((taxRate*0.01)*allPrice).toFixed(2)-0 |
||||
taxLastMoney = (allPrice + taxMoney).toFixed(2)-0 |
||||
target.setValues([{rowKey: row.id, values: {taxMoney: taxMoney, taxLastMoney: taxLastMoney}}]) |
||||
target.recalcAllStatisticsColumns() |
||||
that.autoChangePrice(target) |
||||
break; |
||||
case "taxLastMoney": |
||||
operNumber = row.operNumber-0 //数量
|
||||
taxLastMoney = value-0 |
||||
taxRate = row.taxRate-0 //税率
|
||||
unitPrice = (taxLastMoney/operNumber/(1+taxRate*0.01)).toFixed(6)-0 |
||||
allPrice = (unitPrice*operNumber).toFixed(2)-0 |
||||
taxMoney =(taxLastMoney-allPrice).toFixed(2)-0 |
||||
target.setValues([{rowKey: row.id, values: {unitPrice: unitPrice, allPrice: allPrice, taxMoney: taxMoney}}]) |
||||
target.recalcAllStatisticsColumns() |
||||
that.autoChangePrice(target) |
||||
break; |
||||
case "currentStock" :
|
||||
target.setValues([{rowKey: row.id, values: {operNumber: Number(value) - Number(row.stock)}}]) |
||||
break |
||||
} |
||||
if(value.split(',').length > 1) { |
||||
this.clearKh() |
||||
} |
||||
}, |
||||
//转为商品对象
|
||||
parseInfoToObj(mInfo) { |
||||
return { |
||||
barCode: mInfo.mBarCode, |
||||
name: mInfo.name, |
||||
standard: mInfo.standard, |
||||
model: mInfo.model, |
||||
color: mInfo.color, |
||||
materialOther: mInfo.materialOther, |
||||
unit: mInfo.commodityUnit, |
||||
sku: mInfo.sku, |
||||
operNumber: 1, |
||||
unitPrice: mInfo.billPrice, |
||||
allPrice: mInfo.billPrice, |
||||
taxRate: 0, |
||||
taxMoney: 0, |
||||
taxLastMoney: mInfo.billPrice |
||||
} |
||||
}, |
||||
//使得型号、颜色、扩展信息、sku等为隐藏
|
||||
changeColumnHide() { |
||||
this.changeFormTypes(this.materialTable.columns, 'model', 0) |
||||
this.changeFormTypes(this.materialTable.columns, 'color', 0) |
||||
this.changeFormTypes(this.materialTable.columns, 'materialOther', 0) |
||||
this.changeFormTypes(this.materialTable.columns, 'sku', 0) |
||||
}, |
||||
//使得sku、序列号、批号、到期日等为显示
|
||||
changeColumnShow(info) { |
||||
if(info.model) { |
||||
this.changeFormTypes(this.materialTable.columns, 'model', 1) |
||||
} |
||||
if(info.color) { |
||||
this.changeFormTypes(this.materialTable.columns, 'color', 1) |
||||
} |
||||
if(info.materialOther) { |
||||
this.changeFormTypes(this.materialTable.columns, 'materialOther', 1) |
||||
} |
||||
if(info.sku) { |
||||
this.changeFormTypes(this.materialTable.columns, 'sku', 1) |
||||
} |
||||
if(info.enableSerialNumber === "1") { |
||||
this.changeFormTypes(this.materialTable.columns, 'snList', 1) |
||||
} |
||||
if(info.enableBatchNumber === "1") { |
||||
this.changeFormTypes(this.materialTable.columns, 'batchNumber', 1) |
||||
this.changeFormTypes(this.materialTable.columns, 'expirationDate', 1) |
||||
} |
||||
}, |
||||
//删除一行或多行的时候触发
|
||||
onDeleted(ids, target) { |
||||
target.recalcAllStatisticsColumns() |
||||
this.autoChangePrice(target) |
||||
}, |
||||
//根据仓库和物料编码查询库存
|
||||
getStockByDepotBarCode(row, target){ |
||||
findStockByDepotAndBarCode({ depotId: row.depotId, barCode: row.barCode }).then((res) => { |
||||
if (res && res.code === 200) { |
||||
console.log(res.data.stock,'res.data.stock') |
||||
target.setValues([{rowKey: row.id, values: {stock: res.data.stock}}]) |
||||
target.recalcAllStatisticsColumns() |
||||
} |
||||
}) |
||||
}, |
||||
//改变优惠、本次付款、欠款的值
|
||||
autoChangePrice(target) { |
||||
let allTaxLastMoney = target.statisticsColumns.taxLastMoney-0 |
||||
let discount = this.form.getFieldValue('discount')-0 |
||||
let otherMoney = this.form.getFieldValue('otherMoney')-0 |
||||
let discountMoney = (discount*0.01*allTaxLastMoney).toFixed(2)-0 |
||||
let discountLastMoney = (allTaxLastMoney-discountMoney).toFixed(2)-0 |
||||
let changeAmountNew = (discountLastMoney + otherMoney).toFixed(2)-0 |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue({'discount':discount,'discountMoney':discountMoney,'discountLastMoney':discountLastMoney, |
||||
'changeAmount':changeAmountNew,'debt':0}) |
||||
}); |
||||
}, |
||||
//改变优惠率
|
||||
onKeyUpDiscount(e) { |
||||
const value = e.target.value-0 |
||||
let discountMoney = this.form.getFieldValue('discountMoney')-0 |
||||
let discountLastMoney = this.form.getFieldValue('discountLastMoney')-0 |
||||
let otherMoney = this.form.getFieldValue('otherMoney')-0 |
||||
let allTaxLastMoney = (discountMoney + discountLastMoney).toFixed(2)-0 |
||||
let discountMoneyNew = (allTaxLastMoney*value*0.01).toFixed(2)-0 |
||||
let discountLastMoneyNew = (allTaxLastMoney - discountMoneyNew).toFixed(2)-0 |
||||
let changeAmountNew = (discountLastMoneyNew + otherMoney).toFixed(2)-0 |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue({'discountMoney':discountMoneyNew,'discountLastMoney':discountLastMoneyNew, |
||||
'changeAmount':changeAmountNew,'debt':0}) |
||||
}); |
||||
}, |
||||
//改变付款优惠
|
||||
onKeyUpDiscountMoney(e) { |
||||
const value = e.target.value-0 |
||||
let discount = this.form.getFieldValue('discount')-0 |
||||
let discountLastMoney = this.form.getFieldValue('discountLastMoney')-0 |
||||
let otherMoney = this.form.getFieldValue('otherMoney')-0 |
||||
if(discount !== 100) { |
||||
let allTaxLastMoney = (discountLastMoney/(1-discount/100)).toFixed(2)-0 |
||||
let discountNew = (value/allTaxLastMoney*100).toFixed(2)-0 |
||||
let discountLastMoneyNew = (allTaxLastMoney - value).toFixed(2)-0 |
||||
let changeAmountNew = (discountLastMoneyNew + otherMoney).toFixed(2)-0 |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue({'discount':discountNew,'discountLastMoney':discountLastMoneyNew, |
||||
'changeAmount':changeAmountNew,'debt':0}) |
||||
}); |
||||
} |
||||
}, |
||||
//其它费用
|
||||
onKeyUpOtherMoney(e) { |
||||
const value = e.target.value-0 |
||||
let discountLastMoney = this.form.getFieldValue('discountLastMoney')-0 |
||||
let changeAmountNew = (discountLastMoney + value).toFixed(2)-0 |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue({'changeAmount':changeAmountNew, 'debt':0}) |
||||
}); |
||||
}, |
||||
//改变本次付款
|
||||
onKeyUpChangeAmount(e) { |
||||
const value = e.target.value-0 |
||||
let discountLastMoney = this.form.getFieldValue('discountLastMoney')-0 |
||||
let otherMoney = this.form.getFieldValue('otherMoney')-0 |
||||
let debtNew = (discountLastMoney + otherMoney - value).toFixed(2)-0 |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue({'debt':debtNew}) |
||||
}); |
||||
}, |
||||
scanEnter() { |
||||
this.scanStatus = false |
||||
this.$nextTick(() => { |
||||
this.$refs.scanBarCode.focus() |
||||
}) |
||||
}, |
||||
//扫码之后回车
|
||||
scanPressEnter() { |
||||
if(this.scanBarCode) { |
||||
this.getAllTable().then(tables => { |
||||
return getListData(this.form, tables) |
||||
}).then(allValues => { |
||||
let param = { |
||||
barCode: this.scanBarCode, |
||||
mpList: getMpListShort(Vue.ls.get('materialPropertyList')), //扩展属性
|
||||
prefixNo: this.prefixNo |
||||
} |
||||
getMaterialByBarCode(param).then((res) => { |
||||
if (res && res.code === 200) { |
||||
let hasFinished = false |
||||
let allLastMoney = 0 |
||||
let allTaxLastMoney = 0 |
||||
//获取单据明细列表信息
|
||||
let detailArr = allValues.tablesValue[0].values |
||||
//构造新的列表数组,用于存放单据明细信息
|
||||
let newDetailArr = [] |
||||
for(let detail of detailArr){ |
||||
if(detail.barCode) { |
||||
//如果物料编码重复,就在给原来的数量加1
|
||||
if(detail.barCode === this.scanBarCode) { |
||||
detail.operNumber = (detail.operNumber-0)+1 |
||||
//由于改变了商品数量,需要同时更新相关金额和价税合计
|
||||
let taxRate = detail.taxRate-0 //税率
|
||||
let unitPrice = detail.unitPrice-0 //单价
|
||||
detail.allPrice = (unitPrice*detail.operNumber).toFixed(2)-0 |
||||
detail.taxMoney = ((taxRate*0.01)*detail.allPrice).toFixed(2)-0 |
||||
detail.taxLastMoney = (detail.allPrice + detail.taxMoney).toFixed(2)-0 |
||||
hasFinished = true |
||||
} |
||||
newDetailArr.push(detail) |
||||
} |
||||
} |
||||
if(!hasFinished) { |
||||
//将扫码的物料编码对应的商品加入列表
|
||||
let item = {} |
||||
item.barCode = this.scanBarCode |
||||
let mList = res.data |
||||
if(mList && mList.length>0) { |
||||
let mInfo = mList[0] |
||||
this.changeColumnShow(mInfo) |
||||
item.depotId = mInfo.depotId |
||||
item.name = mInfo.name |
||||
item.standard = mInfo.standard |
||||
item.model = mInfo.model |
||||
item.color = mInfo.color |
||||
item.materialOther = mInfo.materialOther |
||||
item.stock = mInfo.stock |
||||
item.unit = mInfo.commodityUnit |
||||
item.sku = mInfo.sku |
||||
item.operNumber = 1 |
||||
item.unitPrice = mInfo.billPrice |
||||
item.allPrice = mInfo.billPrice |
||||
item.taxRate = 0 |
||||
item.taxMoney = 0 |
||||
item.taxLastMoney = mInfo.billPrice |
||||
newDetailArr.push(item) |
||||
} else { |
||||
this.$message.warning('抱歉,此物料编码不存在商品信息!'); |
||||
} |
||||
} |
||||
//组合和拆分单据给商品类型进行重新赋值
|
||||
for(let i=0; i< newDetailArr.length; i++) { |
||||
if(i===0) { |
||||
newDetailArr[0].mType = '组合件' |
||||
} else { |
||||
newDetailArr[i].mType = '普通子件' |
||||
} |
||||
} |
||||
this.materialTable.dataSource = newDetailArr |
||||
//更新优惠后金额、本次付款等信息
|
||||
for(let newDetail of newDetailArr){ |
||||
allLastMoney = allLastMoney + (newDetail.allPrice-0) |
||||
allTaxLastMoney = allTaxLastMoney + (newDetail.taxLastMoney-0) |
||||
} |
||||
let discount = this.form.getFieldValue('discount')-0 |
||||
let otherMoney = this.form.getFieldValue('otherMoney')-0 |
||||
let discountMoney = (discount*0.01*allTaxLastMoney).toFixed(2)-0 |
||||
let discountLastMoney = (allTaxLastMoney-discountMoney).toFixed(2)-0 |
||||
let changeAmountNew = (discountLastMoney + otherMoney).toFixed(2)-0 |
||||
if(this.prefixNo === 'LSCK' || this.prefixNo === 'LSTH') { |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue({'changeAmount':allLastMoney,'getAmount':allLastMoney,'backAmount':0}) |
||||
}); |
||||
} else { |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue({'discount':discount,'discountMoney':discountMoney,'discountLastMoney':discountLastMoney, |
||||
'changeAmount':changeAmountNew,'debt':0}) |
||||
}); |
||||
} |
||||
//置空扫码的内容
|
||||
this.scanBarCode = '' |
||||
this.$refs.scanBarCode.focus() |
||||
} |
||||
}) |
||||
}) |
||||
} |
||||
}, |
||||
stopScan() { |
||||
this.scanStatus = true |
||||
this.scanBarCode = '' |
||||
}, |
||||
//清空空行
|
||||
clearKh() { |
||||
this.getAllTable().then(tables => { |
||||
let inputValues = tables[0].inputValues
|
||||
let ids = []
|
||||
inputValues.forEach((item) => { |
||||
if(!item.barCode && !item.operNumber) { |
||||
ids.push(item.id) |
||||
} |
||||
}) |
||||
tables[0].removeRows(ids) |
||||
}) |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,239 @@
|
||||
<template> |
||||
<j-modal |
||||
:title="title" |
||||
:width="width" |
||||
:visible="visible" |
||||
:confirmLoading="confirmLoading" |
||||
:maskClosable="false" |
||||
:keyboard="false" |
||||
:forceRender="true" |
||||
switchFullscreen |
||||
@ok="handleOk" |
||||
@cancel="handleCancel" |
||||
wrapClassName="ant-modal-cust-warp" |
||||
style="top:5%;height: 100%;overflow-y: hidden"> |
||||
<a-spin :spinning="confirmLoading"> |
||||
<a-form :form="form"> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="单据日期"> |
||||
<j-date v-decorator="['operTime', validatorRules.operTime]" :show-time="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="单据编号"> |
||||
<a-input placeholder="请输入单据编号" v-decorator.trim="[ 'number' ]" :readOnly="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"></a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"></a-col> |
||||
</a-row> |
||||
<j-editable-table id="billModal" |
||||
:ref="refKeys[0]" |
||||
:loading="materialTable.loading" |
||||
:columns="materialTable.columns" |
||||
:dataSource="materialTable.dataSource" |
||||
:maxHeight="300" |
||||
:rowNumber="false" |
||||
:rowSelection="true" |
||||
:actionButton="true" |
||||
:dragSort="true" |
||||
@valueChange="onValueChange" |
||||
@added="onAdded" |
||||
@deleted="onDeleted"> |
||||
<template #buttonAfter> |
||||
<a-row :gutter="24" style="float:left;" data-step="4" data-title="扫码录入" data-intro="此功能支持扫码枪扫描商品条码进行录入"> |
||||
<a-col v-if="scanStatus" :md="6" :sm="24"> |
||||
<a-button @click="scanEnter">扫码录入</a-button> |
||||
</a-col> |
||||
<a-col v-if="!scanStatus" :md="16" :sm="24" style="padding: 0 6px 0 12px"> |
||||
<a-input placeholder="请扫码商品条码并回车" v-model="scanBarCode" @pressEnter="scanPressEnter" ref="scanBarCode"/> |
||||
</a-col> |
||||
<a-col v-if="!scanStatus" :md="6" :sm="24" style="padding: 0px"> |
||||
<a-button @click="stopScan">收起扫码</a-button> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row :gutter="24" style="float:left;"> |
||||
<a-col :md="24" :sm="24"> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" @click="handleBatchSetDepot"><a-icon type="setting"/>批量设置</a-menu-item> |
||||
<a-menu-item v-if="isTenant" key="2" @click="addDepot"><a-icon type="plus"/>新增仓库</a-menu-item> |
||||
</a-menu> |
||||
<a-button style="margin-left: 8px">仓库操作 <a-icon type="down" /></a-button> |
||||
</a-dropdown> |
||||
</a-col> |
||||
</a-row> |
||||
<a-button type="primary" @click="clearKh" style="margin-left:10px">清空空行</a-button> |
||||
</template> |
||||
</j-editable-table> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="24" :md="24" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="{xs: { span: 24 },sm: { span: 24 }}" label=""> |
||||
<a-textarea :rows="1" placeholder="请输入备注" v-decorator="[ 'remark' ]" style="margin-top:8px;"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="附件"> |
||||
<j-upload v-model="fileList" bizPath="bill"></j-upload> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
</a-form> |
||||
</a-spin> |
||||
<depot-modal ref="depotModalForm" @ok="depotModalFormOk"></depot-modal> |
||||
<batch-set-depot ref="batchSetDepotModalForm" @ok="batchSetDepotModalFormOk"></batch-set-depot> |
||||
</j-modal> |
||||
</template> |
||||
<script> |
||||
import pick from 'lodash.pick' |
||||
import DepotModal from '../../system/modules/DepotModal' |
||||
import BatchSetDepot from '../dialog/BatchSetDepot' |
||||
import { FormTypes } from '@/utils/JEditableTableUtil' |
||||
import { JEditableTableMixin } from '@/mixins/JEditableTableMixin' |
||||
import { BillModalMixin } from '../mixins/BillModalMixin' |
||||
import { getMpListShort } from "@/utils/util" |
||||
import JUpload from '@/components/jeecg/JUpload' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
export default { |
||||
name: "AllocationOutModal", |
||||
mixins: [JEditableTableMixin, BillModalMixin], |
||||
components: { |
||||
DepotModal, |
||||
BatchSetDepot, |
||||
JUpload, |
||||
JDate |
||||
}, |
||||
data () { |
||||
return { |
||||
title:"操作", |
||||
width: '100%', |
||||
moreStatus: false, |
||||
// 新增时子表默认添加几行空数据 |
||||
addDefaultRowNum: 10, |
||||
visible: false, |
||||
operTimeStr: '', |
||||
prefixNo: 'DBCK', |
||||
fileList:[], |
||||
model: {}, |
||||
labelCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 8 }, |
||||
}, |
||||
wrapperCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 16 }, |
||||
}, |
||||
refKeys: ['materialDataTable', ], |
||||
activeKey: 'materialDataTable', |
||||
materialTable: { |
||||
loading: false, |
||||
dataSource: [], |
||||
columns: [ |
||||
{ title: '仓库名称', key: 'depotId', width: '7%', type: FormTypes.select, placeholder: '请选择${title}', options: [], |
||||
allowSearch:true, validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '物料编码', key: 'barCode', width: '8%', type: FormTypes.popupJsh, kind: 'material', multi: true, |
||||
validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '名称', key: 'name', width: '20%', type: FormTypes.normal }, |
||||
{ title: '规格', key: 'standard', width: '5%', type: FormTypes.normal }, |
||||
{ title: '型号', key: 'model', width: '5%', type: FormTypes.normal }, |
||||
{ title: '颜色', key: 'color', width: '5%', type: FormTypes.normal }, |
||||
{ title: '扩展信息', key: 'materialOther', width: '5%', type: FormTypes.normal }, |
||||
{ title: '库存', key: 'stock', width: '5%', type: FormTypes.inputNumber, readonly:true }, |
||||
{ title: '调入仓库', key: 'anotherDepotId', width: '7%', type: FormTypes.select, placeholder: '请选择${title}', options: [], allowSearch:true}, |
||||
{ title: '单位', key: 'unit', width: '4%', type: FormTypes.normal }, |
||||
{ title: '多属性', key: 'sku', width: '4%', type: FormTypes.normal }, |
||||
{ title: '数量', key: 'operNumber', width: '5%', type: FormTypes.inputNumber, statistics: true, |
||||
validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '单价', key: 'unitPrice', width: '5%', type: FormTypes.inputNumber}, |
||||
{ title: '金额', key: 'allPrice', width: '5%', type: FormTypes.inputNumber, statistics: true }, |
||||
{ title: '备注', key: 'remark', width: '5%', type: FormTypes.input } |
||||
] |
||||
}, |
||||
confirmLoading: false, |
||||
validatorRules:{ |
||||
operTime:{ |
||||
rules: [ |
||||
{ required: true, message: '请输入单据日期!' } |
||||
] |
||||
}, |
||||
type:{ |
||||
rules: [ |
||||
{ required: true, message: '请选择类型!' } |
||||
] |
||||
} |
||||
}, |
||||
url: { |
||||
add: '/depotHead/addDepotHeadAndDetail', |
||||
edit: '/depotHead/updateDepotHeadAndDetail', |
||||
detailList: '/depotItem/getDetailList' |
||||
} |
||||
} |
||||
}, |
||||
created () { |
||||
}, |
||||
methods: { |
||||
//调用完edit()方法之后会自动调用此方法 |
||||
editAfter() { |
||||
this.changeColumnHide() |
||||
if (this.action === 'add') { |
||||
this.addInit(this.prefixNo) |
||||
this.fileList = [] |
||||
} else { |
||||
this.model.operTime = this.model.operTimeStr |
||||
this.fileList = this.model.fileName |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue(pick(this.model,'organId', 'operTime', 'number', 'remark', |
||||
'discount','discountMoney','discountLastMoney','otherMoney','accountId','changeAmount')) |
||||
}); |
||||
// 加载子表数据 |
||||
let params = { |
||||
headerId: this.model.id, |
||||
mpList: getMpListShort(Vue.ls.get('materialPropertyList')) //扩展属性 |
||||
} |
||||
let url = this.readOnly ? this.url.detailList : this.url.detailList; |
||||
this.requestSubTableData(url, params, this.materialTable); |
||||
} |
||||
//复制新增单据-初始化单号和日期 |
||||
if(this.action === 'copyAdd') { |
||||
this.model.id = '' |
||||
this.model.tenantId = '' |
||||
this.copyAddInit(this.prefixNo) |
||||
} |
||||
this.initDepot() |
||||
}, |
||||
//提交单据时整理成formData |
||||
classifyIntoFormData(allValues) { |
||||
let totalPrice = 0 |
||||
let billMain = Object.assign(this.model, allValues.formValue) |
||||
let detailArr = allValues.tablesValue[0].values |
||||
billMain.type = '出库' |
||||
billMain.subType = '调拨' |
||||
billMain.defaultNumber = billMain.number |
||||
for(let item of detailArr){ |
||||
totalPrice += item.allPrice-0 |
||||
} |
||||
billMain.totalPrice = totalPrice |
||||
if(this.fileList && this.fileList.length > 0) { |
||||
billMain.fileName = this.fileList |
||||
} |
||||
if(this.model.id){ |
||||
billMain.id = this.model.id |
||||
} |
||||
return { |
||||
info: billMain, |
||||
rows: detailArr, |
||||
} |
||||
}, |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,258 @@
|
||||
<template> |
||||
<j-modal |
||||
:title="title" |
||||
:width="width" |
||||
:visible="visible" |
||||
:confirmLoading="confirmLoading" |
||||
:maskClosable="false" |
||||
:keyboard="false" |
||||
:forceRender="true" |
||||
switchFullscreen |
||||
@ok="handleOk" |
||||
@cancel="handleCancel" |
||||
wrapClassName="ant-modal-cust-warp" |
||||
style="top:5%;height: 100%;overflow-y: hidden"> |
||||
<a-spin :spinning="confirmLoading"> |
||||
<a-form :form="form"> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="单据日期"> |
||||
<j-date v-decorator="['operTime', validatorRules.operTime]" :show-time="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="单据编号"> |
||||
<a-input placeholder="请输入单据编号" v-decorator.trim="[ 'number' ]" :readOnly="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"></a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"></a-col> |
||||
</a-row> |
||||
<j-editable-table id="billModal" |
||||
:ref="refKeys[0]" |
||||
:loading="materialTable.loading" |
||||
:columns="materialTable.columns" |
||||
:dataSource="materialTable.dataSource" |
||||
:maxHeight="300" |
||||
:rowNumber="false" |
||||
:rowSelection="true" |
||||
:actionButton="true" |
||||
@valueChange="onValueChange" |
||||
@added="onAdded" |
||||
@deleted="onDeleted"> |
||||
<template #buttonAfter> |
||||
<a-row :gutter="24" style="float:left;" data-step="4" data-title="扫码录入" data-intro="此功能支持扫码枪扫描商品条码进行录入"> |
||||
<a-col v-if="scanStatus" :md="6" :sm="24"> |
||||
<a-button @click="scanEnter">扫码录入</a-button> |
||||
</a-col> |
||||
<a-col v-if="!scanStatus" :md="16" :sm="24" style="padding: 0 6px 0 12px"> |
||||
<a-input placeholder="请扫码商品条码并回车" v-model="scanBarCode" @pressEnter="scanPressEnter" ref="scanBarCode"/> |
||||
</a-col> |
||||
<a-col v-if="!scanStatus" :md="6" :sm="24" style="padding: 0px"> |
||||
<a-button @click="stopScan">收起扫码</a-button> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row :gutter="24" style="float:left;"> |
||||
<a-col :md="24" :sm="24"> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" @click="handleBatchSetDepot"><a-icon type="setting"/>批量设置</a-menu-item> |
||||
<a-menu-item v-if="isTenant" key="2" @click="addDepot"><a-icon type="plus"/>新增仓库</a-menu-item> |
||||
</a-menu> |
||||
<a-button style="margin-left: 8px">仓库操作 <a-icon type="down" /></a-button> |
||||
</a-dropdown> |
||||
</a-col> |
||||
</a-row> |
||||
<a-button type="primary" @click="clearKh" style="margin-left:10px">清空空行</a-button> |
||||
</template> |
||||
</j-editable-table> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="24" :md="24" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="{xs: { span: 24 },sm: { span: 24 }}" label=""> |
||||
<a-textarea :rows="1" placeholder="请输入备注" v-decorator="[ 'remark' ]" style="margin-top:8px;"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="附件"> |
||||
<j-upload v-model="fileList" bizPath="bill"></j-upload> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
</a-form> |
||||
</a-spin> |
||||
<depot-modal ref="depotModalForm" @ok="depotModalFormOk"></depot-modal> |
||||
<batch-set-depot ref="batchSetDepotModalForm" @ok="batchSetDepotModalFormOk"></batch-set-depot> |
||||
</j-modal> |
||||
</template> |
||||
<script> |
||||
import pick from 'lodash.pick' |
||||
import DepotModal from '../../system/modules/DepotModal' |
||||
import BatchSetDepot from '../dialog/BatchSetDepot' |
||||
import { FormTypes } from '@/utils/JEditableTableUtil' |
||||
import { JEditableTableMixin } from '@/mixins/JEditableTableMixin' |
||||
import { BillModalMixin } from '../mixins/BillModalMixin' |
||||
import { getAction } from '@/api/manage' |
||||
import { getMpListShort } from "@/utils/util" |
||||
import JUpload from '@/components/jeecg/JUpload' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
export default { |
||||
name: "AssembleModal", |
||||
mixins: [JEditableTableMixin, BillModalMixin], |
||||
components: { |
||||
DepotModal, |
||||
BatchSetDepot, |
||||
JUpload, |
||||
JDate |
||||
}, |
||||
data () { |
||||
return { |
||||
title:"操作", |
||||
width: '100%', |
||||
moreStatus: false, |
||||
// 新增时子表默认添加几行空数据 |
||||
addDefaultRowNum: 10, |
||||
visible: false, |
||||
operTimeStr: '', |
||||
prefixNo: 'ZZD', |
||||
fileList:[], |
||||
model: {}, |
||||
labelCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 8 }, |
||||
}, |
||||
wrapperCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 16 }, |
||||
}, |
||||
refKeys: ['materialDataTable', ], |
||||
activeKey: 'materialDataTable', |
||||
materialTable: { |
||||
loading: false, |
||||
dataSource: [], |
||||
columns: [ |
||||
{ title: '商品类型',key: 'mType',width:'7%', type: FormTypes.normal }, |
||||
{ title: '仓库名称', key: 'depotId', width: '7%', type: FormTypes.select, placeholder: '请选择${title}', options: [], |
||||
allowSearch:true, validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '物料编码', key: 'barCode', width: '8%', type: FormTypes.popupJsh, kind: 'material', multi: true, |
||||
validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '名称', key: 'name', width: '20%', type: FormTypes.normal }, |
||||
{ title: '规格', key: 'standard', width: '5%', type: FormTypes.normal }, |
||||
{ title: '型号', key: 'model', width: '5%', type: FormTypes.normal }, |
||||
{ title: '颜色', key: 'color', width: '5%', type: FormTypes.normal }, |
||||
{ title: '扩展信息', key: 'materialOther', width: '5%', type: FormTypes.normal }, |
||||
{ title: '库存', key: 'stock', width: '5%', type: FormTypes.inputNumber, readonly:true}, |
||||
{ title: '单位', key: 'unit', width: '4%', type: FormTypes.normal }, |
||||
{ title: '多属性', key: 'sku', width: '4%', type: FormTypes.normal }, |
||||
{ title: '数量', key: 'operNumber', width: '5%', type: FormTypes.inputNumber, statistics: true, |
||||
validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '单价', key: 'unitPrice', width: '5%', type: FormTypes.inputNumber}, |
||||
{ title: '金额', key: 'allPrice', width: '5%', type: FormTypes.inputNumber, statistics: true }, |
||||
{ title: '备注', key: 'remark', width: '5%', type: FormTypes.input } |
||||
] |
||||
}, |
||||
confirmLoading: false, |
||||
validatorRules:{ |
||||
operTime:{ |
||||
rules: [ |
||||
{ required: true, message: '请输入单据日期!' } |
||||
] |
||||
}, |
||||
type:{ |
||||
rules: [ |
||||
{ required: true, message: '请选择类型!' } |
||||
] |
||||
} |
||||
}, |
||||
url: { |
||||
add: '/depotHead/addDepotHeadAndDetail', |
||||
edit: '/depotHead/updateDepotHeadAndDetail', |
||||
detailList: '/depotItem/getDetailList' |
||||
} |
||||
} |
||||
}, |
||||
created () { |
||||
}, |
||||
methods: { |
||||
//调用完edit()方法之后会自动调用此方法 |
||||
editAfter() { |
||||
this.changeColumnHide() |
||||
if (this.action === 'add') { |
||||
this.addInit(this.prefixNo) |
||||
this.fileList = [] |
||||
} else { |
||||
this.model.operTime = this.model.operTimeStr |
||||
this.model.debt = (this.model.discountLastMoney - this.model.changeAmount).toFixed(2) |
||||
this.fileList = this.model.fileName |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue(pick(this.model,'organId', 'operTime', 'number', 'remark', |
||||
'discount','discountMoney','discountLastMoney','otherMoney','accountId','changeAmount','debt')) |
||||
}); |
||||
// 加载子表数据 |
||||
let params = { |
||||
headerId: this.model.id, |
||||
mpList: getMpListShort(Vue.ls.get('materialPropertyList')) //扩展属性 |
||||
} |
||||
let url = this.readOnly ? this.url.detailList : this.url.detailList; |
||||
this.requestSubTableData(url, params, this.materialTable); |
||||
} |
||||
//复制新增单据-初始化单号和日期 |
||||
if(this.action === 'copyAdd') { |
||||
this.model.id = '' |
||||
this.model.tenantId = '' |
||||
this.copyAddInit(this.prefixNo) |
||||
} |
||||
this.initDepot() |
||||
}, |
||||
//提交单据时整理成formData |
||||
classifyIntoFormData(allValues) { |
||||
let totalPrice = 0 |
||||
let billMain = Object.assign(this.model, allValues.formValue) |
||||
let detailArr = allValues.tablesValue[0].values |
||||
billMain.type = '其它' |
||||
billMain.subType = '组装单' |
||||
billMain.defaultNumber = billMain.number |
||||
for(let item of detailArr){ |
||||
totalPrice += item.allPrice-0 |
||||
} |
||||
billMain.totalPrice = totalPrice |
||||
if(this.fileList && this.fileList.length > 0) { |
||||
billMain.fileName = this.fileList |
||||
} |
||||
if(this.model.id){ |
||||
billMain.id = this.model.id |
||||
} |
||||
return { |
||||
info: billMain, |
||||
rows: detailArr, |
||||
} |
||||
}, |
||||
onAdded(event) { |
||||
const { row, target } = event |
||||
getAction('/depot/findDepotByCurrentUser').then((res) => { |
||||
if (res.code === 200) { |
||||
let arr = res.data |
||||
for (let i = 0; i < arr.length; i++) { |
||||
if(arr[i].isDefault){ |
||||
target.setValues([{rowKey: row.id, values: {depotId: arr[i].id+''}}]) |
||||
} |
||||
} |
||||
} |
||||
}) |
||||
if(target.rows.length>=2) { |
||||
target.setValues([{rowKey: row.id, values: {mType: '普通子件'}}]) |
||||
} else { |
||||
target.setValues([{rowKey: row.id, values: {mType: '组合件'}}]) |
||||
} |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,257 @@
|
||||
<template> |
||||
<j-modal |
||||
:title="title" |
||||
:width="width" |
||||
:visible="visible" |
||||
:confirmLoading="confirmLoading" |
||||
:maskClosable="false" |
||||
:keyboard="false" |
||||
:forceRender="true" |
||||
switchFullscreen |
||||
@ok="handleOk" |
||||
@cancel="handleCancel" |
||||
wrapClassName="ant-modal-cust-warp" |
||||
style="top:5%;height: 100%;overflow-y: hidden"> |
||||
<a-spin :spinning="confirmLoading"> |
||||
<a-form :form="form"> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="单据日期"> |
||||
<j-date v-decorator="['operTime', validatorRules.operTime]" :show-time="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="单据编号"> |
||||
<a-input placeholder="请输入单据编号" v-decorator.trim="[ 'number' ]" :readOnly="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"></a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"></a-col> |
||||
</a-row> |
||||
<j-editable-table id="billModal" |
||||
:ref="refKeys[0]" |
||||
:loading="materialTable.loading" |
||||
:columns="materialTable.columns" |
||||
:dataSource="materialTable.dataSource" |
||||
:maxHeight="300" |
||||
:rowNumber="false" |
||||
:rowSelection="true" |
||||
:actionButton="true" |
||||
@valueChange="onValueChange" |
||||
@added="onAdded" |
||||
@deleted="onDeleted"> |
||||
<template #buttonAfter> |
||||
<a-row :gutter="24" style="float:left;" data-step="4" data-title="扫码录入" data-intro="此功能支持扫码枪扫描商品条码进行录入"> |
||||
<a-col v-if="scanStatus" :md="6" :sm="24"> |
||||
<a-button @click="scanEnter">扫码录入</a-button> |
||||
</a-col> |
||||
<a-col v-if="!scanStatus" :md="16" :sm="24" style="padding: 0 6px 0 12px"> |
||||
<a-input placeholder="请扫码商品条码并回车" v-model="scanBarCode" @pressEnter="scanPressEnter" ref="scanBarCode"/> |
||||
</a-col> |
||||
<a-col v-if="!scanStatus" :md="6" :sm="24" style="padding: 0px"> |
||||
<a-button @click="stopScan">收起扫码</a-button> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row :gutter="24" style="float:left;"> |
||||
<a-col :md="24" :sm="24"> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" @click="handleBatchSetDepot"><a-icon type="setting"/>批量设置</a-menu-item> |
||||
<a-menu-item v-if="isTenant" key="2" @click="addDepot"><a-icon type="plus"/>新增仓库</a-menu-item> |
||||
</a-menu> |
||||
<a-button style="margin-left: 8px">仓库操作 <a-icon type="down" /></a-button> |
||||
</a-dropdown> |
||||
</a-col> |
||||
</a-row> |
||||
<a-button type="primary" @click="clearKh" style="margin-left:10px">清空空行</a-button> |
||||
</template> |
||||
</j-editable-table> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="24" :md="24" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="{xs: { span: 24 },sm: { span: 24 }}" label=""> |
||||
<a-textarea :rows="1" placeholder="请输入备注" v-decorator="[ 'remark' ]" style="margin-top:8px;"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="附件"> |
||||
<j-upload v-model="fileList" bizPath="bill"></j-upload> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
</a-form> |
||||
</a-spin> |
||||
<depot-modal ref="depotModalForm" @ok="depotModalFormOk"></depot-modal> |
||||
<batch-set-depot ref="batchSetDepotModalForm" @ok="batchSetDepotModalFormOk"></batch-set-depot> |
||||
</j-modal> |
||||
</template> |
||||
<script> |
||||
import pick from 'lodash.pick' |
||||
import DepotModal from '../../system/modules/DepotModal' |
||||
import BatchSetDepot from '../dialog/BatchSetDepot' |
||||
import { FormTypes } from '@/utils/JEditableTableUtil' |
||||
import { JEditableTableMixin } from '@/mixins/JEditableTableMixin' |
||||
import { BillModalMixin } from '../mixins/BillModalMixin' |
||||
import { getAction } from '@/api/manage' |
||||
import { getMpListShort } from "@/utils/util" |
||||
import JUpload from '@/components/jeecg/JUpload' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
export default { |
||||
name: "DisassembleModal", |
||||
mixins: [JEditableTableMixin, BillModalMixin], |
||||
components: { |
||||
DepotModal, |
||||
BatchSetDepot, |
||||
JUpload, |
||||
JDate |
||||
}, |
||||
data () { |
||||
return { |
||||
title:"操作", |
||||
width: '100%', |
||||
moreStatus: false, |
||||
// 新增时子表默认添加几行空数据 |
||||
addDefaultRowNum: 10, |
||||
visible: false, |
||||
operTimeStr: '', |
||||
prefixNo: 'CXD', |
||||
fileList:[], |
||||
model: {}, |
||||
labelCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 8 }, |
||||
}, |
||||
wrapperCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 16 }, |
||||
}, |
||||
refKeys: ['materialDataTable', ], |
||||
activeKey: 'materialDataTable', |
||||
materialTable: { |
||||
loading: false, |
||||
dataSource: [], |
||||
columns: [ |
||||
{ title: '商品类型',key: 'mType',width:'7%', type: FormTypes.normal }, |
||||
{ title: '仓库名称', key: 'depotId', width: '7%', type: FormTypes.select, placeholder: '请选择${title}', options: [], |
||||
allowSearch:true, validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '物料编码', key: 'barCode', width: '8%', type: FormTypes.popupJsh, kind: 'material', multi: true, |
||||
validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '名称', key: 'name', width: '20%', type: FormTypes.normal }, |
||||
{ title: '规格', key: 'standard', width: '5%', type: FormTypes.normal }, |
||||
{ title: '型号', key: 'model', width: '5%', type: FormTypes.normal }, |
||||
{ title: '颜色', key: 'color', width: '5%', type: FormTypes.normal }, |
||||
{ title: '扩展信息', key: 'materialOther', width: '5%', type: FormTypes.normal }, |
||||
{ title: '库存', key: 'stock', width: '5%', type: FormTypes.inputNumber, readonly:true}, |
||||
{ title: '单位', key: 'unit', width: '4%', type: FormTypes.normal }, |
||||
{ title: '多属性', key: 'sku', width: '4%', type: FormTypes.normal }, |
||||
{ title: '数量', key: 'operNumber', width: '5%', type: FormTypes.inputNumber, statistics: true, |
||||
validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '单价', key: 'unitPrice', width: '5%', type: FormTypes.inputNumber}, |
||||
{ title: '金额', key: 'allPrice', width: '5%', type: FormTypes.inputNumber, statistics: true }, |
||||
{ title: '备注', key: 'remark', width: '5%', type: FormTypes.input } |
||||
] |
||||
}, |
||||
confirmLoading: false, |
||||
validatorRules:{ |
||||
operTime:{ |
||||
rules: [ |
||||
{ required: true, message: '请输入单据日期!' } |
||||
] |
||||
}, |
||||
type:{ |
||||
rules: [ |
||||
{ required: true, message: '请选择类型!' } |
||||
] |
||||
} |
||||
}, |
||||
url: { |
||||
add: '/depotHead/addDepotHeadAndDetail', |
||||
edit: '/depotHead/updateDepotHeadAndDetail', |
||||
detailList: '/depotItem/getDetailList' |
||||
} |
||||
} |
||||
}, |
||||
created () { |
||||
}, |
||||
methods: { |
||||
//调用完edit()方法之后会自动调用此方法 |
||||
editAfter() { |
||||
this.changeColumnHide() |
||||
if (this.action === 'add') { |
||||
this.addInit(this.prefixNo) |
||||
this.fileList = [] |
||||
} else { |
||||
this.model.operTime = this.model.operTimeStr |
||||
this.fileList = this.model.fileName |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue(pick(this.model,'organId', 'operTime', 'number', 'remark', |
||||
'discount','discountMoney','discountLastMoney','otherMoney','accountId','changeAmount')) |
||||
}); |
||||
// 加载子表数据 |
||||
let params = { |
||||
headerId: this.model.id, |
||||
mpList: getMpListShort(Vue.ls.get('materialPropertyList')) //扩展属性 |
||||
} |
||||
let url = this.readOnly ? this.url.detailList : this.url.detailList; |
||||
this.requestSubTableData(url, params, this.materialTable); |
||||
} |
||||
//复制新增单据-初始化单号和日期 |
||||
if(this.action === 'copyAdd') { |
||||
this.model.id = '' |
||||
this.model.tenantId = '' |
||||
this.copyAddInit(this.prefixNo) |
||||
} |
||||
this.initDepot() |
||||
}, |
||||
//提交单据时整理成formData |
||||
classifyIntoFormData(allValues) { |
||||
let totalPrice = 0 |
||||
let billMain = Object.assign(this.model, allValues.formValue) |
||||
let detailArr = allValues.tablesValue[0].values |
||||
billMain.type = '其它' |
||||
billMain.subType = '拆卸单' |
||||
billMain.defaultNumber = billMain.number |
||||
for(let item of detailArr){ |
||||
totalPrice += item.allPrice-0 |
||||
} |
||||
billMain.totalPrice = totalPrice |
||||
if(this.fileList && this.fileList.length > 0) { |
||||
billMain.fileName = this.fileList |
||||
} |
||||
if(this.model.id){ |
||||
billMain.id = this.model.id |
||||
} |
||||
return { |
||||
info: billMain, |
||||
rows: detailArr, |
||||
} |
||||
}, |
||||
onAdded(event) { |
||||
const { row, target } = event |
||||
getAction('/depot/findDepotByCurrentUser').then((res) => { |
||||
if (res.code === 200) { |
||||
let arr = res.data |
||||
for (let i = 0; i < arr.length; i++) { |
||||
if(arr[i].isDefault){ |
||||
target.setValues([{rowKey: row.id, values: {depotId: arr[i].id+''}}]) |
||||
} |
||||
} |
||||
} |
||||
}) |
||||
if(target.rows.length>=2) { |
||||
target.setValues([{rowKey: row.id, values: {mType: '普通子件'}}]) |
||||
} else { |
||||
target.setValues([{rowKey: row.id, values: {mType: '组合件'}}]) |
||||
} |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,287 @@
|
||||
<template> |
||||
<j-modal |
||||
:title="title" |
||||
:width="width" |
||||
:visible="visible" |
||||
:confirmLoading="confirmLoading" |
||||
:maskClosable="false" |
||||
:keyboard="false" |
||||
:forceRender="true" |
||||
switchFullscreen |
||||
@ok="handleOk" |
||||
@cancel="handleCancel" |
||||
wrapClassName="ant-modal-cust-warp" |
||||
style="top:5%;height: 100%;overflow-y: hidden"> |
||||
<a-spin :spinning="confirmLoading"> |
||||
<a-form :form="form"> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="单据日期"> |
||||
<j-date v-decorator="['operTime', validatorRules.operTime]" :show-time="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="单据编号"> |
||||
<a-input placeholder="请输入单据编号" v-decorator.trim="[ 'number' ]" :readOnly="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="关联单据"> |
||||
<a-input-search placeholder="请选择关联单据" v-decorator="[ 'linkNumber' ]" @search="onSearchLinkNumber" :readOnly="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
<j-editable-table id="billModal" |
||||
:ref="refKeys[0]" |
||||
:loading="materialTable.loading" |
||||
:columns="materialTable.columns" |
||||
:dataSource="materialTable.dataSource" |
||||
:maxHeight="300" |
||||
:rowNumber="false" |
||||
:rowSelection="true" |
||||
:actionButton="true" |
||||
:dragSort="true" |
||||
@valueChange="onValueChange" |
||||
@added="onAdded" |
||||
@deleted="onDeleted"> |
||||
<template #buttonAfter> |
||||
<a-row :gutter="24" style="float:left;" data-step="4" data-title="扫码录入" data-intro="此功能支持扫码枪扫描商品物料编码进行录入"> |
||||
<a-col v-if="scanStatus" :md="6" :sm="24"> |
||||
<a-button @click="scanEnter">扫码录入</a-button> |
||||
</a-col> |
||||
<a-col v-if="!scanStatus" :md="16" :sm="24" style="padding: 0 6px 0 12px"> |
||||
<a-input placeholder="请扫码商品物料编码并回车" v-model="scanBarCode" @pressEnter="scanPressEnter" ref="scanBarCode"/> |
||||
</a-col> |
||||
<a-col v-if="!scanStatus" :md="6" :sm="24" style="padding: 0px"> |
||||
<a-button @click="stopScan">收起扫码</a-button> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row :gutter="24" style="float:left;"> |
||||
<a-col :md="24" :sm="24"> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" @click="handleBatchSetDepot"><a-icon type="setting"/>批量设置</a-menu-item> |
||||
<a-menu-item v-if="isTenant" key="2" @click="addDepot"><a-icon type="plus"/>新增仓库</a-menu-item> |
||||
</a-menu> |
||||
<a-button style="margin-left: 8px">仓库操作 <a-icon type="down" /></a-button> |
||||
</a-dropdown> |
||||
</a-col> |
||||
</a-row> |
||||
<a-button type="primary" @click="clearKh" style="margin-left:10px">清空空行</a-button> |
||||
</template> |
||||
</j-editable-table> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="24" :md="24" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="{xs: { span: 24 },sm: { span: 24 }}" label=""> |
||||
<a-textarea :rows="1" placeholder="请输入备注" v-decorator="[ 'remark' ]" style="margin-top:8px;"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="附件"> |
||||
<j-upload v-model="fileList" bizPath="bill"></j-upload> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
</a-form> |
||||
</a-spin> |
||||
<link-bill-list ref="linkBillList" @ok="linkBillListOk"></link-bill-list> |
||||
<vendor-modal ref="vendorModalForm" @ok="vendorModalFormOk"></vendor-modal> |
||||
<depot-modal ref="depotModalForm" @ok="depotModalFormOk"></depot-modal> |
||||
<batch-set-depot ref="batchSetDepotModalForm" @ok="batchSetDepotModalFormOk"></batch-set-depot> |
||||
</j-modal> |
||||
</template> |
||||
<script> |
||||
import pick from 'lodash.pick' |
||||
import LinkBillList from '../dialog/LinkBillList' |
||||
import VendorModal from '../../system/modules/VendorModal' |
||||
import DepotModal from '../../system/modules/DepotModal' |
||||
import BatchSetDepot from '../dialog/BatchSetDepot' |
||||
import { FormTypes } from '@/utils/JEditableTableUtil' |
||||
import { JEditableTableMixin } from '@/mixins/JEditableTableMixin' |
||||
import { BillModalMixin } from '../mixins/BillModalMixin' |
||||
import { getMpListShort } from "@/utils/util" |
||||
import JUpload from '@/components/jeecg/JUpload' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
export default { |
||||
name: "OtherInModal", |
||||
mixins: [JEditableTableMixin, BillModalMixin], |
||||
components: { |
||||
VendorModal, |
||||
DepotModal, |
||||
BatchSetDepot, |
||||
JUpload, |
||||
JDate, |
||||
LinkBillList, |
||||
VNodes: { |
||||
functional: true, |
||||
render: (h, ctx) => ctx.props.vnodes, |
||||
} |
||||
}, |
||||
data () { |
||||
return { |
||||
title:"操作", |
||||
width: '100%', |
||||
moreStatus: false, |
||||
// 新增时子表默认添加几行空数据 |
||||
addDefaultRowNum: 10, |
||||
visible: false, |
||||
operTimeStr: '', |
||||
prefixNo: 'QTRK', |
||||
fileList:[], |
||||
model: {}, |
||||
labelCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 8 }, |
||||
}, |
||||
wrapperCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 16 }, |
||||
}, |
||||
refKeys: ['materialDataTable', ], |
||||
activeKey: 'materialDataTable', |
||||
materialTable: { |
||||
loading: false, |
||||
dataSource: [], |
||||
columns: [ |
||||
{ title: '仓库名称', key: 'depotId', width: '7%', type: FormTypes.select, placeholder: '请选择${title}', options: [], |
||||
allowSearch:true, validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '物料编码', key: 'barCode', width: '8%', type: FormTypes.popupJsh, kind: 'material', multi: true, |
||||
validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '名称', key: 'name', width: '20%', type: FormTypes.normal }, |
||||
{ title: '规格', key: 'standard', width: '5%', type: FormTypes.normal }, |
||||
{ title: '型号', key: 'model', width: '5%', type: FormTypes.normal }, |
||||
{ title: '颜色', key: 'color', width: '5%', type: FormTypes.normal }, |
||||
{ title: '扩展信息', key: 'materialOther', width: '5%', type: FormTypes.normal }, |
||||
{ title: '库存', key: 'stock', width: '5%', type: FormTypes.inputNumber, readonly:true }, |
||||
{ title: '单位', key: 'unit', width: '4%', type: FormTypes.normal }, |
||||
{ title: '实存', key: 'currentStock', width: '4%', type: FormTypes.inputNumber}, |
||||
{ title: '序列号', key: 'snList', width: '12%', type: FormTypes.input, placeholder: '多个序列号请用逗号隔开', |
||||
validateRules: [{ pattern: /^\S{1,100}$/, message: '请小于100位字符' }] |
||||
}, |
||||
{ title: '批号', key: 'batchNumber', width: '5%', type: FormTypes.input }, |
||||
{ title: '有效期', key: 'expirationDate',width: '7%', type: FormTypes.date }, |
||||
{ title: '多属性', key: 'sku', width: '4%', type: FormTypes.normal }, |
||||
{ title: '数量', key: 'operNumber', width: '5%', type: FormTypes.inputNumber,readonly:true, statistics: true, |
||||
validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '单价', key: 'unitPrice', width: '5%', type: FormTypes.inputNumber}, |
||||
{ title: '金额', key: 'allPrice', width: '5%', type: FormTypes.inputNumber, statistics: true }, |
||||
{ title: '备注', key: 'remark', width: '5%', type: FormTypes.input } |
||||
] |
||||
}, |
||||
confirmLoading: false, |
||||
validatorRules:{ |
||||
operTime:{ |
||||
rules: [ |
||||
{ required: true, message: '请输入单据日期!' } |
||||
] |
||||
}, |
||||
type:{ |
||||
rules: [ |
||||
{ required: true, message: '请选择类型!' } |
||||
] |
||||
} |
||||
}, |
||||
url: { |
||||
add: '/depotHead/addDepotHeadAndDetail', |
||||
edit: '/depotHead/updateDepotHeadAndDetail', |
||||
detailList: '/depotItem/getDetailList' |
||||
} |
||||
} |
||||
}, |
||||
created () { |
||||
}, |
||||
methods: { |
||||
//调用完edit()方法之后会自动调用此方法 |
||||
editAfter() { |
||||
this.changeColumnHide() |
||||
this.changeFormTypes(this.materialTable.columns, 'snList', 0) |
||||
this.changeFormTypes(this.materialTable.columns, 'batchNumber', 0) |
||||
this.changeFormTypes(this.materialTable.columns, 'expirationDate', 0) |
||||
if (this.action === 'add') { |
||||
this.addInit(this.prefixNo) |
||||
this.fileList = [] |
||||
} else { |
||||
this.model.operTime = this.model.operTimeStr |
||||
this.fileList = this.model.fileName |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue(pick(this.model,'linkNumber', 'operTime', 'number', 'remark', |
||||
'discount','discountMoney','discountLastMoney','otherMoney','accountId','changeAmount')) |
||||
}); |
||||
// 加载子表数据 |
||||
let params = { |
||||
headerId: this.model.id, |
||||
mpList: getMpListShort(Vue.ls.get('materialPropertyList')) //扩展属性 |
||||
} |
||||
let url = this.readOnly ? this.url.detailList : this.url.detailList; |
||||
this.requestSubTableData(url, params, this.materialTable); |
||||
} |
||||
//复制新增单据-初始化单号和日期 |
||||
if(this.action === 'copyAdd') { |
||||
this.model.id = '' |
||||
this.model.tenantId = '' |
||||
this.copyAddInit(this.prefixNo) |
||||
} |
||||
this.initSupplier() |
||||
this.initDepot() |
||||
}, |
||||
onSearchLinkNumber() { |
||||
this.$refs.linkBillList.show('复盘', '盘点') |
||||
this.$refs.linkBillList.title = "请选择盘点复盘" |
||||
}, |
||||
linkBillListOk(selectBillRows) { |
||||
this.changeFormTypes(this.materialTable.columns, 'preNumber', 1) |
||||
this.changeFormTypes(this.materialTable.columns, 'finishNumber', 1) |
||||
if(selectBillRows && selectBillRows.length>0) { |
||||
let record = selectBillRows[0] |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue({ |
||||
'linkNumber': record.linkNumber, |
||||
'linkNumber': record.number, |
||||
'remark': record.remark, |
||||
'discountLastMoney': record.totalPrice, |
||||
'changeAmount': record.totalPrice |
||||
}) |
||||
}); |
||||
// 加载子表数据 |
||||
let params = { |
||||
headerId: record.id, |
||||
mpList: getMpListShort(Vue.ls.get('materialPropertyList')) //扩展属性 |
||||
} |
||||
this.requestSubTableDataEx(this.url.detailList, params, this.materialTable); |
||||
} |
||||
}, |
||||
//提交单据时整理成formData |
||||
classifyIntoFormData(allValues) { |
||||
let totalPrice = 0 |
||||
let billMain = Object.assign(this.model, allValues.formValue) |
||||
let detailArr = allValues.tablesValue[0].values |
||||
billMain.type = '其它' |
||||
billMain.subType = '盘点复盘' |
||||
billMain.defaultNumber = billMain.number |
||||
for(let item of detailArr){ |
||||
totalPrice += item.allPrice-0 |
||||
} |
||||
billMain.totalPrice = totalPrice |
||||
if(this.fileList && this.fileList.length > 0) { |
||||
billMain.fileName = this.fileList |
||||
} |
||||
if(this.model.id){ |
||||
billMain.id = this.model.id |
||||
} |
||||
return { |
||||
info: billMain, |
||||
rows: detailArr, |
||||
} |
||||
}, |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,269 @@
|
||||
<template> |
||||
<j-modal |
||||
:title="title" |
||||
:width="width" |
||||
:visible="visible" |
||||
:confirmLoading="confirmLoading" |
||||
:maskClosable="false" |
||||
:keyboard="false" |
||||
:forceRender="true" |
||||
switchFullscreen |
||||
@ok="handleOk" |
||||
@cancel="handleCancel" |
||||
wrapClassName="ant-modal-cust-warp" |
||||
style="top:5%;height: 100%;overflow-y: hidden"> |
||||
<a-spin :spinning="confirmLoading"> |
||||
<a-form :form="form"> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="供应商"> |
||||
<a-select placeholder="选择供应商" v-decorator="[ 'organId' ]" |
||||
:dropdownMatchSelectWidth="false" showSearch optionFilterProp="children"> |
||||
<div slot="dropdownRender" slot-scope="menu"> |
||||
<v-nodes :vnodes="menu" /> |
||||
<a-divider style="margin: 4px 0;" /> |
||||
<div v-if="isTenant" style="padding: 4px 8px; cursor: pointer;" |
||||
@mousedown="e => e.preventDefault()" @click="addSupplier"><a-icon type="plus" /> 新增供应商</div> |
||||
</div> |
||||
<a-select-option v-for="(item,index) in supList" :key="index" :value="Number(item.id)"> |
||||
{{ item.supplier }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="单据日期"> |
||||
<j-date v-decorator="['operTime', validatorRules.operTime]" :show-time="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="单据编号"> |
||||
<a-input placeholder="请输入单据编号" v-decorator.trim="[ 'number' ]" :readOnly="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"></a-col> |
||||
</a-row> |
||||
<j-editable-table id="billModal" |
||||
:ref="refKeys[0]" |
||||
:loading="materialTable.loading" |
||||
:columns="materialTable.columns" |
||||
:dataSource="materialTable.dataSource" |
||||
:maxHeight="300" |
||||
:rowNumber="false" |
||||
:rowSelection="true" |
||||
:actionButton="true" |
||||
:dragSort="true" |
||||
@valueChange="onValueChange" |
||||
@added="onAdded" |
||||
@deleted="onDeleted"> |
||||
<template #buttonAfter> |
||||
<a-row :gutter="24" style="float:left;" data-step="4" data-title="扫码录入" data-intro="此功能支持扫码枪扫描商品物料编码进行录入"> |
||||
<a-col v-if="scanStatus" :md="6" :sm="24"> |
||||
<a-button @click="scanEnter">扫码录入</a-button> |
||||
</a-col> |
||||
<a-col v-if="!scanStatus" :md="16" :sm="24" style="padding: 0 6px 0 12px"> |
||||
<a-input placeholder="请扫码商品物料编码并回车" v-model="scanBarCode" @pressEnter="scanPressEnter" ref="scanBarCode"/> |
||||
</a-col> |
||||
<a-col v-if="!scanStatus" :md="6" :sm="24" style="padding: 0px"> |
||||
<a-button @click="stopScan">收起扫码</a-button> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row :gutter="24" style="float:left;"> |
||||
<a-col :md="24" :sm="24"> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" @click="handleBatchSetDepot"><a-icon type="setting"/>批量设置</a-menu-item> |
||||
<a-menu-item v-if="isTenant" key="2" @click="addDepot"><a-icon type="plus"/>新增仓库</a-menu-item> |
||||
</a-menu> |
||||
<a-button style="margin-left: 8px">仓库操作 <a-icon type="down" /></a-button> |
||||
</a-dropdown> |
||||
</a-col> |
||||
</a-row> |
||||
<a-button type="primary" @click="clearKh" style="margin-left:10px">清空空行</a-button> |
||||
</template> |
||||
</j-editable-table> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="24" :md="24" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="{xs: { span: 24 },sm: { span: 24 }}" label=""> |
||||
<a-textarea :rows="1" placeholder="请输入备注" v-decorator="[ 'remark' ]" style="margin-top:8px;"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="附件"> |
||||
<j-upload v-model="fileList" bizPath="bill"></j-upload> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
</a-form> |
||||
</a-spin> |
||||
<vendor-modal ref="vendorModalForm" @ok="vendorModalFormOk"></vendor-modal> |
||||
<depot-modal ref="depotModalForm" @ok="depotModalFormOk"></depot-modal> |
||||
<batch-set-depot ref="batchSetDepotModalForm" @ok="batchSetDepotModalFormOk"></batch-set-depot> |
||||
</j-modal> |
||||
</template> |
||||
<script> |
||||
import pick from 'lodash.pick' |
||||
import VendorModal from '../../system/modules/VendorModal' |
||||
import DepotModal from '../../system/modules/DepotModal' |
||||
import BatchSetDepot from '../dialog/BatchSetDepot' |
||||
import { FormTypes } from '@/utils/JEditableTableUtil' |
||||
import { JEditableTableMixin } from '@/mixins/JEditableTableMixin' |
||||
import { BillModalMixin } from '../mixins/BillModalMixin' |
||||
import { getMpListShort } from "@/utils/util" |
||||
import JUpload from '@/components/jeecg/JUpload' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
export default { |
||||
name: "OtherInModal", |
||||
mixins: [JEditableTableMixin, BillModalMixin], |
||||
components: { |
||||
VendorModal, |
||||
DepotModal, |
||||
BatchSetDepot, |
||||
JUpload, |
||||
JDate, |
||||
VNodes: { |
||||
functional: true, |
||||
render: (h, ctx) => ctx.props.vnodes, |
||||
} |
||||
}, |
||||
data () { |
||||
return { |
||||
title:"操作", |
||||
width: '100%', |
||||
moreStatus: false, |
||||
// 新增时子表默认添加几行空数据 |
||||
addDefaultRowNum: 10, |
||||
visible: false, |
||||
operTimeStr: '', |
||||
prefixNo: 'QTRK', |
||||
fileList:[], |
||||
model: {}, |
||||
labelCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 8 }, |
||||
}, |
||||
wrapperCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 16 }, |
||||
}, |
||||
refKeys: ['materialDataTable', ], |
||||
activeKey: 'materialDataTable', |
||||
materialTable: { |
||||
loading: false, |
||||
dataSource: [], |
||||
columns: [ |
||||
{ title: '仓库名称', key: 'depotId', width: '7%', type: FormTypes.select, placeholder: '请选择${title}', options: [], |
||||
allowSearch:true, validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '物料编码', key: 'barCode', width: '8%', type: FormTypes.popupJsh, kind: 'material', multi: true, |
||||
validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '名称', key: 'name', width: '20%', type: FormTypes.normal }, |
||||
{ title: '规格', key: 'standard', width: '5%', type: FormTypes.normal }, |
||||
{ title: '型号', key: 'model', width: '5%', type: FormTypes.normal }, |
||||
{ title: '颜色', key: 'color', width: '5%', type: FormTypes.normal }, |
||||
{ title: '扩展信息', key: 'materialOther', width: '5%', type: FormTypes.normal }, |
||||
{ title: '库存', key: 'stock', width: '5%', type: FormTypes.inputNumber, readonly:true}, |
||||
{ title: '单位', key: 'unit', width: '4%', type: FormTypes.normal }, |
||||
{ title: '序列号', key: 'snList', width: '12%', type: FormTypes.input, placeholder: '多个序列号请用逗号隔开', |
||||
validateRules: [{ pattern: /^\S{1,100}$/, message: '请小于100位字符' }] |
||||
}, |
||||
{ title: '批号', key: 'batchNumber', width: '5%', type: FormTypes.input }, |
||||
{ title: '有效期', key: 'expirationDate',width: '7%', type: FormTypes.date }, |
||||
{ title: '多属性', key: 'sku', width: '4%', type: FormTypes.normal }, |
||||
{ title: '数量', key: 'operNumber', width: '5%', type: FormTypes.inputNumber, statistics: true, |
||||
validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '单价', key: 'unitPrice', width: '5%', type: FormTypes.inputNumber}, |
||||
{ title: '金额', key: 'allPrice', width: '5%', type: FormTypes.inputNumber, statistics: true }, |
||||
{ title: '备注', key: 'remark', width: '5%', type: FormTypes.input } |
||||
] |
||||
}, |
||||
confirmLoading: false, |
||||
validatorRules:{ |
||||
operTime:{ |
||||
rules: [ |
||||
{ required: true, message: '请输入单据日期!' } |
||||
] |
||||
}, |
||||
type:{ |
||||
rules: [ |
||||
{ required: true, message: '请选择类型!' } |
||||
] |
||||
} |
||||
}, |
||||
url: { |
||||
add: '/depotHead/addDepotHeadAndDetail', |
||||
edit: '/depotHead/updateDepotHeadAndDetail', |
||||
detailList: '/depotItem/getDetailList' |
||||
} |
||||
} |
||||
}, |
||||
created () { |
||||
}, |
||||
methods: { |
||||
//调用完edit()方法之后会自动调用此方法 |
||||
editAfter() { |
||||
this.changeColumnHide() |
||||
this.changeFormTypes(this.materialTable.columns, 'snList', 0) |
||||
this.changeFormTypes(this.materialTable.columns, 'batchNumber', 0) |
||||
this.changeFormTypes(this.materialTable.columns, 'expirationDate', 0) |
||||
if (this.action === 'add') { |
||||
this.addInit(this.prefixNo) |
||||
this.fileList = [] |
||||
} else { |
||||
this.model.operTime = this.model.operTimeStr |
||||
this.fileList = this.model.fileName |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue(pick(this.model,'organId', 'operTime', 'number', 'remark', |
||||
'discount','discountMoney','discountLastMoney','otherMoney','accountId','changeAmount')) |
||||
}); |
||||
// 加载子表数据 |
||||
let params = { |
||||
headerId: this.model.id, |
||||
mpList: getMpListShort(Vue.ls.get('materialPropertyList')) //扩展属性 |
||||
} |
||||
let url = this.readOnly ? this.url.detailList : this.url.detailList; |
||||
this.requestSubTableData(url, params, this.materialTable); |
||||
} |
||||
//复制新增单据-初始化单号和日期 |
||||
if(this.action === 'copyAdd') { |
||||
this.model.id = '' |
||||
this.model.tenantId = '' |
||||
this.copyAddInit(this.prefixNo) |
||||
} |
||||
this.initSupplier() |
||||
this.initDepot() |
||||
}, |
||||
//提交单据时整理成formData |
||||
classifyIntoFormData(allValues) { |
||||
let totalPrice = 0 |
||||
let billMain = Object.assign(this.model, allValues.formValue) |
||||
let detailArr = allValues.tablesValue[0].values |
||||
billMain.type = '入库' |
||||
billMain.subType = '其它' |
||||
billMain.defaultNumber = billMain.number |
||||
for(let item of detailArr){ |
||||
totalPrice += item.allPrice-0 |
||||
} |
||||
billMain.totalPrice = totalPrice |
||||
if(this.fileList && this.fileList.length > 0) { |
||||
billMain.fileName = this.fileList |
||||
} |
||||
if(this.model.id){ |
||||
billMain.id = this.model.id |
||||
} |
||||
return { |
||||
info: billMain, |
||||
rows: detailArr, |
||||
} |
||||
}, |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,267 @@
|
||||
<template> |
||||
<j-modal |
||||
:title="title" |
||||
:width="width" |
||||
:visible="visible" |
||||
:confirmLoading="confirmLoading" |
||||
:maskClosable="false" |
||||
:keyboard="false" |
||||
:forceRender="true" |
||||
switchFullscreen |
||||
@ok="handleOk" |
||||
@cancel="handleCancel" |
||||
wrapClassName="ant-modal-cust-warp" |
||||
style="top:5%;height: 100%;overflow-y: hidden"> |
||||
<a-spin :spinning="confirmLoading"> |
||||
<a-form :form="form"> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="客户"> |
||||
<a-select placeholder="选择客户" v-decorator="[ 'organId' ]" |
||||
:dropdownMatchSelectWidth="false" showSearch optionFilterProp="children"> |
||||
<div slot="dropdownRender" slot-scope="menu"> |
||||
<v-nodes :vnodes="menu" /> |
||||
<a-divider style="margin: 4px 0;" /> |
||||
<div v-if="isTenant" style="padding: 4px 8px; cursor: pointer;" |
||||
@mousedown="e => e.preventDefault()" @click="addCustomer"><a-icon type="plus" /> 新增客户</div> |
||||
</div> |
||||
<a-select-option v-for="(item,index) in cusList" :key="index" :value="item.id"> |
||||
{{ item.supplier }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="单据日期"> |
||||
<j-date v-decorator="['operTime', validatorRules.operTime]" :show-time="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="单据编号"> |
||||
<a-input placeholder="请输入单据编号" v-decorator.trim="[ 'number' ]" :readOnly="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"></a-col> |
||||
</a-row> |
||||
<j-editable-table id="billModal" |
||||
:ref="refKeys[0]" |
||||
:loading="materialTable.loading" |
||||
:columns="materialTable.columns" |
||||
:dataSource="materialTable.dataSource" |
||||
:maxHeight="300" |
||||
:rowNumber="false" |
||||
:rowSelection="true" |
||||
:actionButton="true" |
||||
:dragSort="true" |
||||
@valueChange="onValueChange" |
||||
@added="onAdded" |
||||
@deleted="onDeleted"> |
||||
<template #buttonAfter> |
||||
<a-row :gutter="24" style="float:left;" data-step="4" data-title="扫码录入" data-intro="此功能支持扫码枪扫描商品条码进行录入"> |
||||
<a-col v-if="scanStatus" :md="6" :sm="24"> |
||||
<a-button @click="scanEnter">扫码录入</a-button> |
||||
</a-col> |
||||
<a-col v-if="!scanStatus" :md="16" :sm="24" style="padding: 0 6px 0 12px"> |
||||
<a-input placeholder="请扫码商品条码并回车" v-model="scanBarCode" @pressEnter="scanPressEnter" ref="scanBarCode"/> |
||||
</a-col> |
||||
<a-col v-if="!scanStatus" :md="6" :sm="24" style="padding: 0px"> |
||||
<a-button @click="stopScan">收起扫码</a-button> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row :gutter="24" style="float:left;"> |
||||
<a-col :md="24" :sm="24"> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" @click="handleBatchSetDepot"><a-icon type="setting"/>批量设置</a-menu-item> |
||||
<a-menu-item v-if="isTenant" key="2" @click="addDepot"><a-icon type="plus"/>新增仓库</a-menu-item> |
||||
</a-menu> |
||||
<a-button style="margin-left: 8px">仓库操作 <a-icon type="down" /></a-button> |
||||
</a-dropdown> |
||||
</a-col> |
||||
</a-row> |
||||
<a-button type="primary" @click="clearKh" style="margin-left:10px">清空空行</a-button> |
||||
</template> |
||||
</j-editable-table> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="24" :md="24" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="{xs: { span: 24 },sm: { span: 24 }}" label=""> |
||||
<a-textarea :rows="1" placeholder="请输入备注" v-decorator="[ 'remark' ]" style="margin-top:8px;"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="附件"> |
||||
<j-upload v-model="fileList" bizPath="bill"></j-upload> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
</a-form> |
||||
</a-spin> |
||||
<customer-modal ref="customerModalForm" @ok="customerModalFormOk"></customer-modal> |
||||
<depot-modal ref="depotModalForm" @ok="depotModalFormOk"></depot-modal> |
||||
<batch-set-depot ref="batchSetDepotModalForm" @ok="batchSetDepotModalFormOk"></batch-set-depot> |
||||
</j-modal> |
||||
</template> |
||||
<script> |
||||
import pick from 'lodash.pick' |
||||
import CustomerModal from '../../system/modules/CustomerModal' |
||||
import DepotModal from '../../system/modules/DepotModal' |
||||
import BatchSetDepot from '../dialog/BatchSetDepot' |
||||
import { FormTypes } from '@/utils/JEditableTableUtil' |
||||
import { JEditableTableMixin } from '@/mixins/JEditableTableMixin' |
||||
import { BillModalMixin } from '../mixins/BillModalMixin' |
||||
import { getMpListShort } from "@/utils/util" |
||||
import JUpload from '@/components/jeecg/JUpload' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
export default { |
||||
name: "OtherOutModal", |
||||
mixins: [JEditableTableMixin, BillModalMixin], |
||||
components: { |
||||
CustomerModal, |
||||
DepotModal, |
||||
BatchSetDepot, |
||||
JUpload, |
||||
JDate, |
||||
VNodes: { |
||||
functional: true, |
||||
render: (h, ctx) => ctx.props.vnodes, |
||||
} |
||||
}, |
||||
data () { |
||||
return { |
||||
title:"操作", |
||||
width: '100%', |
||||
moreStatus: false, |
||||
// 新增时子表默认添加几行空数据 |
||||
addDefaultRowNum: 10, |
||||
visible: false, |
||||
operTimeStr: '', |
||||
prefixNo: 'QTCK', |
||||
fileList:[], |
||||
model: {}, |
||||
labelCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 8 }, |
||||
}, |
||||
wrapperCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 16 }, |
||||
}, |
||||
refKeys: ['materialDataTable', ], |
||||
activeKey: 'materialDataTable', |
||||
materialTable: { |
||||
loading: false, |
||||
dataSource: [], |
||||
columns: [ |
||||
{ title: '仓库名称', key: 'depotId', width: '7%', type: FormTypes.select, placeholder: '请选择${title}', options: [], |
||||
allowSearch:true, validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '物料编码', key: 'barCode', width: '8%', type: FormTypes.popupJsh, kind: 'material', multi: true, |
||||
validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '名称', key: 'name', width: '20%', type: FormTypes.normal }, |
||||
{ title: '规格', key: 'standard', width: '5%', type: FormTypes.normal }, |
||||
{ title: '型号', key: 'model', width: '5%', type: FormTypes.normal }, |
||||
{ title: '颜色', key: 'color', width: '5%', type: FormTypes.normal }, |
||||
{ title: '扩展信息', key: 'materialOther', width: '5%', type: FormTypes.normal }, |
||||
{ title: '库存', key: 'stock', width: '5%', type: FormTypes.inputNumber, readonly:true }, |
||||
{ title: '单位', key: 'unit', width: '4%', type: FormTypes.normal }, |
||||
{ title: '序列号', key: 'snList', width: '12%', type: FormTypes.popupJsh, kind: 'sn', multi: true }, |
||||
{ title: '批号', key: 'batchNumber', width: '7%', type: FormTypes.popupJsh, kind: 'batch', multi: false }, |
||||
{ title: '有效期', key: 'expirationDate',width: '6%', type: FormTypes.normal }, |
||||
{ title: '多属性', key: 'sku', width: '4%', type: FormTypes.normal }, |
||||
{ title: '数量', key: 'operNumber', width: '5%', type: FormTypes.inputNumber, statistics: true, |
||||
validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '单价', key: 'unitPrice', width: '5%', type: FormTypes.inputNumber}, |
||||
{ title: '金额', key: 'allPrice', width: '5%', type: FormTypes.inputNumber, statistics: true }, |
||||
{ title: '备注', key: 'remark', width: '5%', type: FormTypes.input } |
||||
] |
||||
}, |
||||
confirmLoading: false, |
||||
validatorRules:{ |
||||
operTime:{ |
||||
rules: [ |
||||
{ required: true, message: '请输入单据日期!' } |
||||
] |
||||
}, |
||||
type:{ |
||||
rules: [ |
||||
{ required: true, message: '请选择类型!' } |
||||
] |
||||
} |
||||
}, |
||||
url: { |
||||
add: '/depotHead/addDepotHeadAndDetail', |
||||
edit: '/depotHead/updateDepotHeadAndDetail', |
||||
detailList: '/depotItem/getDetailList' |
||||
} |
||||
} |
||||
}, |
||||
created () { |
||||
}, |
||||
methods: { |
||||
//调用完edit()方法之后会自动调用此方法 |
||||
editAfter() { |
||||
this.changeColumnHide() |
||||
this.changeFormTypes(this.materialTable.columns, 'snList', 0) |
||||
this.changeFormTypes(this.materialTable.columns, 'batchNumber', 0) |
||||
this.changeFormTypes(this.materialTable.columns, 'expirationDate', 0) |
||||
if (this.action === 'add') { |
||||
this.addInit(this.prefixNo) |
||||
this.fileList = [] |
||||
} else { |
||||
this.model.operTime = this.model.operTimeStr |
||||
this.fileList = this.model.fileName |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue(pick(this.model,'organId', 'operTime', 'number', 'remark', |
||||
'discount','discountMoney','discountLastMoney','otherMoney','accountId','changeAmount')) |
||||
}); |
||||
// 加载子表数据 |
||||
let params = { |
||||
headerId: this.model.id, |
||||
mpList: getMpListShort(Vue.ls.get('materialPropertyList')) //扩展属性 |
||||
} |
||||
let url = this.readOnly ? this.url.detailList : this.url.detailList; |
||||
this.requestSubTableData(url, params, this.materialTable); |
||||
} |
||||
//复制新增单据-初始化单号和日期 |
||||
if(this.action === 'copyAdd') { |
||||
this.model.id = '' |
||||
this.model.tenantId = '' |
||||
this.copyAddInit(this.prefixNo) |
||||
} |
||||
this.initCustomer() |
||||
this.initDepot() |
||||
}, |
||||
//提交单据时整理成formData |
||||
classifyIntoFormData(allValues) { |
||||
let totalPrice = 0 |
||||
let billMain = Object.assign(this.model, allValues.formValue) |
||||
let detailArr = allValues.tablesValue[0].values |
||||
billMain.type = '出库' |
||||
billMain.subType = '其它' |
||||
billMain.defaultNumber = billMain.number |
||||
for(let item of detailArr){ |
||||
totalPrice += item.allPrice-0 |
||||
} |
||||
billMain.totalPrice = totalPrice |
||||
if(this.fileList && this.fileList.length > 0) { |
||||
billMain.fileName = this.fileList |
||||
} |
||||
if(this.model.id){ |
||||
billMain.id = this.model.id |
||||
} |
||||
return { |
||||
info: billMain, |
||||
rows: detailArr, |
||||
} |
||||
}, |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,404 @@
|
||||
<template> |
||||
<j-modal |
||||
:title="title" |
||||
:width="width" |
||||
:visible="visible" |
||||
:confirmLoading="confirmLoading" |
||||
:maskClosable="false" |
||||
:keyboard="false" |
||||
:forceRender="true" |
||||
switchFullscreen |
||||
@ok="handleOk" |
||||
@cancel="handleCancel" |
||||
wrapClassName="ant-modal-cust-warp" |
||||
style="top:5%;height: 100%;overflow-y: hidden"> |
||||
<a-spin :spinning="confirmLoading"> |
||||
<a-form :form="form"> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="供应商"> |
||||
<a-select placeholder="选择供应商" v-decorator="[ 'organId', validatorRules.organId ]" |
||||
:dropdownMatchSelectWidth="false" showSearch optionFilterProp="children"> |
||||
<div slot="dropdownRender" slot-scope="menu"> |
||||
<v-nodes :vnodes="menu" /> |
||||
<a-divider style="margin: 4px 0;" /> |
||||
<div v-if="isTenant" style="padding: 4px 8px; cursor: pointer;" |
||||
@mousedown="e => e.preventDefault()" @click="addSupplier"><a-icon type="plus" /> 新增供应商</div> |
||||
</div> |
||||
<a-select-option v-for="(item,index) in supList" :key="index" :value="Number(item.id)"> |
||||
{{ item.supplier }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="单据日期"> |
||||
<j-date v-decorator="['operTime', validatorRules.operTime]" :show-time="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="单据编号"> |
||||
<a-input placeholder="请输入单据编号" v-decorator.trim="[ 'number' ]" :readOnly="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="关联单据"> |
||||
<a-input-search placeholder="请选择关联单据" v-decorator="[ 'linkNumber' ]" @search="onSearchLinkNumber" :readOnly="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
<j-editable-table id="billModal" |
||||
:ref="refKeys[0]" |
||||
:loading="materialTable.loading" |
||||
:columns="materialTable.columns" |
||||
:dataSource="materialTable.dataSource" |
||||
:maxHeight="300" |
||||
:rowNumber="false" |
||||
:rowSelection="true" |
||||
:actionButton="true" |
||||
:dragSort="true" |
||||
@valueChange="onValueChange" |
||||
@added="onAdded" |
||||
@deleted="onDeleted"> |
||||
<template #buttonAfter> |
||||
<a-row :gutter="24" style="float:left;" data-step="4" data-title="扫码录入" data-intro="此功能支持扫码枪扫描商品条码进行录入"> |
||||
<a-col v-if="scanStatus" :md="6" :sm="24"> |
||||
<a-button @click="scanEnter">扫码录入</a-button> |
||||
</a-col> |
||||
<a-col v-if="!scanStatus" :md="16" :sm="24" style="padding: 0 6px 0 12px"> |
||||
<a-input placeholder="请扫码商品条码并回车" v-model="scanBarCode" @pressEnter="scanPressEnter" ref="scanBarCode"/> |
||||
</a-col> |
||||
<a-col v-if="!scanStatus" :md="6" :sm="24" style="padding: 0px"> |
||||
<a-button @click="stopScan">收起扫码</a-button> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row :gutter="24" style="float:left;"> |
||||
<a-col :md="24" :sm="24"> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" @click="handleBatchSetDepot"><a-icon type="setting"/>批量设置</a-menu-item> |
||||
<a-menu-item v-if="isTenant" key="2" @click="addDepot"><a-icon type="plus"/>新增仓库</a-menu-item> |
||||
</a-menu> |
||||
<a-button style="margin-left: 8px">仓库操作 <a-icon type="down" /></a-button> |
||||
</a-dropdown> |
||||
</a-col> |
||||
</a-row> |
||||
<a-button type="primary" @click="clearKh" style="margin-left:10px">清空空行</a-button> |
||||
</template> |
||||
</j-editable-table> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="24" :md="24" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="{xs: { span: 24 },sm: { span: 24 }}" label=""> |
||||
<a-textarea :rows="1" placeholder="请输入备注" v-decorator="[ 'remark' ]" style="margin-top:8px;"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="优惠率"> |
||||
<a-input style="width:185px;" placeholder="请输入优惠率" v-decorator.trim="[ 'discount' ]" suffix="%" @keyup="onKeyUpDiscount"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="退款优惠"> |
||||
<a-input placeholder="请输入付款优惠" v-decorator.trim="[ 'discountMoney' ]" @keyup="onKeyUpDiscountMoney"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="优惠后金额"> |
||||
<a-input placeholder="请输入优惠后金额" v-decorator.trim="[ 'discountLastMoney' ]" :readOnly="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="其它费用"> |
||||
<a-input placeholder="请输入其它费用" v-decorator.trim="[ 'otherMoney' ]" @keyup="onKeyUpOtherMoney"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="结算账户"> |
||||
<a-select style="width:185px;" placeholder="选择结算账户" v-decorator="[ 'accountId']" |
||||
:dropdownMatchSelectWidth="false" allowClear @select="selectAccount"> |
||||
<div slot="dropdownRender" slot-scope="menu"> |
||||
<v-nodes :vnodes="menu" /> |
||||
<a-divider style="margin: 4px 0;" /> |
||||
<div v-if="isTenant" style="padding: 4px 8px; cursor: pointer;" |
||||
@mousedown="e => e.preventDefault()" @click="addAccount"><a-icon type="plus" /> 新增结算账户</div> |
||||
</div> |
||||
<a-select-option v-for="(item,index) in accountList" :key="index" :value="item.id"> |
||||
{{ item.name }} |
||||
</a-select-option> |
||||
</a-select> |
||||
<a-tooltip title="多账户明细"> |
||||
<a-button type="default" icon="folder" style="margin-left: 8px;" size="small" v-show="manyAccountBtnStatus" @click="handleManyAccount"/> |
||||
</a-tooltip> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="本次退款"> |
||||
<a-input placeholder="请输入本次退款" v-decorator.trim="[ 'changeAmount' ]" @keyup="onKeyUpChangeAmount" :readOnly="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="本次欠款"> |
||||
<a-input placeholder="请输入本次欠款" v-decorator.trim="[ 'debt' ]" :readOnly="true"/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="附件"> |
||||
<j-upload v-model="fileList" bizPath="bill"></j-upload> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
</a-form> |
||||
</a-spin> |
||||
<many-account-modal ref="manyAccountModalForm" @ok="manyAccountModalFormOk"></many-account-modal> |
||||
<link-bill-list ref="linkBillList" @ok="linkBillListOk"></link-bill-list> |
||||
<vendor-modal ref="vendorModalForm" @ok="vendorModalFormOk"></vendor-modal> |
||||
<depot-modal ref="depotModalForm" @ok="depotModalFormOk"></depot-modal> |
||||
<account-modal ref="accountModalForm" @ok="accountModalFormOk"></account-modal> |
||||
<batch-set-depot ref="batchSetDepotModalForm" @ok="batchSetDepotModalFormOk"></batch-set-depot> |
||||
</j-modal> |
||||
</template> |
||||
<script> |
||||
import pick from 'lodash.pick' |
||||
import ManyAccountModal from '../dialog/ManyAccountModal' |
||||
import LinkBillList from '../dialog/LinkBillList' |
||||
import VendorModal from '../../system/modules/VendorModal' |
||||
import DepotModal from '../../system/modules/DepotModal' |
||||
import AccountModal from '../../system/modules/AccountModal' |
||||
import BatchSetDepot from '../dialog/BatchSetDepot' |
||||
import { FormTypes } from '@/utils/JEditableTableUtil' |
||||
import { JEditableTableMixin } from '@/mixins/JEditableTableMixin' |
||||
import { BillModalMixin } from '../mixins/BillModalMixin' |
||||
import { getMpListShort} from "@/utils/util" |
||||
import { getAction } from '@/api/manage' |
||||
import JUpload from '@/components/jeecg/JUpload' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
export default { |
||||
name: "PurchaseBackModal", |
||||
mixins: [JEditableTableMixin, BillModalMixin], |
||||
components: { |
||||
ManyAccountModal, |
||||
LinkBillList, |
||||
VendorModal, |
||||
DepotModal, |
||||
AccountModal, |
||||
BatchSetDepot, |
||||
JUpload, |
||||
JDate, |
||||
VNodes: { |
||||
functional: true, |
||||
render: (h, ctx) => ctx.props.vnodes, |
||||
} |
||||
}, |
||||
data () { |
||||
return { |
||||
title:"操作", |
||||
width: '100%', |
||||
moreStatus: false, |
||||
// 新增时子表默认添加几行空数据 |
||||
addDefaultRowNum: 10, |
||||
visible: false, |
||||
operTimeStr: '', |
||||
prefixNo: 'CGTH', |
||||
fileList:[], |
||||
model: {}, |
||||
labelCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 8 }, |
||||
}, |
||||
wrapperCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 16 }, |
||||
}, |
||||
refKeys: ['materialDataTable', ], |
||||
activeKey: 'materialDataTable', |
||||
materialTable: { |
||||
loading: false, |
||||
dataSource: [], |
||||
columns: [ |
||||
{ title: '仓库名称', key: 'depotId', width: '7%', type: FormTypes.select, placeholder: '请选择${title}', options: [], |
||||
allowSearch:true, validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '物料编码', key: 'barCode', width: '8%', type: FormTypes.popupJsh, kind: 'material', multi: true, |
||||
validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '名称', key: 'name', width: '20%', type: FormTypes.normal }, |
||||
{ title: '规格', key: 'standard', width: '5%', type: FormTypes.normal }, |
||||
{ title: '型号', key: 'model', width: '5%', type: FormTypes.normal }, |
||||
{ title: '颜色', key: 'color', width: '5%', type: FormTypes.normal }, |
||||
{ title: '扩展信息', key: 'materialOther', width: '5%', type: FormTypes.normal }, |
||||
{ title: '库存', key: 'stock', width: '5%', type: FormTypes.inputNumber, readonly:true }, |
||||
{ title: '单位', key: 'unit', width: '4%', type: FormTypes.normal }, |
||||
{ title: '序列号', key: 'snList', width: '12%', type: FormTypes.popupJsh, kind: 'sn', multi: true }, |
||||
{ title: '批号', key: 'batchNumber', width: '7%', type: FormTypes.popupJsh, kind: 'batch', multi: false }, |
||||
{ title: '有效期', key: 'expirationDate',width: '6%', type: FormTypes.normal }, |
||||
{ title: '多属性', key: 'sku', width: '4%', type: FormTypes.normal }, |
||||
{ title: '数量', key: 'operNumber', width: '5%', type: FormTypes.inputNumber, statistics: true, |
||||
validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '单价', key: 'unitPrice', width: '5%', type: FormTypes.inputNumber}, |
||||
{ title: '金额', key: 'allPrice', width: '5%', type: FormTypes.inputNumber, statistics: true }, |
||||
{ title: '税率', key: 'taxRate', width: '4%', type: FormTypes.inputNumber,placeholder: '%'}, |
||||
{ title: '税额', key: 'taxMoney', width: '5%', type: FormTypes.inputNumber, readonly: true, statistics: true }, |
||||
{ title: '价税合计', key: 'taxLastMoney', width: '5%', type: FormTypes.inputNumber, statistics: true }, |
||||
{ title: '备注', key: 'remark', width: '5%', type: FormTypes.input } |
||||
] |
||||
}, |
||||
confirmLoading: false, |
||||
validatorRules:{ |
||||
operTime:{ |
||||
rules: [ |
||||
{ required: true, message: '请输入单据日期!' } |
||||
] |
||||
}, |
||||
organId:{ |
||||
rules: [ |
||||
{ required: true, message: '请选择供应商!' } |
||||
] |
||||
}, |
||||
accountId:{ |
||||
rules: [ |
||||
{ required: true, message: '请选择结算账户!' } |
||||
] |
||||
} |
||||
}, |
||||
url: { |
||||
add: '/depotHead/addDepotHeadAndDetail', |
||||
edit: '/depotHead/updateDepotHeadAndDetail', |
||||
detailList: '/depotItem/getDetailList' |
||||
} |
||||
} |
||||
}, |
||||
created () { |
||||
}, |
||||
methods: { |
||||
//调用完edit()方法之后会自动调用此方法 |
||||
editAfter() { |
||||
this.changeColumnHide() |
||||
this.changeFormTypes(this.materialTable.columns, 'snList', 0) |
||||
this.changeFormTypes(this.materialTable.columns, 'batchNumber', 0) |
||||
this.changeFormTypes(this.materialTable.columns, 'expirationDate', 0) |
||||
if (this.action === 'add') { |
||||
this.addInit(this.prefixNo) |
||||
this.fileList = [] |
||||
} else { |
||||
this.model.operTime = this.model.operTimeStr |
||||
this.model.debt = (this.model.discountLastMoney + this.model.otherMoney - this.model.changeAmount).toFixed(2) |
||||
if(this.model.accountId == null) { |
||||
this.model.accountId = 0 |
||||
this.manyAccountBtnStatus = true |
||||
this.accountIdList = this.model.accountIdList |
||||
this.accountMoneyList = this.model.accountMoneyList |
||||
} else { |
||||
this.manyAccountBtnStatus = false |
||||
} |
||||
this.fileList = this.model.fileName |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue(pick(this.model,'organId', 'operTime', 'number', 'linkNumber', 'remark', |
||||
'discount','discountMoney','discountLastMoney','otherMoney','accountId','changeAmount','debt')) |
||||
}); |
||||
// 加载子表数据 |
||||
let params = { |
||||
headerId: this.model.id, |
||||
mpList: getMpListShort(Vue.ls.get('materialPropertyList')) //扩展属性 |
||||
} |
||||
let url = this.readOnly ? this.url.detailList : this.url.detailList; |
||||
this.requestSubTableData(url, params, this.materialTable); |
||||
} |
||||
//复制新增单据-初始化单号和日期 |
||||
if(this.action === 'copyAdd') { |
||||
this.model.id = '' |
||||
this.model.tenantId = '' |
||||
this.copyAddInit(this.prefixNo) |
||||
} |
||||
this.initSupplier() |
||||
this.initDepot() |
||||
this.initAccount() |
||||
}, |
||||
//提交单据时整理成formData |
||||
classifyIntoFormData(allValues) { |
||||
let totalPrice = 0 |
||||
let billMain = Object.assign(this.model, allValues.formValue) |
||||
let detailArr = allValues.tablesValue[0].values |
||||
billMain.type = '出库' |
||||
billMain.subType = '采购退货' |
||||
billMain.defaultNumber = billMain.number |
||||
for(let item of detailArr){ |
||||
totalPrice += item.allPrice-0 |
||||
} |
||||
billMain.totalPrice = totalPrice |
||||
if(billMain.accountId === 0) { |
||||
billMain.accountId = '' |
||||
} |
||||
billMain.accountIdList = this.accountIdList.length>0 ? JSON.stringify(this.accountIdList) : "" |
||||
billMain.accountMoneyList = this.accountMoneyList.length>0 ? JSON.stringify(this.accountMoneyList) : "" |
||||
if(this.fileList && this.fileList.length > 0) { |
||||
billMain.fileName = this.fileList |
||||
} |
||||
if(this.model.id){ |
||||
billMain.id = this.model.id |
||||
} |
||||
return { |
||||
info: billMain, |
||||
rows: detailArr, |
||||
} |
||||
}, |
||||
onSearchLinkNumber() { |
||||
this.$refs.linkBillList.show('入库', '采购', '供应商', "1") |
||||
this.$refs.linkBillList.title = "选择采购入库" |
||||
}, |
||||
linkBillListOk(selectBillRows) { |
||||
if(selectBillRows && selectBillRows.length>0) { |
||||
let record = selectBillRows[0] |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue({ |
||||
'organId': record.organId, |
||||
'linkNumber': record.number, |
||||
'remark': record.remark, |
||||
'discountLastMoney': record.totalPrice, |
||||
'changeAmount': record.totalPrice |
||||
}) |
||||
}); |
||||
// 加载子表数据 |
||||
let params = { |
||||
headerId: record.id, |
||||
mpList: getMpListShort(Vue.ls.get('materialPropertyList')) //扩展属性 |
||||
} |
||||
this.requestSubTableDataEx(this.url.detailList, params, this.materialTable); |
||||
} |
||||
}, |
||||
/** 查询某个tab的数据,给明细里面的价税合计赋值 */ |
||||
requestSubTableDataEx(url, params, tab, success) { |
||||
tab.loading = true |
||||
getAction(url, params).then(res => { |
||||
if(res && res.code === 200){ |
||||
let list = res.data.rows |
||||
let listEx = [] |
||||
for(let j=0; j<list.length; j++){ |
||||
let info = list[j]; |
||||
info.taxMoney = 0 |
||||
info.taxLastMoney = info.allPrice |
||||
listEx.push(info) |
||||
this.changeColumnShow(info) |
||||
} |
||||
tab.dataSource = listEx |
||||
typeof success === 'function' ? success(res) : '' |
||||
} |
||||
}).finally(() => { |
||||
tab.loading = false |
||||
}) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
|
||||
</style> |
@ -0,0 +1,564 @@
|
||||
<template> |
||||
<j-modal |
||||
:title="title" |
||||
:width="width" |
||||
:visible="visible" |
||||
:confirmLoading="confirmLoading" |
||||
:maskClosable="false" |
||||
:keyboard="false" |
||||
:forceRender="true" |
||||
v-bind:prefixNo="prefixNo" |
||||
switchHelp |
||||
switchFullscreen |
||||
@ok="handleOk" |
||||
@cancel="handleCancel" |
||||
wrapClassName="ant-modal-cust-warp" |
||||
:id="prefixNo" |
||||
style="top: 5%; height: 100%; overflow-y: hidden" |
||||
> |
||||
<a-spin :spinning="confirmLoading"> |
||||
<a-form :form="form"> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item |
||||
:labelCol="labelCol" |
||||
:wrapperCol="wrapperCol" |
||||
label="供应商" |
||||
data-step="1" |
||||
data-title="供应商" |
||||
data-intro="供应商必须选择,如果发现需要选择的供应商尚未录入,可以在下拉框中点击新增供应商进行录入" |
||||
> |
||||
<a-select |
||||
placeholder="选择供应商" |
||||
v-decorator="['organId', validatorRules.organId]" |
||||
:dropdownMatchSelectWidth="false" |
||||
showSearch |
||||
> |
||||
<div slot="dropdownRender" slot-scope="menu"> |
||||
<v-nodes :vnodes="menu" /> |
||||
<a-divider style="margin: 4px 0" /> |
||||
<div |
||||
v-if="isTenant" |
||||
style="padding: 4px 8px; cursor: pointer" |
||||
@mousedown="(e) => e.preventDefault()" |
||||
@click="addSupplier" |
||||
> |
||||
<a-icon type="plus" /> 新增供应商 |
||||
</div> |
||||
</div> |
||||
<a-select-option v-for="(item, index) in supList" :key="index" :value="item.id"> |
||||
{{ item.supplier }} |
||||
</a-select-option> |
||||
</a-select> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="单据日期"> |
||||
<j-date v-decorator="['operTime', validatorRules.operTime]" :show-time="true" /> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item |
||||
:labelCol="labelCol" |
||||
:wrapperCol="wrapperCol" |
||||
label="单据编号" |
||||
data-step="2" |
||||
data-title="单据编号" |
||||
data-intro="单据编号自动生成、自动累加、开头是单据类型的首字母缩写,累加的规则是每次打开页面会自动占用一个新的编号" |
||||
> |
||||
<a-input placeholder="请输入单据编号" v-decorator.trim="['number']" :readOnly="true" /> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item |
||||
:labelCol="labelCol" |
||||
:wrapperCol="wrapperCol" |
||||
label="关联订单" |
||||
data-step="3" |
||||
data-title="关联订单" |
||||
data-intro="采购入库单据可以通过关联订单来选择已录入的订单,选择之后会自动加载订单的内容,然后继续录入仓库等信息完成单据的提交, |
||||
提交之后原来的采购订单会对应的改变单据状态。另外本系统支持订单多次入库,只需选择订单之后修改对应的商品数量即可" |
||||
> |
||||
<a-input-search |
||||
placeholder="请选择关联订单" |
||||
v-decorator="['linkNumber']" |
||||
@search="onSearchLinkNumber" |
||||
:readOnly="true" |
||||
/> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
<j-editable-table |
||||
id="billModal" |
||||
:ref="refKeys[0]" |
||||
:loading="materialTable.loading" |
||||
:columns="materialTable.columns" |
||||
:dataSource="materialTable.dataSource" |
||||
:maxHeight="300" |
||||
:rowNumber="false" |
||||
:rowSelection="true" |
||||
:actionButton="true" |
||||
:dragSort="true" |
||||
@valueChange="onValueChange" |
||||
@added="onAdded" |
||||
@deleted="onDeleted" |
||||
> |
||||
<template #buttonAfter> |
||||
<a-row |
||||
:gutter="24" |
||||
style="float: left" |
||||
data-step="4" |
||||
data-title="扫码录入" |
||||
data-intro="此功能支持扫码枪扫描商品物料编码进行录入" |
||||
> |
||||
<a-col v-if="scanStatus" :md="6" :sm="24"> |
||||
<a-button @click="scanEnter">扫码录入</a-button> |
||||
</a-col> |
||||
<a-col v-if="!scanStatus" :md="16" :sm="24" style="padding: 0 6px 0 12px"> |
||||
<a-input |
||||
placeholder="请扫码商品物料编码并回车" |
||||
v-model="scanBarCode" |
||||
@pressEnter="scanPressEnter" |
||||
ref="scanBarCode" |
||||
/> |
||||
</a-col> |
||||
<a-col v-if="!scanStatus" :md="6" :sm="24" style="padding: 0px"> |
||||
<a-button @click="stopScan">收起扫码</a-button> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row :gutter="24" style="float: left"> |
||||
<a-col :md="24" :sm="24"> |
||||
<a-dropdown> |
||||
<a-menu slot="overlay"> |
||||
<a-menu-item key="1" @click="handleBatchSetDepot"><a-icon type="setting" />批量设置</a-menu-item> |
||||
<a-menu-item v-if="isTenant" key="2" @click="addDepot"><a-icon type="plus" />新增仓库</a-menu-item> |
||||
</a-menu> |
||||
<a-button style="margin-left: 8px">仓库操作 <a-icon type="down" /></a-button> |
||||
</a-dropdown> |
||||
</a-col> |
||||
</a-row> |
||||
<a-button type="primary" @click="clearKh" style="margin-left: 10px">清空空行</a-button> |
||||
</template> |
||||
</j-editable-table> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="24" :md="24" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="{ xs: { span: 24 }, sm: { span: 24 } }" label=""> |
||||
<a-textarea :rows="1" placeholder="请输入备注" v-decorator="['remark']" style="margin-top: 8px" /> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item |
||||
:labelCol="labelCol" |
||||
:wrapperCol="wrapperCol" |
||||
label="优惠率" |
||||
data-step="5" |
||||
data-title="优惠率" |
||||
data-intro="针对单据明细中商品总金额进行优惠的比例" |
||||
> |
||||
<a-input |
||||
style="width: 185px" |
||||
placeholder="请输入优惠率" |
||||
v-decorator.trim="['discount']" |
||||
suffix="%" |
||||
@keyup="onKeyUpDiscount" |
||||
/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item |
||||
:labelCol="labelCol" |
||||
:wrapperCol="wrapperCol" |
||||
label="付款优惠" |
||||
data-step="6" |
||||
data-title="付款优惠" |
||||
data-intro="针对单据明细中商品总金额进行优惠的金额" |
||||
> |
||||
<a-input |
||||
placeholder="请输入付款优惠" |
||||
v-decorator.trim="['discountMoney']" |
||||
@keyup="onKeyUpDiscountMoney" |
||||
/> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item |
||||
:labelCol="labelCol" |
||||
:wrapperCol="wrapperCol" |
||||
label="优惠后金额" |
||||
data-step="7" |
||||
data-title="优惠后金额" |
||||
data-intro="针对单据明细中商品总金额进行优惠后的金额" |
||||
> |
||||
<a-input placeholder="请输入优惠后金额" v-decorator.trim="['discountLastMoney']" :readOnly="true" /> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item |
||||
:labelCol="labelCol" |
||||
:wrapperCol="wrapperCol" |
||||
label="其它费用" |
||||
data-step="8" |
||||
data-title="其它费用" |
||||
data-intro="比如快递费、油费、过路费" |
||||
> |
||||
<a-input placeholder="请输入其它费用" v-decorator.trim="['otherMoney']" @keyup="onKeyUpOtherMoney" /> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item |
||||
:labelCol="labelCol" |
||||
:wrapperCol="wrapperCol" |
||||
label="结算账户" |
||||
data-step="9" |
||||
data-title="结算账户" |
||||
data-intro="如果在下拉框中选择多账户,则可以通过多个结算账户进行结算" |
||||
> |
||||
<a-select |
||||
style="width: 185px" |
||||
placeholder="选择结算账户" |
||||
v-decorator="['accountId']" |
||||
:dropdownMatchSelectWidth="false" |
||||
allowClear |
||||
@select="selectAccount" |
||||
> |
||||
<div slot="dropdownRender" slot-scope="menu"> |
||||
<v-nodes :vnodes="menu" /> |
||||
<a-divider style="margin: 4px 0" /> |
||||
<div |
||||
v-if="isTenant" |
||||
style="padding: 4px 8px; cursor: pointer" |
||||
@mousedown="(e) => e.preventDefault()" |
||||
@click="addAccount" |
||||
> |
||||
<a-icon type="plus" /> 新增结算账户 |
||||
</div> |
||||
</div> |
||||
<a-select-option v-for="(item, index) in accountList" :key="index" :value="item.id"> |
||||
{{ item.name }} |
||||
</a-select-option> |
||||
</a-select> |
||||
<a-tooltip title="多账户明细"> |
||||
<a-button |
||||
type="default" |
||||
icon="folder" |
||||
style="margin-left: 8px" |
||||
size="small" |
||||
v-show="manyAccountBtnStatus" |
||||
@click="handleManyAccount" |
||||
/> |
||||
</a-tooltip> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="本次付款"> |
||||
<a-input placeholder="请输入本次付款" v-decorator.trim="['changeAmount']" @keyup="onKeyUpChangeAmount" /> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item |
||||
:labelCol="labelCol" |
||||
:wrapperCol="wrapperCol" |
||||
label="本次欠款" |
||||
data-step="10" |
||||
data-title="本次欠款" |
||||
data-intro="欠款产生的费用,后续可以在付款单进行支付" |
||||
> |
||||
<a-input placeholder="请输入本次欠款" v-decorator.trim="['debt']" :readOnly="true" /> |
||||
</a-form-item> |
||||
</a-col> |
||||
<a-col :lg="6" :md="12" :sm="24"> </a-col> |
||||
</a-row> |
||||
<a-row class="form-row" :gutter="24"> |
||||
<a-col :lg="6" :md="12" :sm="24"> |
||||
<a-form-item |
||||
:labelCol="labelCol" |
||||
:wrapperCol="wrapperCol" |
||||
label="附件" |
||||
data-step="11" |
||||
data-title="附件" |
||||
data-intro="可以上传与单据相关的图片、文档,支持多个文件" |
||||
> |
||||
<j-upload v-model="fileList" bizPath="bill"></j-upload> |
||||
</a-form-item> |
||||
</a-col> |
||||
</a-row> |
||||
</a-form> |
||||
</a-spin> |
||||
<many-account-modal ref="manyAccountModalForm" @ok="manyAccountModalFormOk"></many-account-modal> |
||||
<link-bill-list ref="linkBillList" @ok="linkBillListOk"></link-bill-list> |
||||
<vendor-modal ref="vendorModalForm" @ok="vendorModalFormOk"></vendor-modal> |
||||
<depot-modal ref="depotModalForm" @ok="depotModalFormOk"></depot-modal> |
||||
<account-modal ref="accountModalForm" @ok="accountModalFormOk"></account-modal> |
||||
<batch-set-depot ref="batchSetDepotModalForm" @ok="batchSetDepotModalFormOk"></batch-set-depot> |
||||
</j-modal> |
||||
</template> |
||||
|
||||
<script> |
||||
import pick from 'lodash.pick' |
||||
import ManyAccountModal from '../dialog/ManyAccountModal' |
||||
import LinkBillList from '../dialog/LinkBillList' |
||||
import VendorModal from '../../system/modules/VendorModal' |
||||
import DepotModal from '../../system/modules/DepotModal' |
||||
import AccountModal from '../../system/modules/AccountModal' |
||||
import BatchSetDepot from '../dialog/BatchSetDepot' |
||||
import { FormTypes } from '@/utils/JEditableTableUtil' |
||||
import { JEditableTableMixin } from '@/mixins/JEditableTableMixin' |
||||
import { BillModalMixin } from '../mixins/BillModalMixin' |
||||
import { getMpListShort, changeListFmtMinus } from "@/utils/util" |
||||
import { getAction } from '@/api/manage' |
||||
import JUpload from '@/components/jeecg/JUpload' |
||||
import JDate from '@/components/jeecg/JDate' |
||||
import Vue from 'vue' |
||||
export default { |
||||
name: "PurchaseInModal", |
||||
mixins: [JEditableTableMixin, BillModalMixin], |
||||
components: { |
||||
ManyAccountModal, |
||||
LinkBillList, |
||||
VendorModal, |
||||
DepotModal, |
||||
AccountModal, |
||||
BatchSetDepot, |
||||
JUpload, |
||||
JDate, |
||||
VNodes: { |
||||
functional: true, |
||||
render: (h, ctx) => ctx.props.vnodes, |
||||
} |
||||
}, |
||||
data () { |
||||
return { |
||||
title:"操作", |
||||
width: '100%', |
||||
moreStatus: false, |
||||
// 新增时子表默认添加几行空数据 |
||||
addDefaultRowNum: 10, |
||||
visible: false, |
||||
operTimeStr: '', |
||||
prefixNo: 'CGRK', |
||||
fileList:[], |
||||
model: {}, |
||||
labelCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 8 }, |
||||
}, |
||||
wrapperCol: { |
||||
xs: { span: 24 }, |
||||
sm: { span: 16 }, |
||||
}, |
||||
refKeys: ['materialDataTable', ], |
||||
activeKey: 'materialDataTable', |
||||
materialTable: { |
||||
loading: false, |
||||
dataSource: [], |
||||
columns: [ |
||||
{ title: '仓库名称', key: 'depotId', width: '7%', type: FormTypes.select, placeholder: '请选择${title}', options: [], |
||||
allowSearch:true, validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '物料编码', key: 'barCode', width: '8%', type: FormTypes.popupJsh, kind: 'material', multi: true, |
||||
validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '名称', key: 'name', width: '20%', type: FormTypes.normal }, |
||||
{ title: '规格', key: 'standard', width: '5%', type: FormTypes.normal }, |
||||
{ title: '型号', key: 'model', width: '5%', type: FormTypes.normal }, |
||||
{ title: '颜色', key: 'color', width: '5%', type: FormTypes.normal }, |
||||
{ title: '扩展信息', key: 'materialOther', width: '5%', type: FormTypes.normal }, |
||||
{ title: '库存', key: 'stock', width: '5%', type: FormTypes.inputNumber, readonly:true }, |
||||
{ title: '单位', key: 'unit', width: '4%', type: FormTypes.normal }, |
||||
{ title: '序列号', key: 'snList', width: '12%', type: FormTypes.input, placeholder: '多个序列号请用逗号隔开', |
||||
validateRules: [{ pattern: /^\S{1,100}$/, message: '请小于100位字符' }] |
||||
}, |
||||
{ title: '批号', key: 'batchNumber', width: '5%', type: FormTypes.input }, |
||||
{ title: '有效期', key: 'expirationDate',width: '7%', type: FormTypes.date }, |
||||
{ title: '多属性', key: 'sku', width: '4%', type: FormTypes.normal }, |
||||
{ title: '原数量', key: 'preNumber', width: '4%', type: FormTypes.normal }, |
||||
{ title: '已入库', key: 'finishNumber', width: '4%', type: FormTypes.normal }, |
||||
{ title: '数量', key: 'operNumber', width: '4%', type: FormTypes.inputNumber, statistics: true, |
||||
validateRules: [{ required: true, message: '${title}不能为空' }] |
||||
}, |
||||
{ title: '单价', key: 'unitPrice', width: '4%', type: FormTypes.inputNumber, validateRules: [{ required: true, message: '${title}不能为空' }]}, |
||||
{ title: '金额', key: 'allPrice', width: '5%', type: FormTypes.inputNumber, statistics: true }, |
||||
{ title: '税率', key: 'taxRate', width: '3%', type: FormTypes.inputNumber,placeholder: '%'}, |
||||
{ title: '税额', key: 'taxMoney', width: '5%', type: FormTypes.inputNumber, readonly: true, statistics: true }, |
||||
{ title: '价税合计', key: 'taxLastMoney', width: '5%', type: FormTypes.inputNumber, statistics: true }, |
||||
{ title: '备注', key: 'remark', width: '5%', type: FormTypes.input } |
||||
] |
||||
}, |
||||
confirmLoading: false, |
||||
validatorRules:{ |
||||
operTime:{ |
||||
rules: [ |
||||
{ required: true, message: '请输入单据日期!' } |
||||
] |
||||
}, |
||||
organId:{ |
||||
rules: [ |
||||
{ required: true, message: '请选择供应商!' } |
||||
] |
||||
}, |
||||
accountId:{ |
||||
rules: [ |
||||
{ required: true, message: '请选择结算账户!' } |
||||
] |
||||
} |
||||
}, |
||||
url: { |
||||
add: '/depotHead/addDepotHeadAndDetail', |
||||
edit: '/depotHead/updateDepotHeadAndDetail', |
||||
detailList: '/depotItem/getDetailList' |
||||
} |
||||
} |
||||
}, |
||||
created () { |
||||
}, |
||||
methods: { |
||||
//调用完edit()方法之后会自动调用此方法 |
||||
editAfter() { |
||||
this.changeColumnHide() |
||||
this.changeFormTypes(this.materialTable.columns, 'snList', 0) |
||||
this.changeFormTypes(this.materialTable.columns, 'batchNumber', 0) |
||||
this.changeFormTypes(this.materialTable.columns, 'expirationDate', 0) |
||||
this.changeFormTypes(this.materialTable.columns, 'preNumber', 0) |
||||
this.changeFormTypes(this.materialTable.columns, 'finishNumber', 0) |
||||
if (this.action === 'add') { |
||||
this.addInit(this.prefixNo) |
||||
this.fileList = [] |
||||
} else { |
||||
this.model.operTime = this.model.operTimeStr |
||||
this.model.debt = (this.model.discountLastMoney + this.model.otherMoney - this.model.changeAmount).toFixed(2) |
||||
if(this.model.accountId == null) { |
||||
this.model.accountId = 0 |
||||
this.manyAccountBtnStatus = true |
||||
this.accountIdList = this.model.accountIdList |
||||
this.accountMoneyList = this.model.accountMoneyList |
||||
} else { |
||||
this.manyAccountBtnStatus = false |
||||
} |
||||
this.fileList = this.model.fileName |
||||
this.$nextTick(() => { |
||||
|
||||
this.form.setFieldsValue(pick(this.model,'organId', 'operTime', 'number', 'linkNumber', 'remark', |
||||
'discount','discountMoney','discountLastMoney','otherMoney','accountId','changeAmount','debt')) |
||||
}); |
||||
// 加载子表数据 |
||||
let params = { |
||||
headerId: this.model.id, |
||||
mpList: getMpListShort(Vue.ls.get('materialPropertyList')) //扩展属性 |
||||
} |
||||
let url = this.readOnly ? this.url.detailList : this.url.detailList; |
||||
this.requestSubTableData(url, params, this.materialTable); |
||||
} |
||||
//复制新增单据-初始化单号和日期 |
||||
if(this.action === 'copyAdd') { |
||||
this.model.id = '' |
||||
this.model.tenantId = '' |
||||
this.copyAddInit(this.prefixNo) |
||||
} |
||||
this.initSupplier() |
||||
this.initDepot() |
||||
this.initAccount() |
||||
}, |
||||
//提交单据时整理成formData |
||||
classifyIntoFormData(allValues) { |
||||
let totalPrice = 0 |
||||
let billMain = Object.assign(this.model, allValues.formValue) |
||||
let detailArr = allValues.tablesValue[0].values |
||||
billMain.type = '入库' |
||||
billMain.subType = '采购' |
||||
billMain.defaultNumber = billMain.number |
||||
for(let item of detailArr){ |
||||
totalPrice += item.allPrice-0 |
||||
} |
||||
billMain.totalPrice = 0-totalPrice |
||||
billMain.changeAmount = 0-billMain.changeAmount |
||||
if(billMain.accountId === 0) { |
||||
billMain.accountId = '' |
||||
} |
||||
this.accountMoneyList = changeListFmtMinus(this.accountMoneyList) |
||||
billMain.accountIdList = this.accountIdList.length>0 ? JSON.stringify(this.accountIdList) : "" |
||||
billMain.accountMoneyList = this.accountMoneyList.length>0 ? JSON.stringify(this.accountMoneyList) : "" |
||||
if(this.fileList && this.fileList.length > 0) { |
||||
billMain.fileName = this.fileList |
||||
} |
||||
if(this.model.id){ |
||||
billMain.id = this.model.id |
||||
} |
||||
return { |
||||
info: billMain, |
||||
rows: detailArr, |
||||
} |
||||
}, |
||||
onSearchLinkNumber() { |
||||
this.$refs.linkBillList.show('其它', '采购订单', '供应商', "1,3") |
||||
this.$refs.linkBillList.title = "选择采购订单" |
||||
}, |
||||
linkBillListOk(selectBillRows) { |
||||
console.log(selectBillRows,'selectBillRows') |
||||
this.changeFormTypes(this.materialTable.columns, 'preNumber', 1) |
||||
this.changeFormTypes(this.materialTable.columns, 'finishNumber', 1) |
||||
if(selectBillRows && selectBillRows.length>0) { |
||||
let record = selectBillRows[0] |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue({ |
||||
'organId': record.organId, |
||||
'linkNumber': record.number, |
||||
'remark': record.remark, |
||||
'discountLastMoney': record.totalPrice, |
||||
'changeAmount': record.totalPrice, |
||||
'operTime': record.operTimeStr |
||||
}) |
||||
}); |
||||
// 加载子表数据 |
||||
let params = { |
||||
headerId: record.id, |
||||
mpList: getMpListShort(Vue.ls.get('materialPropertyList')) //扩展属性 |
||||
} |
||||
this.requestSubTableDataEx(this.url.detailList, params, this.materialTable); |
||||
} |
||||
}, |
||||
/** 查询某个tab的数据,给明细里面的价税合计赋值 */ |
||||
requestSubTableDataEx(url, params, tab, success) { |
||||
tab.loading = true |
||||
getAction(url, params).then(res => { |
||||
if(res && res.code === 200){ |
||||
let list = res.data.rows |
||||
let listEx = [] |
||||
let discountLastMoney = 0 |
||||
for(let j=0; j<list.length; j++){ |
||||
let info = list[j]; |
||||
if(info.preNumber) { |
||||
info.operNumber = info.preNumber - info.finishNumber |
||||
// info.allPrice = info.allPrice; |
||||
// info.allPrice = info.operNumber * info.unitPrice-0; |
||||
discountLastMoney += info.allPrice |
||||
} |
||||
info.taxMoney = 0 |
||||
info.taxLastMoney = info.allPrice |
||||
listEx.push(info) |
||||
this.changeColumnShow(info) |
||||
} |
||||
tab.dataSource = listEx |
||||
//给优惠后金额重新赋值 |
||||
if(discountLastMoney) { |
||||
this.$nextTick(() => { |
||||
this.form.setFieldsValue({ |
||||
'discountLastMoney': discountLastMoney, |
||||
'changeAmount': discountLastMoney |
||||
}) |
||||
}); |
||||
} |
||||
typeof success === 'function' ? success(res) : '' |
||||
} |
||||
}).finally(() => { |
||||
tab.loading = false |
||||
}) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
<style scoped> |
||||
</style> |
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in new issue