Browse Source

子账户前段结构

chenlei1231 1 month ago
parent
commit
c190adac56

+ 46 - 0
src/views/adweb/enterprise/EnterpriseUserManage.api.ts

@@ -0,0 +1,46 @@
+import { defHttp } from '/@/utils/http/axios';
+import { useMessage } from '/@/hooks/web/useMessage';
+
+const { createConfirm } = useMessage();
+
+enum Api {
+  list = '/enterprise/user/list',
+  deleteOne = '/sys/user/delete',
+  deleteBatch = '/sys/user/deleteBatch',
+}
+
+/**
+ * 列表接口
+ * @param params
+ */
+export const list = (params) => defHttp.get({ url: Api.list, params });
+
+export const queryEnterpriseAll = (params) => defHttp.get({ url: '/enterprise/role/queryall', params });
+
+export const addEnterpriseUser = (params) => defHttp.post({ url: '/enterprise/user/add', params });
+/**
+ * 删除单个
+ */
+export const deleteOne = (params, handleSuccess) => {
+  return defHttp.delete({ url: Api.deleteOne, params }, { joinParamsToUrl: true }).then(() => {
+    handleSuccess();
+  });
+};
+/**
+ * 批量删除
+ * @param params
+ */
+export const batchDelete = (params, handleSuccess) => {
+  createConfirm({
+    iconType: 'warning',
+    title: '确认删除',
+    content: '是否删除选中数据',
+    okText: '确认',
+    cancelText: '取消',
+    onOk: () => {
+      return defHttp.delete({ url: Api.deleteBatch, data: params }, { joinParamsToUrl: true }).then(() => {
+        handleSuccess();
+      });
+    },
+  });
+};

+ 168 - 0
src/views/adweb/enterprise/EnterpriseUserManage.data.ts

@@ -0,0 +1,168 @@
+import { BasicColumn } from '@/components/Table';
+
+export const columns: BasicColumn[] = [
+  {
+    title: '序号',
+    width: 60,
+    align: 'center',
+    dataIndex: 'seqNumber',
+  },
+  {
+    title: '用户名',
+    align: 'left',
+    dataIndex: 'realname',
+  },
+  {
+    title: '登录账号',
+    align: 'left',
+    dataIndex: 'username',
+  },
+  {
+    title: '头像',
+    align: 'left',
+    dataIndex: 'avatar',
+    width: 80,
+  },
+  {
+    title: '类型',
+    align: 'left',
+    width: 80,
+    dataIndex: 'primaryAccount',
+    customRender: function (text) {
+      return text ? '主账户' : '子账户';
+    },
+  },
+  {
+    title: '分配的询盘数',
+    align: 'left',
+    dataIndex: 'enquiryNum',
+  },
+  // {
+  //   title: '负责的产品数',
+  //   align: "left",
+  //   dataIndex: 'productNum',
+  //   scopedSlots: {customRender: "countSlot"}
+  // },
+  {
+    title: '手机号码',
+    align: 'left',
+    dataIndex: 'phone',
+    customRender: function (text) {
+      return text === null ? '--' : text;
+    },
+  },
+  {
+    title: '邮箱',
+    align: 'left',
+    dataIndex: 'email',
+    customRender: function (text) {
+      return text === null ? '--' : text;
+    },
+  },
+  {
+    title: '更新时间',
+    align: 'left',
+    dataIndex: 'updateTime',
+    customRender: function (text) {
+      return text === null ? '--' : text;
+    },
+  },
+  {
+    title: '操作',
+    dataIndex: 'action',
+    fixed: 'right',
+
+    align: 'left',
+  },
+];
+
+export const subColumns: BasicColumn[] = [
+  {
+    title: '序号',
+    width: 60,
+    align: 'center',
+    dataIndex: 'seqNumber',
+  },
+  {
+    title: '用户名',
+    align: 'left',
+    dataIndex: 'realname',
+  },
+  {
+    title: '登录账号',
+    align: 'left',
+    dataIndex: 'username',
+  },
+  {
+    title: '头像',
+    align: 'left',
+    dataIndex: 'avatar',
+    width: 80,
+  },
+  {
+    title: '类型',
+    align: 'left',
+    width: 80,
+    dataIndex: 'primaryAccount',
+    customRender: function (text) {
+      return text ? '主账户' : '子账户';
+    },
+  },
+  {
+    title: '性别',
+    align: 'left',
+    dataIndex: 'sex',
+    customRender: function (text) {
+      if (text === null) {
+        return '-';
+      }
+      return text === 1 ? '男' : '女';
+    },
+  },
+  {
+    title: '手机号码',
+    align: 'left',
+    dataIndex: 'phone',
+    customRender: function (text) {
+      return text === null ? '--' : text;
+    },
+  },
+  {
+    title: '邮箱',
+    align: 'left',
+    dataIndex: 'email',
+    customRender: function (text) {
+      return text === null ? '--' : text;
+    },
+  },
+  {
+    title: '更新时间',
+    align: 'left',
+    dataIndex: 'updateTime',
+    customRender: function (text) {
+      return text === null ? '--' : text;
+    },
+  },
+  {
+    title: '主账号询盘',
+    align: 'center',
+    dataIndex: 'isInquiry',
+    customRender: function (text) {
+      return text === 1 ? '不接收' : '接收中';
+    },
+  },
+  {
+    title: '询盘接收',
+    dataIndex: 'action',
+    fixed: 'right',
+
+    align: 'left',
+  },
+  {
+    title: '操作',
+    dataIndex: 'action1',
+    fixed: 'right',
+
+    align: 'left',
+  },
+];

+ 292 - 0
src/views/adweb/enterprise/EnterpriseUserManageList.vue

@@ -0,0 +1,292 @@
+<template>
+  <a-card :bordered="false">
+    <!-- 查询区域 -->
+    <div class="table-page-search-wrapper">
+      <a-form layout="inline" @keyup.enter.native="searchQuery" style="display: block">
+        <a-row :gutter="24">
+          <a-col :md="6" :sm="12">
+            <a-form-item>
+              <a-input placeholder="请输入用户账号/用户名/手机号" v-model="queryParam.queryCondition" />
+            </a-form-item>
+          </a-col>
+          <a-col :md="18" :sm="12">
+            <span style="float: left; overflow: hidden" class="table-page-search-submitButtons">
+              <a-button type="primary" @click="searchQuery" preIcon="ant-design:search-outlined">查询</a-button>
+              <a-button ghost type="primary" @click="searchReset" preIcon="ant-design:reload-outlined" style="margin-left: 8px">重置</a-button>
+              <a-button ghost @click="handleAdd" type="primary" style="margin-left: 8px" preIcon="ant-design:plus-outlined">添加子用户</a-button>
+              <a-button ghost type="primary" @click="showRecycleBin" style="margin-left: 8px" preIcon="ant-design:hdd-outlined">回收站</a-button>
+              <!--              <router-link :to="{ path: '/inquery/rules' }" v-if="siteCode !== dictSiteCode">-->
+              <!--                <a-button ghost type="primary" icon="logout" @click="showRecycleBin" style="margin-left: 8px">配置询盘规则</a-button>-->
+              <!--              </router-link>-->
+            </span>
+          </a-col>
+        </a-row>
+      </a-form>
+    </div>
+
+    <!-- table区域-begin -->
+    <div v-if="siteCode !== dictSiteCode"> <BasicTable @register="registerTable" :rowSelection="rowSelection" /></div>
+    <!-- table区域-end -->
+
+    <!--    table-子账户轮询-begin-->
+    <div v-if="siteCode === dictSiteCode">
+      <h1>关闭配置或新增子账号会重置轮流分配</h1>
+      <a-table
+        ref="table"
+        size="middle"
+        rowKey="id"
+        :scroll="{ x: true }"
+        :columns="subColumns"
+        :dataSource="subAccountList"
+        :pagination="ipagination1"
+        :loading="loading"
+        @change="handleTableChange"
+      >
+        <template #avatarslot="text, record, index">
+          <div class="anty-img-wrap" style="margin-bottom: 5px">
+            <a-avatar shape="square" :src="getAvatarView(record.avatar)" icon="user" />
+          </div>
+        </template>
+
+        <template #action="text, record">
+          <span>
+            <div>
+              <a-switch
+                :checked="record.isInquiry === 2"
+                @change="(checked) => handleSwitchChange(record, checked)"
+                checked-children="开"
+                un-checked-children="关"
+                :disabled="record.primaryAccount"
+              />
+            </div>
+          </span>
+        </template>
+
+        <template #action1="text, record">
+          <span>
+            <div style="padding: 8px">
+              <a-tag color="purple" @click="handleEdit(record)" style="cursor: pointer"> 编辑 </a-tag>
+              <!--            <a-tag v-if="!record.primaryAccount" color="orange" @click="handlePerssion(record.id)" style="cursor:pointer;">-->
+              <!--              菜单配置-->
+              <!--            </a-tag>-->
+              <a-tag color="green" @click="handleChangePassword(record.username)" style="cursor: pointer"> 重置密码 </a-tag>
+              <a-popconfirm v-if="!record.primaryAccount" title="确定删除吗?" @confirm="() => handleFrozen(record.id, 2)">
+                <a-tag color="red" style="cursor: pointer"> 删除 </a-tag>
+              </a-popconfirm>
+            </div>
+          </span>
+        </template>
+        <template #countSlot="text, record, index">
+          <span style="margin-left: 20px">{{ text }} 个</span>
+        </template>
+      </a-table>
+    </div>
+    <!--    table-子账户轮询-begin-->
+
+    <!-- 编辑、新增用户 -->
+    <enterprise-user-modal ref="registerModal" @ok="modalFormOk" />
+
+    <!-- 权限分配 -->
+    <enterprise-user-role-modal ref="modalUserRole" />
+    <!-- 用户回收站 -->
+    <enterprise-user-recycle-bin-modal ref="recycleBin" @ok="modalFormOk" />
+  </a-card>
+</template>
+
+<script lang="ts" setup name="EnterpriseUserManageList">
+  import '@/assets/less/common.less';
+  import EnterpriseUserModal from './modules/EnterpriseUserModal.vue';
+
+  import { getAction, getFileAccessHttpUrl, putAction } from '@/api/manage/manage';
+  import EnterpriseUserRecycleBinModal from './modules/EnterpriseUserRecycleBinModal.vue';
+  import EnterpriseUserRoleModal from './modules/EnterpriseUserRoleModal.vue';
+
+  import { ref, reactive, onMounted, nextTick, watch, toRaw, toRefs, computed, onBeforeMount } from 'vue';
+  import { BasicTable } from '@/components/Table';
+  import { useMessage } from '@/hooks/web/useMessage';
+  import { useListPage } from '@/hooks/system/useListPage';
+  import { batchDelete, deleteOne, list } from '@/views/adweb/enquiry/AdwebEnquiry.api';
+  import { columns, subColumns } from '@/views/adweb/enterprise/EnterpriseUserManage.data';
+
+  let subAccountList = reactive([]);
+  const registerModal = ref();
+  const ipagination1 = reactive({
+    current: 1,
+    pageSize: 10,
+    pageSizeOptions: ['10', '20', '30'],
+    showTotal: (total, range) => {
+      return range[0] + '-' + range[1] + ' 共' + total + '条';
+    },
+  });
+  const queryParam = reactive<any>({});
+  const loading = ref(false);
+  const siteCode = ref();
+  const dictSiteCode = ref();
+
+  const { createMessage } = useMessage();
+  //注册table数据
+  const { prefixCls, tableContext } = useListPage({
+    tableProps: {
+      title: '账户管理',
+      api: list,
+      columns,
+      canResize: false,
+      useSearchForm: false,
+      actionColumn: {
+        width: 180,
+        fixed: 'right',
+      },
+      striped: true,
+      bordered: false,
+      immediate: false, // 不直接触发,通过reload事件触发接口
+      beforeFetch: (params) => {
+        return Object.assign(params, queryParam);
+      },
+    },
+  });
+  const [
+    registerTable,
+    { reload, clearSelectedRowKeys, updateTableDataRecord, findTableDataRecord, getDataSource },
+    { rowSelection, selectedRowKeys },
+  ] = tableContext;
+
+  onBeforeMount(() => {
+    subList();
+    siteCode.value = localStorage.getItem('siteCode');
+    getTurnInquiryCode();
+  });
+
+  //获取需要轮流询盘的站点code
+  function getTurnInquiryCode() {
+    getAction('/adweb/adwebEnquiry/getTurnInquiryCode', {}).then((res) => {
+      if (res.code === 200) {
+        dictSiteCode.value = res.result[0].value;
+      }
+    });
+  }
+  function handleFrozen(id, status) {
+    loading.value = true;
+    putAction('/enterprise/user/frozen', { ids: id, status: status }).then((res) => {
+      if (res.success) {
+        createMessage.success(res.message);
+        if (siteCode.value === dictSiteCode.value) {
+          subList();
+        } else {
+          loadData();
+        }
+      } else {
+        createMessage.warning(res.message);
+      }
+      loading.value = false;
+    });
+  }
+  function getAvatarView(avatar) {
+    return getFileAccessHttpUrl(avatar, 'https');
+  }
+  function showRecycleBin() {
+    this.$refs.recycleBin.show();
+  }
+  function searchQuery() {
+    if (siteCode.value === dictSiteCode.value) {
+      subList();
+    } else {
+      loadData();
+    }
+    onClearSelected();
+  }
+  function searchReset() {
+    queryParam = {};
+    if (siteCode.value === dictSiteCode.value) {
+      subList();
+    } else {
+      loadData(1);
+    }
+  }
+  function handleChangePassword(username) {
+    this.$refs.passwordmodal.show(username);
+  }
+  function handlePerssion(userId) {
+    this.$refs.modalUserRole.show(userId);
+  }
+
+  function modalFormOk() {
+    // 新增/修改 成功时,重载列表
+    if (siteCode.value === dictSiteCode.value) {
+      subList();
+    } else {
+      loadData();
+    }
+  }
+  //获取子账户
+  function subList() {
+    let params = queryParam;
+    params.siteCode = localStorage.getItem('siteCode');
+    getAction('/enterprise/user/sub/list', params).then((res) => {
+      if (res.success) {
+        subAccountList = res.result.records;
+        ipagination1.total = res.result.total;
+      }
+      if (siteCode.value === dictSiteCode.value) {
+        loading.value = false;
+      }
+    });
+  }
+  function handleSwitchChange(record, checked) {
+    // 处理开关切换操作
+    record.isInquiry = checked ? 2 : 1;
+    putAction('/enterprise/user/updateIsInquiry', record).then((res) => {
+      if (res.success) {
+        this.$message.success(res.message);
+        this.subList();
+      } else {
+        this.$message.warning(res.message);
+        this.subList();
+      }
+    });
+  }
+
+  /**
+   * 新增事件
+   */
+  function handleAdd() {
+    registerModal.value.disableSubmit = false;
+    registerModal.value.add();
+  }
+
+  /**
+   * 编辑事件
+   */
+  function handleEdit(record: Recordable) {
+    registerModal.value.disableSubmit = false;
+    registerModal.value.edit(record);
+  }
+
+  /**
+   * 详情
+   */
+  function handleDetail(record: Recordable) {
+    registerModal.value.disableSubmit = true;
+    registerModal.value.edit(record);
+  }
+
+  /**
+   * 删除事件
+   */
+  async function handleDelete(record) {
+    await deleteOne({ id: record.id }, handleSuccess);
+  }
+
+  /**
+   * 批量删除事件
+   */
+  async function batchHandleDelete() {
+    await batchDelete({ id: selectedRowKeys.value.join(',') }, handleSuccess);
+  }
+
+  /**
+   * 成功回调
+   */
+  function handleSuccess() {
+    (selectedRowKeys.value = []) && reload();
+  }
+</script>

+ 182 - 0
src/views/adweb/enterprise/modules/EnterpriseDepartWindow.vue

@@ -0,0 +1,182 @@
+<template>
+  <a-modal
+    :width="modalWidth"
+    :visible="visible"
+    title="部门搜索"
+    :confirmLoading="confirmLoading"
+    @ok="handleSubmit"
+    @cancel="handleCancel"
+    cancelText="关闭"
+    wrapClassName="ant-modal-cust-warp"
+  >
+    <!--部门树-->
+    <template>
+      <a-form :form="form">
+        <a-form-item :labelCol="labelCol" :wrapperCol="wrapperCol" label="上级部门">
+          <a-tree
+            multiple
+            treeCheckable="tree"
+            checkable
+            @expand="onExpand"
+            :expandedKeys="expandedKeysss"
+            :checkedKeys="checkedKeys"
+            allowClear="true"
+            :checkStrictly="true"
+            @check="onCheck"
+            :dropdownStyle="{ maxHeight: '200px', overflow: 'auto' }"
+            :treeData="departTree"
+            placeholder="请选择上级部门"
+          />
+        </a-form-item>
+      </a-form>
+    </template>
+  </a-modal>
+</template>
+
+<script lang="ts">
+  import { getAction } from '@/api/manage/manage';
+  import { queryIdTree } from '@/views/system/departUser/depart.user.api';
+
+  export default {
+    name: 'EnterpriseDepartWindow',
+
+    data() {
+      return {
+        checkedKeys: [], // 存储选中的部门id
+        expandedKeysss: [], //展开的节点
+        userId: '', // 存储用户id
+        model: {}, // 存储SysUserDepartsVO表
+        userDepartModel: { userId: '', departIdList: [] }, // 存储用户id一对多部门信息的对象
+        departList: [], // 存储部门信息
+        modalWidth: 400,
+        departTree: [],
+        title: '操作',
+        visible: false,
+        labelCol: {
+          xs: { span: 24 },
+          sm: { span: 5 },
+        },
+        wrapperCol: {
+          xs: { span: 24 },
+          sm: { span: 16 },
+        },
+        confirmLoading: false,
+        headers: {},
+        form: this.$form.createForm(this),
+        url: {
+          userId: '/sys/user/generateUserId', // 引入生成添加用户情况下的url
+        },
+      };
+    },
+    methods: {
+      add(checkedDepartKeys, userId) {
+        this.checkedKeys = checkedDepartKeys;
+        this.userId = userId;
+        this.edit({});
+      },
+      edit(record) {
+        this.departList = [];
+        this.queryDepartTree();
+        this.form.resetFields();
+        this.visible = true;
+        this.model = Object.assign({}, record);
+        let filedsVal = pick(this.model, 'id', 'userId', 'departIdList');
+        this.$nextTick(() => {
+          this.form.setFieldsValue(filedsVal);
+        });
+      },
+      close() {
+        this.$emit('close');
+        this.visible = false;
+        this.departList = [];
+        this.checkedKeys = [];
+      },
+      handleSubmit() {
+        const that = this;
+        // 触发表单验证
+        this.form.validateFields((err) => {
+          if (!err) {
+            that.confirmLoading = true;
+            if (this.userId == null) {
+              getAction(this.url.userId)
+                .then((res) => {
+                  if (res.success) {
+                    let formData = { userId: res.result, departIdList: this.departList };
+                    console.log(formData);
+                    that.$emit('ok', formData);
+                  }
+                })
+                .finally(() => {
+                  that.departList = [];
+                  that.confirmLoading = false;
+                  that.close();
+                });
+            } else {
+              let formData = { userId: this.userId, departIdList: this.departList };
+              console.log(formData);
+              that.departList = [];
+              that.$emit('ok', formData);
+              that.confirmLoading = false;
+              that.close();
+            }
+          }
+        });
+      },
+      handleCancel() {
+        this.close();
+      },
+
+      // 选择部门时作用的API
+      onCheck(checkedKeys, info) {
+        this.departList = [];
+        this.checkedKeys = checkedKeys.checked;
+        let checkedNodes = info.checkedNodes;
+        for (let i = 0; i < checkedNodes.length; i++) {
+          let de = checkedNodes[i].data.props;
+          let depart = { key: '', value: '', title: '' };
+          depart.key = de.value;
+          depart.value = de.value;
+          depart.title = de.title;
+          this.departList.push(depart);
+        }
+        console.log('onCheck', checkedKeys, info);
+      },
+      queryDepartTree() {
+        queryIdTree().then((res) => {
+          if (res.success) {
+            this.departTree = res.result;
+            if (this.checkedKeys && this.checkedKeys.length > 0) {
+              let treekey = [];
+              let arr = res.result;
+              if (arr && arr.length > 0) {
+                arr.forEach((item) => {
+                  treekey.push(item.key);
+                  /*     if(item.children&&item.children.length>0){
+                    item.children.forEach(item1 => {
+                      treekey.push(item1.key);
+                    })
+                  }*/
+                });
+                this.expandedKeysss = treekey;
+              }
+            }
+          }
+        });
+      },
+      onExpand(expandedKeys) {
+        this.expandedKeysss = expandedKeys;
+      },
+      modalFormOk() {},
+    },
+  };
+</script>
+
+<style lang="less" scoped>
+  .ant-table-tbody .ant-table-row td {
+    padding-top: 10px;
+    padding-bottom: 10px;
+  }
+  /deep/ .ant-modal {
+    height: 700px;
+  }
+</style>

+ 120 - 0
src/views/adweb/enterprise/modules/EnterpriseRoleDataruleModal.vue

@@ -0,0 +1,120 @@
+<template>
+  <a-drawer
+    title="数据规则/按钮权限配置"
+    width="365"
+    :closable="false"
+    @close="onClose"
+    :visible="visible"
+  >
+
+    <a-tabs defaultActiveKey="1">
+      <a-tab-pane tab="数据规则" key="1">
+
+        <a-checkbox-group v-model="dataruleChecked" v-if="dataruleList.length>0">
+          <a-row>
+            <a-col :span="24" v-for="(item,index) in dataruleList" :key=" 'dr'+index ">
+              <a-checkbox :value="item.id">{{ item.ruleName }}</a-checkbox>
+            </a-col>
+
+            <a-col :span="24">
+              <div style="width: 100%;margin-top: 15px">
+                <a-button @click="saveDataruleForRole" type="primary" size="small" icon="save">点击保存</a-button>
+              </div>
+            </a-col>
+          </a-row>
+        </a-checkbox-group>
+        <div v-else><h3>无配置信息!</h3></div>
+
+      </a-tab-pane>
+      <!--<a-tab-pane tab="按钮权限" key="2">敬请期待!!!</a-tab-pane>-->
+    </a-tabs>
+
+  </a-drawer>
+</template>
+
+<script>
+  import ARow from 'ant-design-vue/es/grid/Row'
+  import ACol from 'ant-design-vue/es/grid/Col'
+  import { getAction,postAction } from '@/api/manage'
+
+  export default {
+    name: 'EnterpriseRoleDataruleModal',
+    components: { ACol, ARow },
+    data(){
+      return {
+        functionId:'',
+        roleId:'',
+        visible:false,
+        tabList: [{
+          key: '1',
+          tab: '数据规则',
+        }, {
+          key: '2',
+          tab: '按钮权限',
+        }],
+        activeTabKey: '1',
+        url:{
+          datarule:"/enterprise/role/datarule",
+        },
+        dataruleList:[],
+        dataruleChecked:[]
+      }
+    },
+    methods:{
+      loadData(){
+        getAction(`${this.url.datarule}/${this.functionId}/${this.roleId}`).then(res=>{
+          console.log(res)
+          if(res.success){
+            this.dataruleList = res.result.datarule
+            let drChecked = res.result.drChecked
+            if(drChecked){
+              this.dataruleChecked = drChecked.split(",")
+            }
+          }
+        })
+      },
+      saveDataruleForRole(){
+        if(!this.dataruleChecked || this.dataruleChecked.length==0){
+          this.$message.warning("请注意,现未勾选任何数据权限!")
+        }
+        let params = {
+          permissionId:this.functionId,
+          roleId:this.roleId,
+          dataRuleIds:this.dataruleChecked.join(",")
+        }
+        console.log("保存数据权限",params)
+        postAction(this.url.datarule,params).then(res=>{
+          if(res.success){
+            this.$message.success(res.message)
+          }else{
+            this.$message.error(res.message)
+          }
+        })
+      },
+      show(functionId,roleId){
+        this.onReset()
+        this.functionId = functionId
+        this.roleId = roleId
+        this.visible=true
+        this.loadData()
+      },
+      onClose(){
+        this.visible=false
+        this.onReset()
+      },
+      onTabChange (key) {
+        this.activeTabKey = key
+      },
+      onReset(){
+        this.functionId=''
+        this.roleId=''
+        this.dataruleList=[]
+        this.dataruleChecked=[]
+      }
+    }
+  }
+</script>
+
+<style scoped>
+
+</style>

+ 292 - 0
src/views/adweb/enterprise/modules/EnterpriseSelectUserModal.vue

@@ -0,0 +1,292 @@
+<template>
+  <div>
+    <a-modal
+      centered
+      :title="title"
+      :width="1000"
+      :visible="visible"
+      @ok="handleOk"
+      @cancel="handleCancel"
+      cancelText="关闭">
+
+
+      <!-- 查询区域 -->
+      <div class="table-page-search-wrapper">
+        <a-form layout="inline"  @keyup.enter.native="searchQuery">
+          <a-row :gutter="24">
+
+            <a-col :span="10">
+              <a-form-item label="用户账号">
+                <a-input placeholder="请输入用户账号" v-model="queryParam.username"></a-input>
+              </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>
+      <!-- table区域-begin -->
+      <div>
+        <a-table
+          size="small"
+          bordered
+          rowKey="id"
+          :columns="columns1"
+          :dataSource="dataSource1"
+          :pagination="ipagination"
+          :loading="loading"
+          :scroll="{ y: 240 }"
+          :rowSelection="{selectedRowKeys: selectedRowKeys,onSelectAll:onSelectAll,onSelect:onSelect,onChange: onSelectChange}"
+          @change="handleTableChange">
+
+        </a-table>
+      </div>
+      <!-- table区域-end -->
+
+
+    </a-modal>
+  </div>
+</template>
+
+<script>
+  import {filterObj} from '@/utils/util'
+  import {getAction} from '@/api/manage'
+  import store from '@/store/'
+
+  export default {
+    name: "EnterpriseSelectUserModal",
+    data() {
+      return {
+        title: "添加已有用户",
+        names: [],
+        visible: false,
+        placement: 'right',
+        description: '',
+        // 查询条件
+        queryParam: {
+          id: store.getters.userInfo.id
+        },
+        // 表头
+        columns1: [
+          {
+            title: '#',
+            dataIndex: '',
+            key: 'rowIndex',
+            width: 50,
+            align: "center",
+            customRender: function (t, r, index) {
+              return parseInt(index) + 1;
+            }
+          },
+          {
+            title: '用户账号',
+            align: "center",
+            width: 100,
+            dataIndex: 'username'
+          },
+          {
+            title: '用户名称',
+            align: "center",
+            width: 100,
+            dataIndex: 'realname'
+          },
+          {
+            title: '性别',
+            align: "center",
+            width: 100,
+            dataIndex: 'sex_dictText'
+          },
+          {
+            title: '电话',
+            align: "center",
+            width: 100,
+            dataIndex: 'phone'
+          },
+          {
+            title: '部门',
+            align: "center",
+            width: 150,
+            dataIndex: 'orgCode'
+          }
+        ],
+        columns2: [
+          {
+            title: '用户账号',
+            align: "center",
+            dataIndex: 'username',
+
+          },
+          {
+            title: '用户名称',
+            align: "center",
+            dataIndex: 'realname',
+          },
+          {
+            title: '操作',
+            dataIndex: 'action',
+            align: "center",
+            width: 100,
+            scopedSlots: {customRender: 'action'},
+          }
+        ],
+        //数据集
+        dataSource1: [],
+        dataSource2: [],
+        // 分页参数
+        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',
+        },
+        loading: false,
+        selectedRowKeys: [],
+        selectedRows: [],
+        url: {
+          list: "/enterprise/user/list",
+        }
+      }
+    },
+    created() {
+      this.loadData();
+    },
+    methods: {
+      searchQuery() {
+        this.loadData(1);
+      },
+      searchReset() {
+        this.queryParam = {
+          id: store.getters.userInfo.id
+        };
+        this.loadData(1);
+      },
+      handleCancel() {
+        this.visible = false;
+      },
+      handleOk() {
+        this.dataSource2 = this.selectedRowKeys;
+        console.log("data:" + this.dataSource2);
+        this.$emit("selectFinished", this.dataSource2);
+        this.visible = false;
+      },
+      add() {
+        this.visible = true;
+      },
+      loadData(arg) {
+        //加载数据 若传入参数1则加载第一页的内容
+        if (arg === 1) {
+          this.ipagination.current = 1;
+        }
+        var params = this.getQueryParams();//查询条件
+        getAction(this.url.list, params).then((res) => {
+          if (res.success) {
+            this.dataSource1 = res.result.records;
+            this.ipagination.total = res.result.total;
+          }
+        })
+      },
+      getQueryParams() {
+        var 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() {
+        //TODO 字段权限控制
+      },
+      onSelectAll(selected, selectedRows, changeRows) {
+        if (selected === true) {
+          for (var a = 0; a < changeRows.length; a++) {
+            this.dataSource2.push(changeRows[a]);
+          }
+        } else {
+          for (var b = 0; b < changeRows.length; b++) {
+            this.dataSource2.splice(this.dataSource2.indexOf(changeRows[b]), 1);
+          }
+        }
+        // console.log(selected, selectedRows, changeRows);
+      },
+      onSelect(record, selected) {
+        if (selected === true) {
+          this.dataSource2.push(record);
+        } else {
+          var index = this.dataSource2.indexOf(record);
+          //console.log();
+          if (index >= 0) {
+            this.dataSource2.splice(this.dataSource2.indexOf(record), 1);
+          }
+
+        }
+      },
+      onSelectChange(selectedRowKeys, selectedRows) {
+        this.selectedRowKeys = selectedRowKeys;
+        this.selectionRows = selectedRows;
+      },
+      onClearSelected() {
+        this.selectedRowKeys = [];
+        this.selectionRows = [];
+      },
+      handleDelete: function (record) {
+        this.dataSource2.splice(this.dataSource2.indexOf(record), 1);
+      },
+      handleTableChange(pagination, filters, sorter) {
+        //分页、排序、筛选变化时触发
+        console.log(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();
+      }
+    }
+  }
+</script>
+<style lang="less" scoped>
+  .ant-card-body .table-operator {
+    margin-bottom: 18px;
+  }
+
+  .ant-table-tbody .ant-table-row td {
+    padding-top: 15px;
+    padding-bottom: 15px;
+  }
+
+  .anty-row-operator button {
+    margin: 0 5px
+  }
+
+  .ant-btn-danger {
+    background-color: #ffffff
+  }
+
+  .ant-modal-cust-warp {
+    height: 100%
+  }
+
+  .ant-modal-cust-warp .ant-modal-body {
+    height: calc(100% - 110px) !important;
+    overflow-y: auto
+  }
+
+  .ant-modal-cust-warp .ant-modal-content {
+    height: 90% !important;
+    overflow-y: hidden
+  }
+</style>

+ 615 - 0
src/views/adweb/enterprise/modules/EnterpriseUserModal.vue

@@ -0,0 +1,615 @@
+<template>
+  <a-drawer
+    :title="title"
+    :maskClosable="true"
+    :width="drawerWidth"
+    placement="right"
+    :closable="true"
+    @close="handleCancel"
+    :visible="visible"
+    style="height: 100%; overflow: auto; padding-bottom: 53px"
+  >
+    <template #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" />
+        </span>
+      </div>
+    </template>
+
+    <a-spin :spinning="confirmLoading">
+      <a-form :form="form">
+        <a-form-item label="用户账号" :labelCol="labelCol" :wrapperCol="wrapperCol">
+          <a-input placeholder="请输入用户账号" v-decorator.trim="['username', validatorRules.username]" :disabled="!!model.id" />
+        </a-form-item>
+        <a-form-item label="用户姓名" :labelCol="labelCol" :wrapperCol="wrapperCol">
+          <a-input placeholder="请输入用户姓名" v-decorator.trim="['realname', validatorRules.realname]" />
+        </a-form-item>
+        <template v-if="!model.id">
+          <a-form-item label="登录密码" :labelCol="labelCol" :wrapperCol="wrapperCol">
+            <a-input type="password" autocomplete="new-password" placeholder="请输入登录密码" v-decorator="['password', validatorRules.password]" />
+          </a-form-item>
+          <a-form-item label="确认密码" :labelCol="labelCol" :wrapperCol="wrapperCol">
+            <a-input
+              type="password"
+              @blur="handleConfirmBlur"
+              placeholder="请重新输入登录密码"
+              v-decorator="['confirmpassword', validatorRules.confirmpassword]"
+            />
+          </a-form-item>
+        </template>
+        <template v-if="siteCode === dictSiteCode">
+          <a-form-item label="性别" :labelCol="labelCol" :wrapperCol="wrapperCol">
+            <a-radio-group v-decorator="['sex', { rules: [{ required: true, message: '请选择性别!' }] }]">
+              <a-radio :value="1">男</a-radio>
+              <a-radio :value="2">女</a-radio>
+            </a-radio-group>
+          </a-form-item>
+          <a-form-item label="工作手机" :labelCol="labelCol" :wrapperCol="wrapperCol">
+            <a-input
+              placeholder="请输入手机号码"
+              v-decorator="['phone', { rules: [{ required: true, message: '请输入工作手机!' }, { validator: validatePhone }] }]"
+            />
+          </a-form-item>
+          <a-form-item label="企业邮箱" :labelCol="labelCol" :wrapperCol="wrapperCol">
+            <a-input
+              placeholder="请输入邮箱"
+              v-decorator="['email', { rules: [{ required: true, message: '请输入企业邮箱!' }, { validator: validateEmail }] }]"
+            />
+          </a-form-item>
+        </template>
+        <template v-if="siteCode !== dictSiteCode">
+          <a-form-item label="手机号码" :labelCol="labelCol" :wrapperCol="wrapperCol">
+            <a-input placeholder="请输入手机号码" v-decorator="['phone', validatorRules.phone]" />
+          </a-form-item>
+          <a-form-item label="邮箱" :labelCol="labelCol" :wrapperCol="wrapperCol">
+            <a-input
+              placeholder="请输入邮箱"
+              v-decorator="['email', { rules: [{ type: 'email', message: '请输入正确的邮箱地址', required: isMasterAccount }] }]"
+            />
+          </a-form-item>
+        </template>
+        <a-form-item label="头像" :labelCol="labelCol" :wrapperCol="wrapperCol">
+          <j-image-upload class="avatar-uploader" text="上传" v-model="fileList" />
+        </a-form-item>
+      </a-form>
+    </a-spin>
+    <enterprise-depart-window ref="enterpriseDepartWindow" @ok="modalFormOk" />
+    <div class="drawer-bootom-button" v-show="!disableSubmit">
+      <a-button @click="handleCancel" style="margin-right: 0.8rem">取消</a-button>
+      <a-button @click="handleSubmit" type="primary" :loading="confirmLoading">提交</a-button>
+    </div>
+  </a-drawer>
+</template>
+
+<script lang="ts" setup name="EnterpriseUserModal">
+  // 引入搜索部门弹出框的组件
+  import enterpriseDepartWindow from './EnterpriseDepartWindow.vue';
+
+  import { getAction, postAction } from '@/api/manage/manage';
+  import { addEnterpriseUser, queryEnterpriseAll } from '../EnterpriseUserManage.api';
+  import { duplicateCheck } from '@/views/system/user/user.api';
+  import JImageUpload from '/@/components/Form/src/jeecg/components/JImageUpload.vue';
+  import {onBeforeMount, reactive, ref} from 'vue';
+
+  const isMasterAccount = ref(false);
+  const departDisabled = ref(false);
+  const roleDisabled = ref(false);
+  const modaltoggleFlag = ref(true);
+  const confirmDirty = ref(false);
+  const modalWidth = ref(800);
+  const drawerWidth = ref(700);
+  const queryParam = ref({});
+  const selectedDepartKeys = reactive([]);
+  const checkedDepartKeys = reactive([]);
+  const checkedDepartNames = reactive([]);
+  const checkedDepartNameString = ref('');
+  const resultDepartOptions = reactive([]);
+  const userId = ref('');
+  const disableSubmit = ref(false);
+  const userDepartModel = reactive({ userId: '', departIdList: [] }); // 保存SysUserDepart的用户部门中间表数据需要的对象
+  const dateFormat = ref('YYYY-MM-DD');
+  const validatorRules = reactive({
+    username: {
+      rules: [
+        {
+          required: true,
+          message: '请输入用户账号!',
+        },
+        {
+          validator: this.validateUsername,
+        },
+      ],
+    },
+    password: {
+      rules: [
+        {
+          required: true,
+          pattern: /^(?=.*[a-zA-Z])(?=.*\d)(?=.*[~!@#$%^&*()_+`\-={}:";'<>?,./]).{8,}$/,
+          message: '密码由8位数字、大小写字母和特殊符号组成!',
+        },
+        {
+          validator: this.validateToNextPassword,
+        },
+      ],
+    },
+    confirmpassword: {
+      rules: [
+        {
+          required: true,
+          message: '请重新输入登录密码!',
+        },
+        {
+          validator: this.compareToFirstPassword,
+        },
+      ],
+    },
+    realname: { rules: [{ required: true, message: '请输入用户名称!' }] },
+    phone: { rules: [{ validator: this.validatePhone }] },
+    email: {
+      rules: [
+        {
+          validator: this.validateEmail,
+        },
+      ],
+    },
+    roles: {},
+    //  sex:{initialValue:((!this.model.sex)?"": (this.model.sex+""))}
+    // workNo: {
+    //   rules: [
+    //     { required: true, message: '请输入工号' },
+    //     { validator: this.validateWorkNo }
+    //   ]
+    // },
+    telephone: {
+      rules: [{ pattern: /^0\d{2,3}-[1-9]\d{6,7}$/, message: '请输入正确的座机号码' }],
+    },
+  });
+
+  const departIdShow = ref(false);
+  const departIds = reactive([]);
+  const title = '操作';
+  const visible = ref(false);
+  const model = reactive({});
+  const roleList = reactive([]);
+  const selectedRole = ref([]);
+  const labelCol = reactive({
+    xs: { span: 24 },
+    sm: { span: 5 },
+  });
+
+  const wrapperCol = reactive({
+    xs: { span: 24 },
+    sm: { span: 16 },
+  });
+
+  const uploadLoading = false;
+  const confirmLoading = false;
+
+  const headers = reactive({});
+  const picUrl = ref('');
+  const identity = ref('1');
+  const fileList = reactive([]);
+  const tenantList = reactive([]);
+  const currentTenant = reactive([]);
+  const siteCode = ref('');
+  const dictSiteCode = ref('');
+  const url = {
+    fileUpload: window._CONFIG['domianURL'] + '/sys/common/upload',
+    userId: '/sys/user/generateUserId', // 引入生成添加用户情况下的url
+    syncUserByUserName: '/act/process/extActProcess/doSyncUserByUserName', //同步用户到工作流
+    queryTenantList: '/sys/tenant/queryList',
+  };
+
+  onBeforeMount(() => {
+    initTenantList();
+    siteCode.value = <string>localStorage.getItem('siteCode');
+    getTurnInquiryCode();
+  });
+
+  function isDisabledAuth(code) {
+    return disabledAuthFilter(code);
+  }
+  function  initTenantList() {
+    getAction(this.url.queryTenantList).then((res) => {
+      if (res.success) {
+        this.tenantList = res.result;
+      }
+    });
+  }
+  //窗口最大化切换
+  function toggleScreen() {
+    if (this.modaltoggleFlag) {
+      this.modalWidth = window.innerWidth;
+    } else {
+      this.modalWidth = 800;
+    }
+    this.modaltoggleFlag = !this.modaltoggleFlag;
+  }
+  function initialRoleList() {
+    queryEnterpriseAll({ username: this.queryParam.username }).then((res) => {
+      if (res.success) {
+        this.roleList = res.result;
+      } else {
+        console.log(res.message);
+      }
+    });
+  }
+  function loadUserRoles(userid) {
+    queryUserRole({ userid: userid }).then((res) => {
+      if (res.success) {
+        this.selectedRole = res.result;
+      } else {
+        console.log(res.message);
+      }
+    });
+  }
+  function refresh() {
+    this.selectedDepartKeys = [];
+    this.checkedDepartKeys = [];
+    this.checkedDepartNames = [];
+    this.checkedDepartNameString = '';
+    this.userId = '';
+    this.resultDepartOptions = [];
+    this.departId = [];
+    this.departIdShow = false;
+    this.currentTenant = [];
+  },
+  function add() {
+    this.picUrl = '';
+    this.refresh();
+    this.edit({ activitiSync: '1' });
+  },
+  function edit(record) {
+    this.resetScreenSize(); // 调用此方法,根据屏幕宽度自适应调整抽屉的宽度
+    let that = this;
+    that.initialRoleList();
+    that.checkedDepartNameString = '';
+    that.form.resetFields();
+    if (record.hasOwnProperty('id')) {
+      that.loadUserRoles(record.id);
+      setTimeout(() => {
+        this.fileList = record.avatar;
+      }, 5);
+    }
+    that.userId = record.id;
+    that.model = Object.assign({}, record);
+    that.$nextTick(() => {
+      that.form.setFieldsValue(pick(this.model, 'username', 'sex', 'realname', 'email', 'phone', 'activitiSync', 'telephone', 'post'));
+    });
+    that.visible = true;
+    //身份为上级显示负责部门,否则不显示
+    if (this.model.userIdentity == '2') {
+      this.identity = '2';
+      this.departIdShow = true;
+    } else {
+      this.identity = '1';
+      this.departIdShow = false;
+    }
+    // 调用查询用户对应的部门信息的方法
+    that.checkedDepartKeys = [];
+    //update-begin-author:taoyan date:2020710 for:多租户配置
+    if (!record.relTenantIds || record.relTenantIds.length == 0) {
+      this.currentTenant = [];
+    } else {
+      this.currentTenant = record.relTenantIds.split(',').map(Number);
+    }
+    //update-end-author:taoyan date:2020710 for:多租户配置
+  }
+
+  close() {
+    this.$emit('close');
+    this.visible = false;
+    this.disableSubmit = false;
+    this.selectedRole = [];
+    this.userDepartModel = { userId: '', departIdList: [] };
+    this.checkedDepartNames = [];
+    this.checkedDepartNameString = '';
+    this.checkedDepartKeys = [];
+    this.selectedDepartKeys = [];
+    this.resultDepartOptions = [];
+    this.departIds = [];
+    this.departIdShow = false;
+    this.identity = '1';
+    this.fileList = [];
+  }
+
+  function handleSubmit() {
+    const that = this;
+    // 触发表单验证
+    this.form.validateFields((err, values) => {
+      if (!err) {
+        that.confirmLoading = true;
+        if (!values.birthday) {
+          values.birthday = '';
+        } else {
+          values.birthday = values.birthday.format(this.dateFormat);
+        }
+        let formData = Object.assign(this.model, values);
+        if (that.fileList != '') {
+          formData.avatar = that.fileList;
+        } else {
+          formData.avatar = null;
+        }
+        if (formData.email == '') {
+          formData.email = null;
+        }
+        //update-begin-author:taoyan date:2020710 for:多租户配置
+        formData.relTenantIds = this.currentTenant.length > 0 ? this.currentTenant.join(',') : '';
+        //update-end-author:taoyan date:2020710 for:多租户配置
+        formData.selectedroles = this.selectedRole.length > 0 ? this.selectedRole.join(',') : '';
+        formData.selecteddeparts = this.userDepartModel.departIdList.length > 0 ? this.userDepartModel.departIdList.join(',') : '';
+        formData.userIdentity = this.identity;
+        //如果是上级择传入departIds,否则为空
+        if (this.identity === '2') {
+          formData.departIds = this.departIds.join(',');
+        } else {
+          formData.departIds = '';
+        }
+        // that.addDepartsToUser(that,formData); // 调用根据当前用户添加部门信息的方法
+        let obj;
+        formData.masterId = store.getters.userInfo.id;
+        formData.masterName = store.getters.userInfo.username;
+        if (!this.model.id) {
+          formData.id = this.userId;
+          formData.siteCode = this.siteCode;
+          obj = addEnterpriseUser(formData);
+        } else {
+          formData.masterEmail = store.getters.userInfo.email;
+          formData.siteCode = this.siteCode;
+          obj = postAction('/enterprise/user/edit', formData);
+        }
+        obj
+          .then((res) => {
+            if (res.success) {
+              that.$message.success(res.message);
+              that.$emit('ok');
+              that.confirmLoading = false;
+              that.checkedDepartNames = [];
+              that.userDepartModel.departIdList = { userId: '', departIdList: [] };
+              that.close();
+            } else {
+              that.confirmLoading = false;
+              that.$message.warning(res.message);
+            }
+          })
+          .finally(() => {});
+      }
+    });
+  }
+  function handleCancel() {
+    this.close();
+  }
+  function validateToNextPassword(rule, value, callback) {
+    const form = this.form;
+    const confirmpassword = form.getFieldValue('confirmpassword');
+
+    if (value && confirmpassword && value !== confirmpassword) {
+      callback('两次输入的密码不一样!');
+    }
+    if (value && this.confirmDirty) {
+      form.validateFields(['confirm'], { force: true });
+    }
+    callback();
+  }
+  function compareToFirstPassword(rule, value, callback) {
+    const form = this.form;
+    if (value && value !== form.getFieldValue('password')) {
+      callback('两次输入的密码不一样!');
+    } else {
+      callback();
+    }
+  }
+  function validatePhone(rule, value, callback) {
+    if (!value) {
+      callback();
+    } else {
+      //update-begin--Author:kangxiaolin  Date:20190826 for:[05] 手机号不支持199号码段--------------------
+      if (new RegExp(/^1[3|4|5|7|8|9][0-9]\d{8}$/).test(value)) {
+        //update-end--Author:kangxiaolin  Date:20190826 for:[05] 手机号不支持199号码段--------------------
+
+        var params = {
+          tableName: 'sys_user',
+          fieldName: 'phone',
+          fieldVal: value,
+          dataId: this.userId,
+        };
+        duplicateCheck(params).then((res) => {
+          if (res.success) {
+            callback();
+          } else {
+            callback('手机号已存在!');
+          }
+        });
+      } else {
+        callback('请输入正确格式的手机号码!');
+      }
+    }
+  }
+  function validateEmail(rule, value, callback) {
+    if (!value) {
+      callback();
+    } else {
+      if (
+        new RegExp(
+          /^(([^<>()\[\]\\.,;:\s@"]+(\.[^<>()\[\]\\.,;:\s@"]+)*)|(".+"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/
+        ).test(value)
+      ) {
+        var params = {
+          tableName: 'sys_user',
+          fieldName: 'email',
+          fieldVal: value,
+          dataId: this.userId,
+        };
+        duplicateCheck(params).then((res) => {
+          console.log(res);
+          if (res.success) {
+            callback();
+          } else {
+            callback('邮箱已存在!');
+          }
+        });
+      } else {
+        callback('请输入正确格式的邮箱!');
+      }
+    }
+  }
+  function validateUsername(rule, value, callback) {
+    var params = {
+      tableName: 'sys_user',
+      fieldName: 'username',
+      fieldVal: value,
+      dataId: this.userId,
+    };
+    duplicateCheck(params).then((res) => {
+      if (res.success) {
+        callback();
+      } else {
+        callback('用户名已存在!');
+      }
+    });
+  }
+  // validateWorkNo(rule, value, callback){
+  //   var params = {
+  //     tableName: 'sys_user',
+  //     fieldName: 'work_no',
+  //     fieldVal: value,
+  //     dataId: this.userId
+  //   };
+  //   duplicateCheck(params).then((res) => {
+  //     if (res.success) {
+  //       callback()
+  //     } else {
+  //       callback("工号已存在!")
+  //     }
+  //   })
+  // },
+  function handleConfirmBlur(e) {
+    const value = e.target.value;
+    this.confirmDirty = this.confirmDirty || !!value;
+  }
+
+  function normFile(e) {
+    console.log('Upload event:', e);
+    if (Array.isArray(e)) {
+      return e;
+    }
+    return e && e.fileList;
+  }
+  beforeUpload: function (file) {
+    var fileType = file.type;
+    if (fileType.indexOf('image') < 0) {
+      this.$message.warning('请上传图片');
+      return false;
+    }
+    //TODO 验证文件大小
+  }
+  function handleChange(info) {
+    this.picUrl = '';
+    if (info.file.status === 'uploading') {
+      this.uploadLoading = true;
+      return;
+    }
+    if (info.file.status === 'done') {
+      var response = info.file.response;
+      this.uploadLoading = false;
+      console.log(response);
+      if (response.success) {
+        this.model.avatar = response.message;
+        this.picUrl = 'Has no pic url yet';
+      } else {
+        this.$message.warning(response.message);
+      }
+    }
+  }
+  // 搜索用户对应的部门API
+  function onSearch() {
+    this.$refs.enterpriseDepartWindow.add(this.checkedDepartKeys, this.userId);
+  }
+
+  // 获取用户对应部门弹出框提交给返回的数据
+  function modalFormOk(formData) {
+    this.checkedDepartNames = [];
+    this.selectedDepartKeys = [];
+    this.checkedDepartNameString = '';
+    this.userId = formData.userId;
+    this.userDepartModel.userId = formData.userId;
+    this.departIds = [];
+    this.resultDepartOptions = [];
+    var depart = [];
+    for (let i = 0; i < formData.departIdList.length; i++) {
+      this.selectedDepartKeys.push(formData.departIdList[i].key);
+      this.checkedDepartNames.push(formData.departIdList[i].title);
+      this.checkedDepartNameString = this.checkedDepartNames.join(',');
+      //新增部门选择,如果上面部门选择后不为空直接付给负责部门
+      depart.push({
+        key: formData.departIdList[i].key,
+        title: formData.departIdList[i].title,
+      });
+      this.departIds.push(formData.departIdList[i].key);
+    }
+    this.resultDepartOptions = depart;
+    this.userDepartModel.departIdList = this.selectedDepartKeys;
+    this.checkedDepartKeys = this.selectedDepartKeys; //更新当前的选择keys
+  }
+  // 根据屏幕变化,设置抽屉尺寸
+  function resetScreenSize() {
+    let screenWidth = document.body.clientWidth;
+    if (screenWidth < 500) {
+      this.drawerWidth = screenWidth;
+    } else {
+      this.drawerWidth = 700;
+    }
+  }
+  function identityChange(e) {
+    if (e.target.value === '1') {
+      this.departIdShow = false;
+    } else {
+      this.departIdShow = true;
+    }
+  }
+  //获取需要轮流询盘的站点code
+  function getTurnInquiryCode() {
+    getAction('/adweb/adwebEnquiry/getTurnInquiryCode').then((res) => {
+      if (res.code === 200) {
+        this.dictSiteCode = res.result[0].value;
+      }
+    });
+  }
+
+</script>
+
+<style scoped>
+  .avatar-uploader > .ant-upload {
+    width: 104px;
+    height: 104px;
+  }
+
+  .ant-upload-select-picture-card i {
+    font-size: 49px;
+    color: #999;
+  }
+
+  .ant-upload-select-picture-card .ant-upload-text {
+    margin-top: 8px;
+    color: #666;
+  }
+
+  .ant-table-tbody .ant-table-row td {
+    padding-top: 10px;
+    padding-bottom: 10px;
+  }
+
+  .drawer-bootom-button {
+    position: absolute;
+    bottom: -8px;
+    width: 100%;
+    border-top: 1px solid #e8e8e8;
+    padding: 10px 16px;
+    text-align: right;
+    left: 0;
+    background: #fff;
+    border-radius: 0 0 2px 2px;
+  }
+</style>

+ 162 - 0
src/views/adweb/enterprise/modules/EnterpriseUserRecycleBinModal.vue

@@ -0,0 +1,162 @@
+<template>
+  <a-drawer :title="title" :mask-closable="true" :width="1000" placement="right" :closable="true" @close="handleCancel" :visible="visible">
+    <a-table
+      ref="table"
+      size="middle"
+      rowKey="id"
+      :scroll="{ x: true }"
+      :columns="columns"
+      :dataSource="dataSource"
+      :pagination="ipagination"
+      :loading="loading"
+    >
+      <template #avatarslot="text, record, index">
+        <div class="anty-img-wrap" style="margin-bottom: 5px">
+          <a-avatar shape="square" :src="getAvatarView(record.avatar)" icon="user" />
+        </div>
+      </template>
+      <template #action="text, record">
+        <span>
+          <div style="padding: 8px">
+            <a-tag color="purple" @click="handleFrozen(record.id, 1)" style="cursor: pointer">恢复</a-tag>
+            <a-popconfirm title="确定删除吗(同时删除wp中的账户,账户下的产品会转移至主账户)?" @confirm="() => handleDelete(record.id)">
+              <a-tag color="red" style="cursor: pointer">删除</a-tag>
+            </a-popconfirm>
+          </div>
+        </span>
+      </template>
+      <template #countSlot="text, record, index">
+        <span style="margin-left: 20px">{{ text }} 个</span>
+      </template>
+    </a-table>
+  </a-drawer>
+</template>
+
+<script>
+  import { putAction, getFileAccessHttpUrl, deleteAction } from '@/api/manage/manage';
+
+  export default {
+    name: 'EnterpriseUserRecycleBinModal',
+
+    data() {
+      return {
+        title: '用户回收站',
+        loading: false,
+        visible: false,
+        selectedRowKeys: [],
+        columns: [
+          {
+            title: '序号',
+            dataIndex: '',
+            key: 'rowIndex',
+            width: 60,
+            align: 'center',
+            customRender: function (t, r, index) {
+              return parseInt(index) + 1;
+            },
+          },
+          {
+            title: '用户名',
+            align: 'left',
+            dataIndex: 'realname',
+          },
+          {
+            title: '登录账号',
+            align: 'left',
+            dataIndex: 'username',
+          },
+          {
+            title: '头像',
+            align: 'left',
+            dataIndex: 'avatar',
+            width: 80,
+            scopedSlots: { customRender: 'avatarslot' },
+          },
+          {
+            title: '类型',
+            align: 'left',
+            width: 80,
+            dataIndex: 'primaryAccount',
+            customRender: function (text) {
+              return text ? '主账户' : '子账户';
+            },
+          },
+          // {
+          //   title: '分配的询盘数',
+          //   align: "left",
+          //   dataIndex: 'enquiryNum',
+          //   scopedSlots: {customRender: "countSlot"}
+          // },
+          // {
+          //   title: '负责的产品数',
+          //   align: "left",
+          //   dataIndex: 'productNum',
+          //   scopedSlots: {customRender: "countSlot"}
+          // },
+          {
+            title: '手机号码',
+            align: 'left',
+            dataIndex: 'phone',
+            customRender: function (text) {
+              return text === null ? '--' : text;
+            },
+          },
+          {
+            title: '邮箱',
+            align: 'left',
+            dataIndex: 'email',
+            customRender: function (text) {
+              return text === null ? '--' : text;
+            },
+          },
+          {
+            title: '更新时间',
+            align: 'left',
+            dataIndex: 'updateTime',
+            customRender: function (text) {
+              return text === null ? '--' : text;
+            },
+          },
+          {
+            title: '操作',
+            dataIndex: 'action',
+            fixed: 'right',
+            scopedSlots: { customRender: 'action' },
+            align: 'center',
+          },
+        ],
+        url: {
+          getAvatar: (path) => getFileAccessHttpUrl(`${path}`),
+          list: '/enterprise/user/recycleBin',
+          delete: '/sys/user/delete',
+        },
+      };
+    },
+    methods: {
+      show() {
+        this.visible = true;
+        this.loadData(1);
+      },
+      handleCancel() {
+        this.visible = false;
+      },
+      handleFrozen: function (id, status) {
+        let that = this;
+        putAction('/enterprise/user/frozen', { ids: id, status: status }).then((res) => {
+          if (res.success) {
+            that.$message.success(res.message);
+            that.loadData();
+            that.$emit('ok');
+          } else {
+            that.$message.warning(res.message);
+          }
+        });
+      },
+      getAvatarView: function (avatar) {
+        return getFileAccessHttpUrl(avatar);
+      },
+    },
+  };
+</script>
+
+<style lang="less" scoped></style>

+ 183 - 0
src/views/adweb/enterprise/modules/EnterpriseUserRoleModal.vue

@@ -0,0 +1,183 @@
+<template>
+  <a-drawer
+    :title="title"
+    :maskClosable="true"
+    width="650"
+    placement="right"
+    :closable="true"
+    @close="close"
+    :visible="visible"
+    style="overflow: auto; padding-bottom: 53px"
+  >
+    <a-form>
+      <a-form-item>
+        <a-tree
+          checkable
+          @check="onCheck"
+          :checkedKeys="checkedKeys"
+          :treeData="treeData"
+          @expand="onExpand"
+          @select="onTreeNodeSelect"
+          :selectedKeys="selectedKeys"
+          :expandedKeys="expandedKeysss"
+          :checkStrictly="checkStrictly"
+        >
+          <template #hasDatarule="{ slotTitle, ruleFlag }">
+            <span> {{ slotTitle }}<a-icon v-if="ruleFlag" type="align-left" style="margin-left: 5px; color: red" /> </span>
+          </template>
+        </a-tree>
+      </a-form-item>
+    </a-form>
+
+    <div class="drawer-bootom-button">
+      <a-dropdown style="float: left" :trigger="['click']" placement="topCenter">
+        <template #overlay>
+          <a-menu>
+            <!--          <a-menu-item key="1" @click="switchCheckStrictly(1)">父子关联</a-menu-item>-->
+            <!--          <a-menu-item key="2" @click="switchCheckStrictly(2)">取消关联</a-menu-item>-->
+            <a-menu-item key="3" @click="checkALL">全部勾选</a-menu-item>
+            <a-menu-item key="4" @click="cancelCheckALL">取消全选</a-menu-item>
+            <a-menu-item key="5" @click="expandAll">展开所有</a-menu-item>
+            <a-menu-item key="6" @click="closeAll">合并所有</a-menu-item>
+          </a-menu>
+        </template>
+        <a-button> 快捷操作 <a-icon type="up" /> </a-button>
+      </a-dropdown>
+      <a-button style="margin-right: 0.8rem" @click="close">取消</a-button>
+      <!--      <a-button @click="handleSubmit(false)" type="primary" :loading="loading" ghost style="margin-right: 0.8rem">仅保存</a-button>-->
+      <a-button @click="handleSubmit" type="primary" :loading="loading">保存</a-button>
+    </div>
+  </a-drawer>
+</template>
+
+<script>
+  import { getAction, postAction } from '@/api/manage/manage';
+
+  export default {
+    name: 'EnterpriseUserRoleModal',
+    components: {},
+    data() {
+      return {
+        userId: '',
+        treeData: [],
+        defaultCheckedKeys: [],
+        checkedKeys: [],
+        expandedKeysss: [],
+        allTreeKeys: [],
+        autoExpandParent: true,
+        checkStrictly: false,
+        title: '菜单权限配置',
+        visible: false,
+        loading: false,
+        selectedKeys: [],
+      };
+    },
+    watch: {
+      visible() {
+        if (this.visible) {
+          this.loadData();
+        }
+      },
+    },
+    methods: {
+      onTreeNodeSelect(id) {
+        if (id && id.length > 0) {
+          this.selectedKeys = id;
+        }
+      },
+      onCheck(o) {
+        if (this.checkStrictly) {
+          this.checkedKeys = o.checked;
+        } else {
+          this.checkedKeys = o;
+        }
+      },
+      show(userId) {
+        this.userId = userId;
+        this.visible = true;
+      },
+      close() {
+        this.reset();
+        this.$emit('close');
+        this.visible = false;
+      },
+      onExpand(expandedKeys) {
+        this.expandedKeysss = expandedKeys;
+        this.autoExpandParent = false;
+      },
+      reset() {
+        this.expandedKeysss = [];
+        this.checkedKeys = [];
+        this.defaultCheckedKeys = [];
+        this.loading = false;
+      },
+      expandAll() {
+        this.expandedKeysss = this.allTreeKeys;
+      },
+      closeAll() {
+        this.expandedKeysss = [];
+      },
+      checkALL() {
+        this.checkedKeys = this.allTreeKeys;
+      },
+      cancelCheckALL() {
+        //this.checkedKeys = this.defaultCheckedKeys
+        this.checkedKeys = [];
+      },
+      // switchCheckStrictly(v) {
+      //   if (v == 1) {
+      //     this.checkStrictly = false
+      //   } else if (v == 2) {
+      //     this.checkStrictly = true
+      //   }
+      // },
+      handleCancel() {
+        this.close();
+      },
+      handleSubmit() {
+        let that = this;
+        let params = {
+          userId: that.userId,
+          permissionIds: that.checkedKeys.join(','),
+        };
+        that.loading = true;
+        console.log('请求参数:', params);
+        postAction('/enterprise/user/saveRolePermission', params).then((res) => {
+          if (res.success) {
+            that.$message.success(res.message);
+          } else {
+            that.$message.error(res.message);
+          }
+          that.loading = false;
+          that.close();
+          this.loadData();
+        });
+      },
+      loadData() {
+        getAction('/enterprise/user/queryTreeList').then((res) => {
+          this.treeData = res.result.treeList;
+          this.allTreeKeys = res.result.ids;
+          getAction('/enterprise/user/queryRolePermission', { userId: this.userId }).then((res) => {
+            this.checkedKeys = [...res.result];
+            this.defaultCheckedKeys = [...res.result];
+            this.expandedKeysss = this.allTreeKeys;
+            console.log(this.defaultCheckedKeys);
+          });
+        });
+      },
+    },
+  };
+</script>
+<style lang="less" scoped>
+  .drawer-bootom-button {
+    position: absolute;
+    bottom: 0;
+    width: 100%;
+    border-top: 1px solid #e8e8e8;
+    padding: 10px 16px;
+    text-align: right;
+    left: 0;
+    background: #fff;
+    border-radius: 0 0 2px 2px;
+  }
+</style>

+ 59 - 68
src/views/adweb/keywords/SeoKeywords.data.ts

@@ -1,28 +1,23 @@
-import { BasicColumn } from "/@/components/Table";
-import { FormSchema } from "/@/components/Table";
-import { rules } from "/@/utils/helper/validator";
-import { render } from "/@/utils/common/renderUtils";
-import { getWeekMonthQuarterYear } from "/@/utils";
-import { defHttp } from "@/utils/http/axios";
+import { BasicColumn, FormSchema } from '/@/components/Table';
 //列表数据
 export const columns: BasicColumn[] = [
   {
-    title: "关键词",
-    align: "center",
-    dataIndex: "keywords",
+    title: '关键词',
+    align: 'center',
+    dataIndex: 'keywords',
     sorter: true,
     width: 300,
   },
   {
-    title: "关键词所在URL",
-    align: "center",
-    dataIndex: "positionUrl",
+    title: '关键词所在URL',
+    align: 'center',
+    dataIndex: 'positionUrl',
     width: 500,
   },
   {
-    title: "关键词类型",
-    align: "center",
-    dataIndex: "keywordType",
+    title: '关键词类型',
+    align: 'center',
+    dataIndex: 'keywordType',
     width: 230,
   },
   {
@@ -54,16 +49,16 @@ export const columns: BasicColumn[] = [
 //查询数据
 export const searchFormSchema: FormSchema[] = [
   {
-    label: "关键词",
+    label: '关键词',
     field: 'keywords',
     component: 'Input',
     //colProps: {span: 6},
   },
   {
-    label: "关键词类型",
+    label: '关键词类型',
     field: 'keywordType',
     component: 'JDictSelectTag',
-    componentProps:{
+    componentProps: {
       dictCode: 'keyword_type',
       placeholder: '请选择关键词类型',
       stringToNumber: true,
@@ -83,15 +78,15 @@ export const searchFormSchema: FormSchema[] = [
   //       { label: '前十页(TOP 100)', value: 10 },
   //     ],
   //   },
-    //colProps: {span: 6},
+  //colProps: {span: 6},
   // },
 ];
 //表单数据
 export const formSchema: FormSchema[] = [
   {
-    label: "关键词类型",
-    field: "keywordType",
-    component: "JDictSelectTag",
+    label: '关键词类型',
+    field: 'keywordType',
+    component: 'JDictSelectTag',
     componentProps: {
       dictCode: 'keyword_type',
       placeholder: '请选择关键词类型',
@@ -99,75 +94,71 @@ export const formSchema: FormSchema[] = [
     },
   },
   {
-    label: "关键词",
-    field: "keywords",
-    component: "Input",
-    dynamicRules: ({ model, schema }) => {
-      return [
-        { required: true, message: "请输入关键词!" }
-      ];
-    }
+    label: '关键词',
+    field: 'keywords',
+    component: 'Input',
+    dynamicRules: ({}) => {
+      return [{ required: true, message: '请输入关键词!' }];
+    },
   },
   {
-    label: "站点ID",
-    field: "siteCode",
-    component: "Input",
+    label: '站点ID',
+    field: 'siteCode',
+    component: 'Input',
     show: false,
     defaultValue: localStorage.getItem('siteCode'),
-    dynamicRules: ({ model, schema }) => {
-      return [
-        { required: true, message: "没有站点ID!" }
-      ];
+    dynamicRules: ({}) => {
+      return [{ required: true, message: '没有站点ID!' }];
     },
   },
   // TODO 主键隐藏字段,目前写死为ID
   {
-    label: "",
-    field: "id",
-    component: "Input",
-    show: false
-  }
+    label: '',
+    field: 'id',
+    component: 'Input',
+    show: false,
+  },
 ];
 
 // 高级查询数据
 export const superQuerySchema = {
-  keywords: { title: "关键词", order: 2, view: "text", type: "string" },
-  lastSearchTime: { title: "最新搜索时间", order: 5, view: "datetime", type: "string" },
+  keywords: { title: '关键词', order: 2, view: 'text', type: 'string' },
+  lastSearchTime: { title: '最新搜索时间', order: 5, view: 'datetime', type: 'string' },
   timerLastSearchTime: {
-    title: "定时器的最新搜索时间",
+    title: '定时器的最新搜索时间',
     order: 6,
-    view: "datetime",
-    type: "string"
-  },
-  status: { title: "0删除,1有效,2过期", order: 7, view: "number", type: "number" },
-  sourceId: { title: "数据复制源ID", order: 8, view: "number", type: "number" },
-  priority: { title: "关键词优先级(1:高;2:中;3:低)", order: 9, view: "number", type: "number" },
-  finishTime: { title: "SEO关键词优化完成时间", order: 10, view: "datetime", type: "string" },
-  optimizeProcess: { title: "优化进度(默认0)", order: 11, view: "number", type: "number" },
+    view: 'datetime',
+    type: 'string',
+  },
+  status: { title: '0删除,1有效,2过期', order: 7, view: 'number', type: 'number' },
+  sourceId: { title: '数据复制源ID', order: 8, view: 'number', type: 'number' },
+  priority: { title: '关键词优先级(1:高;2:中;3:低)', order: 9, view: 'number', type: 'number' },
+  finishTime: { title: 'SEO关键词优化完成时间', order: 10, view: 'datetime', type: 'string' },
+  optimizeProcess: { title: '优化进度(默认0)', order: 11, view: 'number', type: 'number' },
   optimizeStatus: {
-    title: "优化状态(0:未完成;1:已完成)",
+    title: '优化状态(0:未完成;1:已完成)',
     order: 12,
-    view: "number",
-    type: "number"
+    view: 'number',
+    type: 'number',
   },
-  keywordType: { title: "关键词类型", order: 13, view: "number", type: "number" },
-  lastRank: { title: "最近一次搜索排名", order: 14, view: "number", type: "number" },
+  keywordType: { title: '关键词类型', order: 13, view: 'number', type: 'number' },
+  lastRank: { title: '最近一次搜索排名', order: 14, view: 'number', type: 'number' },
   searchStatus: {
-    title: "搜索状态,0为普通状态,1为搜索进行中状态",
+    title: '搜索状态,0为普通状态,1为搜索进行中状态',
     order: 15,
-    view: "number",
-    type: "number"
+    view: 'number',
+    type: 'number',
   },
-  positionUrl: { title: "关键词所在URL", order: 16, view: "textarea", type: "string" },
+  positionUrl: { title: '关键词所在URL', order: 16, view: 'textarea', type: 'string' },
   relatedKeywordId: {
-    title: "相关关键词关联的关键词的id",
+    title: '相关关键词关联的关键词的id',
     order: 17,
-    view: "number",
-    type: "number"
+    view: 'number',
+    type: 'number',
   },
-  planId: { title: "关键词对应套餐的ID", order: 18, view: "text", type: "string" },
-  subscriptionId: { title: "订购表ID", order: 19, view: "text", type: "string" },
-  siteId: { title: "网站表ID", order: 20, view: "text", type: "string" }
+  planId: { title: '关键词对应套餐的ID', order: 18, view: 'text', type: 'string' },
+  subscriptionId: { title: '订购表ID', order: 19, view: 'text', type: 'string' },
+  siteId: { title: '网站表ID', order: 20, view: 'text', type: 'string' },
 };
 
 /**