Browse Source

结果反馈

master
parent
commit
ae6ee80964
  1. 20
      api/inspections/index.js
  2. 12
      components/cs-dict-tag/index.vue
  3. 2
      main.js
  4. 16
      pages.json
  5. 7
      pages/index.vue
  6. 134
      sub/task/detail.vue
  7. 84
      sub/task/enforce.vue
  8. 77
      sub/task/locate.vue
  9. 102
      sub/task/record.vue
  10. 115
      uni_modules/uni-easyinput/changelog.md
  11. 54
      uni_modules/uni-easyinput/components/uni-easyinput/common.js
  12. 676
      uni_modules/uni-easyinput/components/uni-easyinput/uni-easyinput.vue
  13. 88
      uni_modules/uni-easyinput/package.json
  14. 11
      uni_modules/uni-easyinput/readme.md
  15. 1
      utils/ruoyi.js
  16. 6
      wxcomponents/vant/dropdown-menu/index.wxss

20
api/inspections/index.js

@ -48,5 +48,25 @@ export const InspectionsApi = {
method: "POST",
data
})
},
/**
* 转发
*/
replay: (data) => {
return request({
url: `/system/enterprise-inspections/passOn`,
method: "POST",
data
})
},
/**
* 开始执法
*/
beginEnforce: (data) => {
return request({
url: `/system/inspections-log/confirmSignIn`,
method: 'PUT',
data
})
}
}

12
components/cs-dict-tag/index.vue

@ -1,5 +1,9 @@
<template>
<view :class="[getClass, 'dict-tag']" v-if="current.label">
<view
:class="[getClass, 'dict-tag']"
v-if="current.label"
:style="{ color: color, backgroundColor: bgColor }"
>
{{ current.label }}
</view>
</template>
@ -25,6 +29,12 @@ export default {
return []
}
},
color: {
type: String
},
bgColor: {
type: String
},
value: {
type: Number,
required: true

2
main.js

@ -5,6 +5,7 @@ import plugins from './plugins' // plugins
import './permission' // permission
import * as dict from '@/utils/dict.js'
import * as util from '@/utils/ruoyi.js'
import * as roles from '@/utils/permission.js'
// 引入uView
import uView from '@/uni_modules/uview-ui'
// import mpShare from '@/uni_modules/uview-ui/libs/mixin/mpShare.js'
@ -21,6 +22,7 @@ Vue.config.productionTip = false
Vue.prototype.$store = store
Vue.prototype.$dict = dict
Vue.prototype.$util = util
Vue.prototype.$roles = roles
App.mpType = 'app'

16
pages.json

@ -93,7 +93,7 @@
}, {
"path": "task/detail",
"style": {
"navigationBarTitleText": "任务详情",
"navigationBarTitleText": "执法记录",
"onReachBottonDistance": 50,
"enablePullDownRefresh": true,
"usingComponents": {
@ -109,7 +109,7 @@
{
"path": "task/record",
"style": {
"navigationBarTitleText": "记录详情",
"navigationBarTitleText": "记录详情 ",
"usingComponents": {},
"componentPlaceholder": {}
}
@ -124,6 +124,18 @@
"u-modal": "view"
}
}
},
{
"path": "task/enforce",
"style": {
"navigationBarTitleText": "结果反馈",
"usingComponents": {
"van-action-sheet": "/wxcomponents/vant/action-sheet/index"
},
"componentPlaceholder": {
"u-modal": "view"
}
}
}
]
}],

7
pages/index.vue

@ -4,7 +4,12 @@
<script>
export default {
onLoad: function () {}
onLoad: async function () {
await uni.hideTabBar()
uni.hideTabBar({
animation: false
})
}
}
</script>

134
sub/task/detail.vue

@ -87,21 +87,24 @@
style="flex: 1"
>
<van-dropdown-item
:value="queryParams.dept"
v-if="showItem(['director'])"
:value="queryParams.departmentId"
:options="getDropdownOption('dept')"
@change="
v => {
querySelect(v, 'dept')
querySelect(v, 'departmentId')
}
"
/>
<van-dropdown-item
v-if=""
:value="queryParams.dept"
v-if="showItem(['queue', 'director'])"
:value="queryParams.userId"
use-before-toggle
@before-toggle="getUserOption"
:options="userList"
@change="
v => {
querySelect(v, 'dept')
querySelect(v, 'userId')
}
"
/>
@ -180,16 +183,22 @@
<view class="staff-list">
<view
class="staff"
v-for="(staff, index) in 10"
:key="staff.id"
v-for="staff in model.list"
:key="staff.userId"
:style="{
'--select-color': index == 0 ? '#17c653' : '#f1f1f4'
'--select-color':
model.isSelect == staff.userId ? '#17c653' : '#f1f1f4'
}"
@click="selectUser(staff.userId)"
>
<u-avatar size="40" shape="circle"></u-avatar>
<u-avatar
size="40"
shape="circle"
:src="staff.avatar"
></u-avatar>
<view class="info">
<view class="name">姓名</view>
<view class="dept">部门</view>
<view class="name">{{ staff.realName }}</view>
<view class="dept">{{ staff.roleName[0] }}</view>
</view>
<view class="isSelect">
<u-icon
@ -203,7 +212,7 @@
</view>
<template #confirmButton>
<view class="wd-flex wd-flex-center">
<view class="confirm-btn">转移给TA</view>
<view class="confirm-btn" @tap="submitReplay">转移给TA</view>
</view>
</template>
</u-modal>
@ -237,23 +246,29 @@ export default {
},
//
queryParams: {
dept: '',
departmentId: '',
Inspections_status: '',
pageSize: 8,
pageNo: 1,
taskId: ''
userId: ''
},
load: 'loadmore',
model: {
show: false
show: false,
list: [],
isSelect: ''
},
userList: []
userList: [
{
value: '',
text: '按人员'
}
]
}
},
onLoad(res) {
this.queryParams.taskId = res.id
this.detail.id = res.id
this.getDict()
this.init()
},
onPageScroll(e) {
this.OptionsOffset.isTop = e.scrollTop + 10 > this.OptionsOffset.top
@ -267,6 +282,10 @@ export default {
onPullDownRefresh() {
this.reset()
},
onShow() {
this.init()
this.reset()
},
methods: {
getOptionOffset() {
const query = uni.createSelectorQuery().in(this)
@ -277,6 +296,9 @@ export default {
})
.exec()
},
showItem(query) {
return this.$roles.checkRole(query)
},
/**
* 获取字典
*/
@ -325,25 +347,34 @@ export default {
]
},
querySelect(v, key) {
if (key == 'departmentId') {
this.queryParams.userId = ''
}
this.queryParams[key] = v.detail
this.queryList()
},
async init() {
const res = await TaskApi.getDetail(this.queryParams.taskId)
const res = await TaskApi.getDetail(this.detail.id)
this.detail = res.data
this.getList()
},
async getUser(deptId) {
const res = await getUserList(deptId)
console.log(res)
async getUser() {
const res = await getUserList(this.queryParams.departmentId)
return res.data
},
async getList() {
if (!this.$roles.checkRole(['director'])) {
this.queryParams.departmentId = this.$store.getters.deptId
}
uni.showToast({
title: '加载中',
mask: true,
icon: 'loading'
})
this.load = 'loading'
const { data } = await InspectionsApi.getList(this.queryParams)
const { data } = await InspectionsApi.getList({
...this.queryParams,
taskId: this.detail.id
})
this.list.push(...data.list)
this.load = 'loadmore'
if (this.list.length == data.total) {
@ -351,6 +382,24 @@ export default {
}
uni.hideToast()
},
getUserOption({ detail }) {
this.getUser().then(res => {
this.userList = [
...res.map(i => {
return {
...i,
value: i.userId,
text: i.realName
}
}),
{
value: '',
text: '按人员'
}
]
detail.callback(true)
})
},
goBack() {
uni.switchTab({
url: '/pages/task'
@ -376,11 +425,11 @@ export default {
},
async reset() {
this.queryParams = {
dept: '',
departmentId: '',
Inspections_status: '',
pageSize: 8,
pageNo: 1,
taskId: ''
userId: ''
}
await this.queryList()
},
@ -391,15 +440,42 @@ export default {
this.getList()
},
replay() {
this.model.show = true
this.getUser().then(res => {
this.model.list = res
this.model.show = true
})
},
closeModel() {
this.model.show = false
this.model = {
show: false,
list: [],
isSelect: ''
}
},
selectUser(id) {
if (this.model.isSelect == id) {
this.model.isSelect = null
} else {
this.model.isSelect = id
}
},
goRecord(record) {
uni.navigateTo({
url: `/sub/task/record?taskId=${record.taskId}&recordId=${record.id}&enterpriseId=${record.enterpriseId}`
})
},
submitReplay() {
InspectionsApi.replay({
inspectionsId: this.isSelect,
userId: this.model.isSelect
}).then(res => {
uni.showToast({
icon: 'none',
title: '任务转移成功'
})
this.closeModel()
this.reset()
})
}
}
}
@ -614,4 +690,4 @@ export default {
margin: 12px 0;
border-radius: 0px;
}
</style>
</style>

84
sub/task/enforce.vue

@ -0,0 +1,84 @@
<template>
<view class="view-container">
<view class="box">
<view
class="wd-flex"
style="justify-content: space-between; align-items: center"
>
<view style="font-weight: bold">
<text style="color: #f8285a; margin-right: 4px">*</text>
处理意见
</view>
<text style="color: #78829d">最多500字符</text>
</view>
<uni-easyinput
type="textarea"
v-model="form.advice"
placeholder="请输入内容"
></uni-easyinput>
</view>
<cs-bottom-wrapper>
<view class="operation">
<view :class="['btn', form.state == 2 ? 'green' : 'red']">
{{ form.state == 2 ? '审批通过' : '整改处理' }}
</view>
</view>
</cs-bottom-wrapper>
</view>
</template>
<script>
export default {
data() {
return {
form: {
state: undefined,
inspectionsId: undefined,
advice: undefined
}
}
},
onLoad(res) {
this.form.state = res.state
this.inspectionsId = res.inspectionsId
}
}
</script>
<style lang="scss" scoped>
.view-container {
padding: 12px;
height: 100vh;
position: relative;
.box {
padding: 12px;
border: 1px solid #f9f9f9;
border-radius: 12px;
background-color: #fff;
}
.operation {
padding: 12px;
display: flex;
align-items: center;
justify-content: center;
gap: 12px;
.btn {
flex: 1;
border-radius: 8px;
display: flex;
padding: 12px 0;
align-items: center;
justify-content: center;
}
.green {
background-color: $cs-color-main;
color: #fff;
}
.red {
background-color: #f8285a;
color: #fff;
}
}
}
</style>

77
sub/task/locate.vue

@ -73,8 +73,8 @@
</view>
</view>
<cs-bottom-wrapper v-if="showBtn">
<view class="operation">
<cs-bottom-wrapper v-if="!isLocate">
<view class="operation" v-if="">
<button
class="btn green"
v-if="distance < radius"
@ -87,6 +87,14 @@
</view>
</cs-bottom-wrapper>
<cs-bottom-wrapper v-if="isLocate && isLeadLocate">
<view class="operation">
<view class="btn grey" v-if="list.length < 2">开始执法</view>
<button class="btn green" v-else :loading="loading" @tap="goNext">
开始执法
</button>
</view>
</cs-bottom-wrapper>
<u-modal
:show="showTip"
width="295px"
@ -155,15 +163,11 @@ export default {
title: '',
path: '',
imageUrl: ''
}
}
},
computed: {
showBtn() {
return (
this.list.findIndex(i => i.userId == this.$store.getters.userId) <
-1
)
},
isLeadLocate: false,
isLocate: false,
timer: null,
time: 0
}
},
onLoad(res) {
@ -196,6 +200,17 @@ export default {
onShareTimeline() {
return this.share
},
onShow() {
this.refresh()
this.echoQueset()
},
onHide() {
this.timer && clearInterval(this.timer)
},
onBackPress() {
this.timer && clearInterval(this.timer)
return false
},
methods: {
async init() {
const miniCode = await getMiniCode({
@ -205,6 +220,7 @@ export default {
env_version: 'develop',
is_hyaline: true
})
this.miniCode = miniCode
const enterprise = await getEnterPrise(this.enterpriseId)
this.enterprisesName = enterprise.data.enterprisesName
this.enterpriseGps = enterprise.data.gpsLocation
@ -213,15 +229,19 @@ export default {
})
this.share.title = `执法签到-${this.enterprisesName}`
this.getLocateList()
this.miniCode = miniCode
this.echoQueset()
},
async getLocateList() {
const res = await InspectionsApi.getLocate(this.inspectionsId)
//
const isLead = res.data.findIndex(i => {
i.userId == this.$store.getters.userId && i.isInspect
const locate = res.data.find(i => {
return i.userId == this.$store.getters.userId
})
console.log(isLead)
if (locate) {
//
this.isLocate = true
//
if (locate.isInspect) this.isLeadLocate = true
}
this.list = res.data
},
useAuth() {
@ -295,6 +315,7 @@ export default {
refresh() {
this.getDistance()
this.getLocateList()
this.echoQueset()
},
async locate() {
this.loading = true
@ -307,6 +328,30 @@ export default {
title: res.data
})
this.refresh()
},
echoQueset() {
const timeout = 3 * 60 * 1000
this.timer && clearInterval(this.timer)
// this.timer = setInterval(() => {
// this.getLocateList()
// this.time += 5000
// if (this.time > timeout) {
// console.log('timeout')
// clearInterval(this.timer)
// uni.navigateBack({
// delta: 1
// })
// }
// }, 5 * 1000)
},
goNext() {
InspectionsApi.beginEnforce({
inspectionsId: this.inspectionsId
}).then(res => {
uni.navigateBack({
delta: 1
})
})
}
}
}

102
sub/task/record.vue

@ -103,7 +103,43 @@
<view class="box record">
<view class="wd-text-14 wd-font-800 wd-pb-12">结果反馈</view>
<view class="emty">
<view class="row" v-for="item in list" :key="item.id">
<view style="margin-bottom: 12px">
<text
:style="{
marginRight: '10px',
fontWeight: 'bold',
color: setColor(item.status)
}"
>
{{
$dict.echoDicValue(dictMap.Inspections_status, item.status)
}}
</text>
<text>
{{ $util.formatDate(item.createTime, 'YYYY/M/D h:m') }}
</text>
</view>
<scroll-view scroll-x="true" enable-flex="true">
<view
class="row"
v-for="people in item.userList"
:key="people.userId"
style="display: inline-flex; gap: 12px; margin-right: 12px"
>
<u-avatar
:src="people.avtar"
size="40px"
shape="circle"
></u-avatar>
<view class="wd-flex wd-flex-col" style="gap: 2px">
<text>{{ people.realName }}</text>
<text>{{ people.deptName }}</text>
</view>
</view>
</scroll-view>
</view>
<view class="emty" v-if="list.length == 0">
<image
class="image"
src="/static/images/emty.png"
@ -111,11 +147,26 @@
></image>
<text class="wd-text-14" style="color: #99a1b7">暂无处理结果</text>
</view>
<view class="audit">
<cs-dict-tag
:dict="dictMap.Inspections_status"
:value="inspectionsState"
color="#fff"
:bgColor="setColor(inspectionsState)"
></cs-dict-tag>
</view>
</view>
<cs-bottom-wrapper>
<view class="operation">
<view class="btn" @tap="locate">执法签到</view>
<view class="" v-if="[null, 3].includes(inspectionsState)">
<view class="btn green" @tap="locate">执法签到</view>
</view>
<view class="operation" v-else-if="inspectionsState == 1">
<view class="btn red" @tap="tackle(3)">整改处理</view>
<view class="btn green" @tap="tackle(2)">审批通过</view>
</view>
<view class="operation" v-else>
<view class="btn grey">任务完成</view>
</view>
</cs-bottom-wrapper>
</view>
@ -143,7 +194,8 @@ export default {
recordId: '',
enterpriseId: ''
},
list: []
list: [],
inspectionsState: null
}
},
onLoad(res) {
@ -153,13 +205,16 @@ export default {
this.getDict()
this.init()
},
onShow() {
this.init()
},
methods: {
/**
* 获取字典
*/
async getDict() {
const dict = await getDictBatchByType({
type: ['Inspections_status'].join(',')
type: ['Inspections_status', 'task_state'].join(',')
})
const dept = await getDeptTree()
this.dictMap = {
@ -177,9 +232,10 @@ export default {
const feedBack = await InspectionsApi.getFeedBack(
this.queryParams.recordId
)
this.list = feedBack.data.list
this.inspectionsState = this.list[0].status
},
goEnterprise(enterprise) {
uni.navigateTo({
url: `/sub/enterprise/detail?id=${enterprise.id}`
@ -199,6 +255,15 @@ export default {
uni.navigateTo({
url: `/sub/task/locate?inspectionsId=${this.queryParams.recordId}&enterpriseId=${this.queryParams.enterpriseId}`
})
},
setColor(status) {
if (status == 1) return '#F6B100'
if (status == 3) return '#F8285A'
},
tackle(state) {
uni.navigateTo({
url: `/sub/task/enforce?state=${state}&inspectionsId=${this.queryParams.recordId}`
})
}
}
}
@ -217,7 +282,11 @@ export default {
border-radius: $cs-border-radius;
padding: 16px;
}
.row {
border-radius: $cs-border-radius;
padding: 12px;
border: 1px solid #f9f9f9;
}
.detail {
display: flex;
flex-flow: column nowrap;
@ -296,6 +365,8 @@ export default {
}
.record {
position: relative;
overflow: hidden;
.emty {
display: flex;
flex-direction: column;
@ -310,6 +381,13 @@ export default {
height: 64px;
}
}
.audit {
position: absolute;
right: 0;
top: 0;
transform: translateX(30px) translateY(8px) rotateZ(45deg);
transform-origin: 50% 50%;
}
}
.operation {
@ -325,9 +403,19 @@ export default {
padding: 12px 0;
align-items: center;
justify-content: center;
}
.green {
background-color: $cs-color-main;
color: #fff;
}
.red {
background-color: #f8285a;
color: #fff;
}
.grey {
background-color: #f1f1f4;
color: #99a1b7;
}
}
}
</style>

115
uni_modules/uni-easyinput/changelog.md

@ -0,0 +1,115 @@
## 1.1.19(2024-07-18)
- 修复 初始值传入 null 导致input报错的bug
## 1.1.18(2024-04-11)
- 修复 easyinput组件双向绑定问题
## 1.1.17(2024-03-28)
- 修复 在头条小程序下丢失事件绑定的问题
## 1.1.16(2024-03-20)
- 修复 在密码输入情况下 清除和小眼睛覆盖bug 在edge浏览器下显示双眼睛bug
## 1.1.15(2024-02-21)
- 新增 左侧插槽:left
## 1.1.14(2024-02-19)
- 修复 onBlur的emit传值错误
## 1.1.12(2024-01-29)
- 补充 adjust-position文档属性补充
## 1.1.11(2024-01-29)
- 补充 adjust-position属性传递值:(Boolean)当键盘弹起时,是否自动上推页面
## 1.1.10(2024-01-22)
- 去除 移除无用的log输出
## 1.1.9(2023-04-11)
- 修复 vue3 下 keyboardheightchange 事件报错的bug
## 1.1.8(2023-03-29)
- 优化 trim 属性默认值
## 1.1.7(2023-03-29)
- 新增 cursor-spacing 属性
## 1.1.6(2023-01-28)
- 新增 keyboardheightchange 事件,可监听键盘高度变化
## 1.1.5(2022-11-29)
- 优化 主题样式
## 1.1.4(2022-10-27)
- 修复 props 中背景颜色无默认值的bug
## 1.1.0(2022-06-30)
- 新增 在 uni-forms 1.4.0 中使用可以在 blur 时校验内容
- 新增 clear 事件,点击右侧叉号图标触发
- 新增 change 事件 ,仅在输入框失去焦点或用户按下回车时触发
- 优化 组件样式,组件获取焦点时高亮显示,图标颜色调整等
## 1.0.5(2022-06-07)
- 优化 clearable 显示策略
## 1.0.4(2022-06-07)
- 优化 clearable 显示策略
## 1.0.3(2022-05-20)
- 修复 关闭图标某些情况下无法取消的 bug
## 1.0.2(2022-04-12)
- 修复 默认值不生效的 bug
## 1.0.1(2022-04-02)
- 修复 value 不能为 0 的 bug
## 1.0.0(2021-11-19)
- 优化 组件 UI,并提供设计资源,详见:[https://uniapp.dcloud.io/component/uniui/resource](https://uniapp.dcloud.io/component/uniui/resource)
- 文档迁移,详见:[https://uniapp.dcloud.io/component/uniui/uni-easyinput](https://uniapp.dcloud.io/component/uniui/uni-easyinput)
## 0.1.4(2021-08-20)
- 修复 在 uni-forms 的动态表单中默认值校验不通过的 bug
## 0.1.3(2021-08-11)
- 修复 在 uni-forms 中重置表单,错误信息无法清除的问题
## 0.1.2(2021-07-30)
- 优化 vue3 下事件警告的问题
## 0.1.1
- 优化 errorMessage 属性支持 Boolean 类型
## 0.1.0(2021-07-13)
- 组件兼容 vue3,如何创建 vue3 项目,详见 [uni-app 项目支持 vue3 介绍](https://ask.dcloud.net.cn/article/37834)
## 0.0.16(2021-06-29)
- 修复 confirmType 属性(仅 type="text" 生效)导致多行文本框无法换行的 bug
## 0.0.15(2021-06-21)
- 修复 passwordIcon 属性拼写错误的 bug
## 0.0.14(2021-06-18)
- 新增 passwordIcon 属性,当 type=password 时是否显示小眼睛图标
- 修复 confirmType 属性不生效的问题
## 0.0.13(2021-06-04)
- 修复 disabled 状态可清出内容的 bug
## 0.0.12(2021-05-12)
- 新增 组件示例地址
## 0.0.11(2021-05-07)
- 修复 input-border 属性不生效的问题
## 0.0.10(2021-04-30)
- 修复 ios 遮挡文字、显示一半的问题
## 0.0.9(2021-02-05)
- 调整为 uni_modules 目录规范
- 优化 兼容 nvue 页面

54
uni_modules/uni-easyinput/components/uni-easyinput/common.js

@ -0,0 +1,54 @@
/**
* @desc 函数防抖
* @param func 目标函数
* @param wait 延迟执行毫秒数
* @param immediate true - 立即执行 false - 延迟执行
*/
export const debounce = function(func, wait = 1000, immediate = true) {
let timer;
return function() {
let context = this,
args = arguments;
if (timer) clearTimeout(timer);
if (immediate) {
let callNow = !timer;
timer = setTimeout(() => {
timer = null;
}, wait);
if (callNow) func.apply(context, args);
} else {
timer = setTimeout(() => {
func.apply(context, args);
}, wait)
}
}
}
/**
* @desc 函数节流
* @param func 函数
* @param wait 延迟执行毫秒数
* @param type 1 使用表时间戳在时间段开始的时候触发 2 使用表定时器在时间段结束的时候触发
*/
export const throttle = (func, wait = 1000, type = 1) => {
let previous = 0;
let timeout;
return function() {
let context = this;
let args = arguments;
if (type === 1) {
let now = Date.now();
if (now - previous > wait) {
func.apply(context, args);
previous = now;
}
} else if (type === 2) {
if (!timeout) {
timeout = setTimeout(() => {
timeout = null;
func.apply(context, args)
}, wait)
}
}
}
}

676
uni_modules/uni-easyinput/components/uni-easyinput/uni-easyinput.vue

@ -0,0 +1,676 @@
<template>
<view class="uni-easyinput" :class="{ 'uni-easyinput-error': msg }" :style="boxStyle">
<view class="uni-easyinput__content" :class="inputContentClass" :style="inputContentStyle">
<uni-icons v-if="prefixIcon" class="content-clear-icon" :type="prefixIcon" color="#c0c4cc" @click="onClickIcon('prefix')" size="22"></uni-icons>
<slot name="left">
</slot>
<!-- #ifdef MP-ALIPAY -->
<textarea :enableNative="enableNative" v-if="type === 'textarea'" class="uni-easyinput__content-textarea" :class="{ 'input-padding': inputBorder }" :name="name" :value="val" :placeholder="placeholder" :placeholderStyle="placeholderStyle" :disabled="disabled" placeholder-class="uni-easyinput__placeholder-class" :maxlength="inputMaxlength" :focus="focused" :autoHeight="autoHeight" :cursor-spacing="cursorSpacing" :adjust-position="adjustPosition" @input="onInput" @blur="_Blur" @focus="_Focus" @confirm="onConfirm" @keyboardheightchange="onkeyboardheightchange"></textarea>
<input :enableNative="enableNative" v-else :type="type === 'password' ? 'text' : type" class="uni-easyinput__content-input" :style="inputStyle" :name="name" :value="val" :password="!showPassword && type === 'password'" :placeholder="placeholder" :placeholderStyle="placeholderStyle" placeholder-class="uni-easyinput__placeholder-class" :disabled="disabled" :maxlength="inputMaxlength" :focus="focused" :confirmType="confirmType" :cursor-spacing="cursorSpacing" :adjust-position="adjustPosition" @focus="_Focus" @blur="_Blur" @input="onInput" @confirm="onConfirm" @keyboardheightchange="onkeyboardheightchange" />
<!-- #endif -->
<!-- #ifndef MP-ALIPAY -->
<textarea v-if="type === 'textarea'" class="uni-easyinput__content-textarea" :class="{ 'input-padding': inputBorder }" :name="name" :value="val" :placeholder="placeholder" :placeholderStyle="placeholderStyle" :disabled="disabled" placeholder-class="uni-easyinput__placeholder-class" :maxlength="inputMaxlength" :focus="focused" :autoHeight="autoHeight" :cursor-spacing="cursorSpacing" :adjust-position="adjustPosition" @input="onInput" @blur="_Blur" @focus="_Focus" @confirm="onConfirm" @keyboardheightchange="onkeyboardheightchange"></textarea>
<input v-else :type="type === 'password' ? 'text' : type" class="uni-easyinput__content-input" :style="inputStyle" :name="name" :value="val" :password="!showPassword && type === 'password'" :placeholder="placeholder" :placeholderStyle="placeholderStyle" placeholder-class="uni-easyinput__placeholder-class" :disabled="disabled" :maxlength="inputMaxlength" :focus="focused" :confirmType="confirmType" :cursor-spacing="cursorSpacing" :adjust-position="adjustPosition" @focus="_Focus" @blur="_Blur" @input="onInput" @confirm="onConfirm" @keyboardheightchange="onkeyboardheightchange" />
<!-- #endif -->
<template v-if="type === 'password' && passwordIcon">
<!-- 开启密码时显示小眼睛 -->
<uni-icons v-if="isVal" class="content-clear-icon" :class="{ 'is-textarea-icon': type === 'textarea' }" :type="showPassword ? 'eye-slash-filled' : 'eye-filled'" :size="22" :color="focusShow ? primaryColor : '#c0c4cc'" @click="onEyes"></uni-icons>
</template>
<template v-if="suffixIcon">
<uni-icons v-if="suffixIcon" class="content-clear-icon" :type="suffixIcon" color="#c0c4cc" @click="onClickIcon('suffix')" size="22"></uni-icons>
</template>
<template v-else>
<uni-icons v-if="clearable && isVal && !disabled && type !== 'textarea'" class="content-clear-icon" :class="{ 'is-textarea-icon': type === 'textarea' }" type="clear" :size="clearSize" :color="msg ? '#dd524d' : focusShow ? primaryColor : '#c0c4cc'" @click="onClear"></uni-icons>
</template>
<slot name="right"></slot>
</view>
</view>
</template>
<script>
/**
* Easyinput 输入框
* @description 此组件可以实现表单的输入与校验包括 "text" "textarea" 类型
* @tutorial https://ext.dcloud.net.cn/plugin?id=3455
* @property {String} value 输入内容
* @property {String } type 输入框的类型默认text password/text/textarea/..
* @value text 文本输入键盘
* @value textarea 多行文本输入键盘
* @value password 密码输入键盘
* @value number 数字输入键盘注意iOS上app-vue弹出的数字键盘并非9宫格方式
* @value idcard 身份证输入键盘支付宝百度QQ小程序
* @value digit 带小数点的数字键盘 App的nvue页面微信支付宝百度头条QQ小程序支持
* @property {Boolean} clearable 是否显示右侧清空内容的图标控件点击可清空输入框内容默认true
* @property {Boolean} autoHeight 是否自动增高输入区域type为textarea时有效默认true
* @property {String } placeholder 输入框的提示文字
* @property {String } placeholderStyle placeholder的样式(内联样式字符串)"color: #ddd"
* @property {Boolean} focus 是否自动获得焦点默认false
* @property {Boolean} disabled 是否禁用默认false
* @property {Number } maxlength 最大输入长度设置为 -1 的时候不限制最大长度默认140
* @property {String } confirmType 设置键盘右下角按钮的文字仅在type="text"时生效默认done
* @property {Number } clearSize 清除图标的大小单位px默认15
* @property {String} prefixIcon 输入框头部图标
* @property {String} suffixIcon 输入框尾部图标
* @property {String} primaryColor 设置主题色默认#2979ff
* @property {Boolean} trim 是否自动去除两端的空格
* @property {Boolean} cursorSpacing 指定光标与键盘的距离单位 px
* @property {Boolean} ajust-position 当键盘弹起时是否上推内容默认值true
* @value both 去除两端空格
* @value left 去除左侧空格
* @value right 去除右侧空格
* @value start 去除左侧空格
* @value end 去除右侧空格
* @value all 去除全部空格
* @value none 不去除空格
* @property {Boolean} inputBorder 是否显示input输入框的边框默认true
* @property {Boolean} passwordIcon type=password时是否显示小眼睛图标
* @property {Object} styles 自定义颜色
* @event {Function} input 输入框内容发生变化时触发
* @event {Function} focus 输入框获得焦点时触发
* @event {Function} blur 输入框失去焦点时触发
* @event {Function} confirm 点击完成按钮时触发
* @event {Function} iconClick 点击图标时触发
* @example <uni-easyinput v-model="mobile"></uni-easyinput>
*/
function obj2strClass(obj) {
let classess = '';
for (let key in obj) {
const val = obj[key];
if (val) {
classess += `${key} `;
}
}
return classess;
}
function obj2strStyle(obj) {
let style = '';
for (let key in obj) {
const val = obj[key];
style += `${key}:${val};`;
}
return style;
}
export default {
name: 'uni-easyinput',
emits: [
'click',
'iconClick',
'update:modelValue',
'input',
'focus',
'blur',
'confirm',
'clear',
'eyes',
'change',
'keyboardheightchange'
],
model: {
prop: 'modelValue',
event: 'update:modelValue'
},
options: {
// #ifdef MP-TOUTIAO
virtualHost: false,
// #endif
// #ifndef MP-TOUTIAO
virtualHost: true
// #endif
},
inject: {
form: {
from: 'uniForm',
default: null
},
formItem: {
from: 'uniFormItem',
default: null
}
},
props: {
name: String,
value: [Number, String],
modelValue: [Number, String],
type: {
type: String,
default: 'text'
},
clearable: {
type: Boolean,
default: true
},
autoHeight: {
type: Boolean,
default: false
},
placeholder: {
type: String,
default: ' '
},
placeholderStyle: String,
focus: {
type: Boolean,
default: false
},
disabled: {
type: Boolean,
default: false
},
maxlength: {
type: [Number, String],
default: 140
},
confirmType: {
type: String,
default: 'done'
},
clearSize: {
type: [Number, String],
default: 24
},
inputBorder: {
type: Boolean,
default: true
},
prefixIcon: {
type: String,
default: ''
},
suffixIcon: {
type: String,
default: ''
},
trim: {
type: [Boolean, String],
default: false
},
cursorSpacing: {
type: Number,
default: 0
},
passwordIcon: {
type: Boolean,
default: true
},
adjustPosition: {
type: Boolean,
default: true
},
primaryColor: {
type: String,
default: '#2979ff'
},
styles: {
type: Object,
default () {
return {
color: '#333',
backgroundColor: '#fff',
disableColor: '#F7F6F6',
borderColor: '#e5e5e5'
};
}
},
errorMessage: {
type: [String, Boolean],
default: ''
},
// #ifdef MP-ALIPAY
enableNative: {
type: Boolean,
default: false
}
// #endif
},
data() {
return {
focused: false,
val: '',
showMsg: '',
border: false,
isFirstBorder: false,
showClearIcon: false,
showPassword: false,
focusShow: false,
localMsg: '',
isEnter: false // 使
};
},
computed: {
//
isVal() {
const val = this.val;
// fixed by mehaotian 00
if (val || val === 0) {
return true;
}
return false;
},
msg() {
// console.log('computed', this.form, this.formItem);
// if (this.form) {
// return this.errorMessage || this.formItem.errMsg;
// }
// TODO formItem errMsg
return this.localMsg || this.errorMessage;
},
// uniappinputmaxlength
inputMaxlength() {
return Number(this.maxlength);
},
// style
boxStyle() {
return `color:${
this.inputBorder && this.msg ? '#e43d33' : this.styles.color
};`;
},
// input
inputContentClass() {
return obj2strClass({
'is-input-border': this.inputBorder,
'is-input-error-border': this.inputBorder && this.msg,
'is-textarea': this.type === 'textarea',
'is-disabled': this.disabled,
'is-focused': this.focusShow
});
},
inputContentStyle() {
const focusColor = this.focusShow ?
this.primaryColor :
this.styles.borderColor;
const borderColor =
this.inputBorder && this.msg ? '#dd524d' : focusColor;
return obj2strStyle({
'border-color': borderColor || '#e5e5e5',
'background-color': this.disabled ?
this.styles.disableColor : this.styles.backgroundColor
});
},
// input
inputStyle() {
const paddingRight =
this.type === 'password' || this.clearable || this.prefixIcon ?
'' :
'10px';
return obj2strStyle({
'padding-right': paddingRight,
'padding-left': this.prefixIcon ? '' : '10px'
});
}
},
watch: {
value(newVal) {
// fix by mehaotian nullinputbug
if (newVal === null) {
this.val = '';
return
}
this.val = newVal;
},
modelValue(newVal) {
if (newVal === null) {
this.val = '';
return
}
this.val = newVal;
},
focus(newVal) {
this.$nextTick(() => {
this.focused = this.focus;
this.focusShow = this.focus;
});
}
},
created() {
this.init();
// TODO vue3 computed inject formItem.errMsg
if (this.form && this.formItem) {
this.$watch('formItem.errMsg', newVal => {
this.localMsg = newVal;
});
}
},
mounted() {
this.$nextTick(() => {
this.focused = this.focus;
this.focusShow = this.focus;
});
},
methods: {
/**
* 初始化变量值
*/
init() {
if (this.value || this.value === 0) {
this.val = this.value;
} else if (
this.modelValue ||
this.modelValue === 0 ||
this.modelValue === ''
) {
this.val = this.modelValue;
} else {
// fix by ht nullinput
this.val = '';
}
},
/**
* 点击图标时触发
* @param {Object} type
*/
onClickIcon(type) {
this.$emit('iconClick', type);
},
/**
* 显示隐藏内容密码框时生效
*/
onEyes() {
this.showPassword = !this.showPassword;
this.$emit('eyes', this.showPassword);
},
/**
* 输入时触发
* @param {Object} event
*/
onInput(event) {
let value = event.detail.value;
//
if (this.trim) {
if (typeof this.trim === 'boolean' && this.trim) {
value = this.trimStr(value);
}
if (typeof this.trim === 'string') {
value = this.trimStr(value, this.trim);
}
}
if (this.errMsg) this.errMsg = '';
this.val = value;
// TODO vue2
this.$emit('input', value);
// TODO  vue3
this.$emit('update:modelValue', value);
},
/**
* 外部调用方法
* 获取焦点时触发
* @param {Object} event
*/
onFocus() {
this.$nextTick(() => {
this.focused = true;
});
this.$emit('focus', null);
},
_Focus(event) {
this.focusShow = true;
this.$emit('focus', event);
},
/**
* 外部调用方法
* 失去焦点时触发
* @param {Object} event
*/
onBlur() {
this.focused = false;
this.$emit('blur', null);
},
_Blur(event) {
let value = event.detail.value;
this.focusShow = false;
this.$emit('blur', event);
// eventstring
if (this.isEnter === false) {
this.$emit('change', this.val);
}
//
if (this.form && this.formItem) {
const { validateTrigger } = this.form;
if (validateTrigger === 'blur') {
this.formItem.onFieldChange();
}
}
},
/**
* 按下键盘的发送键
* @param {Object} e
*/
onConfirm(e) {
this.$emit('confirm', this.val);
this.isEnter = true;
this.$emit('change', this.val);
this.$nextTick(() => {
this.isEnter = false;
});
},
/**
* 清理内容
* @param {Object} event
*/
onClear(event) {
this.val = '';
// TODO vue2
this.$emit('input', '');
// TODO vue2
// TODO  vue3
this.$emit('update:modelValue', '');
//
this.$emit('clear');
},
/**
* 键盘高度发生变化的时候触发此事件
* 兼容性微信小程序2.7.0+App 3.1.0+
* @param {Object} event
*/
onkeyboardheightchange(event) {
this.$emit('keyboardheightchange', event);
},
/**
* 去除空格
*/
trimStr(str, pos = 'both') {
if (pos === 'both') {
return str.trim();
} else if (pos === 'left') {
return str.trimLeft();
} else if (pos === 'right') {
return str.trimRight();
} else if (pos === 'start') {
return str.trimStart();
} else if (pos === 'end') {
return str.trimEnd();
} else if (pos === 'all') {
return str.replace(/\s+/g, '');
} else if (pos === 'none') {
return str;
}
return str;
}
}
};
</script>
<style lang="scss">
$uni-error: #e43d33;
$uni-border-1: #dcdfe6 !default;
.uni-easyinput {
/* #ifndef APP-NVUE */
width: 100%;
/* #endif */
flex: 1;
position: relative;
text-align: left;
color: #333;
font-size: 14px;
}
.uni-easyinput__content {
flex: 1;
/* #ifndef APP-NVUE */
width: 100%;
display: flex;
box-sizing: border-box;
// min-height: 36px;
/* #endif */
flex-direction: row;
align-items: center;
// border
border-color: #fff;
transition-property: border-color;
transition-duration: 0.3s;
}
.uni-easyinput__content-input {
/* #ifndef APP-NVUE */
width: auto;
/* #endif */
position: relative;
overflow: hidden;
flex: 1;
line-height: 1;
font-size: 14px;
height: 35px;
// min-height: 36px;
/*ifdef H5*/
& ::-ms-reveal {
display: none;
}
& ::-ms-clear {
display: none;
}
& ::-o-clear {
display: none;
}
/*endif*/
}
.uni-easyinput__placeholder-class {
color: #999;
font-size: 12px;
// font-weight: 200;
}
.is-textarea {
align-items: flex-start;
}
.is-textarea-icon {
margin-top: 5px;
}
.uni-easyinput__content-textarea {
position: relative;
overflow: hidden;
flex: 1;
line-height: 1.5;
font-size: 14px;
margin: 6px;
margin-left: 0;
height: 80px;
min-height: 80px;
/* #ifndef APP-NVUE */
min-height: 80px;
width: auto;
/* #endif */
}
.input-padding {
padding-left: 10px;
}
.content-clear-icon {
padding: 0 5px;
}
.label-icon {
margin-right: 5px;
margin-top: -1px;
}
//
.is-input-border {
/* #ifndef APP-NVUE */
display: flex;
box-sizing: border-box;
/* #endif */
flex-direction: row;
align-items: center;
border: 1px solid $uni-border-1;
border-radius: 4px;
/* #ifdef MP-ALIPAY */
overflow: hidden;
/* #endif */
}
.uni-error-message {
position: absolute;
bottom: -17px;
left: 0;
line-height: 12px;
color: $uni-error;
font-size: 12px;
text-align: left;
}
.uni-error-msg--boeder {
position: relative;
bottom: 0;
line-height: 22px;
}
.is-input-error-border {
border-color: $uni-error;
.uni-easyinput__placeholder-class {
color: mix(#fff, $uni-error, 50%);
}
}
.uni-easyinput--border {
margin-bottom: 0;
padding: 10px 15px;
// padding-bottom: 0;
border-top: 1px #eee solid;
}
.uni-easyinput-error {
padding-bottom: 0;
}
.is-first-border {
/* #ifndef APP-NVUE */
border: none;
/* #endif */
/* #ifdef APP-NVUE */
border-width: 0;
/* #endif */
}
.is-disabled {
background-color: #f7f6f6;
color: #d5d5d5;
.uni-easyinput__placeholder-class {
color: #d5d5d5;
font-size: 12px;
}
}
</style>

88
uni_modules/uni-easyinput/package.json

@ -0,0 +1,88 @@
{
"id": "uni-easyinput",
"displayName": "uni-easyinput 增强输入框",
"version": "1.1.19",
"description": "Easyinput 组件是对原生input组件的增强",
"keywords": [
"uni-ui",
"uniui",
"input",
"uni-easyinput",
"输入框"
],
"repository": "https://github.com/dcloudio/uni-ui",
"engines": {
"HBuilderX": ""
},
"directories": {
"example": "../../temps/example_temps"
},
"dcloudext": {
"sale": {
"regular": {
"price": "0.00"
},
"sourcecode": {
"price": "0.00"
}
},
"contact": {
"qq": ""
},
"declaration": {
"ads": "无",
"data": "无",
"permissions": "无"
},
"npmurl": "https://www.npmjs.com/package/@dcloudio/uni-ui",
"type": "component-vue"
},
"uni_modules": {
"dependencies": [
"uni-scss",
"uni-icons"
],
"encrypt": [],
"platforms": {
"cloud": {
"tcb": "y",
"aliyun": "y",
"alipay": "n"
},
"client": {
"App": {
"app-vue": "y",
"app-nvue": "y"
},
"H5-mobile": {
"Safari": "y",
"Android Browser": "y",
"微信浏览器(Android)": "y",
"QQ浏览器(Android)": "y"
},
"H5-pc": {
"Chrome": "y",
"IE": "y",
"Edge": "y",
"Firefox": "y",
"Safari": "y"
},
"小程序": {
"微信": "y",
"阿里": "y",
"百度": "y",
"字节跳动": "y",
"QQ": "y"
},
"快应用": {
"华为": "u",
"联盟": "u"
},
"Vue": {
"vue2": "y",
"vue3": "y"
}
}
}
}
}

11
uni_modules/uni-easyinput/readme.md

@ -0,0 +1,11 @@
### Easyinput 增强输入框
> **组件名:uni-easyinput**
> 代码块: `uEasyinput`
easyinput 组件是对原生input组件的增强 ,是专门为配合表单组件[uni-forms](https://ext.dcloud.net.cn/plugin?id=2773)而设计的,easyinput 内置了边框,图标等,同时包含 input 所有功能
### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-easyinput)
#### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839

1
utils/ruoyi.js

@ -4,7 +4,6 @@
*/
import dayjs from '@/uni_modules/uview-ui/libs/util/dayjs.js'
// 日期格式化
export function parseTime(time, pattern) {
if (arguments.length === 0 || !time) {

6
wxcomponents/vant/dropdown-menu/index.wxss

@ -6,15 +6,15 @@
display: flex;
height: var(--dropdown-menu-height, 50px);
-webkit-user-select: none;
user-select: none
user-select: none;
justify-content: space-between;
}
.van-dropdown-menu__item {
align-items: center;
display: flex;
flex: 1;
justify-content: center;
min-width: 0
flex: 1;
}
.van-dropdown-menu__item:active {

Loading…
Cancel
Save