parent
f91386d39d
commit
b83976b5d1
|
|
@ -0,0 +1,134 @@
|
||||||
|
<template>
|
||||||
|
<!-- 不能用v-if (i: 每个tab页的专属下标; index: 当前tab的下标 )-->
|
||||||
|
<view v-show="i === index">
|
||||||
|
<!-- top="120"下拉布局往下偏移,防止被悬浮菜单遮住 -->
|
||||||
|
<mescroll-body
|
||||||
|
:height="`${scrollHeight}px`"
|
||||||
|
@init="mescrollInit"
|
||||||
|
:down="downOption"
|
||||||
|
@down="downCallback"
|
||||||
|
:up="upOption"
|
||||||
|
@up="upCallback"
|
||||||
|
|
||||||
|
>
|
||||||
|
<slot :list="list"></slot>
|
||||||
|
</mescroll-body>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup>
|
||||||
|
import { http } from '@/utils/request'
|
||||||
|
import { sys } from '@climblee/uv-ui/libs/function'
|
||||||
|
import { ref, watch, nextTick, computed } from 'vue'
|
||||||
|
import useMescroll from '@/uni_modules/mescroll-uni/hooks/useMescroll.js'
|
||||||
|
// import { apiGoods } from '@/api/mock.js'
|
||||||
|
|
||||||
|
const props = defineProps({
|
||||||
|
i: Number,
|
||||||
|
index: {
|
||||||
|
type: Number,
|
||||||
|
default() {
|
||||||
|
return 0
|
||||||
|
},
|
||||||
|
},
|
||||||
|
top: {
|
||||||
|
type: Number,
|
||||||
|
default: 44,
|
||||||
|
},
|
||||||
|
apiUrl: {
|
||||||
|
type: String,
|
||||||
|
default: '',
|
||||||
|
},
|
||||||
|
params: {
|
||||||
|
type: Object,
|
||||||
|
default: () => {},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
const scrollHeight = computed(() => {
|
||||||
|
return sys().screenHeight - props.top
|
||||||
|
})
|
||||||
|
|
||||||
|
// 子组件的mescroll-body无需传入onPageScroll, onReachBottom
|
||||||
|
const { mescrollInit, downCallback, getMescroll } = useMescroll() // 调用mescroll的hook
|
||||||
|
|
||||||
|
defineExpose({ getMescroll }) // 使父组件可以通过ref调用到getMescroll方法 (必须)
|
||||||
|
|
||||||
|
const isAutoInit = props.i === props.index // 自动加载当前tab的数据
|
||||||
|
const downOption = {
|
||||||
|
auto: isAutoInit, // 自动加载当前tab的数据
|
||||||
|
}
|
||||||
|
|
||||||
|
const upOption = {
|
||||||
|
auto: false, // 不自动加载
|
||||||
|
noMoreSize: 4, //如果列表已无数据,可设置列表的总数量要大于半页才显示无更多数据;避免列表数据过少(比如只有一条数据),显示无更多数据会不好看; 默认5
|
||||||
|
empty: {
|
||||||
|
tip: '~ 空空如也 ~', // 提示
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
const isInit = ref(isAutoInit) // 当前tab是否已初始化
|
||||||
|
const list = ref([]) // 数据列表
|
||||||
|
|
||||||
|
// 监听下标的变化
|
||||||
|
watch(
|
||||||
|
() => props.index,
|
||||||
|
(val) => {
|
||||||
|
if (props.i === val && !isInit.value) mescrollTrigger()
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// 主动触发加载
|
||||||
|
const mescrollTrigger = () => {
|
||||||
|
isInit.value = true // 标记为true
|
||||||
|
const mescroll = getMescroll()
|
||||||
|
if (mescroll) {
|
||||||
|
if (mescroll.optDown.use) {
|
||||||
|
mescroll.triggerDownScroll()
|
||||||
|
} else {
|
||||||
|
mescroll.triggerUpScroll()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*上拉加载的回调: 其中mescroll.num:当前页 从1开始, mescroll.size:每页数据条数,默认10 */
|
||||||
|
const upCallback = async (mescroll) => {
|
||||||
|
await nextTick()
|
||||||
|
if (!props.apiUrl) return
|
||||||
|
http.request({
|
||||||
|
url: props.apiUrl,
|
||||||
|
method: 'GET',
|
||||||
|
params: {
|
||||||
|
per_page: mescroll.size,
|
||||||
|
page: mescroll.num,
|
||||||
|
...props.params,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.then((res) => {
|
||||||
|
const arr = res.data || [] // 当前页数据
|
||||||
|
if (mescroll.num == 1) list.value = [] //如果是第一页需手动制空列表
|
||||||
|
list.value = list.value.concat(arr) //追加新数据
|
||||||
|
mescroll.endSuccess(arr.length) // 请求成功, 结束加载
|
||||||
|
})
|
||||||
|
.catch(() => {
|
||||||
|
mescroll.endErr() // 请求失败, 结束加载
|
||||||
|
})
|
||||||
|
// let word = props.tabs[props.i].name // 具体项目中,您可能取的是tab中的type,status等字段
|
||||||
|
// apiGoods(mescroll.num, mescroll.size, word)
|
||||||
|
// .then((res) => {
|
||||||
|
// const list = res.list || [] // 当前页数据
|
||||||
|
// if (mescroll.num == 1) goods.value = [] //如果是第一页需手动制空列表
|
||||||
|
// goods.value = goods.value.concat(list) //追加新数据
|
||||||
|
// mescroll.endSuccess(list.length) // 请求成功, 结束加载
|
||||||
|
// })
|
||||||
|
// .catch(() => {
|
||||||
|
// mescroll.endErr() // 请求失败, 结束加载
|
||||||
|
// })
|
||||||
|
}
|
||||||
|
|
||||||
|
//点击空布局按钮的回调
|
||||||
|
const emptyClick = () => {
|
||||||
|
uni.showToast({
|
||||||
|
title: '点击了按钮,具体逻辑自行实现',
|
||||||
|
})
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
@ -0,0 +1,65 @@
|
||||||
|
<template>
|
||||||
|
<!-- 当mescroll-body写在子组件时,父页面需引入useMescrollComp.js -->
|
||||||
|
<mescroll-body
|
||||||
|
:height="`${scrollHeight}px`"
|
||||||
|
@init="mescrollInit"
|
||||||
|
@down="downCallback"
|
||||||
|
@up="upCallback"
|
||||||
|
>
|
||||||
|
<slot :list="list"></slot>
|
||||||
|
</mescroll-body>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup>
|
||||||
|
import { sys } from '@climblee/uv-ui/libs/function'
|
||||||
|
import { ref, computed, nextTick } from 'vue'
|
||||||
|
import useMescroll from '@/uni_modules/mescroll-uni/hooks/useMescroll.js'
|
||||||
|
import { http } from '@/utils/request'
|
||||||
|
const props = defineProps({
|
||||||
|
top: {
|
||||||
|
type: Number,
|
||||||
|
default: 44,
|
||||||
|
},
|
||||||
|
apiUrl: {
|
||||||
|
type: String,
|
||||||
|
default: '',
|
||||||
|
},
|
||||||
|
params: {
|
||||||
|
type: Object,
|
||||||
|
default: () => {},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
const scrollHeight = computed(() => {
|
||||||
|
return sys().screenHeight - props.top
|
||||||
|
})
|
||||||
|
const list = ref([]) // 数据列表
|
||||||
|
|
||||||
|
// 子组件的mescroll-body无需传入onPageScroll, onReachBottom
|
||||||
|
const { mescrollInit, downCallback, getMescroll } = useMescroll() // 调用mescroll的hook
|
||||||
|
|
||||||
|
// 使父组件可以通过ref调用到getMescroll方法 (必须)
|
||||||
|
defineExpose({ getMescroll })
|
||||||
|
|
||||||
|
// 上拉加载的回调: 其中num:当前页 从1开始, size:每页数据条数,默认10
|
||||||
|
const upCallback = async (mescroll) => {
|
||||||
|
await nextTick()
|
||||||
|
if (!props.apiUrl) return
|
||||||
|
http
|
||||||
|
.get(props.apiUrl, {
|
||||||
|
params: {
|
||||||
|
per_page: mescroll.size,
|
||||||
|
page: mescroll.num,
|
||||||
|
...props.params,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.then((res) => {
|
||||||
|
const curPageData = res.data || [] // 当前页数据
|
||||||
|
if (mescroll.num == 1) list.value = [] // 第一页需手动制空列表
|
||||||
|
list.value = list.value.concat(curPageData) // 追加新数据
|
||||||
|
mescroll.endSuccess(curPageData.length) // 隐藏加载进度
|
||||||
|
})
|
||||||
|
.catch(() => {
|
||||||
|
mescroll.endErr() // 隐藏加载进度
|
||||||
|
})
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
@ -121,6 +121,12 @@
|
||||||
"navigationBarTitleText": "任务提交"
|
"navigationBarTitleText": "任务提交"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"path": "task_hygienes_submit",
|
||||||
|
"style": {
|
||||||
|
"navigationBarTitleText": "清洁任务提交"
|
||||||
|
}
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"path": "detail",
|
"path": "detail",
|
||||||
"style": {
|
"style": {
|
||||||
|
|
@ -161,6 +167,7 @@
|
||||||
"navigationBarBackgroundColor": "#FFFFFF",
|
"navigationBarBackgroundColor": "#FFFFFF",
|
||||||
"backgroundColor": "#FFFFFF",
|
"backgroundColor": "#FFFFFF",
|
||||||
"navigationStyle": "custom",
|
"navigationStyle": "custom",
|
||||||
|
"backgroundColorTop":"#FFFFFF",
|
||||||
"app-plus": {
|
"app-plus": {
|
||||||
"bounce": "none",
|
"bounce": "none",
|
||||||
"scrollIndicator": "none"
|
"scrollIndicator": "none"
|
||||||
|
|
|
||||||
|
|
@ -2,7 +2,11 @@
|
||||||
<view>
|
<view>
|
||||||
<CuNavbar title="报销管理">
|
<CuNavbar title="报销管理">
|
||||||
<template #right>
|
<template #right>
|
||||||
<view @click="goPath('/pages/expense-account/submit')" class="text-24rpx text-white">申请</view>
|
<view
|
||||||
|
@click="goPath('/pages/expense-account/submit')"
|
||||||
|
class="text-24rpx text-white"
|
||||||
|
>申请</view
|
||||||
|
>
|
||||||
</template>
|
</template>
|
||||||
</CuNavbar>
|
</CuNavbar>
|
||||||
<uv-sticky bgColor="#fff">
|
<uv-sticky bgColor="#fff">
|
||||||
|
|
@ -13,36 +17,63 @@
|
||||||
:list="tabList"
|
:list="tabList"
|
||||||
></uv-tabs>
|
></uv-tabs>
|
||||||
</uv-sticky>
|
</uv-sticky>
|
||||||
|
<mescroll-body @init="mescrollInit" @down="downCallback" @up="upCallback">
|
||||||
<view class="px-base space-y-20rpx mt-30rpx">
|
<view class="px-base space-y-20rpx mt-30rpx">
|
||||||
<view
|
<view
|
||||||
v-for="item in 4"
|
v-for="item in list"
|
||||||
|
:key="item.id"
|
||||||
class="card-shadow bg-white rounded-19rpx p-base space-y-10rpx"
|
class="card-shadow bg-white rounded-19rpx p-base space-y-10rpx"
|
||||||
>
|
>
|
||||||
<view class="flex items-center justify-between">
|
<view class="flex items-center justify-between">
|
||||||
<view class="text-30rpx"> 报销申请 </view>
|
<view class="text-30rpx"> {{ item.type.name }} </view>
|
||||||
<view class="text-24rpx text-hex-999999">待完成</view>
|
<view
|
||||||
|
:style="[
|
||||||
|
{
|
||||||
|
color: statusFun(
|
||||||
|
item.workflow_check.check_status,
|
||||||
|
'statusExpense',
|
||||||
|
'color'
|
||||||
|
),
|
||||||
|
},
|
||||||
|
]"
|
||||||
|
class="text-24rpx"
|
||||||
|
>{{ item.workflow_check.check_status_text }}</view
|
||||||
|
>
|
||||||
</view>
|
</view>
|
||||||
<view class="text-24rpx text-hex-999999 flex">
|
<view class="text-24rpx text-hex-999999 flex">
|
||||||
<view class="w-140rpx">报销金额</view>
|
<view class="w-140rpx">报销金额</view>
|
||||||
<view class="text-primary">12313</view>
|
<view class="text-primary">{{ item.expense }}</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="text-24rpx text-hex-999999 flex">
|
<view class="text-24rpx text-hex-999999 flex">
|
||||||
<view class="w-140rpx">报销时间</view>
|
<view class="w-140rpx">报销时间</view>
|
||||||
<view class="text-hex-333">2022-01-01</view>
|
<view class="text-hex-333">{{ timeFormat(item.created_at) }}</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="text-24rpx text-hex-999999">
|
<view class="text-24rpx text-hex-999999">
|
||||||
<view class="">
|
<view class="">
|
||||||
<text class="w-140rpx inline-block">报销原因:</text>
|
<text class="w-140rpx inline-block">报销原因:</text>
|
||||||
<text class="text-hex-333 leading-27rpx">报销原因报销原因报销原因报销原因报销原因报销原因报销原因报销原因报销原因报销原因</text>
|
<text class="text-hex-333 leading-27rpx">{{ item.reason }}</text>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
</mescroll-body>
|
||||||
</view>
|
</view>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import CuNavbar from '@/components/cu-navbar/index'
|
import CuNavbar from '@/components/cu-navbar/index'
|
||||||
|
import { http } from '@/utils/request'
|
||||||
import { ref } from 'vue'
|
import { ref } from 'vue'
|
||||||
|
import { onLoad } from '@dcloudio/uni-app'
|
||||||
|
import { onPageScroll, onReachBottom } from '@dcloudio/uni-app'
|
||||||
|
import useMescroll from '@/uni_modules/mescroll-uni/hooks/useMescroll.js'
|
||||||
|
import { timeFormat } from '@climblee/uv-ui/libs/function'
|
||||||
|
import statusFun from '@/utils/status'
|
||||||
|
const { mescrollInit, downCallback, getMescroll } = useMescroll(
|
||||||
|
onPageScroll,
|
||||||
|
onReachBottom
|
||||||
|
)
|
||||||
|
const list = ref([])
|
||||||
|
|
||||||
const tabList = ref([
|
const tabList = ref([
|
||||||
{
|
{
|
||||||
name: '我的报销',
|
name: '我的报销',
|
||||||
|
|
@ -52,9 +83,31 @@ const tabList = ref([
|
||||||
},
|
},
|
||||||
])
|
])
|
||||||
|
|
||||||
|
const upCallback = async (mescroll) => {
|
||||||
|
const { size, num } = mescroll
|
||||||
|
|
||||||
|
try {
|
||||||
|
const resData = await http.get('/reimbursements', {
|
||||||
|
params: {
|
||||||
|
per_page: size,
|
||||||
|
page: num,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
const curPageData = resData || []
|
||||||
|
if (num === 1) list.value = []
|
||||||
|
list.value = list.value.concat(curPageData)
|
||||||
|
mescroll.endSuccess(curPageData.length)
|
||||||
|
} catch (error) {
|
||||||
|
console.log(error)
|
||||||
|
mescroll.endErr() // 请求失败, 结束加载
|
||||||
|
} finally {
|
||||||
|
// firstloading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const goPath = (url) => {
|
const goPath = (url) => {
|
||||||
uni.navigateTo({
|
uni.navigateTo({
|
||||||
url
|
url,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
|
||||||
|
|
@ -11,40 +11,37 @@
|
||||||
ref="formRef"
|
ref="formRef"
|
||||||
labelPosition="left"
|
labelPosition="left"
|
||||||
>
|
>
|
||||||
<uv-form-item
|
<uv-form-item label="报销分类" required prop="reimbursement_type_id">
|
||||||
label="报销分类"
|
<view
|
||||||
required
|
@click="openType"
|
||||||
:borderBottom="true"
|
keyName="name"
|
||||||
prop="content"
|
class="h-full w-full flex justify-end"
|
||||||
>
|
>
|
||||||
<uv-input
|
{{ type.name }}
|
||||||
border="none"
|
<uv-icon name="arrow-right"></uv-icon>
|
||||||
placeholder="请输入报销分类"
|
</view>
|
||||||
v-model="form.content"
|
|
||||||
></uv-input>
|
|
||||||
</uv-form-item>
|
</uv-form-item>
|
||||||
<uv-form-item
|
<uv-line color="#f5f5f5"></uv-line>
|
||||||
:borderBottom="true"
|
<uv-form-item label="报销金额" required prop="expense">
|
||||||
label="报销金额"
|
|
||||||
required
|
|
||||||
prop="content"
|
|
||||||
>
|
|
||||||
<uv-input
|
<uv-input
|
||||||
border="none"
|
:border="`none`"
|
||||||
placeholder="请输入报销金额"
|
placeholder="请输入报销金额"
|
||||||
v-model="form.content"
|
type="digit"
|
||||||
|
input-align="right"
|
||||||
|
v-model="form.expense"
|
||||||
></uv-input>
|
></uv-input>
|
||||||
</uv-form-item>
|
</uv-form-item>
|
||||||
|
<uv-line color="#f5f5f5"></uv-line>
|
||||||
<uv-form-item
|
<uv-form-item
|
||||||
label="报销原因"
|
label="报销原因"
|
||||||
required
|
required
|
||||||
prop="content"
|
prop="reason"
|
||||||
:borderBottom="true"
|
:borderBottom="true"
|
||||||
labelPosition="top"
|
labelPosition="top"
|
||||||
>
|
>
|
||||||
<uv-textarea
|
<uv-textarea
|
||||||
border="none"
|
:border="`none`"
|
||||||
v-model="form.content"
|
v-model="form.reason"
|
||||||
placeholder="请输入报销原因"
|
placeholder="请输入报销原因"
|
||||||
></uv-textarea>
|
></uv-textarea>
|
||||||
</uv-form-item>
|
</uv-form-item>
|
||||||
|
|
@ -55,7 +52,14 @@
|
||||||
required
|
required
|
||||||
>
|
>
|
||||||
<view class="w-full mt-15rpx">
|
<view class="w-full mt-15rpx">
|
||||||
<uv-upload></uv-upload>
|
<uv-upload
|
||||||
|
:maxCount="9"
|
||||||
|
multiple
|
||||||
|
:fileList="form.photos"
|
||||||
|
@afterRead="afterRead"
|
||||||
|
@delete="deletePic"
|
||||||
|
name="photos"
|
||||||
|
></uv-upload>
|
||||||
</view>
|
</view>
|
||||||
</uv-form-item>
|
</uv-form-item>
|
||||||
</uv-form>
|
</uv-form>
|
||||||
|
|
@ -65,11 +69,18 @@
|
||||||
<uv-button type="primary" @click="submit">提交</uv-button>
|
<uv-button type="primary" @click="submit">提交</uv-button>
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
|
<uv-picker
|
||||||
|
ref="typeRef"
|
||||||
|
keyName="name"
|
||||||
|
:columns="[typeList]"
|
||||||
|
@confirm="typeConfirm"
|
||||||
|
></uv-picker>
|
||||||
|
|
||||||
<uv-modal
|
<uv-modal
|
||||||
ref="modalRef"
|
ref="modalRef"
|
||||||
title="提示"
|
title="提示"
|
||||||
content="确定提交投诉?"
|
content="确定提交报销申请?"
|
||||||
@confirm="changePassword"
|
@confirm="onSubmit"
|
||||||
:showCancelButton="true"
|
:showCancelButton="true"
|
||||||
></uv-modal>
|
></uv-modal>
|
||||||
</view>
|
</view>
|
||||||
|
|
@ -79,42 +90,125 @@ import CuNavbar from '@/components/cu-navbar/index'
|
||||||
import Cell from '@/components/cell/index'
|
import Cell from '@/components/cell/index'
|
||||||
import { ref, reactive } from 'vue'
|
import { ref, reactive } from 'vue'
|
||||||
import { http } from '@/utils/request'
|
import { http } from '@/utils/request'
|
||||||
|
import { onLoad } from '@dcloudio/uni-app'
|
||||||
|
const typeRef = ref(null)
|
||||||
|
const typeList = ref([])
|
||||||
|
const type = ref({})
|
||||||
const modalRef = ref(null)
|
const modalRef = ref(null)
|
||||||
const formRef = ref(null)
|
const formRef = ref(null)
|
||||||
|
const loading = ref(false)
|
||||||
const form = reactive({
|
const form = reactive({
|
||||||
content: '',
|
reimbursement_type_id: '',
|
||||||
|
expense: '',
|
||||||
|
reason: '',
|
||||||
photos: [],
|
photos: [],
|
||||||
})
|
})
|
||||||
const rules = reactive({
|
const rules = reactive({
|
||||||
content: [
|
reimbursement_type_id: [
|
||||||
{ required: true, message: '请输入投诉内容' },
|
{
|
||||||
{ min: 20, message: '至少20字' },
|
required: true,
|
||||||
{ max: 200, message: '最多200字' },
|
message: '请选择报销分类',
|
||||||
|
},
|
||||||
],
|
],
|
||||||
|
expense: [{ required: true, message: '请输入报销金额' }],
|
||||||
|
reason: [{ required: true, message: '请输入报销原因' }],
|
||||||
photos: {
|
photos: {
|
||||||
type: 'array',
|
type: 'array',
|
||||||
|
required: true,
|
||||||
|
message: '请上传报销凭证',
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
onLoad(() => {
|
||||||
|
getTypes()
|
||||||
|
})
|
||||||
|
|
||||||
const submit = () => {
|
const submit = () => {
|
||||||
formRef.value.validate().then((res) => {
|
formRef.value.validate().then((res) => {
|
||||||
modalRef.value.open()
|
modalRef.value.open()
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
const onSubmit = async () => {
|
||||||
|
if (loading.value) return
|
||||||
|
loading.value = true
|
||||||
|
try {
|
||||||
|
const resData = await http.post('/reimbursements', {
|
||||||
|
reimbursement_type_id: form.reimbursement_type_id,
|
||||||
|
expense: form.expense,
|
||||||
|
reason: form.reason,
|
||||||
|
photos: form.photos.map((item) => item.url),
|
||||||
|
})
|
||||||
|
uni.showToast({
|
||||||
|
title: '提交成功',
|
||||||
|
icon: 'none',
|
||||||
|
})
|
||||||
|
formRef.value.resetFields()
|
||||||
|
uni.$emit('ex:submit', resData)
|
||||||
|
uni.navigateBack()
|
||||||
|
} catch (error) {
|
||||||
|
} finally {
|
||||||
|
loading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const changePassword = () => {
|
const afterRead = async (event) => {
|
||||||
// http
|
let lists = [].concat(event.file)
|
||||||
// .post('/auth/profile', {
|
let fileListLen = form[event.name].length
|
||||||
// password: form.password,
|
|
||||||
// password_confirmation: form.password2,
|
lists.map((item) => {
|
||||||
// })
|
form[event.name].push({
|
||||||
// .then((ress) => {
|
...item,
|
||||||
// uni.showToast({
|
status: 'uploading',
|
||||||
// title: '修改成功',
|
message: '上传中',
|
||||||
// duration: 2000,
|
})
|
||||||
// icon: 'none',
|
})
|
||||||
// })
|
for (let i = 0; i < lists.length; i++) {
|
||||||
// formRef.value.resetFields()
|
const result = await uploadFilePromise(lists[i].url)
|
||||||
// })
|
let item = form[event.name][fileListLen]
|
||||||
|
form[event.name].splice(
|
||||||
|
fileListLen,
|
||||||
|
1,
|
||||||
|
Object.assign(item, {
|
||||||
|
status: 'success',
|
||||||
|
message: '',
|
||||||
|
url: result,
|
||||||
|
})
|
||||||
|
)
|
||||||
|
fileListLen++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const uploadFilePromise = (url) => {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
http
|
||||||
|
.upload('/fileupload', {
|
||||||
|
filePath: url,
|
||||||
|
name: 'file',
|
||||||
|
})
|
||||||
|
.then((res) => {
|
||||||
|
resolve(res.url)
|
||||||
|
})
|
||||||
|
.catch((err) => {
|
||||||
|
reject(err)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const deletePic = (event) => {
|
||||||
|
form[event.name].splice(event.index, 1)
|
||||||
|
}
|
||||||
|
const getTypes = () => {
|
||||||
|
http.get('/keyword?parent_key=reimbursement_type').then((res) => {
|
||||||
|
typeList.value = res
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const typeConfirm = ({ value }) => {
|
||||||
|
type.value = value[0]
|
||||||
|
form.reimbursement_type_id = type.value.id
|
||||||
|
}
|
||||||
|
|
||||||
|
const openType = () => {
|
||||||
|
typeRef.value.open()
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
|
||||||
|
|
@ -1,7 +1,7 @@
|
||||||
<template>
|
<template>
|
||||||
<view>
|
<view>
|
||||||
<uv-sticky bgColor="#fff">
|
<uv-sticky bgColor="#fff">
|
||||||
<view class="flex-center h-88rpx">
|
<view class="flex-center h-44px">
|
||||||
<view class="flex-center flex-1" @click="selectMenu({ name: 'shore' })">
|
<view class="flex-center flex-1" @click="selectMenu({ name: 'shore' })">
|
||||||
<view>全部区域</view>
|
<view>全部区域</view>
|
||||||
<uv-icon
|
<uv-icon
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,50 @@
|
||||||
|
<template>
|
||||||
|
<view
|
||||||
|
class="card-shadow bg-white rounded-19rpx p-base space-y-10rpx"
|
||||||
|
@click.stop="onClick"
|
||||||
|
>
|
||||||
|
<view class="flex items-center justify-between">
|
||||||
|
<view class="text-30rpx">{{ item.name }}</view>
|
||||||
|
<view
|
||||||
|
:style="{
|
||||||
|
color: statusFun(item.taskable.status, item.taskable_type, 'color'),
|
||||||
|
}"
|
||||||
|
class="text-24rpx"
|
||||||
|
>{{ statusFun(item.taskable.status, item.taskable_type, 'name') }}</view
|
||||||
|
>
|
||||||
|
</view>
|
||||||
|
<view class="text-24rpx text-hex-999999">
|
||||||
|
任务时间:{{ timeFormat(item.start_at, 'yyyy年mm月dd日') }} -
|
||||||
|
{{ timeFormat(item.end_at, 'yyyy年mm月dd日') }}
|
||||||
|
</view>
|
||||||
|
<template v-if="item.taskable.status == 2">
|
||||||
|
<view class="py-10rpx">
|
||||||
|
<uv-line></uv-line>
|
||||||
|
</view>
|
||||||
|
<view class="flex justify-end">
|
||||||
|
<view @click.stop="onTask">
|
||||||
|
<uv-button type="primary" size="mini">去完成</uv-button>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
<script setup>
|
||||||
|
import statusFun from '@/utils/status'
|
||||||
|
import { timeFormat } from '@climblee/uv-ui/libs/function/index'
|
||||||
|
const props = defineProps({
|
||||||
|
item: Object,
|
||||||
|
})
|
||||||
|
|
||||||
|
const onClick = () => {
|
||||||
|
const type = props.item.taskable_type
|
||||||
|
console.log(type)
|
||||||
|
}
|
||||||
|
|
||||||
|
const onTask = (e) => {
|
||||||
|
const type = props.item.taskable_type
|
||||||
|
uni.navigateTo({
|
||||||
|
url: `/pages/task/${type}_submit?id=${props.item.id}`,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
@ -0,0 +1,74 @@
|
||||||
|
<template>
|
||||||
|
<view>
|
||||||
|
<CuNavbar title="我的任务"></CuNavbar>
|
||||||
|
<uv-sticky bgColor="#fff">
|
||||||
|
<uv-tabs
|
||||||
|
:activeStyle="{ color: '#ee2c37' }"
|
||||||
|
:scrollable="false"
|
||||||
|
lineColor="#ee2c37"
|
||||||
|
:list="tabList"
|
||||||
|
></uv-tabs>
|
||||||
|
</uv-sticky>
|
||||||
|
<mescroll-body @init="mescrollInit" @down="downCallback" @up="upCallback">
|
||||||
|
<view class="px-base space-y-20rpx mt-30rpx">
|
||||||
|
<view
|
||||||
|
v-for="item in 4"
|
||||||
|
class="card-shadow bg-white rounded-19rpx p-base space-y-10rpx"
|
||||||
|
>
|
||||||
|
<view class="text-30rpx">月度清洁任务</view>
|
||||||
|
<view class="text-24rpx text-hex-999999">
|
||||||
|
任务时间:2022年03月01号 - 2022年03月31号
|
||||||
|
</view>
|
||||||
|
<view class="text-24rpx text-hex-999999">待完成</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</mescroll-body>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
<script setup>
|
||||||
|
import { http } from '@/utils/request'
|
||||||
|
import CuNavbar from '@/components/cu-navbar/index'
|
||||||
|
import { computed, reactive, ref } from 'vue'
|
||||||
|
import { onPageScroll, onReachBottom } from '@dcloudio/uni-app'
|
||||||
|
import useMescroll from '@/uni_modules/mescroll-uni/hooks/useMescroll.js'
|
||||||
|
const { mescrollInit, downCallback, getMescroll } = useMescroll(
|
||||||
|
onPageScroll,
|
||||||
|
onReachBottom
|
||||||
|
)
|
||||||
|
const list = ref([])
|
||||||
|
const tabList = ref([
|
||||||
|
{
|
||||||
|
name: '任务列表',
|
||||||
|
apiUrl: '/tasks',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: '任务审核',
|
||||||
|
apiUrl: '/workflow',
|
||||||
|
params:{
|
||||||
|
|
||||||
|
}
|
||||||
|
},
|
||||||
|
])
|
||||||
|
|
||||||
|
const upCallback = async (mescroll) => {
|
||||||
|
const { size, num } = mescroll
|
||||||
|
|
||||||
|
try {
|
||||||
|
const resData = await http.get('/tasks', {
|
||||||
|
params: {
|
||||||
|
per_page: size,
|
||||||
|
page: num,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
const curPageData = resData.data || []
|
||||||
|
if (num === 1) list.value = []
|
||||||
|
list.value = list.value.concat(curPageData)
|
||||||
|
mescroll.endSuccess(curPageData.length)
|
||||||
|
} catch (error) {
|
||||||
|
console.log(error)
|
||||||
|
mescroll.endErr() // 请求失败, 结束加载
|
||||||
|
} finally {
|
||||||
|
// firstloading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
@ -3,32 +3,72 @@
|
||||||
<CuNavbar title="我的任务"></CuNavbar>
|
<CuNavbar title="我的任务"></CuNavbar>
|
||||||
<uv-sticky bgColor="#fff">
|
<uv-sticky bgColor="#fff">
|
||||||
<uv-tabs
|
<uv-tabs
|
||||||
|
height="44"
|
||||||
:activeStyle="{ color: '#ee2c37' }"
|
:activeStyle="{ color: '#ee2c37' }"
|
||||||
:scrollable="false"
|
:scrollable="false"
|
||||||
|
:current="tabIndex"
|
||||||
lineColor="#ee2c37"
|
lineColor="#ee2c37"
|
||||||
:list="tabList"
|
:list="tabList"
|
||||||
|
@change="tabChange"
|
||||||
></uv-tabs>
|
></uv-tabs>
|
||||||
</uv-sticky>
|
</uv-sticky>
|
||||||
<view class="px-base space-y-20rpx mt-30rpx">
|
|
||||||
<view v-for="item in 4" class="card-shadow bg-white rounded-19rpx p-base space-y-10rpx">
|
<MescrollItem
|
||||||
<view class="text-30rpx">月度清洁任务</view>
|
ref="mescrollItem0"
|
||||||
<view class="text-24rpx text-hex-999999">
|
:top="88"
|
||||||
任务时间:2022年03月01号 - 2022年03月31号
|
:i="0"
|
||||||
</view>
|
:index="tabIndex"
|
||||||
<view class="text-24rpx text-hex-999999">待完成</view>
|
:apiUrl="tabList[0].apiUrl"
|
||||||
|
>
|
||||||
|
<template v-slot="{ list }">
|
||||||
|
<view class="space-y-15rpx p-base">
|
||||||
|
<view v-for="(item, i) in list" :key="i">
|
||||||
|
<Item :item="item"></Item>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
</template>
|
||||||
|
</MescrollItem>
|
||||||
|
<MescrollItem
|
||||||
|
ref="mescrollItem1"
|
||||||
|
:i="1"
|
||||||
|
:top="88"
|
||||||
|
:index="tabIndex"
|
||||||
|
:apiUrl="tabList[1].apiUrl"
|
||||||
|
:params="tabList[1].params"
|
||||||
|
></MescrollItem>
|
||||||
</view>
|
</view>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import CuNavbar from '@/components/cu-navbar/index'
|
import CuNavbar from '@/components/cu-navbar/index'
|
||||||
import { ref } from 'vue'
|
import { computed, reactive, ref } from 'vue'
|
||||||
|
import Item from './components/item.vue'
|
||||||
|
import { onPageScroll, onReachBottom, onShow } from '@dcloudio/uni-app'
|
||||||
|
import useMescrollMore from '@/uni_modules/mescroll-uni/hooks/useMescrollMore.js'
|
||||||
|
import MescrollItem from '@/components/mescroll-api/more.vue'
|
||||||
|
|
||||||
|
const mescrollItem0 = ref(null)
|
||||||
|
const mescrollItem1 = ref(null)
|
||||||
|
|
||||||
|
const mescrollItems = [mescrollItem0, mescrollItem1]
|
||||||
|
const { tabIndex, getMescroll, scrollToLastY } = useMescrollMore(
|
||||||
|
mescrollItems,
|
||||||
|
onPageScroll,
|
||||||
|
onReachBottom
|
||||||
|
)
|
||||||
const tabList = ref([
|
const tabList = ref([
|
||||||
{
|
{
|
||||||
name: '任务列表',
|
name: '任务列表',
|
||||||
|
apiUrl: '/tasks',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: '任务审核',
|
name: '任务审核',
|
||||||
|
apiUrl: '/workflow',
|
||||||
|
params: {},
|
||||||
},
|
},
|
||||||
])
|
])
|
||||||
|
|
||||||
|
const tabChange = ({ index }) => {
|
||||||
|
tabIndex.value = index
|
||||||
|
scrollToLastY()
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,154 @@
|
||||||
|
<template>
|
||||||
|
<view class="p-base">
|
||||||
|
<CuNavbar title="清洁任务提交"></CuNavbar>
|
||||||
|
<view class="card-shadow px-base">
|
||||||
|
<uv-form
|
||||||
|
labelPosition="left"
|
||||||
|
:model="form"
|
||||||
|
:rules="rules"
|
||||||
|
ref="formRef"
|
||||||
|
errorType="toast"
|
||||||
|
labelWidth="150rpx"
|
||||||
|
>
|
||||||
|
<uv-form-item required label="清洁范围" prop="description">
|
||||||
|
<uv-input
|
||||||
|
placeholder="请输入清洁范围"
|
||||||
|
inputAlign="right"
|
||||||
|
:border="`none`"
|
||||||
|
v-model="form.description"
|
||||||
|
>
|
||||||
|
</uv-input>
|
||||||
|
</uv-form-item>
|
||||||
|
<uv-line color="#f5f5f5"></uv-line>
|
||||||
|
<uv-form-item
|
||||||
|
label="报销凭证"
|
||||||
|
labelPosition="top"
|
||||||
|
prop="photos"
|
||||||
|
required
|
||||||
|
>
|
||||||
|
<view class="w-full mt-15rpx">
|
||||||
|
<uv-upload
|
||||||
|
:maxCount="9"
|
||||||
|
multiple
|
||||||
|
:fileList="form.photos"
|
||||||
|
@afterRead="afterRead"
|
||||||
|
@delete="deletePic"
|
||||||
|
name="photos"
|
||||||
|
></uv-upload>
|
||||||
|
</view>
|
||||||
|
</uv-form-item>
|
||||||
|
</uv-form>
|
||||||
|
</view>
|
||||||
|
<view class="mt-100rpx">
|
||||||
|
<uv-button type="primary" @click="submit">提交</uv-button>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<uv-modal
|
||||||
|
ref="modalRef"
|
||||||
|
title="提示"
|
||||||
|
content="确定提交吗?"
|
||||||
|
@confirm="onSubmit"
|
||||||
|
:showCancelButton="true"
|
||||||
|
></uv-modal>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
<script setup>
|
||||||
|
import CuNavbar from '@/components/cu-navbar/index'
|
||||||
|
import { ref, reactive, computed } from 'vue'
|
||||||
|
import { onLoad } from '@dcloudio/uni-app'
|
||||||
|
import { http } from '@/utils/request'
|
||||||
|
const formRef = ref(null)
|
||||||
|
const modalRef = ref(null)
|
||||||
|
const id = ref(0)
|
||||||
|
const loading = ref(false)
|
||||||
|
const form = reactive({
|
||||||
|
description: '',
|
||||||
|
photos: [],
|
||||||
|
})
|
||||||
|
const rules = reactive({
|
||||||
|
description: [{ required: true, message: '请输入清洁范围' }],
|
||||||
|
photos: {
|
||||||
|
type: 'array',
|
||||||
|
required: true,
|
||||||
|
message: '请上传报销凭证',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
onLoad((options) => {
|
||||||
|
id.value = options.id
|
||||||
|
})
|
||||||
|
|
||||||
|
const submit = () => {
|
||||||
|
formRef.value.validate().then((res) => {
|
||||||
|
modalRef.value.open()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const onSubmit = async () => {
|
||||||
|
if (loading.value) return
|
||||||
|
loading.value = true
|
||||||
|
try {
|
||||||
|
const resData = await http.post(`/tasks/${id.value}/submit`, {
|
||||||
|
id: id.value,
|
||||||
|
description: form.description,
|
||||||
|
photos: form.photos.map((item) => item.url),
|
||||||
|
})
|
||||||
|
uni.showToast({
|
||||||
|
title: '提交成功',
|
||||||
|
icon: 'none',
|
||||||
|
})
|
||||||
|
formRef.value.resetFields()
|
||||||
|
uni.$emit('ex:submit', resData)
|
||||||
|
uni.navigateBack()
|
||||||
|
} catch (error) {
|
||||||
|
} finally {
|
||||||
|
loading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const afterRead = async (event) => {
|
||||||
|
let lists = [].concat(event.file)
|
||||||
|
let fileListLen = form[event.name].length
|
||||||
|
|
||||||
|
lists.map((item) => {
|
||||||
|
form[event.name].push({
|
||||||
|
...item,
|
||||||
|
status: 'uploading',
|
||||||
|
message: '上传中',
|
||||||
|
})
|
||||||
|
})
|
||||||
|
for (let i = 0; i < lists.length; i++) {
|
||||||
|
const result = await uploadFilePromise(lists[i].url)
|
||||||
|
let item = form[event.name][fileListLen]
|
||||||
|
form[event.name].splice(
|
||||||
|
fileListLen,
|
||||||
|
1,
|
||||||
|
Object.assign(item, {
|
||||||
|
status: 'success',
|
||||||
|
message: '',
|
||||||
|
url: result,
|
||||||
|
})
|
||||||
|
)
|
||||||
|
fileListLen++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const uploadFilePromise = (url) => {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
http
|
||||||
|
.upload('/fileupload', {
|
||||||
|
filePath: url,
|
||||||
|
name: 'file',
|
||||||
|
})
|
||||||
|
.then((res) => {
|
||||||
|
resolve(res.url)
|
||||||
|
})
|
||||||
|
.catch((err) => {
|
||||||
|
reject(err)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const deletePic = (event) => {
|
||||||
|
form[event.name].splice(event.index, 1)
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
@ -0,0 +1,40 @@
|
||||||
|
<template>
|
||||||
|
<view class="flex rounded-19rpx box-content card" @click="goPage(`/pages/user/detail?id=${item.id}`)">
|
||||||
|
<view class="rounded-full w-100rpx h-100rpx overflow-hidden">
|
||||||
|
<image class="w-full h-full" :src="item.avatar"></image>
|
||||||
|
</view>
|
||||||
|
<view class="flex-1 ml-20rpx flex flex-col justify-between">
|
||||||
|
<view class="flex text-sm">
|
||||||
|
<view>{{ item.name }}</view>
|
||||||
|
<template v-if="item.jobs">
|
||||||
|
<uv-tags
|
||||||
|
v-for="job in item.jobs"
|
||||||
|
class="ml-20rpx"
|
||||||
|
:text="job.name"
|
||||||
|
:key="job.id"
|
||||||
|
plain
|
||||||
|
size="mini"
|
||||||
|
type="primary"
|
||||||
|
></uv-tags>
|
||||||
|
</template>
|
||||||
|
</view>
|
||||||
|
<view class="text-24rpx text-hex-999">
|
||||||
|
<view class="">电话:{{ item.phone }}</view>
|
||||||
|
<view class="" v-if="item.store">门店:{{ item.store.address }}</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
<script setup>
|
||||||
|
defineProps({
|
||||||
|
item: {
|
||||||
|
type: Object,
|
||||||
|
default: () => ({}),
|
||||||
|
},
|
||||||
|
})
|
||||||
|
const goPage = (url) => {
|
||||||
|
uni.navigateTo({
|
||||||
|
url,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
@ -38,13 +38,27 @@
|
||||||
cancelText="取消"
|
cancelText="取消"
|
||||||
>
|
>
|
||||||
</uv-action-sheet>
|
</uv-action-sheet>
|
||||||
|
|
||||||
|
<uv-modal
|
||||||
|
ref="modalRef"
|
||||||
|
:title="modalOptin.title"
|
||||||
|
:content="modalOptin.content"
|
||||||
|
@confirm="modalOptin.confirm"
|
||||||
|
:showCancelButton="true"
|
||||||
|
></uv-modal>
|
||||||
</view>
|
</view>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import { ref } from 'vue'
|
import { ref, reactive } from 'vue'
|
||||||
import CuNavbar from '@/components/cu-navbar/index'
|
import CuNavbar from '@/components/cu-navbar/index'
|
||||||
import { http } from '@/utils/request'
|
import { http } from '@/utils/request'
|
||||||
import { onLoad, onShow } from '@dcloudio/uni-app'
|
import { onLoad, onShow } from '@dcloudio/uni-app'
|
||||||
|
const modalRef = ref(null)
|
||||||
|
const modalOptin = reactive({
|
||||||
|
title: '提示',
|
||||||
|
content: '',
|
||||||
|
confirm: () => {},
|
||||||
|
})
|
||||||
const actionSheet = ref(null)
|
const actionSheet = ref(null)
|
||||||
const id = ref(null)
|
const id = ref(null)
|
||||||
const detail = ref({})
|
const detail = ref({})
|
||||||
|
|
@ -72,7 +86,6 @@ onShow(() => {
|
||||||
|
|
||||||
const getDetail = () => {
|
const getDetail = () => {
|
||||||
http.get(`/hr/employee/${id.value}`).then((res) => {
|
http.get(`/hr/employee/${id.value}`).then((res) => {
|
||||||
console.log(res)
|
|
||||||
detail.value = res
|
detail.value = res
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
@ -86,6 +99,36 @@ const select = (e) => {
|
||||||
uni.navigateTo({
|
uni.navigateTo({
|
||||||
url: `/pages/user/update?id=${id.value}`,
|
url: `/pages/user/update?id=${id.value}`,
|
||||||
})
|
})
|
||||||
|
} else if (value === 'quit') {
|
||||||
|
modalOptin.content = '是否确认离职该员工?'
|
||||||
|
modalOptin.confirm = onQuick
|
||||||
|
modalRef.value.open()
|
||||||
|
} else if (value === 'delete') {
|
||||||
|
modalOptin.content = '是否确认删除该员工?'
|
||||||
|
modalOptin.confirm = onDelete
|
||||||
|
modalRef.value.open()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
const onDelete = () => {
|
||||||
|
http.delete(`/hr/employee/${id.value}`).then((res) => {
|
||||||
|
uni.$emit('user:onRefresh')
|
||||||
|
uni.showToast({
|
||||||
|
title: '删除成功',
|
||||||
|
duration: 2000,
|
||||||
|
icon: 'none',
|
||||||
|
})
|
||||||
|
uni.navigateBack()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
const onQuick = () => {
|
||||||
|
http.post(`/hr/employee/${id.value}/leave`).then((res) => {
|
||||||
|
uni.showToast({
|
||||||
|
title: '离职成功',
|
||||||
|
duration: 2000,
|
||||||
|
icon: 'none',
|
||||||
|
})
|
||||||
|
uni.$emit('user:onRefresh')
|
||||||
|
uni.navigateBack()
|
||||||
|
})
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,128 @@
|
||||||
|
<template>
|
||||||
|
<view>
|
||||||
|
<CuNavbar title="员工管理">
|
||||||
|
<template #right>
|
||||||
|
<text @click="goPage('/pages/user/update')" class="text-white"
|
||||||
|
>添加</text
|
||||||
|
>
|
||||||
|
</template>
|
||||||
|
</CuNavbar>
|
||||||
|
<StoreDropDown></StoreDropDown>
|
||||||
|
|
||||||
|
<mescroll-body @init="mescrollInit" @down="downCallback" @up="upCallback">
|
||||||
|
<view class="mt-15rpx px-base">
|
||||||
|
<!-- <uv-swipe-action> -->
|
||||||
|
<view class="space-y-15rpx">
|
||||||
|
<view
|
||||||
|
class="rounded-19rpx card p-0 overflow-hidden"
|
||||||
|
v-for="(item, i) in list"
|
||||||
|
:key="i"
|
||||||
|
@click="goPage(`/pages/user/detail?id=${item.id}`)"
|
||||||
|
>
|
||||||
|
<!-- <uv-swipe-action-item :options="options"> -->
|
||||||
|
<view class="flex rounded-19rpx box-content card">
|
||||||
|
<view class="rounded-4rpx w-100rpx h-100rpx">
|
||||||
|
<image class="w-full h-full" :src="item.avatar"></image>
|
||||||
|
</view>
|
||||||
|
<view
|
||||||
|
class="flex-1 ml-20rpx flex flex-col justify-between py-10rpx"
|
||||||
|
>
|
||||||
|
<view class="flex items-center text-sm">
|
||||||
|
<view>{{ item.name }}</view>
|
||||||
|
<template v-if="item.jobs">
|
||||||
|
<uv-tags
|
||||||
|
v-for="job in item.jobs"
|
||||||
|
class="ml-20rpx"
|
||||||
|
:text="job.name"
|
||||||
|
:key="job.id"
|
||||||
|
plain
|
||||||
|
size="mini"
|
||||||
|
type="primary"
|
||||||
|
></uv-tags>
|
||||||
|
</template>
|
||||||
|
</view>
|
||||||
|
<view class="flex justify-between">
|
||||||
|
<view class="text-28rpx text-hex-999 text-xs">{{
|
||||||
|
item.phone
|
||||||
|
}}</view>
|
||||||
|
<view v-if="item.store">{{ item.store.address }}</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<!-- </uv-swipe-action-item> -->
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<!-- </uv-swipe-action> -->
|
||||||
|
</view>
|
||||||
|
</mescroll-body>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
<script setup>
|
||||||
|
import CuNavbar from '@/components/cu-navbar/index'
|
||||||
|
import { http } from '@/utils/request'
|
||||||
|
import { computed, reactive, ref } from 'vue'
|
||||||
|
import StoreDropDown from '@/pages/home/components/store-drop-down/index.vue'
|
||||||
|
import { onPageScroll, onReachBottom } from '@dcloudio/uni-app'
|
||||||
|
import useMescroll from '@/uni_modules/mescroll-uni/hooks/useMescroll.js'
|
||||||
|
import { onShow } from '@dcloudio/uni-app'
|
||||||
|
const { mescrollInit, downCallback, getMescroll } = useMescroll(
|
||||||
|
onPageScroll,
|
||||||
|
onReachBottom
|
||||||
|
)
|
||||||
|
const firstloading = ref(true)
|
||||||
|
const list = ref([])
|
||||||
|
const options = [
|
||||||
|
{
|
||||||
|
text: '删除',
|
||||||
|
style: {
|
||||||
|
backgroundColor: '#f56c6c',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: '修改',
|
||||||
|
style: {
|
||||||
|
backgroundColor: '#3c9cff',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: '离职',
|
||||||
|
style: {
|
||||||
|
backgroundColor: '#f9ae3d',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
onShow(() => {
|
||||||
|
if (!firstloading.value) {
|
||||||
|
getMescroll().resetUpScroll()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
const upCallback = async (mescroll) => {
|
||||||
|
const { size, num } = mescroll
|
||||||
|
|
||||||
|
try {
|
||||||
|
const resData = await http.get('/hr/employee', {
|
||||||
|
params: {
|
||||||
|
per_page: size,
|
||||||
|
page: num,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
const curPageData = resData.data || []
|
||||||
|
if (num === 1) list.value = []
|
||||||
|
list.value = list.value.concat(curPageData)
|
||||||
|
mescroll.endSuccess(curPageData.length)
|
||||||
|
} catch (error) {
|
||||||
|
console.log(error)
|
||||||
|
mescroll.endErr() // 请求失败, 结束加载
|
||||||
|
} finally {
|
||||||
|
firstloading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const goPage = (url) => {
|
||||||
|
uni.navigateTo({
|
||||||
|
url,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
@ -8,110 +8,44 @@
|
||||||
</template>
|
</template>
|
||||||
</CuNavbar>
|
</CuNavbar>
|
||||||
<StoreDropDown></StoreDropDown>
|
<StoreDropDown></StoreDropDown>
|
||||||
|
<MescrollApiOne
|
||||||
<mescroll-body @init="mescrollInit" @down="downCallback" @up="upCallback">
|
:top="88"
|
||||||
<view class="mt-15rpx px-base">
|
ref="mescrollItem"
|
||||||
<!-- <uv-swipe-action> -->
|
apiUrl="/hr/employee"
|
||||||
<view class="space-y-15rpx">
|
:params="params"
|
||||||
<view
|
|
||||||
class="rounded-19rpx card p-0 overflow-hidden"
|
|
||||||
v-for="(item, i) in list"
|
|
||||||
:key="i"
|
|
||||||
@click="goPage(`/pages/user/detail?id=${item.id}`)"
|
|
||||||
>
|
>
|
||||||
<!-- <uv-swipe-action-item :options="options"> -->
|
<template v-slot="{ list }">
|
||||||
<view class="flex rounded-19rpx box-content card">
|
<view class="space-y-15rpx p-base">
|
||||||
<view class="rounded-4rpx w-100rpx h-100rpx">
|
<view v-for="(item, i) in list" :key="i">
|
||||||
<image class="w-full h-full" :src="item.avatar"></image>
|
<Item :item="item"></Item>
|
||||||
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view
|
|
||||||
class="flex-1 ml-20rpx flex flex-col justify-between py-10rpx"
|
|
||||||
>
|
|
||||||
<view class="flex items-center text-sm">
|
|
||||||
<view>{{ item.name }}</view>
|
|
||||||
<template v-if="item.jobs">
|
|
||||||
<uv-tags
|
|
||||||
v-for="job in item.jobs"
|
|
||||||
class="ml-20rpx"
|
|
||||||
:text="job.name"
|
|
||||||
:key="job.id"
|
|
||||||
plain
|
|
||||||
size="mini"
|
|
||||||
type="primary"
|
|
||||||
></uv-tags>
|
|
||||||
</template>
|
</template>
|
||||||
</view>
|
</MescrollApiOne>
|
||||||
<view class="flex justify-between">
|
|
||||||
<view class="text-28rpx text-hex-999 text-xs">{{
|
|
||||||
item.phone
|
|
||||||
}}</view>
|
|
||||||
<view v-if="item.store">{{ item.store.address }}</view>
|
|
||||||
</view>
|
|
||||||
</view>
|
|
||||||
</view>
|
|
||||||
<!-- </uv-swipe-action-item> -->
|
|
||||||
</view>
|
|
||||||
</view>
|
|
||||||
<!-- </uv-swipe-action> -->
|
|
||||||
</view>
|
|
||||||
</mescroll-body>
|
|
||||||
</view>
|
</view>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
|
import { ref } from 'vue'
|
||||||
import CuNavbar from '@/components/cu-navbar/index'
|
import CuNavbar from '@/components/cu-navbar/index'
|
||||||
import { http } from '@/utils/request'
|
|
||||||
import { computed, reactive, ref } from 'vue'
|
|
||||||
import StoreDropDown from '@/pages/home/components/store-drop-down/index.vue'
|
import StoreDropDown from '@/pages/home/components/store-drop-down/index.vue'
|
||||||
import { onPageScroll, onReachBottom } from '@dcloudio/uni-app'
|
import Item from './components/item.vue'
|
||||||
import useMescroll from '@/uni_modules/mescroll-uni/hooks/useMescroll.js'
|
import MescrollApiOne from '@/components/mescroll-api/one'
|
||||||
|
import { onPageScroll, onReachBottom, onLoad } from '@dcloudio/uni-app'
|
||||||
|
import useMescrollComp from '@/uni_modules/mescroll-uni/hooks/useMescrollComp.js'
|
||||||
|
const { mescrollItem } = useMescrollComp(onPageScroll, onReachBottom)
|
||||||
|
const params = ref(null)
|
||||||
|
|
||||||
const { mescrollInit, downCallback, getMescroll } = useMescroll(
|
onLoad(()=>{
|
||||||
onPageScroll,
|
uni.$on('user:onRefresh',onRefresh)
|
||||||
onReachBottom
|
})
|
||||||
)
|
|
||||||
|
|
||||||
const list = ref([])
|
const onRefresh = () => {
|
||||||
const options = [
|
// params.value = {
|
||||||
{
|
// city_code: 11,
|
||||||
text: '删除',
|
// store_id: 12,
|
||||||
style: {
|
// }
|
||||||
backgroundColor: '#f56c6c',
|
mescrollItem.value.getMescroll().resetUpScroll()
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
text: '修改',
|
|
||||||
style: {
|
|
||||||
backgroundColor: '#3c9cff',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
text: '离职',
|
|
||||||
style: {
|
|
||||||
backgroundColor: '#f9ae3d',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
]
|
|
||||||
|
|
||||||
const upCallback = async (mescroll) => {
|
|
||||||
const { size, num } = mescroll
|
|
||||||
|
|
||||||
try {
|
|
||||||
const resData = await http.get('/hr/employee', {
|
|
||||||
params: {
|
|
||||||
per_page: size,
|
|
||||||
page: num,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
const curPageData = resData.data || []
|
|
||||||
if (num === 1) list.value = []
|
|
||||||
list.value = list.value.concat(curPageData)
|
|
||||||
mescroll.endSuccess(curPageData.length)
|
|
||||||
} catch (error) {
|
|
||||||
console.log(error)
|
|
||||||
mescroll.endErr() // 请求失败, 结束加载
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const goPage = (url) => {
|
const goPage = (url) => {
|
||||||
uni.navigateTo({
|
uni.navigateTo({
|
||||||
url,
|
url,
|
||||||
|
|
|
||||||
|
|
@ -36,7 +36,7 @@
|
||||||
</uv-input>
|
</uv-input>
|
||||||
</uv-form-item>
|
</uv-form-item>
|
||||||
<uv-line color="#f5f5f5"></uv-line>
|
<uv-line color="#f5f5f5"></uv-line>
|
||||||
<uv-form-item required label="登录用户名" prop="name">
|
<uv-form-item required label="登录用户名" prop="username">
|
||||||
<uv-input
|
<uv-input
|
||||||
placeholder="请输入登录用户名"
|
placeholder="请输入登录用户名"
|
||||||
inputAlign="right"
|
inputAlign="right"
|
||||||
|
|
@ -56,6 +56,9 @@
|
||||||
>
|
>
|
||||||
</uv-input>
|
</uv-input>
|
||||||
</uv-form-item>
|
</uv-form-item>
|
||||||
|
<uv-form-item :required="!isEdit" label="门店" prop="password">
|
||||||
|
|
||||||
|
</uv-form-item>
|
||||||
</uv-form>
|
</uv-form>
|
||||||
</view>
|
</view>
|
||||||
<view class="mt-100rpx">
|
<view class="mt-100rpx">
|
||||||
|
|
@ -115,7 +118,7 @@ const rules = computed(() => {
|
||||||
})
|
})
|
||||||
onLoad((options) => {
|
onLoad((options) => {
|
||||||
id.value = options.id
|
id.value = options.id
|
||||||
getDetail()
|
if(isEdit.value) getDetail()
|
||||||
})
|
})
|
||||||
const isEdit = computed(() => !!id.value)
|
const isEdit = computed(() => !!id.value)
|
||||||
const title = computed(() => (isEdit.value ? '员工修改' : '员工添加'))
|
const title = computed(() => (isEdit.value ? '员工修改' : '员工添加'))
|
||||||
|
|
@ -142,6 +145,7 @@ const updateUser = async () => {
|
||||||
await http.put(`/hr/employee/${id.value}`, {
|
await http.put(`/hr/employee/${id.value}`, {
|
||||||
...form,
|
...form,
|
||||||
})
|
})
|
||||||
|
uni.$emit('user:onRefresh')
|
||||||
uni.navigateBack()
|
uni.navigateBack()
|
||||||
formRef.value.resetFields()
|
formRef.value.resetFields()
|
||||||
uni.showToast({
|
uni.showToast({
|
||||||
|
|
|
||||||
|
|
@ -45,6 +45,16 @@ http.interceptors.request.use((config) => {
|
||||||
// 必须使用异步函数,注意
|
// 必须使用异步函数,注意
|
||||||
http.interceptors.response.use(async (response) => {
|
http.interceptors.response.use(async (response) => {
|
||||||
|
|
||||||
|
// 处理响应结果
|
||||||
|
const { statusCode, data } = response
|
||||||
|
if (statusCode !== 200 && statusCode !== 201) {
|
||||||
|
uni.showToast({
|
||||||
|
title: data.message || '服务器错误',
|
||||||
|
icon: 'none'
|
||||||
|
})
|
||||||
|
return Promise.reject(response)
|
||||||
|
}
|
||||||
|
|
||||||
const { isTransformResponse = true, isReturnNativeResponse } = response.config.custom;
|
const { isTransformResponse = true, isReturnNativeResponse } = response.config.custom;
|
||||||
// 是否返回原生响应头
|
// 是否返回原生响应头
|
||||||
if (isReturnNativeResponse) {
|
if (isReturnNativeResponse) {
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,79 @@
|
||||||
|
const data = {
|
||||||
|
statusExpense: [{
|
||||||
|
value: 1,
|
||||||
|
name: '待提审',
|
||||||
|
color: '#f56c6c'
|
||||||
|
}, {
|
||||||
|
value: 2,
|
||||||
|
name: '审核中',
|
||||||
|
color: '#f56c6c'
|
||||||
|
}, {
|
||||||
|
value: 3,
|
||||||
|
name: '审核通过',
|
||||||
|
color: '#3c9cff'
|
||||||
|
}, {
|
||||||
|
value: 4,
|
||||||
|
name: '未通过',
|
||||||
|
color: '#999999'
|
||||||
|
}, {
|
||||||
|
value: 5,
|
||||||
|
name: '已取消',
|
||||||
|
color: '#999999'
|
||||||
|
}],
|
||||||
|
//清洁任务
|
||||||
|
task_hygienes: [{
|
||||||
|
value: 1,
|
||||||
|
name: '未开始',
|
||||||
|
color: '#f56c6c'
|
||||||
|
}, {
|
||||||
|
value: 2,
|
||||||
|
name: '待完成',
|
||||||
|
color: '#f56c6c'
|
||||||
|
}, {
|
||||||
|
value: 3,
|
||||||
|
name: '审核中',
|
||||||
|
color: '#3c9cff'
|
||||||
|
}, {
|
||||||
|
value: 4,
|
||||||
|
name: '未通过',
|
||||||
|
color: '#999999'
|
||||||
|
}, {
|
||||||
|
value: 5,
|
||||||
|
name: '已完成',
|
||||||
|
color: '#3c9cff'
|
||||||
|
}, {
|
||||||
|
value: 6,
|
||||||
|
name: '未完成',
|
||||||
|
color: '#999999'
|
||||||
|
}],
|
||||||
|
//总账录入
|
||||||
|
task_ledgers: [{
|
||||||
|
value: 1,
|
||||||
|
name: '未开始',
|
||||||
|
color: '#f56c6c'
|
||||||
|
}, {
|
||||||
|
value: 2,
|
||||||
|
name: '待完成',
|
||||||
|
color: '#f56c6c'
|
||||||
|
}, {
|
||||||
|
value: 3,
|
||||||
|
name: '已完成',
|
||||||
|
color: '#3c9cff'
|
||||||
|
}, {
|
||||||
|
value: 4,
|
||||||
|
name: '未完成',
|
||||||
|
color: '#999999'
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
export default function (value, status, key) {
|
||||||
|
if (!status) {
|
||||||
|
return value
|
||||||
|
}
|
||||||
|
|
||||||
|
const item = data[status]?.find((item) => item.value === value)
|
||||||
|
|
||||||
|
if (item && key) {
|
||||||
|
return item[key] ?? value
|
||||||
|
}
|
||||||
|
return item ?? value
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue