Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
J
jiedao-app-customer-new-master
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
lanbaoming
jiedao-app-customer-new-master
Commits
64e10593
Commit
64e10593
authored
Mar 15, 2025
by
knight
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
集运首页+列表+创建+编辑功能
parent
c68f7fc3
Changes
40
Hide whitespace changes
Inline
Side-by-side
Showing
40 changed files
with
5761 additions
and
19 deletions
+5761
-19
warehouse.js
src/api/ecw/warehouse.js
+10
-0
cons.js
src/api/order/cons.js
+17
-0
lang.js
src/common/lang.js
+5
-0
pages.json
src/pages.json
+6
-0
signInPicture.vue
src/pages/cons/components/signInPicture.vue
+1
-1
create.vue
src/pages/cons/create.vue
+574
-4
index.vue
src/pages/cons/index.vue
+6
-2
create_order.vue
src/pages/create_order/create_order.vue
+2
-0
cons_create.css
src/static/css/cons_create.css
+250
-0
add_white.png
src/static/img/add_white.png
+0
-0
cons.js
src/static/lang/zh/cons.js
+40
-7
changelog.md
src/uni_modules/uni-data-checkbox/changelog.md
+51
-0
clientdb.js
...ni-data-checkbox/components/uni-data-checkbox/clientdb.js
+316
-0
uni-data-checkbox.vue
...eckbox/components/uni-data-checkbox/uni-data-checkbox.vue
+853
-0
package.json
src/uni_modules/uni-data-checkbox/package.json
+87
-0
readme.md
src/uni_modules/uni-data-checkbox/readme.md
+18
-0
changelog.md
src/uni_modules/uni-data-picker/changelog.md
+79
-0
keypress.js
...es/uni-data-picker/components/uni-data-picker/keypress.js
+45
-0
uni-data-picker.uvue
...ta-picker/components/uni-data-picker/uni-data-picker.uvue
+381
-0
uni-data-picker.vue
...ata-picker/components/uni-data-picker/uni-data-picker.vue
+551
-0
uni-data-picker.js
...-picker/components/uni-data-pickerview/uni-data-picker.js
+622
-0
uni-data-picker.uts
...picker/components/uni-data-pickerview/uni-data-picker.uts
+692
-0
uni-data-pickerview.css
...er/components/uni-data-pickerview/uni-data-pickerview.css
+76
-0
uni-data-pickerview.uvue
...r/components/uni-data-pickerview/uni-data-pickerview.uvue
+69
-0
uni-data-pickerview.vue
...er/components/uni-data-pickerview/uni-data-pickerview.vue
+323
-0
package.json
src/uni_modules/uni-data-picker/package.json
+91
-0
readme.md
src/uni_modules/uni-data-picker/readme.md
+22
-0
changelog.md
src/uni_modules/uni-forms/changelog.md
+6
-0
uni-forms-item.vue
...es/uni-forms/components/uni-forms-item/uni-forms-item.vue
+5
-0
uni-forms.vue
src/uni_modules/uni-forms/components/uni-forms/uni-forms.vue
+8
-1
package.json
src/uni_modules/uni-forms/package.json
+3
-2
changelog.md
src/uni_modules/uni-load-more/changelog.md
+25
-0
en.json
...dules/uni-load-more/components/uni-load-more/i18n/en.json
+5
-0
index.js
...ules/uni-load-more/components/uni-load-more/i18n/index.js
+8
-0
zh-Hans.json
.../uni-load-more/components/uni-load-more/i18n/zh-Hans.json
+5
-0
zh-Hant.json
.../uni-load-more/components/uni-load-more/i18n/zh-Hant.json
+5
-0
uni-load-more.vue
.../uni-load-more/components/uni-load-more/uni-load-more.vue
+404
-0
package.json
src/uni_modules/uni-load-more/package.json
+84
-0
readme.md
src/uni_modules/uni-load-more/readme.md
+14
-0
config.js
src/utils/config.js
+2
-2
No files found.
src/api/ecw/warehouse.js
0 → 100644
View file @
64e10593
import
request
from
'
@/utils/request
'
import
lang
from
'
@/common/lang
'
export
function
getWarehouseList
(
query
)
{
return
request
.
get
(
'
/app-api/ecw/warehouse/list
'
,
query
);
}
export
function
getCountryTree
(
query
)
{
return
request
.
get
(
'
/app-api/ecw/warehouse/getWarehouseTreeList
'
,
query
);
}
src/api/order/cons.js
0 → 100644
View file @
64e10593
import
request
from
'
@/utils/request
'
export
function
getExpressNo
(){
return
request
.
get
(
'
/app-api/order/cons/express_no
'
);
}
export
function
createCons
(
param
){
return
request
.
post
(
'
/app-api/order/cons/create
'
,
param
);
}
export
function
updateCons
(
param
){
return
request
.
put
(
'
/app-api/order/cons/update
'
,
param
);
}
export
function
getCons
(
param
){
return
request
.
get
(
'
/app-api/order/cons/get
'
,
param
);
}
src/common/lang.js
View file @
64e10593
...
@@ -29,6 +29,7 @@ const languageData = {
...
@@ -29,6 +29,7 @@ const languageData = {
startAddress
:
'
startAddressZh
'
,
startAddress
:
'
startAddressZh
'
,
msgTitle
:
'
title
'
,
msgTitle
:
'
title
'
,
msgContent
:
'
content
'
,
msgContent
:
'
content
'
,
warehouseTitle
:
'
warehouseTitleZh
'
,
},
},
'
en
'
:
{
'
en
'
:
{
label
:
'
labelEn
'
,
label
:
'
labelEn
'
,
...
@@ -49,6 +50,7 @@ const languageData = {
...
@@ -49,6 +50,7 @@ const languageData = {
startAddress
:
'
startAddressEn
'
,
startAddress
:
'
startAddressEn
'
,
msgTitle
:
'
titleEn
'
,
msgTitle
:
'
titleEn
'
,
msgContent
:
'
contentEn
'
,
msgContent
:
'
contentEn
'
,
warehouseTitle
:
'
warehouseTitleEn
'
,
},
},
'
fr
'
:
{
'
fr
'
:
{
label
:
'
labelFr
'
,
label
:
'
labelFr
'
,
...
@@ -69,6 +71,8 @@ const languageData = {
...
@@ -69,6 +71,8 @@ const languageData = {
startAddress
:
'
startAddressFr
'
,
startAddress
:
'
startAddressFr
'
,
msgTitle
:
'
titleFr
'
,
msgTitle
:
'
titleFr
'
,
msgContent
:
'
contentFr
'
,
msgContent
:
'
contentFr
'
,
warehouseTitle
:
'
warehouseTitleFr
'
,
}
}
};
};
...
@@ -97,6 +101,7 @@ function getLanguageData(local = 'zh') {
...
@@ -97,6 +101,7 @@ function getLanguageData(local = 'zh') {
startAddress
:
languageData
[
local
].
startAddress
,
startAddress
:
languageData
[
local
].
startAddress
,
msgTitle
:
languageData
[
local
].
msgTitle
,
msgTitle
:
languageData
[
local
].
msgTitle
,
msgContent
:
languageData
[
local
].
msgContent
,
msgContent
:
languageData
[
local
].
msgContent
,
warehouseTitle
:
languageData
[
local
].
warehouseTitle
,
};
};
}
}
...
...
src/pages.json
View file @
64e10593
...
@@ -349,6 +349,12 @@
...
@@ -349,6 +349,12 @@
"style"
:
{
"style"
:
{
"navigationStyle"
:
"custom"
"navigationStyle"
:
"custom"
}
}
},
{
"path"
:
"pages/cons/create"
,
"style"
:
{
"navigationStyle"
:
"custom"
}
}
}
],
],
"globalStyle"
:
{
"globalStyle"
:
{
...
...
src/pages/cons/components/signInPicture.vue
View file @
64e10593
...
@@ -20,7 +20,7 @@
...
@@ -20,7 +20,7 @@
<
script
>
<
script
>
export
default
{
export
default
{
name
:
'
s
ignInPicture
'
,
name
:
'
S
ignInPicture
'
,
data
(){
data
(){
return
{
return
{
list
:[],
list
:[],
...
...
src/pages/cons/create.vue
View file @
64e10593
<
script
setup
>
<
template
>
<view>
<view
class=
"header-toper"
>
<view
class=
"start-bar"
></view>
<view
class=
"header-toper-title"
>
<image
src=
"../../static/img/back.png"
mode=
""
@
click=
"handleBack"
></image>
<text>
{{
$lang
.
lang
.
cons
.
create
}}
</text>
</view>
</view>
</
script
>
<view
class=
"container"
>
<view
class=
"form-section"
>
<
template
>
<!-- 运输Tab -->
<view
class=
"form-tab"
>
<!-- 标题 -->
<view
class=
"form-tab-top"
>
<image
src=
"../../static/img/corder-1.png"
mode=
""
></image>
<view
class=
""
>
{{
$lang
.
lang
.
cons
.
ship
}}
</view>
</view>
<view
class=
"form-tab-content"
>
<!-- 包裹编号(编辑才有) -->
<view
class=
"form-item"
v-if=
"form.id"
>
<view
class=
"form-label"
>
包裹编号:
</view>
<text>
{{
this
.
form
.
consNum
}}
</text>
</view>
<!-- 集运仓库 -->
<view
class=
"form-item"
>
<view
class=
"form-label"
><span
style=
"color: #ff5b5a;"
>
*
</span>
{{
$lang
.
lang
.
cons
.
consWarehouse
}}
:
</view>
<uni-data-checkbox
mode=
"tag"
v-model=
"form.wareId"
:localdata=
"warehouseList"
></uni-data-checkbox>
</view>
<!-- 快递单号 -->
<view
class=
"form-item"
>
<view
class=
"form-label"
><span
style=
"color: #ff5b5a;"
>
*
</span>
{{
$lang
.
lang
.
cons
.
expressNo
}}
:
</view>
<input
class=
"from-item-textarea"
v-model=
"form.expressNo"
@
input=
"handleExpressNo"
:placeholder=
"$lang.lang.cons.expressNoPlaceholder"
/>
<view
class=
"express-no-btn"
@
click=
"generateExpressNo"
>
{{
$lang
.
lang
.
cons
.
generateExpressNo
}}
</view>
</view>
<!-- 快递单号输入提示 -->
<view
class=
"express-no-notice"
><span
style=
"color: #ff5b5a;"
>
{{
expressNoNotice
}}
</span>
</view>
<!-- 运输方式 -->
<view
class=
"form-item"
>
<view
class=
"form-label"
><span
style=
"color: #ff5b5a;"
>
*
</span>
{{
$lang
.
lang
.
cons
.
transportType
}}
:
</view>
<radio-group
v-model=
"form.transportId"
@
change=
"radioChange"
>
<label
class=
"from-item-radio"
v-for=
"(item, index) in transportType"
>
<radio
:value=
"item.value"
:checked=
"parseInt(item.value) === parseInt(form.transportId)"
color=
"#007AFF"
/>
{{
item
.
label
}}
</label>
</radio-group>
</view>
<!-- 常用提货点 -->
<view
class=
"form-item"
>
<view
class=
"form-label"
><span
style=
"color: #ff5b5a;"
>
*
</span>
{{
$lang
.
lang
.
cons
.
pickupPoint
}}
:
</view>
<uni-data-picker
:placeholder=
"$lang.lang.cons.pickupPointNotice"
:popup-title=
"$lang.lang.cons.pickupPointNotice"
:localdata=
"countryTree"
v-model=
"pickupPoint"
></uni-data-picker>
</view>
</view>
</view>
<!-- 商品Tab -->
<view
class=
"form-tab"
>
<!-- 标题 -->
<view
class=
"form-tab-top"
>
<image
src=
"../../static/img/corder-1.png"
mode=
""
></image>
<view
class=
"no-divider"
>
<text>
{{
$lang
.
lang
.
cons
.
product
}}
</text>
<view
class=
"add-product-btn"
@
tap=
"addItem"
>
{{
$lang
.
lang
.
cons
.
addOneLine
}}
</view>
</view>
</view>
<view
class=
"form-tab-content"
>
<!-- 如果为编辑操作且商品信息为空且包裹状态已签收则提示 -->
<view
v-if=
"form.id && form.consItemVOList.length === 0"
class=
"not-goods-notice"
>
<span
style=
"color: #ff5b5a;"
>
{{
$lang
.
lang
.
cons
.
notGoodsNotice
}}
</span>
<view
class=
"contact-btn"
>
{{
$lang
.
lang
.
cons
.
goConcat
}}
</view>
</view>
<!-- 商品内容 -->
<view
class=
"form-item"
>
<view
class=
"goods-table"
v-if=
"form.consItemVOList.length > 0"
>
<view
class=
"table-header"
>
<text
class=
"header-cell"
>
{{
$lang
.
lang
.
cons
.
number
}}
</text>
<text
class=
"header-cell"
>
{{
$lang
.
lang
.
cons
.
goodsName
}}
</text>
<text
class=
"header-cell"
>
{{
$lang
.
lang
.
cons
.
num
}}
</text>
<text
class=
"header-cell"
>
{{
$lang
.
lang
.
cons
.
option
}}
</text>
</view>
<view
class=
"table-row"
v-for=
"(item, index) in form.consItemVOList"
:key=
"index"
>
<text
class=
"cell"
>
{{
index
+
1
}}
</text>
<view
class=
"cell"
style=
"width: 100%"
>
<cuihai-combox
class=
"goods-choose"
:label=
"$lang.lang.notices.please"
:emptyTips=
"$lang.lang.create.noSelect"
:placeholder=
"$lang.lang.notices.please"
:candidates=
"productData.label"
:value=
"item.prodName"
@
getValue=
"(e) => getValue(e, index)"
></cuihai-combox>
</view>
<view
class=
"cell"
>
<input
v-model=
"item.quantity"
type=
"number"
/>
</view>
<text
class=
"cell delete-btn"
@
tap=
"deleteItem(index)"
>
{{
$lang
.
lang
.
cons
.
deleted
}}
</text>
</view>
<!-- 统计商品总数 -->
<view
class=
"total-section"
>
<text
class=
"label"
>
{{
$lang
.
lang
.
cons
.
bookInTotal
}}
:
</text>
<text
class=
"total-number"
>
{{
totalQuantity
}}
个
</text>
</view>
</view>
</view>
<!-- 订单货值 -->
<view
class=
"form-item"
>
<view
class=
"form-label"
style=
"width: 260upx;"
><span
style=
"color: #ff5b5a;"
>
*
</span>
{{
$lang
.
lang
.
cons
.
orderTotalWorth
}}
:
</view>
<input
class=
"form-item-input"
v-model=
"form.worth"
:placeholder=
"$lang.lang.cons.orderTotalWorthNotice"
@
input=
"handleTotalValue"
type=
"number"
/>
</view>
</view>
</view>
<!-- 商品Tab -->
<view
class=
"form-tab"
>
<!-- 标题 -->
<view
class=
"form-tab-top"
>
<image
src=
"../../static/img/corder-1.png"
mode=
""
></image>
<view
class=
""
>
{{
$lang
.
lang
.
cons
.
other
}}
</view>
</view>
<view
class=
"form-tab-content"
>
<!-- 备注 -->
<view
class=
"form-item"
>
<view
class=
"form-label"
>
{{
$lang
.
lang
.
cons
.
remark
}}
:
</view>
<textarea
class=
"from-item-textarea"
style=
"width: 100%"
v-model=
"form.remarks"
:placeholder=
"$lang.lang.cons.remarkNotice"
/>
</view>
</view>
</view>
<!-- 同意协议 -->
<label
class=
"agree-checkbox"
>
<checkbox
:checked=
"agree"
@
tap=
"agree = !agree"
color=
"#007AFF"
/>
<text>
{{
$lang
.
lang
.
cons
.
warehouseInAgree
}}
</text>
</label>
<!-- 创建显示的按钮 -->
<view
v-if=
"!form.id"
>
<!-- 提交按钮 -->
<button
class=
"submit-btn"
type=
"primary"
@
tap=
"handleSubmit"
>
{{
$lang
.
lang
.
cons
.
confirmAdd
}}
</button>
<!-- 其他按钮 -->
<view
class=
"footer-nav"
>
<text
style=
"background-color: #F59123"
class=
"footer-btn"
@
tap=
"navigateTo('address')"
text=
""
>
{{
$lang
.
lang
.
cons
.
myConsAddress
}}
</text>
<text
style=
"margin-left: 20upx;background-color: #910012"
@
tap=
"navigateTo('address')"
class=
"footer-btn"
text=
""
>
{{
$lang
.
lang
.
cons
.
warehouseInNotice
}}
</text>
<text
style=
"margin-left: 20upx;background-color: red"
class=
"footer-btn"
@
tap=
"navigateTo('prohibited')"
text=
""
>
{{
$lang
.
lang
.
cons
.
banGoods
}}
</text>
</view>
</view>
<!-- 编辑显示的按钮 -->
<view
v-else
style=
"display: flex;padding: 0 80upx 0 80upx;margin-bottom: 20upx;"
>
<button
class=
"submit-btn"
type=
"primary"
@
tap=
"handleSubmit"
>
编辑
</button>
<button
class=
"cancel-btn"
type=
"warning"
@
tap=
"handleCancel"
>
取消
</button>
</view>
</view>
</view>
</view>
</
template
>
</
template
>
<
style
scoped
>
<
script
>
import
{
getWarehouseList
,
getCountryTree
}
from
"
@/api/ecw/warehouse
"
;
import
{
createCons
,
getExpressNo
,
getCons
,
updateCons
}
from
"
@/api/order/cons
"
;
import
lang
from
"
@/common/lang
"
;
export
default
{
data
()
{
const
initForm
=
{
id
:
null
,
// 包裹id(编辑才有)
consNum
:
''
,
// 包裹编号(编辑才有)
expressNo
:
''
,
transportId
:
''
,
// 常用提货点
consigneeWarehouseId
:
''
,
consigneeCountryId
:
''
,
consigneeCityId
:
''
,
customerId
:
''
,
customerNumber
:
''
,
worth
:
''
,
remarks
:
''
,
// 始发仓库id
wareId
:
null
,
consItemVOList
:
[],
sumQuantity
:
this
.
totalQuantity
,
}
return
{
form
:
{...
initForm
},
initForm
,
agree
:
true
,
// 商品列表信息
productData
:
{
index
:
0
,
label
:
[],
value
:
[],
data
:
[]
},
// 常用提货点,由(国家id-城市id-仓库id)组成
pickupPoint
:
''
,
// 集运仓库列表
warehouseList
:
[],
// 运输方式
transportType
:
null
,
// 快递单号输入提示
expressNoNotice
:
''
,
// 提货点树状结构
countryTree
:
[],
}
},
// option 请求链接的参数列表
onLoad
(
option
)
{
this
.
getProductData
();
this
.
getWarehouseList
();
// 显示快递单号输入提示
this
.
expressNoNotice
=
this
.
$lang
.
lang
.
cons
.
expressNoNotice_1
;
// 获取运输方式
this
.
getTransportType
();
this
.
getCountryTree
();
// 设置包裹id
this
.
form
.
id
=
option
.
id
;
// 初始化编辑页面数据
this
.
init
();
},
computed
:
{
totalQuantity
()
{
this
.
form
.
sumQuantity
=
this
.
form
.
consItemVOList
.
reduce
((
sum
,
item
)
=>
sum
+
(
Number
(
item
.
quantity
)
||
0
),
0
);
return
this
.
form
.
sumQuantity
;
}
},
methods
:
{
/**
* 编辑包裹初始化
*/
init
()
{
if
(
this
.
form
.
id
>
0
)
{
getCons
({
'
id
'
:
this
.
form
.
id
}).
then
(
res
=>
{
this
.
form
=
res
.
data
;
// 组装常用提货点所需数据,设置选中常用提货点
this
.
pickupPoint
=
[
this
.
form
.
consigneeCountryId
,
this
.
form
.
consigneeCityId
,
this
.
form
.
consigneeWarehouseId
].
join
(
'
-
'
)
});
}
},
/**
* 获取运输方式
*/
getTransportType
()
{
this
.
$request
.
getConfig
(
"
transport_type
"
).
then
(
res
=>
{
this
.
transportType
=
res
.
data
.
list
;
if
(
this
.
form
.
transportId
===
''
){
// 如果为空则设置第一个为默认值
this
.
form
.
transportId
=
res
.
data
.
list
[
0
].
value
;
}
})
},
/**
* 返回按钮提示
*/
handleBack
()
{
uni
.
showModal
({
// title: '提示',
content
:
this
.
$lang
.
lang
.
cons
.
backNotice
,
success
:
(
res
)
=>
{
if
(
res
.
confirm
)
{
// 用户点击确定,执行返回
uni
.
navigateBack
({
delta
:
1
// 返回层数,1表示返回上一页
});
}
}
});
},
/**
* 获取始发仓库
*/
getWarehouseList
()
{
getWarehouseList
({
tradeType
:
2
}).
then
(
res
=>
{
res
.
data
.
map
(
item
=>
{
// 如果没有选中则设置第一个为默认选中
if
(
this
.
form
.
wareId
===
null
&&
item
.
isConsService
===
1
)
this
.
form
.
wareId
=
1
// 组装组件所需参数
this
.
warehouseList
.
push
({
'
text
'
:
item
[
this
.
$lang
.
title
],
'
value
'
:
item
.
id
,
'
disable
'
:
!
Boolean
(
item
.
isConsService
)
// 是否开启集运服务
})
})
})
},
getProductData
()
{
this
.
$request
.
get
(
'
/app-api/ecw/product/list
'
,
{
auditStatus
:
2
}).
then
(
res
=>
{
if
(
res
.
code
===
0
&&
res
.
data
&&
res
.
data
.
length
>
0
)
{
let
d
=
res
.
data
for
(
let
i
in
d
)
{
this
.
productData
.
label
.
push
(
d
[
i
][
this
.
$lang
.
title
])
this
.
productData
.
value
.
push
(
d
[
i
][
this
.
$lang
.
title
])
this
.
productData
.
data
.
push
(
d
[
i
])
}
}
})
},
/**
* 生成快递单号
*/
generateExpressNo
()
{
getExpressNo
().
then
(
res
=>
{
this
.
form
.
expressNo
=
res
.
data
// 自动生成快递单号需提示用户填写到包裹面单
this
.
expressNoNotice
=
this
.
$lang
.
lang
.
cons
.
expressNoNotice_2
})
},
/**
* 获取国家树状结构数据
*/
getCountryTree
()
{
getCountryTree
({
tradeType
:
1
}).
then
(
res
=>
{
this
.
countryTree
=
res
.
data
.
filter
(
country
=>
country
.
guojia
)
// 过滤无效国家
.
sort
((
a
,
b
)
=>
(
a
.
guojiaSort
??
0
)
-
(
b
.
guojiaSort
??
0
))
.
map
(
country
=>
({
text
:
country
[
lang
.
guojiaName
].
replace
(
/
\([^
)
]
*
\)
/g
,
''
)
||
country
.
guojiaName
.
replace
(
/
\([^
)
]
*
\)
/g
,
''
),
value
:
String
(
country
.
guojia
),
children
:
(
country
.
children
||
[])
.
filter
(
city
=>
city
.
shi
)
// 过滤无效城市
.
sort
((
a
,
b
)
=>
(
a
.
shiSort
??
0
)
-
(
b
.
shiSort
??
0
))
.
map
(
city
=>
({
text
:
city
[
lang
.
shiName
].
replace
(
/(
[^
)
]
*)/g
,
''
)
||
city
.
shiName
.
replace
(
/(
[^
)
]
*)/g
,
''
),
value
:
String
(
city
.
shi
),
// 注意:示例中的"shi"应该实际用城市ID
children
:
(
city
.
children
||
[])
.
filter
(
warehouse
=>
warehouse
.
warehouseId
)
// 过滤无效仓库
.
sort
((
a
,
b
)
=>
(
a
.
sort
??
0
)
-
(
b
.
sort
??
0
))
.
map
(
warehouse
=>
({
text
:
warehouse
[
lang
.
warehouseTitle
]
||
warehouse
.
warehouseTitleZh
,
value
:
String
(
country
.
guojia
)
+
'
-
'
+
String
(
city
.
shi
)
+
'
-
'
+
String
(
warehouse
.
warehouseId
)
}))
}))
}));
})
},
/**
* 获取运输方式的值
*/
radioChange
(
e
)
{
this
.
form
.
transportId
=
e
.
detail
.
value
},
/**
* 添加一行
*/
addItem
()
{
this
.
form
.
consItemVOList
.
push
({
prodId
:
null
,
quantity
:
''
});
},
getValue
(
e
,
index
)
{
if
(
this
.
form
.
consItemVOList
[
index
])
{
// 根据商品名称获取商品信息
this
.
productData
.
data
.
forEach
((
item
)
=>
{
if
(
item
[
this
.
$lang
.
title
]
===
e
)
{
// 保存商品信息
this
.
form
.
consItemVOList
[
index
].
prodId
=
item
[
'
id
'
];
this
.
form
.
consItemVOList
[
index
].
prodAttrIds
=
item
[
'
attrId
'
];
this
.
form
.
consItemVOList
[
index
].
prodTitleEn
=
item
[
'
titleEn
'
];
this
.
form
.
consItemVOList
[
index
].
prodTitleZh
=
item
[
'
titleZh
'
];
this
.
form
.
consItemVOList
[
index
].
prodTitleFr
=
item
[
'
titleFr
'
];
this
.
form
.
consItemVOList
[
index
].
prodType
=
item
[
'
typeId
'
];
}
});
}
},
/**
* 删除一行
*/
deleteItem
(
index
)
{
this
.
form
.
consItemVOList
.
splice
(
index
,
1
)
},
/**
* 处理快递订单输入框,替换特殊字符
*/
handleExpressNo
(
e
)
{
// 获取输入值并过滤非字母数字字符
const
cleanedValue
=
e
.
detail
.
value
.
replace
(
/
[^
A-Za-z0-9
]
/g
,
''
);
// 立即更新模型数据
this
.
$nextTick
(()
=>
{
this
.
form
.
expressNo
=
cleanedValue
;
});
// iOS兼容性处理(解决某些情况下输入法问题)
if
(
uni
.
getSystemInfoSync
().
platform
===
'
ios
'
)
{
this
.
$forceUpdate
();
}
},
/**
* 处理货值输入框
*/
handleTotalValue
(
e
)
{
// 获取输入值并过滤非数字
const
cleanedValue
=
e
.
detail
.
value
.
replace
(
/
[^
0-9
]
/g
,
''
);
// 立即更新模型数据
this
.
$nextTick
(()
=>
{
this
.
form
.
worth
=
cleanedValue
;
});
// iOS兼容性处理(解决某些情况下输入法问题)
if
(
uni
.
getSystemInfoSync
().
platform
===
'
ios
'
)
{
this
.
$forceUpdate
();
}
},
// 校验商品选择后数量是否填写
validateGoodsList
()
{
return
this
.
form
.
consItemVOList
.
every
(
item
=>
{
// 如果选择了商品(product 非空)
if
(
item
.
prodId
!==
null
)
{
// 则数量必须为有效数字且 > 0
return
typeof
parseInt
(
item
.
quantity
)
===
'
number
'
&&
item
.
quantity
>
0
;
}
// 未选择商品时,忽略 quantity 的值
return
true
;
});
},
/**
* 提交表单
*/
handleSubmit
()
{
// 校验快递单号位数
if
(
this
.
form
.
expressNo
>=
0
&&
this
.
form
.
expressNo
<
7
)
{
uni
.
showToast
({
title
:
'
快递单号不得少于7位
'
,
icon
:
'
none
'
});
return
;
}
// 校验常用提货点
if
(
this
.
pickupPoint
===
''
)
{
uni
.
showToast
({
title
:
'
请选择常用提货点
'
,
icon
:
'
none
'
});
return
;
}
else
{
// 处理提货点字符串
const
pickupPointArr
=
this
.
pickupPoint
.
split
(
'
-
'
);
this
.
form
.
consigneeCountryId
=
pickupPointArr
[
0
]
this
.
form
.
consigneeCityId
=
pickupPointArr
[
1
]
this
.
form
.
consigneeWarehouseId
=
pickupPointArr
[
2
]
}
// 校验商品列表中的数量
if
(
!
this
.
validateGoodsList
())
{
uni
.
showToast
({
title
:
'
请确保已选商品的数量大于0
'
,
icon
:
'
none
'
});
return
;
}
// 如果货值小于10则提示
if
(
this
.
form
.
worth
<
10
)
{
uni
.
showModal
({
title
:
'
提示
'
,
content
:
this
.
$lang
.
lang
.
cons
.
orderTotalValueNotice
,
confirmText
:
'
忽略,继续提交
'
,
cancelText
:
'
前去检查
'
,
success
:
(
res
)
=>
{
if
(
res
.
confirm
)
{
// 用户点击确定,执行提交
this
.
submitForm
();
}
}
});
return
;
}
// 提交表单
this
.
submitForm
();
},
submitForm
()
{
console
.
log
({...
this
.
form
})
if
(
this
.
form
.
id
>
0
){
// 更新包裹
updateCons
({...
this
.
form
}).
then
(
res
=>
{
if
(
res
.
code
===
0
)
{
uni
.
showToast
({
title
:
'
编辑成功
'
,
icon
:
'
none
'
});
}
else
{
uni
.
showToast
({
title
:
res
.
msg
,
icon
:
'
none
'
});
}
})
}
else
{
// 创建包裹
createCons
({...
this
.
form
}).
then
((
res
)
=>
{
if
(
res
.
code
===
0
)
{
// 创建成功
uni
.
showModal
({
title
:
'
提示
'
,
content
:
'
集运包裹添加成功
'
,
confirmText
:
'
继续添加
'
,
cancelText
:
'
返回首页
'
,
success
:
(
res
)
=>
{
if
(
!
res
.
confirm
)
{
// 用户点击取消则返回首页
uni
.
navigateTo
({
url
:
'
../index/index
'
})
}
// 用户点击确定,停留在当前页面并清空数据
this
.
form
=
{...
this
.
initForm
};
this
.
form
.
consItemVOList
=
[];
this
.
pickupPoint
=
''
;
}
})
}
else
{
uni
.
showToast
({
title
:
res
.
msg
,
icon
:
'
none
'
});
}
});
}
},
/**
* 取消按钮操作
*/
handleCancel
(){
uni
.
navigateBack
({
delta
:
1
});
// 返回上一页
},
navigateTo
(
page
)
{
// 这里添加页面跳转逻辑
}
},
}
</
script
>
<
style
scoped
>
@import
url(../../static/css/cons_create.css)
;
</
style
>
</
style
>
src/pages/cons/index.vue
View file @
64e10593
...
@@ -5,7 +5,7 @@
...
@@ -5,7 +5,7 @@
<view
class=
"header-toper-title"
>
<view
class=
"header-toper-title"
>
<image
src=
"../../static/img/back.png"
mode=
""
@
click=
"goBack"
></image>
<image
src=
"../../static/img/back.png"
mode=
""
@
click=
"goBack"
></image>
<text>
{{
$lang
.
lang
.
cons
.
myConsPackage
}}
</text>
<text>
{{
$lang
.
lang
.
cons
.
myConsPackage
}}
</text>
<image
src=
"../../static/img/add
.png"
mode=
"
"
></image>
<image
src=
"../../static/img/add
_white.png"
@
click=
"goCreate
"
></image>
</view>
</view>
</view>
</view>
...
@@ -206,7 +206,11 @@ export default {
...
@@ -206,7 +206,11 @@ export default {
},
},
methods
:
{
methods
:
{
goCreate
(){
uni
.
navigateTo
({
url
:
'
/pages/cons/create
'
})
},
/**
/**
* 返回上一页
* 返回上一页
...
...
src/pages/create_order/create_order.vue
View file @
64e10593
...
@@ -368,7 +368,9 @@
...
@@ -368,7 +368,9 @@
<image
src=
"../../static/img/corder-1.png"
mode=
""
></image>
<image
src=
"../../static/img/corder-1.png"
mode=
""
></image>
<view
class=
""
>
{{$lang.lang.create.orderInfo}}
</view>
<view
class=
""
>
{{$lang.lang.create.orderInfo}}
</view>
</view>
</view>
<view
class=
"corder-v5"
>
<view
class=
"corder-v5"
>
<view
class=
"corder-v5-v"
>
<view
class=
"corder-v5-v"
>
<view
class=
"corder-v5-label"
><span
class=
"redcolor"
>
*
</span>
{{$lang.lang.create.marks}}
</view>
<view
class=
"corder-v5-label"
><span
class=
"redcolor"
>
*
</span>
{{$lang.lang.create.marks}}
</view>
<input
type=
"text"
v-model=
"params.marks"
maxlength=
"255"
:placeholder=
"$lang.lang.notices.marks"
@
blur=
"trimMarks"
/>
<input
type=
"text"
v-model=
"params.marks"
maxlength=
"255"
:placeholder=
"$lang.lang.notices.marks"
@
blur=
"trimMarks"
/>
...
...
src/static/css/cons_create.css
0 → 100644
View file @
64e10593
.container
{
padding
:
20
rpx
;
background-color
:
#f5f5f5
;
min-height
:
100vh
;
margin-top
:
100
upx
;
}
/*Tab样式*/
.form-tab
{
margin-bottom
:
30
rpx
;
background-color
:
#fff
;
padding
:
20
rpx
;
border-radius
:
20
rpx
;
}
/*标题Tab样式*/
.form-tab-top
{
display
:
flex
;
align-items
:
center
;
font-size
:
var
(
--f30
);
font-weight
:
700
;
padding
:
30
upx
;
position
:
relative
;
}
.form-tab-top
image
{
width
:
38
upx
;
height
:
38
upx
;
margin
:
0
20
upx
16
upx
0
;
}
.form-tab-top
view
{
/*display: block;*/
width
:
90%
;
border-bottom
:
1px
solid
var
(
--cdf
);
padding-bottom
:
16
upx
;
}
/*标题Tab样式-end*/
/*Tab内容样式*/
.form-tab-content
{
width
:
90%
;
margin
:
0
auto
;
}
.form-item
{
display
:
flex
;
align-items
:
center
;
font-size
:
var
(
--f24
);
color
:
var
(
--c-1
);
margin
:
0
upx
0
50
upx
0
;
}
.express-no-notice
{
/*display: none !important;*/
/*flex: 1;*/
align-items
:
center
;
font-size
:
var
(
--f24
);
margin
:
0
upx
0
50
upx
20
upx
;
}
.not-goods-notice
{
display
:
flex
;
align-items
:
center
;
font-size
:
var
(
--f24
);
}
.contact-btn
{
padding
:
16
upx
0
;
width
:
20%
;
margin-left
:
8
upx
;
border-radius
:
10
upx
;
text-align
:
center
;
color
:
#fff
;
background-color
:
#1373eb
;
}
.form-label
{
width
:
160
upx
;
display
:
flex
;
align-items
:
center
;
position
:
relative
;
}
.from-item-textarea
{
width
:
50%
;
height
:
30
upx
;
border-radius
:
12
upx
;
font-size
:
var
(
--f24
);
padding
:
12
upx
;
border
:
1px
solid
var
(
--cdf
);
}
.form-item-input
{
font-weight
:
500
;
font-size
:
var
(
--f26
);
border
:
1px
solid
var
(
--cdf
);
padding
:
20
upx
10
upx
;
width
:
80%
;
border-radius
:
12
upx
;
margin-right
:
10
upx
;
}
/*生成快递单号按钮*/
.express-no-btn
{
padding
:
16
upx
0
;
width
:
21%
;
margin-left
:
8
upx
;
border-radius
:
8
upx
;
text-align
:
center
;
color
:
#fff
;
background-color
:
#1373eb
;
}
.add-product-btn
{
background
:
var
(
--c036
);
color
:
var
(
--c0
);
padding
:
10
upx
20
upx
;
font-size
:
var
(
--f24
);
border-radius
:
12
upx
;
position
:
absolute
;
text-align
:
center
;
width
:
21%
!important
;
margin-top
:
-45
upx
;
right
:
30
upx
;
}
.no-divider
{
border-bottom
:
none
!important
;
}
.from-item-radio
{
align-items
:
center
;
margin-bottom
:
10
upx
;
margin-right
:
10
upx
;
position
:
relative
;
}
/*Tab内容样式-end*/
/*table*/
.goods-table
{
background-color
:
#fff
;
width
:
100%
;
border-radius
:
10
rpx
;
box-shadow
:
0
2
rpx
12
rpx
rgba
(
0
,
0
,
0
,
0.1
);
}
.table-header
,
.table-row
{
display
:
flex
;
padding
:
20
rpx
;
border-bottom
:
1
rpx
solid
#eee
;
}
.header-cell
{
flex
:
1
;
font-weight
:
bold
;
text-align
:
center
;
}
.cell
{
flex
:
1
;
text-align
:
center
;
display
:
flex
;
align-items
:
center
;
justify-content
:
center
;
}
.cell
input
{
width
:
50%
;
font-size
:
var
(
--f24
);
text-align
:
center
;
border
:
1
rpx
solid
#eee
;
/*padding-right: 15upx;*/
margin-left
:
15
upx
;
padding
:
8
rpx
;
border-radius
:
8
rpx
;
}
.goods-choose
{
width
:
150
upx
;
font-size
:
var
(
--f24
);
text-align
:
center
;
border
:
1
rpx
solid
#eee
;
padding
:
8
rpx
;
border-radius
:
8
rpx
;
}
.delete-btn
{
color
:
red
;
cursor
:
pointer
;
}
.total-section
{
padding
:
20
rpx
;
display
:
flex
;
justify-content
:
flex-end
;
align-items
:
center
;
}
.total-number
{
font-weight
:
bold
;
color
:
#007AFF
;
margin-left
:
10
rpx
;
}
/*table-end*/
.agree-checkbox
{
display
:
flex
;
align-items
:
center
;
margin
:
20
rpx
0
;
}
.submit-btn
{
margin-top
:
20
rpx
;
width
:
100%
;
border-radius
:
8
upx
;
text-align
:
center
;
color
:
#fff
;
background-color
:
#1373eb
;
font-size
:
35
upx
;
}
.cancel-btn
{
margin-top
:
20
rpx
;
width
:
100%
;
border-radius
:
8
upx
;
text-align
:
center
;
/*color: #fff;*/
background-color
:
#D3D3D3
;
color
:
#fff
;
font-size
:
35
upx
;
margin-left
:
50
upx
;
}
.footer-nav
{
display
:
flex
;
border-top
:
1
rpx
solid
#eee
;
}
.footer-btn
{
padding
:
25
upx
;
width
:
100%
;
margin-top
:
20
upx
;
border-radius
:
8
upx
;
text-align
:
center
;
color
:
#fff
;
/*background-color: #1373eb;*/
}
src/static/img/add_white.png
0 → 100644
View file @
64e10593
2.22 KB
src/static/lang/zh/cons.js
View file @
64e10593
export
default
{
export
default
{
"
myCons
"
:
"
我的集运
"
,
"
myCons
"
:
"
我的集运
"
,
"
myConsPackage
"
:
"
我的集运包裹
"
,
"
myConsPackage
"
:
"
我的集运包裹
"
,
"
address
"
:
"
专属地址
"
,
"
address
"
:
"
专属地址
"
,
"
turnOnWaitConfirm
"
:
"
转运待确认
"
,
"
turnOnWaitConfirm
"
:
"
转运待确认
"
,
"
consOrder
"
:
"
集运地址
"
,
"
consOrder
"
:
"
集运地址
"
,
"
searchNotice
"
:
"
支持包裹快递单号/提单/订单号查询/商品名称
"
,
"
searchNotice
"
:
"
支持包裹快递单号/提单/订单号查询/商品名称
"
,
"
closePopup
"
:
"
关闭弹窗
"
,
"
closePopup
"
:
"
关闭弹窗
"
,
"
create
"
:
"
包裹预报
"
,
"
backNotice
"
:
"
是否退出添加包裹页面?
"
,
"
ship
"
:
"
运输
"
,
"
consWarehouse
"
:
"
集运仓库
"
,
"
expressNo
"
:
"
快递单号
"
,
"
expressNoPlaceholder
"
:
"
请输入快递单号
"
,
"
generateExpressNo
"
:
"
生成单号
"
,
"
expressNoNotice_1
"
:
"
*请完整填写快递单号,包括字母快递单号需大于7位
"
,
"
expressNoNotice_2
"
:
"
*请于包裹面单上填写此快递单号
"
,
"
transportType
"
:
"
运输方式
"
,
"
pickupPoint
"
:
"
常用提货点
"
,
"
pickupPointNotice
"
:
"
请选择常用提货点
"
,
"
product
"
:
"
商品
"
,
"
addOneLine
"
:
"
添加一行
"
,
"
notGoodsNotice
"
:
"
不确定包裹内容?请联系客服申请验货
"
,
"
goConcat
"
:
"
去联系
"
,
"
number
"
:
"
序号
"
,
"
goodsName
"
:
"
品名
"
,
"
num
"
:
"
数量
"
,
"
option
"
:
"
操作
"
,
"
deleted
"
:
"
删除
"
,
"
bookInTotal
"
:
"
登记总数量
"
,
"
orderTotalWorth
"
:
"
订单总货值 RMB
"
,
"
orderTotalWorthNotice
"
:
"
填写订单人民币总货值
"
,
"
other
"
:
"
其他
"
,
"
remark
"
:
"
备注
"
,
"
remarkNotice
"
:
"
请输入备注
"
,
"
warehouseInAgree
"
:
"
我已仔细阅读并同意《入仓须知》
"
,
"
confirmAdd
"
:
"
确认添加
"
,
"
myConsAddress
"
:
"
我的集运地址
"
,
"
warehouseInNotice
"
:
"
入仓须知
"
,
"
banGoods
"
:
"
禁运物品
"
,
"
orderTotalValueNotice
"
:
"
货值需要大于等于10,请检查
"
,
}
}
src/uni_modules/uni-data-checkbox/changelog.md
0 → 100644
View file @
64e10593
## 1.0.6(2024-10-22)
-
新增 当 multiple 为 false 且传递的 value 为 数组时,使用数组第一项用作反显
## 1.0.5(2024-03-20)
-
修复 单选模式下选中样式不生效的bug
## 1.0.4(2024-01-27)
-
修复 修复错别字chagne为change
## 1.0.3(2022-09-16)
-
可以使用 uni-scss 控制主题色
## 1.0.2(2022-06-30)
-
优化 在 uni-forms 中的依赖注入方式
## 1.0.1(2022-02-07)
-
修复 multiple 为 true 时,v-model 的值为 null 报错的 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-data-checkbox
](
https://uniapp.dcloud.io/component/uniui/uni-data-checkbox
)
## 0.2.5(2021-08-23)
-
修复 在uni-forms中 modelValue 中不存在当前字段,当前字段必填写也不参与校验的问题
## 0.2.4(2021-08-17)
-
修复 单选 list 模式下 ,icon 为 left 时,选中图标不显示的问题
## 0.2.3(2021-08-11)
-
修复 在 uni-forms 中重置表单,错误信息无法清除的问题
## 0.2.2(2021-07-30)
-
优化 在uni-forms组件,与label不对齐的问题
## 0.2.1(2021-07-27)
-
修复 单选默认值为0不能选中的Bug
## 0.2.0(2021-07-13)
-
组件兼容 vue3,如何创建vue3项目,详见
[
uni-app 项目支持 vue3 介绍
](
https://ask.dcloud.net.cn/article/37834
)
## 0.1.11(2021-07-06)
-
优化 删除无用日志
## 0.1.10(2021-07-05)
-
修复 由 0.1.9 引起的非 nvue 端图标不显示的问题
## 0.1.9(2021-07-05)
-
修复 nvue 黑框样式问题
## 0.1.8(2021-06-28)
-
修复 selectedTextColor 属性不生效的Bug
## 0.1.7(2021-06-02)
-
新增 map 属性,可以方便映射text/value属性
## 0.1.6(2021-05-26)
-
修复 不关联服务空间的情况下组件报错的Bug
## 0.1.5(2021-05-12)
-
新增 组件示例地址
## 0.1.4(2021-04-09)
-
修复 nvue 下无法选中的问题
## 0.1.3(2021-03-22)
-
新增 disabled属性
## 0.1.2(2021-02-24)
-
优化 默认颜色显示
## 0.1.1(2021-02-24)
-
新增 支持nvue
## 0.1.0(2021-02-18)
-
“暂无数据”显示居中
src/uni_modules/uni-data-checkbox/components/uni-data-checkbox/clientdb.js
0 → 100644
View file @
64e10593
const
events
=
{
load
:
'
load
'
,
error
:
'
error
'
}
const
pageMode
=
{
add
:
'
add
'
,
replace
:
'
replace
'
}
const
attrs
=
[
'
pageCurrent
'
,
'
pageSize
'
,
'
collection
'
,
'
action
'
,
'
field
'
,
'
getcount
'
,
'
orderby
'
,
'
where
'
]
export
default
{
data
()
{
return
{
loading
:
false
,
listData
:
this
.
getone
?
{}
:
[],
paginationInternal
:
{
current
:
this
.
pageCurrent
,
size
:
this
.
pageSize
,
count
:
0
},
errorMessage
:
''
}
},
created
()
{
let
db
=
null
;
let
dbCmd
=
null
;
if
(
this
.
collection
){
this
.
db
=
uniCloud
.
database
();
this
.
dbCmd
=
this
.
db
.
command
;
}
this
.
_isEnded
=
false
this
.
$watch
(()
=>
{
let
al
=
[]
attrs
.
forEach
(
key
=>
{
al
.
push
(
this
[
key
])
})
return
al
},
(
newValue
,
oldValue
)
=>
{
this
.
paginationInternal
.
pageSize
=
this
.
pageSize
let
needReset
=
false
for
(
let
i
=
2
;
i
<
newValue
.
length
;
i
++
)
{
if
(
newValue
[
i
]
!=
oldValue
[
i
])
{
needReset
=
true
break
}
}
if
(
needReset
)
{
this
.
clear
()
this
.
reset
()
}
if
(
newValue
[
0
]
!=
oldValue
[
0
])
{
this
.
paginationInternal
.
current
=
this
.
pageCurrent
}
this
.
_execLoadData
()
})
// #ifdef H5
if
(
process
.
env
.
NODE_ENV
===
'
development
'
)
{
this
.
_debugDataList
=
[]
if
(
!
window
.
unidev
)
{
window
.
unidev
=
{
clientDB
:
{
data
:
[]
}
}
}
unidev
.
clientDB
.
data
.
push
(
this
.
_debugDataList
)
}
// #endif
// #ifdef MP-TOUTIAO
let
changeName
let
events
=
this
.
$scope
.
dataset
.
eventOpts
for
(
let
i
=
0
;
i
<
events
.
length
;
i
++
)
{
let
event
=
events
[
i
]
if
(
event
[
0
].
includes
(
'
^load
'
))
{
changeName
=
event
[
1
][
0
][
0
]
}
}
if
(
changeName
)
{
let
parent
=
this
.
$parent
let
maxDepth
=
16
this
.
_changeDataFunction
=
null
while
(
parent
&&
maxDepth
>
0
)
{
let
fun
=
parent
[
changeName
]
if
(
fun
&&
typeof
fun
===
'
function
'
)
{
this
.
_changeDataFunction
=
fun
maxDepth
=
0
break
}
parent
=
parent
.
$parent
maxDepth
--
;
}
}
// #endif
// if (!this.manual) {
// this.loadData()
// }
},
// #ifdef H5
beforeDestroy
()
{
if
(
process
.
env
.
NODE_ENV
===
'
development
'
&&
window
.
unidev
)
{
let
cd
=
this
.
_debugDataList
let
dl
=
unidev
.
clientDB
.
data
for
(
let
i
=
dl
.
length
-
1
;
i
>=
0
;
i
--
)
{
if
(
dl
[
i
]
===
cd
)
{
dl
.
splice
(
i
,
1
)
break
}
}
}
},
// #endif
methods
:
{
loadData
(
args1
,
args2
)
{
let
callback
=
null
if
(
typeof
args1
===
'
object
'
)
{
if
(
args1
.
clear
)
{
this
.
clear
()
this
.
reset
()
}
if
(
args1
.
current
!==
undefined
)
{
this
.
paginationInternal
.
current
=
args1
.
current
}
if
(
typeof
args2
===
'
function
'
)
{
callback
=
args2
}
}
else
if
(
typeof
args1
===
'
function
'
)
{
callback
=
args1
}
this
.
_execLoadData
(
callback
)
},
loadMore
()
{
if
(
this
.
_isEnded
)
{
return
}
this
.
_execLoadData
()
},
refresh
()
{
this
.
clear
()
this
.
_execLoadData
()
},
clear
()
{
this
.
_isEnded
=
false
this
.
listData
=
[]
},
reset
()
{
this
.
paginationInternal
.
current
=
1
},
remove
(
id
,
{
action
,
callback
,
confirmTitle
,
confirmContent
}
=
{})
{
if
(
!
id
||
!
id
.
length
)
{
return
}
uni
.
showModal
({
title
:
confirmTitle
||
'
提示
'
,
content
:
confirmContent
||
'
是否删除该数据
'
,
showCancel
:
true
,
success
:
(
res
)
=>
{
if
(
!
res
.
confirm
)
{
return
}
this
.
_execRemove
(
id
,
action
,
callback
)
}
})
},
_execLoadData
(
callback
)
{
if
(
this
.
loading
)
{
return
}
this
.
loading
=
true
this
.
errorMessage
=
''
this
.
_getExec
().
then
((
res
)
=>
{
this
.
loading
=
false
const
{
data
,
count
}
=
res
.
result
this
.
_isEnded
=
data
.
length
<
this
.
pageSize
callback
&&
callback
(
data
,
this
.
_isEnded
)
this
.
_dispatchEvent
(
events
.
load
,
data
)
if
(
this
.
getone
)
{
this
.
listData
=
data
.
length
?
data
[
0
]
:
undefined
}
else
if
(
this
.
pageData
===
pageMode
.
add
)
{
this
.
listData
.
push
(...
data
)
if
(
this
.
listData
.
length
)
{
this
.
paginationInternal
.
current
++
}
}
else
if
(
this
.
pageData
===
pageMode
.
replace
)
{
this
.
listData
=
data
this
.
paginationInternal
.
count
=
count
}
// #ifdef H5
if
(
process
.
env
.
NODE_ENV
===
'
development
'
)
{
this
.
_debugDataList
.
length
=
0
this
.
_debugDataList
.
push
(...
JSON
.
parse
(
JSON
.
stringify
(
this
.
listData
)))
}
// #endif
}).
catch
((
err
)
=>
{
this
.
loading
=
false
this
.
errorMessage
=
err
callback
&&
callback
()
this
.
$emit
(
events
.
error
,
err
)
})
},
_getExec
()
{
let
exec
=
this
.
db
if
(
this
.
action
)
{
exec
=
exec
.
action
(
this
.
action
)
}
exec
=
exec
.
collection
(
this
.
collection
)
if
(
!
(
!
this
.
where
||
!
Object
.
keys
(
this
.
where
).
length
))
{
exec
=
exec
.
where
(
this
.
where
)
}
if
(
this
.
field
)
{
exec
=
exec
.
field
(
this
.
field
)
}
if
(
this
.
orderby
)
{
exec
=
exec
.
orderBy
(
this
.
orderby
)
}
const
{
current
,
size
}
=
this
.
paginationInternal
exec
=
exec
.
skip
(
size
*
(
current
-
1
)).
limit
(
size
).
get
({
getCount
:
this
.
getcount
})
return
exec
},
_execRemove
(
id
,
action
,
callback
)
{
if
(
!
this
.
collection
||
!
id
)
{
return
}
const
ids
=
Array
.
isArray
(
id
)
?
id
:
[
id
]
if
(
!
ids
.
length
)
{
return
}
uni
.
showLoading
({
mask
:
true
})
let
exec
=
this
.
db
if
(
action
)
{
exec
=
exec
.
action
(
action
)
}
exec
.
collection
(
this
.
collection
).
where
({
_id
:
dbCmd
.
in
(
ids
)
}).
remove
().
then
((
res
)
=>
{
callback
&&
callback
(
res
.
result
)
if
(
this
.
pageData
===
pageMode
.
replace
)
{
this
.
refresh
()
}
else
{
this
.
removeData
(
ids
)
}
}).
catch
((
err
)
=>
{
uni
.
showModal
({
content
:
err
.
message
,
showCancel
:
false
})
}).
finally
(()
=>
{
uni
.
hideLoading
()
})
},
removeData
(
ids
)
{
let
il
=
ids
.
slice
(
0
)
let
dl
=
this
.
listData
for
(
let
i
=
dl
.
length
-
1
;
i
>=
0
;
i
--
)
{
let
index
=
il
.
indexOf
(
dl
[
i
].
_id
)
if
(
index
>=
0
)
{
dl
.
splice
(
i
,
1
)
il
.
splice
(
index
,
1
)
}
}
},
_dispatchEvent
(
type
,
data
)
{
if
(
this
.
_changeDataFunction
)
{
this
.
_changeDataFunction
(
data
,
this
.
_isEnded
)
}
else
{
this
.
$emit
(
type
,
data
,
this
.
_isEnded
)
}
}
}
}
src/uni_modules/uni-data-checkbox/components/uni-data-checkbox/uni-data-checkbox.vue
0 → 100644
View file @
64e10593
<
template
>
<view
class=
"uni-data-checklist"
:style=
"
{'margin-top':isTop+'px'}">
<template
v-if=
"!isLocal"
>
<view
class=
"uni-data-loading"
>
<uni-load-more
v-if=
"!mixinDatacomErrorMessage"
status=
"loading"
iconType=
"snow"
:iconSize=
"18"
:content-text=
"contentText"
></uni-load-more>
<text
v-else
>
{{
mixinDatacomErrorMessage
}}
</text>
</view>
</
template
>
<
template
v-else
>
<checkbox-group
v-if=
"multiple"
class=
"checklist-group"
:class=
"
{'is-list':mode==='list' || wrap}"
@change="change">
<label
class=
"checklist-box"
:class=
"['is--'+mode,item.selected?'is-checked':'',(disabled || !!item.disabled)?'is-disable':'',index!==0&&mode==='list'?'is-list-border':'']"
:style=
"item.styleBackgroud"
v-for=
"(item,index) in dataList"
:key=
"index"
>
<checkbox
class=
"hidden"
hidden
:disabled=
"disabled || !!item.disabled"
:value=
"item[map.value]+''"
:checked=
"item.selected"
/>
<view
v-if=
"(mode !=='tag' && mode !== 'list') || ( mode === 'list' && icon === 'left')"
class=
"checkbox__inner"
:style=
"item.styleIcon"
>
<view
class=
"checkbox__inner-icon"
></view>
</view>
<view
class=
"checklist-content"
:class=
"
{'list-content':mode === 'list'
&&
icon ==='left'}">
<text
class=
"checklist-text"
:style=
"item.styleIconText"
>
{{
item
[
map
.
text
]
}}
</text>
<view
v-if=
"mode === 'list' && icon === 'right'"
class=
"checkobx__list"
:style=
"item.styleBackgroud"
></view>
</view>
</label>
</checkbox-group>
<radio-group
v-else
class=
"checklist-group"
:class=
"
{'is-list':mode==='list','is-wrap':wrap}" @change="change">
<label
class=
"checklist-box"
:class=
"['is--'+mode,item.selected?'is-checked':'',(disabled || !!item.disabled)?'is-disable':'',index!==0&&mode==='list'?'is-list-border':'']"
:style=
"item.styleBackgroud"
v-for=
"(item,index) in dataList"
:key=
"index"
>
<radio
class=
"hidden"
hidden
:disabled=
"disabled || item.disabled"
:value=
"item[map.value]+''"
:checked=
"item.selected"
/>
<view
v-if=
"(mode !=='tag' && mode !== 'list') || ( mode === 'list' && icon === 'left')"
class=
"radio__inner"
:style=
"item.styleBackgroud"
>
<view
class=
"radio__inner-icon"
:style=
"item.styleIcon"
></view>
</view>
<view
class=
"checklist-content"
:class=
"
{'list-content':mode === 'list'
&&
icon ==='left'}">
<text
class=
"checklist-text"
:style=
"item.styleIconText"
>
{{
item
[
map
.
text
]
}}
</text>
<view
v-if=
"mode === 'list' && icon === 'right'"
:style=
"item.styleRightIcon"
class=
"checkobx__list"
></view>
</view>
</label>
</radio-group>
</
template
>
</view>
</template>
<
script
>
/**
* DataChecklist 数据选择器
* @description 通过数据渲染 checkbox 和 radio
* @tutorial https://ext.dcloud.net.cn/plugin?id=xxx
* @property {String} mode = [default| list | button | tag] 显示模式
* @value default 默认横排模式
* @value list 列表模式
* @value button 按钮模式
* @value tag 标签模式
* @property {Boolean} multiple = [true|false] 是否多选
* @property {Array|String|Number} value 默认值
* @property {Array} localdata 本地数据 ,格式 [{text:'',value:''}]
* @property {Number|String} min 最小选择个数 ,multiple为true时生效
* @property {Number|String} max 最大选择个数 ,multiple为true时生效
* @property {Boolean} wrap 是否换行显示
* @property {String} icon = [left|right] list 列表模式下icon显示位置
* @property {Boolean} selectedColor 选中颜色
* @property {Boolean} emptyText 没有数据时显示的文字 ,本地数据无效
* @property {Boolean} selectedTextColor 选中文本颜色,如不填写则自动显示
* @property {Object} map 字段映射, 默认 map={text:'text',value:'value'}
* @value left 左侧显示
* @value right 右侧显示
* @event {Function} change 选中发生变化触发
*/
export
default
{
name
:
'
uniDataChecklist
'
,
mixins
:
[
uniCloud
.
mixinDatacom
||
{}],
emits
:
[
'
input
'
,
'
update:modelValue
'
,
'
change
'
],
props
:
{
mode
:
{
type
:
String
,
default
:
'
default
'
},
multiple
:
{
type
:
Boolean
,
default
:
false
},
value
:
{
type
:
[
Array
,
String
,
Number
],
default
()
{
return
''
}
},
// TODO vue3
modelValue
:
{
type
:
[
Array
,
String
,
Number
],
default
()
{
return
''
;
}
},
localdata
:
{
type
:
Array
,
default
()
{
return
[]
}
},
min
:
{
type
:
[
Number
,
String
],
default
:
''
},
max
:
{
type
:
[
Number
,
String
],
default
:
''
},
wrap
:
{
type
:
Boolean
,
default
:
false
},
icon
:
{
type
:
String
,
default
:
'
left
'
},
selectedColor
:
{
type
:
String
,
default
:
''
},
selectedTextColor
:
{
type
:
String
,
default
:
''
},
emptyText
:
{
type
:
String
,
default
:
'
暂无数据
'
},
disabled
:
{
type
:
Boolean
,
default
:
false
},
map
:
{
type
:
Object
,
default
()
{
return
{
text
:
'
text
'
,
value
:
'
value
'
}
}
}
},
watch
:
{
localdata
:
{
handler
(
newVal
)
{
this
.
range
=
newVal
this
.
dataList
=
this
.
getDataList
(
this
.
getSelectedValue
(
newVal
))
},
deep
:
true
},
mixinDatacomResData
(
newVal
)
{
this
.
range
=
newVal
this
.
dataList
=
this
.
getDataList
(
this
.
getSelectedValue
(
newVal
))
},
value
(
newVal
)
{
this
.
dataList
=
this
.
getDataList
(
newVal
)
// fix by mehaotian is_reset 在 uni-forms 中定义
// if(!this.is_reset){
// this.is_reset = false
// this.formItem && this.formItem.setValue(newVal)
// }
},
modelValue
(
newVal
)
{
this
.
dataList
=
this
.
getDataList
(
newVal
);
// if(!this.is_reset){
// this.is_reset = false
// this.formItem && this.formItem.setValue(newVal)
// }
}
},
data
()
{
return
{
dataList
:
[],
range
:
[],
contentText
:
{
contentdown
:
'
查看更多
'
,
contentrefresh
:
'
加载中
'
,
contentnomore
:
'
没有更多
'
},
isLocal
:
true
,
styles
:
{
selectedColor
:
'
#2979ff
'
,
selectedTextColor
:
'
#666
'
,
},
isTop
:
0
};
},
computed
:
{
dataValue
()
{
if
(
this
.
value
===
''
)
return
this
.
modelValue
if
(
this
.
modelValue
===
''
)
return
this
.
value
return
this
.
value
}
},
created
()
{
// this.form = this.getForm('uniForms')
// this.formItem = this.getForm('uniFormsItem')
// this.formItem && this.formItem.setValue(this.value)
// if (this.formItem) {
// this.isTop = 6
// if (this.formItem.name) {
// // 如果存在name添加默认值,否则formData 中不存在这个字段不校验
// if(!this.is_reset){
// this.is_reset = false
// this.formItem.setValue(this.dataValue)
// }
// this.rename = this.formItem.name
// this.form.inputChildrens.push(this)
// }
// }
if
(
this
.
localdata
&&
this
.
localdata
.
length
!==
0
)
{
this
.
isLocal
=
true
this
.
range
=
this
.
localdata
this
.
dataList
=
this
.
getDataList
(
this
.
getSelectedValue
(
this
.
range
))
}
else
{
if
(
this
.
collection
)
{
this
.
isLocal
=
false
this
.
loadData
()
}
}
},
methods
:
{
loadData
()
{
this
.
mixinDatacomGet
().
then
(
res
=>
{
this
.
mixinDatacomResData
=
res
.
result
.
data
if
(
this
.
mixinDatacomResData
.
length
===
0
)
{
this
.
isLocal
=
false
this
.
mixinDatacomErrorMessage
=
this
.
emptyText
}
else
{
this
.
isLocal
=
true
}
}).
catch
(
err
=>
{
this
.
mixinDatacomErrorMessage
=
err
.
message
})
},
/**
* 获取父元素实例
*/
getForm
(
name
=
'
uniForms
'
)
{
let
parent
=
this
.
$parent
;
let
parentName
=
parent
.
$options
.
name
;
while
(
parentName
!==
name
)
{
parent
=
parent
.
$parent
;
if
(
!
parent
)
return
false
parentName
=
parent
.
$options
.
name
;
}
return
parent
;
},
change
(
e
)
{
const
values
=
e
.
detail
.
value
let
detail
=
{
value
:
[],
data
:
[]
}
if
(
this
.
multiple
)
{
this
.
range
.
forEach
(
item
=>
{
if
(
values
.
includes
(
item
[
this
.
map
.
value
]
+
''
))
{
detail
.
value
.
push
(
item
[
this
.
map
.
value
])
detail
.
data
.
push
(
item
)
}
})
}
else
{
const
range
=
this
.
range
.
find
(
item
=>
(
item
[
this
.
map
.
value
]
+
''
)
===
values
)
if
(
range
)
{
detail
=
{
value
:
range
[
this
.
map
.
value
],
data
:
range
}
}
}
// this.formItem && this.formItem.setValue(detail.value)
// TODO 兼容 vue2
this
.
$emit
(
'
input
'
,
detail
.
value
);
// // TOTO 兼容 vue3
this
.
$emit
(
'
update:modelValue
'
,
detail
.
value
);
this
.
$emit
(
'
change
'
,
{
detail
})
if
(
this
.
multiple
)
{
// 如果 v-model 没有绑定 ,则走内部逻辑
// if (this.value.length === 0) {
this
.
dataList
=
this
.
getDataList
(
detail
.
value
,
true
)
// }
}
else
{
this
.
dataList
=
this
.
getDataList
(
detail
.
value
)
}
},
/**
* 获取渲染的新数组
* @param {Object} value 选中内容
*/
getDataList
(
value
)
{
// 解除引用关系,破坏原引用关系,避免污染源数据
let
dataList
=
JSON
.
parse
(
JSON
.
stringify
(
this
.
range
))
let
list
=
[]
if
(
this
.
multiple
)
{
if
(
!
Array
.
isArray
(
value
))
{
value
=
[]
}
}
else
{
if
(
Array
.
isArray
(
value
)
&&
value
.
length
)
{
value
=
value
[
0
]
}
}
dataList
.
forEach
((
item
,
index
)
=>
{
item
.
disabled
=
item
.
disable
||
item
.
disabled
||
false
if
(
this
.
multiple
)
{
if
(
value
.
length
>
0
)
{
let
have
=
value
.
find
(
val
=>
val
===
item
[
this
.
map
.
value
])
item
.
selected
=
have
!==
undefined
}
else
{
item
.
selected
=
false
}
}
else
{
item
.
selected
=
value
===
item
[
this
.
map
.
value
]
}
list
.
push
(
item
)
})
return
this
.
setRange
(
list
)
},
/**
* 处理最大最小值
* @param {Object} list
*/
setRange
(
list
)
{
let
selectList
=
list
.
filter
(
item
=>
item
.
selected
)
let
min
=
Number
(
this
.
min
)
||
0
let
max
=
Number
(
this
.
max
)
||
''
list
.
forEach
((
item
,
index
)
=>
{
if
(
this
.
multiple
)
{
if
(
selectList
.
length
<=
min
)
{
let
have
=
selectList
.
find
(
val
=>
val
[
this
.
map
.
value
]
===
item
[
this
.
map
.
value
])
if
(
have
!==
undefined
)
{
item
.
disabled
=
true
}
}
if
(
selectList
.
length
>=
max
&&
max
!==
''
)
{
let
have
=
selectList
.
find
(
val
=>
val
[
this
.
map
.
value
]
===
item
[
this
.
map
.
value
])
if
(
have
===
undefined
)
{
item
.
disabled
=
true
}
}
}
this
.
setStyles
(
item
,
index
)
list
[
index
]
=
item
})
return
list
},
/**
* 设置 class
* @param {Object} item
* @param {Object} index
*/
setStyles
(
item
,
index
)
{
// 设置自定义样式
item
.
styleBackgroud
=
this
.
setStyleBackgroud
(
item
)
item
.
styleIcon
=
this
.
setStyleIcon
(
item
)
item
.
styleIconText
=
this
.
setStyleIconText
(
item
)
item
.
styleRightIcon
=
this
.
setStyleRightIcon
(
item
)
},
/**
* 获取选中值
* @param {Object} range
*/
getSelectedValue
(
range
)
{
if
(
!
this
.
multiple
)
return
this
.
dataValue
let
selectedArr
=
[]
range
.
forEach
((
item
)
=>
{
if
(
item
.
selected
)
{
selectedArr
.
push
(
item
[
this
.
map
.
value
])
}
})
return
this
.
dataValue
.
length
>
0
?
this
.
dataValue
:
selectedArr
},
/**
* 设置背景样式
*/
setStyleBackgroud
(
item
)
{
let
styles
=
{}
let
selectedColor
=
this
.
selectedColor
?
this
.
selectedColor
:
'
#2979ff
'
if
(
this
.
selectedColor
)
{
if
(
this
.
mode
!==
'
list
'
)
{
styles
[
'
border-color
'
]
=
item
.
selected
?
selectedColor
:
'
#DCDFE6
'
}
if
(
this
.
mode
===
'
tag
'
)
{
styles
[
'
background-color
'
]
=
item
.
selected
?
selectedColor
:
'
#f5f5f5
'
}
}
let
classles
=
''
for
(
let
i
in
styles
)
{
classles
+=
`
${
i
}
:
${
styles
[
i
]}
;`
}
return
classles
},
setStyleIcon
(
item
)
{
let
styles
=
{}
let
classles
=
''
if
(
this
.
selectedColor
)
{
let
selectedColor
=
this
.
selectedColor
?
this
.
selectedColor
:
'
#2979ff
'
styles
[
'
background-color
'
]
=
item
.
selected
?
selectedColor
:
'
#fff
'
styles
[
'
border-color
'
]
=
item
.
selected
?
selectedColor
:
'
#DCDFE6
'
if
(
!
item
.
selected
&&
item
.
disabled
)
{
styles
[
'
background-color
'
]
=
'
#F2F6FC
'
styles
[
'
border-color
'
]
=
item
.
selected
?
selectedColor
:
'
#DCDFE6
'
}
}
for
(
let
i
in
styles
)
{
classles
+=
`
${
i
}
:
${
styles
[
i
]}
;`
}
return
classles
},
setStyleIconText
(
item
)
{
let
styles
=
{}
let
classles
=
''
if
(
this
.
selectedColor
)
{
let
selectedColor
=
this
.
selectedColor
?
this
.
selectedColor
:
'
#2979ff
'
if
(
this
.
mode
===
'
tag
'
)
{
styles
.
color
=
item
.
selected
?
(
this
.
selectedTextColor
?
this
.
selectedTextColor
:
'
#fff
'
)
:
'
#666
'
}
else
{
styles
.
color
=
item
.
selected
?
(
this
.
selectedTextColor
?
this
.
selectedTextColor
:
selectedColor
)
:
'
#666
'
}
if
(
!
item
.
selected
&&
item
.
disabled
)
{
styles
.
color
=
'
#999
'
}
}
for
(
let
i
in
styles
)
{
classles
+=
`
${
i
}
:
${
styles
[
i
]}
;`
}
return
classles
},
setStyleRightIcon
(
item
)
{
let
styles
=
{}
let
classles
=
''
if
(
this
.
mode
===
'
list
'
)
{
styles
[
'
border-color
'
]
=
item
.
selected
?
this
.
styles
.
selectedColor
:
'
#DCDFE6
'
}
for
(
let
i
in
styles
)
{
classles
+=
`
${
i
}
:
${
styles
[
i
]}
;`
}
return
classles
}
}
}
</
script
>
<
style
lang=
"scss"
>
$uni-primary
:
#2979ff
!
default
;
$border-color
:
#DCDFE6
;
$disable
:
0
.4
;
@mixin
flex
{
/* #ifndef APP-NVUE */
display
:
flex
;
/* #endif */
}
.uni-data-loading
{
@include
flex
;
flex-direction
:
row
;
justify-content
:
center
;
align-items
:
center
;
height
:
36px
;
padding-left
:
10px
;
color
:
#999
;
}
.uni-data-checklist
{
position
:
relative
;
z-index
:
0
;
flex
:
1
;
// 多选样式
.checklist-group
{
@include
flex
;
flex-direction
:
row
;
flex-wrap
:
wrap
;
&
.is-list
{
flex-direction
:
column
;
}
.checklist-box
{
@include
flex
;
flex-direction
:
row
;
align-items
:
center
;
position
:
relative
;
margin
:
5px
0
;
margin-right
:
25px
;
.hidden
{
position
:
absolute
;
opacity
:
0
;
}
// 文字样式
.checklist-content
{
@include
flex
;
flex
:
1
;
flex-direction
:
row
;
align-items
:
center
;
justify-content
:
space-between
;
.checklist-text
{
font-size
:
14px
;
color
:
#666
;
margin-left
:
5px
;
line-height
:
14px
;
}
.checkobx__list
{
border-right-width
:
1px
;
border-right-color
:
#007aff
;
border-right-style
:
solid
;
border-bottom-width
:
1px
;
border-bottom-color
:
#007aff
;
border-bottom-style
:
solid
;
height
:
12px
;
width
:
6px
;
left
:
-5px
;
transform-origin
:
center
;
transform
:
rotate
(
45deg
);
opacity
:
0
;
}
}
// 多选样式
.checkbox__inner
{
/* #ifndef APP-NVUE */
flex-shrink
:
0
;
box-sizing
:
border-box
;
/* #endif */
position
:
relative
;
width
:
16px
;
height
:
16px
;
border
:
1px
solid
$border-color
;
border-radius
:
4px
;
background-color
:
#fff
;
z-index
:
1
;
.checkbox__inner-icon
{
position
:
absolute
;
/* #ifdef APP-NVUE */
top
:
2px
;
/* #endif */
/* #ifndef APP-NVUE */
top
:
1px
;
/* #endif */
left
:
5px
;
height
:
8px
;
width
:
4px
;
border-right-width
:
1px
;
border-right-color
:
#fff
;
border-right-style
:
solid
;
border-bottom-width
:
1px
;
border-bottom-color
:
#fff
;
border-bottom-style
:
solid
;
opacity
:
0
;
transform-origin
:
center
;
transform
:
rotate
(
40deg
);
}
}
// 单选样式
.radio__inner
{
@include
flex
;
/* #ifndef APP-NVUE */
flex-shrink
:
0
;
box-sizing
:
border-box
;
/* #endif */
justify-content
:
center
;
align-items
:
center
;
position
:
relative
;
width
:
16px
;
height
:
16px
;
border
:
1px
solid
$border-color
;
border-radius
:
16px
;
background-color
:
#fff
;
z-index
:
1
;
.radio__inner-icon
{
width
:
8px
;
height
:
8px
;
border-radius
:
10px
;
opacity
:
0
;
}
}
// 默认样式
&
.is--default
{
// 禁用
&
.is-disable
{
/* #ifdef H5 */
cursor
:
not
-
allowed
;
/* #endif */
.checkbox__inner
{
background-color
:
#F2F6FC
;
border-color
:
$border-color
;
/* #ifdef H5 */
cursor
:
not
-
allowed
;
/* #endif */
}
.radio__inner
{
background-color
:
#F2F6FC
;
border-color
:
$border-color
;
}
.checklist-text
{
color
:
#999
;
}
}
// 选中
&
.is-checked
{
.checkbox__inner
{
border-color
:
$uni-primary
;
background-color
:
$uni-primary
;
.checkbox__inner-icon
{
opacity
:
1
;
transform
:
rotate
(
45deg
);
}
}
.radio__inner
{
border-color
:
$uni-primary
;
.radio__inner-icon
{
opacity
:
1
;
background-color
:
$uni-primary
;
}
}
.checklist-text
{
color
:
$uni-primary
;
}
// 选中禁用
&
.is-disable
{
.checkbox__inner
{
opacity
:
$disable
;
}
.checklist-text
{
opacity
:
$disable
;
}
.radio__inner
{
opacity
:
$disable
;
}
}
}
}
// 按钮样式
&
.is--button
{
margin-right
:
10px
;
padding
:
5px
10px
;
border
:
1px
$border-color
solid
;
border-radius
:
3px
;
transition
:
border-color
0
.2s
;
// 禁用
&
.is-disable
{
/* #ifdef H5 */
cursor
:
not
-
allowed
;
/* #endif */
border
:
1px
#eee
solid
;
opacity
:
$disable
;
.checkbox__inner
{
background-color
:
#F2F6FC
;
border-color
:
$border-color
;
/* #ifdef H5 */
cursor
:
not
-
allowed
;
/* #endif */
}
.radio__inner
{
background-color
:
#F2F6FC
;
border-color
:
$border-color
;
/* #ifdef H5 */
cursor
:
not
-
allowed
;
/* #endif */
}
.checklist-text
{
color
:
#999
;
}
}
&
.is-checked
{
border-color
:
$uni-primary
;
.checkbox__inner
{
border-color
:
$uni-primary
;
background-color
:
$uni-primary
;
.checkbox__inner-icon
{
opacity
:
1
;
transform
:
rotate
(
45deg
);
}
}
.radio__inner
{
border-color
:
$uni-primary
;
.radio__inner-icon
{
opacity
:
1
;
background-color
:
$uni-primary
;
}
}
.checklist-text
{
color
:
$uni-primary
;
}
// 选中禁用
&
.is-disable
{
opacity
:
$disable
;
}
}
}
// 标签样式
&
.is--tag
{
margin-right
:
10px
;
padding
:
5px
10px
;
border
:
1px
$border-color
solid
;
border-radius
:
3px
;
background-color
:
#f5f5f5
;
.checklist-text
{
margin
:
0
;
color
:
#666
;
}
// 禁用
&
.is-disable
{
/* #ifdef H5 */
cursor
:
not
-
allowed
;
/* #endif */
opacity
:
$disable
;
}
&
.is-checked
{
background-color
:
$uni-primary
;
border-color
:
$uni-primary
;
.checklist-text
{
color
:
#fff
;
}
}
}
// 列表样式
&
.is--list
{
/* #ifndef APP-NVUE */
display
:
flex
;
/* #endif */
padding
:
10px
15px
;
padding-left
:
0
;
margin
:
0
;
&
.is-list-border
{
border-top
:
1px
#eee
solid
;
}
// 禁用
&
.is-disable
{
/* #ifdef H5 */
cursor
:
not
-
allowed
;
/* #endif */
.checkbox__inner
{
background-color
:
#F2F6FC
;
border-color
:
$border-color
;
/* #ifdef H5 */
cursor
:
not
-
allowed
;
/* #endif */
}
.checklist-text
{
color
:
#999
;
}
}
&
.is-checked
{
.checkbox__inner
{
border-color
:
$uni-primary
;
background-color
:
$uni-primary
;
.checkbox__inner-icon
{
opacity
:
1
;
transform
:
rotate
(
45deg
);
}
}
.radio__inner
{
border-color
:
$uni-primary
;
.radio__inner-icon
{
opacity
:
1
;
background-color
:
$uni-primary
;
}
}
.checklist-text
{
color
:
$uni-primary
;
}
.checklist-content
{
.checkobx__list
{
opacity
:
1
;
border-color
:
$uni-primary
;
}
}
// 选中禁用
&
.is-disable
{
.checkbox__inner
{
opacity
:
$disable
;
}
.checklist-text
{
opacity
:
$disable
;
}
}
}
}
}
}
}
</
style
>
src/uni_modules/uni-data-checkbox/package.json
0 → 100644
View file @
64e10593
{
"id"
:
"uni-data-checkbox"
,
"displayName"
:
"uni-data-checkbox 数据选择器"
,
"version"
:
"1.0.6"
,
"description"
:
"通过数据驱动的单选框和复选框"
,
"keywords"
:
[
"uni-ui"
,
"checkbox"
,
"单选"
,
"多选"
,
"单选多选"
],
"repository"
:
"https://github.com/dcloudio/uni-ui"
,
"engines"
:
{
"HBuilderX"
:
"^3.1.1"
},
"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-load-more"
,
"uni-scss"
],
"encrypt"
:
[],
"platforms"
:
{
"cloud"
:
{
"tcb"
:
"y"
,
"aliyun"
:
"y"
,
"alipay"
:
"n"
},
"client"
:
{
"App"
:
{
"app-vue"
:
"y"
,
"app-nvue"
:
"y"
,
"app-harmony"
:
"u"
,
"app-uvue"
:
"u"
},
"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"
}
}
}
}
}
src/uni_modules/uni-data-checkbox/readme.md
0 → 100644
View file @
64e10593
## DataCheckbox 数据驱动的单选复选框
> **组件名:uni-data-checkbox**
> 代码块: `uDataCheckbox`
本组件是基于uni-app基础组件checkbox的封装。本组件要解决问题包括:
1.
数据绑定型组件:给本组件绑定一个data,会自动渲染一组候选内容。再以往,开发者需要编写不少代码实现类似功能
2.
自动的表单校验:组件绑定了data,且符合
[
uni-forms
](
https://ext.dcloud.net.cn/plugin?id=2773
)
组件的表单校验规范,搭配使用会自动实现表单校验
3.
本组件合并了单选多选
4.
本组件有若干风格选择,如普通的单选多选框、并列button风格、tag风格。开发者可以快速选择需要的风格。但作为一个封装组件,样式代码虽然不用自己写了,却会牺牲一定的样式自定义性
在uniCloud开发中,
`DB Schema`
中配置了enum枚举等类型后,在web控制台的
[
自动生成表单
](
https://uniapp.dcloud.io/uniCloud/schema?id=autocode
)
功能中,会自动生成
``uni-data-checkbox``
组件并绑定好data
### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-data-checkbox)
#### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839
\ No newline at end of file
src/uni_modules/uni-data-picker/changelog.md
0 → 100644
View file @
64e10593
## 2.0.1(2024-08-22)
-
修复 uni-app-x v-model 没有更新传入值的 bug
## 2.0.0(2023-12-11)
-
新增 支持 uni-app-x
## 1.1.2(2023-04-11)
-
修复 更改 modelValue 报错的 bug
-
修复 v-for 未使用 key 值控制台 warning
## 1.1.1(2023-02-21)
-
修复代码合并时引发 value 属性为空时不渲染数据的问题
## 1.1.0(2023-02-15)
-
修复 localdata 不支持动态更新的bug
## 1.0.9(2023-02-15)
-
修复 localdata 不支持动态更新的bug
## 1.0.8(2022-09-16)
-
可以使用 uni-scss 控制主题色
## 1.0.7(2022-07-06)
-
优化 pc端图标位置不正确的问题
## 1.0.6(2022-07-05)
-
优化 显示样式
## 1.0.5(2022-07-04)
-
修复 uni-data-picker 在 uni-forms-item 中宽度不正确的bug
## 1.0.4(2022-04-19)
-
修复 字节小程序 本地数据无法选择下一级的Bug
## 1.0.3(2022-02-25)
-
修复 nvue 不支持的 v-show 的 bug
## 1.0.2(2022-02-25)
-
修复 条件编译 nvue 不支持的 css 样式
## 1.0.1(2021-11-23)
-
修复 由上个版本引发的map、v-model等属性不生效的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-data-picker
](
https://uniapp.dcloud.io/component/uniui/uni-data-picker
)
## 0.4.9(2021-10-28)
-
修复 VUE2 v-model 概率无效的 bug
## 0.4.8(2021-10-27)
-
修复 v-model 概率无效的 bug
## 0.4.7(2021-10-25)
-
新增 属性 spaceInfo 服务空间配置 HBuilderX 3.2.11+
-
修复 树型 uniCloud 数据类型为 int 时报错的 bug
## 0.4.6(2021-10-19)
-
修复 非 VUE3 v-model 为 0 时无法选中的 bug
## 0.4.5(2021-09-26)
-
新增 清除已选项的功能(通过 clearIcon 属性配置是否显示按钮),同时提供 clear 方法以供调用,二者等效
-
修复 readonly 为 true 时报错的 bug
## 0.4.4(2021-09-26)
-
修复 上一版本造成的 map 属性失效的 bug
-
新增 ellipsis 属性,支持配置 tab 选项长度过长时是否自动省略
## 0.4.3(2021-09-24)
-
修复 某些情况下级联未触发的 bug
## 0.4.2(2021-09-23)
-
新增 提供 show 和 hide 方法,开发者可以通过 ref 调用
-
新增 选项内容过长自动添加省略号
## 0.4.1(2021-09-15)
-
新增 map 属性 字段映射,将 text/value 映射到数据中的其他字段
## 0.4.0(2021-07-13)
-
组件兼容 vue3,如何创建 vue3 项目,详见
[
uni-app 项目支持 vue3 介绍
](
https://ask.dcloud.net.cn/article/37834
)
## 0.3.5(2021-06-04)
-
修复 无法加载云端数据的问题
## 0.3.4(2021-05-28)
-
修复 v-model 无效问题
-
修复 loaddata 为空数据组时加载时间过长问题
-
修复 上个版本引出的本地数据无法选择带有 children 的 2 级节点
## 0.3.3(2021-05-12)
-
新增 组件示例地址
## 0.3.2(2021-04-22)
-
修复 非树形数据有 where 属性查询报错的问题
## 0.3.1(2021-04-15)
-
修复 本地数据概率无法回显时问题
## 0.3.0(2021-04-07)
-
新增 支持云端非树形表结构数据
-
修复 根节点 parent_field 字段等于 null 时选择界面错乱问题
## 0.2.0(2021-03-15)
-
修复 nodeclick、popupopened、popupclosed 事件无法触发的问题
## 0.1.9(2021-03-09)
-
修复 微信小程序某些情况下无法选择的问题
## 0.1.8(2021-02-05)
-
优化 部分样式在 nvue 上的兼容表现
## 0.1.7(2021-02-05)
-
调整为 uni_modules 目录规范
src/uni_modules/uni-data-picker/components/uni-data-picker/keypress.js
0 → 100644
View file @
64e10593
// #ifdef H5
export
default
{
name
:
'
Keypress
'
,
props
:
{
disable
:
{
type
:
Boolean
,
default
:
false
}
},
mounted
()
{
const
keyNames
=
{
esc
:
[
'
Esc
'
,
'
Escape
'
],
tab
:
'
Tab
'
,
enter
:
'
Enter
'
,
space
:
[
'
'
,
'
Spacebar
'
],
up
:
[
'
Up
'
,
'
ArrowUp
'
],
left
:
[
'
Left
'
,
'
ArrowLeft
'
],
right
:
[
'
Right
'
,
'
ArrowRight
'
],
down
:
[
'
Down
'
,
'
ArrowDown
'
],
delete
:
[
'
Backspace
'
,
'
Delete
'
,
'
Del
'
]
}
const
listener
=
(
$event
)
=>
{
if
(
this
.
disable
)
{
return
}
const
keyName
=
Object
.
keys
(
keyNames
).
find
(
key
=>
{
const
keyName
=
$event
.
key
const
value
=
keyNames
[
key
]
return
value
===
keyName
||
(
Array
.
isArray
(
value
)
&&
value
.
includes
(
keyName
))
})
if
(
keyName
)
{
// 避免和其他按键事件冲突
setTimeout
(()
=>
{
this
.
$emit
(
keyName
,
{})
},
0
)
}
}
document
.
addEventListener
(
'
keyup
'
,
listener
)
this
.
$once
(
'
hook:beforeDestroy
'
,
()
=>
{
document
.
removeEventListener
(
'
keyup
'
,
listener
)
})
},
render
:
()
=>
{}
}
// #endif
src/uni_modules/uni-data-picker/components/uni-data-picker/uni-data-picker.uvue
0 → 100644
View file @
64e10593
<template>
<view class="uni-data-tree">
<view class="uni-data-tree-input" @click="handleInput">
<slot :data="selectedPaths" :error="error">
<view class="input-value" :class="{'input-value-border': border}">
<text v-if="error!=null" class="error-text">{{error!.errMsg}}</text>
<scroll-view v-if="selectedPaths.length" class="selected-path" scroll-x="true">
<view class="selected-list">
<template v-for="(item, index) in selectedPaths">
<text class="text-color">{{item[mappingTextName]}}</text>
<text v-if="index<selectedPaths.length-1" class="input-split-line">{{split}}</text>
</template>
</view>
</scroll-view>
<text v-else-if="error==null&&!loading" class="placeholder">{{placeholder}}</text>
<view v-if="!readonly" class="arrow-area">
<view class="input-arrow"></view>
</view>
</view>
</slot>
<view v-if="loading && !isOpened" class="selected-loading">
<slot name="picker-loading" :loading="loading"></slot>
</view>
</view>
<view class="uni-data-tree-cover" v-if="isOpened" @click="handleClose"></view>
<view class="uni-data-tree-dialog" v-if="isOpened">
<view class="uni-popper__arrow"></view>
<view class="dialog-caption">
<view class="dialog-title-view">
<text class="dialog-title">{{popupTitle}}</text>
</view>
<view class="dialog-close" @click="handleClose">
<view class="dialog-close-plus" data-id="close"></view>
<view class="dialog-close-plus dialog-close-rotate" data-id="close"></view>
</view>
</view>
<view ref="pickerView" class="uni-data-pickerview">
<view v-if="error!=null" class="error">
<text class="error-text">{{error!.errMsg}}</text>
</view>
<scroll-view v-if="!isCloudDataList" :scroll-x="true">
<view class="selected-node-list">
<template v-for="(item, index) in selectedNodes">
<text class="selected-node-item" :class="{'selected-node-item-active':index==selectedIndex}"
@click="onTabSelect(index)">
{{item[mappingTextName]}}
</text>
</template>
</view>
</scroll-view>
<list-view class="list-view" :scroll-y="true">
<list-item class="list-item" v-for="(item, _) in currentDataList" @click="onNodeClick(item)">
<text class="item-text" :class="{'item-text-disabled': item['disable']}">{{item[mappingTextName]}}</text>
<text class="check" v-if="item[mappingValueName] == selectedNodes[selectedIndex][mappingValueName]"></text>
</list-item>
</list-view>
<view class="loading-cover" v-if="loading">
<slot name="pickerview-loading" :loading="loading"></slot>
</view>
</view>
</view>
</view>
</template>
<script>
import { dataPicker } from "../uni-data-pickerview/uni-data-picker.uts"
/**
* DataPicker 级联选择
* @description 支持单列、和多列级联选择。列数没有限制,如果屏幕显示不全,顶部tab区域会左右滚动。
* @tutorial https://ext.dcloud.net.cn/plugin?id=3796
* @property {String} popup-title 弹出窗口标题
* @property {Array} localdata 本地数据,参考
* @property {Boolean} border = [true|false] 是否有边框
* @property {Boolean} readonly = [true|false] 是否仅读
* @property {Boolean} preload = [true|false] 是否预加载数据
* @value true 开启预加载数据,点击弹出窗口后显示已加载数据
* @value false 关闭预加载数据,点击弹出窗口后开始加载数据
* @property {Boolean} step-searh = [true|false] 是否分布查询
* @value true 启用分布查询,仅查询当前选中节点
* @value false 关闭分布查询,一次查询出所有数据
* @property {String|DBFieldString} self-field 分布查询当前字段名称
* @property {String|DBFieldString} parent-field 分布查询父字段名称
* @property {String|DBCollectionString} collection 表名
* @property {String|DBFieldString} field 查询字段,多个字段用 `,` 分割
* @property {String} orderby 排序字段及正序倒叙设置
* @property {String|JQLString} where 查询条件
* @event {Function} popupshow 弹出的选择窗口打开时触发此事件
* @event {Function} popuphide 弹出的选择窗口关闭时触发此事件
*/
export default {
name: 'UniDataPicker',
emits: ['popupopened', 'popupclosed', 'nodeclick', 'change', 'input', 'update:modelValue', 'inputclick'],
mixins: [dataPicker],
props: {
popupTitle: {
type: String,
default: '请选择'
},
placeholder: {
type: String,
default: '请选择'
},
heightMobile: {
type: String,
default: ''
},
readonly: {
type: Boolean,
default: false
},
clearIcon: {
type: Boolean,
default: true
},
border: {
type: Boolean,
default: true
},
split: {
type: String,
default: '/'
},
ellipsis: {
type: Boolean,
default: true
}
},
data() {
return {
isOpened: false
}
},
computed: {
isShowClearIcon() : boolean {
if (this.readonly) {
return false
}
if (this.clearIcon && this.selectedPaths.length > 0) {
return true
}
return false
}
},
created() {
this.load()
},
methods: {
clear() {
},
load() {
if (this.isLocalData) {
this.loadLocalData()
} else if (this.isCloudDataList || this.isCloudDataTree) {
this.loadCloudDataPath()
}
},
show() {
this.isOpened = true
this.$emit('popupopened')
if (!this.hasCloudTreeData) {
this.loadData()
}
},
hide() {
this.isOpened = false
this.$emit('popupclosed')
},
handleInput() {
if (this.readonly) {
this.$emit('inputclick')
} else {
this.show()
}
},
handleClose() {
this.hide()
},
onFinish() {
this.selectedPaths = this.getChangeNodes()
this.$emit('update:modelValue', this.selectedPaths)
this.$emit('change', this.selectedPaths)
this.hide()
}
}
}
</script>
<style>
@import url("../uni-data-pickerview/uni-data-pickerview.css");
.uni-data-tree {
position: relative;
}
.uni-data-tree-input {
position: relative;
}
.selected-loading {
display: flex;
justify-content: center;
position: absolute;
left: 0;
top: 0;
right: 0;
bottom: 0;
}
.error-text {
flex: 1;
font-size: 12px;
color: #DD524D;
}
.input-value {
flex-direction: row;
align-items: center;
flex-wrap: nowrap;
padding: 5px 5px;
padding-right: 5px;
overflow: hidden;
min-height: 28px;
}
.input-value-border {
border: 1px solid #e5e5e5;
border-radius: 5px;
}
.selected-path {
flex: 1;
flex-direction: row;
overflow: hidden;
}
.load-more {
width: 40px;
}
.selected-list {
flex-direction: row;
flex-wrap: nowrap;
}
.selected-item {
flex-direction: row;
flex-wrap: nowrap;
}
.text-color {
font-size: 14px;
color: #333;
}
.placeholder {
color: grey;
font-size: 14px;
}
.input-split-line {
opacity: .5;
margin-left: 1px;
margin-right: 1px;
}
.arrow-area {
position: relative;
padding: 0 12px;
margin-left: auto;
justify-content: center;
transform: rotate(-45deg);
transform-origin: center;
}
.input-arrow {
width: 8px;
height: 8px;
border-left: 2px solid #999;
border-bottom: 2px solid #999;
}
.uni-data-tree-cover {
position: fixed;
left: 0;
top: 0;
right: 0;
bottom: 0;
background-color: rgba(0, 0, 0, .4);
flex-direction: column;
z-index: 100;
}
.uni-data-tree-dialog {
position: fixed;
left: 0;
top: 20%;
right: 0;
bottom: 0;
background-color: #FFFFFF;
border-top-left-radius: 10px;
border-top-right-radius: 10px;
flex-direction: column;
z-index: 102;
overflow: hidden;
}
.dialog-caption {
position: relative;
flex-direction: row;
}
.dialog-title-view {
flex: 1;
}
.dialog-title {
align-self: center;
padding: 0 10px;
line-height: 44px;
}
.dialog-close {
position: absolute;
top: 0;
right: 0;
bottom: 0;
flex-direction: row;
align-items: center;
padding: 0 15px;
}
.dialog-close-plus {
width: 16px;
height: 2px;
background-color: #666;
border-radius: 2px;
transform: rotate(45deg);
}
.dialog-close-rotate {
position: absolute;
transform: rotate(-45deg);
}
.uni-data-pickerview {
flex: 1;
}
.icon-clear {
display: flex;
align-items: center;
}
/* #ifdef H5 */
@media all and (min-width: 768px) {
.uni-data-tree-cover {
background-color: transparent;
}
.uni-data-tree-dialog {
position: absolute;
top: 55px;
height: auto;
min-height: 400px;
max-height: 50vh;
background-color: #fff;
border: 1px solid #EBEEF5;
box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1);
border-radius: 4px;
overflow: unset;
}
.dialog-caption {
display: none;
}
}
/* #endif */
</style>
src/uni_modules/uni-data-picker/components/uni-data-picker/uni-data-picker.vue
0 → 100644
View file @
64e10593
<
template
>
<view
class=
"uni-data-tree"
>
<view
class=
"uni-data-tree-input"
@
click=
"handleInput"
>
<slot
:options=
"options"
:data=
"inputSelected"
:error=
"errorMessage"
>
<view
class=
"input-value"
:class=
"
{'input-value-border': border}">
<text
v-if=
"errorMessage"
class=
"selected-area error-text"
>
{{
errorMessage
}}
</text>
<view
v-else-if=
"loading && !isOpened"
class=
"selected-area"
>
<uni-load-more
class=
"load-more"
:contentText=
"loadMore"
status=
"loading"
></uni-load-more>
</view>
<scroll-view
v-else-if=
"inputSelected.length"
class=
"selected-area"
scroll-x=
"true"
>
<view
class=
"selected-list"
>
<view
class=
"selected-item"
v-for=
"(item,index) in inputSelected"
:key=
"index"
>
<text
class=
"text-color"
>
{{
item
.
text
}}
</text><text
v-if=
"index
<inputSelected
.
length-1
"
class=
"input-split-line"
>
{{
split
}}
</text>
</view>
</view>
</scroll-view>
<text
v-else
class=
"selected-area placeholder"
>
{{
placeholder
}}
</text>
<view
v-if=
"clearIcon && !readonly && inputSelected.length"
class=
"icon-clear"
@
click.stop=
"clear"
>
<uni-icons
type=
"clear"
color=
"#c0c4cc"
size=
"24"
></uni-icons>
</view>
<view
class=
"arrow-area"
v-if=
"(!clearIcon || !inputSelected.length) && !readonly "
>
<view
class=
"input-arrow"
></view>
</view>
</view>
</slot>
</view>
<view
class=
"uni-data-tree-cover"
v-if=
"isOpened"
@
click=
"handleClose"
></view>
<view
class=
"uni-data-tree-dialog"
v-if=
"isOpened"
>
<view
class=
"uni-popper__arrow"
></view>
<view
class=
"dialog-caption"
>
<view
class=
"title-area"
>
<text
class=
"dialog-title"
>
{{
popupTitle
}}
</text>
</view>
<view
class=
"dialog-close"
@
click=
"handleClose"
>
<view
class=
"dialog-close-plus"
data-id=
"close"
></view>
<view
class=
"dialog-close-plus dialog-close-rotate"
data-id=
"close"
></view>
</view>
</view>
<data-picker-view
class=
"picker-view"
ref=
"pickerView"
v-model=
"dataValue"
:localdata=
"localdata"
:preload=
"preload"
:collection=
"collection"
:field=
"field"
:orderby=
"orderby"
:where=
"where"
:step-searh=
"stepSearh"
:self-field=
"selfField"
:parent-field=
"parentField"
:managed-mode=
"true"
:map=
"map"
:ellipsis=
"ellipsis"
@
change=
"onchange"
@
datachange=
"ondatachange"
@
nodeclick=
"onnodeclick"
>
</data-picker-view>
</view>
</view>
</
template
>
<
script
>
import
dataPicker
from
"
../uni-data-pickerview/uni-data-picker.js
"
import
DataPickerView
from
"
../uni-data-pickerview/uni-data-pickerview.vue
"
/**
* DataPicker 级联选择
* @description 支持单列、和多列级联选择。列数没有限制,如果屏幕显示不全,顶部tab区域会左右滚动。
* @tutorial https://ext.dcloud.net.cn/plugin?id=3796
* @property {String} popup-title 弹出窗口标题
* @property {Array} localdata 本地数据,参考
* @property {Boolean} border = [true|false] 是否有边框
* @property {Boolean} readonly = [true|false] 是否仅读
* @property {Boolean} preload = [true|false] 是否预加载数据
* @value true 开启预加载数据,点击弹出窗口后显示已加载数据
* @value false 关闭预加载数据,点击弹出窗口后开始加载数据
* @property {Boolean} step-searh = [true|false] 是否分布查询
* @value true 启用分布查询,仅查询当前选中节点
* @value false 关闭分布查询,一次查询出所有数据
* @property {String|DBFieldString} self-field 分布查询当前字段名称
* @property {String|DBFieldString} parent-field 分布查询父字段名称
* @property {String|DBCollectionString} collection 表名
* @property {String|DBFieldString} field 查询字段,多个字段用 `,` 分割
* @property {String} orderby 排序字段及正序倒叙设置
* @property {String|JQLString} where 查询条件
* @event {Function} popupshow 弹出的选择窗口打开时触发此事件
* @event {Function} popuphide 弹出的选择窗口关闭时触发此事件
*/
export
default
{
name
:
'
UniDataPicker
'
,
emits
:
[
'
popupopened
'
,
'
popupclosed
'
,
'
nodeclick
'
,
'
input
'
,
'
change
'
,
'
update:modelValue
'
,
'
inputclick
'
],
mixins
:
[
dataPicker
],
components
:
{
DataPickerView
},
props
:
{
options
:
{
type
:
[
Object
,
Array
],
default
()
{
return
{}
}
},
popupTitle
:
{
type
:
String
,
default
:
'
请选择
'
},
placeholder
:
{
type
:
String
,
default
:
'
请选择
'
},
heightMobile
:
{
type
:
String
,
default
:
''
},
readonly
:
{
type
:
Boolean
,
default
:
false
},
clearIcon
:
{
type
:
Boolean
,
default
:
true
},
border
:
{
type
:
Boolean
,
default
:
true
},
split
:
{
type
:
String
,
default
:
'
/
'
},
ellipsis
:
{
type
:
Boolean
,
default
:
true
}
},
data
()
{
return
{
isOpened
:
false
,
inputSelected
:
[]
}
},
created
()
{
this
.
$nextTick
(()
=>
{
this
.
load
();
})
},
watch
:
{
localdata
:
{
handler
()
{
this
.
load
()
},
deep
:
true
},
},
methods
:
{
clear
()
{
this
.
_dispatchEvent
([]);
},
onPropsChange
()
{
this
.
_treeData
=
[];
this
.
selectedIndex
=
0
;
this
.
load
();
},
load
()
{
if
(
this
.
readonly
)
{
this
.
_processReadonly
(
this
.
localdata
,
this
.
dataValue
);
return
;
}
// 回显本地数据
if
(
this
.
isLocalData
)
{
this
.
loadData
();
this
.
inputSelected
=
this
.
selected
.
slice
(
0
);
}
else
if
(
this
.
isCloudDataList
||
this
.
isCloudDataTree
)
{
// 回显 Cloud 数据
this
.
loading
=
true
;
this
.
getCloudDataValue
().
then
((
res
)
=>
{
this
.
loading
=
false
;
this
.
inputSelected
=
res
;
}).
catch
((
err
)
=>
{
this
.
loading
=
false
;
this
.
errorMessage
=
err
;
})
}
},
show
()
{
this
.
isOpened
=
true
setTimeout
(()
=>
{
this
.
$refs
.
pickerView
.
updateData
({
treeData
:
this
.
_treeData
,
selected
:
this
.
selected
,
selectedIndex
:
this
.
selectedIndex
})
},
200
)
this
.
$emit
(
'
popupopened
'
)
},
hide
()
{
this
.
isOpened
=
false
this
.
$emit
(
'
popupclosed
'
)
},
handleInput
()
{
if
(
this
.
readonly
)
{
this
.
$emit
(
'
inputclick
'
)
return
}
this
.
show
()
},
handleClose
(
e
)
{
this
.
hide
()
},
onnodeclick
(
e
)
{
this
.
$emit
(
'
nodeclick
'
,
e
)
},
ondatachange
(
e
)
{
this
.
_treeData
=
this
.
$refs
.
pickerView
.
_treeData
},
onchange
(
e
)
{
this
.
hide
()
this
.
$nextTick
(()
=>
{
this
.
inputSelected
=
e
;
})
this
.
_dispatchEvent
(
e
)
},
_processReadonly
(
dataList
,
value
)
{
var
isTree
=
dataList
.
findIndex
((
item
)
=>
{
return
item
.
children
})
if
(
isTree
>
-
1
)
{
let
inputValue
if
(
Array
.
isArray
(
value
))
{
inputValue
=
value
[
value
.
length
-
1
]
if
(
typeof
inputValue
===
'
object
'
&&
inputValue
.
value
)
{
inputValue
=
inputValue
.
value
}
}
else
{
inputValue
=
value
}
this
.
inputSelected
=
this
.
_findNodePath
(
inputValue
,
this
.
localdata
)
return
}
if
(
!
this
.
hasValue
)
{
this
.
inputSelected
=
[]
return
}
let
result
=
[]
for
(
let
i
=
0
;
i
<
value
.
length
;
i
++
)
{
var
val
=
value
[
i
]
var
item
=
dataList
.
find
((
v
)
=>
{
return
v
.
value
==
val
})
if
(
item
)
{
result
.
push
(
item
)
}
}
if
(
result
.
length
)
{
this
.
inputSelected
=
result
}
},
_filterForArray
(
data
,
valueArray
)
{
var
result
=
[]
for
(
let
i
=
0
;
i
<
valueArray
.
length
;
i
++
)
{
var
value
=
valueArray
[
i
]
var
found
=
data
.
find
((
item
)
=>
{
return
item
.
value
==
value
})
if
(
found
)
{
result
.
push
(
found
)
}
}
return
result
},
_dispatchEvent
(
selected
)
{
let
item
=
{}
if
(
selected
.
length
)
{
var
value
=
new
Array
(
selected
.
length
)
for
(
var
i
=
0
;
i
<
selected
.
length
;
i
++
)
{
value
[
i
]
=
selected
[
i
].
value
}
item
=
selected
[
selected
.
length
-
1
]
}
else
{
item
.
value
=
''
}
if
(
this
.
formItem
)
{
this
.
formItem
.
setValue
(
item
.
value
)
}
this
.
$emit
(
'
input
'
,
item
.
value
)
this
.
$emit
(
'
update:modelValue
'
,
item
.
value
)
this
.
$emit
(
'
change
'
,
{
detail
:
{
value
:
selected
}
})
}
}
}
</
script
>
<
style
>
.uni-data-tree
{
flex
:
1
;
position
:
relative
;
font-size
:
var
(
--f24
);
}
.error-text
{
color
:
#DD524D
;
}
.input-value
{
/* #ifndef APP-NVUE */
display
:
flex
;
/* #endif */
flex-direction
:
row
;
align-items
:
center
;
flex-wrap
:
nowrap
;
font-size
:
var
(
--f24
);
/* line-height: 35px; */
padding
:
0
10px
;
padding-right
:
5px
;
overflow
:
hidden
;
height
:
35px
;
/* #ifndef APP-NVUE */
box-sizing
:
border-box
;
/* #endif */
}
.input-value-border
{
border
:
1px
solid
#e5e5e5
;
border-radius
:
5px
;
}
.selected-area
{
flex
:
1
;
overflow
:
hidden
;
/* #ifndef APP-NVUE */
display
:
flex
;
/* #endif */
flex-direction
:
row
;
}
.load-more
{
/* #ifndef APP-NVUE */
margin-right
:
auto
;
/* #endif */
/* #ifdef APP-NVUE */
width
:
40px
;
/* #endif */
}
.selected-list
{
/* #ifndef APP-NVUE */
display
:
flex
;
/* #endif */
flex-direction
:
row
;
flex-wrap
:
nowrap
;
/* padding: 0 5px; */
}
.selected-item
{
flex-direction
:
row
;
/* padding: 0 1px; */
/* #ifndef APP-NVUE */
white-space
:
nowrap
;
/* #endif */
}
.text-color
{
color
:
#333
;
}
.placeholder
{
color
:
grey
;
font-size
:
12px
;
}
.input-split-line
{
opacity
:
.5
;
}
.arrow-area
{
position
:
relative
;
width
:
20px
;
/* #ifndef APP-NVUE */
margin-bottom
:
5px
;
margin-left
:
auto
;
display
:
flex
;
/* #endif */
justify-content
:
center
;
transform
:
rotate
(
-45deg
);
transform-origin
:
center
;
}
.input-arrow
{
width
:
7px
;
height
:
7px
;
border-left
:
1px
solid
#999
;
border-bottom
:
1px
solid
#999
;
}
.uni-data-tree-cover
{
position
:
fixed
;
left
:
0
;
top
:
0
;
right
:
0
;
bottom
:
0
;
background-color
:
rgba
(
0
,
0
,
0
,
.4
);
/* #ifndef APP-NVUE */
display
:
flex
;
/* #endif */
flex-direction
:
column
;
z-index
:
100
;
}
.uni-data-tree-dialog
{
position
:
fixed
;
left
:
0
;
/* #ifndef APP-NVUE */
top
:
20%
;
/* #endif */
/* #ifdef APP-NVUE */
top
:
200px
;
/* #endif */
right
:
0
;
bottom
:
0
;
background-color
:
#FFFFFF
;
border-top-left-radius
:
10px
;
border-top-right-radius
:
10px
;
/* #ifndef APP-NVUE */
display
:
flex
;
/* #endif */
flex-direction
:
column
;
z-index
:
102
;
overflow
:
hidden
;
/* #ifdef APP-NVUE */
width
:
750
rpx
;
/* #endif */
}
.dialog-caption
{
position
:
relative
;
/* #ifndef APP-NVUE */
display
:
flex
;
/* #endif */
flex-direction
:
row
;
/* border-bottom: 1px solid #f0f0f0; */
}
.title-area
{
/* #ifndef APP-NVUE */
display
:
flex
;
/* #endif */
align-items
:
center
;
/* #ifndef APP-NVUE */
margin
:
auto
;
/* #endif */
padding
:
0
10px
;
}
.dialog-title
{
/* font-weight: bold; */
line-height
:
44px
;
}
.dialog-close
{
position
:
absolute
;
top
:
0
;
right
:
0
;
bottom
:
0
;
/* #ifndef APP-NVUE */
display
:
flex
;
/* #endif */
flex-direction
:
row
;
align-items
:
center
;
padding
:
0
15px
;
}
.dialog-close-plus
{
width
:
16px
;
height
:
2px
;
background-color
:
#666
;
border-radius
:
2px
;
transform
:
rotate
(
45deg
);
}
.dialog-close-rotate
{
position
:
absolute
;
transform
:
rotate
(
-45deg
);
}
.picker-view
{
flex
:
1
;
overflow
:
hidden
;
}
.icon-clear
{
display
:
flex
;
align-items
:
center
;
}
/* #ifdef H5 */
@media
all
and
(
min-width
:
768px
)
{
.uni-data-tree-cover
{
background-color
:
transparent
;
}
.uni-data-tree-dialog
{
position
:
absolute
;
top
:
55px
;
height
:
auto
;
min-height
:
400px
;
max-height
:
50vh
;
background-color
:
#fff
;
border
:
1px
solid
#EBEEF5
;
box-shadow
:
0
2px
12px
0
rgba
(
0
,
0
,
0
,
0.1
);
border-radius
:
4px
;
overflow
:
unset
;
}
.dialog-caption
{
display
:
none
;
}
.icon-clear
{
/* margin-right: 5px; */
}
}
/* #endif */
/* picker 弹出层通用的指示小三角, todo:扩展至上下左右方向定位 */
/* #ifndef APP-NVUE */
.uni-popper__arrow
,
.uni-popper__arrow
::after
{
position
:
absolute
;
display
:
block
;
width
:
0
;
height
:
0
;
border-color
:
transparent
;
border-style
:
solid
;
border-width
:
6px
;
}
.uni-popper__arrow
{
filter
:
drop-shadow
(
0
2px
12px
rgba
(
0
,
0
,
0
,
0.03
));
top
:
-6px
;
left
:
10%
;
margin-right
:
3px
;
border-top-width
:
0
;
border-bottom-color
:
#EBEEF5
;
}
.uni-popper__arrow
::after
{
content
:
" "
;
top
:
1px
;
margin-left
:
-6px
;
border-top-width
:
0
;
border-bottom-color
:
#fff
;
}
/* #endif */
</
style
>
src/uni_modules/uni-data-picker/components/uni-data-pickerview/uni-data-picker.js
0 → 100644
View file @
64e10593
export
default
{
props
:
{
localdata
:
{
type
:
[
Array
,
Object
],
default
()
{
return
[]
}
},
spaceInfo
:
{
type
:
Object
,
default
()
{
return
{}
}
},
collection
:
{
type
:
String
,
default
:
''
},
action
:
{
type
:
String
,
default
:
''
},
field
:
{
type
:
String
,
default
:
''
},
orderby
:
{
type
:
String
,
default
:
''
},
where
:
{
type
:
[
String
,
Object
],
default
:
''
},
pageData
:
{
type
:
String
,
default
:
'
add
'
},
pageCurrent
:
{
type
:
Number
,
default
:
1
},
pageSize
:
{
type
:
Number
,
default
:
500
},
getcount
:
{
type
:
[
Boolean
,
String
],
default
:
false
},
getone
:
{
type
:
[
Boolean
,
String
],
default
:
false
},
gettree
:
{
type
:
[
Boolean
,
String
],
default
:
false
},
manual
:
{
type
:
Boolean
,
default
:
false
},
value
:
{
type
:
[
Array
,
String
,
Number
],
default
()
{
return
[]
}
},
modelValue
:
{
type
:
[
Array
,
String
,
Number
],
default
()
{
return
[]
}
},
preload
:
{
type
:
Boolean
,
default
:
false
},
stepSearh
:
{
type
:
Boolean
,
default
:
true
},
selfField
:
{
type
:
String
,
default
:
''
},
parentField
:
{
type
:
String
,
default
:
''
},
multiple
:
{
type
:
Boolean
,
default
:
false
},
map
:
{
type
:
Object
,
default
()
{
return
{
text
:
"
text
"
,
value
:
"
value
"
}
}
}
},
data
()
{
return
{
loading
:
false
,
errorMessage
:
''
,
loadMore
:
{
contentdown
:
''
,
contentrefresh
:
''
,
contentnomore
:
''
},
dataList
:
[],
selected
:
[],
selectedIndex
:
0
,
page
:
{
current
:
this
.
pageCurrent
,
size
:
this
.
pageSize
,
count
:
0
}
}
},
computed
:
{
isLocalData
()
{
return
!
this
.
collection
.
length
;
},
isCloudData
()
{
return
this
.
collection
.
length
>
0
;
},
isCloudDataList
()
{
return
(
this
.
isCloudData
&&
(
!
this
.
parentField
&&
!
this
.
selfField
));
},
isCloudDataTree
()
{
return
(
this
.
isCloudData
&&
this
.
parentField
&&
this
.
selfField
);
},
dataValue
()
{
let
isModelValue
=
Array
.
isArray
(
this
.
modelValue
)
?
(
this
.
modelValue
.
length
>
0
)
:
(
this
.
modelValue
!==
null
||
this
.
modelValue
!==
undefined
);
return
isModelValue
?
this
.
modelValue
:
this
.
value
;
},
hasValue
()
{
if
(
typeof
this
.
dataValue
===
'
number
'
)
{
return
true
}
return
(
this
.
dataValue
!=
null
)
&&
(
this
.
dataValue
.
length
>
0
)
}
},
created
()
{
this
.
$watch
(()
=>
{
var
al
=
[];
[
'
pageCurrent
'
,
'
pageSize
'
,
'
spaceInfo
'
,
'
value
'
,
'
modelValue
'
,
'
localdata
'
,
'
collection
'
,
'
action
'
,
'
field
'
,
'
orderby
'
,
'
where
'
,
'
getont
'
,
'
getcount
'
,
'
gettree
'
].
forEach
(
key
=>
{
al
.
push
(
this
[
key
])
});
return
al
},
(
newValue
,
oldValue
)
=>
{
let
needReset
=
false
for
(
let
i
=
2
;
i
<
newValue
.
length
;
i
++
)
{
if
(
newValue
[
i
]
!=
oldValue
[
i
])
{
needReset
=
true
break
}
}
if
(
newValue
[
0
]
!=
oldValue
[
0
])
{
this
.
page
.
current
=
this
.
pageCurrent
}
this
.
page
.
size
=
this
.
pageSize
this
.
onPropsChange
()
})
this
.
_treeData
=
[]
},
methods
:
{
onPropsChange
()
{
this
.
_treeData
=
[];
},
// 填充 pickview 数据
async
loadData
()
{
if
(
this
.
isLocalData
)
{
this
.
loadLocalData
();
}
else
if
(
this
.
isCloudDataList
)
{
this
.
loadCloudDataList
();
}
else
if
(
this
.
isCloudDataTree
)
{
this
.
loadCloudDataTree
();
}
},
// 加载本地数据
async
loadLocalData
()
{
this
.
_treeData
=
[];
this
.
_extractTree
(
this
.
localdata
,
this
.
_treeData
);
let
inputValue
=
this
.
dataValue
;
if
(
inputValue
===
undefined
)
{
return
;
}
if
(
Array
.
isArray
(
inputValue
))
{
inputValue
=
inputValue
[
inputValue
.
length
-
1
];
if
(
typeof
inputValue
===
'
object
'
&&
inputValue
[
this
.
map
.
value
])
{
inputValue
=
inputValue
[
this
.
map
.
value
];
}
}
this
.
selected
=
this
.
_findNodePath
(
inputValue
,
this
.
localdata
);
},
// 加载 Cloud 数据 (单列)
async
loadCloudDataList
()
{
if
(
this
.
loading
)
{
return
;
}
this
.
loading
=
true
;
try
{
let
response
=
await
this
.
getCommand
();
let
responseData
=
response
.
result
.
data
;
this
.
_treeData
=
responseData
;
this
.
_updateBindData
();
this
.
_updateSelected
();
this
.
onDataChange
();
}
catch
(
e
)
{
this
.
errorMessage
=
e
;
}
finally
{
this
.
loading
=
false
;
}
},
// 加载 Cloud 数据 (树形)
async
loadCloudDataTree
()
{
if
(
this
.
loading
)
{
return
;
}
this
.
loading
=
true
;
try
{
let
commandOptions
=
{
field
:
this
.
_cloudDataPostField
(),
where
:
this
.
_cloudDataTreeWhere
()
};
if
(
this
.
gettree
)
{
commandOptions
.
startwith
=
`
${
this
.
selfField
}
=='
${
this
.
dataValue
}
'`
;
}
let
response
=
await
this
.
getCommand
(
commandOptions
);
let
responseData
=
response
.
result
.
data
;
this
.
_treeData
=
responseData
;
this
.
_updateBindData
();
this
.
_updateSelected
();
this
.
onDataChange
();
}
catch
(
e
)
{
this
.
errorMessage
=
e
;
}
finally
{
this
.
loading
=
false
;
}
},
// 加载 Cloud 数据 (节点)
async
loadCloudDataNode
(
callback
)
{
if
(
this
.
loading
)
{
return
;
}
this
.
loading
=
true
;
try
{
let
commandOptions
=
{
field
:
this
.
_cloudDataPostField
(),
where
:
this
.
_cloudDataNodeWhere
()
};
let
response
=
await
this
.
getCommand
(
commandOptions
);
let
responseData
=
response
.
result
.
data
;
callback
(
responseData
);
}
catch
(
e
)
{
this
.
errorMessage
=
e
;
}
finally
{
this
.
loading
=
false
;
}
},
// 回显 Cloud 数据
getCloudDataValue
()
{
if
(
this
.
isCloudDataList
)
{
return
this
.
getCloudDataListValue
();
}
if
(
this
.
isCloudDataTree
)
{
return
this
.
getCloudDataTreeValue
();
}
},
// 回显 Cloud 数据 (单列)
getCloudDataListValue
()
{
// 根据 field's as value标识匹配 where 条件
let
where
=
[];
let
whereField
=
this
.
_getForeignKeyByField
();
if
(
whereField
)
{
where
.
push
(
`
${
whereField
}
== '
${
this
.
dataValue
}
'`
)
}
where
=
where
.
join
(
'
||
'
);
if
(
this
.
where
)
{
where
=
`(
${
this
.
where
}
) && (
${
where
}
)`
}
return
this
.
getCommand
({
field
:
this
.
_cloudDataPostField
(),
where
}).
then
((
res
)
=>
{
this
.
selected
=
res
.
result
.
data
;
return
res
.
result
.
data
;
});
},
// 回显 Cloud 数据 (树形)
getCloudDataTreeValue
()
{
return
this
.
getCommand
({
field
:
this
.
_cloudDataPostField
(),
getTreePath
:
{
startWith
:
`
${
this
.
selfField
}
=='
${
this
.
dataValue
}
'`
}
}).
then
((
res
)
=>
{
let
treePath
=
[];
this
.
_extractTreePath
(
res
.
result
.
data
,
treePath
);
this
.
selected
=
treePath
;
return
treePath
;
});
},
getCommand
(
options
=
{})
{
/* eslint-disable no-undef */
let
db
=
uniCloud
.
database
(
this
.
spaceInfo
)
const
action
=
options
.
action
||
this
.
action
if
(
action
)
{
db
=
db
.
action
(
action
)
}
const
collection
=
options
.
collection
||
this
.
collection
db
=
db
.
collection
(
collection
)
const
where
=
options
.
where
||
this
.
where
if
(
!
(
!
where
||
!
Object
.
keys
(
where
).
length
))
{
db
=
db
.
where
(
where
)
}
const
field
=
options
.
field
||
this
.
field
if
(
field
)
{
db
=
db
.
field
(
field
)
}
const
orderby
=
options
.
orderby
||
this
.
orderby
if
(
orderby
)
{
db
=
db
.
orderBy
(
orderby
)
}
const
current
=
options
.
pageCurrent
!==
undefined
?
options
.
pageCurrent
:
this
.
page
.
current
const
size
=
options
.
pageSize
!==
undefined
?
options
.
pageSize
:
this
.
page
.
size
const
getCount
=
options
.
getcount
!==
undefined
?
options
.
getcount
:
this
.
getcount
const
getTree
=
options
.
gettree
!==
undefined
?
options
.
gettree
:
this
.
gettree
const
getOptions
=
{
getCount
,
getTree
}
if
(
options
.
getTreePath
)
{
getOptions
.
getTreePath
=
options
.
getTreePath
}
db
=
db
.
skip
(
size
*
(
current
-
1
)).
limit
(
size
).
get
(
getOptions
)
return
db
},
_cloudDataPostField
()
{
let
fields
=
[
this
.
field
];
if
(
this
.
parentField
)
{
fields
.
push
(
`
${
this
.
parentField
}
as parent_value`
);
}
return
fields
.
join
(
'
,
'
);
},
_cloudDataTreeWhere
()
{
let
result
=
[]
let
selected
=
this
.
selected
let
parentField
=
this
.
parentField
if
(
parentField
)
{
result
.
push
(
`
${
parentField
}
== null ||
${
parentField
}
== ""`
)
}
if
(
selected
.
length
)
{
for
(
var
i
=
0
;
i
<
selected
.
length
-
1
;
i
++
)
{
result
.
push
(
`
${
parentField
}
== '
${
selected
[
i
].
value
}
'`
)
}
}
let
where
=
[]
if
(
this
.
where
)
{
where
.
push
(
`(
${
this
.
where
}
)`
)
}
if
(
result
.
length
)
{
where
.
push
(
`(
${
result
.
join
(
'
||
'
)}
)`
)
}
return
where
.
join
(
'
&&
'
)
},
_cloudDataNodeWhere
()
{
let
where
=
[]
let
selected
=
this
.
selected
;
if
(
selected
.
length
)
{
where
.
push
(
`
${
this
.
parentField
}
== '
${
selected
[
selected
.
length
-
1
].
value
}
'`
);
}
where
=
where
.
join
(
'
||
'
);
if
(
this
.
where
)
{
return
`(
${
this
.
where
}
) && (
${
where
}
)`
}
return
where
},
_getWhereByForeignKey
()
{
let
result
=
[]
let
whereField
=
this
.
_getForeignKeyByField
();
if
(
whereField
)
{
result
.
push
(
`
${
whereField
}
== '
${
this
.
dataValue
}
'`
)
}
if
(
this
.
where
)
{
return
`(
${
this
.
where
}
) && (
${
result
.
join
(
'
||
'
)}
)`
}
return
result
.
join
(
'
||
'
)
},
_getForeignKeyByField
()
{
let
fields
=
this
.
field
.
split
(
'
,
'
);
let
whereField
=
null
;
for
(
let
i
=
0
;
i
<
fields
.
length
;
i
++
)
{
const
items
=
fields
[
i
].
split
(
'
as
'
);
if
(
items
.
length
<
2
)
{
continue
;
}
if
(
items
[
1
].
trim
()
===
'
value
'
)
{
whereField
=
items
[
0
].
trim
();
break
;
}
}
return
whereField
;
},
_updateBindData
(
node
)
{
const
{
dataList
,
hasNodes
}
=
this
.
_filterData
(
this
.
_treeData
,
this
.
selected
)
let
isleaf
=
this
.
_stepSearh
===
false
&&
!
hasNodes
if
(
node
)
{
node
.
isleaf
=
isleaf
}
this
.
dataList
=
dataList
this
.
selectedIndex
=
dataList
.
length
-
1
if
(
!
isleaf
&&
this
.
selected
.
length
<
dataList
.
length
)
{
this
.
selected
.
push
({
value
:
null
,
text
:
"
请选择
"
})
}
return
{
isleaf
,
hasNodes
}
},
_updateSelected
()
{
let
dl
=
this
.
dataList
let
sl
=
this
.
selected
let
textField
=
this
.
map
.
text
let
valueField
=
this
.
map
.
value
for
(
let
i
=
0
;
i
<
sl
.
length
;
i
++
)
{
let
value
=
sl
[
i
].
value
let
dl2
=
dl
[
i
]
for
(
let
j
=
0
;
j
<
dl2
.
length
;
j
++
)
{
let
item2
=
dl2
[
j
]
if
(
item2
[
valueField
]
===
value
)
{
sl
[
i
].
text
=
item2
[
textField
]
break
}
}
}
},
_filterData
(
data
,
paths
)
{
let
dataList
=
[]
let
hasNodes
=
true
dataList
.
push
(
data
.
filter
((
item
)
=>
{
return
(
item
.
parent_value
===
null
||
item
.
parent_value
===
undefined
||
item
.
parent_value
===
''
)
}))
for
(
let
i
=
0
;
i
<
paths
.
length
;
i
++
)
{
let
value
=
paths
[
i
].
value
let
nodes
=
data
.
filter
((
item
)
=>
{
return
item
.
parent_value
===
value
})
if
(
nodes
.
length
)
{
dataList
.
push
(
nodes
)
}
else
{
hasNodes
=
false
}
}
return
{
dataList
,
hasNodes
}
},
_extractTree
(
nodes
,
result
,
parent_value
)
{
let
list
=
result
||
[]
let
valueField
=
this
.
map
.
value
for
(
let
i
=
0
;
i
<
nodes
.
length
;
i
++
)
{
let
node
=
nodes
[
i
]
let
child
=
{}
for
(
let
key
in
node
)
{
if
(
key
!==
'
children
'
)
{
child
[
key
]
=
node
[
key
]
}
}
if
(
parent_value
!==
null
&&
parent_value
!==
undefined
&&
parent_value
!==
''
)
{
child
.
parent_value
=
parent_value
}
result
.
push
(
child
)
let
children
=
node
.
children
if
(
children
)
{
this
.
_extractTree
(
children
,
result
,
node
[
valueField
])
}
}
},
_extractTreePath
(
nodes
,
result
)
{
let
list
=
result
||
[]
for
(
let
i
=
0
;
i
<
nodes
.
length
;
i
++
)
{
let
node
=
nodes
[
i
]
let
child
=
{}
for
(
let
key
in
node
)
{
if
(
key
!==
'
children
'
)
{
child
[
key
]
=
node
[
key
]
}
}
result
.
push
(
child
)
let
children
=
node
.
children
if
(
children
)
{
this
.
_extractTreePath
(
children
,
result
)
}
}
},
_findNodePath
(
key
,
nodes
,
path
=
[])
{
let
textField
=
this
.
map
.
text
let
valueField
=
this
.
map
.
value
for
(
let
i
=
0
;
i
<
nodes
.
length
;
i
++
)
{
let
node
=
nodes
[
i
]
let
children
=
node
.
children
let
text
=
node
[
textField
]
let
value
=
node
[
valueField
]
path
.
push
({
value
,
text
})
if
(
value
===
key
)
{
return
path
}
if
(
children
)
{
const
p
=
this
.
_findNodePath
(
key
,
children
,
path
)
if
(
p
.
length
)
{
return
p
}
}
path
.
pop
()
}
return
[]
}
}
}
src/uni_modules/uni-data-picker/components/uni-data-pickerview/uni-data-picker.uts
0 → 100644
View file @
64e10593
export type PaginationType = {
current : number,
size : number,
count : number
}
export type LoadMoreType = {
contentdown : string,
contentrefresh : string,
contentnomore : string
}
export type SelectedItemType = {
name : string,
value : string,
}
export type GetCommandOptions = {
collection ?: UTSJSONObject,
field ?: string,
orderby ?: string,
where ?: any,
pageData ?: string,
pageCurrent ?: number,
pageSize ?: number,
getCount ?: boolean,
getTree ?: any,
getTreePath ?: UTSJSONObject,
startwith ?: string,
limitlevel ?: number,
groupby ?: string,
groupField ?: string,
distinct ?: boolean,
pageIndistinct ?: boolean,
foreignKey ?: string,
loadtime ?: string,
manual ?: boolean
}
const DefaultSelectedNode = {
text: '请选择',
value: ''
}
export const dataPicker = defineMixin({
props: {
localdata: {
type: Array as PropType<Array<UTSJSONObject>>,
default: [] as Array<UTSJSONObject>
},
collection: {
type: Object,
default: ''
},
field: {
type: String,
default: ''
},
orderby: {
type: String,
default: ''
},
where: {
type: Object,
default: ''
},
pageData: {
type: String,
default: 'add'
},
pageCurrent: {
type: Number,
default: 1
},
pageSize: {
type: Number,
default: 20
},
getcount: {
type: Boolean,
default: false
},
gettree: {
type: Object,
default: ''
},
gettreepath: {
type: Object,
default: ''
},
startwith: {
type: String,
default: ''
},
limitlevel: {
type: Number,
default: 10
},
groupby: {
type: String,
default: ''
},
groupField: {
type: String,
default: ''
},
distinct: {
type: Boolean,
default: false
},
pageIndistinct: {
type: Boolean,
default: false
},
foreignKey: {
type: String,
default: ''
},
loadtime: {
type: String,
default: 'auto'
},
manual: {
type: Boolean,
default: false
},
preload: {
type: Boolean,
default: false
},
stepSearh: {
type: Boolean,
default: true
},
selfField: {
type: String,
default: ''
},
parentField: {
type: String,
default: ''
},
multiple: {
type: Boolean,
default: false
},
value: {
type: Object,
default: ''
},
modelValue: {
type: Object,
default: ''
},
defaultProps: {
type: Object as PropType<UTSJSONObject>,
}
},
data() {
return {
loading: false,
error: null as UniCloudError | null,
treeData: [] as Array<UTSJSONObject>,
selectedIndex: 0,
selectedNodes: [] as Array<UTSJSONObject>,
selectedPages: [] as Array<UTSJSONObject>[],
selectedValue: '',
selectedPaths: [] as Array<UTSJSONObject>,
pagination: {
current: 1,
size: 20,
count: 0
} as PaginationType
}
},
computed: {
mappingTextName() : string {
// TODO
return (this.defaultProps != null) ? this.defaultProps!.getString('text', 'text') : 'text'
},
mappingValueName() : string {
// TODO
return (this.defaultProps != null) ? this.defaultProps!.getString('value', 'value') : 'value'
},
currentDataList() : Array<UTSJSONObject> {
if (this.selectedIndex > this.selectedPages.length - 1) {
return [] as Array<UTSJSONObject>
}
return this.selectedPages[this.selectedIndex]
},
isLocalData() : boolean {
return this.localdata.length > 0
},
isCloudData() : boolean {
return this._checkIsNotNull(this.collection)
},
isCloudDataList() : boolean {
return (this.isCloudData && (this.parentField.length == 0 && this.selfField.length == 0))
},
isCloudDataTree() : boolean {
return (this.isCloudData && this.parentField.length > 0 && this.selfField.length > 0)
},
dataValue() : any {
return this.hasModelValue ? this.modelValue : this.value
},
hasCloudTreeData() : boolean {
return this.treeData.length > 0
},
hasModelValue() : boolean {
if (typeof this.modelValue == 'string') {
const valueString = this.modelValue as string
return (valueString.length > 0)
} else if (Array.isArray(this.modelValue)) {
const valueArray = this.modelValue as Array<string>
return (valueArray.length > 0)
}
return false
},
hasCloudDataValue() : boolean {
if (typeof this.dataValue == 'string') {
const valueString = this.dataValue as string
return (valueString.length > 0)
}
return false
}
},
created() {
this.pagination.current = this.pageCurrent
this.pagination.size = this.pageSize
this.$watch(
() : any => [
this.pageCurrent,
this.pageSize,
this.localdata,
this.value,
this.collection,
this.field,
this.getcount,
this.orderby,
this.where,
this.groupby,
this.groupField,
this.distinct
],
(newValue : Array<any>, oldValue : Array<any>) => {
this.pagination.size = this.pageSize
if (newValue[0] !== oldValue[0]) {
this.pagination.current = this.pageCurrent
}
this.onPropsChange()
}
)
},
methods: {
onPropsChange() {
this.selectedIndex = 0
this.selectedNodes.length = 0
this.selectedPages.length = 0
this.selectedPaths.length = 0
// 加载数据
this.$nextTick(() => {
this.loadData()
})
},
onTabSelect(index : number) {
this.selectedIndex = index
},
onNodeClick(nodeData : UTSJSONObject) {
if (nodeData.getBoolean('disable', false)) {
return
}
const isLeaf = this._checkIsLeafNode(nodeData)
this._trimSelectedNodes(nodeData)
this.$emit('nodeclick', nodeData)
if (this.isLocalData) {
if (isLeaf || !this._checkHasChildren(nodeData)) {
this.onFinish()
}
} else if (this.isCloudDataList) {
this.onFinish()
} else if (this.isCloudDataTree) {
if (isLeaf) {
this.onFinish()
} else if (!this._checkHasChildren(nodeData)) {
// 尝试请求一次,如果没有返回数据标记为叶子节点
this.loadCloudDataNode(nodeData)
}
}
},
getChangeNodes(): Array<UTSJSONObject> {
const nodes: Array<UTSJSONObject> = []
this.selectedNodes.forEach((node : UTSJSONObject) => {
const newNode: UTSJSONObject = {}
newNode[this.mappingTextName] = node.getString(this.mappingTextName)
newNode[this.mappingValueName] = node.getString(this.mappingValueName)
nodes.push(newNode)
})
return nodes
},
onFinish() { },
// 加载数据(自动判定环境)
loadData() {
if (this.isLocalData) {
this.loadLocalData()
} else if (this.isCloudDataList) {
this.loadCloudDataList()
} else if (this.isCloudDataTree) {
this.loadCloudDataTree()
}
},
// 加载本地数据
loadLocalData() {
this.treeData = this.localdata
if (Array.isArray(this.dataValue)) {
const value = this.dataValue as Array<UTSJSONObject>
this.selectedPaths = value.slice(0)
this._pushSelectedTreeNodes(value, this.localdata)
} else {
this._pushSelectedNodes(this.localdata)
}
},
// 加载 Cloud 数据 (单列)
loadCloudDataList() {
this._loadCloudData(null, (data : Array<UTSJSONObject>) => {
this.treeData = data
this._pushSelectedNodes(data)
})
},
// 加载 Cloud 数据 (树形)
loadCloudDataTree() {
let commandOptions = {
field: this._cloudDataPostField(),
where: this._cloudDataTreeWhere(),
getTree: true
} as GetCommandOptions
if (this._checkIsNotNull(this.gettree)) {
commandOptions.startwith = `${this.selfField}=='${this.dataValue as string}'`
}
this._loadCloudData(commandOptions, (data : Array<UTSJSONObject>) => {
this.treeData = data
if (this.selectedPaths.length > 0) {
this._pushSelectedTreeNodes(this.selectedPaths, data)
} else {
this._pushSelectedNodes(data)
}
})
},
// 加载 Cloud 数据 (节点)
loadCloudDataNode(nodeData : UTSJSONObject) {
const commandOptions = {
field: this._cloudDataPostField(),
where: this._cloudDataNodeWhere()
} as GetCommandOptions
this._loadCloudData(commandOptions, (data : Array<UTSJSONObject>) => {
nodeData['children'] = data
if (data.length == 0) {
nodeData['isleaf'] = true
this.onFinish()
} else {
this._pushSelectedNodes(data)
}
})
},
// 回显 Cloud Tree Path
loadCloudDataPath() {
if (!this.hasCloudDataValue) {
return
}
const command : GetCommandOptions = {}
// 单列
if (this.isCloudDataList) {
// 根据 field's as value标识匹配 where 条件
let where : Array<string> = [];
let whereField = this._getForeignKeyByField();
if (whereField.length > 0) {
where.push(`${whereField} == '${this.dataValue as string}'`)
}
let whereString = where.join(' || ')
if (this._checkIsNotNull(this.where)) {
whereString = `(${this.where}) && (${whereString})`
}
command.field = this._cloudDataPostField()
command.where = whereString
}
// 树形
if (this.isCloudDataTree) {
command.field = this._cloudDataPostField()
command.getTreePath = {
startWith: `${this.selfField}=='${this.dataValue as string}'`
}
}
this._loadCloudData(command, (data : Array<UTSJSONObject>) => {
this._extractTreePath(data, this.selectedPaths)
})
},
_loadCloudData(options ?: GetCommandOptions, callback ?: ((data : Array<UTSJSONObject>) => void)) {
if (this.loading) {
return
}
this.loading = true
this.error = null
this._getCommand(options).then((response : UniCloudDBGetResult) => {
callback?.(response.data)
}).catch((err : any | null) => {
this.error = err as UniCloudError
}).finally(() => {
this.loading = false
})
},
_cloudDataPostField() : string {
let fields = [this.field];
if (this.parentField.length > 0) {
fields.push(`${this.parentField} as parent_value`)
}
return fields.join(',')
},
_cloudDataTreeWhere() : string {
let result : Array<string> = []
let selectedNodes = this.selectedNodes.length > 0 ? this.selectedNodes : this.selectedPaths
let parentField = this.parentField
if (parentField.length > 0) {
result.push(`${parentField} == null || ${parentField} == ""`)
}
if (selectedNodes.length > 0) {
for (var i = 0; i < selectedNodes.length - 1; i++) {
const parentFieldValue = selectedNodes[i].getString('value', '')
result.push(`${parentField} == '${parentFieldValue}'`)
}
}
let where : Array<string> = []
if (this._checkIsNotNull(this.where)) {
where.push(`(${this.where as string})`)
}
if (result.length > 0) {
where.push(`(${result.join(' || ')})`)
}
return where.join(' && ')
},
_cloudDataNodeWhere() : string {
const where : Array<string> = []
if (this.selectedNodes.length > 0) {
const value = this.selectedNodes[this.selectedNodes.length - 1].getString('value', '')
where.push(`${this.parentField} == '${value}'`)
}
let whereString = where.join(' || ')
if (this._checkIsNotNull(this.where)) {
return `(${this.where as string}) && (${whereString})`
}
return whereString
},
_getWhereByForeignKey() : string {
let result : Array<string> = []
let whereField = this._getForeignKeyByField();
if (whereField.length > 0) {
result.push(`${whereField} == '${this.dataValue as string}'`)
}
if (this._checkIsNotNull(this.where)) {
return `(${this.where}) && (${result.join(' || ')})`
}
return result.join(' || ')
},
_getForeignKeyByField() : string {
const fields = this.field.split(',')
let whereField = ''
for (let i = 0; i < fields.length; i++) {
const items = fields[i].split('as')
if (items.length < 2) {
continue
}
if (items[1].trim() === 'value') {
whereField = items[0].trim()
break
}
}
return whereField
},
_getCommand(options ?: GetCommandOptions) : Promise<UniCloudDBGetResult> {
let db = uniCloud.databaseForJQL()
let collection = Array.isArray(this.collection) ? db.collection(...(this.collection as Array<any>)) : db.collection(this.collection)
let filter : UniCloudDBFilter | null = null
if (this.foreignKey.length > 0) {
filter = collection.foreignKey(this.foreignKey)
}
const where : any = options?.where ?? this.where
if (typeof where == 'string') {
const whereString = where as string
if (whereString.length > 0) {
filter = (filter != null) ? filter.where(where) : collection.where(where)
}
} else {
filter = (filter != null) ? filter.where(where) : collection.where(where)
}
let query : UniCloudDBQuery | null = null
if (this.field.length > 0) {
query = (filter != null) ? filter.field(this.field) : collection.field(this.field)
}
if (this.groupby.length > 0) {
if (query != null) {
query = query.groupBy(this.groupby)
} else if (filter != null) {
query = filter.groupBy(this.groupby)
}
}
if (this.groupField.length > 0) {
if (query != null) {
query = query.groupField(this.groupField)
} else if (filter != null) {
query = filter.groupField(this.groupField)
}
}
if (this.distinct == true) {
if (query != null) {
query = query.distinct(this.field)
} else if (filter != null) {
query = filter.distinct(this.field)
}
}
if (this.orderby.length > 0) {
if (query != null) {
query = query.orderBy(this.orderby)
} else if (filter != null) {
query = filter.orderBy(this.orderby)
}
}
const size = this.pagination.size
const current = this.pagination.current
if (query != null) {
query = query.skip(size * (current - 1)).limit(size)
} else if (filter != null) {
query = filter.skip(size * (current - 1)).limit(size)
} else {
query = collection.skip(size * (current - 1)).limit(size)
}
const getOptions = {}
const treeOptions = {
limitLevel: this.limitlevel,
startWith: this.startwith
}
if (this.getcount == true) {
getOptions['getCount'] = this.getcount
}
const getTree : any = options?.getTree ?? this.gettree
if (typeof getTree == 'string') {
const getTreeString = getTree as string
if (getTreeString.length > 0) {
getOptions['getTree'] = treeOptions
}
} else if (typeof getTree == 'object') {
getOptions['getTree'] = treeOptions
} else {
getOptions['getTree'] = getTree
}
const getTreePath = options?.getTreePath ?? this.gettreepath
if (typeof getTreePath == 'string') {
const getTreePathString = getTreePath as string
if (getTreePathString.length > 0) {
getOptions['getTreePath'] = getTreePath
}
} else {
getOptions['getTreePath'] = getTreePath
}
return query.get(getOptions)
},
_checkIsNotNull(value : any) : boolean {
if (typeof value == 'string') {
const valueString = value as string
return (valueString.length > 0)
} else if (value instanceof UTSJSONObject) {
return true
}
return false
},
_checkIsLeafNode(nodeData : UTSJSONObject) : boolean {
if (this.selectedIndex >= this.limitlevel) {
return true
}
if (nodeData.getBoolean('isleaf', false)) {
return true
}
return false
},
_checkHasChildren(nodeData : UTSJSONObject) : boolean {
const children = nodeData.getArray('children') ?? ([] as Array<any>)
return children.length > 0
},
_pushSelectedNodes(nodes : Array<UTSJSONObject>) {
this.selectedNodes.push(DefaultSelectedNode)
this.selectedPages.push(nodes)
this.selectedIndex = this.selectedPages.length - 1
},
_trimSelectedNodes(nodeData : UTSJSONObject) {
this.selectedNodes.splice(this.selectedIndex)
this.selectedNodes.push(nodeData)
if (this.selectedPages.length > 0) {
this.selectedPages.splice(this.selectedIndex + 1)
}
const children = nodeData.getArray<UTSJSONObject>('children') ?? ([] as Array<UTSJSONObject>)
if (children.length > 0) {
this.selectedNodes.push(DefaultSelectedNode)
this.selectedPages.push(children)
}
this.selectedIndex = this.selectedPages.length - 1
},
_pushSelectedTreeNodes(paths : Array<UTSJSONObject>, nodes : Array<UTSJSONObject>) {
let children : Array<UTSJSONObject> = nodes
paths.forEach((node : UTSJSONObject) => {
const findNode = children.find((item : UTSJSONObject) : boolean => {
return (item.getString(this.mappingValueName) == node.getString(this.mappingValueName))
})
if (findNode != null) {
this.selectedPages.push(children)
this.selectedNodes.push(node)
children = findNode.getArray<UTSJSONObject>('children') ?? ([] as Array<UTSJSONObject>)
}
})
this.selectedIndex = this.selectedPages.length - 1
},
_extractTreePath(nodes : Array<UTSJSONObject>, result : Array<UTSJSONObject>) {
if (nodes.length == 0) {
return
}
const node = nodes[0]
result.push(node)
const children = node.getArray<UTSJSONObject>('children')
if (Array.isArray(children) && children!.length > 0) {
this._extractTreePath(children, result)
}
}
}
})
src/uni_modules/uni-data-picker/components/uni-data-pickerview/uni-data-pickerview.css
0 → 100644
View file @
64e10593
.uni-data-pickerview
{
position
:
relative
;
flex-direction
:
column
;
overflow
:
hidden
;
}
.loading-cover
{
position
:
absolute
;
left
:
0
;
top
:
0
;
right
:
0
;
bottom
:
0
;
align-items
:
center
;
justify-content
:
center
;
background-color
:
rgba
(
150
,
150
,
150
,
.1
);
}
.error
{
background-color
:
#fff
;
padding
:
15px
;
}
.error-text
{
color
:
#DD524D
;
}
.selected-node-list
{
flex-direction
:
row
;
flex-wrap
:
nowrap
;
}
.selected-node-item
{
margin-left
:
10px
;
margin-right
:
10px
;
padding
:
8px
10px
8px
10px
;
border-bottom
:
2px
solid
transparent
;
}
.selected-node-item-active
{
color
:
#007aff
;
border-bottom-color
:
#007aff
;
}
.list-view
{
flex
:
1
;
}
.list-item
{
flex-direction
:
row
;
justify-content
:
space-between
;
padding
:
12px
15px
;
border-bottom
:
1px
solid
#f0f0f0
;
}
.item-text
{
color
:
#333333
;
}
.item-text-disabled
{
opacity
:
.5
;
}
.item-text-overflow
{
overflow
:
hidden
;
}
.check
{
margin-right
:
5px
;
border
:
2px
solid
#007aff
;
border-left
:
0
;
border-top
:
0
;
height
:
12px
;
width
:
6px
;
transform-origin
:
center
;
transform
:
rotate
(
45deg
);
}
src/uni_modules/uni-data-picker/components/uni-data-pickerview/uni-data-pickerview.uvue
0 → 100644
View file @
64e10593
<template>
<view class="uni-data-pickerview">
<view v-if="error!=null" class="error">
<text class="error-text">{{error!.errMsg}}</text>
</view>
<scroll-view v-if="!isCloudDataList" :scroll-x="true">
<view class="selected-node-list">
<template v-for="(item, index) in selectedNodes">
<text class="selected-node-item" :class="{'selected-node-item-active':index==selectedIndex}"
@click="onTabSelect(index)">
{{item[mappingTextName]}}
</text>
</template>
</view>
</scroll-view>
<list-view class="list-view" :scroll-y="true">
<list-item class="list-item" v-for="(item, _) in currentDataList" @click="onNodeClick(item)">
<text class="item-text" :class="{'item-text-disabled': item['disable']}">{{item[mappingTextName]}}</text>
<text class="check" v-if="item[mappingValueName] == selectedNodes[selectedIndex][mappingValueName]"></text>
</list-item>
</list-view>
<view class="loading-cover" v-if="loading">
<slot name="pickerview-loading" :loading="loading"></slot>
</view>
</view>
</template>
<script>
import { dataPicker } from "./uni-data-picker.uts"
/**
* DataPickerview
* @description uni-data-pickerview
* @tutorial https://ext.dcloud.net.cn/plugin?id=3796
* @property {Array} localdata 本地数据,参考
* @property {Boolean} step-searh = [true|false] 是否分布查询
* @value true 启用分布查询,仅查询当前选中节点
* @value false 关闭分布查询,一次查询出所有数据
* @property {String|DBFieldString} self-field 分布查询当前字段名称
* @property {String|DBFieldString} parent-field 分布查询父字段名称
* @property {String|DBCollectionString} collection 表名
* @property {String|DBFieldString} field 查询字段,多个字段用 `,` 分割
* @property {String} orderby 排序字段及正序倒叙设置
* @property {String|JQLString} where 查询条件
*/
export default {
name: 'UniDataPickerView',
emits: ['nodeclick', 'change', 'update:modelValue'],
mixins: [dataPicker],
props: {
ellipsis: {
type: Boolean,
default: true
}
},
created() {
this.loadData()
},
methods: {
onFinish() {
this.$emit('change', this.getChangeNodes())
}
}
}
</script>
<style>
@import url("uni-data-pickerview.css");
</style>
src/uni_modules/uni-data-picker/components/uni-data-pickerview/uni-data-pickerview.vue
0 → 100644
View file @
64e10593
<
template
>
<view
class=
"uni-data-pickerview"
>
<scroll-view
v-if=
"!isCloudDataList"
class=
"selected-area"
scroll-x=
"true"
>
<view
class=
"selected-list"
>
<view
class=
"selected-item"
v-for=
"(item,index) in selected"
:key=
"index"
:class=
"
{
'selected-item-active':index == selectedIndex
}"
@click="handleSelect(index)"
>
<text>
{{
item
.
text
||
''
}}
</text>
</view>
</view>
</scroll-view>
<view
class=
"tab-c"
>
<scroll-view
class=
"list"
:scroll-y=
"true"
>
<view
class=
"item"
:class=
"
{'is-disabled': !!item.disable}" v-for="(item, j) in dataList[selectedIndex]" :key="j"
@click="handleNodeClick(item, selectedIndex, j)">
<text
class=
"item-text"
>
{{
item
[
map
.
text
]
}}
</text>
<view
class=
"check"
v-if=
"selected.length > selectedIndex && item[map.value] == selected[selectedIndex].value"
></view>
</view>
</scroll-view>
<view
class=
"loading-cover"
v-if=
"loading"
>
<uni-load-more
class=
"load-more"
:contentText=
"loadMore"
status=
"loading"
></uni-load-more>
</view>
<view
class=
"error-message"
v-if=
"errorMessage"
>
<text
class=
"error-text"
>
{{
errorMessage
}}
</text>
</view>
</view>
</view>
</
template
>
<
script
>
import
dataPicker
from
"
./uni-data-picker.js
"
/**
* DataPickerview
* @description uni-data-pickerview
* @tutorial https://ext.dcloud.net.cn/plugin?id=3796
* @property {Array} localdata 本地数据,参考
* @property {Boolean} step-searh = [true|false] 是否分布查询
* @value true 启用分布查询,仅查询当前选中节点
* @value false 关闭分布查询,一次查询出所有数据
* @property {String|DBFieldString} self-field 分布查询当前字段名称
* @property {String|DBFieldString} parent-field 分布查询父字段名称
* @property {String|DBCollectionString} collection 表名
* @property {String|DBFieldString} field 查询字段,多个字段用 `,` 分割
* @property {String} orderby 排序字段及正序倒叙设置
* @property {String|JQLString} where 查询条件
*/
export
default
{
name
:
'
UniDataPickerView
'
,
emits
:
[
'
nodeclick
'
,
'
change
'
,
'
datachange
'
,
'
update:modelValue
'
],
mixins
:
[
dataPicker
],
props
:
{
managedMode
:
{
type
:
Boolean
,
default
:
false
},
ellipsis
:
{
type
:
Boolean
,
default
:
true
}
},
created
()
{
if
(
!
this
.
managedMode
)
{
this
.
$nextTick
(()
=>
{
this
.
loadData
();
})
}
},
methods
:
{
onPropsChange
()
{
this
.
_treeData
=
[];
this
.
selectedIndex
=
0
;
this
.
$nextTick
(()
=>
{
this
.
loadData
();
})
},
handleSelect
(
index
)
{
this
.
selectedIndex
=
index
;
},
handleNodeClick
(
item
,
i
,
j
)
{
if
(
item
.
disable
)
{
return
;
}
const
node
=
this
.
dataList
[
i
][
j
];
const
text
=
node
[
this
.
map
.
text
];
const
value
=
node
[
this
.
map
.
value
];
if
(
i
<
this
.
selected
.
length
-
1
)
{
this
.
selected
.
splice
(
i
,
this
.
selected
.
length
-
i
)
this
.
selected
.
push
({
text
,
value
})
}
else
if
(
i
===
this
.
selected
.
length
-
1
)
{
this
.
selected
.
splice
(
i
,
1
,
{
text
,
value
})
}
if
(
node
.
isleaf
)
{
this
.
onSelectedChange
(
node
,
node
.
isleaf
)
return
}
const
{
isleaf
,
hasNodes
}
=
this
.
_updateBindData
()
// 本地数据
if
(
this
.
isLocalData
)
{
this
.
onSelectedChange
(
node
,
(
!
hasNodes
||
isleaf
))
}
else
if
(
this
.
isCloudDataList
)
{
// Cloud 数据 (单列)
this
.
onSelectedChange
(
node
,
true
)
}
else
if
(
this
.
isCloudDataTree
)
{
// Cloud 数据 (树形)
if
(
isleaf
)
{
this
.
onSelectedChange
(
node
,
node
.
isleaf
)
}
else
if
(
!
hasNodes
)
{
// 请求一次服务器以确定是否为叶子节点
this
.
loadCloudDataNode
((
data
)
=>
{
if
(
!
data
.
length
)
{
node
.
isleaf
=
true
}
else
{
this
.
_treeData
.
push
(...
data
)
this
.
_updateBindData
(
node
)
}
this
.
onSelectedChange
(
node
,
node
.
isleaf
)
})
}
}
},
updateData
(
data
)
{
this
.
_treeData
=
data
.
treeData
this
.
selected
=
data
.
selected
if
(
!
this
.
_treeData
.
length
)
{
this
.
loadData
()
}
else
{
//this.selected = data.selected
this
.
_updateBindData
()
}
},
onDataChange
()
{
this
.
$emit
(
'
datachange
'
);
},
onSelectedChange
(
node
,
isleaf
)
{
if
(
isleaf
)
{
this
.
_dispatchEvent
()
}
if
(
node
)
{
this
.
$emit
(
'
nodeclick
'
,
node
)
}
},
_dispatchEvent
()
{
this
.
$emit
(
'
change
'
,
this
.
selected
.
slice
(
0
))
}
}
}
</
script
>
<
style
lang=
"scss"
>
$uni-primary
:
#007aff
!
default
;
.uni-data-pickerview
{
flex
:
1
;
/* #ifndef APP-NVUE */
display
:
flex
;
/* #endif */
flex-direction
:
column
;
overflow
:
hidden
;
height
:
100%
;
}
.error-text
{
color
:
#DD524D
;
}
.loading-cover
{
position
:
absolute
;
left
:
0
;
top
:
0
;
right
:
0
;
bottom
:
0
;
background-color
:
rgba
(
255
,
255
,
255
,
.5
);
/* #ifndef APP-NVUE */
display
:
flex
;
/* #endif */
flex-direction
:
column
;
align-items
:
center
;
z-index
:
1001
;
}
.load-more
{
/* #ifndef APP-NVUE */
margin
:
auto
;
/* #endif */
}
.error-message
{
background-color
:
#fff
;
position
:
absolute
;
left
:
0
;
top
:
0
;
right
:
0
;
bottom
:
0
;
padding
:
15px
;
opacity
:
.9
;
z-index
:
102
;
}
/* #ifdef APP-NVUE */
.selected-area
{
width
:
750rpx
;
}
/* #endif */
.selected-list
{
/* #ifndef APP-NVUE */
display
:
flex
;
flex-wrap
:
nowrap
;
/* #endif */
flex-direction
:
row
;
padding
:
0
5px
;
border-bottom
:
1px
solid
#f8f8f8
;
}
.selected-item
{
margin-left
:
10px
;
margin-right
:
10px
;
padding
:
12px
0
;
text-align
:
center
;
/* #ifndef APP-NVUE */
white-space
:
nowrap
;
/* #endif */
}
.selected-item-text-overflow
{
width
:
168px
;
/* fix nvue */
overflow
:
hidden
;
/* #ifndef APP-NVUE */
width
:
6em
;
white-space
:
nowrap
;
text-overflow
:
ellipsis
;
-o-text-overflow
:
ellipsis
;
/* #endif */
}
.selected-item-active
{
border-bottom
:
2px
solid
$uni-primary
;
}
.selected-item-text
{
color
:
$uni-primary
;
}
.tab-c
{
position
:
relative
;
flex
:
1
;
/* #ifndef APP-NVUE */
display
:
flex
;
/* #endif */
flex-direction
:
row
;
overflow
:
hidden
;
}
.list
{
flex
:
1
;
}
.item
{
padding
:
12px
15px
;
/* border-bottom: 1px solid #f0f0f0; */
/* #ifndef APP-NVUE */
display
:
flex
;
/* #endif */
flex-direction
:
row
;
justify-content
:
space-between
;
}
.is-disabled
{
opacity
:
.5
;
}
.item-text
{
/* flex: 1; */
color
:
#333333
;
}
.item-text-overflow
{
width
:
280px
;
/* fix nvue */
overflow
:
hidden
;
/* #ifndef APP-NVUE */
width
:
20em
;
white-space
:
nowrap
;
text-overflow
:
ellipsis
;
-o-text-overflow
:
ellipsis
;
/* #endif */
}
.check
{
margin-right
:
5px
;
border
:
2px
solid
$uni-primary
;
border-left
:
0
;
border-top
:
0
;
height
:
12px
;
width
:
6px
;
transform-origin
:
center
;
/* #ifndef APP-NVUE */
transition
:
all
0
.3s
;
/* #endif */
transform
:
rotate
(
45deg
);
}
</
style
>
src/uni_modules/uni-data-picker/package.json
0 → 100644
View file @
64e10593
{
"id"
:
"uni-data-picker"
,
"displayName"
:
"uni-data-picker 数据驱动的picker选择器"
,
"version"
:
"2.0.1"
,
"description"
:
"单列、多列级联选择器,常用于省市区城市选择、公司部门选择、多级分类等场景"
,
"keywords"
:
[
"uni-ui"
,
"uniui"
,
"picker"
,
"级联"
,
"省市区"
,
""
],
"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-load-more"
,
"uni-icons"
,
"uni-scss"
],
"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"
,
"联盟"
:
"u"
},
"Vue"
:
{
"vue2"
:
"y"
,
"vue3"
:
"y"
}
}
}
}
}
\ No newline at end of file
src/uni_modules/uni-data-picker/readme.md
0 → 100644
View file @
64e10593
## DataPicker 级联选择
> **组件名:uni-data-picker**
> 代码块: `uDataPicker`
> 关联组件:`uni-data-pickerview`、`uni-load-more`。
`<uni-data-picker>`
是一个选择类
[
datacom组件
](
https://uniapp.dcloud.net.cn/component/datacom
)
。
支持单列、和多列级联选择。列数没有限制,如果屏幕显示不全,顶部tab区域会左右滚动。
候选数据支持一次性加载完毕,也支持懒加载,比如示例图中,选择了“北京”后,动态加载北京的区县数据。
`<uni-data-picker>`
组件尤其适用于地址选择、分类选择等选择类。
`<uni-data-picker>`
支持本地数据、云端静态数据(json),uniCloud云数据库数据。
`<uni-data-picker>`
可以通过JQL直连uniCloud云数据库,配套
[
DB Schema
](
https://uniapp.dcloud.net.cn/uniCloud/schema
)
,可在schema2code中自动生成前端页面,还支持服务器端校验。
在uniCloud数据表中新建表“uni-id-address”和“opendb-city-china”,这2个表的schema自带foreignKey关联。在“uni-id-address”表的表结构页面使用schema2code生成前端页面,会自动生成地址管理的维护页面,自动从“opendb-city-china”表包含的中国所有省市区信息里选择地址。
### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-data-picker)
#### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839
\ No newline at end of file
src/uni_modules/uni-forms/changelog.md
View file @
64e10593
## 1.4.13(2024-10-08)
-
修复 校验规则在抖音开发者工具上不生效的bug,详见:
[
https://ask.dcloud.net.cn/question/191933
](
https://ask.dcloud.net.cn/question/191933
)
## 1.4.12 (2024-9-21)
-
修复 form上次修改的问题
## 1.4.11 (2024-9-14)
-
修复 binddata的兼容性问题
## 1.4.10(2023-11-03)
## 1.4.10(2023-11-03)
-
优化 labelWidth 描述错误
-
优化 labelWidth 描述错误
## 1.4.9(2023-02-10)
## 1.4.9(2023-02-10)
...
...
src/uni_modules/uni-forms/components/uni-forms-item/uni-forms-item.vue
View file @
64e10593
...
@@ -57,7 +57,12 @@
...
@@ -57,7 +57,12 @@
export
default
{
export
default
{
name
:
'
uniFormsItem
'
,
name
:
'
uniFormsItem
'
,
options
:
{
options
:
{
// #ifdef MP-TOUTIAO
virtualHost
:
false
,
// #endif
// #ifndef MP-TOUTIAO
virtualHost
:
true
virtualHost
:
true
// #endif
},
},
provide
()
{
provide
()
{
return
{
return
{
...
...
src/uni_modules/uni-forms/components/uni-forms/uni-forms.vue
View file @
64e10593
...
@@ -68,7 +68,12 @@
...
@@ -68,7 +68,12 @@
name
:
'
uniForms
'
,
name
:
'
uniForms
'
,
emits
:
[
'
validate
'
,
'
submit
'
],
emits
:
[
'
validate
'
,
'
submit
'
],
options
:
{
options
:
{
// #ifdef MP-TOUTIAO
virtualHost
:
false
,
// #endif
// #ifndef MP-TOUTIAO
virtualHost
:
true
virtualHost
:
true
// #endif
},
},
props
:
{
props
:
{
// 即将弃用
// 即将弃用
...
@@ -180,7 +185,9 @@
...
@@ -180,7 +185,9 @@
}
}
}
}
if
(
!
formVm
)
return
console
.
error
(
'
当前 uni-froms 组件缺少 ref 属性
'
);
if
(
!
formVm
)
return
console
.
error
(
'
当前 uni-froms 组件缺少 ref 属性
'
);
formVm
.
setValue
(
name
,
value
);
if
(
formVm
.
model
)
formVm
.
model
[
name
]
=
value
if
(
formVm
.
modelValue
)
formVm
.
modelValue
[
name
]
=
value
if
(
formVm
.
value
)
formVm
.
value
[
name
]
=
value
}
}
}
}
}
}
...
...
src/uni_modules/uni-forms/package.json
View file @
64e10593
{
{
"id"
:
"uni-forms"
,
"id"
:
"uni-forms"
,
"displayName"
:
"uni-forms 表单"
,
"displayName"
:
"uni-forms 表单"
,
"version"
:
"1.4.1
0
"
,
"version"
:
"1.4.1
3
"
,
"description"
:
"由输入框、选择器、单选框、多选框等控件组成,用以收集、校验、提交数据"
,
"description"
:
"由输入框、选择器、单选框、多选框等控件组成,用以收集、校验、提交数据"
,
"keywords"
:
[
"keywords"
:
[
"uni-ui"
,
"uni-ui"
,
...
@@ -46,7 +46,8 @@
...
@@ -46,7 +46,8 @@
"platforms"
:
{
"platforms"
:
{
"cloud"
:
{
"cloud"
:
{
"tcb"
:
"y"
,
"tcb"
:
"y"
,
"aliyun"
:
"y"
"aliyun"
:
"y"
,
"alipay"
:
"n"
},
},
"client"
:
{
"client"
:
{
"App"
:
{
"App"
:
{
...
...
src/uni_modules/uni-load-more/changelog.md
0 → 100644
View file @
64e10593
## 1.3.6(2024-10-15)
-
修复 微信小程序中的getSystemInfo警告
## 1.3.5(2024-10-12)
-
修复 微信小程序中的getSystemInfo警告
## 1.3.4(2024-10-12)
-
修复 微信小程序中的getSystemInfo警告
## 1.3.3(2022-01-20)
-
新增 showText属性 ,是否显示文本
## 1.3.2(2022-01-19)
-
修复 nvue 平台下不显示文本的bug
## 1.3.1(2022-01-19)
-
修复 微信小程序平台样式选择器报警告的问题
## 1.3.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-load-more
](
https://uniapp.dcloud.io/component/uniui/uni-load-more
)
## 1.2.1(2021-08-24)
-
新增 支持国际化
## 1.2.0(2021-07-30)
-
组件兼容 vue3,如何创建vue3项目,详见
[
uni-app 项目支持 vue3 介绍
](
https://ask.dcloud.net.cn/article/37834
)
## 1.1.8(2021-05-12)
-
新增 组件示例地址
## 1.1.7(2021-03-30)
-
修复 uni-load-more 在首页使用时,h5 平台报 'uni is not defined' 的 bug
## 1.1.6(2021-02-05)
-
调整为uni_modules目录规范
src/uni_modules/uni-load-more/components/uni-load-more/i18n/en.json
0 → 100644
View file @
64e10593
{
"uni-load-more.contentdown"
:
"Pull up to show more"
,
"uni-load-more.contentrefresh"
:
"loading..."
,
"uni-load-more.contentnomore"
:
"No more data"
}
src/uni_modules/uni-load-more/components/uni-load-more/i18n/index.js
0 → 100644
View file @
64e10593
import
en
from
'
./en.json
'
import
zhHans
from
'
./zh-Hans.json
'
import
zhHant
from
'
./zh-Hant.json
'
export
default
{
en
,
'
zh-Hans
'
:
zhHans
,
'
zh-Hant
'
:
zhHant
}
src/uni_modules/uni-load-more/components/uni-load-more/i18n/zh-Hans.json
0 → 100644
View file @
64e10593
{
"uni-load-more.contentdown"
:
"上拉显示更多"
,
"uni-load-more.contentrefresh"
:
"正在加载..."
,
"uni-load-more.contentnomore"
:
"没有更多数据了"
}
src/uni_modules/uni-load-more/components/uni-load-more/i18n/zh-Hant.json
0 → 100644
View file @
64e10593
{
"uni-load-more.contentdown"
:
"上拉顯示更多"
,
"uni-load-more.contentrefresh"
:
"正在加載..."
,
"uni-load-more.contentnomore"
:
"沒有更多數據了"
}
src/uni_modules/uni-load-more/components/uni-load-more/uni-load-more.vue
0 → 100644
View file @
64e10593
<
template
>
<view
class=
"uni-load-more"
@
click=
"onClick"
>
<!-- #ifdef APP-NVUE -->
<loading-indicator
v-if=
"!webviewHide && status === 'loading' && showIcon"
:style=
"
{color: color,width:iconSize+'px',height:iconSize+'px'}" :animating="true"
class="uni-load-more__img uni-load-more__img--nvue">
</loading-indicator>
<!-- #endif -->
<!-- #ifdef H5 -->
<svg
width=
"24"
height=
"24"
viewBox=
"25 25 50 50"
v-if=
"!webviewHide && (iconType==='circle' || iconType==='auto' && platform === 'android') && status === 'loading' && showIcon"
:style=
"
{width:iconSize+'px',height:iconSize+'px'}"
class="uni-load-more__img uni-load-more__img--android-H5">
<circle
cx=
"50"
cy=
"50"
r=
"20"
fill=
"none"
:style=
"
{color:color}" :stroke-width="3">
</circle>
</svg>
<!-- #endif -->
<!-- #ifndef APP-NVUE || H5 -->
<view
v-if=
"!webviewHide && (iconType==='circle' || iconType==='auto' && platform === 'android') && status === 'loading' && showIcon"
:style=
"
{width:iconSize+'px',height:iconSize+'px'}"
class="uni-load-more__img uni-load-more__img--android-MP">
<view
class=
"uni-load-more__img-icon"
:style=
"
{borderTopColor:color,borderTopWidth:iconSize/12}">
</view>
<view
class=
"uni-load-more__img-icon"
:style=
"
{borderTopColor:color,borderTopWidth:iconSize/12}">
</view>
<view
class=
"uni-load-more__img-icon"
:style=
"
{borderTopColor:color,borderTopWidth:iconSize/12}">
</view>
</view>
<!-- #endif -->
<!-- #ifndef APP-NVUE -->
<view
v-else-if=
"!webviewHide && status === 'loading' && showIcon"
:style=
"
{width:iconSize+'px',height:iconSize+'px'}" class="uni-load-more__img uni-load-more__img--ios-H5">
<image
:src=
"imgBase64"
mode=
"widthFix"
></image>
</view>
<!-- #endif -->
<text
v-if=
"showText"
class=
"uni-load-more__text"
:style=
"
{color: color}">
{{
status
===
'
more
'
?
contentdownText
:
status
===
'
loading
'
?
contentrefreshText
:
contentnomoreText
}}
</text>
</view>
</
template
>
<
script
>
let
platform
setTimeout
(()
=>
{
// #ifdef MP-WEIXIN
platform
=
uni
.
getDeviceInfo
().
platform
// #endif
// #ifndef MP-WEIXIN
platform
=
uni
.
getSystemInfoSync
().
platform
// #endif
},
16
)
import
{
initVueI18n
}
from
'
@dcloudio/uni-i18n
'
import
messages
from
'
./i18n/index.js
'
const
{
t
}
=
initVueI18n
(
messages
)
/**
* LoadMore 加载更多
* @description 用于列表中,做滚动加载使用,展示 loading 的各种状态
* @tutorial https://ext.dcloud.net.cn/plugin?id=29
* @property {String} status = [more|loading|noMore] loading 的状态
* @value more loading前
* @value loading loading中
* @value noMore 没有更多了
* @property {Number} iconSize 指定图标大小
* @property {Boolean} iconSize = [true|false] 是否显示 loading 图标
* @property {String} iconType = [snow|circle|auto] 指定图标样式
* @value snow ios雪花加载样式
* @value circle 安卓唤醒加载样式
* @value auto 根据平台自动选择加载样式
* @property {String} color 图标和文字颜色
* @property {Object} contentText 各状态文字说明,值为:{contentdown: "上拉显示更多",contentrefresh: "正在加载...",contentnomore: "没有更多数据了"}
* @event {Function} clickLoadMore 点击加载更多时触发
*/
export
default
{
name
:
'
UniLoadMore
'
,
emits
:
[
'
clickLoadMore
'
],
props
:
{
status
:
{
// 上拉的状态:more-loading前;loading-loading中;noMore-没有更多了
type
:
String
,
default
:
'
more
'
},
showIcon
:
{
type
:
Boolean
,
default
:
true
},
iconType
:
{
type
:
String
,
default
:
'
auto
'
},
iconSize
:
{
type
:
Number
,
default
:
24
},
color
:
{
type
:
String
,
default
:
'
#777777
'
},
contentText
:
{
type
:
Object
,
default
()
{
return
{
contentdown
:
''
,
contentrefresh
:
''
,
contentnomore
:
''
}
}
},
showText
:
{
type
:
Boolean
,
default
:
true
}
},
data
()
{
return
{
webviewHide
:
false
,
platform
:
platform
,
imgBase64
:
'

'
}
},
computed
:
{
iconSnowWidth
()
{
return
(
Math
.
floor
(
this
.
iconSize
/
24
)
||
1
)
*
2
},
contentdownText
()
{
return
this
.
contentText
.
contentdown
||
t
(
"
uni-load-more.contentdown
"
)
},
contentrefreshText
()
{
return
this
.
contentText
.
contentrefresh
||
t
(
"
uni-load-more.contentrefresh
"
)
},
contentnomoreText
()
{
return
this
.
contentText
.
contentnomore
||
t
(
"
uni-load-more.contentnomore
"
)
}
},
mounted
()
{
// #ifdef APP-PLUS
var
pages
=
getCurrentPages
();
var
page
=
pages
[
pages
.
length
-
1
];
var
currentWebview
=
page
.
$getAppWebview
();
currentWebview
.
addEventListener
(
'
hide
'
,
()
=>
{
this
.
webviewHide
=
true
})
currentWebview
.
addEventListener
(
'
show
'
,
()
=>
{
this
.
webviewHide
=
false
})
// #endif
},
methods
:
{
onClick
()
{
this
.
$emit
(
'
clickLoadMore
'
,
{
detail
:
{
status
:
this
.
status
,
}
})
}
}
}
</
script
>
<
style
lang=
"scss"
>
.uni-load-more
{
/* #ifndef APP-NVUE */
display
:
flex
;
/* #endif */
flex-direction
:
row
;
height
:
40px
;
align-items
:
center
;
justify-content
:
center
;
}
.uni-load-more__text
{
font-size
:
14px
;
margin-left
:
8px
;
}
.uni-load-more__img
{
width
:
24px
;
height
:
24px
;
// margin-right: 8px;
}
.uni-load-more__img--nvue
{
color
:
#666666
;
}
.uni-load-more__img--android
,
.uni-load-more__img--ios
{
width
:
24px
;
height
:
24px
;
transform
:
rotate
(
0deg
);
}
/* #ifndef APP-NVUE */
.uni-load-more__img--android
{
animation
:
loading-ios
1s
0s
linear
infinite
;
}
@keyframes
loading-android
{
0
%
{
transform
:
rotate
(
0deg
);
}
100
%
{
transform
:
rotate
(
360deg
);
}
}
.uni-load-more__img--ios-H5
{
position
:
relative
;
animation
:
loading-ios-H5
1s
0s
step-end
infinite
;
}
.uni-load-more__img--ios-H5
image
{
position
:
absolute
;
width
:
100%
;
height
:
100%
;
left
:
0
;
top
:
0
;
}
@keyframes
loading-ios-H5
{
0
%
{
transform
:
rotate
(
0deg
);
}
8
%
{
transform
:
rotate
(
30deg
);
}
16
%
{
transform
:
rotate
(
60deg
);
}
24
%
{
transform
:
rotate
(
90deg
);
}
32
%
{
transform
:
rotate
(
120deg
);
}
40
%
{
transform
:
rotate
(
150deg
);
}
48
%
{
transform
:
rotate
(
180deg
);
}
56
%
{
transform
:
rotate
(
210deg
);
}
64
%
{
transform
:
rotate
(
240deg
);
}
73
%
{
transform
:
rotate
(
270deg
);
}
82
%
{
transform
:
rotate
(
300deg
);
}
91
%
{
transform
:
rotate
(
330deg
);
}
100
%
{
transform
:
rotate
(
360deg
);
}
}
/* #endif */
/* #ifdef H5 */
.uni-load-more__img--android-H5
{
animation
:
loading-android-H5-rotate
2s
linear
infinite
;
transform-origin
:
center
center
;
}
.uni-load-more__img--android-H5
circle
{
display
:
inline-block
;
animation
:
loading-android-H5-dash
1
.5s
ease-in-out
infinite
;
stroke
:
currentColor
;
stroke-linecap
:
round
;
}
@keyframes
loading-android-H5-rotate
{
0
%
{
transform
:
rotate
(
0deg
);
}
100
%
{
transform
:
rotate
(
360deg
);
}
}
@keyframes
loading-android-H5-dash
{
0
%
{
stroke-dasharray
:
1
,
200
;
stroke-dashoffset
:
0
;
}
50
%
{
stroke-dasharray
:
90
,
150
;
stroke-dashoffset
:
-40
;
}
100
%
{
stroke-dasharray
:
90
,
150
;
stroke-dashoffset
:
-120
;
}
}
/* #endif */
/* #ifndef APP-NVUE || H5 */
.uni-load-more__img--android-MP
{
position
:
relative
;
width
:
24px
;
height
:
24px
;
transform
:
rotate
(
0deg
);
animation
:
loading-ios
1s
0s
ease
infinite
;
}
.uni-load-more__img--android-MP
.uni-load-more__img-icon
{
position
:
absolute
;
box-sizing
:
border-box
;
width
:
100%
;
height
:
100%
;
border-radius
:
50%
;
border
:
solid
2px
transparent
;
border-top
:
solid
2px
#777777
;
transform-origin
:
center
;
}
.uni-load-more__img--android-MP
.uni-load-more__img-icon
:nth-child
(
1
)
{
animation
:
loading-android-MP-1
1s
0s
linear
infinite
;
}
.uni-load-more__img--android-MP
.uni-load-more__img-icon
:nth-child
(
2
)
{
animation
:
loading-android-MP-2
1s
0s
linear
infinite
;
}
.uni-load-more__img--android-MP
.uni-load-more__img-icon
:nth-child
(
3
)
{
animation
:
loading-android-MP-3
1s
0s
linear
infinite
;
}
@keyframes
loading-android
{
0
%
{
transform
:
rotate
(
0deg
);
}
100
%
{
transform
:
rotate
(
360deg
);
}
}
@keyframes
loading-android-MP-1
{
0
%
{
transform
:
rotate
(
0deg
);
}
50
%
{
transform
:
rotate
(
90deg
);
}
100
%
{
transform
:
rotate
(
360deg
);
}
}
@keyframes
loading-android-MP-2
{
0
%
{
transform
:
rotate
(
0deg
);
}
50
%
{
transform
:
rotate
(
180deg
);
}
100
%
{
transform
:
rotate
(
360deg
);
}
}
@keyframes
loading-android-MP-3
{
0
%
{
transform
:
rotate
(
0deg
);
}
50
%
{
transform
:
rotate
(
270deg
);
}
100
%
{
transform
:
rotate
(
360deg
);
}
}
/* #endif */
</
style
>
src/uni_modules/uni-load-more/package.json
0 → 100644
View file @
64e10593
{
"id"
:
"uni-load-more"
,
"displayName"
:
"uni-load-more 加载更多"
,
"version"
:
"1.3.6"
,
"description"
:
"LoadMore 组件,常用在列表里面,做滚动加载使用。"
,
"keywords"
:
[
"uni-ui"
,
"uniui"
,
"加载更多"
,
"load-more"
],
"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"
],
"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"
}
}
}
}
}
src/uni_modules/uni-load-more/readme.md
0 → 100644
View file @
64e10593
### LoadMore 加载更多
> **组件名:uni-load-more**
> 代码块: `uLoadMore`
用于列表中,做滚动加载使用,展示 loading 的各种状态。
### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-load-more)
#### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839
src/utils/config.js
View file @
64e10593
// const domain = 'https://api2.groupage.cn'
// const domain = 'https://api2.groupage.cn'
// const domain = 'https://api.jd.jdshangmen.com'
// const domain = 'https://api.jd.jdshangmen.com'
// const domain = 'https://api.sit.jdshangmen.com'
// const domain = 'https://api.sit.jdshangmen.com'
const
domain
=
'
https://api2.groupage.cn
'
//
const domain = 'https://api2.groupage.cn'
//
const domain = 'http://127.0.0.1:48080'
const
domain
=
'
http://127.0.0.1:48080
'
//const domain = 'https://admtest.groupage.cn'
//const domain = 'https://admtest.groupage.cn'
const
config
=
{
const
config
=
{
CDN_DOMAIN
:
domain
,
CDN_DOMAIN
:
domain
,
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment