commit
888a152204
302 changed files with 38116 additions and 0 deletions
@ -0,0 +1,17 @@ |
||||
<script> |
||||
export default { |
||||
onLaunch: function() { |
||||
console.log('App Launch') |
||||
}, |
||||
onShow: function() { |
||||
console.log('App Show') |
||||
}, |
||||
onHide: function() { |
||||
console.log('App Hide') |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style> |
||||
/*每个页面公共css */ |
||||
</style> |
@ -0,0 +1,52 @@ |
||||
<template> |
||||
<view class="content"> |
||||
<view class="title"> |
||||
<text class="number">{{num}}</text> |
||||
<text>{{title}}</text> |
||||
</view> |
||||
<view class="underline"> |
||||
</view> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
export default { |
||||
name:"sectionTitle", |
||||
//props即为需要从组件外部传递进来的参数,包括标题文字title和序号num |
||||
props: { |
||||
num: { |
||||
type: String, |
||||
}, |
||||
title: { |
||||
type: String, |
||||
}, |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style> |
||||
.content { |
||||
display: flex; |
||||
flex-direction: column; |
||||
padding-top: 5px; |
||||
} |
||||
|
||||
.title { |
||||
color: #c5c5c5; |
||||
font-size: 16px; |
||||
font-style: italic; |
||||
font-weight: 600; |
||||
padding-bottom: 5px; |
||||
} |
||||
|
||||
.underline { |
||||
width: 40px; |
||||
height: 4px; |
||||
background-color: #ebebeb; |
||||
margin-bottom: 20px; |
||||
} |
||||
|
||||
.number { |
||||
font-size: 30px; |
||||
} |
||||
</style> |
@ -0,0 +1,784 @@ |
||||
<template> |
||||
<view v-show="show" class="t-wrapper" @touchmove.stop.prevent="moveHandle"> |
||||
<view class="t-mask" :class="{active:active}" @click.stop="close"></view> |
||||
<view class="t-box" :class="{active:active}"> |
||||
<view class="t-header"> |
||||
<view class="t-header-button" @click="close">取消</view> |
||||
<view class="t-header-button" @click="confirm">确认</view> |
||||
</view> |
||||
<view class="t-color__box" :style="{ background: 'rgb(' + bgcolor.r + ',' + bgcolor.g + ',' + bgcolor.b + ')'}"> |
||||
<view class="t-background boxs" @touchstart="touchstart($event, 0)" @touchmove="touchmove($event, 0)" @touchend="touchend($event, 0)"> |
||||
<view class="t-color-mask"></view> |
||||
<view class="t-pointer" :style="{ top: site[0].top - 8 + 'px', left: site[0].left - 8 + 'px' }"></view> |
||||
</view> |
||||
</view> |
||||
<view class="t-control__box"> |
||||
<view class="t-control__color"> |
||||
<view class="t-control__color-content" :style="{ background: 'rgba(' + rgba.r + ',' + rgba.g + ',' + rgba.b + ',' + rgba.a + ')' }"></view> |
||||
</view> |
||||
<view class="t-control-box__item"> |
||||
<view class="t-controller boxs" @touchstart="touchstart($event, 1)" @touchmove="touchmove($event, 1)" @touchend="touchend($event, 1)"> |
||||
<view class="t-hue"> |
||||
<view class="t-circle" :style="{ left: site[1].left - 12 + 'px' }"></view> |
||||
</view> |
||||
</view> |
||||
<view class="t-controller boxs" @touchstart="touchstart($event, 2)" @touchmove="touchmove($event, 2)" @touchend="touchend($event, 2)"> |
||||
<view class="t-transparency"> |
||||
<view class="t-circle" :style="{ left: site[2].left - 12 + 'px' }"></view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
<view class="t-result__box"> |
||||
<view v-if="mode" class="t-result__item"> |
||||
<view class="t-result__box-input">{{hex}}</view> |
||||
<view class="t-result__box-text">HEX</view> |
||||
</view> |
||||
<template v-else> |
||||
<view class="t-result__item"> |
||||
<view class="t-result__box-input">{{rgba.r}}</view> |
||||
<view class="t-result__box-text">R</view> |
||||
</view> |
||||
<view class="t-result__item"> |
||||
<view class="t-result__box-input">{{rgba.g}}</view> |
||||
<view class="t-result__box-text">G</view> |
||||
</view> |
||||
<view class="t-result__item"> |
||||
<view class="t-result__box-input">{{rgba.b}}</view> |
||||
<view class="t-result__box-text">B</view> |
||||
</view> |
||||
<view class="t-result__item"> |
||||
<view class="t-result__box-input">{{rgba.a}}</view> |
||||
<view class="t-result__box-text">A</view> |
||||
</view> |
||||
</template> |
||||
|
||||
<view class="t-result__item t-select" @click="select"> |
||||
<view class="t-result__box-input"> |
||||
<view>切换</view> |
||||
<view>模式</view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
<view class="t-alternative"> |
||||
<view class="t-alternative__item" v-for="(item,index) in colorList" :key="index"> |
||||
<view class="t-alternative__item-content" :style="{ background: 'rgba(' + item.r + ',' + item.g + ',' + item.b + ',' + item.a + ')' }" |
||||
@click="selectColor(item)"> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
export default { |
||||
props: { |
||||
color: { |
||||
type: Object, |
||||
default () { |
||||
return { |
||||
r: 0, |
||||
g: 0, |
||||
b: 0, |
||||
a: 0 |
||||
} |
||||
} |
||||
}, |
||||
spareColor: { |
||||
type: Array, |
||||
default () { |
||||
return [] |
||||
} |
||||
} |
||||
}, |
||||
data() { |
||||
return { |
||||
show: false, |
||||
active: false, |
||||
// rgba 颜色 |
||||
rgba: { |
||||
r: 0, |
||||
g: 0, |
||||
b: 0, |
||||
a: 1 |
||||
}, |
||||
// hsb 颜色 |
||||
hsb: { |
||||
h: 0, |
||||
s: 0, |
||||
b: 0 |
||||
}, |
||||
site: [{ |
||||
top: 0, |
||||
left: 0 |
||||
}, { |
||||
left: 0 |
||||
}, { |
||||
left: 0 |
||||
}], |
||||
index: 0, |
||||
bgcolor: { |
||||
r: 255, |
||||
g: 0, |
||||
b: 0, |
||||
a: 1 |
||||
}, |
||||
hex: '#000000', |
||||
mode: true, |
||||
colorList: [{ |
||||
r: 244, |
||||
g: 67, |
||||
b: 54, |
||||
a: 1 |
||||
}, { |
||||
r: 233, |
||||
g: 30, |
||||
b: 99, |
||||
a: 1 |
||||
}, { |
||||
r: 156, |
||||
g: 39, |
||||
b: 176, |
||||
a: 1 |
||||
}, { |
||||
r: 103, |
||||
g: 58, |
||||
b: 183, |
||||
a: 1 |
||||
}, { |
||||
r: 63, |
||||
g: 81, |
||||
b: 181, |
||||
a: 1 |
||||
}, { |
||||
r: 33, |
||||
g: 150, |
||||
b: 243, |
||||
a: 1 |
||||
}, { |
||||
r: 3, |
||||
g: 169, |
||||
b: 244, |
||||
a: 1 |
||||
}, { |
||||
r: 0, |
||||
g: 188, |
||||
b: 212, |
||||
a: 1 |
||||
}, { |
||||
r: 0, |
||||
g: 150, |
||||
b: 136, |
||||
a: 1 |
||||
}, { |
||||
r: 76, |
||||
g: 175, |
||||
b: 80, |
||||
a: 1 |
||||
}, { |
||||
r: 139, |
||||
g: 195, |
||||
b: 74, |
||||
a: 1 |
||||
}, { |
||||
r: 205, |
||||
g: 220, |
||||
b: 57, |
||||
a: 1 |
||||
}, { |
||||
r: 255, |
||||
g: 235, |
||||
b: 59, |
||||
a: 1 |
||||
}, { |
||||
r: 255, |
||||
g: 193, |
||||
b: 7, |
||||
a: 1 |
||||
}, { |
||||
r: 255, |
||||
g: 152, |
||||
b: 0, |
||||
a: 1 |
||||
}, { |
||||
r: 255, |
||||
g: 87, |
||||
b: 34, |
||||
a: 1 |
||||
}, { |
||||
r: 121, |
||||
g: 85, |
||||
b: 72, |
||||
a: 1 |
||||
}, { |
||||
r: 158, |
||||
g: 158, |
||||
b: 158, |
||||
a: 1 |
||||
}, { |
||||
r: 0, |
||||
g: 0, |
||||
b: 0, |
||||
a: 0.5 |
||||
}, { |
||||
r: 0, |
||||
g: 0, |
||||
b: 0, |
||||
a: 0 |
||||
}, ] |
||||
}; |
||||
}, |
||||
created() { |
||||
this.rgba = this.color; |
||||
if (this.spareColor.length !== 0) { |
||||
this.colorList = this.spareColor; |
||||
} |
||||
}, |
||||
methods: { |
||||
/** |
||||
* 初始化 |
||||
*/ |
||||
init() { |
||||
// hsb 颜色 |
||||
this.hsb = this.rgbToHex(this.rgba); |
||||
// this.setColor(); |
||||
this.setValue(this.rgba); |
||||
}, |
||||
moveHandle() {}, |
||||
open() { |
||||
this.show = true; |
||||
this.$nextTick(() => { |
||||
this.init(); |
||||
setTimeout(() => { |
||||
this.active = true; |
||||
setTimeout(() => { |
||||
this.getSelectorQuery(); |
||||
}, 350) |
||||
}, 50) |
||||
}) |
||||
|
||||
}, |
||||
close() { |
||||
this.active = false; |
||||
this.$nextTick(() => { |
||||
setTimeout(() => { |
||||
this.show = false; |
||||
}, 500) |
||||
}) |
||||
}, |
||||
confirm() { |
||||
this.close(); |
||||
this.$emit('confirm', { |
||||
rgba: this.rgba, |
||||
hex: this.hex |
||||
}) |
||||
}, |
||||
// 选择模式 |
||||
select() { |
||||
this.mode = !this.mode |
||||
}, |
||||
// 常用颜色选择 |
||||
selectColor(item) { |
||||
this.setColorBySelect(item) |
||||
}, |
||||
touchstart(e, index) { |
||||
const { |
||||
pageX, |
||||
pageY |
||||
} = e.touches[0]; |
||||
this.pageX = pageX; |
||||
this.pageY = pageY; |
||||
this.setPosition(pageX, pageY, index); |
||||
}, |
||||
touchmove(e, index) { |
||||
const { |
||||
pageX, |
||||
pageY |
||||
} = e.touches[0]; |
||||
this.moveX = pageX; |
||||
this.moveY = pageY; |
||||
this.setPosition(pageX, pageY, index); |
||||
}, |
||||
touchend(e, index) {}, |
||||
/** |
||||
* 设置位置 |
||||
*/ |
||||
setPosition(x, y, index) { |
||||
this.index = index; |
||||
const { |
||||
top, |
||||
left, |
||||
width, |
||||
height |
||||
} = this.position[index]; |
||||
// 设置最大最小值 |
||||
|
||||
this.site[index].left = Math.max(0, Math.min(parseInt(x - left), width)); |
||||
if (index === 0) { |
||||
this.site[index].top = Math.max(0, Math.min(parseInt(y - top), height)); |
||||
// 设置颜色 |
||||
this.hsb.s = parseInt((100 * this.site[index].left) / width); |
||||
this.hsb.b = parseInt(100 - (100 * this.site[index].top) / height); |
||||
this.setColor(); |
||||
this.setValue(this.rgba); |
||||
} else { |
||||
this.setControl(index, this.site[index].left); |
||||
} |
||||
}, |
||||
/** |
||||
* 设置 rgb 颜色 |
||||
*/ |
||||
setColor() { |
||||
const rgb = this.HSBToRGB(this.hsb); |
||||
this.rgba.r = rgb.r; |
||||
this.rgba.g = rgb.g; |
||||
this.rgba.b = rgb.b; |
||||
}, |
||||
/** |
||||
* 设置二进制颜色 |
||||
* @param {Object} rgb |
||||
*/ |
||||
setValue(rgb) { |
||||
this.hex = '#' + this.rgbToHex(rgb); |
||||
}, |
||||
setControl(index, x) { |
||||
const { |
||||
top, |
||||
left, |
||||
width, |
||||
height |
||||
} = this.position[index]; |
||||
|
||||
if (index === 1) { |
||||
this.hsb.h = parseInt((360 * x) / width); |
||||
this.bgcolor = this.HSBToRGB({ |
||||
h: this.hsb.h, |
||||
s: 100, |
||||
b: 100 |
||||
}); |
||||
this.setColor() |
||||
} else { |
||||
this.rgba.a = (x / width).toFixed(1); |
||||
} |
||||
this.setValue(this.rgba); |
||||
}, |
||||
/** |
||||
* rgb 转 二进制 hex |
||||
* @param {Object} rgb |
||||
*/ |
||||
rgbToHex(rgb) { |
||||
let hex = [rgb.r.toString(16), rgb.g.toString(16), rgb.b.toString(16)]; |
||||
hex.map(function(str, i) { |
||||
if (str.length == 1) { |
||||
hex[i] = '0' + str; |
||||
} |
||||
}); |
||||
return hex.join(''); |
||||
}, |
||||
setColorBySelect(getrgb) { |
||||
const { |
||||
r, |
||||
g, |
||||
b, |
||||
a |
||||
} = getrgb; |
||||
let rgb = {} |
||||
rgb = { |
||||
r: r ? parseInt(r) : 0, |
||||
g: g ? parseInt(g) : 0, |
||||
b: b ? parseInt(b) : 0, |
||||
a: a ? a : 0, |
||||
}; |
||||
this.rgba = rgb; |
||||
this.hsb = this.rgbToHsb(rgb); |
||||
this.changeViewByHsb(); |
||||
}, |
||||
changeViewByHsb() { |
||||
const [a, b, c] = this.position; |
||||
this.site[0].left = parseInt(this.hsb.s * a.width / 100); |
||||
this.site[0].top = parseInt((100 - this.hsb.b) * a.height / 100); |
||||
this.setColor(this.hsb.h); |
||||
this.setValue(this.rgba); |
||||
this.bgcolor = this.HSBToRGB({ |
||||
h: this.hsb.h, |
||||
s: 100, |
||||
b: 100 |
||||
}); |
||||
|
||||
this.site[1].left = this.hsb.h / 360 * b.width; |
||||
this.site[2].left = this.rgba.a * c.width; |
||||
|
||||
}, |
||||
/** |
||||
* hsb 转 rgb |
||||
* @param {Object} 颜色模式 H(hues)表示色相,S(saturation)表示饱和度,B(brightness)表示亮度 |
||||
*/ |
||||
HSBToRGB(hsb) { |
||||
let rgb = {}; |
||||
let h = Math.round(hsb.h); |
||||
let s = Math.round((hsb.s * 255) / 100); |
||||
let v = Math.round((hsb.b * 255) / 100); |
||||
if (s == 0) { |
||||
rgb.r = rgb.g = rgb.b = v; |
||||
} else { |
||||
let t1 = v; |
||||
let t2 = ((255 - s) * v) / 255; |
||||
let t3 = ((t1 - t2) * (h % 60)) / 60; |
||||
if (h == 360) h = 0; |
||||
if (h < 60) { |
||||
rgb.r = t1; |
||||
rgb.b = t2; |
||||
rgb.g = t2 + t3; |
||||
} else if (h < 120) { |
||||
rgb.g = t1; |
||||
rgb.b = t2; |
||||
rgb.r = t1 - t3; |
||||
} else if (h < 180) { |
||||
rgb.g = t1; |
||||
rgb.r = t2; |
||||
rgb.b = t2 + t3; |
||||
} else if (h < 240) { |
||||
rgb.b = t1; |
||||
rgb.r = t2; |
||||
rgb.g = t1 - t3; |
||||
} else if (h < 300) { |
||||
rgb.b = t1; |
||||
rgb.g = t2; |
||||
rgb.r = t2 + t3; |
||||
} else if (h < 360) { |
||||
rgb.r = t1; |
||||
rgb.g = t2; |
||||
rgb.b = t1 - t3; |
||||
} else { |
||||
rgb.r = 0; |
||||
rgb.g = 0; |
||||
rgb.b = 0; |
||||
} |
||||
} |
||||
return { |
||||
r: Math.round(rgb.r), |
||||
g: Math.round(rgb.g), |
||||
b: Math.round(rgb.b) |
||||
}; |
||||
}, |
||||
rgbToHsb(rgb) { |
||||
let hsb = { |
||||
h: 0, |
||||
s: 0, |
||||
b: 0 |
||||
}; |
||||
let min = Math.min(rgb.r, rgb.g, rgb.b); |
||||
let max = Math.max(rgb.r, rgb.g, rgb.b); |
||||
let delta = max - min; |
||||
hsb.b = max; |
||||
hsb.s = max != 0 ? 255 * delta / max : 0; |
||||
if (hsb.s != 0) { |
||||
if (rgb.r == max) hsb.h = (rgb.g - rgb.b) / delta; |
||||
else if (rgb.g == max) hsb.h = 2 + (rgb.b - rgb.r) / delta; |
||||
else hsb.h = 4 + (rgb.r - rgb.g) / delta; |
||||
} else hsb.h = -1; |
||||
hsb.h *= 60; |
||||
if (hsb.h < 0) hsb.h = 0; |
||||
hsb.s *= 100 / 255; |
||||
hsb.b *= 100 / 255; |
||||
return hsb; |
||||
}, |
||||
getSelectorQuery() { |
||||
const views = uni.createSelectorQuery().in(this); |
||||
views |
||||
.selectAll('.boxs') |
||||
.boundingClientRect(data => { |
||||
if (!data || data.length === 0) { |
||||
setTimeout(() => this.getSelectorQuery(), 20) |
||||
return |
||||
} |
||||
this.position = data; |
||||
// this.site[0].top = data[0].height; |
||||
// this.site[0].left = 0; |
||||
// this.site[1].left = data[1].width; |
||||
// this.site[2].left = data[2].width; |
||||
this.setColorBySelect(this.rgba); |
||||
}) |
||||
.exec(); |
||||
} |
||||
}, |
||||
watch: { |
||||
spareColor(newVal) { |
||||
this.colorList = newVal; |
||||
} |
||||
} |
||||
}; |
||||
</script> |
||||
|
||||
<style> |
||||
.t-wrapper { |
||||
position: fixed; |
||||
top: 0; |
||||
bottom: 0; |
||||
left: 0; |
||||
width: 100%; |
||||
box-sizing: border-box; |
||||
z-index: 9999; |
||||
} |
||||
|
||||
.t-box { |
||||
width: 100%; |
||||
position: absolute; |
||||
bottom: 0; |
||||
padding: 30upx 0; |
||||
padding-top: 0; |
||||
background: #fff; |
||||
transition: all 0.3s; |
||||
transform: translateY(100%); |
||||
} |
||||
|
||||
.t-box.active { |
||||
transform: translateY(0%); |
||||
} |
||||
|
||||
.t-header { |
||||
display: flex; |
||||
justify-content: space-between; |
||||
width: 100%; |
||||
height: 100upx; |
||||
border-bottom: 1px #eee solid; |
||||
box-shadow: 1px 0 2px rgba(0, 0, 0, 0.1); |
||||
background: #fff; |
||||
} |
||||
|
||||
.t-header-button { |
||||
display: flex; |
||||
align-items: center; |
||||
width: 150upx; |
||||
height: 100upx; |
||||
font-size: 30upx; |
||||
color: #666; |
||||
padding-left: 20upx; |
||||
} |
||||
|
||||
.t-header-button:last-child { |
||||
justify-content: flex-end; |
||||
padding-right: 20upx; |
||||
} |
||||
|
||||
.t-mask { |
||||
position: absolute; |
||||
top: 0; |
||||
left: 0; |
||||
right: 0; |
||||
bottom: 0; |
||||
background: rgba(0, 0, 0, 0.6); |
||||
z-index: -1; |
||||
transition: all 0.3s; |
||||
opacity: 0; |
||||
} |
||||
|
||||
.t-mask.active { |
||||
opacity: 1; |
||||
} |
||||
|
||||
.t-color__box { |
||||
position: relative; |
||||
height: 400upx; |
||||
background: rgb(255, 0, 0); |
||||
overflow: hidden; |
||||
box-sizing: border-box; |
||||
margin: 0 20upx; |
||||
margin-top: 20upx; |
||||
box-sizing: border-box; |
||||
} |
||||
|
||||
.t-background { |
||||
position: absolute; |
||||
top: 0; |
||||
left: 0; |
||||
right: 0; |
||||
bottom: 0; |
||||
background: linear-gradient(to right, #fff, rgba(255, 255, 255, 0)); |
||||
} |
||||
|
||||
.t-color-mask { |
||||
position: absolute; |
||||
top: 0; |
||||
left: 0; |
||||
right: 0; |
||||
bottom: 0; |
||||
width: 100%; |
||||
height: 400upx; |
||||
background: linear-gradient(to top, #000, rgba(0, 0, 0, 0)); |
||||
} |
||||
|
||||
.t-pointer { |
||||
position: absolute; |
||||
bottom: -8px; |
||||
left: -8px; |
||||
z-index: 2; |
||||
width: 15px; |
||||
height: 15px; |
||||
border: 1px #fff solid; |
||||
border-radius: 50%; |
||||
} |
||||
|
||||
.t-show-color { |
||||
width: 100upx; |
||||
height: 50upx; |
||||
} |
||||
|
||||
.t-control__box { |
||||
margin-top: 50upx; |
||||
width: 100%; |
||||
display: flex; |
||||
padding-left: 20upx; |
||||
box-sizing: border-box; |
||||
} |
||||
|
||||
.t-control__color { |
||||
flex-shrink: 0; |
||||
width: 100upx; |
||||
height: 100upx; |
||||
border-radius: 50%; |
||||
background-color: #fff; |
||||
background-image: linear-gradient(45deg, #eee 25%, transparent 25%, transparent 75%, #eee 75%, #eee), |
||||
linear-gradient(45deg, #eee 25%, transparent 25%, transparent 75%, #eee 75%, #eee); |
||||
background-size: 36upx 36upx; |
||||
background-position: 0 0, 18upx 18upx; |
||||
border: 1px #eee solid; |
||||
overflow: hidden; |
||||
} |
||||
|
||||
.t-control__color-content { |
||||
width: 100%; |
||||
height: 100%; |
||||
} |
||||
|
||||
.t-control-box__item { |
||||
display: flex; |
||||
flex-direction: column; |
||||
justify-content: space-between; |
||||
width: 100%; |
||||
padding: 0 30upx; |
||||
} |
||||
|
||||
.t-controller { |
||||
position: relative; |
||||
width: 100%; |
||||
height: 16px; |
||||
background-color: #fff; |
||||
background-image: linear-gradient(45deg, #eee 25%, transparent 25%, transparent 75%, #eee 75%, #eee), |
||||
linear-gradient(45deg, #eee 25%, transparent 25%, transparent 75%, #eee 75%, #eee); |
||||
background-size: 32upx 32upx; |
||||
background-position: 0 0, 16upx 16upx; |
||||
} |
||||
|
||||
.t-hue { |
||||
width: 100%; |
||||
height: 100%; |
||||
background: linear-gradient(to right, #f00 0%, #ff0 17%, #0f0 33%, #0ff 50%, #00f 67%, #f0f 83%, #f00 100%); |
||||
} |
||||
|
||||
.t-transparency { |
||||
width: 100%; |
||||
height: 100%; |
||||
background: linear-gradient(to right, rgba(0, 0, 0, 0) 0%, rgb(0, 0, 0)); |
||||
} |
||||
|
||||
.t-circle { |
||||
position: absolute; |
||||
/* right: -10px; */ |
||||
top: -2px; |
||||
width: 20px; |
||||
height: 20px; |
||||
box-sizing: border-box; |
||||
border-radius: 50%; |
||||
background: #fff; |
||||
box-shadow: 0 0 2px 1px rgba(0, 0, 0, 0.1); |
||||
} |
||||
|
||||
.t-result__box { |
||||
margin-top: 20upx; |
||||
padding: 10upx; |
||||
width: 100%; |
||||
display: flex; |
||||
box-sizing: border-box; |
||||
} |
||||
|
||||
.t-result__item { |
||||
display: flex; |
||||
flex-direction: column; |
||||
align-items: center; |
||||
justify-content: center; |
||||
padding: 10upx; |
||||
width: 100%; |
||||
box-sizing: border-box; |
||||
} |
||||
|
||||
.t-result__box-input { |
||||
padding: 10upx 0; |
||||
width: 100%; |
||||
font-size: 28upx; |
||||
box-shadow: 0 0 1px 1px rgba(0, 0, 0, 0.1); |
||||
color: #999; |
||||
text-align: center; |
||||
background: #fff; |
||||
} |
||||
|
||||
.t-result__box-text { |
||||
margin-top: 10upx; |
||||
font-size: 28upx; |
||||
line-height: 2; |
||||
} |
||||
|
||||
.t-select { |
||||
flex-shrink: 0; |
||||
width: 150upx; |
||||
padding: 0 30upx; |
||||
} |
||||
|
||||
.t-select .t-result__box-input { |
||||
border-radius: 10upx; |
||||
border: none; |
||||
color: #999; |
||||
box-shadow: 1px 1px 2px 1px rgba(0, 0, 0, 0.1); |
||||
background: #fff; |
||||
} |
||||
|
||||
.t-select .t-result__box-input:active { |
||||
box-shadow: 0px 0px 1px 0px rgba(0, 0, 0, 0.1); |
||||
} |
||||
|
||||
.t-alternative { |
||||
display: flex; |
||||
flex-wrap: wrap; |
||||
/* justify-content: space-between; */ |
||||
width: 100%; |
||||
padding-right: 10upx; |
||||
box-sizing: border-box; |
||||
} |
||||
|
||||
.t-alternative__item { |
||||
margin-left: 12upx; |
||||
margin-top: 10upx; |
||||
width: 50upx; |
||||
height: 50upx; |
||||
border-radius: 10upx; |
||||
background-color: #fff; |
||||
background-image: linear-gradient(45deg, #eee 25%, transparent 25%, transparent 75%, #eee 75%, #eee), |
||||
linear-gradient(45deg, #eee 25%, transparent 25%, transparent 75%, #eee 75%, #eee); |
||||
background-size: 36upx 36upx; |
||||
background-position: 0 0, 18upx 18upx; |
||||
border: 1px #eee solid; |
||||
overflow: hidden; |
||||
} |
||||
|
||||
.t-alternative__item-content { |
||||
width: 50upx; |
||||
height: 50upx; |
||||
background: rgba(255, 0, 0, 0.5); |
||||
} |
||||
|
||||
.t-alternative__item:active { |
||||
transition: all 0.3s; |
||||
transform: scale(1.1); |
||||
} |
||||
</style> |
@ -0,0 +1,20 @@ |
||||
<!DOCTYPE html> |
||||
<html lang="en"> |
||||
<head> |
||||
<meta charset="UTF-8" /> |
||||
<script> |
||||
var coverSupport = 'CSS' in window && typeof CSS.supports === 'function' && (CSS.supports('top: env(a)') || |
||||
CSS.supports('top: constant(a)')) |
||||
document.write( |
||||
'<meta name="viewport" content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0' + |
||||
(coverSupport ? ', viewport-fit=cover' : '') + '" />') |
||||
</script> |
||||
<title></title> |
||||
<!--preload-links--> |
||||
<!--app-context--> |
||||
</head> |
||||
<body> |
||||
<div id="app"><!--app-html--></div> |
||||
<script type="module" src="/main.js"></script> |
||||
</body> |
||||
</html> |
@ -0,0 +1,21 @@ |
||||
import App from './App' |
||||
|
||||
// #ifndef VUE3
|
||||
import Vue from 'vue' |
||||
Vue.config.productionTip = false |
||||
App.mpType = 'app' |
||||
const app = new Vue({ |
||||
...App |
||||
}) |
||||
app.$mount() |
||||
// #endif
|
||||
|
||||
// #ifdef VUE3
|
||||
import { createSSRApp } from 'vue' |
||||
export function createApp() { |
||||
const app = createSSRApp(App) |
||||
return { |
||||
app |
||||
} |
||||
} |
||||
// #endif
|
@ -0,0 +1,72 @@ |
||||
{ |
||||
"name" : "Chameleon", |
||||
"appid" : "", |
||||
"description" : "", |
||||
"versionName" : "1.0.0", |
||||
"versionCode" : "100", |
||||
"transformPx" : false, |
||||
/* 5+App特有相关 */ |
||||
"app-plus" : { |
||||
"usingComponents" : true, |
||||
"nvueStyleCompiler" : "uni-app", |
||||
"compilerVersion" : 3, |
||||
"splashscreen" : { |
||||
"alwaysShowBeforeRender" : true, |
||||
"waiting" : true, |
||||
"autoclose" : true, |
||||
"delay" : 0 |
||||
}, |
||||
/* 模块配置 */ |
||||
"modules" : {}, |
||||
/* 应用发布信息 */ |
||||
"distribute" : { |
||||
/* android打包配置 */ |
||||
"android" : { |
||||
"permissions" : [ |
||||
"<uses-permission android:name=\"android.permission.CHANGE_NETWORK_STATE\"/>", |
||||
"<uses-permission android:name=\"android.permission.MOUNT_UNMOUNT_FILESYSTEMS\"/>", |
||||
"<uses-permission android:name=\"android.permission.VIBRATE\"/>", |
||||
"<uses-permission android:name=\"android.permission.READ_LOGS\"/>", |
||||
"<uses-permission android:name=\"android.permission.ACCESS_WIFI_STATE\"/>", |
||||
"<uses-feature android:name=\"android.hardware.camera.autofocus\"/>", |
||||
"<uses-permission android:name=\"android.permission.ACCESS_NETWORK_STATE\"/>", |
||||
"<uses-permission android:name=\"android.permission.CAMERA\"/>", |
||||
"<uses-permission android:name=\"android.permission.GET_ACCOUNTS\"/>", |
||||
"<uses-permission android:name=\"android.permission.READ_PHONE_STATE\"/>", |
||||
"<uses-permission android:name=\"android.permission.CHANGE_WIFI_STATE\"/>", |
||||
"<uses-permission android:name=\"android.permission.WAKE_LOCK\"/>", |
||||
"<uses-permission android:name=\"android.permission.FLASHLIGHT\"/>", |
||||
"<uses-feature android:name=\"android.hardware.camera\"/>", |
||||
"<uses-permission android:name=\"android.permission.WRITE_SETTINGS\"/>" |
||||
] |
||||
}, |
||||
/* ios打包配置 */ |
||||
"ios" : {}, |
||||
/* SDK配置 */ |
||||
"sdkConfigs" : {} |
||||
} |
||||
}, |
||||
/* 快应用特有相关 */ |
||||
"quickapp" : {}, |
||||
/* 小程序特有相关 */ |
||||
"mp-weixin" : { |
||||
"appid" : "", |
||||
"setting" : { |
||||
"urlCheck" : false |
||||
}, |
||||
"usingComponents" : true |
||||
}, |
||||
"mp-alipay" : { |
||||
"usingComponents" : true |
||||
}, |
||||
"mp-baidu" : { |
||||
"usingComponents" : true |
||||
}, |
||||
"mp-toutiao" : { |
||||
"usingComponents" : true |
||||
}, |
||||
"uniStatistics" : { |
||||
"enable" : false |
||||
}, |
||||
"vueVersion" : "2" |
||||
} |
@ -0,0 +1,17 @@ |
||||
{ |
||||
"pages": [ //pages数组中第一项表示应用启动页,参考:https://uniapp.dcloud.io/collocation/pages |
||||
{ |
||||
"path": "pages/index/index", |
||||
"style": { |
||||
"navigationBarTitleText": "变色龙" |
||||
} |
||||
} |
||||
], |
||||
"globalStyle": { |
||||
"navigationBarTextStyle": "white", |
||||
"navigationBarTitleText": "变色龙", |
||||
"navigationBarBackgroundColor": "#000", |
||||
"backgroundColor": "#fff" |
||||
}, |
||||
"uniIdRouter": {} |
||||
} |
@ -0,0 +1,179 @@ |
||||
<template> |
||||
<view class="padding-10"> |
||||
<sectionTitle title="当前颜色" num='01'></sectionTitle> |
||||
<view class="container"> |
||||
<!-- 展示当前颜色的画布 --> |
||||
<view class="container canvas" :style="canvasStyle" @click="onClickCanvas"> |
||||
<view :style="canvasHexStyle"> |
||||
{{canvasHex}} |
||||
</view> |
||||
</view> |
||||
<button @click="openPicker">打开色盘</button> |
||||
<!-- ref定义了组件的名字 --> |
||||
<!-- @confirm将调色盘的确认按钮绑定到confirmPicker()方法 --> |
||||
<t-color-picker ref="colorPicker" :color="canvasColor" @confirm="confirmPicker"></t-color-picker> |
||||
|
||||
<view class="form padding-10"> |
||||
<text class="padding-r-10 input-title">Hex色码</text> |
||||
<uni-easyinput v-model="hexValue" placeholder="#ffc107" class="padding-r-10"></uni-easyinput> |
||||
<button @click="hexSubmit" size="mini" type="primary">确认</button> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
export default { |
||||
data() { |
||||
return { |
||||
canvasColor: { |
||||
r: 255, |
||||
g: 193, |
||||
b: 7, |
||||
a: 1 |
||||
}, |
||||
hexValue: "", |
||||
}; |
||||
}, |
||||
// 页面启动时的生命周期函数 |
||||
mounted() { |
||||
this.hexValue = "#" + this.rgbToHex(this.canvasColor) |
||||
}, |
||||
methods: { |
||||
// methods中的方法需要通过某些事件、动作才能触发运行 |
||||
openPicker(item) { |
||||
// 打开颜色选择器 |
||||
this.$refs.colorPicker.open(); |
||||
}, |
||||
confirmPicker(e) { |
||||
// 色盘点击确定后 |
||||
// 将选取的颜色赋值给canvasColor状态 |
||||
this.canvasColor = e.rgba; |
||||
// 更新hex输入框的数据 |
||||
this.hexValue = e.hex; |
||||
}, |
||||
// rgb转为hex |
||||
rgbToHex(rgb) { |
||||
let hex = [rgb.r.toString(16), rgb.g.toString(16), rgb.b.toString(16)]; |
||||
hex.map(function(str, i) { |
||||
if (str.length == 1) { |
||||
hex[i] = '0' + str; |
||||
} |
||||
}); |
||||
return hex.join(''); |
||||
}, |
||||
// 将hex转换为rgb |
||||
hexToRgb(hex) { |
||||
// 将hex切割为rgb数组 |
||||
const hexArr = hex.match(/[\s\S]{1,2}/g) || [] |
||||
// 转换为10进制数组 |
||||
const rgbArr = hexArr.map(hex => parseInt(hex, 16)); |
||||
return { |
||||
r: rgbArr[0], |
||||
g: rgbArr[1], |
||||
b: rgbArr[2], |
||||
a: 1, |
||||
} |
||||
}, |
||||
// 将hex值复制到剪切板 |
||||
onClickCanvas() { |
||||
uni.setClipboardData({ |
||||
data: this.canvasHex, |
||||
}) |
||||
}, |
||||
// 提交hex值 |
||||
hexSubmit() { |
||||
// 预处理 |
||||
let hexStr = this.hexValue.trim().replace('#', "").toLowerCase(); |
||||
// 将3位hex值转化为6位hex |
||||
if (hexStr.length === 3) { |
||||
hexStr = hexStr.split("").reduce((total, current) => total + current.repeat(2), ""); |
||||
} |
||||
if (hexStr.length !== 6) { |
||||
return uni.showToast({ |
||||
title: "Hex色码必须为3位或6位", |
||||
icon: "none", |
||||
}); |
||||
} |
||||
// 验证hex是否合法 |
||||
const alphabet = "0123456789abcdef"; |
||||
for (let elem of hexStr) { |
||||
if (!alphabet.includes(elem)) { |
||||
return uni.showToast({ |
||||
title: "请输入正确的Hex色码", |
||||
icon: "none", |
||||
}) |
||||
} |
||||
} |
||||
|
||||
const rgb = this.hexToRgb(hexStr); |
||||
|
||||
// 更新canvas颜色 |
||||
this.canvasColor = rgb; |
||||
// 更新色盘颜色 |
||||
this.$refs.colorPicker.rgba = rgb; |
||||
}, |
||||
}, |
||||
computed: { |
||||
// computed 中的方法被称为计算属性,它会实时监听自身所依赖的状态 |
||||
// 每当canvasColor发生变化时,设置canvas的颜色 |
||||
canvasStyle() { |
||||
return `background: #${this.rgbToHex(this.canvasColor)}` |
||||
}, |
||||
// 设置canvas的文本颜色 |
||||
// 使其根据hex颜色的深浅自动变化 |
||||
canvasHexStyle() { |
||||
for (let key in this.canvasColor) { |
||||
if (this.canvasColor[key] >= 130) { |
||||
return "color: black"; |
||||
} |
||||
return "color: white"; |
||||
} |
||||
}, |
||||
// 设置canvas的hex文本(16进制表示) |
||||
canvasHex() { |
||||
return "#" + this.rgbToHex(this.canvasColor); |
||||
} |
||||
} |
||||
|
||||
}; |
||||
</script> |
||||
|
||||
<style scoped> |
||||
.container { |
||||
display: flex; |
||||
flex-direction: column; |
||||
align-items: center; |
||||
justify-content: center; |
||||
padding: 10px; |
||||
} |
||||
|
||||
.padding-10 { |
||||
padding: 10px; |
||||
} |
||||
|
||||
.canvas { |
||||
height: 80px; |
||||
width: 100%; |
||||
margin-bottom: 10px; |
||||
border-radius: 10px; |
||||
border-color: #484848; |
||||
border-style: dashed; |
||||
} |
||||
|
||||
.form { |
||||
display: flex; |
||||
flex-direction: row; |
||||
align-items: center; |
||||
} |
||||
|
||||
.padding-r-10 { |
||||
padding-right: 10px; |
||||
} |
||||
|
||||
.input-title { |
||||
font-size: 14px; |
||||
color: gray; |
||||
font-weight: 700; |
||||
} |
||||
</style> |
After Width: | Height: | Size: 3.9 KiB |
@ -0,0 +1,76 @@ |
||||
/** |
||||
* 这里是uni-app内置的常用样式变量 |
||||
* |
||||
* uni-app 官方扩展插件及插件市场(https://ext.dcloud.net.cn)上很多三方插件均使用了这些样式变量 |
||||
* 如果你是插件开发者,建议你使用scss预处理,并在插件代码中直接使用这些变量(无需 import 这个文件),方便用户通过搭积木的方式开发整体风格一致的App |
||||
* |
||||
*/ |
||||
|
||||
/** |
||||
* 如果你是App开发者(插件使用者),你可以通过修改这些变量来定制自己的插件主题,实现自定义主题功能 |
||||
* |
||||
* 如果你的项目同样使用了scss预处理,你也可以直接在你的 scss 代码中使用如下变量,同时无需 import 这个文件 |
||||
*/ |
||||
|
||||
/* 颜色变量 */ |
||||
|
||||
/* 行为相关颜色 */ |
||||
$uni-color-primary: #007aff; |
||||
$uni-color-success: #4cd964; |
||||
$uni-color-warning: #f0ad4e; |
||||
$uni-color-error: #dd524d; |
||||
|
||||
/* 文字基本颜色 */ |
||||
$uni-text-color:#333;//基本色 |
||||
$uni-text-color-inverse:#fff;//反色 |
||||
$uni-text-color-grey:#999;//辅助灰色,如加载更多的提示信息 |
||||
$uni-text-color-placeholder: #808080; |
||||
$uni-text-color-disable:#c0c0c0; |
||||
|
||||
/* 背景颜色 */ |
||||
$uni-bg-color:#ffffff; |
||||
$uni-bg-color-grey:#f8f8f8; |
||||
$uni-bg-color-hover:#f1f1f1;//点击状态颜色 |
||||
$uni-bg-color-mask:rgba(0, 0, 0, 0.4);//遮罩颜色 |
||||
|
||||
/* 边框颜色 */ |
||||
$uni-border-color:#c8c7cc; |
||||
|
||||
/* 尺寸变量 */ |
||||
|
||||
/* 文字尺寸 */ |
||||
$uni-font-size-sm:12px; |
||||
$uni-font-size-base:14px; |
||||
$uni-font-size-lg:16; |
||||
|
||||
/* 图片尺寸 */ |
||||
$uni-img-size-sm:20px; |
||||
$uni-img-size-base:26px; |
||||
$uni-img-size-lg:40px; |
||||
|
||||
/* Border Radius */ |
||||
$uni-border-radius-sm: 2px; |
||||
$uni-border-radius-base: 3px; |
||||
$uni-border-radius-lg: 6px; |
||||
$uni-border-radius-circle: 50%; |
||||
|
||||
/* 水平间距 */ |
||||
$uni-spacing-row-sm: 5px; |
||||
$uni-spacing-row-base: 10px; |
||||
$uni-spacing-row-lg: 15px; |
||||
|
||||
/* 垂直间距 */ |
||||
$uni-spacing-col-sm: 4px; |
||||
$uni-spacing-col-base: 8px; |
||||
$uni-spacing-col-lg: 12px; |
||||
|
||||
/* 透明度 */ |
||||
$uni-opacity-disabled: 0.3; // 组件禁用态的透明度 |
||||
|
||||
/* 文章场景相关 */ |
||||
$uni-color-title: #2C405A; // 文章标题颜色 |
||||
$uni-font-size-title:20px; |
||||
$uni-color-subtitle: #555555; // 二级标题颜色 |
||||
$uni-font-size-subtitle:26px; |
||||
$uni-color-paragraph: #3F536E; // 文章段落颜色 |
||||
$uni-font-size-paragraph:15px; |
@ -0,0 +1,31 @@ |
||||
## 1.2.1(2022-09-05) |
||||
- 修复 当 text 超过 max-num 时,badge 的宽度计算是根据 text 的长度计算,更改为 css 计算实际展示宽度,详见:[https://ask.dcloud.net.cn/question/150473](https://ask.dcloud.net.cn/question/150473) |
||||
## 1.2.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-badge](https://uniapp.dcloud.io/component/uniui/uni-badge) |
||||
## 1.1.7(2021-11-08) |
||||
- 优化 升级ui |
||||
- 修改 size 属性默认值调整为 small |
||||
- 修改 type 属性,默认值调整为 error,info 替换 default |
||||
## 1.1.6(2021-09-22) |
||||
- 修复 在字节小程序上样式不生效的 bug |
||||
## 1.1.5(2021-07-30) |
||||
- 组件兼容 vue3,如何创建vue3项目,详见 [uni-app 项目支持 vue3 介绍](https://ask.dcloud.net.cn/article/37834) |
||||
## 1.1.4(2021-07-29) |
||||
- 修复 去掉 nvue 不支持css 的 align-self 属性,nvue 下不暂支持 absolute 属性 |
||||
## 1.1.3(2021-06-24) |
||||
- 优化 示例项目 |
||||
## 1.1.1(2021-05-12) |
||||
- 新增 组件示例地址 |
||||
## 1.1.0(2021-05-12) |
||||
- 新增 uni-badge 的 absolute 属性,支持定位 |
||||
- 新增 uni-badge 的 offset 属性,支持定位偏移 |
||||
- 新增 uni-badge 的 is-dot 属性,支持仅显示有一个小点 |
||||
- 新增 uni-badge 的 max-num 属性,支持自定义封顶的数字值,超过 99 显示99+ |
||||
- 优化 uni-badge 属性 custom-style, 支持以对象形式自定义样式 |
||||
## 1.0.7(2021-05-07) |
||||
- 修复 uni-badge 在 App 端,数字小于10时不是圆形的bug |
||||
- 修复 uni-badge 在父元素不是 flex 布局时,宽度缩小的bug |
||||
- 新增 uni-badge 属性 custom-style, 支持自定义样式 |
||||
## 1.0.6(2021-02-04) |
||||
- 调整为uni_modules目录规范 |
@ -0,0 +1,268 @@ |
||||
<template> |
||||
<view class="uni-badge--x"> |
||||
<slot /> |
||||
<text v-if="text" :class="classNames" :style="[positionStyle, customStyle, dotStyle]" |
||||
class="uni-badge" @click="onClick()">{{displayValue}}</text> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
/** |
||||
* Badge 数字角标 |
||||
* @description 数字角标一般和其它控件(列表、9宫格等)配合使用,用于进行数量提示,默认为实心灰色背景 |
||||
* @tutorial https://ext.dcloud.net.cn/plugin?id=21 |
||||
* @property {String} text 角标内容 |
||||
* @property {String} size = [normal|small] 角标内容 |
||||
* @property {String} type = [info|primary|success|warning|error] 颜色类型 |
||||
* @value info 灰色 |
||||
* @value primary 蓝色 |
||||
* @value success 绿色 |
||||
* @value warning 黄色 |
||||
* @value error 红色 |
||||
* @property {String} inverted = [true|false] 是否无需背景颜色 |
||||
* @property {Number} maxNum 展示封顶的数字值,超过 99 显示 99+ |
||||
* @property {String} absolute = [rightTop|rightBottom|leftBottom|leftTop] 开启绝对定位, 角标将定位到其包裹的标签的四角上 |
||||
* @value rightTop 右上 |
||||
* @value rightBottom 右下 |
||||
* @value leftTop 左上 |
||||
* @value leftBottom 左下 |
||||
* @property {Array[number]} offset 距定位角中心点的偏移量,只有存在 absolute 属性时有效,例如:[-10, -10] 表示向外偏移 10px,[10, 10] 表示向 absolute 指定的内偏移 10px |
||||
* @property {String} isDot = [true|false] 是否显示为一个小点 |
||||
* @event {Function} click 点击 Badge 触发事件 |
||||
* @example <uni-badge text="1"></uni-badge> |
||||
*/ |
||||
|
||||
export default { |
||||
name: 'UniBadge', |
||||
emits: ['click'], |
||||
props: { |
||||
type: { |
||||
type: String, |
||||
default: 'error' |
||||
}, |
||||
inverted: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
isDot: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
maxNum: { |
||||
type: Number, |
||||
default: 99 |
||||
}, |
||||
absolute: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
offset: { |
||||
type: Array, |
||||
default () { |
||||
return [0, 0] |
||||
} |
||||
}, |
||||
text: { |
||||
type: [String, Number], |
||||
default: '' |
||||
}, |
||||
size: { |
||||
type: String, |
||||
default: 'small' |
||||
}, |
||||
customStyle: { |
||||
type: Object, |
||||
default () { |
||||
return {} |
||||
} |
||||
} |
||||
}, |
||||
data() { |
||||
return {}; |
||||
}, |
||||
computed: { |
||||
width() { |
||||
return String(this.text).length * 8 + 12 |
||||
}, |
||||
classNames() { |
||||
const { |
||||
inverted, |
||||
type, |
||||
size, |
||||
absolute |
||||
} = this |
||||
return [ |
||||
inverted ? 'uni-badge--' + type + '-inverted' : '', |
||||
'uni-badge--' + type, |
||||
'uni-badge--' + size, |
||||
absolute ? 'uni-badge--absolute' : '' |
||||
].join(' ') |
||||
}, |
||||
positionStyle() { |
||||
if (!this.absolute) return {} |
||||
let w = this.width / 2, |
||||
h = 10 |
||||
if (this.isDot) { |
||||
w = 5 |
||||
h = 5 |
||||
} |
||||
const x = `${- w + this.offset[0]}px` |
||||
const y = `${- h + this.offset[1]}px` |
||||
|
||||
const whiteList = { |
||||
rightTop: { |
||||
right: x, |
||||
top: y |
||||
}, |
||||
rightBottom: { |
||||
right: x, |
||||
bottom: y |
||||
}, |
||||
leftBottom: { |
||||
left: x, |
||||
bottom: y |
||||
}, |
||||
leftTop: { |
||||
left: x, |
||||
top: y |
||||
} |
||||
} |
||||
const match = whiteList[this.absolute] |
||||
return match ? match : whiteList['rightTop'] |
||||
}, |
||||
dotStyle() { |
||||
if (!this.isDot) return {} |
||||
return { |
||||
width: '10px', |
||||
minWidth: '0', |
||||
height: '10px', |
||||
padding: '0', |
||||
borderRadius: '10px' |
||||
} |
||||
}, |
||||
displayValue() { |
||||
const { |
||||
isDot, |
||||
text, |
||||
maxNum |
||||
} = this |
||||
return isDot ? '' : (Number(text) > maxNum ? `${maxNum}+` : text) |
||||
} |
||||
}, |
||||
methods: { |
||||
onClick() { |
||||
this.$emit('click'); |
||||
} |
||||
} |
||||
}; |
||||
</script> |
||||
|
||||
<style lang="scss" > |
||||
$uni-primary: #2979ff !default; |
||||
$uni-success: #4cd964 !default; |
||||
$uni-warning: #f0ad4e !default; |
||||
$uni-error: #dd524d !default; |
||||
$uni-info: #909399 !default; |
||||
|
||||
|
||||
$bage-size: 12px; |
||||
$bage-small: scale(0.8); |
||||
|
||||
.uni-badge--x { |
||||
/* #ifdef APP-NVUE */ |
||||
// align-self: flex-start; |
||||
/* #endif */ |
||||
/* #ifndef APP-NVUE */ |
||||
display: inline-block; |
||||
/* #endif */ |
||||
position: relative; |
||||
} |
||||
|
||||
.uni-badge--absolute { |
||||
position: absolute; |
||||
} |
||||
|
||||
.uni-badge--small { |
||||
transform: $bage-small; |
||||
transform-origin: center center; |
||||
} |
||||
|
||||
.uni-badge { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
overflow: hidden; |
||||
box-sizing: border-box; |
||||
/* #endif */ |
||||
justify-content: center; |
||||
flex-direction: row; |
||||
height: 20px; |
||||
min-width: 20px; |
||||
padding: 0 4px; |
||||
line-height: 18px; |
||||
color: #fff; |
||||
border-radius: 100px; |
||||
background-color: $uni-info; |
||||
background-color: transparent; |
||||
border: 1px solid #fff; |
||||
text-align: center; |
||||
font-family: 'Helvetica Neue', Helvetica, sans-serif; |
||||
font-feature-settings: "tnum"; |
||||
font-size: $bage-size; |
||||
/* #ifdef H5 */ |
||||
z-index: 999; |
||||
cursor: pointer; |
||||
/* #endif */ |
||||
|
||||
&--info { |
||||
color: #fff; |
||||
background-color: $uni-info; |
||||
} |
||||
|
||||
&--primary { |
||||
background-color: $uni-primary; |
||||
} |
||||
|
||||
&--success { |
||||
background-color: $uni-success; |
||||
} |
||||
|
||||
&--warning { |
||||
background-color: $uni-warning; |
||||
} |
||||
|
||||
&--error { |
||||
background-color: $uni-error; |
||||
} |
||||
|
||||
&--inverted { |
||||
padding: 0 5px 0 0; |
||||
color: $uni-info; |
||||
} |
||||
|
||||
&--info-inverted { |
||||
color: $uni-info; |
||||
background-color: transparent; |
||||
} |
||||
|
||||
&--primary-inverted { |
||||
color: $uni-primary; |
||||
background-color: transparent; |
||||
} |
||||
|
||||
&--success-inverted { |
||||
color: $uni-success; |
||||
background-color: transparent; |
||||
} |
||||
|
||||
&--warning-inverted { |
||||
color: $uni-warning; |
||||
background-color: transparent; |
||||
} |
||||
|
||||
&--error-inverted { |
||||
color: $uni-error; |
||||
background-color: transparent; |
||||
} |
||||
|
||||
} |
||||
</style> |
@ -0,0 +1,85 @@ |
||||
{ |
||||
"id": "uni-badge", |
||||
"displayName": "uni-badge 数字角标", |
||||
"version": "1.2.1", |
||||
"description": "数字角标(徽章)组件,在元素周围展示消息提醒,一般用于列表、九宫格、按钮等地方。", |
||||
"keywords": [ |
||||
"", |
||||
"badge", |
||||
"uni-ui", |
||||
"uniui", |
||||
"数字角标", |
||||
"徽章" |
||||
], |
||||
"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" |
||||
}, |
||||
"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" |
||||
}, |
||||
"快应用": { |
||||
"华为": "y", |
||||
"联盟": "y" |
||||
}, |
||||
"Vue": { |
||||
"vue2": "y", |
||||
"vue3": "y" |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,10 @@ |
||||
## Badge 数字角标 |
||||
> **组件名:uni-badge** |
||||
> 代码块: `uBadge` |
||||
|
||||
数字角标一般和其它控件(列表、9宫格等)配合使用,用于进行数量提示,默认为实心灰色背景, |
||||
|
||||
### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-badge) |
||||
#### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839 |
||||
|
||||
|
@ -0,0 +1,20 @@ |
||||
## 1.4.7(2022-09-16) |
||||
- 可以使用 uni-scss 控制主题色 |
||||
## 1.4.6(2022-09-08) |
||||
- fix: 表头年月切换,导致改变当前日期为选择月1号,且未触发change事件 |
||||
## 1.4.5(2022-02-25) |
||||
- 修复 条件编译 nvue 不支持的 css 样式 |
||||
## 1.4.4(2022-02-25) |
||||
- 修复 条件编译 nvue 不支持的 css 样式 |
||||
## 1.4.3(2021-09-22) |
||||
- 修复 startDate、 endDate 属性失效的 bug |
||||
## 1.4.2(2021-08-24) |
||||
- 新增 支持国际化 |
||||
## 1.4.1(2021-08-05) |
||||
- 修复 弹出层被 tabbar 遮盖 bug |
||||
## 1.4.0(2021-07-30) |
||||
- 组件兼容 vue3,如何创建vue3项目,详见 [uni-app 项目支持 vue3 介绍](https://ask.dcloud.net.cn/article/37834) |
||||
## 1.3.16(2021-05-12) |
||||
- 新增 组件示例地址 |
||||
## 1.3.15(2021-02-04) |
||||
- 调整为uni_modules目录规范 |
@ -0,0 +1,546 @@ |
||||
/** |
||||
* @1900-2100区间内的公历、农历互转 |
||||
* @charset UTF-8 |
||||
* @github https://github.com/jjonline/calendar.js
|
||||
* @Author Jea杨(JJonline@JJonline.Cn) |
||||
* @Time 2014-7-21 |
||||
* @Time 2016-8-13 Fixed 2033hex、Attribution Annals |
||||
* @Time 2016-9-25 Fixed lunar LeapMonth Param Bug |
||||
* @Time 2017-7-24 Fixed use getTerm Func Param Error.use solar year,NOT lunar year |
||||
* @Version 1.0.3 |
||||
* @公历转农历:calendar.solar2lunar(1987,11,01); //[you can ignore params of prefix 0]
|
||||
* @农历转公历:calendar.lunar2solar(1987,09,10); //[you can ignore params of prefix 0]
|
||||
*/ |
||||
/* eslint-disable */ |
||||
var calendar = { |
||||
|
||||
/** |
||||
* 农历1900-2100的润大小信息表 |
||||
* @Array Of Property |
||||
* @return Hex |
||||
*/ |
||||
lunarInfo: [0x04bd8, 0x04ae0, 0x0a570, 0x054d5, 0x0d260, 0x0d950, 0x16554, 0x056a0, 0x09ad0, 0x055d2, // 1900-1909
|
||||
0x04ae0, 0x0a5b6, 0x0a4d0, 0x0d250, 0x1d255, 0x0b540, 0x0d6a0, 0x0ada2, 0x095b0, 0x14977, // 1910-1919
|
||||
0x04970, 0x0a4b0, 0x0b4b5, 0x06a50, 0x06d40, 0x1ab54, 0x02b60, 0x09570, 0x052f2, 0x04970, // 1920-1929
|
||||
0x06566, 0x0d4a0, 0x0ea50, 0x06e95, 0x05ad0, 0x02b60, 0x186e3, 0x092e0, 0x1c8d7, 0x0c950, // 1930-1939
|
||||
0x0d4a0, 0x1d8a6, 0x0b550, 0x056a0, 0x1a5b4, 0x025d0, 0x092d0, 0x0d2b2, 0x0a950, 0x0b557, // 1940-1949
|
||||
0x06ca0, 0x0b550, 0x15355, 0x04da0, 0x0a5b0, 0x14573, 0x052b0, 0x0a9a8, 0x0e950, 0x06aa0, // 1950-1959
|
||||
0x0aea6, 0x0ab50, 0x04b60, 0x0aae4, 0x0a570, 0x05260, 0x0f263, 0x0d950, 0x05b57, 0x056a0, // 1960-1969
|
||||
0x096d0, 0x04dd5, 0x04ad0, 0x0a4d0, 0x0d4d4, 0x0d250, 0x0d558, 0x0b540, 0x0b6a0, 0x195a6, // 1970-1979
|
||||
0x095b0, 0x049b0, 0x0a974, 0x0a4b0, 0x0b27a, 0x06a50, 0x06d40, 0x0af46, 0x0ab60, 0x09570, // 1980-1989
|
||||
0x04af5, 0x04970, 0x064b0, 0x074a3, 0x0ea50, 0x06b58, 0x05ac0, 0x0ab60, 0x096d5, 0x092e0, // 1990-1999
|
||||
0x0c960, 0x0d954, 0x0d4a0, 0x0da50, 0x07552, 0x056a0, 0x0abb7, 0x025d0, 0x092d0, 0x0cab5, // 2000-2009
|
||||
0x0a950, 0x0b4a0, 0x0baa4, 0x0ad50, 0x055d9, 0x04ba0, 0x0a5b0, 0x15176, 0x052b0, 0x0a930, // 2010-2019
|
||||
0x07954, 0x06aa0, 0x0ad50, 0x05b52, 0x04b60, 0x0a6e6, 0x0a4e0, 0x0d260, 0x0ea65, 0x0d530, // 2020-2029
|
||||
0x05aa0, 0x076a3, 0x096d0, 0x04afb, 0x04ad0, 0x0a4d0, 0x1d0b6, 0x0d250, 0x0d520, 0x0dd45, // 2030-2039
|
||||
0x0b5a0, 0x056d0, 0x055b2, 0x049b0, 0x0a577, 0x0a4b0, 0x0aa50, 0x1b255, 0x06d20, 0x0ada0, // 2040-2049
|
||||
/** Add By JJonline@JJonline.Cn**/ |
||||
0x14b63, 0x09370, 0x049f8, 0x04970, 0x064b0, 0x168a6, 0x0ea50, 0x06b20, 0x1a6c4, 0x0aae0, // 2050-2059
|
||||
0x0a2e0, 0x0d2e3, 0x0c960, 0x0d557, 0x0d4a0, 0x0da50, 0x05d55, 0x056a0, 0x0a6d0, 0x055d4, // 2060-2069
|
||||
0x052d0, 0x0a9b8, 0x0a950, 0x0b4a0, 0x0b6a6, 0x0ad50, 0x055a0, 0x0aba4, 0x0a5b0, 0x052b0, // 2070-2079
|
||||
0x0b273, 0x06930, 0x07337, 0x06aa0, 0x0ad50, 0x14b55, 0x04b60, 0x0a570, 0x054e4, 0x0d160, // 2080-2089
|
||||
0x0e968, 0x0d520, 0x0daa0, 0x16aa6, 0x056d0, 0x04ae0, 0x0a9d4, 0x0a2d0, 0x0d150, 0x0f252, // 2090-2099
|
||||
0x0d520], // 2100
|
||||
|
||||
/** |
||||
* 公历每个月份的天数普通表 |
||||
* @Array Of Property |
||||
* @return Number |
||||
*/ |
||||
solarMonth: [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31], |
||||
|
||||
/** |
||||
* 天干地支之天干速查表 |
||||
* @Array Of Property trans["甲","乙","丙","丁","戊","己","庚","辛","壬","癸"] |
||||
* @return Cn string |
||||
*/ |
||||
Gan: ['\u7532', '\u4e59', '\u4e19', '\u4e01', '\u620a', '\u5df1', '\u5e9a', '\u8f9b', '\u58ec', '\u7678'], |
||||
|
||||
/** |
||||
* 天干地支之地支速查表 |
||||
* @Array Of Property |
||||
* @trans["子","丑","寅","卯","辰","巳","午","未","申","酉","戌","亥"] |
||||
* @return Cn string |
||||
*/ |
||||
Zhi: ['\u5b50', '\u4e11', '\u5bc5', '\u536f', '\u8fb0', '\u5df3', '\u5348', '\u672a', '\u7533', '\u9149', '\u620c', '\u4ea5'], |
||||
|
||||
/** |
||||
* 天干地支之地支速查表<=>生肖 |
||||
* @Array Of Property |
||||
* @trans["鼠","牛","虎","兔","龙","蛇","马","羊","猴","鸡","狗","猪"] |
||||
* @return Cn string |
||||
*/ |
||||
Animals: ['\u9f20', '\u725b', '\u864e', '\u5154', '\u9f99', '\u86c7', '\u9a6c', '\u7f8a', '\u7334', '\u9e21', '\u72d7', '\u732a'], |
||||
|
||||
/** |
||||
* 24节气速查表 |
||||
* @Array Of Property |
||||
* @trans["小寒","大寒","立春","雨水","惊蛰","春分","清明","谷雨","立夏","小满","芒种","夏至","小暑","大暑","立秋","处暑","白露","秋分","寒露","霜降","立冬","小雪","大雪","冬至"] |
||||
* @return Cn string |
||||
*/ |
||||
solarTerm: ['\u5c0f\u5bd2', '\u5927\u5bd2', '\u7acb\u6625', '\u96e8\u6c34', '\u60ca\u86f0', '\u6625\u5206', '\u6e05\u660e', '\u8c37\u96e8', '\u7acb\u590f', '\u5c0f\u6ee1', '\u8292\u79cd', '\u590f\u81f3', '\u5c0f\u6691', '\u5927\u6691', '\u7acb\u79cb', '\u5904\u6691', '\u767d\u9732', '\u79cb\u5206', '\u5bd2\u9732', '\u971c\u964d', '\u7acb\u51ac', '\u5c0f\u96ea', '\u5927\u96ea', '\u51ac\u81f3'], |
||||
|
||||
/** |
||||
* 1900-2100各年的24节气日期速查表 |
||||
* @Array Of Property |
||||
* @return 0x string For splice |
||||
*/ |
||||
sTermInfo: ['9778397bd097c36b0b6fc9274c91aa', '97b6b97bd19801ec9210c965cc920e', '97bcf97c3598082c95f8c965cc920f', |
||||
'97bd0b06bdb0722c965ce1cfcc920f', 'b027097bd097c36b0b6fc9274c91aa', '97b6b97bd19801ec9210c965cc920e', |
||||
'97bcf97c359801ec95f8c965cc920f', '97bd0b06bdb0722c965ce1cfcc920f', 'b027097bd097c36b0b6fc9274c91aa', |
||||
'97b6b97bd19801ec9210c965cc920e', '97bcf97c359801ec95f8c965cc920f', '97bd0b06bdb0722c965ce1cfcc920f', |
||||
'b027097bd097c36b0b6fc9274c91aa', '9778397bd19801ec9210c965cc920e', '97b6b97bd19801ec95f8c965cc920f', |
||||
'97bd09801d98082c95f8e1cfcc920f', '97bd097bd097c36b0b6fc9210c8dc2', '9778397bd197c36c9210c9274c91aa', |
||||
'97b6b97bd19801ec95f8c965cc920e', '97bd09801d98082c95f8e1cfcc920f', '97bd097bd097c36b0b6fc9210c8dc2', |
||||
'9778397bd097c36c9210c9274c91aa', '97b6b97bd19801ec95f8c965cc920e', '97bcf97c3598082c95f8e1cfcc920f', |
||||
'97bd097bd097c36b0b6fc9210c8dc2', '9778397bd097c36c9210c9274c91aa', '97b6b97bd19801ec9210c965cc920e', |
||||
'97bcf97c3598082c95f8c965cc920f', '97bd097bd097c35b0b6fc920fb0722', '9778397bd097c36b0b6fc9274c91aa', |
||||
'97b6b97bd19801ec9210c965cc920e', '97bcf97c3598082c95f8c965cc920f', '97bd097bd097c35b0b6fc920fb0722', |
||||
'9778397bd097c36b0b6fc9274c91aa', '97b6b97bd19801ec9210c965cc920e', '97bcf97c359801ec95f8c965cc920f', |
||||
'97bd097bd097c35b0b6fc920fb0722', '9778397bd097c36b0b6fc9274c91aa', '97b6b97bd19801ec9210c965cc920e', |
||||
'97bcf97c359801ec95f8c965cc920f', '97bd097bd097c35b0b6fc920fb0722', '9778397bd097c36b0b6fc9274c91aa', |
||||
'97b6b97bd19801ec9210c965cc920e', '97bcf97c359801ec95f8c965cc920f', '97bd097bd07f595b0b6fc920fb0722', |
||||
'9778397bd097c36b0b6fc9210c8dc2', '9778397bd19801ec9210c9274c920e', '97b6b97bd19801ec95f8c965cc920f', |
||||
'97bd07f5307f595b0b0bc920fb0722', '7f0e397bd097c36b0b6fc9210c8dc2', '9778397bd097c36c9210c9274c920e', |
||||
'97b6b97bd19801ec95f8c965cc920f', '97bd07f5307f595b0b0bc920fb0722', '7f0e397bd097c36b0b6fc9210c8dc2', |
||||
'9778397bd097c36c9210c9274c91aa', '97b6b97bd19801ec9210c965cc920e', '97bd07f1487f595b0b0bc920fb0722', |
||||
'7f0e397bd097c36b0b6fc9210c8dc2', '9778397bd097c36b0b6fc9274c91aa', '97b6b97bd19801ec9210c965cc920e', |
||||
'97bcf7f1487f595b0b0bb0b6fb0722', '7f0e397bd097c35b0b6fc920fb0722', '9778397bd097c36b0b6fc9274c91aa', |
||||
'97b6b97bd19801ec9210c965cc920e', '97bcf7f1487f595b0b0bb0b6fb0722', '7f0e397bd097c35b0b6fc920fb0722', |
||||
'9778397bd097c36b0b6fc9274c91aa', '97b6b97bd19801ec9210c965cc920e', '97bcf7f1487f531b0b0bb0b6fb0722', |
||||
'7f0e397bd097c35b0b6fc920fb0722', '9778397bd097c36b0b6fc9274c91aa', '97b6b97bd19801ec9210c965cc920e', |
||||
'97bcf7f1487f531b0b0bb0b6fb0722', '7f0e397bd07f595b0b6fc920fb0722', '9778397bd097c36b0b6fc9274c91aa', |
||||
'97b6b97bd19801ec9210c9274c920e', '97bcf7f0e47f531b0b0bb0b6fb0722', '7f0e397bd07f595b0b0bc920fb0722', |
||||
'9778397bd097c36b0b6fc9210c91aa', '97b6b97bd197c36c9210c9274c920e', '97bcf7f0e47f531b0b0bb0b6fb0722', |
||||
'7f0e397bd07f595b0b0bc920fb0722', '9778397bd097c36b0b6fc9210c8dc2', '9778397bd097c36c9210c9274c920e', |
||||
'97b6b7f0e47f531b0723b0b6fb0722', '7f0e37f5307f595b0b0bc920fb0722', '7f0e397bd097c36b0b6fc9210c8dc2', |
||||
'9778397bd097c36b0b70c9274c91aa', '97b6b7f0e47f531b0723b0b6fb0721', '7f0e37f1487f595b0b0bb0b6fb0722', |
||||
'7f0e397bd097c35b0b6fc9210c8dc2', '9778397bd097c36b0b6fc9274c91aa', '97b6b7f0e47f531b0723b0b6fb0721', |
||||
'7f0e27f1487f595b0b0bb0b6fb0722', '7f0e397bd097c35b0b6fc920fb0722', '9778397bd097c36b0b6fc9274c91aa', |
||||
'97b6b7f0e47f531b0723b0b6fb0721', '7f0e27f1487f531b0b0bb0b6fb0722', '7f0e397bd097c35b0b6fc920fb0722', |
||||
'9778397bd097c36b0b6fc9274c91aa', '97b6b7f0e47f531b0723b0b6fb0721', '7f0e27f1487f531b0b0bb0b6fb0722', |
||||
'7f0e397bd097c35b0b6fc920fb0722', '9778397bd097c36b0b6fc9274c91aa', '97b6b7f0e47f531b0723b0b6fb0721', |
||||
'7f0e27f1487f531b0b0bb0b6fb0722', '7f0e397bd07f595b0b0bc920fb0722', '9778397bd097c36b0b6fc9274c91aa', |
||||
'97b6b7f0e47f531b0723b0787b0721', '7f0e27f0e47f531b0b0bb0b6fb0722', '7f0e397bd07f595b0b0bc920fb0722', |
||||
'9778397bd097c36b0b6fc9210c91aa', '97b6b7f0e47f149b0723b0787b0721', '7f0e27f0e47f531b0723b0b6fb0722', |
||||
'7f0e397bd07f595b0b0bc920fb0722', '9778397bd097c36b0b6fc9210c8dc2', '977837f0e37f149b0723b0787b0721', |
||||
'7f07e7f0e47f531b0723b0b6fb0722', '7f0e37f5307f595b0b0bc920fb0722', '7f0e397bd097c35b0b6fc9210c8dc2', |
||||
'977837f0e37f14998082b0787b0721', '7f07e7f0e47f531b0723b0b6fb0721', '7f0e37f1487f595b0b0bb0b6fb0722', |
||||
'7f0e397bd097c35b0b6fc9210c8dc2', '977837f0e37f14998082b0787b06bd', '7f07e7f0e47f531b0723b0b6fb0721', |
||||
'7f0e27f1487f531b0b0bb0b6fb0722', '7f0e397bd097c35b0b6fc920fb0722', '977837f0e37f14998082b0787b06bd', |
||||
'7f07e7f0e47f531b0723b0b6fb0721', '7f0e27f1487f531b0b0bb0b6fb0722', '7f0e397bd097c35b0b6fc920fb0722', |
||||
'977837f0e37f14998082b0787b06bd', '7f07e7f0e47f531b0723b0b6fb0721', '7f0e27f1487f531b0b0bb0b6fb0722', |
||||
'7f0e397bd07f595b0b0bc920fb0722', '977837f0e37f14998082b0787b06bd', '7f07e7f0e47f531b0723b0b6fb0721', |
||||
'7f0e27f1487f531b0b0bb0b6fb0722', '7f0e397bd07f595b0b0bc920fb0722', '977837f0e37f14998082b0787b06bd', |
||||
'7f07e7f0e47f149b0723b0787b0721', '7f0e27f0e47f531b0b0bb0b6fb0722', '7f0e397bd07f595b0b0bc920fb0722', |
||||
'977837f0e37f14998082b0723b06bd', '7f07e7f0e37f149b0723b0787b0721', '7f0e27f0e47f531b0723b0b6fb0722', |
||||
'7f0e397bd07f595b0b0bc920fb0722', '977837f0e37f14898082b0723b02d5', '7ec967f0e37f14998082b0787b0721', |
||||
'7f07e7f0e47f531b0723b0b6fb0722', '7f0e37f1487f595b0b0bb0b6fb0722', '7f0e37f0e37f14898082b0723b02d5', |
||||
'7ec967f0e37f14998082b0787b0721', '7f07e7f0e47f531b0723b0b6fb0722', '7f0e37f1487f531b0b0bb0b6fb0722', |
||||
'7f0e37f0e37f14898082b0723b02d5', '7ec967f0e37f14998082b0787b06bd', '7f07e7f0e47f531b0723b0b6fb0721', |
||||
'7f0e37f1487f531b0b0bb0b6fb0722', '7f0e37f0e37f14898082b072297c35', '7ec967f0e37f14998082b0787b06bd', |
||||
'7f07e7f0e47f531b0723b0b6fb0721', '7f0e27f1487f531b0b0bb0b6fb0722', '7f0e37f0e37f14898082b072297c35', |
||||
'7ec967f0e37f14998082b0787b06bd', '7f07e7f0e47f531b0723b0b6fb0721', '7f0e27f1487f531b0b0bb0b6fb0722', |
||||
'7f0e37f0e366aa89801eb072297c35', '7ec967f0e37f14998082b0787b06bd', '7f07e7f0e47f149b0723b0787b0721', |
||||
'7f0e27f1487f531b0b0bb0b6fb0722', '7f0e37f0e366aa89801eb072297c35', '7ec967f0e37f14998082b0723b06bd', |
||||
'7f07e7f0e47f149b0723b0787b0721', '7f0e27f0e47f531b0723b0b6fb0722', '7f0e37f0e366aa89801eb072297c35', |
||||
'7ec967f0e37f14998082b0723b06bd', '7f07e7f0e37f14998083b0787b0721', '7f0e27f0e47f531b0723b0b6fb0722', |
||||
'7f0e37f0e366aa89801eb072297c35', '7ec967f0e37f14898082b0723b02d5', '7f07e7f0e37f14998082b0787b0721', |
||||
'7f07e7f0e47f531b0723b0b6fb0722', '7f0e36665b66aa89801e9808297c35', '665f67f0e37f14898082b0723b02d5', |
||||
'7ec967f0e37f14998082b0787b0721', '7f07e7f0e47f531b0723b0b6fb0722', '7f0e36665b66a449801e9808297c35', |
||||
'665f67f0e37f14898082b0723b02d5', '7ec967f0e37f14998082b0787b06bd', '7f07e7f0e47f531b0723b0b6fb0721', |
||||
'7f0e36665b66a449801e9808297c35', '665f67f0e37f14898082b072297c35', '7ec967f0e37f14998082b0787b06bd', |
||||
'7f07e7f0e47f531b0723b0b6fb0721', '7f0e26665b66a449801e9808297c35', '665f67f0e37f1489801eb072297c35', |
||||
'7ec967f0e37f14998082b0787b06bd', '7f07e7f0e47f531b0723b0b6fb0721', '7f0e27f1487f531b0b0bb0b6fb0722'], |
||||
|
||||
/** |
||||
* 数字转中文速查表 |
||||
* @Array Of Property |
||||
* @trans ['日','一','二','三','四','五','六','七','八','九','十'] |
||||
* @return Cn string |
||||
*/ |
||||
nStr1: ['\u65e5', '\u4e00', '\u4e8c', '\u4e09', '\u56db', '\u4e94', '\u516d', '\u4e03', '\u516b', '\u4e5d', '\u5341'], |
||||
|
||||
/** |
||||
* 日期转农历称呼速查表 |
||||
* @Array Of Property |
||||
* @trans ['初','十','廿','卅'] |
||||
* @return Cn string |
||||
*/ |
||||
nStr2: ['\u521d', '\u5341', '\u5eff', '\u5345'], |
||||
|
||||
/** |
||||
* 月份转农历称呼速查表 |
||||
* @Array Of Property |
||||
* @trans ['正','一','二','三','四','五','六','七','八','九','十','冬','腊'] |
||||
* @return Cn string |
||||
*/ |
||||
nStr3: ['\u6b63', '\u4e8c', '\u4e09', '\u56db', '\u4e94', '\u516d', '\u4e03', '\u516b', '\u4e5d', '\u5341', '\u51ac', '\u814a'], |
||||
|
||||
/** |
||||
* 返回农历y年一整年的总天数 |
||||
* @param lunar Year |
||||
* @return Number |
||||
* @eg:var count = calendar.lYearDays(1987) ;//count=387
|
||||
*/ |
||||
lYearDays: function (y) { |
||||
var i; var sum = 348 |
||||
for (i = 0x8000; i > 0x8; i >>= 1) { sum += (this.lunarInfo[y - 1900] & i) ? 1 : 0 } |
||||
return (sum + this.leapDays(y)) |
||||
}, |
||||
|
||||
/** |
||||
* 返回农历y年闰月是哪个月;若y年没有闰月 则返回0 |
||||
* @param lunar Year |
||||
* @return Number (0-12) |
||||
* @eg:var leapMonth = calendar.leapMonth(1987) ;//leapMonth=6
|
||||
*/ |
||||
leapMonth: function (y) { // 闰字编码 \u95f0
|
||||
return (this.lunarInfo[y - 1900] & 0xf) |
||||
}, |
||||
|
||||
/** |
||||
* 返回农历y年闰月的天数 若该年没有闰月则返回0 |
||||
* @param lunar Year |
||||
* @return Number (0、29、30) |
||||
* @eg:var leapMonthDay = calendar.leapDays(1987) ;//leapMonthDay=29
|
||||
*/ |
||||
leapDays: function (y) { |
||||
if (this.leapMonth(y)) { |
||||
return ((this.lunarInfo[y - 1900] & 0x10000) ? 30 : 29) |
||||
} |
||||
return (0) |
||||
}, |
||||
|
||||
/** |
||||
* 返回农历y年m月(非闰月)的总天数,计算m为闰月时的天数请使用leapDays方法 |
||||
* @param lunar Year |
||||
* @return Number (-1、29、30) |
||||
* @eg:var MonthDay = calendar.monthDays(1987,9) ;//MonthDay=29
|
||||
*/ |
||||
monthDays: function (y, m) { |
||||
if (m > 12 || m < 1) { return -1 }// 月份参数从1至12,参数错误返回-1
|
||||
return ((this.lunarInfo[y - 1900] & (0x10000 >> m)) ? 30 : 29) |
||||
}, |
||||
|
||||
/** |
||||
* 返回公历(!)y年m月的天数 |
||||
* @param solar Year |
||||
* @return Number (-1、28、29、30、31) |
||||
* @eg:var solarMonthDay = calendar.leapDays(1987) ;//solarMonthDay=30
|
||||
*/ |
||||
solarDays: function (y, m) { |
||||
if (m > 12 || m < 1) { return -1 } // 若参数错误 返回-1
|
||||
var ms = m - 1 |
||||
if (ms == 1) { // 2月份的闰平规律测算后确认返回28或29
|
||||
return (((y % 4 == 0) && (y % 100 != 0) || (y % 400 == 0)) ? 29 : 28) |
||||
} else { |
||||
return (this.solarMonth[ms]) |
||||
} |
||||
}, |
||||
|
||||
/** |
||||
* 农历年份转换为干支纪年 |
||||
* @param lYear 农历年的年份数 |
||||
* @return Cn string |
||||
*/ |
||||
toGanZhiYear: function (lYear) { |
||||
var ganKey = (lYear - 3) % 10 |
||||
var zhiKey = (lYear - 3) % 12 |
||||
if (ganKey == 0) ganKey = 10// 如果余数为0则为最后一个天干
|
||||
if (zhiKey == 0) zhiKey = 12// 如果余数为0则为最后一个地支
|
||||
return this.Gan[ganKey - 1] + this.Zhi[zhiKey - 1] |
||||
}, |
||||
|
||||
/** |
||||
* 公历月、日判断所属星座 |
||||
* @param cMonth [description] |
||||
* @param cDay [description] |
||||
* @return Cn string |
||||
*/ |
||||
toAstro: function (cMonth, cDay) { |
||||
var s = '\u9b54\u7faf\u6c34\u74f6\u53cc\u9c7c\u767d\u7f8a\u91d1\u725b\u53cc\u5b50\u5de8\u87f9\u72ee\u5b50\u5904\u5973\u5929\u79e4\u5929\u874e\u5c04\u624b\u9b54\u7faf' |
||||
var arr = [20, 19, 21, 21, 21, 22, 23, 23, 23, 23, 22, 22] |
||||
return s.substr(cMonth * 2 - (cDay < arr[cMonth - 1] ? 2 : 0), 2) + '\u5ea7'// 座
|
||||
}, |
||||
|
||||
/** |
||||
* 传入offset偏移量返回干支 |
||||
* @param offset 相对甲子的偏移量 |
||||
* @return Cn string |
||||
*/ |
||||
toGanZhi: function (offset) { |
||||
return this.Gan[offset % 10] + this.Zhi[offset % 12] |
||||
}, |
||||
|
||||
/** |
||||
* 传入公历(!)y年获得该年第n个节气的公历日期 |
||||
* @param y公历年(1900-2100);n二十四节气中的第几个节气(1~24);从n=1(小寒)算起 |
||||
* @return day Number |
||||
* @eg:var _24 = calendar.getTerm(1987,3) ;//_24=4;意即1987年2月4日立春
|
||||
*/ |
||||
getTerm: function (y, n) { |
||||
if (y < 1900 || y > 2100) { return -1 } |
||||
if (n < 1 || n > 24) { return -1 } |
||||
var _table = this.sTermInfo[y - 1900] |
||||
var _info = [ |
||||
parseInt('0x' + _table.substr(0, 5)).toString(), |
||||
parseInt('0x' + _table.substr(5, 5)).toString(), |
||||
parseInt('0x' + _table.substr(10, 5)).toString(), |
||||
parseInt('0x' + _table.substr(15, 5)).toString(), |
||||
parseInt('0x' + _table.substr(20, 5)).toString(), |
||||
parseInt('0x' + _table.substr(25, 5)).toString() |
||||
] |
||||
var _calday = [ |
||||
_info[0].substr(0, 1), |
||||
_info[0].substr(1, 2), |
||||
_info[0].substr(3, 1), |
||||
_info[0].substr(4, 2), |
||||
|
||||
_info[1].substr(0, 1), |
||||
_info[1].substr(1, 2), |
||||
_info[1].substr(3, 1), |
||||
_info[1].substr(4, 2), |
||||
|
||||
_info[2].substr(0, 1), |
||||
_info[2].substr(1, 2), |
||||
_info[2].substr(3, 1), |
||||
_info[2].substr(4, 2), |
||||
|
||||
_info[3].substr(0, 1), |
||||
_info[3].substr(1, 2), |
||||
_info[3].substr(3, 1), |
||||
_info[3].substr(4, 2), |
||||
|
||||
_info[4].substr(0, 1), |
||||
_info[4].substr(1, 2), |
||||
_info[4].substr(3, 1), |
||||
_info[4].substr(4, 2), |
||||
|
||||
_info[5].substr(0, 1), |
||||
_info[5].substr(1, 2), |
||||
_info[5].substr(3, 1), |
||||
_info[5].substr(4, 2) |
||||
] |
||||
return parseInt(_calday[n - 1]) |
||||
}, |
||||
|
||||
/** |
||||
* 传入农历数字月份返回汉语通俗表示法 |
||||
* @param lunar month |
||||
* @return Cn string |
||||
* @eg:var cnMonth = calendar.toChinaMonth(12) ;//cnMonth='腊月'
|
||||
*/ |
||||
toChinaMonth: function (m) { // 月 => \u6708
|
||||
if (m > 12 || m < 1) { return -1 } // 若参数错误 返回-1
|
||||
var s = this.nStr3[m - 1] |
||||
s += '\u6708'// 加上月字
|
||||
return s |
||||
}, |
||||
|
||||
/** |
||||
* 传入农历日期数字返回汉字表示法 |
||||
* @param lunar day |
||||
* @return Cn string |
||||
* @eg:var cnDay = calendar.toChinaDay(21) ;//cnMonth='廿一'
|
||||
*/ |
||||
toChinaDay: function (d) { // 日 => \u65e5
|
||||
var s |
||||
switch (d) { |
||||
case 10: |
||||
s = '\u521d\u5341'; break |
||||
case 20: |
||||
s = '\u4e8c\u5341'; break |
||||
break |
||||
case 30: |
||||
s = '\u4e09\u5341'; break |
||||
break |
||||
default : |
||||
s = this.nStr2[Math.floor(d / 10)] |
||||
s += this.nStr1[d % 10] |
||||
} |
||||
return (s) |
||||
}, |
||||
|
||||
/** |
||||
* 年份转生肖[!仅能大致转换] => 精确划分生肖分界线是“立春” |
||||
* @param y year |
||||
* @return Cn string |
||||
* @eg:var animal = calendar.getAnimal(1987) ;//animal='兔'
|
||||
*/ |
||||
getAnimal: function (y) { |
||||
return this.Animals[(y - 4) % 12] |
||||
}, |
||||
|
||||
/** |
||||
* 传入阳历年月日获得详细的公历、农历object信息 <=>JSON |
||||
* @param y solar year |
||||
* @param m solar month |
||||
* @param d solar day |
||||
* @return JSON object |
||||
* @eg:console.log(calendar.solar2lunar(1987,11,01)); |
||||
*/ |
||||
solar2lunar: function (y, m, d) { // 参数区间1900.1.31~2100.12.31
|
||||
// 年份限定、上限
|
||||
if (y < 1900 || y > 2100) { |
||||
return -1// undefined转换为数字变为NaN
|
||||
} |
||||
// 公历传参最下限
|
||||
if (y == 1900 && m == 1 && d < 31) { |
||||
return -1 |
||||
} |
||||
// 未传参 获得当天
|
||||
if (!y) { |
||||
var objDate = new Date() |
||||
} else { |
||||
var objDate = new Date(y, parseInt(m) - 1, d) |
||||
} |
||||
var i; var leap = 0; var temp = 0 |
||||
// 修正ymd参数
|
||||
var y = objDate.getFullYear() |
||||
var m = objDate.getMonth() + 1 |
||||
var d = objDate.getDate() |
||||
var offset = (Date.UTC(objDate.getFullYear(), objDate.getMonth(), objDate.getDate()) - Date.UTC(1900, 0, 31)) / 86400000 |
||||
for (i = 1900; i < 2101 && offset > 0; i++) { |
||||
temp = this.lYearDays(i) |
||||
offset -= temp |
||||
} |
||||
if (offset < 0) { |
||||
offset += temp; i-- |
||||
} |
||||
|
||||
// 是否今天
|
||||
var isTodayObj = new Date() |
||||
var isToday = false |
||||
if (isTodayObj.getFullYear() == y && isTodayObj.getMonth() + 1 == m && isTodayObj.getDate() == d) { |
||||
isToday = true |
||||
} |
||||
// 星期几
|
||||
var nWeek = objDate.getDay() |
||||
var cWeek = this.nStr1[nWeek] |
||||
// 数字表示周几顺应天朝周一开始的惯例
|
||||
if (nWeek == 0) { |
||||
nWeek = 7 |
||||
} |
||||
// 农历年
|
||||
var year = i |
||||
var leap = this.leapMonth(i) // 闰哪个月
|
||||
var isLeap = false |
||||
|
||||
// 效验闰月
|
||||
for (i = 1; i < 13 && offset > 0; i++) { |
||||
// 闰月
|
||||
if (leap > 0 && i == (leap + 1) && isLeap == false) { |
||||
--i |
||||
isLeap = true; temp = this.leapDays(year) // 计算农历闰月天数
|
||||
} else { |
||||
temp = this.monthDays(year, i)// 计算农历普通月天数
|
||||
} |
||||
// 解除闰月
|
||||
if (isLeap == true && i == (leap + 1)) { isLeap = false } |
||||
offset -= temp |
||||
} |
||||
// 闰月导致数组下标重叠取反
|
||||
if (offset == 0 && leap > 0 && i == leap + 1) { |
||||
if (isLeap) { |
||||
isLeap = false |
||||
} else { |
||||
isLeap = true; --i |
||||
} |
||||
} |
||||
if (offset < 0) { |
||||
offset += temp; --i |
||||
} |
||||
// 农历月
|
||||
var month = i |
||||
// 农历日
|
||||
var day = offset + 1 |
||||
// 天干地支处理
|
||||
var sm = m - 1 |
||||
var gzY = this.toGanZhiYear(year) |
||||
|
||||
// 当月的两个节气
|
||||
// bugfix-2017-7-24 11:03:38 use lunar Year Param `y` Not `year`
|
||||
var firstNode = this.getTerm(y, (m * 2 - 1))// 返回当月「节」为几日开始
|
||||
var secondNode = this.getTerm(y, (m * 2))// 返回当月「节」为几日开始
|
||||
|
||||
// 依据12节气修正干支月
|
||||
var gzM = this.toGanZhi((y - 1900) * 12 + m + 11) |
||||
if (d >= firstNode) { |
||||
gzM = this.toGanZhi((y - 1900) * 12 + m + 12) |
||||
} |
||||
|
||||
// 传入的日期的节气与否
|
||||
var isTerm = false |
||||
var Term = null |
||||
if (firstNode == d) { |
||||
isTerm = true |
||||
Term = this.solarTerm[m * 2 - 2] |
||||
} |
||||
if (secondNode == d) { |
||||
isTerm = true |
||||
Term = this.solarTerm[m * 2 - 1] |
||||
} |
||||
// 日柱 当月一日与 1900/1/1 相差天数
|
||||
var dayCyclical = Date.UTC(y, sm, 1, 0, 0, 0, 0) / 86400000 + 25567 + 10 |
||||
var gzD = this.toGanZhi(dayCyclical + d - 1) |
||||
// 该日期所属的星座
|
||||
var astro = this.toAstro(m, d) |
||||
|
||||
return { 'lYear': year, 'lMonth': month, 'lDay': day, 'Animal': this.getAnimal(year), 'IMonthCn': (isLeap ? '\u95f0' : '') + this.toChinaMonth(month), 'IDayCn': this.toChinaDay(day), 'cYear': y, 'cMonth': m, 'cDay': d, 'gzYear': gzY, 'gzMonth': gzM, 'gzDay': gzD, 'isToday': isToday, 'isLeap': isLeap, 'nWeek': nWeek, 'ncWeek': '\u661f\u671f' + cWeek, 'isTerm': isTerm, 'Term': Term, 'astro': astro } |
||||
}, |
||||
|
||||
/** |
||||
* 传入农历年月日以及传入的月份是否闰月获得详细的公历、农历object信息 <=>JSON |
||||
* @param y lunar year |
||||
* @param m lunar month |
||||
* @param d lunar day |
||||
* @param isLeapMonth lunar month is leap or not.[如果是农历闰月第四个参数赋值true即可] |
||||
* @return JSON object |
||||
* @eg:console.log(calendar.lunar2solar(1987,9,10)); |
||||
*/ |
||||
lunar2solar: function (y, m, d, isLeapMonth) { // 参数区间1900.1.31~2100.12.1
|
||||
var isLeapMonth = !!isLeapMonth |
||||
var leapOffset = 0 |
||||
var leapMonth = this.leapMonth(y) |
||||
var leapDay = this.leapDays(y) |
||||
if (isLeapMonth && (leapMonth != m)) { return -1 }// 传参要求计算该闰月公历 但该年得出的闰月与传参的月份并不同
|
||||
if (y == 2100 && m == 12 && d > 1 || y == 1900 && m == 1 && d < 31) { return -1 }// 超出了最大极限值
|
||||
var day = this.monthDays(y, m) |
||||
var _day = day |
||||
// bugFix 2016-9-25
|
||||
// if month is leap, _day use leapDays method
|
||||
if (isLeapMonth) { |
||||
_day = this.leapDays(y, m) |
||||
} |
||||
if (y < 1900 || y > 2100 || d > _day) { return -1 }// 参数合法性效验
|
||||
|
||||
// 计算农历的时间差
|
||||
var offset = 0 |
||||
for (var i = 1900; i < y; i++) { |
||||
offset += this.lYearDays(i) |
||||
} |
||||
var leap = 0; var isAdd = false |
||||
for (var i = 1; i < m; i++) { |
||||
leap = this.leapMonth(y) |
||||
if (!isAdd) { // 处理闰月
|
||||
if (leap <= i && leap > 0) { |
||||
offset += this.leapDays(y); isAdd = true |
||||
} |
||||
} |
||||
offset += this.monthDays(y, i) |
||||
} |
||||
// 转换闰月农历 需补充该年闰月的前一个月的时差
|
||||
if (isLeapMonth) { offset += day } |
||||
// 1900年农历正月一日的公历时间为1900年1月30日0时0分0秒(该时间也是本农历的最开始起始点)
|
||||
var stmap = Date.UTC(1900, 1, 30, 0, 0, 0) |
||||
var calObj = new Date((offset + d - 31) * 86400000 + stmap) |
||||
var cY = calObj.getUTCFullYear() |
||||
var cM = calObj.getUTCMonth() + 1 |
||||
var cD = calObj.getUTCDate() |
||||
|
||||
return this.solar2lunar(cY, cM, cD) |
||||
} |
||||
} |
||||
|
||||
export default calendar |
@ -0,0 +1,12 @@ |
||||
{ |
||||
"uni-calender.ok": "ok", |
||||
"uni-calender.cancel": "cancel", |
||||
"uni-calender.today": "today", |
||||
"uni-calender.MON": "MON", |
||||
"uni-calender.TUE": "TUE", |
||||
"uni-calender.WED": "WED", |
||||
"uni-calender.THU": "THU", |
||||
"uni-calender.FRI": "FRI", |
||||
"uni-calender.SAT": "SAT", |
||||
"uni-calender.SUN": "SUN" |
||||
} |
@ -0,0 +1,8 @@ |
||||
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 |
||||
} |
@ -0,0 +1,12 @@ |
||||
{ |
||||
"uni-calender.ok": "确定", |
||||
"uni-calender.cancel": "取消", |
||||
"uni-calender.today": "今日", |
||||
"uni-calender.SUN": "日", |
||||
"uni-calender.MON": "一", |
||||
"uni-calender.TUE": "二", |
||||
"uni-calender.WED": "三", |
||||
"uni-calender.THU": "四", |
||||
"uni-calender.FRI": "五", |
||||
"uni-calender.SAT": "六" |
||||
} |
@ -0,0 +1,12 @@ |
||||
{ |
||||
"uni-calender.ok": "確定", |
||||
"uni-calender.cancel": "取消", |
||||
"uni-calender.today": "今日", |
||||
"uni-calender.SUN": "日", |
||||
"uni-calender.MON": "一", |
||||
"uni-calender.TUE": "二", |
||||
"uni-calender.WED": "三", |
||||
"uni-calender.THU": "四", |
||||
"uni-calender.FRI": "五", |
||||
"uni-calender.SAT": "六" |
||||
} |
@ -0,0 +1,188 @@ |
||||
<template> |
||||
<view class="uni-calendar-item__weeks-box" :class="{ |
||||
'uni-calendar-item--disable':weeks.disable, |
||||
'uni-calendar-item--isDay':calendar.fullDate === weeks.fullDate && weeks.isDay, |
||||
'uni-calendar-item--checked':(calendar.fullDate === weeks.fullDate && !weeks.isDay) , |
||||
'uni-calendar-item--before-checked':weeks.beforeMultiple, |
||||
'uni-calendar-item--multiple': weeks.multiple, |
||||
'uni-calendar-item--after-checked':weeks.afterMultiple, |
||||
}" |
||||
@click="choiceDate(weeks)"> |
||||
<view class="uni-calendar-item__weeks-box-item"> |
||||
<text v-if="selected&&weeks.extraInfo" class="uni-calendar-item__weeks-box-circle"></text> |
||||
<text class="uni-calendar-item__weeks-box-text" :class="{ |
||||
'uni-calendar-item--isDay-text': weeks.isDay, |
||||
'uni-calendar-item--isDay':calendar.fullDate === weeks.fullDate && weeks.isDay, |
||||
'uni-calendar-item--checked':calendar.fullDate === weeks.fullDate && !weeks.isDay, |
||||
'uni-calendar-item--before-checked':weeks.beforeMultiple, |
||||
'uni-calendar-item--multiple': weeks.multiple, |
||||
'uni-calendar-item--after-checked':weeks.afterMultiple, |
||||
'uni-calendar-item--disable':weeks.disable, |
||||
}">{{weeks.date}}</text> |
||||
<text v-if="!lunar&&!weeks.extraInfo && weeks.isDay" class="uni-calendar-item__weeks-lunar-text" :class="{ |
||||
'uni-calendar-item--isDay-text':weeks.isDay, |
||||
'uni-calendar-item--isDay':calendar.fullDate === weeks.fullDate && weeks.isDay, |
||||
'uni-calendar-item--checked':calendar.fullDate === weeks.fullDate && !weeks.isDay, |
||||
'uni-calendar-item--before-checked':weeks.beforeMultiple, |
||||
'uni-calendar-item--multiple': weeks.multiple, |
||||
'uni-calendar-item--after-checked':weeks.afterMultiple, |
||||
}">{{todayText}}</text> |
||||
<text v-if="lunar&&!weeks.extraInfo" class="uni-calendar-item__weeks-lunar-text" :class="{ |
||||
'uni-calendar-item--isDay-text':weeks.isDay, |
||||
'uni-calendar-item--isDay':calendar.fullDate === weeks.fullDate && weeks.isDay, |
||||
'uni-calendar-item--checked':calendar.fullDate === weeks.fullDate && !weeks.isDay, |
||||
'uni-calendar-item--before-checked':weeks.beforeMultiple, |
||||
'uni-calendar-item--multiple': weeks.multiple, |
||||
'uni-calendar-item--after-checked':weeks.afterMultiple, |
||||
'uni-calendar-item--disable':weeks.disable, |
||||
}">{{weeks.isDay ? todayText : (weeks.lunar.IDayCn === '初一'?weeks.lunar.IMonthCn:weeks.lunar.IDayCn)}}</text> |
||||
<text v-if="weeks.extraInfo&&weeks.extraInfo.info" class="uni-calendar-item__weeks-lunar-text" :class="{ |
||||
'uni-calendar-item--extra':weeks.extraInfo.info, |
||||
'uni-calendar-item--isDay-text':weeks.isDay, |
||||
'uni-calendar-item--isDay':calendar.fullDate === weeks.fullDate && weeks.isDay, |
||||
'uni-calendar-item--checked':calendar.fullDate === weeks.fullDate && !weeks.isDay, |
||||
'uni-calendar-item--before-checked':weeks.beforeMultiple, |
||||
'uni-calendar-item--multiple': weeks.multiple, |
||||
'uni-calendar-item--after-checked':weeks.afterMultiple, |
||||
'uni-calendar-item--disable':weeks.disable, |
||||
}">{{weeks.extraInfo.info}}</text> |
||||
</view> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
import { |
||||
initVueI18n |
||||
} from '@dcloudio/uni-i18n' |
||||
import messages from './i18n/index.js' |
||||
const { t } = initVueI18n(messages) |
||||
export default { |
||||
emits:['change'], |
||||
props: { |
||||
weeks: { |
||||
type: Object, |
||||
default () { |
||||
return {} |
||||
} |
||||
}, |
||||
calendar: { |
||||
type: Object, |
||||
default: () => { |
||||
return {} |
||||
} |
||||
}, |
||||
selected: { |
||||
type: Array, |
||||
default: () => { |
||||
return [] |
||||
} |
||||
}, |
||||
lunar: { |
||||
type: Boolean, |
||||
default: false |
||||
} |
||||
}, |
||||
computed: { |
||||
todayText() { |
||||
return t("uni-calender.today") |
||||
}, |
||||
}, |
||||
methods: { |
||||
choiceDate(weeks) { |
||||
this.$emit('change', weeks) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss" scoped> |
||||
$uni-font-size-base:14px; |
||||
$uni-text-color:#333; |
||||
$uni-font-size-sm:12px; |
||||
$uni-color-error: #e43d33; |
||||
$uni-opacity-disabled: 0.3; |
||||
$uni-text-color-disable:#c0c0c0; |
||||
$uni-primary: #2979ff !default; |
||||
.uni-calendar-item__weeks-box { |
||||
flex: 1; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: column; |
||||
justify-content: center; |
||||
align-items: center; |
||||
} |
||||
|
||||
.uni-calendar-item__weeks-box-text { |
||||
font-size: $uni-font-size-base; |
||||
color: $uni-text-color; |
||||
} |
||||
|
||||
.uni-calendar-item__weeks-lunar-text { |
||||
font-size: $uni-font-size-sm; |
||||
color: $uni-text-color; |
||||
} |
||||
|
||||
.uni-calendar-item__weeks-box-item { |
||||
position: relative; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: column; |
||||
justify-content: center; |
||||
align-items: center; |
||||
width: 100rpx; |
||||
height: 100rpx; |
||||
} |
||||
|
||||
.uni-calendar-item__weeks-box-circle { |
||||
position: absolute; |
||||
top: 5px; |
||||
right: 5px; |
||||
width: 8px; |
||||
height: 8px; |
||||
border-radius: 8px; |
||||
background-color: $uni-color-error; |
||||
|
||||
} |
||||
|
||||
.uni-calendar-item--disable { |
||||
background-color: rgba(249, 249, 249, $uni-opacity-disabled); |
||||
color: $uni-text-color-disable; |
||||
} |
||||
|
||||
.uni-calendar-item--isDay-text { |
||||
color: $uni-primary; |
||||
} |
||||
|
||||
.uni-calendar-item--isDay { |
||||
background-color: $uni-primary; |
||||
opacity: 0.8; |
||||
color: #fff; |
||||
} |
||||
|
||||
.uni-calendar-item--extra { |
||||
color: $uni-color-error; |
||||
opacity: 0.8; |
||||
} |
||||
|
||||
.uni-calendar-item--checked { |
||||
background-color: $uni-primary; |
||||
color: #fff; |
||||
opacity: 0.8; |
||||
} |
||||
|
||||
.uni-calendar-item--multiple { |
||||
background-color: $uni-primary; |
||||
color: #fff; |
||||
opacity: 0.8; |
||||
} |
||||
.uni-calendar-item--before-checked { |
||||
background-color: #ff5a5f; |
||||
color: #fff; |
||||
} |
||||
.uni-calendar-item--after-checked { |
||||
background-color: #ff5a5f; |
||||
color: #fff; |
||||
} |
||||
</style> |
@ -0,0 +1,562 @@ |
||||
<template> |
||||
<view class="uni-calendar"> |
||||
<view v-if="!insert&&show" class="uni-calendar__mask" :class="{'uni-calendar--mask-show':aniMaskShow}" @click="clean"></view> |
||||
<view v-if="insert || show" class="uni-calendar__content" :class="{'uni-calendar--fixed':!insert,'uni-calendar--ani-show':aniMaskShow}"> |
||||
<view v-if="!insert" class="uni-calendar__header uni-calendar--fixed-top"> |
||||
<view class="uni-calendar__header-btn-box" @click="close"> |
||||
<text class="uni-calendar__header-text uni-calendar--fixed-width">{{cancelText}}</text> |
||||
</view> |
||||
<view class="uni-calendar__header-btn-box" @click="confirm"> |
||||
<text class="uni-calendar__header-text uni-calendar--fixed-width">{{okText}}</text> |
||||
</view> |
||||
</view> |
||||
<view class="uni-calendar__header"> |
||||
<view class="uni-calendar__header-btn-box" @click.stop="pre"> |
||||
<view class="uni-calendar__header-btn uni-calendar--left"></view> |
||||
</view> |
||||
<picker mode="date" :value="date" fields="month" @change="bindDateChange"> |
||||
<text class="uni-calendar__header-text">{{ (nowDate.year||'') +' / '+( nowDate.month||'')}}</text> |
||||
</picker> |
||||
<view class="uni-calendar__header-btn-box" @click.stop="next"> |
||||
<view class="uni-calendar__header-btn uni-calendar--right"></view> |
||||
</view> |
||||
<text class="uni-calendar__backtoday" @click="backtoday">{{todayText}}</text> |
||||
|
||||
</view> |
||||
<view class="uni-calendar__box"> |
||||
<view v-if="showMonth" class="uni-calendar__box-bg"> |
||||
<text class="uni-calendar__box-bg-text">{{nowDate.month}}</text> |
||||
</view> |
||||
<view class="uni-calendar__weeks"> |
||||
<view class="uni-calendar__weeks-day"> |
||||
<text class="uni-calendar__weeks-day-text">{{SUNText}}</text> |
||||
</view> |
||||
<view class="uni-calendar__weeks-day"> |
||||
<text class="uni-calendar__weeks-day-text">{{monText}}</text> |
||||
</view> |
||||
<view class="uni-calendar__weeks-day"> |
||||
<text class="uni-calendar__weeks-day-text">{{TUEText}}</text> |
||||
</view> |
||||
<view class="uni-calendar__weeks-day"> |
||||
<text class="uni-calendar__weeks-day-text">{{WEDText}}</text> |
||||
</view> |
||||
<view class="uni-calendar__weeks-day"> |
||||
<text class="uni-calendar__weeks-day-text">{{THUText}}</text> |
||||
</view> |
||||
<view class="uni-calendar__weeks-day"> |
||||
<text class="uni-calendar__weeks-day-text">{{FRIText}}</text> |
||||
</view> |
||||
<view class="uni-calendar__weeks-day"> |
||||
<text class="uni-calendar__weeks-day-text">{{SATText}}</text> |
||||
</view> |
||||
</view> |
||||
<view class="uni-calendar__weeks" v-for="(item,weekIndex) in weeks" :key="weekIndex"> |
||||
<view class="uni-calendar__weeks-item" v-for="(weeks,weeksIndex) in item" :key="weeksIndex"> |
||||
<calendar-item class="uni-calendar-item--hook" :weeks="weeks" :calendar="calendar" :selected="selected" :lunar="lunar" @change="choiceDate"></calendar-item> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
import Calendar from './util.js'; |
||||
import calendarItem from './uni-calendar-item.vue' |
||||
import { |
||||
initVueI18n |
||||
} from '@dcloudio/uni-i18n' |
||||
import messages from './i18n/index.js' |
||||
const { t } = initVueI18n(messages) |
||||
/** |
||||
* Calendar 日历 |
||||
* @description 日历组件可以查看日期,选择任意范围内的日期,打点操作。常用场景如:酒店日期预订、火车机票选择购买日期、上下班打卡等 |
||||
* @tutorial https://ext.dcloud.net.cn/plugin?id=56 |
||||
* @property {String} date 自定义当前时间,默认为今天 |
||||
* @property {Boolean} lunar 显示农历 |
||||
* @property {String} startDate 日期选择范围-开始日期 |
||||
* @property {String} endDate 日期选择范围-结束日期 |
||||
* @property {Boolean} range 范围选择 |
||||
* @property {Boolean} insert = [true|false] 插入模式,默认为false |
||||
* @value true 弹窗模式 |
||||
* @value false 插入模式 |
||||
* @property {Boolean} clearDate = [true|false] 弹窗模式是否清空上次选择内容 |
||||
* @property {Array} selected 打点,期待格式[{date: '2019-06-27', info: '签到', data: { custom: '自定义信息', name: '自定义消息头',xxx:xxx... }}] |
||||
* @property {Boolean} showMonth 是否选择月份为背景 |
||||
* @event {Function} change 日期改变,`insert :ture` 时生效 |
||||
* @event {Function} confirm 确认选择`insert :false` 时生效 |
||||
* @event {Function} monthSwitch 切换月份时触发 |
||||
* @example <uni-calendar :insert="true":lunar="true" :start-date="'2019-3-2'":end-date="'2019-5-20'"@change="change" /> |
||||
*/ |
||||
export default { |
||||
components: { |
||||
calendarItem |
||||
}, |
||||
emits:['close','confirm','change','monthSwitch'], |
||||
props: { |
||||
date: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
selected: { |
||||
type: Array, |
||||
default () { |
||||
return [] |
||||
} |
||||
}, |
||||
lunar: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
startDate: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
endDate: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
range: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
insert: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
showMonth: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
clearDate: { |
||||
type: Boolean, |
||||
default: true |
||||
} |
||||
}, |
||||
data() { |
||||
return { |
||||
show: false, |
||||
weeks: [], |
||||
calendar: {}, |
||||
nowDate: '', |
||||
aniMaskShow: false |
||||
} |
||||
}, |
||||
computed:{ |
||||
/** |
||||
* for i18n |
||||
*/ |
||||
|
||||
okText() { |
||||
return t("uni-calender.ok") |
||||
}, |
||||
cancelText() { |
||||
return t("uni-calender.cancel") |
||||
}, |
||||
todayText() { |
||||
return t("uni-calender.today") |
||||
}, |
||||
monText() { |
||||
return t("uni-calender.MON") |
||||
}, |
||||
TUEText() { |
||||
return t("uni-calender.TUE") |
||||
}, |
||||
WEDText() { |
||||
return t("uni-calender.WED") |
||||
}, |
||||
THUText() { |
||||
return t("uni-calender.THU") |
||||
}, |
||||
FRIText() { |
||||
return t("uni-calender.FRI") |
||||
}, |
||||
SATText() { |
||||
return t("uni-calender.SAT") |
||||
}, |
||||
SUNText() { |
||||
return t("uni-calender.SUN") |
||||
}, |
||||
}, |
||||
watch: { |
||||
date(newVal) { |
||||
// this.cale.setDate(newVal) |
||||
this.init(newVal) |
||||
}, |
||||
startDate(val){ |
||||
this.cale.resetSatrtDate(val) |
||||
this.cale.setDate(this.nowDate.fullDate) |
||||
this.weeks = this.cale.weeks |
||||
}, |
||||
endDate(val){ |
||||
this.cale.resetEndDate(val) |
||||
this.cale.setDate(this.nowDate.fullDate) |
||||
this.weeks = this.cale.weeks |
||||
}, |
||||
selected(newVal) { |
||||
this.cale.setSelectInfo(this.nowDate.fullDate, newVal) |
||||
this.weeks = this.cale.weeks |
||||
} |
||||
}, |
||||
created() { |
||||
// 获取日历方法实例 |
||||
this.cale = new Calendar({ |
||||
// date: new Date(), |
||||
selected: this.selected, |
||||
startDate: this.startDate, |
||||
endDate: this.endDate, |
||||
range: this.range, |
||||
}) |
||||
// 选中某一天 |
||||
// this.cale.setDate(this.date) |
||||
this.init(this.date) |
||||
// this.setDay |
||||
}, |
||||
methods: { |
||||
// 取消穿透 |
||||
clean() {}, |
||||
bindDateChange(e) { |
||||
const value = e.detail.value + '-1' |
||||
console.log(this.cale.getDate(value)); |
||||
this.setDate(value) |
||||
}, |
||||
/** |
||||
* 初始化日期显示 |
||||
* @param {Object} date |
||||
*/ |
||||
init(date) { |
||||
this.cale.setDate(date) |
||||
this.weeks = this.cale.weeks |
||||
this.nowDate = this.calendar = this.cale.getInfo(date) |
||||
}, |
||||
/** |
||||
* 打开日历弹窗 |
||||
*/ |
||||
open() { |
||||
// 弹窗模式并且清理数据 |
||||
if (this.clearDate && !this.insert) { |
||||
this.cale.cleanMultipleStatus() |
||||
// this.cale.setDate(this.date) |
||||
this.init(this.date) |
||||
} |
||||
this.show = true |
||||
this.$nextTick(() => { |
||||
setTimeout(() => { |
||||
this.aniMaskShow = true |
||||
}, 50) |
||||
}) |
||||
}, |
||||
/** |
||||
* 关闭日历弹窗 |
||||
*/ |
||||
close() { |
||||
this.aniMaskShow = false |
||||
this.$nextTick(() => { |
||||
setTimeout(() => { |
||||
this.show = false |
||||
this.$emit('close') |
||||
}, 300) |
||||
}) |
||||
}, |
||||
/** |
||||
* 确认按钮 |
||||
*/ |
||||
confirm() { |
||||
this.setEmit('confirm') |
||||
this.close() |
||||
}, |
||||
/** |
||||
* 变化触发 |
||||
*/ |
||||
change() { |
||||
if (!this.insert) return |
||||
this.setEmit('change') |
||||
}, |
||||
/** |
||||
* 选择月份触发 |
||||
*/ |
||||
monthSwitch() { |
||||
let { |
||||
year, |
||||
month |
||||
} = this.nowDate |
||||
this.$emit('monthSwitch', { |
||||
year, |
||||
month: Number(month) |
||||
}) |
||||
}, |
||||
/** |
||||
* 派发事件 |
||||
* @param {Object} name |
||||
*/ |
||||
setEmit(name) { |
||||
let { |
||||
year, |
||||
month, |
||||
date, |
||||
fullDate, |
||||
lunar, |
||||
extraInfo |
||||
} = this.calendar |
||||
this.$emit(name, { |
||||
range: this.cale.multipleStatus, |
||||
year, |
||||
month, |
||||
date, |
||||
fulldate: fullDate, |
||||
lunar, |
||||
extraInfo: extraInfo || {} |
||||
}) |
||||
}, |
||||
/** |
||||
* 选择天触发 |
||||
* @param {Object} weeks |
||||
*/ |
||||
choiceDate(weeks) { |
||||
if (weeks.disable) return |
||||
this.calendar = weeks |
||||
// 设置多选 |
||||
this.cale.setMultiple(this.calendar.fullDate) |
||||
this.weeks = this.cale.weeks |
||||
this.change() |
||||
}, |
||||
/** |
||||
* 回到今天 |
||||
*/ |
||||
backtoday() { |
||||
console.log(this.cale.getDate(new Date()).fullDate); |
||||
let date = this.cale.getDate(new Date()).fullDate |
||||
// this.cale.setDate(date) |
||||
this.init(date) |
||||
this.change() |
||||
}, |
||||
/** |
||||
* 上个月 |
||||
*/ |
||||
pre() { |
||||
const preDate = this.cale.getDate(this.nowDate.fullDate, -1, 'month').fullDate |
||||
this.setDate(preDate) |
||||
this.monthSwitch() |
||||
|
||||
}, |
||||
/** |
||||
* 下个月 |
||||
*/ |
||||
next() { |
||||
const nextDate = this.cale.getDate(this.nowDate.fullDate, +1, 'month').fullDate |
||||
this.setDate(nextDate) |
||||
this.monthSwitch() |
||||
}, |
||||
/** |
||||
* 设置日期 |
||||
* @param {Object} date |
||||
*/ |
||||
setDate(date) { |
||||
this.cale.setDate(date) |
||||
this.weeks = this.cale.weeks |
||||
this.nowDate = this.cale.getInfo(date) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss" scoped> |
||||
$uni-bg-color-mask: rgba($color: #000000, $alpha: 0.4); |
||||
$uni-border-color: #EDEDED; |
||||
$uni-text-color: #333; |
||||
$uni-bg-color-hover:#f1f1f1; |
||||
$uni-font-size-base:14px; |
||||
$uni-text-color-placeholder: #808080; |
||||
$uni-color-subtitle: #555555; |
||||
$uni-text-color-grey:#999; |
||||
.uni-calendar { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: column; |
||||
} |
||||
|
||||
.uni-calendar__mask { |
||||
position: fixed; |
||||
bottom: 0; |
||||
top: 0; |
||||
left: 0; |
||||
right: 0; |
||||
background-color: $uni-bg-color-mask; |
||||
transition-property: opacity; |
||||
transition-duration: 0.3s; |
||||
opacity: 0; |
||||
/* #ifndef APP-NVUE */ |
||||
z-index: 99; |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-calendar--mask-show { |
||||
opacity: 1 |
||||
} |
||||
|
||||
.uni-calendar--fixed { |
||||
position: fixed; |
||||
/* #ifdef APP-NVUE */ |
||||
bottom: 0; |
||||
/* #endif */ |
||||
left: 0; |
||||
right: 0; |
||||
transition-property: transform; |
||||
transition-duration: 0.3s; |
||||
transform: translateY(460px); |
||||
/* #ifndef APP-NVUE */ |
||||
bottom: calc(var(--window-bottom)); |
||||
z-index: 99; |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-calendar--ani-show { |
||||
transform: translateY(0); |
||||
} |
||||
|
||||
.uni-calendar__content { |
||||
background-color: #fff; |
||||
} |
||||
|
||||
.uni-calendar__header { |
||||
position: relative; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
justify-content: center; |
||||
align-items: center; |
||||
height: 50px; |
||||
border-bottom-color: $uni-border-color; |
||||
border-bottom-style: solid; |
||||
border-bottom-width: 1px; |
||||
} |
||||
|
||||
.uni-calendar--fixed-top { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
justify-content: space-between; |
||||
border-top-color: $uni-border-color; |
||||
border-top-style: solid; |
||||
border-top-width: 1px; |
||||
} |
||||
|
||||
.uni-calendar--fixed-width { |
||||
width: 50px; |
||||
// padding: 0 15px; |
||||
} |
||||
|
||||
.uni-calendar__backtoday { |
||||
position: absolute; |
||||
right: 0; |
||||
top: 25rpx; |
||||
padding: 0 5px; |
||||
padding-left: 10px; |
||||
height: 25px; |
||||
line-height: 25px; |
||||
font-size: 12px; |
||||
border-top-left-radius: 25px; |
||||
border-bottom-left-radius: 25px; |
||||
color: $uni-text-color; |
||||
background-color: $uni-bg-color-hover; |
||||
} |
||||
|
||||
.uni-calendar__header-text { |
||||
text-align: center; |
||||
width: 100px; |
||||
font-size: $uni-font-size-base; |
||||
color: $uni-text-color; |
||||
} |
||||
|
||||
.uni-calendar__header-btn-box { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
align-items: center; |
||||
justify-content: center; |
||||
width: 50px; |
||||
height: 50px; |
||||
} |
||||
|
||||
.uni-calendar__header-btn { |
||||
width: 10px; |
||||
height: 10px; |
||||
border-left-color: $uni-text-color-placeholder; |
||||
border-left-style: solid; |
||||
border-left-width: 2px; |
||||
border-top-color: $uni-color-subtitle; |
||||
border-top-style: solid; |
||||
border-top-width: 2px; |
||||
} |
||||
|
||||
.uni-calendar--left { |
||||
transform: rotate(-45deg); |
||||
} |
||||
|
||||
.uni-calendar--right { |
||||
transform: rotate(135deg); |
||||
} |
||||
|
||||
|
||||
.uni-calendar__weeks { |
||||
position: relative; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
} |
||||
|
||||
.uni-calendar__weeks-item { |
||||
flex: 1; |
||||
} |
||||
|
||||
.uni-calendar__weeks-day { |
||||
flex: 1; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: column; |
||||
justify-content: center; |
||||
align-items: center; |
||||
height: 45px; |
||||
border-bottom-color: #F5F5F5; |
||||
border-bottom-style: solid; |
||||
border-bottom-width: 1px; |
||||
} |
||||
|
||||
.uni-calendar__weeks-day-text { |
||||
font-size: 14px; |
||||
} |
||||
|
||||
.uni-calendar__box { |
||||
position: relative; |
||||
} |
||||
|
||||
.uni-calendar__box-bg { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
justify-content: center; |
||||
align-items: center; |
||||
position: absolute; |
||||
top: 0; |
||||
left: 0; |
||||
right: 0; |
||||
bottom: 0; |
||||
} |
||||
|
||||
.uni-calendar__box-bg-text { |
||||
font-size: 200px; |
||||
font-weight: bold; |
||||
color: $uni-text-color-grey; |
||||
opacity: 0.1; |
||||
text-align: center; |
||||
/* #ifndef APP-NVUE */ |
||||
line-height: 1; |
||||
/* #endif */ |
||||
} |
||||
</style> |
@ -0,0 +1,350 @@ |
||||
import CALENDAR from './calendar.js' |
||||
|
||||
class Calendar { |
||||
constructor({ |
||||
date, |
||||
selected, |
||||
startDate, |
||||
endDate, |
||||
range |
||||
} = {}) { |
||||
// 当前日期
|
||||
this.date = this.getDate(new Date()) // 当前初入日期
|
||||
// 打点信息
|
||||
this.selected = selected || []; |
||||
// 范围开始
|
||||
this.startDate = startDate |
||||
// 范围结束
|
||||
this.endDate = endDate |
||||
this.range = range |
||||
// 多选状态
|
||||
this.cleanMultipleStatus() |
||||
// 每周日期
|
||||
this.weeks = {} |
||||
// this._getWeek(this.date.fullDate)
|
||||
} |
||||
/** |
||||
* 设置日期 |
||||
* @param {Object} date |
||||
*/ |
||||
setDate(date) { |
||||
this.selectDate = this.getDate(date) |
||||
this._getWeek(this.selectDate.fullDate) |
||||
} |
||||
|
||||
/** |
||||
* 清理多选状态 |
||||
*/ |
||||
cleanMultipleStatus() { |
||||
this.multipleStatus = { |
||||
before: '', |
||||
after: '', |
||||
data: [] |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 重置开始日期 |
||||
*/ |
||||
resetSatrtDate(startDate) { |
||||
// 范围开始
|
||||
this.startDate = startDate |
||||
|
||||
} |
||||
|
||||
/** |
||||
* 重置结束日期 |
||||
*/ |
||||
resetEndDate(endDate) { |
||||
// 范围结束
|
||||
this.endDate = endDate |
||||
} |
||||
|
||||
/** |
||||
* 获取任意时间 |
||||
*/ |
||||
getDate(date, AddDayCount = 0, str = 'day') { |
||||
if (!date) { |
||||
date = new Date() |
||||
} |
||||
if (typeof date !== 'object') { |
||||
date = date.replace(/-/g, '/') |
||||
} |
||||
const dd = new Date(date) |
||||
switch (str) { |
||||
case 'day': |
||||
dd.setDate(dd.getDate() + AddDayCount) // 获取AddDayCount天后的日期
|
||||
break |
||||
case 'month': |
||||
if (dd.getDate() === 31) { |
||||
dd.setDate(dd.getDate() + AddDayCount) |
||||
} else { |
||||
dd.setMonth(dd.getMonth() + AddDayCount) // 获取AddDayCount天后的日期
|
||||
} |
||||
break |
||||
case 'year': |
||||
dd.setFullYear(dd.getFullYear() + AddDayCount) // 获取AddDayCount天后的日期
|
||||
break |
||||
} |
||||
const y = dd.getFullYear() |
||||
const m = dd.getMonth() + 1 < 10 ? '0' + (dd.getMonth() + 1) : dd.getMonth() + 1 // 获取当前月份的日期,不足10补0
|
||||
const d = dd.getDate() < 10 ? '0' + dd.getDate() : dd.getDate() // 获取当前几号,不足10补0
|
||||
return { |
||||
fullDate: y + '-' + m + '-' + d, |
||||
year: y, |
||||
month: m, |
||||
date: d, |
||||
day: dd.getDay() |
||||
} |
||||
} |
||||
|
||||
|
||||
/** |
||||
* 获取上月剩余天数 |
||||
*/ |
||||
_getLastMonthDays(firstDay, full) { |
||||
let dateArr = [] |
||||
for (let i = firstDay; i > 0; i--) { |
||||
const beforeDate = new Date(full.year, full.month - 1, -i + 1).getDate() |
||||
dateArr.push({ |
||||
date: beforeDate, |
||||
month: full.month - 1, |
||||
lunar: this.getlunar(full.year, full.month - 1, beforeDate), |
||||
disable: true |
||||
}) |
||||
} |
||||
return dateArr |
||||
} |
||||
/** |
||||
* 获取本月天数 |
||||
*/ |
||||
_currentMonthDys(dateData, full) { |
||||
let dateArr = [] |
||||
let fullDate = this.date.fullDate |
||||
for (let i = 1; i <= dateData; i++) { |
||||
let nowDate = full.year + '-' + (full.month < 10 ? |
||||
full.month : full.month) + '-' + (i < 10 ? |
||||
'0' + i : i) |
||||
// 是否今天
|
||||
let isDay = fullDate === nowDate |
||||
// 获取打点信息
|
||||
let info = this.selected && this.selected.find((item) => { |
||||
if (this.dateEqual(nowDate, item.date)) { |
||||
return item |
||||
} |
||||
}) |
||||
|
||||
// 日期禁用
|
||||
let disableBefore = true |
||||
let disableAfter = true |
||||
if (this.startDate) { |
||||
// let dateCompBefore = this.dateCompare(this.startDate, fullDate)
|
||||
// disableBefore = this.dateCompare(dateCompBefore ? this.startDate : fullDate, nowDate)
|
||||
disableBefore = this.dateCompare(this.startDate, nowDate) |
||||
} |
||||
|
||||
if (this.endDate) { |
||||
// let dateCompAfter = this.dateCompare(fullDate, this.endDate)
|
||||
// disableAfter = this.dateCompare(nowDate, dateCompAfter ? this.endDate : fullDate)
|
||||
disableAfter = this.dateCompare(nowDate, this.endDate) |
||||
} |
||||
let multiples = this.multipleStatus.data |
||||
let checked = false |
||||
let multiplesStatus = -1 |
||||
if (this.range) { |
||||
if (multiples) { |
||||
multiplesStatus = multiples.findIndex((item) => { |
||||
return this.dateEqual(item, nowDate) |
||||
}) |
||||
} |
||||
if (multiplesStatus !== -1) { |
||||
checked = true |
||||
} |
||||
} |
||||
let data = { |
||||
fullDate: nowDate, |
||||
year: full.year, |
||||
date: i, |
||||
multiple: this.range ? checked : false, |
||||
beforeMultiple: this.dateEqual(this.multipleStatus.before, nowDate), |
||||
afterMultiple: this.dateEqual(this.multipleStatus.after, nowDate), |
||||
month: full.month, |
||||
lunar: this.getlunar(full.year, full.month, i), |
||||
disable: !(disableBefore && disableAfter), |
||||
isDay |
||||
} |
||||
if (info) { |
||||
data.extraInfo = info |
||||
} |
||||
|
||||
dateArr.push(data) |
||||
} |
||||
return dateArr |
||||
} |
||||
/** |
||||
* 获取下月天数 |
||||
*/ |
||||
_getNextMonthDays(surplus, full) { |
||||
let dateArr = [] |
||||
for (let i = 1; i < surplus + 1; i++) { |
||||
dateArr.push({ |
||||
date: i, |
||||
month: Number(full.month) + 1, |
||||
lunar: this.getlunar(full.year, Number(full.month) + 1, i), |
||||
disable: true |
||||
}) |
||||
} |
||||
return dateArr |
||||
} |
||||
|
||||
/** |
||||
* 获取当前日期详情 |
||||
* @param {Object} date |
||||
*/ |
||||
getInfo(date) { |
||||
if (!date) { |
||||
date = new Date() |
||||
} |
||||
const dateInfo = this.canlender.find(item => item.fullDate === this.getDate(date).fullDate) |
||||
return dateInfo |
||||
} |
||||
|
||||
/** |
||||
* 比较时间大小 |
||||
*/ |
||||
dateCompare(startDate, endDate) { |
||||
// 计算截止时间
|
||||
startDate = new Date(startDate.replace('-', '/').replace('-', '/')) |
||||
// 计算详细项的截止时间
|
||||
endDate = new Date(endDate.replace('-', '/').replace('-', '/')) |
||||
if (startDate <= endDate) { |
||||
return true |
||||
} else { |
||||
return false |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 比较时间是否相等 |
||||
*/ |
||||
dateEqual(before, after) { |
||||
// 计算截止时间
|
||||
before = new Date(before.replace('-', '/').replace('-', '/')) |
||||
// 计算详细项的截止时间
|
||||
after = new Date(after.replace('-', '/').replace('-', '/')) |
||||
if (before.getTime() - after.getTime() === 0) { |
||||
return true |
||||
} else { |
||||
return false |
||||
} |
||||
} |
||||
|
||||
|
||||
/** |
||||
* 获取日期范围内所有日期 |
||||
* @param {Object} begin |
||||
* @param {Object} end |
||||
*/ |
||||
geDateAll(begin, end) { |
||||
var arr = [] |
||||
var ab = begin.split('-') |
||||
var ae = end.split('-') |
||||
var db = new Date() |
||||
db.setFullYear(ab[0], ab[1] - 1, ab[2]) |
||||
var de = new Date() |
||||
de.setFullYear(ae[0], ae[1] - 1, ae[2]) |
||||
var unixDb = db.getTime() - 24 * 60 * 60 * 1000 |
||||
var unixDe = de.getTime() - 24 * 60 * 60 * 1000 |
||||
for (var k = unixDb; k <= unixDe;) { |
||||
k = k + 24 * 60 * 60 * 1000 |
||||
arr.push(this.getDate(new Date(parseInt(k))).fullDate) |
||||
} |
||||
return arr |
||||
} |
||||
/** |
||||
* 计算阴历日期显示 |
||||
*/ |
||||
getlunar(year, month, date) { |
||||
return CALENDAR.solar2lunar(year, month, date) |
||||
} |
||||
/** |
||||
* 设置打点 |
||||
*/ |
||||
setSelectInfo(data, value) { |
||||
this.selected = value |
||||
this._getWeek(data) |
||||
} |
||||
|
||||
/** |
||||
* 获取多选状态 |
||||
*/ |
||||
setMultiple(fullDate) { |
||||
let { |
||||
before, |
||||
after |
||||
} = this.multipleStatus |
||||
|
||||
if (!this.range) return |
||||
if (before && after) { |
||||
this.multipleStatus.before = '' |
||||
this.multipleStatus.after = '' |
||||
this.multipleStatus.data = [] |
||||
} else { |
||||
if (!before) { |
||||
this.multipleStatus.before = fullDate |
||||
} else { |
||||
this.multipleStatus.after = fullDate |
||||
if (this.dateCompare(this.multipleStatus.before, this.multipleStatus.after)) { |
||||
this.multipleStatus.data = this.geDateAll(this.multipleStatus.before, this.multipleStatus.after); |
||||
} else { |
||||
this.multipleStatus.data = this.geDateAll(this.multipleStatus.after, this.multipleStatus.before); |
||||
} |
||||
} |
||||
} |
||||
this._getWeek(fullDate) |
||||
} |
||||
|
||||
/** |
||||
* 获取每周数据 |
||||
* @param {Object} dateData |
||||
*/ |
||||
_getWeek(dateData) { |
||||
const { |
||||
year, |
||||
month |
||||
} = this.getDate(dateData) |
||||
let firstDay = new Date(year, month - 1, 1).getDay() |
||||
let currentDay = new Date(year, month, 0).getDate() |
||||
let dates = { |
||||
lastMonthDays: this._getLastMonthDays(firstDay, this.getDate(dateData)), // 上个月末尾几天
|
||||
currentMonthDys: this._currentMonthDys(currentDay, this.getDate(dateData)), // 本月天数
|
||||
nextMonthDays: [], // 下个月开始几天
|
||||
weeks: [] |
||||
} |
||||
let canlender = [] |
||||
const surplus = 42 - (dates.lastMonthDays.length + dates.currentMonthDys.length) |
||||
dates.nextMonthDays = this._getNextMonthDays(surplus, this.getDate(dateData)) |
||||
canlender = canlender.concat(dates.lastMonthDays, dates.currentMonthDys, dates.nextMonthDays) |
||||
let weeks = {} |
||||
// 拼接数组 上个月开始几天 + 本月天数+ 下个月开始几天
|
||||
for (let i = 0; i < canlender.length; i++) { |
||||
if (i % 7 === 0) { |
||||
weeks[parseInt(i / 7)] = new Array(7) |
||||
} |
||||
weeks[parseInt(i / 7)][i % 7] = canlender[i] |
||||
} |
||||
this.canlender = canlender |
||||
this.weeks = weeks |
||||
} |
||||
|
||||
//静态方法
|
||||
// static init(date) {
|
||||
// if (!this.instance) {
|
||||
// this.instance = new Calendar(date);
|
||||
// }
|
||||
// return this.instance;
|
||||
// }
|
||||
} |
||||
|
||||
|
||||
export default Calendar |
@ -0,0 +1,85 @@ |
||||
{ |
||||
"id": "uni-calendar", |
||||
"displayName": "uni-calendar 日历", |
||||
"version": "1.4.7", |
||||
"description": "日历组件", |
||||
"keywords": [ |
||||
"uni-ui", |
||||
"uniui", |
||||
"日历", |
||||
"", |
||||
"打卡", |
||||
"日历选择" |
||||
], |
||||
"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": [], |
||||
"encrypt": [], |
||||
"platforms": { |
||||
"cloud": { |
||||
"tcb": "y", |
||||
"aliyun": "y" |
||||
}, |
||||
"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" |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,103 @@ |
||||
|
||||
|
||||
## Calendar 日历 |
||||
> **组件名:uni-calendar** |
||||
> 代码块: `uCalendar` |
||||
|
||||
|
||||
日历组件 |
||||
|
||||
> **注意事项** |
||||
> 为了避免错误使用,给大家带来不好的开发体验,请在使用组件前仔细阅读下面的注意事项,可以帮你避免一些错误。 |
||||
> - 本组件农历转换使用的js是 [@1900-2100区间内的公历、农历互转](https://github.com/jjonline/calendar.js) |
||||
> - 仅支持自定义组件模式 |
||||
> - `date`属性传入的应该是一个 String ,如: 2019-06-27 ,而不是 new Date() |
||||
> - 通过 `insert` 属性来确定当前的事件是 @change 还是 @confirm 。理应合并为一个事件,但是为了区分模式,现使用两个事件,这里需要注意 |
||||
> - 弹窗模式下无法阻止后面的元素滚动,如有需要阻止,请在弹窗弹出后,手动设置滚动元素为不可滚动 |
||||
|
||||
|
||||
### 安装方式 |
||||
|
||||
本组件符合[easycom](https://uniapp.dcloud.io/collocation/pages?id=easycom)规范,`HBuilderX 2.5.5`起,只需将本组件导入项目,在页面`template`中即可直接使用,无需在页面中`import`和注册`components`。 |
||||
|
||||
如需通过`npm`方式使用`uni-ui`组件,另见文档:[https://ext.dcloud.net.cn/plugin?id=55](https://ext.dcloud.net.cn/plugin?id=55) |
||||
|
||||
### 基本用法 |
||||
|
||||
在 ``template`` 中使用组件 |
||||
|
||||
```html |
||||
<view> |
||||
<uni-calendar |
||||
:insert="true" |
||||
:lunar="true" |
||||
:start-date="'2019-3-2'" |
||||
:end-date="'2019-5-20'" |
||||
@change="change" |
||||
/> |
||||
</view> |
||||
``` |
||||
|
||||
### 通过方法打开日历 |
||||
|
||||
需要设置 `insert` 为 `false` |
||||
|
||||
```html |
||||
<view> |
||||
<uni-calendar |
||||
ref="calendar" |
||||
:insert="false" |
||||
@confirm="confirm" |
||||
/> |
||||
<button @click="open">打开日历</button> |
||||
</view> |
||||
``` |
||||
|
||||
```javascript |
||||
|
||||
export default { |
||||
data() { |
||||
return {}; |
||||
}, |
||||
methods: { |
||||
open(){ |
||||
this.$refs.calendar.open(); |
||||
}, |
||||
confirm(e) { |
||||
console.log(e); |
||||
} |
||||
} |
||||
}; |
||||
|
||||
``` |
||||
|
||||
|
||||
## API |
||||
|
||||
### Calendar Props |
||||
|
||||
| 属性名 | 类型 | 默认值| 说明 | |
||||
| | | |
||||
| date | String |- | 自定义当前时间,默认为今天 | |
||||
| lunar | Boolean | false | 显示农历 | |
||||
| startDate | String |- | 日期选择范围-开始日期 | |
||||
| endDate | String |- | 日期选择范围-结束日期 | |
||||
| range | Boolean | false | 范围选择 | |
||||
| insert | Boolean | false | 插入模式,可选值,ture:插入模式;false:弹窗模式;默认为插入模式 | |
||||
|clearDate |Boolean |true |弹窗模式是否清空上次选择内容 | |
||||
| selected | Array |- | 打点,期待格式[{date: '2019-06-27', info: '签到', data: { custom: '自定义信息', name: '自定义消息头',xxx:xxx... }}] | |
||||
|showMonth | Boolean | true | 是否显示月份为背景 | |
||||
|
||||
### Calendar Events |
||||
|
||||
| 事件名 | 说明 |返回值| |
||||
| | | | |
||||
| open | 弹出日历组件,`insert :false` 时生效|- | |
||||
|
||||
|
||||
|
||||
|
||||
|
||||
## 组件示例 |
||||
|
||||
点击查看:[https://hellouniapp.dcloud.net.cn/pages/extUI/calendar/calendar](https://hellouniapp.dcloud.net.cn/pages/extUI/calendar/calendar) |
@ -0,0 +1,26 @@ |
||||
## 1.3.1(2021-12-20) |
||||
- 修复 在vue页面下略缩图显示不正常的bug |
||||
## 1.3.0(2021-11-19) |
||||
- 重构插槽的用法 ,header 替换为 title |
||||
- 新增 actions 插槽 |
||||
- 新增 cover 封面图属性和插槽 |
||||
- 新增 padding 内容默认内边距离 |
||||
- 新增 margin 卡片默认外边距离 |
||||
- 新增 spacing 卡片默认内边距 |
||||
- 新增 shadow 卡片阴影属性 |
||||
- 取消 mode 属性,可使用组合插槽代替 |
||||
- 取消 note 属性 ,使用actions插槽代替 |
||||
- 优化 组件UI,并提供设计资源,详见:[https://uniapp.dcloud.io/component/uniui/resource](https://uniapp.dcloud.io/component/uniui/resource) |
||||
- 文档迁移,详见:[https://uniapp.dcloud.io/component/uniui/uni-card](https://uniapp.dcloud.io/component/uniui/uni-card) |
||||
## 1.2.1(2021-07-30) |
||||
- 优化 vue3下事件警告的问题 |
||||
## 1.2.0(2021-07-13) |
||||
- 组件兼容 vue3,如何创建vue3项目详见 [uni-app 项目支持 vue3 介绍](https://ask.dcloud.net.cn/article/37834) |
||||
## 1.1.8(2021-07-01) |
||||
- 优化 图文卡片无图片加载时,提供占位图标 |
||||
- 新增 header 插槽,自定义卡片头部( 图文卡片 mode="style" 时,不支持) |
||||
- 修复 thumbnail 不存在仍然占位的 bug |
||||
## 1.1.7(2021-05-12) |
||||
- 新增 组件示例地址 |
||||
## 1.1.6(2021-02-04) |
||||
- 调整为uni_modules目录规范 |
@ -0,0 +1,270 @@ |
||||
<template> |
||||
<view class="uni-card" :class="{ 'uni-card--full': isFull, 'uni-card--shadow': isShadow,'uni-card--border':border}" |
||||
:style="{'margin':isFull?0:margin,'padding':spacing,'box-shadow':isShadow?shadow:''}"> |
||||
<!-- 封面 --> |
||||
<slot name="cover"> |
||||
<view v-if="cover" class="uni-card__cover"> |
||||
<image class="uni-card__cover-image" mode="widthFix" @click="onClick('cover')" :src="cover"></image> |
||||
</view> |
||||
</slot> |
||||
<slot name="title"> |
||||
<view v-if="title || extra" class="uni-card__header"> |
||||
<!-- 卡片标题 --> |
||||
<view class="uni-card__header-box" @click="onClick('title')"> |
||||
<view v-if="thumbnail" class="uni-card__header-avatar"> |
||||
<image class="uni-card__header-avatar-image" :src="thumbnail" mode="aspectFit" /> |
||||
</view> |
||||
<view class="uni-card__header-content"> |
||||
<text class="uni-card__header-content-title uni-ellipsis">{{ title }}</text> |
||||
<text v-if="title&&subTitle" |
||||
class="uni-card__header-content-subtitle uni-ellipsis">{{ subTitle }}</text> |
||||
</view> |
||||
</view> |
||||
<view class="uni-card__header-extra" @click="onClick('extra')"> |
||||
<text class="uni-card__header-extra-text">{{ extra }}</text> |
||||
</view> |
||||
</view> |
||||
</slot> |
||||
<!-- 卡片内容 --> |
||||
<view class="uni-card__content" :style="{padding:padding}" @click="onClick('content')"> |
||||
<slot></slot> |
||||
</view> |
||||
<view class="uni-card__actions" @click="onClick('actions')"> |
||||
<slot name="actions"></slot> |
||||
</view> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
/** |
||||
* Card 卡片 |
||||
* @description 卡片视图组件 |
||||
* @tutorial https://ext.dcloud.net.cn/plugin?id=22 |
||||
* @property {String} title 标题文字 |
||||
* @property {String} subTitle 副标题 |
||||
* @property {Number} padding 内容内边距 |
||||
* @property {Number} margin 卡片外边距 |
||||
* @property {Number} spacing 卡片内边距 |
||||
* @property {String} extra 标题额外信息 |
||||
* @property {String} cover 封面图(本地路径需要引入) |
||||
* @property {String} thumbnail 标题左侧缩略图 |
||||
* @property {Boolean} is-full = [true | false] 卡片内容是否通栏,为 true 时将去除padding值 |
||||
* @property {Boolean} is-shadow = [true | false] 卡片内容是否开启阴影 |
||||
* @property {String} shadow 卡片阴影 |
||||
* @property {Boolean} border 卡片边框 |
||||
* @event {Function} click 点击 Card 触发事件 |
||||
*/ |
||||
export default { |
||||
name: 'UniCard', |
||||
emits: ['click'], |
||||
props: { |
||||
title: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
subTitle: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
padding: { |
||||
type: String, |
||||
default: '10px' |
||||
}, |
||||
margin: { |
||||
type: String, |
||||
default: '15px' |
||||
}, |
||||
spacing: { |
||||
type: String, |
||||
default: '0 10px' |
||||
}, |
||||
extra: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
cover: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
thumbnail: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
isFull: { |
||||
// 内容区域是否通栏 |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
isShadow: { |
||||
// 是否开启阴影 |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
shadow: { |
||||
type: String, |
||||
default: '0px 0px 3px 1px rgba(0, 0, 0, 0.08)' |
||||
}, |
||||
border: { |
||||
type: Boolean, |
||||
default: true |
||||
} |
||||
}, |
||||
methods: { |
||||
onClick(type) { |
||||
this.$emit('click', type) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss"> |
||||
$uni-border-3: #EBEEF5 !default; |
||||
$uni-shadow-base:0 0px 6px 1px rgba($color: #a5a5a5, $alpha: 0.2) !default; |
||||
$uni-main-color: #3a3a3a !default; |
||||
$uni-base-color: #6a6a6a !default; |
||||
$uni-secondary-color: #909399 !default; |
||||
$uni-spacing-sm: 8px !default; |
||||
$uni-border-color:$uni-border-3; |
||||
$uni-shadow: $uni-shadow-base; |
||||
$uni-card-title: 15px; |
||||
$uni-cart-title-color:$uni-main-color; |
||||
$uni-card-subtitle: 12px; |
||||
$uni-cart-subtitle-color:$uni-secondary-color; |
||||
$uni-card-spacing: 10px; |
||||
$uni-card-content-color: $uni-base-color; |
||||
|
||||
.uni-card { |
||||
margin: $uni-card-spacing; |
||||
padding: 0 $uni-spacing-sm; |
||||
border-radius: 4px; |
||||
overflow: hidden; |
||||
font-family: Helvetica Neue, Helvetica, PingFang SC, Hiragino Sans GB, Microsoft YaHei, SimSun, sans-serif; |
||||
background-color: #fff; |
||||
flex: 1; |
||||
|
||||
.uni-card__cover { |
||||
position: relative; |
||||
margin-top: $uni-card-spacing; |
||||
flex-direction: row; |
||||
overflow: hidden; |
||||
border-radius: 4px; |
||||
.uni-card__cover-image { |
||||
flex: 1; |
||||
// width: 100%; |
||||
/* #ifndef APP-PLUS */ |
||||
vertical-align: middle; |
||||
/* #endif */ |
||||
} |
||||
} |
||||
|
||||
.uni-card__header { |
||||
display: flex; |
||||
border-bottom: 1px $uni-border-color solid; |
||||
flex-direction: row; |
||||
align-items: center; |
||||
padding: $uni-card-spacing; |
||||
overflow: hidden; |
||||
|
||||
.uni-card__header-box { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex: 1; |
||||
flex-direction: row; |
||||
align-items: center; |
||||
overflow: hidden; |
||||
} |
||||
|
||||
.uni-card__header-avatar { |
||||
width: 40px; |
||||
height: 40px; |
||||
overflow: hidden; |
||||
border-radius: 5px; |
||||
margin-right: $uni-card-spacing; |
||||
.uni-card__header-avatar-image { |
||||
flex: 1; |
||||
width: 40px; |
||||
height: 40px; |
||||
} |
||||
} |
||||
|
||||
.uni-card__header-content { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: column; |
||||
justify-content: center; |
||||
flex: 1; |
||||
// height: 40px; |
||||
overflow: hidden; |
||||
|
||||
.uni-card__header-content-title { |
||||
font-size: $uni-card-title; |
||||
color: $uni-cart-title-color; |
||||
// line-height: 22px; |
||||
} |
||||
|
||||
.uni-card__header-content-subtitle { |
||||
font-size: $uni-card-subtitle; |
||||
margin-top: 5px; |
||||
color: $uni-cart-subtitle-color; |
||||
} |
||||
} |
||||
|
||||
.uni-card__header-extra { |
||||
line-height: 12px; |
||||
|
||||
.uni-card__header-extra-text { |
||||
font-size: 12px; |
||||
color: $uni-cart-subtitle-color; |
||||
} |
||||
} |
||||
} |
||||
|
||||
.uni-card__content { |
||||
padding: $uni-card-spacing; |
||||
font-size: 14px; |
||||
color: $uni-card-content-color; |
||||
line-height: 22px; |
||||
} |
||||
|
||||
.uni-card__actions { |
||||
font-size: 12px; |
||||
} |
||||
} |
||||
|
||||
.uni-card--border { |
||||
border: 1px solid $uni-border-color; |
||||
} |
||||
|
||||
.uni-card--shadow { |
||||
position: relative; |
||||
/* #ifndef APP-NVUE */ |
||||
box-shadow: $uni-shadow; |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-card--full { |
||||
margin: 0; |
||||
border-left-width: 0; |
||||
border-left-width: 0; |
||||
border-radius: 0; |
||||
} |
||||
|
||||
/* #ifndef APP-NVUE */ |
||||
.uni-card--full:after { |
||||
border-radius: 0; |
||||
} |
||||
|
||||
/* #endif */ |
||||
.uni-ellipsis { |
||||
/* #ifndef APP-NVUE */ |
||||
overflow: hidden; |
||||
white-space: nowrap; |
||||
text-overflow: ellipsis; |
||||
/* #endif */ |
||||
/* #ifdef APP-NVUE */ |
||||
lines: 1; |
||||
/* #endif */ |
||||
} |
||||
</style> |
@ -0,0 +1,90 @@ |
||||
{ |
||||
"id": "uni-card", |
||||
"displayName": "uni-card 卡片", |
||||
"version": "1.3.1", |
||||
"description": "Card 组件,提供常见的卡片样式。", |
||||
"keywords": [ |
||||
"uni-ui", |
||||
"uniui", |
||||
"card", |
||||
"", |
||||
"卡片" |
||||
], |
||||
"repository": "https://github.com/dcloudio/uni-ui", |
||||
"engines": { |
||||
"HBuilderX": "" |
||||
}, |
||||
"directories": { |
||||
"example": "../../temps/example_temps" |
||||
}, |
||||
"dcloudext": { |
||||
"category": [ |
||||
"前端组件", |
||||
"通用组件" |
||||
], |
||||
"sale": { |
||||
"regular": { |
||||
"price": "0.00" |
||||
}, |
||||
"sourcecode": { |
||||
"price": "0.00" |
||||
} |
||||
}, |
||||
"contact": { |
||||
"qq": "" |
||||
}, |
||||
"declaration": { |
||||
"ads": "无", |
||||
"data": "无", |
||||
"permissions": "无" |
||||
}, |
||||
"npmurl": "https://www.npmjs.com/package/@dcloudio/uni-ui" |
||||
}, |
||||
"uni_modules": { |
||||
"dependencies": [ |
||||
"uni-icons", |
||||
"uni-scss" |
||||
], |
||||
"encrypt": [], |
||||
"platforms": { |
||||
"cloud": { |
||||
"tcb": "y", |
||||
"aliyun": "y" |
||||
}, |
||||
"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" |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,12 @@ |
||||
|
||||
|
||||
## Card 卡片 |
||||
> **组件名:uni-card** |
||||
> 代码块: `uCard` |
||||
|
||||
卡片视图组件。 |
||||
|
||||
### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-card) |
||||
#### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839 |
||||
|
||||
|
@ -0,0 +1,36 @@ |
||||
## 1.4.3(2022-01-25) |
||||
- 修复 初始化的时候 ,open 属性失效的bug |
||||
## 1.4.2(2022-01-21) |
||||
- 修复 微信小程序resize后组件收起的bug |
||||
## 1.4.1(2021-11-22) |
||||
- 修复 vue3中个别scss变量无法找到的问题 |
||||
## 1.4.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-collapse](https://uniapp.dcloud.io/component/uniui/uni-collapse) |
||||
## 1.3.3(2021-08-17) |
||||
- 优化 show-arrow 属性默认为true |
||||
## 1.3.2(2021-08-17) |
||||
- 新增 show-arrow 属性,控制是否显示右侧箭头 |
||||
## 1.3.1(2021-07-30) |
||||
- 优化 vue3下小程序事件警告的问题 |
||||
## 1.3.0(2021-07-30) |
||||
- 组件兼容 vue3,如何创建vue3项目,详见 [uni-app 项目支持 vue3 介绍](https://ask.dcloud.net.cn/article/37834) |
||||
## 1.2.2(2021-07-21) |
||||
- 修复 由1.2.0版本引起的 change 事件返回 undefined 的Bug |
||||
## 1.2.1(2021-07-21) |
||||
- 优化 组件示例 |
||||
## 1.2.0(2021-07-21) |
||||
- 新增 组件折叠动画 |
||||
- 新增 value\v-model 属性 ,动态修改面板折叠状态 |
||||
- 新增 title 插槽 ,可定义面板标题 |
||||
- 新增 border 属性 ,显示隐藏面板内容分隔线 |
||||
- 新增 title-border 属性 ,显示隐藏面板标题分隔线 |
||||
- 修复 resize 方法失效的Bug |
||||
- 修复 change 事件返回参数不正确的Bug |
||||
- 优化 H5、App 平台自动更具内容更新高度,无需调用 reszie() 方法 |
||||
## 1.1.7(2021-05-12) |
||||
- 新增 组件示例地址 |
||||
## 1.1.6(2021-02-05) |
||||
- 优化 组件引用关系,通过uni_modules引用组件 |
||||
## 1.1.5(2021-02-05) |
||||
- 调整为uni_modules目录规范 |
@ -0,0 +1,402 @@ |
||||
<template> |
||||
<view class="uni-collapse-item"> |
||||
<!-- onClick(!isOpen) --> |
||||
<view @click="onClick(!isOpen)" class="uni-collapse-item__title" |
||||
:class="{'is-open':isOpen &&titleBorder === 'auto' ,'uni-collapse-item-border':titleBorder !== 'none'}"> |
||||
<view class="uni-collapse-item__title-wrap"> |
||||
<slot name="title"> |
||||
<view class="uni-collapse-item__title-box" :class="{'is-disabled':disabled}"> |
||||
<image v-if="thumb" :src="thumb" class="uni-collapse-item__title-img" /> |
||||
<text class="uni-collapse-item__title-text">{{ title }}</text> |
||||
</view> |
||||
</slot> |
||||
</view> |
||||
<view v-if="showArrow" |
||||
:class="{ 'uni-collapse-item__title-arrow-active': isOpen, 'uni-collapse-item--animation': showAnimation === true }" |
||||
class="uni-collapse-item__title-arrow"> |
||||
<uni-icons :color="disabled?'#ddd':'#bbb'" size="14" type="bottom" /> |
||||
</view> |
||||
</view> |
||||
<view class="uni-collapse-item__wrap" :class="{'is--transition':showAnimation}" |
||||
:style="{height: (isOpen?height:0) +'px'}"> |
||||
<view :id="elId" ref="collapse--hook" class="uni-collapse-item__wrap-content" |
||||
:class="{open:isheight,'uni-collapse-item--border':border&&isOpen}"> |
||||
<slot></slot> |
||||
</view> |
||||
</view> |
||||
|
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
// #ifdef APP-NVUE |
||||
const dom = weex.requireModule('dom') |
||||
// #endif |
||||
/** |
||||
* CollapseItem 折叠面板子组件 |
||||
* @description 折叠面板子组件 |
||||
* @property {String} title 标题文字 |
||||
* @property {String} thumb 标题左侧缩略图 |
||||
* @property {String} name 唯一标志符 |
||||
* @property {Boolean} open = [true|false] 是否展开组件 |
||||
* @property {Boolean} titleBorder = [true|false] 是否显示标题分隔线 |
||||
* @property {Boolean} border = [true|false] 是否显示分隔线 |
||||
* @property {Boolean} disabled = [true|false] 是否展开面板 |
||||
* @property {Boolean} showAnimation = [true|false] 开启动画 |
||||
* @property {Boolean} showArrow = [true|false] 是否显示右侧箭头 |
||||
*/ |
||||
export default { |
||||
name: 'uniCollapseItem', |
||||
props: { |
||||
// 列表标题 |
||||
title: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
name: { |
||||
type: [Number, String], |
||||
default: '' |
||||
}, |
||||
// 是否禁用 |
||||
disabled: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
// #ifdef APP-PLUS |
||||
// 是否显示动画,app 端默认不开启动画,卡顿严重 |
||||
showAnimation: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
// #endif |
||||
// #ifndef APP-PLUS |
||||
// 是否显示动画 |
||||
showAnimation: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
// #endif |
||||
// 是否展开 |
||||
open: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
// 缩略图 |
||||
thumb: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
// 标题分隔线显示类型 |
||||
titleBorder: { |
||||
type: String, |
||||
default: 'auto' |
||||
}, |
||||
border: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
showArrow: { |
||||
type: Boolean, |
||||
default: true |
||||
} |
||||
}, |
||||
data() { |
||||
// TODO 随机生生元素ID,解决百度小程序获取同一个元素位置信息的bug |
||||
const elId = `Uni_${Math.ceil(Math.random() * 10e5).toString(36)}` |
||||
return { |
||||
isOpen: false, |
||||
isheight: null, |
||||
height: 0, |
||||
elId, |
||||
nameSync: 0 |
||||
} |
||||
}, |
||||
watch: { |
||||
open(val) { |
||||
this.isOpen = val |
||||
this.onClick(val, 'init') |
||||
} |
||||
}, |
||||
updated(e) { |
||||
this.$nextTick(() => { |
||||
this.init(true) |
||||
}) |
||||
}, |
||||
created() { |
||||
this.collapse = this.getCollapse() |
||||
this.oldHeight = 0 |
||||
this.onClick(this.open, 'init') |
||||
}, |
||||
// #ifndef VUE3 |
||||
// TODO vue2 |
||||
destroyed() { |
||||
if (this.__isUnmounted) return |
||||
this.uninstall() |
||||
}, |
||||
// #endif |
||||
// #ifdef VUE3 |
||||
// TODO vue3 |
||||
unmounted() { |
||||
this.__isUnmounted = true |
||||
this.uninstall() |
||||
}, |
||||
// #endif |
||||
mounted() { |
||||
if (!this.collapse) return |
||||
if (this.name !== '') { |
||||
this.nameSync = this.name |
||||
} else { |
||||
this.nameSync = this.collapse.childrens.length + '' |
||||
} |
||||
if (this.collapse.names.indexOf(this.nameSync) === -1) { |
||||
this.collapse.names.push(this.nameSync) |
||||
} else { |
||||
console.warn(`name 值 ${this.nameSync} 重复`); |
||||
} |
||||
if (this.collapse.childrens.indexOf(this) === -1) { |
||||
this.collapse.childrens.push(this) |
||||
} |
||||
this.init() |
||||
}, |
||||
methods: { |
||||
init(type) { |
||||
// #ifndef APP-NVUE |
||||
this.getCollapseHeight(type) |
||||
// #endif |
||||
// #ifdef APP-NVUE |
||||
this.getNvueHwight(type) |
||||
// #endif |
||||
}, |
||||
uninstall() { |
||||
if (this.collapse) { |
||||
this.collapse.childrens.forEach((item, index) => { |
||||
if (item === this) { |
||||
this.collapse.childrens.splice(index, 1) |
||||
} |
||||
}) |
||||
this.collapse.names.forEach((item, index) => { |
||||
if (item === this.nameSync) { |
||||
this.collapse.names.splice(index, 1) |
||||
} |
||||
}) |
||||
} |
||||
}, |
||||
onClick(isOpen, type) { |
||||
if (this.disabled) return |
||||
this.isOpen = isOpen |
||||
if (this.isOpen && this.collapse) { |
||||
this.collapse.setAccordion(this) |
||||
} |
||||
if (type !== 'init') { |
||||
this.collapse.onChange(isOpen, this) |
||||
} |
||||
}, |
||||
getCollapseHeight(type, index = 0) { |
||||
const views = uni.createSelectorQuery().in(this) |
||||
views |
||||
.select(`#${this.elId}`) |
||||
.fields({ |
||||
size: true |
||||
}, data => { |
||||
// TODO 百度中可能获取不到节点信息 ,需要循环获取 |
||||
if (index >= 10) return |
||||
if (!data) { |
||||
index++ |
||||
this.getCollapseHeight(false, index) |
||||
return |
||||
} |
||||
// #ifdef APP-NVUE |
||||
this.height = data.height + 1 |
||||
// #endif |
||||
// #ifndef APP-NVUE |
||||
this.height = data.height |
||||
// #endif |
||||
this.isheight = true |
||||
if (type) return |
||||
this.onClick(this.isOpen, 'init') |
||||
}) |
||||
.exec() |
||||
}, |
||||
getNvueHwight(type) { |
||||
const result = dom.getComponentRect(this.$refs['collapse--hook'], option => { |
||||
if (option && option.result && option.size) { |
||||
// #ifdef APP-NVUE |
||||
this.height = option.size.height + 1 |
||||
// #endif |
||||
// #ifndef APP-NVUE |
||||
this.height = option.size.height |
||||
// #endif |
||||
this.isheight = true |
||||
if (type) return |
||||
this.onClick(this.open, 'init') |
||||
} |
||||
}) |
||||
}, |
||||
/** |
||||
* 获取父元素实例 |
||||
*/ |
||||
getCollapse(name = 'uniCollapse') { |
||||
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; |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss"> |
||||
.uni-collapse-item { |
||||
/* #ifndef APP-NVUE */ |
||||
box-sizing: border-box; |
||||
|
||||
/* #endif */ |
||||
&__title { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
width: 100%; |
||||
box-sizing: border-box; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
align-items: center; |
||||
transition: border-bottom-color .3s; |
||||
|
||||
// transition-property: border-bottom-color; |
||||
// transition-duration: 5s; |
||||
&-wrap { |
||||
width: 100%; |
||||
flex: 1; |
||||
|
||||
} |
||||
|
||||
&-box { |
||||
padding: 0 15px; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
width: 100%; |
||||
box-sizing: border-box; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
justify-content: space-between; |
||||
align-items: center; |
||||
height: 48px; |
||||
line-height: 48px; |
||||
background-color: #fff; |
||||
color: #303133; |
||||
font-size: 13px; |
||||
font-weight: 500; |
||||
/* #ifdef H5 */ |
||||
cursor: pointer; |
||||
outline: none; |
||||
|
||||
/* #endif */ |
||||
&.is-disabled { |
||||
.uni-collapse-item__title-text { |
||||
color: #999; |
||||
} |
||||
} |
||||
|
||||
} |
||||
|
||||
&.uni-collapse-item-border { |
||||
border-bottom: 1px solid #ebeef5; |
||||
} |
||||
|
||||
&.is-open { |
||||
border-bottom-color: transparent; |
||||
} |
||||
|
||||
&-img { |
||||
height: 22px; |
||||
width: 22px; |
||||
margin-right: 10px; |
||||
} |
||||
|
||||
&-text { |
||||
flex: 1; |
||||
font-size: 14px; |
||||
/* #ifndef APP-NVUE */ |
||||
white-space: nowrap; |
||||
color: inherit; |
||||
/* #endif */ |
||||
/* #ifdef APP-NVUE */ |
||||
lines: 1; |
||||
/* #endif */ |
||||
overflow: hidden; |
||||
text-overflow: ellipsis; |
||||
} |
||||
|
||||
&-arrow { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
box-sizing: border-box; |
||||
/* #endif */ |
||||
align-items: center; |
||||
justify-content: center; |
||||
width: 20px; |
||||
height: 20px; |
||||
margin-right: 10px; |
||||
transform: rotate(0deg); |
||||
|
||||
&-active { |
||||
transform: rotate(-180deg); |
||||
} |
||||
} |
||||
|
||||
|
||||
} |
||||
|
||||
&__wrap { |
||||
/* #ifndef APP-NVUE */ |
||||
will-change: height; |
||||
box-sizing: border-box; |
||||
/* #endif */ |
||||
background-color: #fff; |
||||
overflow: hidden; |
||||
position: relative; |
||||
height: 0; |
||||
|
||||
&.is--transition { |
||||
// transition: all 0.3s; |
||||
transition-property: height, border-bottom-width; |
||||
transition-duration: 0.3s; |
||||
/* #ifndef APP-NVUE */ |
||||
will-change: height; |
||||
/* #endif */ |
||||
} |
||||
|
||||
|
||||
|
||||
&-content { |
||||
position: absolute; |
||||
font-size: 13px; |
||||
color: #303133; |
||||
// transition: height 0.3s; |
||||
border-bottom-color: transparent; |
||||
border-bottom-style: solid; |
||||
border-bottom-width: 0; |
||||
|
||||
&.uni-collapse-item--border { |
||||
border-bottom-width: 1px; |
||||
border-bottom-color: red; |
||||
border-bottom-color: #ebeef5; |
||||
} |
||||
|
||||
&.open { |
||||
position: relative; |
||||
} |
||||
} |
||||
} |
||||
|
||||
&--animation { |
||||
transition-property: transform; |
||||
transition-duration: 0.3s; |
||||
transition-timing-function: ease; |
||||
} |
||||
|
||||
} |
||||
</style> |
@ -0,0 +1,147 @@ |
||||
<template> |
||||
<view class="uni-collapse"> |
||||
<slot /> |
||||
</view> |
||||
</template> |
||||
<script> |
||||
/** |
||||
* Collapse 折叠面板 |
||||
* @description 展示可以折叠 / 展开的内容区域 |
||||
* @tutorial https://ext.dcloud.net.cn/plugin?id=23 |
||||
* @property {String|Array} value 当前激活面板改变时触发(如果是手风琴模式,参数类型为string,否则为array) |
||||
* @property {Boolean} accordion = [true|false] 是否开启手风琴效果是否开启手风琴效果 |
||||
* @event {Function} change 切换面板时触发,如果是手风琴模式,返回类型为string,否则为array |
||||
*/ |
||||
export default { |
||||
name: 'uniCollapse', |
||||
emits:['change','activeItem','input','update:modelValue'], |
||||
props: { |
||||
value: { |
||||
type: [String, Array], |
||||
default: '' |
||||
}, |
||||
modelValue: { |
||||
type: [String, Array], |
||||
default: '' |
||||
}, |
||||
accordion: { |
||||
// 是否开启手风琴效果 |
||||
type: [Boolean, String], |
||||
default: false |
||||
}, |
||||
}, |
||||
data() { |
||||
return {} |
||||
}, |
||||
computed: { |
||||
// TODO 兼容 vue2 和 vue3 |
||||
dataValue() { |
||||
let value = (typeof this.value === 'string' && this.value === '') || |
||||
(Array.isArray(this.value) && this.value.length === 0) |
||||
let modelValue = (typeof this.modelValue === 'string' && this.modelValue === '') || |
||||
(Array.isArray(this.modelValue) && this.modelValue.length === 0) |
||||
if (value) { |
||||
return this.modelValue |
||||
} |
||||
if (modelValue) { |
||||
return this.value |
||||
} |
||||
|
||||
return this.value |
||||
} |
||||
}, |
||||
watch: { |
||||
dataValue(val) { |
||||
this.setOpen(val) |
||||
} |
||||
}, |
||||
created() { |
||||
this.childrens = [] |
||||
this.names = [] |
||||
}, |
||||
mounted() { |
||||
this.$nextTick(()=>{ |
||||
this.setOpen(this.dataValue) |
||||
}) |
||||
}, |
||||
methods: { |
||||
setOpen(val) { |
||||
let str = typeof val === 'string' |
||||
let arr = Array.isArray(val) |
||||
this.childrens.forEach((vm, index) => { |
||||
if (str) { |
||||
if (val === vm.nameSync) { |
||||
if (!this.accordion) { |
||||
console.warn('accordion 属性为 false ,v-model 类型应该为 array') |
||||
return |
||||
} |
||||
vm.isOpen = true |
||||
} |
||||
} |
||||
if (arr) { |
||||
val.forEach(v => { |
||||
if (v === vm.nameSync) { |
||||
if (this.accordion) { |
||||
console.warn('accordion 属性为 true ,v-model 类型应该为 string') |
||||
return |
||||
} |
||||
vm.isOpen = true |
||||
} |
||||
}) |
||||
} |
||||
}) |
||||
this.emit(val) |
||||
}, |
||||
setAccordion(self) { |
||||
if (!this.accordion) return |
||||
this.childrens.forEach((vm, index) => { |
||||
if (self !== vm) { |
||||
vm.isOpen = false |
||||
} |
||||
}) |
||||
}, |
||||
resize() { |
||||
this.childrens.forEach((vm, index) => { |
||||
// #ifndef APP-NVUE |
||||
vm.getCollapseHeight() |
||||
// #endif |
||||
// #ifdef APP-NVUE |
||||
vm.getNvueHwight() |
||||
// #endif |
||||
}) |
||||
}, |
||||
onChange(isOpen, self) { |
||||
let activeItem = [] |
||||
|
||||
if (this.accordion) { |
||||
activeItem = isOpen ? self.nameSync : '' |
||||
} else { |
||||
this.childrens.forEach((vm, index) => { |
||||
if (vm.isOpen) { |
||||
activeItem.push(vm.nameSync) |
||||
} |
||||
}) |
||||
} |
||||
this.$emit('change', activeItem) |
||||
this.emit(activeItem) |
||||
}, |
||||
emit(val){ |
||||
this.$emit('input', val) |
||||
this.$emit('update:modelValue', val) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
<style lang="scss" > |
||||
.uni-collapse { |
||||
/* #ifndef APP-NVUE */ |
||||
width: 100%; |
||||
display: flex; |
||||
/* #endif */ |
||||
/* #ifdef APP-NVUE */ |
||||
flex: 1; |
||||
/* #endif */ |
||||
flex-direction: column; |
||||
background-color: #fff; |
||||
} |
||||
</style> |
@ -0,0 +1,89 @@ |
||||
{ |
||||
"id": "uni-collapse", |
||||
"displayName": "uni-collapse 折叠面板", |
||||
"version": "1.4.3", |
||||
"description": "Collapse 组件,可以折叠 / 展开的内容区域。", |
||||
"keywords": [ |
||||
"uni-ui", |
||||
"折叠", |
||||
"折叠面板", |
||||
"手风琴" |
||||
], |
||||
"repository": "https://github.com/dcloudio/uni-ui", |
||||
"engines": { |
||||
"HBuilderX": "" |
||||
}, |
||||
"directories": { |
||||
"example": "../../temps/example_temps" |
||||
}, |
||||
"dcloudext": { |
||||
"category": [ |
||||
"前端组件", |
||||
"通用组件" |
||||
], |
||||
"sale": { |
||||
"regular": { |
||||
"price": "0.00" |
||||
}, |
||||
"sourcecode": { |
||||
"price": "0.00" |
||||
} |
||||
}, |
||||
"contact": { |
||||
"qq": "" |
||||
}, |
||||
"declaration": { |
||||
"ads": "无", |
||||
"data": "无", |
||||
"permissions": "无" |
||||
}, |
||||
"npmurl": "https://www.npmjs.com/package/@dcloudio/uni-ui" |
||||
}, |
||||
"uni_modules": { |
||||
"dependencies": [ |
||||
"uni-scss", |
||||
"uni-icons" |
||||
], |
||||
"encrypt": [], |
||||
"platforms": { |
||||
"cloud": { |
||||
"tcb": "y", |
||||
"aliyun": "y" |
||||
}, |
||||
"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" |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,12 @@ |
||||
|
||||
|
||||
## Collapse 折叠面板 |
||||
> **组件名:uni-collapse** |
||||
> 代码块: `uCollapse` |
||||
> 关联组件:`uni-collapse-item`、`uni-icons`。 |
||||
|
||||
|
||||
折叠面板用来折叠/显示过长的内容或者是列表。通常是在多内容分类项使用,折叠不重要的内容,显示重要内容。点击可以展开折叠部分。 |
||||
|
||||
### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-collapse) |
||||
#### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839 |
@ -0,0 +1,15 @@ |
||||
## 1.0.1(2021-11-23) |
||||
- 优化 label、label-width 属性 |
||||
## 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-combox](https://uniapp.dcloud.io/component/uniui/uni-combox) |
||||
## 0.1.0(2021-07-30) |
||||
- 组件兼容 vue3,如何创建vue3项目,详见 [uni-app 项目支持 vue3 介绍](https://ask.dcloud.net.cn/article/37834) |
||||
## 0.0.6(2021-05-12) |
||||
- 新增 组件示例地址 |
||||
## 0.0.5(2021-04-21) |
||||
- 优化 添加依赖 uni-icons, 导入后自动下载依赖 |
||||
## 0.0.4(2021-02-05) |
||||
- 优化 组件引用关系,通过uni_modules引用组件 |
||||
## 0.0.3(2021-02-04) |
||||
- 调整为uni_modules目录规范 |
@ -0,0 +1,275 @@ |
||||
<template> |
||||
<view class="uni-combox" :class="border ? '' : 'uni-combox__no-border'"> |
||||
<view v-if="label" class="uni-combox__label" :style="labelStyle"> |
||||
<text>{{label}}</text> |
||||
</view> |
||||
<view class="uni-combox__input-box"> |
||||
<input class="uni-combox__input" type="text" :placeholder="placeholder" |
||||
placeholder-class="uni-combox__input-plac" v-model="inputVal" @input="onInput" @focus="onFocus" |
||||
@blur="onBlur" /> |
||||
<uni-icons :type="showSelector? 'top' : 'bottom'" size="14" color="#999" @click="toggleSelector"> |
||||
</uni-icons> |
||||
</view> |
||||
<view class="uni-combox__selector" v-if="showSelector"> |
||||
<view class="uni-popper__arrow"></view> |
||||
<scroll-view scroll-y="true" class="uni-combox__selector-scroll"> |
||||
<view class="uni-combox__selector-empty" v-if="filterCandidatesLength === 0"> |
||||
<text>{{emptyTips}}</text> |
||||
</view> |
||||
<view class="uni-combox__selector-item" v-for="(item,index) in filterCandidates" :key="index" |
||||
@click="onSelectorClick(index)"> |
||||
<text>{{item}}</text> |
||||
</view> |
||||
</scroll-view> |
||||
</view> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
/** |
||||
* Combox 组合输入框 |
||||
* @description 组合输入框一般用于既可以输入也可以选择的场景 |
||||
* @tutorial https://ext.dcloud.net.cn/plugin?id=1261 |
||||
* @property {String} label 左侧文字 |
||||
* @property {String} labelWidth 左侧内容宽度 |
||||
* @property {String} placeholder 输入框占位符 |
||||
* @property {Array} candidates 候选项列表 |
||||
* @property {String} emptyTips 筛选结果为空时显示的文字 |
||||
* @property {String} value 组合框的值 |
||||
*/ |
||||
export default { |
||||
name: 'uniCombox', |
||||
emits: ['input', 'update:modelValue'], |
||||
props: { |
||||
border: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
label: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
labelWidth: { |
||||
type: String, |
||||
default: 'auto' |
||||
}, |
||||
placeholder: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
candidates: { |
||||
type: Array, |
||||
default () { |
||||
return [] |
||||
} |
||||
}, |
||||
emptyTips: { |
||||
type: String, |
||||
default: '无匹配项' |
||||
}, |
||||
// #ifndef VUE3 |
||||
value: { |
||||
type: [String, Number], |
||||
default: '' |
||||
}, |
||||
// #endif |
||||
// #ifdef VUE3 |
||||
modelValue: { |
||||
type: [String, Number], |
||||
default: '' |
||||
}, |
||||
// #endif |
||||
}, |
||||
data() { |
||||
return { |
||||
showSelector: false, |
||||
inputVal: '' |
||||
} |
||||
}, |
||||
computed: { |
||||
labelStyle() { |
||||
if (this.labelWidth === 'auto') { |
||||
return "" |
||||
} |
||||
return `width: ${this.labelWidth}` |
||||
}, |
||||
filterCandidates() { |
||||
return this.candidates.filter((item) => { |
||||
return item.toString().indexOf(this.inputVal) > -1 |
||||
}) |
||||
}, |
||||
filterCandidatesLength() { |
||||
return this.filterCandidates.length |
||||
} |
||||
}, |
||||
watch: { |
||||
// #ifndef VUE3 |
||||
value: { |
||||
handler(newVal) { |
||||
this.inputVal = newVal |
||||
}, |
||||
immediate: true |
||||
}, |
||||
// #endif |
||||
// #ifdef VUE3 |
||||
modelValue: { |
||||
handler(newVal) { |
||||
this.inputVal = newVal |
||||
}, |
||||
immediate: true |
||||
}, |
||||
// #endif |
||||
}, |
||||
methods: { |
||||
toggleSelector() { |
||||
this.showSelector = !this.showSelector |
||||
}, |
||||
onFocus() { |
||||
this.showSelector = true |
||||
}, |
||||
onBlur() { |
||||
setTimeout(() => { |
||||
this.showSelector = false |
||||
}, 153) |
||||
}, |
||||
onSelectorClick(index) { |
||||
this.inputVal = this.filterCandidates[index] |
||||
this.showSelector = false |
||||
this.$emit('input', this.inputVal) |
||||
this.$emit('update:modelValue', this.inputVal) |
||||
}, |
||||
onInput() { |
||||
setTimeout(() => { |
||||
this.$emit('input', this.inputVal) |
||||
this.$emit('update:modelValue', this.inputVal) |
||||
}) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss" scoped> |
||||
.uni-combox { |
||||
font-size: 14px; |
||||
border: 1px solid #DCDFE6; |
||||
border-radius: 4px; |
||||
padding: 6px 10px; |
||||
position: relative; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
// height: 40px; |
||||
flex-direction: row; |
||||
align-items: center; |
||||
// border-bottom: solid 1px #DDDDDD; |
||||
} |
||||
|
||||
.uni-combox__label { |
||||
font-size: 16px; |
||||
line-height: 22px; |
||||
padding-right: 10px; |
||||
color: #999999; |
||||
} |
||||
|
||||
.uni-combox__input-box { |
||||
position: relative; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex: 1; |
||||
flex-direction: row; |
||||
align-items: center; |
||||
} |
||||
|
||||
.uni-combox__input { |
||||
flex: 1; |
||||
font-size: 14px; |
||||
height: 22px; |
||||
line-height: 22px; |
||||
} |
||||
|
||||
.uni-combox__input-plac { |
||||
font-size: 14px; |
||||
color: #999; |
||||
} |
||||
|
||||
.uni-combox__selector { |
||||
/* #ifndef APP-NVUE */ |
||||
box-sizing: border-box; |
||||
/* #endif */ |
||||
position: absolute; |
||||
top: calc(100% + 12px); |
||||
left: 0; |
||||
width: 100%; |
||||
background-color: #FFFFFF; |
||||
border: 1px solid #EBEEF5; |
||||
border-radius: 6px; |
||||
box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1); |
||||
z-index: 2; |
||||
padding: 4px 0; |
||||
} |
||||
|
||||
.uni-combox__selector-scroll { |
||||
/* #ifndef APP-NVUE */ |
||||
max-height: 200px; |
||||
box-sizing: border-box; |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-combox__selector-empty, |
||||
.uni-combox__selector-item { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
cursor: pointer; |
||||
/* #endif */ |
||||
line-height: 36px; |
||||
font-size: 14px; |
||||
text-align: center; |
||||
// border-bottom: solid 1px #DDDDDD; |
||||
padding: 0px 10px; |
||||
} |
||||
|
||||
.uni-combox__selector-item:hover { |
||||
background-color: #f9f9f9; |
||||
} |
||||
|
||||
.uni-combox__selector-empty:last-child, |
||||
.uni-combox__selector-item:last-child { |
||||
/* #ifndef APP-NVUE */ |
||||
border-bottom: none; |
||||
/* #endif */ |
||||
} |
||||
|
||||
// picker 弹出层通用的指示小三角 |
||||
.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; |
||||
} |
||||
|
||||
.uni-combox__no-border { |
||||
border: none; |
||||
} |
||||
</style> |
@ -0,0 +1,90 @@ |
||||
{ |
||||
"id": "uni-combox", |
||||
"displayName": "uni-combox 组合框", |
||||
"version": "1.0.1", |
||||
"description": "可以选择也可以输入的表单项 ", |
||||
"keywords": [ |
||||
"uni-ui", |
||||
"uniui", |
||||
"combox", |
||||
"组合框", |
||||
"select" |
||||
], |
||||
"repository": "https://github.com/dcloudio/uni-ui", |
||||
"engines": { |
||||
"HBuilderX": "" |
||||
}, |
||||
"directories": { |
||||
"example": "../../temps/example_temps" |
||||
}, |
||||
"dcloudext": { |
||||
"category": [ |
||||
"前端组件", |
||||
"通用组件" |
||||
], |
||||
"sale": { |
||||
"regular": { |
||||
"price": "0.00" |
||||
}, |
||||
"sourcecode": { |
||||
"price": "0.00" |
||||
} |
||||
}, |
||||
"contact": { |
||||
"qq": "" |
||||
}, |
||||
"declaration": { |
||||
"ads": "无", |
||||
"data": "无", |
||||
"permissions": "无" |
||||
}, |
||||
"npmurl": "https://www.npmjs.com/package/@dcloudio/uni-ui" |
||||
}, |
||||
"uni_modules": { |
||||
"dependencies": [ |
||||
"uni-scss", |
||||
"uni-icons" |
||||
], |
||||
"encrypt": [], |
||||
"platforms": { |
||||
"cloud": { |
||||
"tcb": "y", |
||||
"aliyun": "y" |
||||
}, |
||||
"client": { |
||||
"App": { |
||||
"app-vue": "y", |
||||
"app-nvue": "n" |
||||
}, |
||||
"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" |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,11 @@ |
||||
|
||||
|
||||
## Combox 组合框 |
||||
> **组件名:uni-combox** |
||||
> 代码块: `uCombox` |
||||
|
||||
|
||||
组合框组件。 |
||||
|
||||
### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-combox) |
||||
#### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839 |
@ -0,0 +1,24 @@ |
||||
## 1.2.2(2022-01-19) |
||||
- 修复 在微信小程序中样式不生效的bug |
||||
## 1.2.1(2022-01-18) |
||||
- 新增 update 方法 ,在动态更新时间后,刷新组件 |
||||
## 1.2.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-countdown](https://uniapp.dcloud.io/component/uniui/uni-countdown) |
||||
## 1.1.3(2021-10-18) |
||||
- 重构 |
||||
- 新增 font-size 支持自定义字体大小 |
||||
## 1.1.2(2021-08-24) |
||||
- 新增 支持国际化 |
||||
## 1.1.1(2021-07-30) |
||||
- 优化 vue3下小程序事件警告的问题 |
||||
## 1.1.0(2021-07-30) |
||||
- 组件兼容 vue3,如何创建vue3项目,详见 [uni-app 项目支持 vue3 介绍](https://ask.dcloud.net.cn/article/37834) |
||||
## 1.0.5(2021-06-18) |
||||
- 修复 uni-countdown 重复赋值跳两秒的 bug |
||||
## 1.0.4(2021-05-12) |
||||
- 新增 组件示例地址 |
||||
## 1.0.3(2021-05-08) |
||||
- 修复 uni-countdown 不能控制倒计时的 bug |
||||
## 1.0.2(2021-02-04) |
||||
- 调整为uni_modules目录规范 |
@ -0,0 +1,6 @@ |
||||
{ |
||||
"uni-countdown.day": "day", |
||||
"uni-countdown.h": "h", |
||||
"uni-countdown.m": "m", |
||||
"uni-countdown.s": "s" |
||||
} |
@ -0,0 +1,8 @@ |
||||
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 |
||||
} |
@ -0,0 +1,6 @@ |
||||
{ |
||||
"uni-countdown.day": "天", |
||||
"uni-countdown.h": "时", |
||||
"uni-countdown.m": "分", |
||||
"uni-countdown.s": "秒" |
||||
} |
@ -0,0 +1,6 @@ |
||||
{ |
||||
"uni-countdown.day": "天", |
||||
"uni-countdown.h": "時", |
||||
"uni-countdown.m": "分", |
||||
"uni-countdown.s": "秒" |
||||
} |
@ -0,0 +1,271 @@ |
||||
<template> |
||||
<view class="uni-countdown"> |
||||
<text v-if="showDay" :style="[timeStyle]" class="uni-countdown__number">{{ d }}</text> |
||||
<text v-if="showDay" :style="[splitorStyle]" class="uni-countdown__splitor">{{dayText}}</text> |
||||
<text :style="[timeStyle]" class="uni-countdown__number">{{ h }}</text> |
||||
<text :style="[splitorStyle]" class="uni-countdown__splitor">{{ showColon ? ':' : hourText }}</text> |
||||
<text :style="[timeStyle]" class="uni-countdown__number">{{ i }}</text> |
||||
<text :style="[splitorStyle]" class="uni-countdown__splitor">{{ showColon ? ':' : minuteText }}</text> |
||||
<text :style="[timeStyle]" class="uni-countdown__number">{{ s }}</text> |
||||
<text v-if="!showColon" :style="[splitorStyle]" class="uni-countdown__splitor">{{secondText}}</text> |
||||
</view> |
||||
</template> |
||||
<script> |
||||
import { |
||||
initVueI18n |
||||
} from '@dcloudio/uni-i18n' |
||||
import messages from './i18n/index.js' |
||||
const { |
||||
t |
||||
} = initVueI18n(messages) |
||||
/** |
||||
* Countdown 倒计时 |
||||
* @description 倒计时组件 |
||||
* @tutorial https://ext.dcloud.net.cn/plugin?id=25 |
||||
* @property {String} backgroundColor 背景色 |
||||
* @property {String} color 文字颜色 |
||||
* @property {Number} day 天数 |
||||
* @property {Number} hour 小时 |
||||
* @property {Number} minute 分钟 |
||||
* @property {Number} second 秒 |
||||
* @property {Number} timestamp 时间戳 |
||||
* @property {Boolean} showDay = [true|false] 是否显示天数 |
||||
* @property {Boolean} show-colon = [true|false] 是否以冒号为分隔符 |
||||
* @property {String} splitorColor 分割符号颜色 |
||||
* @event {Function} timeup 倒计时时间到触发事件 |
||||
* @example <uni-countdown :day="1" :hour="1" :minute="12" :second="40"></uni-countdown> |
||||
*/ |
||||
export default { |
||||
name: 'UniCountdown', |
||||
emits: ['timeup'], |
||||
props: { |
||||
showDay: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
showColon: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
start: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
backgroundColor: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
color: { |
||||
type: String, |
||||
default: '#333' |
||||
}, |
||||
fontSize: { |
||||
type: Number, |
||||
default: 14 |
||||
}, |
||||
splitorColor: { |
||||
type: String, |
||||
default: '#333' |
||||
}, |
||||
day: { |
||||
type: Number, |
||||
default: 0 |
||||
}, |
||||
hour: { |
||||
type: Number, |
||||
default: 0 |
||||
}, |
||||
minute: { |
||||
type: Number, |
||||
default: 0 |
||||
}, |
||||
second: { |
||||
type: Number, |
||||
default: 0 |
||||
}, |
||||
timestamp: { |
||||
type: Number, |
||||
default: 0 |
||||
} |
||||
}, |
||||
data() { |
||||
return { |
||||
timer: null, |
||||
syncFlag: false, |
||||
d: '00', |
||||
h: '00', |
||||
i: '00', |
||||
s: '00', |
||||
leftTime: 0, |
||||
seconds: 0 |
||||
} |
||||
}, |
||||
computed: { |
||||
dayText() { |
||||
return t("uni-countdown.day") |
||||
}, |
||||
hourText(val) { |
||||
return t("uni-countdown.h") |
||||
}, |
||||
minuteText(val) { |
||||
return t("uni-countdown.m") |
||||
}, |
||||
secondText(val) { |
||||
return t("uni-countdown.s") |
||||
}, |
||||
timeStyle() { |
||||
const { |
||||
color, |
||||
backgroundColor, |
||||
fontSize |
||||
} = this |
||||
return { |
||||
color, |
||||
backgroundColor, |
||||
fontSize: `${fontSize}px`, |
||||
width: `${fontSize * 22 / 14}px`, // 按字体大小为 14px 时的比例缩放 |
||||
lineHeight: `${fontSize * 20 / 14}px`, |
||||
borderRadius: `${fontSize * 3 / 14}px`, |
||||
} |
||||
}, |
||||
splitorStyle() { |
||||
const { splitorColor, fontSize, backgroundColor } = this |
||||
return { |
||||
color: splitorColor, |
||||
fontSize: `${fontSize * 12 / 14}px`, |
||||
margin: backgroundColor ? `${fontSize * 4 / 14}px` : '' |
||||
} |
||||
} |
||||
}, |
||||
watch: { |
||||
day(val) { |
||||
this.changeFlag() |
||||
}, |
||||
hour(val) { |
||||
this.changeFlag() |
||||
}, |
||||
minute(val) { |
||||
this.changeFlag() |
||||
}, |
||||
second(val) { |
||||
this.changeFlag() |
||||
}, |
||||
start: { |
||||
immediate: true, |
||||
handler(newVal, oldVal) { |
||||
if (newVal) { |
||||
this.startData(); |
||||
} else { |
||||
if (!oldVal) return |
||||
clearInterval(this.timer) |
||||
} |
||||
} |
||||
|
||||
} |
||||
}, |
||||
created: function(e) { |
||||
this.seconds = this.toSeconds(this.timestamp, this.day, this.hour, this.minute, this.second) |
||||
this.countDown() |
||||
}, |
||||
// #ifndef VUE3 |
||||
destroyed() { |
||||
clearInterval(this.timer) |
||||
}, |
||||
// #endif |
||||
// #ifdef VUE3 |
||||
unmounted() { |
||||
clearInterval(this.timer) |
||||
}, |
||||
// #endif |
||||
methods: { |
||||
toSeconds(timestamp, day, hours, minutes, seconds) { |
||||
if (timestamp) { |
||||
return timestamp - parseInt(new Date().getTime() / 1000, 10) |
||||
} |
||||
return day * 60 * 60 * 24 + hours * 60 * 60 + minutes * 60 + seconds |
||||
}, |
||||
timeUp() { |
||||
clearInterval(this.timer) |
||||
this.$emit('timeup') |
||||
}, |
||||
countDown() { |
||||
let seconds = this.seconds |
||||
let [day, hour, minute, second] = [0, 0, 0, 0] |
||||
if (seconds > 0) { |
||||
day = Math.floor(seconds / (60 * 60 * 24)) |
||||
hour = Math.floor(seconds / (60 * 60)) - (day * 24) |
||||
minute = Math.floor(seconds / 60) - (day * 24 * 60) - (hour * 60) |
||||
second = Math.floor(seconds) - (day * 24 * 60 * 60) - (hour * 60 * 60) - (minute * 60) |
||||
} else { |
||||
this.timeUp() |
||||
} |
||||
if (day < 10) { |
||||
day = '0' + day |
||||
} |
||||
if (hour < 10) { |
||||
hour = '0' + hour |
||||
} |
||||
if (minute < 10) { |
||||
minute = '0' + minute |
||||
} |
||||
if (second < 10) { |
||||
second = '0' + second |
||||
} |
||||
this.d = day |
||||
this.h = hour |
||||
this.i = minute |
||||
this.s = second |
||||
}, |
||||
startData() { |
||||
this.seconds = this.toSeconds(this.timestamp, this.day, this.hour, this.minute, this.second) |
||||
if (this.seconds <= 0) { |
||||
this.seconds = this.toSeconds(0, 0, 0, 0, 0) |
||||
this.countDown() |
||||
return |
||||
} |
||||
clearInterval(this.timer) |
||||
this.countDown() |
||||
this.timer = setInterval(() => { |
||||
this.seconds-- |
||||
if (this.seconds < 0) { |
||||
this.timeUp() |
||||
return |
||||
} |
||||
this.countDown() |
||||
}, 1000) |
||||
}, |
||||
update(){ |
||||
this.startData(); |
||||
}, |
||||
changeFlag() { |
||||
if (!this.syncFlag) { |
||||
this.seconds = this.toSeconds(this.timestamp, this.day, this.hour, this.minute, this.second) |
||||
this.startData(); |
||||
this.syncFlag = true; |
||||
} |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
<style lang="scss" scoped> |
||||
$font-size: 14px; |
||||
|
||||
.uni-countdown { |
||||
display: flex; |
||||
flex-direction: row; |
||||
justify-content: flex-start; |
||||
align-items: center; |
||||
|
||||
&__splitor { |
||||
margin: 0 2px; |
||||
font-size: $font-size; |
||||
color: #333; |
||||
} |
||||
|
||||
&__number { |
||||
border-radius: 3px; |
||||
text-align: center; |
||||
font-size: $font-size; |
||||
} |
||||
} |
||||
</style> |
@ -0,0 +1,86 @@ |
||||
{ |
||||
"id": "uni-countdown", |
||||
"displayName": "uni-countdown 倒计时", |
||||
"version": "1.2.2", |
||||
"description": "CountDown 倒计时组件", |
||||
"keywords": [ |
||||
"uni-ui", |
||||
"uniui", |
||||
"countdown", |
||||
"倒计时" |
||||
], |
||||
"repository": "https://github.com/dcloudio/uni-ui", |
||||
"engines": { |
||||
"HBuilderX": "" |
||||
}, |
||||
"directories": { |
||||
"example": "../../temps/example_temps" |
||||
}, |
||||
"dcloudext": { |
||||
"category": [ |
||||
"前端组件", |
||||
"通用组件" |
||||
], |
||||
"sale": { |
||||
"regular": { |
||||
"price": "0.00" |
||||
}, |
||||
"sourcecode": { |
||||
"price": "0.00" |
||||
} |
||||
}, |
||||
"contact": { |
||||
"qq": "" |
||||
}, |
||||
"declaration": { |
||||
"ads": "无", |
||||
"data": "无", |
||||
"permissions": "无" |
||||
}, |
||||
"npmurl": "https://www.npmjs.com/package/@dcloudio/uni-ui" |
||||
}, |
||||
"uni_modules": { |
||||
"dependencies": ["uni-scss"], |
||||
"encrypt": [], |
||||
"platforms": { |
||||
"cloud": { |
||||
"tcb": "y", |
||||
"aliyun": "y" |
||||
}, |
||||
"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" |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,10 @@ |
||||
|
||||
|
||||
## CountDown 倒计时 |
||||
> **组件名:uni-countdown** |
||||
> 代码块: `uCountDown` |
||||
|
||||
倒计时组件。 |
||||
|
||||
### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-countdown) |
||||
#### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839 |
@ -0,0 +1,45 @@ |
||||
## 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) |
||||
- “暂无数据”显示居中 |
@ -0,0 +1,821 @@ |
||||
<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="chagne"> |
||||
<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="chagne"> |
||||
<!-- --> |
||||
<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; |
||||
}, |
||||
chagne(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 = [] |
||||
} |
||||
} |
||||
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 { |
||||
.radio__inner-icon { |
||||
opacity: 1; |
||||
} |
||||
} |
||||
.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> |
@ -0,0 +1,84 @@ |
||||
{ |
||||
"id": "uni-data-checkbox", |
||||
"displayName": "uni-data-checkbox 数据选择器", |
||||
"version": "1.0.3", |
||||
"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" |
||||
}, |
||||
"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" |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,18 @@ |
||||
|
||||
|
||||
## 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 |
@ -0,0 +1,66 @@ |
||||
## 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 目录规范 |
@ -0,0 +1,45 @@ |
||||
// #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
|
@ -0,0 +1,554 @@ |
||||
<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'], |
||||
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.form = this.getForm('uniForms') |
||||
this.formItem = this.getForm('uniFormsItem') |
||||
if (this.formItem) { |
||||
if (this.formItem.name) { |
||||
this.rename = this.formItem.name |
||||
this.form.inputChildrens.push(this) |
||||
} |
||||
} |
||||
|
||||
this.$nextTick(() => { |
||||
this.load() |
||||
}) |
||||
}, |
||||
methods: { |
||||
clear() { |
||||
this.inputSelected.splice(0) |
||||
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.parentField && !this.selfField && this.hasValue) { |
||||
this.getNodeData(() => { |
||||
this.inputSelected = this.selected.slice(0) |
||||
}) |
||||
} else if (this.hasValue) { |
||||
this.getTreePath(() => { |
||||
this.inputSelected = this.selected.slice(0) |
||||
}) |
||||
} |
||||
}, |
||||
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; |
||||
}, |
||||
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) { |
||||
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: 14px; |
||||
} |
||||
|
||||
.error-text { |
||||
color: #DD524D; |
||||
} |
||||
|
||||
.input-value { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
align-items: center; |
||||
flex-wrap: nowrap; |
||||
font-size: 14px; |
||||
/* 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; |
||||
top: 20%; |
||||
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: 750rpx; |
||||
/* #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> |
@ -0,0 +1,563 @@ |
||||
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: 20 |
||||
}, |
||||
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 |
||||
}, |
||||
postField() { |
||||
let fields = [this.field]; |
||||
if (this.parentField) { |
||||
fields.push(`${this.parentField} as parent_value`); |
||||
} |
||||
return fields.join(','); |
||||
}, |
||||
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 = [] |
||||
}, |
||||
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 |
||||
}, |
||||
getNodeData(callback) { |
||||
if (this.loading) { |
||||
return |
||||
} |
||||
this.loading = true |
||||
this.getCommand({ |
||||
field: this.postField, |
||||
where: this._pathWhere() |
||||
}).then((res) => { |
||||
this.loading = false |
||||
this.selected = res.result.data |
||||
callback && callback() |
||||
}).catch((err) => { |
||||
this.loading = false |
||||
this.errorMessage = err |
||||
}) |
||||
}, |
||||
getTreePath(callback) { |
||||
if (this.loading) { |
||||
return |
||||
} |
||||
this.loading = true |
||||
|
||||
this.getCommand({ |
||||
field: this.postField, |
||||
getTreePath: { |
||||
startWith: `${this.selfField}=='${this.dataValue}'` |
||||
} |
||||
}).then((res) => { |
||||
this.loading = false |
||||
let treePath = [] |
||||
this._extractTreePath(res.result.data, treePath) |
||||
this.selected = treePath |
||||
callback && callback() |
||||
}).catch((err) => { |
||||
this.loading = false |
||||
this.errorMessage = err |
||||
}) |
||||
}, |
||||
loadData() { |
||||
if (this.isLocaldata) { |
||||
this._processLocalData() |
||||
return |
||||
} |
||||
|
||||
if (this.dataValue != null) { |
||||
this._loadNodeData((data) => { |
||||
this._treeData = data |
||||
this._updateBindData() |
||||
this._updateSelected() |
||||
}) |
||||
return |
||||
} |
||||
|
||||
if (this.stepSearh) { |
||||
this._loadNodeData((data) => { |
||||
this._treeData = data |
||||
this._updateBindData() |
||||
}) |
||||
} else { |
||||
this._loadAllData((data) => { |
||||
this._treeData = [] |
||||
this._extractTree(data, this._treeData, null) |
||||
this._updateBindData() |
||||
}) |
||||
} |
||||
}, |
||||
_loadAllData(callback) { |
||||
if (this.loading) { |
||||
return |
||||
} |
||||
this.loading = true |
||||
|
||||
this.getCommand({ |
||||
field: this.postField, |
||||
gettree: true, |
||||
startwith: `${this.selfField}=='${this.dataValue}'` |
||||
}).then((res) => { |
||||
this.loading = false |
||||
callback(res.result.data) |
||||
this.onDataChange() |
||||
}).catch((err) => { |
||||
this.loading = false |
||||
this.errorMessage = err |
||||
}) |
||||
}, |
||||
_loadNodeData(callback, pw) { |
||||
if (this.loading) { |
||||
return |
||||
} |
||||
this.loading = true |
||||
|
||||
this.getCommand({ |
||||
field: this.postField, |
||||
where: pw || this._postWhere(), |
||||
pageSize: 500 |
||||
}).then((res) => { |
||||
this.loading = false |
||||
callback(res.result.data) |
||||
this.onDataChange() |
||||
}).catch((err) => { |
||||
this.loading = false |
||||
this.errorMessage = err |
||||
}) |
||||
}, |
||||
_pathWhere() { |
||||
let result = [] |
||||
let where_field = this._getParentNameByField(); |
||||
if (where_field) { |
||||
result.push(`${where_field} == '${this.dataValue}'`) |
||||
} |
||||
|
||||
if (this.where) { |
||||
return `(${this.where}) && (${result.join(' || ')})` |
||||
} |
||||
|
||||
return result.join(' || ') |
||||
}, |
||||
_postWhere() { |
||||
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(' && ') |
||||
}, |
||||
_nodeWhere() { |
||||
let result = [] |
||||
let selected = this.selected |
||||
if (selected.length) { |
||||
result.push(`${this.parentField} == '${selected[selected.length - 1].value}'`) |
||||
} |
||||
|
||||
if (this.where) { |
||||
return `(${this.where}) && (${result.join(' || ')})` |
||||
} |
||||
|
||||
return result.join(' || ') |
||||
}, |
||||
_getParentNameByField() { |
||||
const fields = this.field.split(','); |
||||
let where_field = 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') { |
||||
where_field = items[0].trim(); |
||||
break; |
||||
} |
||||
} |
||||
return where_field |
||||
}, |
||||
_isTreeView() { |
||||
return (this.parentField && this.selfField) |
||||
}, |
||||
_updateSelected() { |
||||
var dl = this.dataList |
||||
var sl = this.selected |
||||
let textField = this.map.text |
||||
let valueField = this.map.value |
||||
for (var i = 0; i < sl.length; i++) { |
||||
var value = sl[i].value |
||||
var dl2 = dl[i] |
||||
for (var j = 0; j < dl2.length; j++) { |
||||
var item2 = dl2[j] |
||||
if (item2[valueField] === value) { |
||||
sl[i].text = item2[textField] |
||||
break |
||||
} |
||||
} |
||||
} |
||||
}, |
||||
_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 |
||||
} |
||||
}, |
||||
_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++) { |
||||
var value = paths[i].value |
||||
var 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 [] |
||||
}, |
||||
_processLocalData() { |
||||
this._treeData = [] |
||||
this._extractTree(this.localdata, this._treeData) |
||||
|
||||
var 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) |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,335 @@ |
||||
<template> |
||||
<view class="uni-data-pickerview"> |
||||
<scroll-view class="selected-area" scroll-x="true" scroll-y="false" :show-scrollbar="false"> |
||||
<view class="selected-list"> |
||||
<template v-for="(item,index) in selected"> |
||||
<view class="selected-item" |
||||
:class="{'selected-item-active':index==selectedIndex, 'selected-item-text-overflow': ellipsis}" |
||||
v-if="item.text" @click="handleSelect(index)"> |
||||
<text class="">{{item.text}}</text> |
||||
</view> |
||||
</template> |
||||
</view> |
||||
</scroll-view> |
||||
<view class="tab-c"> |
||||
<template v-for="(child, i) in dataList" > |
||||
<scroll-view class="list" :key="i" v-if="i==selectedIndex" :scroll-y="true"> |
||||
<view class="item" :class="{'is-disabled': !!item.disable}" v-for="(item, j) in child" |
||||
@click="handleNodeClick(item, i, j)"> |
||||
<text class="item-text item-text-overflow">{{item[map.text]}}</text> |
||||
<view class="check" v-if="selected.length > i && item[map.value] == selected[i].value"></view> |
||||
</view> |
||||
</scroll-view> |
||||
</template> |
||||
|
||||
<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 |
||||
} |
||||
}, |
||||
data() { |
||||
return {} |
||||
}, |
||||
created() { |
||||
if (this.managedMode) { |
||||
return |
||||
} |
||||
|
||||
this.$nextTick(() => { |
||||
this.load() |
||||
}) |
||||
}, |
||||
methods: { |
||||
onPropsChange() { |
||||
this._treeData = [] |
||||
this.selectedIndex = 0 |
||||
this.load() |
||||
}, |
||||
load() { |
||||
if (this.isLocaldata) { |
||||
this.loadData() |
||||
} else if (this.dataValue.length) { |
||||
this.getTreePath((res) => { |
||||
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._isTreeView() && !hasNodes) { |
||||
this.onSelectedChange(node, true) |
||||
return |
||||
} |
||||
|
||||
if (this.isLocaldata && (!hasNodes || isleaf)) { |
||||
this.onSelectedChange(node, true) |
||||
return |
||||
} |
||||
|
||||
if (!isleaf && !hasNodes) { |
||||
this._loadNodeData((data) => { |
||||
if (!data.length) { |
||||
node.isleaf = true |
||||
} else { |
||||
this._treeData.push(...data) |
||||
this._updateBindData(node) |
||||
} |
||||
this.onSelectedChange(node, node.isleaf) |
||||
}, this._nodeWhere()) |
||||
return |
||||
} |
||||
|
||||
this.onSelectedChange(node, false) |
||||
}, |
||||
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; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
flex-wrap: nowrap; |
||||
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> |
@ -0,0 +1,90 @@ |
||||
{ |
||||
"id": "uni-data-picker", |
||||
"displayName": "uni-data-picker 数据驱动的picker选择器", |
||||
"version": "1.0.8", |
||||
"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" |
||||
}, |
||||
"client": { |
||||
"App": { |
||||
"app-vue": "y", |
||||
"app-nvue": "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", |
||||
"联盟": "u" |
||||
}, |
||||
"Vue": { |
||||
"vue2": "y", |
||||
"vue3": "y" |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,22 @@ |
||||
## 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 |
@ -0,0 +1,22 @@ |
||||
## 0.1.9(2022-09-05) |
||||
- 修复 微信小程序下拉框出现后选择会点击到蒙板后面的输入框 |
||||
## 0.1.8(2022-08-29) |
||||
- 修复 点击的位置不准确 |
||||
## 0.1.7(2022-08-12) |
||||
- 新增 支持 disabled 属性 |
||||
## 0.1.6(2022-07-06) |
||||
- 修复 pc端宽度异常的bug |
||||
## 0.1.5 |
||||
- 修复 pc端宽度异常的bug |
||||
## 0.1.4(2022-07-05) |
||||
- 优化 显示样式 |
||||
## 0.1.3(2022-06-02) |
||||
- 修复 localdata 赋值不生效的 bug |
||||
- 新增 支持 uni.scss 修改颜色 |
||||
- 新增 支持选项禁用(数据选项设置 disabled: true 即禁用) |
||||
## 0.1.2(2022-05-08) |
||||
- 修复 当 value 为 0 时选择不生效的 bug |
||||
## 0.1.1(2022-05-07) |
||||
- 新增 记住上次的选项(仅 collection 存在时有效) |
||||
## 0.1.0(2022-04-22) |
||||
- 初始化 |
@ -0,0 +1,440 @@ |
||||
<template> |
||||
<view class="uni-stat__select"> |
||||
<span v-if="label" class="uni-label-text hide-on-phone">{{label + ':'}}</span> |
||||
<view class="uni-stat-box" :class="{'uni-stat__actived': current}"> |
||||
<view class="uni-select" :class="{'uni-select--disabled':disabled}"> |
||||
<view class="uni-select__input-box" @click="toggleSelector"> |
||||
<view v-if="current" class="uni-select__input-text">{{current}}</view> |
||||
<view v-else class="uni-select__input-text uni-select__input-placeholder">{{typePlaceholder}}</view> |
||||
<uni-icons v-if="current && clear" type="clear" color="#c0c4cc" size="24" @click="clearVal" /> |
||||
<uni-icons v-else :type="showSelector? 'top' : 'bottom'" size="14" color="#999" /> |
||||
</view> |
||||
<view class="uni-select--mask" v-if="showSelector" @click="toggleSelector" /> |
||||
<view class="uni-select__selector" v-if="showSelector"> |
||||
<view class="uni-popper__arrow"></view> |
||||
<scroll-view scroll-y="true" class="uni-select__selector-scroll"> |
||||
<view class="uni-select__selector-empty" v-if="mixinDatacomResData.length === 0"> |
||||
<text>{{emptyTips}}</text> |
||||
</view> |
||||
<view v-else class="uni-select__selector-item" v-for="(item,index) in mixinDatacomResData" |
||||
:key="index" @click="change(item)"> |
||||
<text |
||||
:class="{'uni-select__selector__disabled': item.disable}">{{formatItemName(item)}}</text> |
||||
</view> |
||||
</scroll-view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
/** |
||||
* DataChecklist 数据选择器 |
||||
* @description 通过数据渲染的下拉框组件 |
||||
* @tutorial https://uniapp.dcloud.io/component/uniui/uni-data-select |
||||
* @property {String} value 默认值 |
||||
* @property {Array} localdata 本地数据 ,格式 [{text:'',value:''}] |
||||
* @property {Boolean} clear 是否可以清空已选项 |
||||
* @property {Boolean} emptyText 没有数据时显示的文字 ,本地数据无效 |
||||
* @property {String} label 左侧标题 |
||||
* @property {String} placeholder 输入框的提示文字 |
||||
* @property {Boolean} disabled 是否禁用 |
||||
* @event {Function} change 选中发生变化触发 |
||||
*/ |
||||
|
||||
export default { |
||||
name: "uni-stat-select", |
||||
mixins: [uniCloud.mixinDatacom || {}], |
||||
data() { |
||||
return { |
||||
showSelector: false, |
||||
current: '', |
||||
mixinDatacomResData: [], |
||||
apps: [], |
||||
channels: [] |
||||
}; |
||||
}, |
||||
props: { |
||||
localdata: { |
||||
type: Array, |
||||
default () { |
||||
return [] |
||||
} |
||||
}, |
||||
value: { |
||||
type: [String, Number], |
||||
default: '' |
||||
}, |
||||
modelValue: { |
||||
type: [String, Number], |
||||
default: '' |
||||
}, |
||||
label: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
placeholder: { |
||||
type: String, |
||||
default: '请选择' |
||||
}, |
||||
emptyTips: { |
||||
type: String, |
||||
default: '无选项' |
||||
}, |
||||
clear: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
defItem: { |
||||
type: Number, |
||||
default: 0 |
||||
}, |
||||
disabled: { |
||||
type: Boolean, |
||||
default: false |
||||
} |
||||
}, |
||||
created() { |
||||
this.last = `${this.collection}_last_selected_option_value` |
||||
if (this.collection && !this.localdata.length) { |
||||
this.mixinDatacomEasyGet() |
||||
} |
||||
}, |
||||
computed: { |
||||
typePlaceholder() { |
||||
const text = { |
||||
'opendb-stat-app-versions': '版本', |
||||
'opendb-app-channels': '渠道', |
||||
'opendb-app-list': '应用' |
||||
} |
||||
const common = this.placeholder |
||||
const placeholder = text[this.collection] |
||||
return placeholder ? |
||||
common + placeholder : |
||||
common |
||||
} |
||||
}, |
||||
watch: { |
||||
localdata: { |
||||
immediate: true, |
||||
handler(val, old) { |
||||
if (Array.isArray(val) && old !== val) { |
||||
this.mixinDatacomResData = val |
||||
} |
||||
} |
||||
}, |
||||
// #ifndef VUE3 |
||||
value() { |
||||
this.initDefVal() |
||||
}, |
||||
// #endif |
||||
// #ifdef VUE3 |
||||
modelValue() { |
||||
this.initDefVal() |
||||
}, |
||||
// #endif |
||||
mixinDatacomResData: { |
||||
immediate: true, |
||||
handler(val) { |
||||
if (val.length) { |
||||
this.initDefVal() |
||||
} |
||||
} |
||||
} |
||||
}, |
||||
methods: { |
||||
initDefVal() { |
||||
let defValue = '' |
||||
if ((this.value || this.value === 0) && !this.isDisabled(this.value)) { |
||||
defValue = this.value |
||||
} else if ((this.modelValue || this.modelValue === 0) && !this.isDisabled(this.modelValue)) { |
||||
defValue = this.modelValue |
||||
} else { |
||||
let strogeValue |
||||
if (this.collection) { |
||||
strogeValue = uni.getStorageSync(this.last) |
||||
} |
||||
if (strogeValue || strogeValue === 0) { |
||||
defValue = strogeValue |
||||
} else { |
||||
let defItem = '' |
||||
if (this.defItem > 0 && this.defItem < this.mixinDatacomResData.length) { |
||||
defItem = this.mixinDatacomResData[this.defItem - 1].value |
||||
} |
||||
defValue = defItem |
||||
} |
||||
this.emit(defValue) |
||||
} |
||||
const def = this.mixinDatacomResData.find(item => item.value === defValue) |
||||
this.current = def ? this.formatItemName(def) : '' |
||||
}, |
||||
|
||||
/** |
||||
* @param {[String, Number]} value |
||||
* 判断用户给的 value 是否同时为禁用状态 |
||||
*/ |
||||
isDisabled(value) { |
||||
let isDisabled = false; |
||||
|
||||
this.mixinDatacomResData.forEach(item => { |
||||
if (item.value === value) { |
||||
isDisabled = item.disable |
||||
} |
||||
}) |
||||
|
||||
return isDisabled; |
||||
}, |
||||
|
||||
clearVal() { |
||||
this.emit('') |
||||
if (this.collection) { |
||||
uni.removeStorageSync(this.last) |
||||
} |
||||
}, |
||||
change(item) { |
||||
if (!item.disable) { |
||||
this.showSelector = false |
||||
this.current = this.formatItemName(item) |
||||
this.emit(item.value) |
||||
} |
||||
}, |
||||
emit(val) { |
||||
this.$emit('change', val) |
||||
this.$emit('input', val) |
||||
this.$emit('update:modelValue', val) |
||||
if (this.collection) { |
||||
uni.setStorageSync(this.last, val) |
||||
} |
||||
}, |
||||
|
||||
toggleSelector() { |
||||
if(this.disabled){ |
||||
return |
||||
} |
||||
|
||||
this.showSelector = !this.showSelector |
||||
}, |
||||
formatItemName(item) { |
||||
let { |
||||
text, |
||||
value, |
||||
channel_code |
||||
} = item |
||||
channel_code = channel_code ? `(${channel_code})` : '' |
||||
return this.collection.indexOf('app-list') > 0 ? |
||||
`${text}(${value})` : |
||||
( |
||||
text ? |
||||
text : |
||||
`未命名${channel_code}` |
||||
) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss"> |
||||
$uni-base-color: #6a6a6a !default; |
||||
$uni-main-color: #333 !default; |
||||
$uni-secondary-color: #909399 !default; |
||||
$uni-border-3: #e5e5e5; |
||||
|
||||
|
||||
/* #ifndef APP-NVUE */ |
||||
@media screen and (max-width: 500px) { |
||||
.hide-on-phone { |
||||
display: none; |
||||
} |
||||
} |
||||
|
||||
/* #endif */ |
||||
.uni-stat__select { |
||||
display: flex; |
||||
align-items: center; |
||||
// padding: 15px; |
||||
cursor: pointer; |
||||
width: 100%; |
||||
flex: 1; |
||||
box-sizing: border-box; |
||||
} |
||||
|
||||
.uni-stat-box { |
||||
width: 100%; |
||||
flex: 1; |
||||
} |
||||
|
||||
.uni-stat__actived { |
||||
width: 100%; |
||||
flex: 1; |
||||
// outline: 1px solid #2979ff; |
||||
} |
||||
|
||||
.uni-label-text { |
||||
font-size: 14px; |
||||
font-weight: bold; |
||||
color: $uni-base-color; |
||||
margin: auto 0; |
||||
margin-right: 5px; |
||||
} |
||||
|
||||
.uni-select { |
||||
font-size: 14px; |
||||
border: 1px solid $uni-border-3; |
||||
box-sizing: border-box; |
||||
border-radius: 4px; |
||||
padding: 0 5px; |
||||
padding-left: 10px; |
||||
position: relative; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
user-select: none; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
align-items: center; |
||||
border-bottom: solid 1px $uni-border-3; |
||||
width: 100%; |
||||
flex: 1; |
||||
height: 35px; |
||||
|
||||
&--disabled{ |
||||
background-color: #f5f7fa; |
||||
cursor: not-allowed; |
||||
} |
||||
} |
||||
|
||||
.uni-select__label { |
||||
font-size: 16px; |
||||
// line-height: 22px; |
||||
height: 35px; |
||||
padding-right: 10px; |
||||
color: $uni-secondary-color; |
||||
} |
||||
|
||||
.uni-select__input-box { |
||||
height: 35px; |
||||
position: relative; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex: 1; |
||||
flex-direction: row; |
||||
align-items: center; |
||||
} |
||||
|
||||
.uni-select__input { |
||||
flex: 1; |
||||
font-size: 14px; |
||||
height: 22px; |
||||
line-height: 22px; |
||||
} |
||||
|
||||
.uni-select__input-plac { |
||||
font-size: 14px; |
||||
color: $uni-secondary-color; |
||||
} |
||||
|
||||
.uni-select__selector { |
||||
/* #ifndef APP-NVUE */ |
||||
box-sizing: border-box; |
||||
/* #endif */ |
||||
position: absolute; |
||||
top: calc(100% + 12px); |
||||
left: 0; |
||||
width: 100%; |
||||
background-color: #FFFFFF; |
||||
border: 1px solid #EBEEF5; |
||||
border-radius: 6px; |
||||
box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1); |
||||
z-index: 3; |
||||
padding: 4px 0; |
||||
} |
||||
|
||||
.uni-select__selector-scroll { |
||||
/* #ifndef APP-NVUE */ |
||||
max-height: 200px; |
||||
box-sizing: border-box; |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-select__selector-empty, |
||||
.uni-select__selector-item { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
cursor: pointer; |
||||
/* #endif */ |
||||
line-height: 35px; |
||||
font-size: 14px; |
||||
text-align: center; |
||||
/* border-bottom: solid 1px $uni-border-3; */ |
||||
padding: 0px 10px; |
||||
} |
||||
|
||||
.uni-select__selector-item:hover { |
||||
background-color: #f9f9f9; |
||||
} |
||||
|
||||
.uni-select__selector-empty:last-child, |
||||
.uni-select__selector-item:last-child { |
||||
/* #ifndef APP-NVUE */ |
||||
border-bottom: none; |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-select__selector__disabled { |
||||
opacity: 0.4; |
||||
cursor: default; |
||||
} |
||||
|
||||
/* picker 弹出层通用的指示小三角 */ |
||||
.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; |
||||
} |
||||
|
||||
.uni-select__input-text { |
||||
// width: 280px; |
||||
width: 100%; |
||||
color: $uni-main-color; |
||||
white-space: nowrap; |
||||
text-overflow: ellipsis; |
||||
-o-text-overflow: ellipsis; |
||||
overflow: hidden; |
||||
} |
||||
|
||||
.uni-select__input-placeholder { |
||||
color: $uni-base-color; |
||||
font-size: 12px; |
||||
} |
||||
|
||||
.uni-select--mask { |
||||
position: fixed; |
||||
top: 0; |
||||
bottom: 0; |
||||
right: 0; |
||||
left: 0; |
||||
} |
||||
</style> |
@ -0,0 +1,85 @@ |
||||
{ |
||||
"id": "uni-data-select", |
||||
"displayName": "uni-data-select 下拉框选择器", |
||||
"version": "0.1.9", |
||||
"description": "通过数据驱动的下拉框选择器", |
||||
"keywords": [ |
||||
"uni-ui", |
||||
"select", |
||||
"uni-data-select", |
||||
"下拉框", |
||||
"下拉选" |
||||
], |
||||
"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"], |
||||
"encrypt": [], |
||||
"platforms": { |
||||
"cloud": { |
||||
"tcb": "y", |
||||
"aliyun": "y" |
||||
}, |
||||
"client": { |
||||
"App": { |
||||
"app-vue": "u", |
||||
"app-nvue": "n" |
||||
}, |
||||
"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", |
||||
"阿里": "u", |
||||
"百度": "u", |
||||
"字节跳动": "u", |
||||
"QQ": "u", |
||||
"京东": "u" |
||||
}, |
||||
"快应用": { |
||||
"华为": "u", |
||||
"联盟": "u" |
||||
}, |
||||
"Vue": { |
||||
"vue2": "y", |
||||
"vue3": "y" |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,8 @@ |
||||
## DataSelect 下拉框选择器 |
||||
> **组件名:uni-data-select** |
||||
> 代码块: `uDataSelect` |
||||
|
||||
当选项过多时,使用下拉菜单展示并选择内容 |
||||
|
||||
### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-data-select) |
||||
#### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839 |
@ -0,0 +1,10 @@ |
||||
## 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-dateformat](https://uniapp.dcloud.io/component/uniui/uni-dateformat) |
||||
## 0.0.5(2021-07-08) |
||||
- 调整 默认时间不再是当前时间,而是显示'-'字符 |
||||
## 0.0.4(2021-05-12) |
||||
- 新增 组件示例地址 |
||||
## 0.0.3(2021-02-04) |
||||
- 调整为uni_modules目录规范 |
||||
- 修复 iOS 平台日期格式化出错的问题 |
@ -0,0 +1,200 @@ |
||||
// yyyy-MM-dd hh:mm:ss.SSS 所有支持的类型
|
||||
function pad(str, length = 2) { |
||||
str += '' |
||||
while (str.length < length) { |
||||
str = '0' + str |
||||
} |
||||
return str.slice(-length) |
||||
} |
||||
|
||||
const parser = { |
||||
yyyy: (dateObj) => { |
||||
return pad(dateObj.year, 4) |
||||
}, |
||||
yy: (dateObj) => { |
||||
return pad(dateObj.year) |
||||
}, |
||||
MM: (dateObj) => { |
||||
return pad(dateObj.month) |
||||
}, |
||||
M: (dateObj) => { |
||||
return dateObj.month |
||||
}, |
||||
dd: (dateObj) => { |
||||
return pad(dateObj.day) |
||||
}, |
||||
d: (dateObj) => { |
||||
return dateObj.day |
||||
}, |
||||
hh: (dateObj) => { |
||||
return pad(dateObj.hour) |
||||
}, |
||||
h: (dateObj) => { |
||||
return dateObj.hour |
||||
}, |
||||
mm: (dateObj) => { |
||||
return pad(dateObj.minute) |
||||
}, |
||||
m: (dateObj) => { |
||||
return dateObj.minute |
||||
}, |
||||
ss: (dateObj) => { |
||||
return pad(dateObj.second) |
||||
}, |
||||
s: (dateObj) => { |
||||
return dateObj.second |
||||
}, |
||||
SSS: (dateObj) => { |
||||
return pad(dateObj.millisecond, 3) |
||||
}, |
||||
S: (dateObj) => { |
||||
return dateObj.millisecond |
||||
}, |
||||
} |
||||
|
||||
// 这都n年了iOS依然不认识2020-12-12,需要转换为2020/12/12
|
||||
function getDate(time) { |
||||
if (time instanceof Date) { |
||||
return time |
||||
} |
||||
switch (typeof time) { |
||||
case 'string': |
||||
{ |
||||
// 2020-12-12T12:12:12.000Z、2020-12-12T12:12:12.000
|
||||
if (time.indexOf('T') > -1) { |
||||
return new Date(time) |
||||
} |
||||
return new Date(time.replace(/-/g, '/')) |
||||
} |
||||
default: |
||||
return new Date(time) |
||||
} |
||||
} |
||||
|
||||
export function formatDate(date, format = 'yyyy/MM/dd hh:mm:ss') { |
||||
if (!date && date !== 0) { |
||||
return '' |
||||
} |
||||
date = getDate(date) |
||||
const dateObj = { |
||||
year: date.getFullYear(), |
||||
month: date.getMonth() + 1, |
||||
day: date.getDate(), |
||||
hour: date.getHours(), |
||||
minute: date.getMinutes(), |
||||
second: date.getSeconds(), |
||||
millisecond: date.getMilliseconds() |
||||
} |
||||
const tokenRegExp = /yyyy|yy|MM|M|dd|d|hh|h|mm|m|ss|s|SSS|SS|S/ |
||||
let flag = true |
||||
let result = format |
||||
while (flag) { |
||||
flag = false |
||||
result = result.replace(tokenRegExp, function(matched) { |
||||
flag = true |
||||
return parser[matched](dateObj) |
||||
}) |
||||
} |
||||
return result |
||||
} |
||||
|
||||
export function friendlyDate(time, { |
||||
locale = 'zh', |
||||
threshold = [60000, 3600000], |
||||
format = 'yyyy/MM/dd hh:mm:ss' |
||||
}) { |
||||
if (time === '-') { |
||||
return time |
||||
} |
||||
if (!time && time !== 0) { |
||||
return '' |
||||
} |
||||
const localeText = { |
||||
zh: { |
||||
year: '年', |
||||
month: '月', |
||||
day: '天', |
||||
hour: '小时', |
||||
minute: '分钟', |
||||
second: '秒', |
||||
ago: '前', |
||||
later: '后', |
||||
justNow: '刚刚', |
||||
soon: '马上', |
||||
template: '{num}{unit}{suffix}' |
||||
}, |
||||
en: { |
||||
year: 'year', |
||||
month: 'month', |
||||
day: 'day', |
||||
hour: 'hour', |
||||
minute: 'minute', |
||||
second: 'second', |
||||
ago: 'ago', |
||||
later: 'later', |
||||
justNow: 'just now', |
||||
soon: 'soon', |
||||
template: '{num} {unit} {suffix}' |
||||
} |
||||
} |
||||
const text = localeText[locale] || localeText.zh |
||||
let date = getDate(time) |
||||
let ms = date.getTime() - Date.now() |
||||
let absMs = Math.abs(ms) |
||||
if (absMs < threshold[0]) { |
||||
return ms < 0 ? text.justNow : text.soon |
||||
} |
||||
if (absMs >= threshold[1]) { |
||||
return formatDate(date, format) |
||||
} |
||||
let num |
||||
let unit |
||||
let suffix = text.later |
||||
if (ms < 0) { |
||||
suffix = text.ago |
||||
ms = -ms |
||||
} |
||||
const seconds = Math.floor((ms) / 1000) |
||||
const minutes = Math.floor(seconds / 60) |
||||
const hours = Math.floor(minutes / 60) |
||||
const days = Math.floor(hours / 24) |
||||
const months = Math.floor(days / 30) |
||||
const years = Math.floor(months / 12) |
||||
switch (true) { |
||||
case years > 0: |
||||
num = years |
||||
unit = text.year |
||||
break |
||||
case months > 0: |
||||
num = months |
||||
unit = text.month |
||||
break |
||||
case days > 0: |
||||
num = days |
||||
unit = text.day |
||||
break |
||||
case hours > 0: |
||||
num = hours |
||||
unit = text.hour |
||||
break |
||||
case minutes > 0: |
||||
num = minutes |
||||
unit = text.minute |
||||
break |
||||
default: |
||||
num = seconds |
||||
unit = text.second |
||||
break |
||||
} |
||||
|
||||
if (locale === 'en') { |
||||
if (num === 1) { |
||||
num = 'a' |
||||
} else { |
||||
unit += 's' |
||||
} |
||||
} |
||||
|
||||
return text.template.replace(/{\s*num\s*}/g, num + '').replace(/{\s*unit\s*}/g, unit).replace(/{\s*suffix\s*}/g, |
||||
suffix) |
||||
} |
@ -0,0 +1,88 @@ |
||||
<template> |
||||
<text>{{dateShow}}</text> |
||||
</template> |
||||
|
||||
<script> |
||||
import {friendlyDate} from './date-format.js' |
||||
/** |
||||
* Dateformat 日期格式化 |
||||
* @description 日期格式化组件 |
||||
* @tutorial https://ext.dcloud.net.cn/plugin?id=3279 |
||||
* @property {Object|String|Number} date 日期对象/日期字符串/时间戳 |
||||
* @property {String} locale 格式化使用的语言 |
||||
* @value zh 中文 |
||||
* @value en 英文 |
||||
* @property {Array} threshold 应用不同类型格式化的阈值 |
||||
* @property {String} format 输出日期字符串时的格式 |
||||
*/ |
||||
export default { |
||||
name: 'uniDateformat', |
||||
props: { |
||||
date: { |
||||
type: [Object, String, Number], |
||||
default () { |
||||
return '-' |
||||
} |
||||
}, |
||||
locale: { |
||||
type: String, |
||||
default: 'zh', |
||||
}, |
||||
threshold: { |
||||
type: Array, |
||||
default () { |
||||
return [0, 0] |
||||
} |
||||
}, |
||||
format: { |
||||
type: String, |
||||
default: 'yyyy/MM/dd hh:mm:ss' |
||||
}, |
||||
// refreshRate使用不当可能导致性能问题,谨慎使用 |
||||
refreshRate: { |
||||
type: [Number, String], |
||||
default: 0 |
||||
} |
||||
}, |
||||
data() { |
||||
return { |
||||
refreshMark: 0 |
||||
} |
||||
}, |
||||
computed: { |
||||
dateShow() { |
||||
this.refreshMark |
||||
return friendlyDate(this.date, { |
||||
locale: this.locale, |
||||
threshold: this.threshold, |
||||
format: this.format |
||||
}) |
||||
} |
||||
}, |
||||
watch: { |
||||
refreshRate: { |
||||
handler() { |
||||
this.setAutoRefresh() |
||||
}, |
||||
immediate: true |
||||
} |
||||
}, |
||||
methods: { |
||||
refresh() { |
||||
this.refreshMark++ |
||||
}, |
||||
setAutoRefresh() { |
||||
clearInterval(this.refreshInterval) |
||||
if (this.refreshRate) { |
||||
this.refreshInterval = setInterval(() => { |
||||
this.refresh() |
||||
}, parseInt(this.refreshRate)) |
||||
} |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style> |
||||
|
||||
</style> |
@ -0,0 +1,88 @@ |
||||
{ |
||||
"id": "uni-dateformat", |
||||
"displayName": "uni-dateformat 日期格式化", |
||||
"version": "1.0.0", |
||||
"description": "日期格式化组件,可以将日期格式化为1分钟前、刚刚等形式", |
||||
"keywords": [ |
||||
"uni-ui", |
||||
"uniui", |
||||
"日期格式化", |
||||
"时间格式化", |
||||
"格式化时间", |
||||
"" |
||||
], |
||||
"repository": "https://github.com/dcloudio/uni-ui", |
||||
"engines": { |
||||
"HBuilderX": "" |
||||
}, |
||||
"directories": { |
||||
"example": "../../temps/example_temps" |
||||
}, |
||||
"dcloudext": { |
||||
"category": [ |
||||
"前端组件", |
||||
"通用组件" |
||||
], |
||||
"sale": { |
||||
"regular": { |
||||
"price": "0.00" |
||||
}, |
||||
"sourcecode": { |
||||
"price": "0.00" |
||||
} |
||||
}, |
||||
"contact": { |
||||
"qq": "" |
||||
}, |
||||
"declaration": { |
||||
"ads": "无", |
||||
"data": "无", |
||||
"permissions": "无" |
||||
}, |
||||
"npmurl": "https://www.npmjs.com/package/@dcloudio/uni-ui" |
||||
}, |
||||
"uni_modules": { |
||||
"dependencies": ["uni-scss"], |
||||
"encrypt": [], |
||||
"platforms": { |
||||
"cloud": { |
||||
"tcb": "y", |
||||
"aliyun": "y" |
||||
}, |
||||
"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" |
||||
}, |
||||
"快应用": { |
||||
"华为": "y", |
||||
"联盟": "y" |
||||
}, |
||||
"Vue": { |
||||
"vue2": "y", |
||||
"vue3": "y" |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,11 @@ |
||||
|
||||
|
||||
### DateFormat 日期格式化 |
||||
> **组件名:uni-dateformat** |
||||
> 代码块: `uDateformat` |
||||
|
||||
|
||||
日期格式化组件。 |
||||
|
||||
### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-dateformat) |
||||
#### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839 |
@ -0,0 +1,103 @@ |
||||
## 2.2.11(2022-09-19) |
||||
- 修复,支付宝小程序样式错乱,[详情](https://github.com/dcloudio/uni-app/issues/3861) |
||||
## 2.2.10(2022-09-19) |
||||
- 修复,反向选择日期范围,日期显示异常,[详情](https://ask.dcloud.net.cn/question/153401?item_id=212892&rf=false) |
||||
## 2.2.9(2022-09-16) |
||||
- 可以使用 uni-scss 控制主题色 |
||||
## 2.2.8(2022-09-08) |
||||
- 修复 close事件无效的 bug |
||||
## 2.2.7(2022-09-05) |
||||
- 修复 移动端 maskClick 无效的 bug,详见:[https://ask.dcloud.net.cn/question/140824?item_id=209458&rf=false](https://ask.dcloud.net.cn/question/140824?item_id=209458&rf=false) |
||||
## 2.2.6(2022-06-30) |
||||
- 优化 组件样式,调整了组件图标大小、高度、颜色等,与uni-ui风格保持一致 |
||||
## 2.2.5(2022-06-24) |
||||
- 修复 日历顶部年月及底部确认未国际化 bug |
||||
## 2.2.4(2022-03-31) |
||||
- 修复 Vue3 下动态赋值,单选类型未响应的 bug |
||||
## 2.2.3(2022-03-28) |
||||
- 修复 Vue3 下动态赋值未响应的 bug |
||||
## 2.2.2(2021-12-10) |
||||
- 修复 clear-icon 属性在小程序平台不生效的 bug |
||||
## 2.2.1(2021-12-10) |
||||
- 修复 日期范围选在小程序平台,必须多点击一次才能取消选中状态的 bug |
||||
## 2.2.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-datetime-picker](https://uniapp.dcloud.io/component/uniui/uni-datetime-picker) |
||||
## 2.1.5(2021-11-09) |
||||
- 新增 提供组件设计资源,组件样式调整 |
||||
## 2.1.4(2021-09-10) |
||||
- 修复 hide-second 在移动端的 bug |
||||
- 修复 单选赋默认值时,赋值日期未高亮的 bug |
||||
- 修复 赋默认值时,移动端未正确显示时间的 bug |
||||
## 2.1.3(2021-09-09) |
||||
- 新增 hide-second 属性,支持只使用时分,隐藏秒 |
||||
## 2.1.2(2021-09-03) |
||||
- 优化 取消选中时(范围选)直接开始下一次选择, 避免多点一次 |
||||
- 优化 移动端支持清除按钮,同时支持通过 ref 调用组件的 clear 方法 |
||||
- 优化 调整字号大小,美化日历界面 |
||||
- 修复 因国际化导致的 placeholder 失效的 bug |
||||
## 2.1.1(2021-08-24) |
||||
- 新增 支持国际化 |
||||
- 优化 范围选择器在 pc 端过宽的问题 |
||||
## 2.1.0(2021-08-09) |
||||
- 新增 适配 vue3 |
||||
## 2.0.19(2021-08-09) |
||||
- 新增 支持作为 uni-forms 子组件相关功能 |
||||
- 修复 在 uni-forms 中使用时,选择时间报 NAN 错误的 bug |
||||
## 2.0.18(2021-08-05) |
||||
- 修复 type 属性动态赋值无效的 bug |
||||
- 修复 ‘确认’按钮被 tabbar 遮盖 bug |
||||
- 修复 组件未赋值时范围选左、右日历相同的 bug |
||||
## 2.0.17(2021-08-04) |
||||
- 修复 范围选未正确显示当前值的 bug |
||||
- 修复 h5 平台(移动端)报错 'cale' of undefined 的 bug |
||||
## 2.0.16(2021-07-21) |
||||
- 新增 return-type 属性支持返回 date 日期对象 |
||||
## 2.0.15(2021-07-14) |
||||
- 修复 单选日期类型,初始赋值后不在当前日历的 bug |
||||
- 新增 clearIcon 属性,显示框的清空按钮可配置显示隐藏(仅 pc 有效) |
||||
- 优化 移动端移除显示框的清空按钮,无实际用途 |
||||
## 2.0.14(2021-07-14) |
||||
- 修复 组件赋值为空,界面未更新的 bug |
||||
- 修复 start 和 end 不能动态赋值的 bug |
||||
- 修复 范围选类型,用户选择后再次选择右侧日历(结束日期)显示不正确的 bug |
||||
## 2.0.13(2021-07-08) |
||||
- 修复 范围选择不能动态赋值的 bug |
||||
## 2.0.12(2021-07-08) |
||||
- 修复 范围选择的初始时间在一个月内时,造成无法选择的bug |
||||
## 2.0.11(2021-07-08) |
||||
- 优化 弹出层在超出视窗边缘定位不准确的问题 |
||||
## 2.0.10(2021-07-08) |
||||
- 修复 范围起始点样式的背景色与今日样式的字体前景色融合,导致日期字体看不清的 bug |
||||
- 优化 弹出层在超出视窗边缘被遮盖的问题 |
||||
## 2.0.9(2021-07-07) |
||||
- 新增 maskClick 事件 |
||||
- 修复 特殊情况日历 rpx 布局错误的 bug,rpx -> px |
||||
- 修复 范围选择时清空返回值不合理的bug,['', ''] -> [] |
||||
## 2.0.8(2021-07-07) |
||||
- 新增 日期时间显示框支持插槽 |
||||
## 2.0.7(2021-07-01) |
||||
- 优化 添加 uni-icons 依赖 |
||||
## 2.0.6(2021-05-22) |
||||
- 修复 图标在小程序上不显示的 bug |
||||
- 优化 重命名引用组件,避免潜在组件命名冲突 |
||||
## 2.0.5(2021-05-20) |
||||
- 优化 代码目录扁平化 |
||||
## 2.0.4(2021-05-12) |
||||
- 新增 组件示例地址 |
||||
## 2.0.3(2021-05-10) |
||||
- 修复 ios 下不识别 '-' 日期格式的 bug |
||||
- 优化 pc 下弹出层添加边框和阴影 |
||||
## 2.0.2(2021-05-08) |
||||
- 修复 在 admin 中获取弹出层定位错误的bug |
||||
## 2.0.1(2021-05-08) |
||||
- 修复 type 属性向下兼容,默认值从 date 变更为 datetime |
||||
## 2.0.0(2021-04-30) |
||||
- 支持日历形式的日期+时间的范围选择 |
||||
> 注意:此版本不向后兼容,不再支持单独时间选择(type=time)及相关的 hide-second 属性(时间选可使用内置组件 picker) |
||||
## 1.0.6(2021-03-18) |
||||
- 新增 hide-second 属性,时间支持仅选择时、分 |
||||
- 修复 选择跟显示的日期不一样的 bug |
||||
- 修复 chang事件触发2次的 bug |
||||
- 修复 分、秒 end 范围错误的 bug |
||||
- 优化 更好的 nvue 适配 |
@ -0,0 +1,187 @@ |
||||
<template> |
||||
<view class="uni-calendar-item__weeks-box" :class="{ |
||||
'uni-calendar-item--disable':weeks.disable, |
||||
'uni-calendar-item--before-checked-x':weeks.beforeMultiple, |
||||
'uni-calendar-item--multiple': weeks.multiple, |
||||
'uni-calendar-item--after-checked-x':weeks.afterMultiple, |
||||
}" @click="choiceDate(weeks)" @mouseenter="handleMousemove(weeks)"> |
||||
<view class="uni-calendar-item__weeks-box-item" :class="{ |
||||
'uni-calendar-item--checked':calendar.fullDate === weeks.fullDate && (calendar.userChecked || !checkHover), |
||||
'uni-calendar-item--checked-range-text': checkHover, |
||||
'uni-calendar-item--before-checked':weeks.beforeMultiple, |
||||
'uni-calendar-item--multiple': weeks.multiple, |
||||
'uni-calendar-item--after-checked':weeks.afterMultiple, |
||||
'uni-calendar-item--disable':weeks.disable, |
||||
}"> |
||||
<text v-if="selected&&weeks.extraInfo" class="uni-calendar-item__weeks-box-circle"></text> |
||||
<text class="uni-calendar-item__weeks-box-text uni-calendar-item__weeks-box-text-disable uni-calendar-item--checked-text">{{weeks.date}}</text> |
||||
</view> |
||||
<view :class="{'uni-calendar-item--isDay': weeks.isDay}"></view> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
export default { |
||||
props: { |
||||
weeks: { |
||||
type: Object, |
||||
default () { |
||||
return {} |
||||
} |
||||
}, |
||||
calendar: { |
||||
type: Object, |
||||
default: () => { |
||||
return {} |
||||
} |
||||
}, |
||||
selected: { |
||||
type: Array, |
||||
default: () => { |
||||
return [] |
||||
} |
||||
}, |
||||
lunar: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
checkHover: { |
||||
type: Boolean, |
||||
default: false |
||||
} |
||||
}, |
||||
methods: { |
||||
choiceDate(weeks) { |
||||
this.$emit('change', weeks) |
||||
}, |
||||
handleMousemove(weeks) { |
||||
this.$emit('handleMouse', weeks) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss" > |
||||
$uni-primary: #007aff !default; |
||||
|
||||
.uni-calendar-item__weeks-box { |
||||
flex: 1; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: column; |
||||
justify-content: center; |
||||
align-items: center; |
||||
margin: 1px 0; |
||||
position: relative; |
||||
} |
||||
|
||||
.uni-calendar-item__weeks-box-text { |
||||
font-size: 14px; |
||||
// font-family: Lato-Bold, Lato; |
||||
font-weight: bold; |
||||
color: darken($color: $uni-primary, $amount: 40%); |
||||
} |
||||
|
||||
.uni-calendar-item__weeks-lunar-text { |
||||
font-size: 12px; |
||||
color: #333; |
||||
} |
||||
|
||||
.uni-calendar-item__weeks-box-item { |
||||
position: relative; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: column; |
||||
justify-content: center; |
||||
align-items: center; |
||||
width: 40px; |
||||
height: 40px; |
||||
/* #ifdef H5 */ |
||||
cursor: pointer; |
||||
/* #endif */ |
||||
} |
||||
|
||||
|
||||
.uni-calendar-item__weeks-box-circle { |
||||
position: absolute; |
||||
top: 5px; |
||||
right: 5px; |
||||
width: 8px; |
||||
height: 8px; |
||||
border-radius: 8px; |
||||
background-color: #dd524d; |
||||
|
||||
} |
||||
|
||||
.uni-calendar-item__weeks-box .uni-calendar-item--disable { |
||||
// background-color: rgba(249, 249, 249, $uni-opacity-disabled); |
||||
cursor: default; |
||||
} |
||||
|
||||
.uni-calendar-item--disable .uni-calendar-item__weeks-box-text-disable { |
||||
color: #D1D1D1; |
||||
} |
||||
|
||||
.uni-calendar-item--isDay { |
||||
position: absolute; |
||||
top: 10px; |
||||
right: 17%; |
||||
background-color: #dd524d; |
||||
width:6px; |
||||
height: 6px; |
||||
border-radius: 50%; |
||||
} |
||||
|
||||
.uni-calendar-item--extra { |
||||
color: #dd524d; |
||||
opacity: 0.8; |
||||
} |
||||
|
||||
.uni-calendar-item__weeks-box .uni-calendar-item--checked { |
||||
background-color: $uni-primary; |
||||
border-radius: 50%; |
||||
box-sizing: border-box; |
||||
border: 3px solid #fff; |
||||
} |
||||
|
||||
.uni-calendar-item--checked .uni-calendar-item--checked-text { |
||||
color: #fff; |
||||
} |
||||
|
||||
.uni-calendar-item--multiple .uni-calendar-item--checked-range-text { |
||||
color: #333; |
||||
} |
||||
|
||||
.uni-calendar-item--multiple { |
||||
background-color: #F6F7FC; |
||||
// color: #fff; |
||||
} |
||||
|
||||
.uni-calendar-item--multiple .uni-calendar-item--before-checked, |
||||
.uni-calendar-item--multiple .uni-calendar-item--after-checked { |
||||
background-color: $uni-primary; |
||||
border-radius: 50%; |
||||
box-sizing: border-box; |
||||
border: 3px solid #F6F7FC; |
||||
} |
||||
|
||||
.uni-calendar-item--before-checked .uni-calendar-item--checked-text, |
||||
.uni-calendar-item--after-checked .uni-calendar-item--checked-text { |
||||
color: #fff; |
||||
} |
||||
|
||||
.uni-calendar-item--before-checked-x { |
||||
border-top-left-radius: 50px; |
||||
border-bottom-left-radius: 50px; |
||||
box-sizing: border-box; |
||||
background-color: #F6F7FC; |
||||
} |
||||
|
||||
.uni-calendar-item--after-checked-x { |
||||
border-top-right-radius: 50px; |
||||
border-bottom-right-radius: 50px; |
||||
background-color: #F6F7FC; |
||||
} |
||||
</style> |
@ -0,0 +1,924 @@ |
||||
<template> |
||||
<view class="uni-calendar" @mouseleave="leaveCale"> |
||||
<view v-if="!insert&&show" class="uni-calendar__mask" :class="{'uni-calendar--mask-show':aniMaskShow}" |
||||
@click="clean();maskClick()"></view> |
||||
<view v-if="insert || show" class="uni-calendar__content" |
||||
:class="{'uni-calendar--fixed':!insert,'uni-calendar--ani-show':aniMaskShow, 'uni-calendar__content-mobile': aniMaskShow}"> |
||||
<view class="uni-calendar__header" :class="{'uni-calendar__header-mobile' :!insert}"> |
||||
<view v-if="left" class="uni-calendar__header-btn-box" @click.stop="pre"> |
||||
<view class="uni-calendar__header-btn uni-calendar--left"></view> |
||||
</view> |
||||
<picker mode="date" :value="date" fields="month" @change="bindDateChange"> |
||||
<text |
||||
class="uni-calendar__header-text">{{ (nowDate.year||'') + yearText + ( nowDate.month||'') + monthText}}</text> |
||||
</picker> |
||||
<view v-if="right" class="uni-calendar__header-btn-box" @click.stop="next"> |
||||
<view class="uni-calendar__header-btn uni-calendar--right"></view> |
||||
</view> |
||||
<view v-if="!insert" class="dialog-close" @click="clean"> |
||||
<view class="dialog-close-plus" data-id="close"></view> |
||||
<view class="dialog-close-plus dialog-close-rotate" data-id="close"></view> |
||||
</view> |
||||
|
||||
<!-- <text class="uni-calendar__backtoday" @click="backtoday">回到今天</text> --> |
||||
</view> |
||||
<view class="uni-calendar__box"> |
||||
<view v-if="showMonth" class="uni-calendar__box-bg"> |
||||
<text class="uni-calendar__box-bg-text">{{nowDate.month}}</text> |
||||
</view> |
||||
<view class="uni-calendar__weeks" style="padding-bottom: 7px;"> |
||||
<view class="uni-calendar__weeks-day"> |
||||
<text class="uni-calendar__weeks-day-text">{{SUNText}}</text> |
||||
</view> |
||||
<view class="uni-calendar__weeks-day"> |
||||
<text class="uni-calendar__weeks-day-text">{{MONText}}</text> |
||||
</view> |
||||
<view class="uni-calendar__weeks-day"> |
||||
<text class="uni-calendar__weeks-day-text">{{TUEText}}</text> |
||||
</view> |
||||
<view class="uni-calendar__weeks-day"> |
||||
<text class="uni-calendar__weeks-day-text">{{WEDText}}</text> |
||||
</view> |
||||
<view class="uni-calendar__weeks-day"> |
||||
<text class="uni-calendar__weeks-day-text">{{THUText}}</text> |
||||
</view> |
||||
<view class="uni-calendar__weeks-day"> |
||||
<text class="uni-calendar__weeks-day-text">{{FRIText}}</text> |
||||
</view> |
||||
<view class="uni-calendar__weeks-day"> |
||||
<text class="uni-calendar__weeks-day-text">{{SATText}}</text> |
||||
</view> |
||||
</view> |
||||
<view class="uni-calendar__weeks" v-for="(item,weekIndex) in weeks" :key="weekIndex"> |
||||
<view class="uni-calendar__weeks-item" v-for="(weeks,weeksIndex) in item" :key="weeksIndex"> |
||||
<calendar-item class="uni-calendar-item--hook" :weeks="weeks" :calendar="calendar" |
||||
:selected="selected" :lunar="lunar" :checkHover="range" @change="choiceDate" |
||||
@handleMouse="handleMouse"> |
||||
</calendar-item> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
<view v-if="!insert && !range && typeHasTime" class="uni-date-changed uni-calendar--fixed-top" |
||||
style="padding: 0 80px;"> |
||||
<view class="uni-date-changed--time-date">{{tempSingleDate ? tempSingleDate : selectDateText}}</view> |
||||
<time-picker type="time" :start="reactStartTime" :end="reactEndTime" v-model="time" |
||||
:disabled="!tempSingleDate" :border="false" :hide-second="hideSecond" class="time-picker-style"> |
||||
</time-picker> |
||||
</view> |
||||
|
||||
<view v-if="!insert && range && typeHasTime" class="uni-date-changed uni-calendar--fixed-top"> |
||||
<view class="uni-date-changed--time-start"> |
||||
<view class="uni-date-changed--time-date">{{tempRange.before ? tempRange.before : startDateText}} |
||||
</view> |
||||
<time-picker type="time" :start="reactStartTime" v-model="timeRange.startTime" :border="false" |
||||
:hide-second="hideSecond" :disabled="!tempRange.before" class="time-picker-style"> |
||||
</time-picker> |
||||
</view> |
||||
<view style="line-height: 50px;"> |
||||
<uni-icons type="arrowthinright" color="#999"></uni-icons> |
||||
</view> |
||||
<view class="uni-date-changed--time-end"> |
||||
<view class="uni-date-changed--time-date">{{tempRange.after ? tempRange.after : endDateText}}</view> |
||||
<time-picker type="time" :end="reactEndTime" v-model="timeRange.endTime" :border="false" |
||||
:hide-second="hideSecond" :disabled="!tempRange.after" class="time-picker-style"> |
||||
</time-picker> |
||||
</view> |
||||
</view> |
||||
<view v-if="!insert" class="uni-date-changed uni-date-btn--ok"> |
||||
<!-- <view class="uni-calendar__header-btn-box"> |
||||
<text class="uni-calendar__button-text uni-calendar--fixed-width">{{okText}}</text> |
||||
</view> --> |
||||
<view class="uni-datetime-picker--btn" @click="confirm">{{confirmText}}</view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
import Calendar from './util.js'; |
||||
import calendarItem from './calendar-item.vue' |
||||
import timePicker from './time-picker.vue' |
||||
import { |
||||
initVueI18n |
||||
} from '@dcloudio/uni-i18n' |
||||
import messages from './i18n/index.js' |
||||
const { |
||||
t |
||||
} = initVueI18n(messages) |
||||
/** |
||||
* Calendar 日历 |
||||
* @description 日历组件可以查看日期,选择任意范围内的日期,打点操作。常用场景如:酒店日期预订、火车机票选择购买日期、上下班打卡等 |
||||
* @tutorial https://ext.dcloud.net.cn/plugin?id=56 |
||||
* @property {String} date 自定义当前时间,默认为今天 |
||||
* @property {Boolean} lunar 显示农历 |
||||
* @property {String} startDate 日期选择范围-开始日期 |
||||
* @property {String} endDate 日期选择范围-结束日期 |
||||
* @property {Boolean} range 范围选择 |
||||
* @property {Boolean} insert = [true|false] 插入模式,默认为false |
||||
* @value true 弹窗模式 |
||||
* @value false 插入模式 |
||||
* @property {Boolean} clearDate = [true|false] 弹窗模式是否清空上次选择内容 |
||||
* @property {Array} selected 打点,期待格式[{date: '2019-06-27', info: '签到', data: { custom: '自定义信息', name: '自定义消息头',xxx:xxx... }}] |
||||
* @property {Boolean} showMonth 是否选择月份为背景 |
||||
* @event {Function} change 日期改变,`insert :ture` 时生效 |
||||
* @event {Function} confirm 确认选择`insert :false` 时生效 |
||||
* @event {Function} monthSwitch 切换月份时触发 |
||||
* @example <uni-calendar :insert="true":lunar="true" :start-date="'2019-3-2'":end-date="'2019-5-20'"@change="change" /> |
||||
*/ |
||||
export default { |
||||
components: { |
||||
calendarItem, |
||||
timePicker |
||||
}, |
||||
props: { |
||||
date: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
defTime: { |
||||
type: [String, Object], |
||||
default: '' |
||||
}, |
||||
selectableTimes: { |
||||
type: [Object], |
||||
default () { |
||||
return {} |
||||
} |
||||
}, |
||||
selected: { |
||||
type: Array, |
||||
default () { |
||||
return [] |
||||
} |
||||
}, |
||||
lunar: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
startDate: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
endDate: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
range: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
typeHasTime: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
insert: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
showMonth: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
clearDate: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
left: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
right: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
checkHover: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
hideSecond: { |
||||
type: [Boolean], |
||||
default: false |
||||
}, |
||||
pleStatus: { |
||||
type: Object, |
||||
default () { |
||||
return { |
||||
before: '', |
||||
after: '', |
||||
data: [], |
||||
fulldate: '' |
||||
} |
||||
} |
||||
} |
||||
}, |
||||
data() { |
||||
return { |
||||
show: false, |
||||
weeks: [], |
||||
calendar: {}, |
||||
nowDate: '', |
||||
aniMaskShow: false, |
||||
firstEnter: true, |
||||
time: '', |
||||
timeRange: { |
||||
startTime: '', |
||||
endTime: '' |
||||
}, |
||||
tempSingleDate: '', |
||||
tempRange: { |
||||
before: '', |
||||
after: '' |
||||
} |
||||
} |
||||
}, |
||||
watch: { |
||||
date: { |
||||
immediate: true, |
||||
handler(newVal, oldVal) { |
||||
if (!this.range) { |
||||
this.tempSingleDate = newVal |
||||
setTimeout(() => { |
||||
this.init(newVal) |
||||
}, 100) |
||||
} |
||||
} |
||||
}, |
||||
defTime: { |
||||
immediate: true, |
||||
handler(newVal, oldVal) { |
||||
if (!this.range) { |
||||
this.time = newVal |
||||
} else { |
||||
// console.log('-----', newVal); |
||||
this.timeRange.startTime = newVal.start |
||||
this.timeRange.endTime = newVal.end |
||||
} |
||||
} |
||||
}, |
||||
startDate(val) { |
||||
this.cale.resetSatrtDate(val) |
||||
this.cale.setDate(this.nowDate.fullDate) |
||||
this.weeks = this.cale.weeks |
||||
}, |
||||
endDate(val) { |
||||
this.cale.resetEndDate(val) |
||||
this.cale.setDate(this.nowDate.fullDate) |
||||
this.weeks = this.cale.weeks |
||||
}, |
||||
selected(newVal) { |
||||
this.cale.setSelectInfo(this.nowDate.fullDate, newVal) |
||||
this.weeks = this.cale.weeks |
||||
}, |
||||
pleStatus: { |
||||
immediate: true, |
||||
handler(newVal, oldVal) { |
||||
const { |
||||
before, |
||||
after, |
||||
fulldate, |
||||
which |
||||
} = newVal |
||||
this.tempRange.before = before |
||||
this.tempRange.after = after |
||||
setTimeout(() => { |
||||
if (fulldate) { |
||||
this.cale.setHoverMultiple(fulldate) |
||||
if (before && after) { |
||||
this.cale.lastHover = true |
||||
if (this.rangeWithinMonth(after, before)) return |
||||
this.setDate(before) |
||||
} else { |
||||
this.cale.setMultiple(fulldate) |
||||
this.setDate(this.nowDate.fullDate) |
||||
this.calendar.fullDate = '' |
||||
this.cale.lastHover = false |
||||
} |
||||
} else { |
||||
this.cale.setDefaultMultiple(before, after) |
||||
if (which === 'left') { |
||||
this.setDate(before) |
||||
this.weeks = this.cale.weeks |
||||
} else { |
||||
this.setDate(after) |
||||
this.weeks = this.cale.weeks |
||||
} |
||||
this.cale.lastHover = true |
||||
} |
||||
}, 16) |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
reactStartTime() { |
||||
const activeDate = this.range ? this.tempRange.before : this.calendar.fullDate |
||||
const res = activeDate === this.startDate ? this.selectableTimes.start : '' |
||||
return res |
||||
}, |
||||
reactEndTime() { |
||||
const activeDate = this.range ? this.tempRange.after : this.calendar.fullDate |
||||
const res = activeDate === this.endDate ? this.selectableTimes.end : '' |
||||
return res |
||||
}, |
||||
/** |
||||
* for i18n |
||||
*/ |
||||
selectDateText() { |
||||
return t("uni-datetime-picker.selectDate") |
||||
}, |
||||
startDateText() { |
||||
return this.startPlaceholder || t("uni-datetime-picker.startDate") |
||||
}, |
||||
endDateText() { |
||||
return this.endPlaceholder || t("uni-datetime-picker.endDate") |
||||
}, |
||||
okText() { |
||||
return t("uni-datetime-picker.ok") |
||||
}, |
||||
yearText() { |
||||
return t("uni-datetime-picker.year") |
||||
}, |
||||
monthText() { |
||||
return t("uni-datetime-picker.month") |
||||
}, |
||||
MONText() { |
||||
return t("uni-calender.MON") |
||||
}, |
||||
TUEText() { |
||||
return t("uni-calender.TUE") |
||||
}, |
||||
WEDText() { |
||||
return t("uni-calender.WED") |
||||
}, |
||||
THUText() { |
||||
return t("uni-calender.THU") |
||||
}, |
||||
FRIText() { |
||||
return t("uni-calender.FRI") |
||||
}, |
||||
SATText() { |
||||
return t("uni-calender.SAT") |
||||
}, |
||||
SUNText() { |
||||
return t("uni-calender.SUN") |
||||
}, |
||||
confirmText() { |
||||
return t("uni-calender.confirm") |
||||
}, |
||||
}, |
||||
created() { |
||||
// 获取日历方法实例 |
||||
this.cale = new Calendar({ |
||||
// date: new Date(), |
||||
selected: this.selected, |
||||
startDate: this.startDate, |
||||
endDate: this.endDate, |
||||
range: this.range, |
||||
// multipleStatus: this.pleStatus |
||||
}) |
||||
// 选中某一天 |
||||
// this.cale.setDate(this.date) |
||||
this.init(this.date) |
||||
// this.setDay |
||||
}, |
||||
methods: { |
||||
leaveCale() { |
||||
this.firstEnter = true |
||||
}, |
||||
handleMouse(weeks) { |
||||
if (weeks.disable) return |
||||
if (this.cale.lastHover) return |
||||
let { |
||||
before, |
||||
after |
||||
} = this.cale.multipleStatus |
||||
if (!before) return |
||||
this.calendar = weeks |
||||
// 设置范围选 |
||||
this.cale.setHoverMultiple(this.calendar.fullDate) |
||||
this.weeks = this.cale.weeks |
||||
// hover时,进入一个日历,更新另一个 |
||||
if (this.firstEnter) { |
||||
this.$emit('firstEnterCale', this.cale.multipleStatus) |
||||
this.firstEnter = false |
||||
} |
||||
}, |
||||
rangeWithinMonth(A, B) { |
||||
const [yearA, monthA] = A.split('-') |
||||
const [yearB, monthB] = B.split('-') |
||||
return yearA === yearB && monthA === monthB |
||||
}, |
||||
|
||||
// 取消穿透 |
||||
clean() { |
||||
this.close() |
||||
}, |
||||
|
||||
// 蒙版点击事件 |
||||
maskClick() { |
||||
this.$emit('maskClose') |
||||
}, |
||||
|
||||
clearCalender() { |
||||
if (this.range) { |
||||
this.timeRange.startTime = '' |
||||
this.timeRange.endTime = '' |
||||
this.tempRange.before = '' |
||||
this.tempRange.after = '' |
||||
this.cale.multipleStatus.before = '' |
||||
this.cale.multipleStatus.after = '' |
||||
this.cale.multipleStatus.data = [] |
||||
this.cale.lastHover = false |
||||
} else { |
||||
this.time = '' |
||||
this.tempSingleDate = '' |
||||
} |
||||
this.calendar.fullDate = '' |
||||
this.setDate() |
||||
}, |
||||
|
||||
bindDateChange(e) { |
||||
const value = e.detail.value + '-1' |
||||
this.init(value) |
||||
}, |
||||
/** |
||||
* 初始化日期显示 |
||||
* @param {Object} date |
||||
*/ |
||||
init(date) { |
||||
this.cale.setDate(date) |
||||
this.weeks = this.cale.weeks |
||||
this.nowDate = this.calendar = this.cale.getInfo(date) |
||||
}, |
||||
// choiceDate(weeks) { |
||||
// if (weeks.disable) return |
||||
// this.calendar = weeks |
||||
// // 设置多选 |
||||
// this.cale.setMultiple(this.calendar.fullDate, true) |
||||
// this.weeks = this.cale.weeks |
||||
// this.tempSingleDate = this.calendar.fullDate |
||||
// this.tempRange.before = this.cale.multipleStatus.before |
||||
// this.tempRange.after = this.cale.multipleStatus.after |
||||
// this.change() |
||||
// }, |
||||
/** |
||||
* 打开日历弹窗 |
||||
*/ |
||||
open() { |
||||
// 弹窗模式并且清理数据 |
||||
if (this.clearDate && !this.insert) { |
||||
this.cale.cleanMultipleStatus() |
||||
// this.cale.setDate(this.date) |
||||
this.init(this.date) |
||||
} |
||||
this.show = true |
||||
this.$nextTick(() => { |
||||
setTimeout(() => { |
||||
this.aniMaskShow = true |
||||
}, 50) |
||||
}) |
||||
}, |
||||
/** |
||||
* 关闭日历弹窗 |
||||
*/ |
||||
close() { |
||||
this.aniMaskShow = false |
||||
this.$nextTick(() => { |
||||
setTimeout(() => { |
||||
this.show = false |
||||
this.$emit('close') |
||||
}, 300) |
||||
}) |
||||
}, |
||||
/** |
||||
* 确认按钮 |
||||
*/ |
||||
confirm() { |
||||
this.setEmit('confirm') |
||||
this.close() |
||||
}, |
||||
/** |
||||
* 变化触发 |
||||
*/ |
||||
change() { |
||||
if (!this.insert) return |
||||
this.setEmit('change') |
||||
}, |
||||
/** |
||||
* 选择月份触发 |
||||
*/ |
||||
monthSwitch() { |
||||
let { |
||||
year, |
||||
month |
||||
} = this.nowDate |
||||
this.$emit('monthSwitch', { |
||||
year, |
||||
month: Number(month) |
||||
}) |
||||
}, |
||||
/** |
||||
* 派发事件 |
||||
* @param {Object} name |
||||
*/ |
||||
setEmit(name) { |
||||
let { |
||||
year, |
||||
month, |
||||
date, |
||||
fullDate, |
||||
lunar, |
||||
extraInfo |
||||
} = this.calendar |
||||
this.$emit(name, { |
||||
range: this.cale.multipleStatus, |
||||
year, |
||||
month, |
||||
date, |
||||
time: this.time, |
||||
timeRange: this.timeRange, |
||||
fulldate: fullDate, |
||||
lunar, |
||||
extraInfo: extraInfo || {} |
||||
}) |
||||
}, |
||||
/** |
||||
* 选择天触发 |
||||
* @param {Object} weeks |
||||
*/ |
||||
choiceDate(weeks) { |
||||
if (weeks.disable) return |
||||
this.calendar = weeks |
||||
this.calendar.userChecked = true |
||||
// 设置多选 |
||||
this.cale.setMultiple(this.calendar.fullDate, true) |
||||
this.weeks = this.cale.weeks |
||||
this.tempSingleDate = this.calendar.fullDate |
||||
const beforeStatus = this.cale.multipleStatus.before |
||||
const beforeDate = new Date(this.cale.multipleStatus.before).getTime() |
||||
const afterDate = new Date(this.cale.multipleStatus.after).getTime() |
||||
if (beforeDate > afterDate && afterDate) { |
||||
this.tempRange.before = this.cale.multipleStatus.after |
||||
this.tempRange.after = this.cale.multipleStatus.before |
||||
} else { |
||||
this.tempRange.before = this.cale.multipleStatus.before |
||||
this.tempRange.after = this.cale.multipleStatus.after |
||||
} |
||||
this.change() |
||||
}, |
||||
/** |
||||
* 回到今天 |
||||
*/ |
||||
backtoday() { |
||||
let date = this.cale.getDate(new Date()).fullDate |
||||
// this.cale.setDate(date) |
||||
this.init(date) |
||||
this.change() |
||||
}, |
||||
/** |
||||
* 比较时间大小 |
||||
*/ |
||||
dateCompare(startDate, endDate) { |
||||
// 计算截止时间 |
||||
startDate = new Date(startDate.replace('-', '/').replace('-', '/')) |
||||
// 计算详细项的截止时间 |
||||
endDate = new Date(endDate.replace('-', '/').replace('-', '/')) |
||||
if (startDate <= endDate) { |
||||
return true |
||||
} else { |
||||
return false |
||||
} |
||||
}, |
||||
/** |
||||
* 上个月 |
||||
*/ |
||||
pre() { |
||||
const preDate = this.cale.getDate(this.nowDate.fullDate, -1, 'month').fullDate |
||||
this.setDate(preDate) |
||||
this.monthSwitch() |
||||
|
||||
}, |
||||
/** |
||||
* 下个月 |
||||
*/ |
||||
next() { |
||||
const nextDate = this.cale.getDate(this.nowDate.fullDate, +1, 'month').fullDate |
||||
this.setDate(nextDate) |
||||
this.monthSwitch() |
||||
}, |
||||
/** |
||||
* 设置日期 |
||||
* @param {Object} date |
||||
*/ |
||||
setDate(date) { |
||||
this.cale.setDate(date) |
||||
this.weeks = this.cale.weeks |
||||
this.nowDate = this.cale.getInfo(date) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss" > |
||||
$uni-primary: #007aff !default; |
||||
|
||||
.uni-calendar { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: column; |
||||
} |
||||
|
||||
.uni-calendar__mask { |
||||
position: fixed; |
||||
bottom: 0; |
||||
top: 0; |
||||
left: 0; |
||||
right: 0; |
||||
background-color: rgba(0, 0, 0, 0.4); |
||||
transition-property: opacity; |
||||
transition-duration: 0.3s; |
||||
opacity: 0; |
||||
/* #ifndef APP-NVUE */ |
||||
z-index: 99; |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-calendar--mask-show { |
||||
opacity: 1 |
||||
} |
||||
|
||||
.uni-calendar--fixed { |
||||
position: fixed; |
||||
bottom: calc(var(--window-bottom)); |
||||
left: 0; |
||||
right: 0; |
||||
transition-property: transform; |
||||
transition-duration: 0.3s; |
||||
transform: translateY(460px); |
||||
/* #ifndef APP-NVUE */ |
||||
z-index: 99; |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-calendar--ani-show { |
||||
transform: translateY(0); |
||||
} |
||||
|
||||
.uni-calendar__content { |
||||
background-color: #fff; |
||||
} |
||||
|
||||
.uni-calendar__content-mobile { |
||||
border-top-left-radius: 10px; |
||||
border-top-right-radius: 10px; |
||||
box-shadow: 0px 0px 5px 3px rgba(0, 0, 0, 0.1); |
||||
} |
||||
|
||||
.uni-calendar__header { |
||||
position: relative; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
justify-content: center; |
||||
align-items: center; |
||||
height: 50px; |
||||
} |
||||
|
||||
.uni-calendar__header-mobile { |
||||
padding: 10px; |
||||
padding-bottom: 0; |
||||
} |
||||
|
||||
.uni-calendar--fixed-top { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
justify-content: space-between; |
||||
border-top-color: rgba(0, 0, 0, 0.4); |
||||
border-top-style: solid; |
||||
border-top-width: 1px; |
||||
} |
||||
|
||||
.uni-calendar--fixed-width { |
||||
width: 50px; |
||||
} |
||||
|
||||
.uni-calendar__backtoday { |
||||
position: absolute; |
||||
right: 0; |
||||
top: 25rpx; |
||||
padding: 0 5px; |
||||
padding-left: 10px; |
||||
height: 25px; |
||||
line-height: 25px; |
||||
font-size: 12px; |
||||
border-top-left-radius: 25px; |
||||
border-bottom-left-radius: 25px; |
||||
color: #fff; |
||||
background-color: #f1f1f1; |
||||
} |
||||
|
||||
.uni-calendar__header-text { |
||||
text-align: center; |
||||
width: 100px; |
||||
font-size: 15px; |
||||
color: #666; |
||||
} |
||||
|
||||
.uni-calendar__button-text { |
||||
text-align: center; |
||||
width: 100px; |
||||
font-size: 14px; |
||||
color: $uni-primary; |
||||
/* #ifndef APP-NVUE */ |
||||
letter-spacing: 3px; |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-calendar__header-btn-box { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
align-items: center; |
||||
justify-content: center; |
||||
width: 50px; |
||||
height: 50px; |
||||
} |
||||
|
||||
.uni-calendar__header-btn { |
||||
width: 9px; |
||||
height: 9px; |
||||
border-left-color: #808080; |
||||
border-left-style: solid; |
||||
border-left-width: 1px; |
||||
border-top-color: #555555; |
||||
border-top-style: solid; |
||||
border-top-width: 1px; |
||||
} |
||||
|
||||
.uni-calendar--left { |
||||
transform: rotate(-45deg); |
||||
} |
||||
|
||||
.uni-calendar--right { |
||||
transform: rotate(135deg); |
||||
} |
||||
|
||||
|
||||
.uni-calendar__weeks { |
||||
position: relative; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
} |
||||
|
||||
.uni-calendar__weeks-item { |
||||
flex: 1; |
||||
} |
||||
|
||||
.uni-calendar__weeks-day { |
||||
flex: 1; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: column; |
||||
justify-content: center; |
||||
align-items: center; |
||||
height: 40px; |
||||
border-bottom-color: #F5F5F5; |
||||
border-bottom-style: solid; |
||||
border-bottom-width: 1px; |
||||
} |
||||
|
||||
.uni-calendar__weeks-day-text { |
||||
font-size: 12px; |
||||
color: #B2B2B2; |
||||
} |
||||
|
||||
.uni-calendar__box { |
||||
position: relative; |
||||
// padding: 0 10px; |
||||
padding-bottom: 7px; |
||||
} |
||||
|
||||
.uni-calendar__box-bg { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
justify-content: center; |
||||
align-items: center; |
||||
position: absolute; |
||||
top: 0; |
||||
left: 0; |
||||
right: 0; |
||||
bottom: 0; |
||||
} |
||||
|
||||
.uni-calendar__box-bg-text { |
||||
font-size: 200px; |
||||
font-weight: bold; |
||||
color: #999; |
||||
opacity: 0.1; |
||||
text-align: center; |
||||
/* #ifndef APP-NVUE */ |
||||
line-height: 1; |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-date-changed { |
||||
padding: 0 10px; |
||||
// line-height: 50px; |
||||
text-align: center; |
||||
color: #333; |
||||
border-top-color: #DCDCDC; |
||||
; |
||||
border-top-style: solid; |
||||
border-top-width: 1px; |
||||
flex: 1; |
||||
} |
||||
|
||||
.uni-date-btn--ok { |
||||
padding: 20px 15px; |
||||
} |
||||
|
||||
.uni-date-changed--time-start { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
align-items: center; |
||||
} |
||||
|
||||
.uni-date-changed--time-end { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
align-items: center; |
||||
} |
||||
|
||||
.uni-date-changed--time-date { |
||||
color: #999; |
||||
line-height: 50px; |
||||
margin-right: 5px; |
||||
// opacity: 0.6; |
||||
} |
||||
|
||||
.time-picker-style { |
||||
// width: 62px; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
justify-content: center; |
||||
align-items: center |
||||
} |
||||
|
||||
.mr-10 { |
||||
margin-right: 10px; |
||||
} |
||||
|
||||
.dialog-close { |
||||
position: absolute; |
||||
top: 0; |
||||
right: 0; |
||||
bottom: 0; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
align-items: center; |
||||
padding: 0 25px; |
||||
margin-top: 10px; |
||||
} |
||||
|
||||
.dialog-close-plus { |
||||
width: 16px; |
||||
height: 2px; |
||||
background-color: #737987; |
||||
border-radius: 2px; |
||||
transform: rotate(45deg); |
||||
} |
||||
|
||||
.dialog-close-rotate { |
||||
position: absolute; |
||||
transform: rotate(-45deg); |
||||
} |
||||
|
||||
.uni-datetime-picker--btn { |
||||
border-radius: 100px; |
||||
height: 40px; |
||||
line-height: 40px; |
||||
background-color: $uni-primary; |
||||
color: #fff; |
||||
font-size: 16px; |
||||
letter-spacing: 2px; |
||||
} |
||||
|
||||
/* #ifndef APP-NVUE */ |
||||
.uni-datetime-picker--btn:active { |
||||
opacity: 0.7; |
||||
} |
||||
/* #endif */ |
||||
</style> |
@ -0,0 +1,22 @@ |
||||
{ |
||||
"uni-datetime-picker.selectDate": "select date", |
||||
"uni-datetime-picker.selectTime": "select time", |
||||
"uni-datetime-picker.selectDateTime": "select datetime", |
||||
"uni-datetime-picker.startDate": "start date", |
||||
"uni-datetime-picker.endDate": "end date", |
||||
"uni-datetime-picker.startTime": "start time", |
||||
"uni-datetime-picker.endTime": "end time", |
||||
"uni-datetime-picker.ok": "ok", |
||||
"uni-datetime-picker.clear": "clear", |
||||
"uni-datetime-picker.cancel": "cancel", |
||||
"uni-datetime-picker.year": "-", |
||||
"uni-datetime-picker.month": "", |
||||
"uni-calender.MON": "MON", |
||||
"uni-calender.TUE": "TUE", |
||||
"uni-calender.WED": "WED", |
||||
"uni-calender.THU": "THU", |
||||
"uni-calender.FRI": "FRI", |
||||
"uni-calender.SAT": "SAT", |
||||
"uni-calender.SUN": "SUN", |
||||
"uni-calender.confirm": "confirm" |
||||
} |
@ -0,0 +1,8 @@ |
||||
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 |
||||
} |
@ -0,0 +1,22 @@ |
||||
{ |
||||
"uni-datetime-picker.selectDate": "选择日期", |
||||
"uni-datetime-picker.selectTime": "选择时间", |
||||
"uni-datetime-picker.selectDateTime": "选择日期时间", |
||||
"uni-datetime-picker.startDate": "开始日期", |
||||
"uni-datetime-picker.endDate": "结束日期", |
||||
"uni-datetime-picker.startTime": "开始时间", |
||||
"uni-datetime-picker.endTime": "结束时间", |
||||
"uni-datetime-picker.ok": "确定", |
||||
"uni-datetime-picker.clear": "清除", |
||||
"uni-datetime-picker.cancel": "取消", |
||||
"uni-datetime-picker.year": "年", |
||||
"uni-datetime-picker.month": "月", |
||||
"uni-calender.SUN": "日", |
||||
"uni-calender.MON": "一", |
||||
"uni-calender.TUE": "二", |
||||
"uni-calender.WED": "三", |
||||
"uni-calender.THU": "四", |
||||
"uni-calender.FRI": "五", |
||||
"uni-calender.SAT": "六", |
||||
"uni-calender.confirm": "确认" |
||||
} |
@ -0,0 +1,22 @@ |
||||
{ |
||||
"uni-datetime-picker.selectDate": "選擇日期", |
||||
"uni-datetime-picker.selectTime": "選擇時間", |
||||
"uni-datetime-picker.selectDateTime": "選擇日期時間", |
||||
"uni-datetime-picker.startDate": "開始日期", |
||||
"uni-datetime-picker.endDate": "結束日期", |
||||
"uni-datetime-picker.startTime": "開始时间", |
||||
"uni-datetime-picker.endTime": "結束时间", |
||||
"uni-datetime-picker.ok": "確定", |
||||
"uni-datetime-picker.clear": "清除", |
||||
"uni-datetime-picker.cancel": "取消", |
||||
"uni-datetime-picker.year": "年", |
||||
"uni-datetime-picker.month": "月", |
||||
"uni-calender.SUN": "日", |
||||
"uni-calender.MON": "一", |
||||
"uni-calender.TUE": "二", |
||||
"uni-calender.WED": "三", |
||||
"uni-calender.THU": "四", |
||||
"uni-calender.FRI": "五", |
||||
"uni-calender.SAT": "六", |
||||
"uni-calender.confirm": "確認" |
||||
} |
@ -0,0 +1,45 @@ |
||||
// #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
|
@ -0,0 +1,930 @@ |
||||
<template> |
||||
<view class="uni-datetime-picker"> |
||||
<view @click="initTimePicker"> |
||||
<slot> |
||||
<view class="uni-datetime-picker-timebox-pointer" |
||||
:class="{'uni-datetime-picker-disabled': disabled, 'uni-datetime-picker-timebox': border}"> |
||||
<text class="uni-datetime-picker-text">{{time}}</text> |
||||
<view v-if="!time" class="uni-datetime-picker-time"> |
||||
<text class="uni-datetime-picker-text">{{selectTimeText}}</text> |
||||
</view> |
||||
</view> |
||||
</slot> |
||||
</view> |
||||
<view v-if="visible" id="mask" class="uni-datetime-picker-mask" @click="tiggerTimePicker"></view> |
||||
<view v-if="visible" class="uni-datetime-picker-popup" :class="[dateShow && timeShow ? '' : 'fix-nvue-height']" |
||||
:style="fixNvueBug"> |
||||
<view class="uni-title"> |
||||
<text class="uni-datetime-picker-text">{{selectTimeText}}</text> |
||||
</view> |
||||
<view v-if="dateShow" class="uni-datetime-picker__container-box"> |
||||
<picker-view class="uni-datetime-picker-view" :indicator-style="indicatorStyle" :value="ymd" |
||||
@change="bindDateChange"> |
||||
<picker-view-column> |
||||
<view class="uni-datetime-picker-item" v-for="(item,index) in years" :key="index"> |
||||
<text class="uni-datetime-picker-item">{{lessThanTen(item)}}</text> |
||||
</view> |
||||
</picker-view-column> |
||||
<picker-view-column> |
||||
<view class="uni-datetime-picker-item" v-for="(item,index) in months" :key="index"> |
||||
<text class="uni-datetime-picker-item">{{lessThanTen(item)}}</text> |
||||
</view> |
||||
</picker-view-column> |
||||
<picker-view-column> |
||||
<view class="uni-datetime-picker-item" v-for="(item,index) in days" :key="index"> |
||||
<text class="uni-datetime-picker-item">{{lessThanTen(item)}}</text> |
||||
</view> |
||||
</picker-view-column> |
||||
</picker-view> |
||||
<!-- 兼容 nvue 不支持伪类 --> |
||||
<text class="uni-datetime-picker-sign sign-left">-</text> |
||||
<text class="uni-datetime-picker-sign sign-right">-</text> |
||||
</view> |
||||
<view v-if="timeShow" class="uni-datetime-picker__container-box"> |
||||
<picker-view class="uni-datetime-picker-view" :class="[hideSecond ? 'time-hide-second' : '']" |
||||
:indicator-style="indicatorStyle" :value="hms" @change="bindTimeChange"> |
||||
<picker-view-column> |
||||
<view class="uni-datetime-picker-item" v-for="(item,index) in hours" :key="index"> |
||||
<text class="uni-datetime-picker-item">{{lessThanTen(item)}}</text> |
||||
</view> |
||||
</picker-view-column> |
||||
<picker-view-column> |
||||
<view class="uni-datetime-picker-item" v-for="(item,index) in minutes" :key="index"> |
||||
<text class="uni-datetime-picker-item">{{lessThanTen(item)}}</text> |
||||
</view> |
||||
</picker-view-column> |
||||
<picker-view-column v-if="!hideSecond"> |
||||
<view class="uni-datetime-picker-item" v-for="(item,index) in seconds" :key="index"> |
||||
<text class="uni-datetime-picker-item">{{lessThanTen(item)}}</text> |
||||
</view> |
||||
</picker-view-column> |
||||
</picker-view> |
||||
<!-- 兼容 nvue 不支持伪类 --> |
||||
<text class="uni-datetime-picker-sign" :class="[hideSecond ? 'sign-center' : 'sign-left']">:</text> |
||||
<text v-if="!hideSecond" class="uni-datetime-picker-sign sign-right">:</text> |
||||
</view> |
||||
<view class="uni-datetime-picker-btn"> |
||||
<view @click="clearTime"> |
||||
<text class="uni-datetime-picker-btn-text">{{clearText}}</text> |
||||
</view> |
||||
<view class="uni-datetime-picker-btn-group"> |
||||
<view class="uni-datetime-picker-cancel" @click="tiggerTimePicker"> |
||||
<text class="uni-datetime-picker-btn-text">{{cancelText}}</text> |
||||
</view> |
||||
<view @click="setTime"> |
||||
<text class="uni-datetime-picker-btn-text">{{okText}}</text> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
</view> |
||||
<!-- #ifdef H5 --> |
||||
<!-- <keypress v-if="visible" @esc="tiggerTimePicker" @enter="setTime" /> --> |
||||
<!-- #endif --> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
// #ifdef H5 |
||||
import keypress from './keypress' |
||||
// #endif |
||||
import { |
||||
initVueI18n |
||||
} from '@dcloudio/uni-i18n' |
||||
import messages from './i18n/index.js' |
||||
const { t } = initVueI18n(messages) |
||||
|
||||
/** |
||||
* DatetimePicker 时间选择器 |
||||
* @description 可以同时选择日期和时间的选择器 |
||||
* @tutorial https://ext.dcloud.net.cn/plugin?id=xxx |
||||
* @property {String} type = [datetime | date | time] 显示模式 |
||||
* @property {Boolean} multiple = [true|false] 是否多选 |
||||
* @property {String|Number} value 默认值 |
||||
* @property {String|Number} start 起始日期或时间 |
||||
* @property {String|Number} end 起始日期或时间 |
||||
* @property {String} return-type = [timestamp | string] |
||||
* @event {Function} change 选中发生变化触发 |
||||
*/ |
||||
|
||||
export default { |
||||
name: 'UniDatetimePicker', |
||||
components: { |
||||
// #ifdef H5 |
||||
keypress |
||||
// #endif |
||||
}, |
||||
data() { |
||||
return { |
||||
indicatorStyle: `height: 50px;`, |
||||
visible: false, |
||||
fixNvueBug: {}, |
||||
dateShow: true, |
||||
timeShow: true, |
||||
title: '日期和时间', |
||||
// 输入框当前时间 |
||||
time: '', |
||||
// 当前的年月日时分秒 |
||||
year: 1920, |
||||
month: 0, |
||||
day: 0, |
||||
hour: 0, |
||||
minute: 0, |
||||
second: 0, |
||||
// 起始时间 |
||||
startYear: 1920, |
||||
startMonth: 1, |
||||
startDay: 1, |
||||
startHour: 0, |
||||
startMinute: 0, |
||||
startSecond: 0, |
||||
// 结束时间 |
||||
endYear: 2120, |
||||
endMonth: 12, |
||||
endDay: 31, |
||||
endHour: 23, |
||||
endMinute: 59, |
||||
endSecond: 59, |
||||
} |
||||
}, |
||||
props: { |
||||
type: { |
||||
type: String, |
||||
default: 'datetime' |
||||
}, |
||||
value: { |
||||
type: [String, Number], |
||||
default: '' |
||||
}, |
||||
modelValue: { |
||||
type: [String, Number], |
||||
default: '' |
||||
}, |
||||
start: { |
||||
type: [Number, String], |
||||
default: '' |
||||
}, |
||||
end: { |
||||
type: [Number, String], |
||||
default: '' |
||||
}, |
||||
returnType: { |
||||
type: String, |
||||
default: 'string' |
||||
}, |
||||
disabled: { |
||||
type: [Boolean, String], |
||||
default: false |
||||
}, |
||||
border: { |
||||
type: [Boolean, String], |
||||
default: true |
||||
}, |
||||
hideSecond: { |
||||
type: [Boolean, String], |
||||
default: false |
||||
} |
||||
}, |
||||
watch: { |
||||
value: { |
||||
handler(newVal, oldVal) { |
||||
if (newVal) { |
||||
this.parseValue(this.fixIosDateFormat(newVal)) //兼容 iOS、safari 日期格式 |
||||
this.initTime(false) |
||||
} else { |
||||
this.time = '' |
||||
this.parseValue(Date.now()) |
||||
} |
||||
}, |
||||
immediate: true |
||||
}, |
||||
type: { |
||||
handler(newValue) { |
||||
if (newValue === 'date') { |
||||
this.dateShow = true |
||||
this.timeShow = false |
||||
this.title = '日期' |
||||
} else if (newValue === 'time') { |
||||
this.dateShow = false |
||||
this.timeShow = true |
||||
this.title = '时间' |
||||
} else { |
||||
this.dateShow = true |
||||
this.timeShow = true |
||||
this.title = '日期和时间' |
||||
} |
||||
}, |
||||
immediate: true |
||||
}, |
||||
start: { |
||||
handler(newVal) { |
||||
this.parseDatetimeRange(this.fixIosDateFormat(newVal), 'start') //兼容 iOS、safari 日期格式 |
||||
}, |
||||
immediate: true |
||||
}, |
||||
end: { |
||||
handler(newVal) { |
||||
this.parseDatetimeRange(this.fixIosDateFormat(newVal), 'end') //兼容 iOS、safari 日期格式 |
||||
}, |
||||
immediate: true |
||||
}, |
||||
|
||||
// 月、日、时、分、秒可选范围变化后,检查当前值是否在范围内,不在则当前值重置为可选范围第一项 |
||||
months(newVal) { |
||||
this.checkValue('month', this.month, newVal) |
||||
}, |
||||
days(newVal) { |
||||
this.checkValue('day', this.day, newVal) |
||||
}, |
||||
hours(newVal) { |
||||
this.checkValue('hour', this.hour, newVal) |
||||
}, |
||||
minutes(newVal) { |
||||
this.checkValue('minute', this.minute, newVal) |
||||
}, |
||||
seconds(newVal) { |
||||
this.checkValue('second', this.second, newVal) |
||||
} |
||||
}, |
||||
computed: { |
||||
// 当前年、月、日、时、分、秒选择范围 |
||||
years() { |
||||
return this.getCurrentRange('year') |
||||
}, |
||||
|
||||
months() { |
||||
return this.getCurrentRange('month') |
||||
}, |
||||
|
||||
days() { |
||||
return this.getCurrentRange('day') |
||||
}, |
||||
|
||||
hours() { |
||||
return this.getCurrentRange('hour') |
||||
}, |
||||
|
||||
minutes() { |
||||
return this.getCurrentRange('minute') |
||||
}, |
||||
|
||||
seconds() { |
||||
return this.getCurrentRange('second') |
||||
}, |
||||
|
||||
// picker 当前值数组 |
||||
ymd() { |
||||
return [this.year - this.minYear, this.month - this.minMonth, this.day - this.minDay] |
||||
}, |
||||
hms() { |
||||
return [this.hour - this.minHour, this.minute - this.minMinute, this.second - this.minSecond] |
||||
}, |
||||
|
||||
// 当前 date 是 start |
||||
currentDateIsStart() { |
||||
return this.year === this.startYear && this.month === this.startMonth && this.day === this.startDay |
||||
}, |
||||
|
||||
// 当前 date 是 end |
||||
currentDateIsEnd() { |
||||
return this.year === this.endYear && this.month === this.endMonth && this.day === this.endDay |
||||
}, |
||||
|
||||
// 当前年、月、日、时、分、秒的最小值和最大值 |
||||
minYear() { |
||||
return this.startYear |
||||
}, |
||||
maxYear() { |
||||
return this.endYear |
||||
}, |
||||
minMonth() { |
||||
if (this.year === this.startYear) { |
||||
return this.startMonth |
||||
} else { |
||||
return 1 |
||||
} |
||||
}, |
||||
maxMonth() { |
||||
if (this.year === this.endYear) { |
||||
return this.endMonth |
||||
} else { |
||||
return 12 |
||||
} |
||||
}, |
||||
minDay() { |
||||
if (this.year === this.startYear && this.month === this.startMonth) { |
||||
return this.startDay |
||||
} else { |
||||
return 1 |
||||
} |
||||
}, |
||||
maxDay() { |
||||
if (this.year === this.endYear && this.month === this.endMonth) { |
||||
return this.endDay |
||||
} else { |
||||
return this.daysInMonth(this.year, this.month) |
||||
} |
||||
}, |
||||
minHour() { |
||||
if (this.type === 'datetime') { |
||||
if (this.currentDateIsStart) { |
||||
return this.startHour |
||||
} else { |
||||
return 0 |
||||
} |
||||
} |
||||
if (this.type === 'time') { |
||||
return this.startHour |
||||
} |
||||
}, |
||||
maxHour() { |
||||
if (this.type === 'datetime') { |
||||
if (this.currentDateIsEnd) { |
||||
return this.endHour |
||||
} else { |
||||
return 23 |
||||
} |
||||
} |
||||
if (this.type === 'time') { |
||||
return this.endHour |
||||
} |
||||
}, |
||||
minMinute() { |
||||
if (this.type === 'datetime') { |
||||
if (this.currentDateIsStart && this.hour === this.startHour) { |
||||
return this.startMinute |
||||
} else { |
||||
return 0 |
||||
} |
||||
} |
||||
if (this.type === 'time') { |
||||
if (this.hour === this.startHour) { |
||||
return this.startMinute |
||||
} else { |
||||
return 0 |
||||
} |
||||
} |
||||
}, |
||||
maxMinute() { |
||||
if (this.type === 'datetime') { |
||||
if (this.currentDateIsEnd && this.hour === this.endHour) { |
||||
return this.endMinute |
||||
} else { |
||||
return 59 |
||||
} |
||||
} |
||||
if (this.type === 'time') { |
||||
if (this.hour === this.endHour) { |
||||
return this.endMinute |
||||
} else { |
||||
return 59 |
||||
} |
||||
} |
||||
}, |
||||
minSecond() { |
||||
if (this.type === 'datetime') { |
||||
if (this.currentDateIsStart && this.hour === this.startHour && this.minute === this.startMinute) { |
||||
return this.startSecond |
||||
} else { |
||||
return 0 |
||||
} |
||||
} |
||||
if (this.type === 'time') { |
||||
if (this.hour === this.startHour && this.minute === this.startMinute) { |
||||
return this.startSecond |
||||
} else { |
||||
return 0 |
||||
} |
||||
} |
||||
}, |
||||
maxSecond() { |
||||
if (this.type === 'datetime') { |
||||
if (this.currentDateIsEnd && this.hour === this.endHour && this.minute === this.endMinute) { |
||||
return this.endSecond |
||||
} else { |
||||
return 59 |
||||
} |
||||
} |
||||
if (this.type === 'time') { |
||||
if (this.hour === this.endHour && this.minute === this.endMinute) { |
||||
return this.endSecond |
||||
} else { |
||||
return 59 |
||||
} |
||||
} |
||||
}, |
||||
|
||||
/** |
||||
* for i18n |
||||
*/ |
||||
selectTimeText() { |
||||
return t("uni-datetime-picker.selectTime") |
||||
}, |
||||
okText() { |
||||
return t("uni-datetime-picker.ok") |
||||
}, |
||||
clearText() { |
||||
return t("uni-datetime-picker.clear") |
||||
}, |
||||
cancelText() { |
||||
return t("uni-datetime-picker.cancel") |
||||
} |
||||
}, |
||||
|
||||
mounted() { |
||||
// #ifdef APP-NVUE |
||||
const res = uni.getSystemInfoSync(); |
||||
this.fixNvueBug = { |
||||
top: res.windowHeight / 2, |
||||
left: res.windowWidth / 2 |
||||
} |
||||
// #endif |
||||
}, |
||||
|
||||
methods: { |
||||
/** |
||||
* @param {Object} item |
||||
* 小于 10 在前面加个 0 |
||||
*/ |
||||
|
||||
lessThanTen(item) { |
||||
return item < 10 ? '0' + item : item |
||||
}, |
||||
|
||||
/** |
||||
* 解析时分秒字符串,例如:00:00:00 |
||||
* @param {String} timeString |
||||
*/ |
||||
parseTimeType(timeString) { |
||||
if (timeString) { |
||||
let timeArr = timeString.split(':') |
||||
this.hour = Number(timeArr[0]) |
||||
this.minute = Number(timeArr[1]) |
||||
this.second = Number(timeArr[2]) |
||||
} |
||||
}, |
||||
|
||||
/** |
||||
* 解析选择器初始值,类型可以是字符串、时间戳,例如:2000-10-02、'08:30:00'、 1610695109000 |
||||
* @param {String | Number} datetime |
||||
*/ |
||||
initPickerValue(datetime) { |
||||
let defaultValue = null |
||||
if (datetime) { |
||||
defaultValue = this.compareValueWithStartAndEnd(datetime, this.start, this.end) |
||||
} else { |
||||
defaultValue = Date.now() |
||||
defaultValue = this.compareValueWithStartAndEnd(defaultValue, this.start, this.end) |
||||
} |
||||
this.parseValue(defaultValue) |
||||
}, |
||||
|
||||
/** |
||||
* 初始值规则: |
||||
* - 用户设置初始值 value |
||||
* - 设置了起始时间 start、终止时间 end,并 start < value < end,初始值为 value, 否则初始值为 start |
||||
* - 只设置了起始时间 start,并 start < value,初始值为 value,否则初始值为 start |
||||
* - 只设置了终止时间 end,并 value < end,初始值为 value,否则初始值为 end |
||||
* - 无起始终止时间,则初始值为 value |
||||
* - 无初始值 value,则初始值为当前本地时间 Date.now() |
||||
* @param {Object} value |
||||
* @param {Object} dateBase |
||||
*/ |
||||
compareValueWithStartAndEnd(value, start, end) { |
||||
let winner = null |
||||
value = this.superTimeStamp(value) |
||||
start = this.superTimeStamp(start) |
||||
end = this.superTimeStamp(end) |
||||
|
||||
if (start && end) { |
||||
if (value < start) { |
||||
winner = new Date(start) |
||||
} else if (value > end) { |
||||
winner = new Date(end) |
||||
} else { |
||||
winner = new Date(value) |
||||
} |
||||
} else if (start && !end) { |
||||
winner = start <= value ? new Date(value) : new Date(start) |
||||
} else if (!start && end) { |
||||
winner = value <= end ? new Date(value) : new Date(end) |
||||
} else { |
||||
winner = new Date(value) |
||||
} |
||||
|
||||
return winner |
||||
}, |
||||
|
||||
/** |
||||
* 转换为可比较的时间戳,接受日期、时分秒、时间戳 |
||||
* @param {Object} value |
||||
*/ |
||||
superTimeStamp(value) { |
||||
let dateBase = '' |
||||
if (this.type === 'time' && value && typeof value === 'string') { |
||||
const now = new Date() |
||||
const year = now.getFullYear() |
||||
const month = now.getMonth() + 1 |
||||
const day = now.getDate() |
||||
dateBase = year + '/' + month + '/' + day + ' ' |
||||
} |
||||
if (Number(value) && typeof value !== NaN) { |
||||
value = parseInt(value) |
||||
dateBase = 0 |
||||
} |
||||
return this.createTimeStamp(dateBase + value) |
||||
}, |
||||
|
||||
/** |
||||
* 解析默认值 value,字符串、时间戳 |
||||
* @param {Object} defaultTime |
||||
*/ |
||||
parseValue(value) { |
||||
if (!value) { |
||||
return |
||||
} |
||||
if (this.type === 'time' && typeof value === "string") { |
||||
this.parseTimeType(value) |
||||
} else { |
||||
let defaultDate = null |
||||
defaultDate = new Date(value) |
||||
if (this.type !== 'time') { |
||||
this.year = defaultDate.getFullYear() |
||||
this.month = defaultDate.getMonth() + 1 |
||||
this.day = defaultDate.getDate() |
||||
} |
||||
if (this.type !== 'date') { |
||||
this.hour = defaultDate.getHours() |
||||
this.minute = defaultDate.getMinutes() |
||||
this.second = defaultDate.getSeconds() |
||||
} |
||||
} |
||||
if (this.hideSecond) { |
||||
this.second = 0 |
||||
} |
||||
}, |
||||
|
||||
/** |
||||
* 解析可选择时间范围 start、end,年月日字符串、时间戳 |
||||
* @param {Object} defaultTime |
||||
*/ |
||||
parseDatetimeRange(point, pointType) { |
||||
// 时间为空,则重置为初始值 |
||||
if (!point) { |
||||
if (pointType === 'start') { |
||||
this.startYear = 1920 |
||||
this.startMonth = 1 |
||||
this.startDay = 1 |
||||
this.startHour = 0 |
||||
this.startMinute = 0 |
||||
this.startSecond = 0 |
||||
} |
||||
if (pointType === 'end') { |
||||
this.endYear = 2120 |
||||
this.endMonth = 12 |
||||
this.endDay = 31 |
||||
this.endHour = 23 |
||||
this.endMinute = 59 |
||||
this.endSecond = 59 |
||||
} |
||||
return |
||||
} |
||||
if (this.type === 'time') { |
||||
const pointArr = point.split(':') |
||||
this[pointType + 'Hour'] = Number(pointArr[0]) |
||||
this[pointType + 'Minute'] = Number(pointArr[1]) |
||||
this[pointType + 'Second'] = Number(pointArr[2]) |
||||
} else { |
||||
if (!point) { |
||||
pointType === 'start' ? this.startYear = this.year - 60 : this.endYear = this.year + 60 |
||||
return |
||||
} |
||||
if (Number(point) && Number(point) !== NaN) { |
||||
point = parseInt(point) |
||||
} |
||||
// datetime 的 end 没有时分秒, 则不限制 |
||||
const hasTime = /[0-9]:[0-9]/ |
||||
if (this.type === 'datetime' && pointType === 'end' && typeof point === 'string' && !hasTime.test( |
||||
point)) { |
||||
point = point + ' 23:59:59' |
||||
} |
||||
const pointDate = new Date(point) |
||||
this[pointType + 'Year'] = pointDate.getFullYear() |
||||
this[pointType + 'Month'] = pointDate.getMonth() + 1 |
||||
this[pointType + 'Day'] = pointDate.getDate() |
||||
if (this.type === 'datetime') { |
||||
this[pointType + 'Hour'] = pointDate.getHours() |
||||
this[pointType + 'Minute'] = pointDate.getMinutes() |
||||
this[pointType + 'Second'] = pointDate.getSeconds() |
||||
} |
||||
} |
||||
}, |
||||
|
||||
// 获取 年、月、日、时、分、秒 当前可选范围 |
||||
getCurrentRange(value) { |
||||
const range = [] |
||||
for (let i = this['min' + this.capitalize(value)]; i <= this['max' + this.capitalize(value)]; i++) { |
||||
range.push(i) |
||||
} |
||||
return range |
||||
}, |
||||
|
||||
// 字符串首字母大写 |
||||
capitalize(str) { |
||||
return str.charAt(0).toUpperCase() + str.slice(1) |
||||
}, |
||||
|
||||
// 检查当前值是否在范围内,不在则当前值重置为可选范围第一项 |
||||
checkValue(name, value, values) { |
||||
if (values.indexOf(value) === -1) { |
||||
this[name] = values[0] |
||||
} |
||||
}, |
||||
|
||||
// 每个月的实际天数 |
||||
daysInMonth(year, month) { // Use 1 for January, 2 for February, etc. |
||||
return new Date(year, month, 0).getDate(); |
||||
}, |
||||
|
||||
//兼容 iOS、safari 日期格式 |
||||
fixIosDateFormat(value) { |
||||
if (typeof value === 'string') { |
||||
value = value.replace(/-/g, '/') |
||||
} |
||||
return value |
||||
}, |
||||
|
||||
/** |
||||
* 生成时间戳 |
||||
* @param {Object} time |
||||
*/ |
||||
createTimeStamp(time) { |
||||
if (!time) return |
||||
if (typeof time === "number") { |
||||
return time |
||||
} else { |
||||
time = time.replace(/-/g, '/') |
||||
if (this.type === 'date') { |
||||
time = time + ' ' + '00:00:00' |
||||
} |
||||
return Date.parse(time) |
||||
} |
||||
}, |
||||
|
||||
/** |
||||
* 生成日期或时间的字符串 |
||||
*/ |
||||
createDomSting() { |
||||
const yymmdd = this.year + |
||||
'-' + |
||||
this.lessThanTen(this.month) + |
||||
'-' + |
||||
this.lessThanTen(this.day) |
||||
|
||||
let hhmmss = this.lessThanTen(this.hour) + |
||||
':' + |
||||
this.lessThanTen(this.minute) |
||||
|
||||
if (!this.hideSecond) { |
||||
hhmmss = hhmmss + ':' + this.lessThanTen(this.second) |
||||
} |
||||
|
||||
if (this.type === 'date') { |
||||
return yymmdd |
||||
} else if (this.type === 'time') { |
||||
return hhmmss |
||||
} else { |
||||
return yymmdd + ' ' + hhmmss |
||||
} |
||||
}, |
||||
|
||||
/** |
||||
* 初始化返回值,并抛出 change 事件 |
||||
*/ |
||||
initTime(emit = true) { |
||||
this.time = this.createDomSting() |
||||
if (!emit) return |
||||
if (this.returnType === 'timestamp' && this.type !== 'time') { |
||||
this.$emit('change', this.createTimeStamp(this.time)) |
||||
this.$emit('input', this.createTimeStamp(this.time)) |
||||
this.$emit('update:modelValue', this.createTimeStamp(this.time)) |
||||
} else { |
||||
this.$emit('change', this.time) |
||||
this.$emit('input', this.time) |
||||
this.$emit('update:modelValue', this.time) |
||||
} |
||||
}, |
||||
|
||||
/** |
||||
* 用户选择日期或时间更新 data |
||||
* @param {Object} e |
||||
*/ |
||||
bindDateChange(e) { |
||||
const val = e.detail.value |
||||
this.year = this.years[val[0]] |
||||
this.month = this.months[val[1]] |
||||
this.day = this.days[val[2]] |
||||
}, |
||||
bindTimeChange(e) { |
||||
const val = e.detail.value |
||||
this.hour = this.hours[val[0]] |
||||
this.minute = this.minutes[val[1]] |
||||
this.second = this.seconds[val[2]] |
||||
}, |
||||
|
||||
/** |
||||
* 初始化弹出层 |
||||
*/ |
||||
initTimePicker() { |
||||
if (this.disabled) return |
||||
const value = this.fixIosDateFormat(this.value) |
||||
this.initPickerValue(value) |
||||
this.visible = !this.visible |
||||
}, |
||||
|
||||
/** |
||||
* 触发或关闭弹框 |
||||
*/ |
||||
tiggerTimePicker(e) { |
||||
this.visible = !this.visible |
||||
}, |
||||
|
||||
/** |
||||
* 用户点击“清空”按钮,清空当前值 |
||||
*/ |
||||
clearTime() { |
||||
this.time = '' |
||||
this.$emit('change', this.time) |
||||
this.$emit('input', this.time) |
||||
this.$emit('update:modelValue', this.time) |
||||
this.tiggerTimePicker() |
||||
}, |
||||
|
||||
/** |
||||
* 用户点击“确定”按钮 |
||||
*/ |
||||
setTime() { |
||||
this.initTime() |
||||
this.tiggerTimePicker() |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss"> |
||||
$uni-primary: #007aff !default; |
||||
|
||||
.uni-datetime-picker { |
||||
/* #ifndef APP-NVUE */ |
||||
/* width: 100%; */ |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-datetime-picker-view { |
||||
height: 130px; |
||||
width: 270px; |
||||
/* #ifndef APP-NVUE */ |
||||
cursor: pointer; |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-datetime-picker-item { |
||||
height: 50px; |
||||
line-height: 50px; |
||||
text-align: center; |
||||
font-size: 14px; |
||||
} |
||||
|
||||
.uni-datetime-picker-btn { |
||||
margin-top: 60px; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
cursor: pointer; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
justify-content: space-between; |
||||
} |
||||
|
||||
.uni-datetime-picker-btn-text { |
||||
font-size: 14px; |
||||
color: $uni-primary; |
||||
} |
||||
|
||||
.uni-datetime-picker-btn-group { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
} |
||||
|
||||
.uni-datetime-picker-cancel { |
||||
margin-right: 30px; |
||||
} |
||||
|
||||
.uni-datetime-picker-mask { |
||||
position: fixed; |
||||
bottom: 0px; |
||||
top: 0px; |
||||
left: 0px; |
||||
right: 0px; |
||||
background-color: rgba(0, 0, 0, 0.4); |
||||
transition-duration: 0.3s; |
||||
z-index: 998; |
||||
} |
||||
|
||||
.uni-datetime-picker-popup { |
||||
border-radius: 8px; |
||||
padding: 30px; |
||||
width: 270px; |
||||
/* #ifdef APP-NVUE */ |
||||
height: 500px; |
||||
/* #endif */ |
||||
/* #ifdef APP-NVUE */ |
||||
width: 330px; |
||||
/* #endif */ |
||||
background-color: #fff; |
||||
position: fixed; |
||||
top: 50%; |
||||
left: 50%; |
||||
transform: translate(-50%, -50%); |
||||
transition-duration: 0.3s; |
||||
z-index: 999; |
||||
} |
||||
|
||||
.fix-nvue-height { |
||||
/* #ifdef APP-NVUE */ |
||||
height: 330px; |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-datetime-picker-time { |
||||
color: grey; |
||||
} |
||||
|
||||
.uni-datetime-picker-column { |
||||
height: 50px; |
||||
} |
||||
|
||||
.uni-datetime-picker-timebox { |
||||
|
||||
border: 1px solid #E5E5E5; |
||||
border-radius: 5px; |
||||
padding: 7px 10px; |
||||
/* #ifndef APP-NVUE */ |
||||
box-sizing: border-box; |
||||
cursor: pointer; |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-datetime-picker-timebox-pointer { |
||||
/* #ifndef APP-NVUE */ |
||||
cursor: pointer; |
||||
/* #endif */ |
||||
} |
||||
|
||||
|
||||
.uni-datetime-picker-disabled { |
||||
opacity: 0.4; |
||||
/* #ifdef H5 */ |
||||
cursor: not-allowed !important; |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-datetime-picker-text { |
||||
font-size: 14px; |
||||
line-height: 50px |
||||
} |
||||
|
||||
.uni-datetime-picker-sign { |
||||
position: absolute; |
||||
top: 53px; |
||||
/* 减掉 10px 的元素高度,兼容nvue */ |
||||
color: #999; |
||||
/* #ifdef APP-NVUE */ |
||||
font-size: 16px; |
||||
/* #endif */ |
||||
} |
||||
|
||||
.sign-left { |
||||
left: 86px; |
||||
} |
||||
|
||||
.sign-right { |
||||
right: 86px; |
||||
} |
||||
|
||||
.sign-center { |
||||
left: 135px; |
||||
} |
||||
|
||||
.uni-datetime-picker__container-box { |
||||
position: relative; |
||||
display: flex; |
||||
align-items: center; |
||||
justify-content: center; |
||||
margin-top: 40px; |
||||
} |
||||
|
||||
.time-hide-second { |
||||
width: 180px; |
||||
} |
||||
</style> |
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,410 @@ |
||||
class Calendar { |
||||
constructor({ |
||||
date, |
||||
selected, |
||||
startDate, |
||||
endDate, |
||||
range, |
||||
// multipleStatus
|
||||
} = {}) { |
||||
// 当前日期
|
||||
this.date = this.getDate(new Date()) // 当前初入日期
|
||||
// 打点信息
|
||||
this.selected = selected || []; |
||||
// 范围开始
|
||||
this.startDate = startDate |
||||
// 范围结束
|
||||
this.endDate = endDate |
||||
this.range = range |
||||
// 多选状态
|
||||
this.cleanMultipleStatus() |
||||
// 每周日期
|
||||
this.weeks = {} |
||||
// this._getWeek(this.date.fullDate)
|
||||
// this.multipleStatus = multipleStatus
|
||||
this.lastHover = false |
||||
} |
||||
/** |
||||
* 设置日期 |
||||
* @param {Object} date |
||||
*/ |
||||
setDate(date) { |
||||
this.selectDate = this.getDate(date) |
||||
this._getWeek(this.selectDate.fullDate) |
||||
} |
||||
|
||||
/** |
||||
* 清理多选状态 |
||||
*/ |
||||
cleanMultipleStatus() { |
||||
this.multipleStatus = { |
||||
before: '', |
||||
after: '', |
||||
data: [] |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 重置开始日期 |
||||
*/ |
||||
resetSatrtDate(startDate) { |
||||
// 范围开始
|
||||
this.startDate = startDate |
||||
|
||||
} |
||||
|
||||
/** |
||||
* 重置结束日期 |
||||
*/ |
||||
resetEndDate(endDate) { |
||||
// 范围结束
|
||||
this.endDate = endDate |
||||
} |
||||
|
||||
/** |
||||
* 获取任意时间 |
||||
*/ |
||||
getDate(date, AddDayCount = 0, str = 'day') { |
||||
if (!date) { |
||||
date = new Date() |
||||
} |
||||
if (typeof date !== 'object') { |
||||
date = date.replace(/-/g, '/') |
||||
} |
||||
const dd = new Date(date) |
||||
switch (str) { |
||||
case 'day': |
||||
dd.setDate(dd.getDate() + AddDayCount) // 获取AddDayCount天后的日期
|
||||
break |
||||
case 'month': |
||||
if (dd.getDate() === 31) { |
||||
dd.setDate(dd.getDate() + AddDayCount) |
||||
} else { |
||||
dd.setMonth(dd.getMonth() + AddDayCount) // 获取AddDayCount天后的日期
|
||||
} |
||||
break |
||||
case 'year': |
||||
dd.setFullYear(dd.getFullYear() + AddDayCount) // 获取AddDayCount天后的日期
|
||||
break |
||||
} |
||||
const y = dd.getFullYear() |
||||
const m = dd.getMonth() + 1 < 10 ? '0' + (dd.getMonth() + 1) : dd.getMonth() + 1 // 获取当前月份的日期,不足10补0
|
||||
const d = dd.getDate() < 10 ? '0' + dd.getDate() : dd.getDate() // 获取当前几号,不足10补0
|
||||
return { |
||||
fullDate: y + '-' + m + '-' + d, |
||||
year: y, |
||||
month: m, |
||||
date: d, |
||||
day: dd.getDay() |
||||
} |
||||
} |
||||
|
||||
|
||||
/** |
||||
* 获取上月剩余天数 |
||||
*/ |
||||
_getLastMonthDays(firstDay, full) { |
||||
let dateArr = [] |
||||
for (let i = firstDay; i > 0; i--) { |
||||
const beforeDate = new Date(full.year, full.month - 1, -i + 1).getDate() |
||||
dateArr.push({ |
||||
date: beforeDate, |
||||
month: full.month - 1, |
||||
disable: true |
||||
}) |
||||
} |
||||
return dateArr |
||||
} |
||||
/** |
||||
* 获取本月天数 |
||||
*/ |
||||
_currentMonthDys(dateData, full) { |
||||
let dateArr = [] |
||||
let fullDate = this.date.fullDate |
||||
for (let i = 1; i <= dateData; i++) { |
||||
let isinfo = false |
||||
let nowDate = full.year + '-' + (full.month < 10 ? |
||||
full.month : full.month) + '-' + (i < 10 ? |
||||
'0' + i : i) |
||||
// 是否今天
|
||||
let isDay = fullDate === nowDate |
||||
// 获取打点信息
|
||||
let info = this.selected && this.selected.find((item) => { |
||||
if (this.dateEqual(nowDate, item.date)) { |
||||
return item |
||||
} |
||||
}) |
||||
|
||||
// 日期禁用
|
||||
let disableBefore = true |
||||
let disableAfter = true |
||||
if (this.startDate) { |
||||
// let dateCompBefore = this.dateCompare(this.startDate, fullDate)
|
||||
// disableBefore = this.dateCompare(dateCompBefore ? this.startDate : fullDate, nowDate)
|
||||
disableBefore = this.dateCompare(this.startDate, nowDate) |
||||
} |
||||
|
||||
if (this.endDate) { |
||||
// let dateCompAfter = this.dateCompare(fullDate, this.endDate)
|
||||
// disableAfter = this.dateCompare(nowDate, dateCompAfter ? this.endDate : fullDate)
|
||||
disableAfter = this.dateCompare(nowDate, this.endDate) |
||||
} |
||||
let multiples = this.multipleStatus.data |
||||
let checked = false |
||||
let multiplesStatus = -1 |
||||
if (this.range) { |
||||
if (multiples) { |
||||
multiplesStatus = multiples.findIndex((item) => { |
||||
return this.dateEqual(item, nowDate) |
||||
}) |
||||
} |
||||
if (multiplesStatus !== -1) { |
||||
checked = true |
||||
} |
||||
} |
||||
let data = { |
||||
fullDate: nowDate, |
||||
year: full.year, |
||||
date: i, |
||||
multiple: this.range ? checked : false, |
||||
beforeMultiple: this.isLogicBefore(nowDate, this.multipleStatus.before, this.multipleStatus.after), |
||||
afterMultiple: this.isLogicAfter(nowDate, this.multipleStatus.before, this.multipleStatus.after), |
||||
month: full.month, |
||||
disable: !(disableBefore && disableAfter), |
||||
isDay, |
||||
userChecked: false |
||||
} |
||||
if (info) { |
||||
data.extraInfo = info |
||||
} |
||||
|
||||
dateArr.push(data) |
||||
} |
||||
return dateArr |
||||
} |
||||
/** |
||||
* 获取下月天数 |
||||
*/ |
||||
_getNextMonthDays(surplus, full) { |
||||
let dateArr = [] |
||||
for (let i = 1; i < surplus + 1; i++) { |
||||
dateArr.push({ |
||||
date: i, |
||||
month: Number(full.month) + 1, |
||||
disable: true |
||||
}) |
||||
} |
||||
return dateArr |
||||
} |
||||
|
||||
/** |
||||
* 获取当前日期详情 |
||||
* @param {Object} date |
||||
*/ |
||||
getInfo(date) { |
||||
if (!date) { |
||||
date = new Date() |
||||
} |
||||
const dateInfo = this.canlender.find(item => item.fullDate === this.getDate(date).fullDate) |
||||
return dateInfo |
||||
} |
||||
|
||||
/** |
||||
* 比较时间大小 |
||||
*/ |
||||
dateCompare(startDate, endDate) { |
||||
// 计算截止时间
|
||||
startDate = new Date(startDate.replace('-', '/').replace('-', '/')) |
||||
// 计算详细项的截止时间
|
||||
endDate = new Date(endDate.replace('-', '/').replace('-', '/')) |
||||
if (startDate <= endDate) { |
||||
return true |
||||
} else { |
||||
return false |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 比较时间是否相等 |
||||
*/ |
||||
dateEqual(before, after) { |
||||
// 计算截止时间
|
||||
before = new Date(before.replace('-', '/').replace('-', '/')) |
||||
// 计算详细项的截止时间
|
||||
after = new Date(after.replace('-', '/').replace('-', '/')) |
||||
if (before.getTime() - after.getTime() === 0) { |
||||
return true |
||||
} else { |
||||
return false |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 比较真实起始日期 |
||||
*/ |
||||
|
||||
isLogicBefore(currentDay, before, after) { |
||||
let logicBefore = before |
||||
if (before && after) { |
||||
logicBefore = this.dateCompare(before, after) ? before : after |
||||
} |
||||
return this.dateEqual(logicBefore, currentDay) |
||||
} |
||||
|
||||
isLogicAfter(currentDay, before, after) { |
||||
let logicAfter = after |
||||
if (before && after) { |
||||
logicAfter = this.dateCompare(before, after) ? after : before |
||||
} |
||||
return this.dateEqual(logicAfter, currentDay) |
||||
} |
||||
|
||||
/** |
||||
* 获取日期范围内所有日期 |
||||
* @param {Object} begin |
||||
* @param {Object} end |
||||
*/ |
||||
geDateAll(begin, end) { |
||||
var arr = [] |
||||
var ab = begin.split('-') |
||||
var ae = end.split('-') |
||||
var db = new Date() |
||||
db.setFullYear(ab[0], ab[1] - 1, ab[2]) |
||||
var de = new Date() |
||||
de.setFullYear(ae[0], ae[1] - 1, ae[2]) |
||||
var unixDb = db.getTime() - 24 * 60 * 60 * 1000 |
||||
var unixDe = de.getTime() - 24 * 60 * 60 * 1000 |
||||
for (var k = unixDb; k <= unixDe;) { |
||||
k = k + 24 * 60 * 60 * 1000 |
||||
arr.push(this.getDate(new Date(parseInt(k))).fullDate) |
||||
} |
||||
return arr |
||||
} |
||||
|
||||
/** |
||||
* 获取多选状态 |
||||
*/ |
||||
setMultiple(fullDate) { |
||||
let { |
||||
before, |
||||
after |
||||
} = this.multipleStatus |
||||
if (!this.range) return |
||||
if (before && after) { |
||||
if (!this.lastHover) { |
||||
this.lastHover = true |
||||
return |
||||
} |
||||
this.multipleStatus.before = fullDate |
||||
this.multipleStatus.after = '' |
||||
this.multipleStatus.data = [] |
||||
this.multipleStatus.fulldate = '' |
||||
this.lastHover = false |
||||
} else { |
||||
if (!before) { |
||||
this.multipleStatus.before = fullDate |
||||
this.lastHover = false |
||||
} else { |
||||
this.multipleStatus.after = fullDate |
||||
if (this.dateCompare(this.multipleStatus.before, this.multipleStatus.after)) { |
||||
this.multipleStatus.data = this.geDateAll(this.multipleStatus.before, this.multipleStatus |
||||
.after); |
||||
} else { |
||||
this.multipleStatus.data = this.geDateAll(this.multipleStatus.after, this.multipleStatus |
||||
.before); |
||||
} |
||||
this.lastHover = true |
||||
} |
||||
} |
||||
this._getWeek(fullDate) |
||||
} |
||||
|
||||
/** |
||||
* 鼠标 hover 更新多选状态 |
||||
*/ |
||||
setHoverMultiple(fullDate) { |
||||
let { |
||||
before, |
||||
after |
||||
} = this.multipleStatus |
||||
|
||||
if (!this.range) return |
||||
if (this.lastHover) return |
||||
|
||||
if (!before) { |
||||
this.multipleStatus.before = fullDate |
||||
} else { |
||||
this.multipleStatus.after = fullDate |
||||
if (this.dateCompare(this.multipleStatus.before, this.multipleStatus.after)) { |
||||
this.multipleStatus.data = this.geDateAll(this.multipleStatus.before, this.multipleStatus.after); |
||||
} else { |
||||
this.multipleStatus.data = this.geDateAll(this.multipleStatus.after, this.multipleStatus.before); |
||||
} |
||||
} |
||||
this._getWeek(fullDate) |
||||
} |
||||
|
||||
/** |
||||
* 更新默认值多选状态 |
||||
*/ |
||||
setDefaultMultiple(before, after) { |
||||
this.multipleStatus.before = before |
||||
this.multipleStatus.after = after |
||||
if (before && after) { |
||||
if (this.dateCompare(before, after)) { |
||||
this.multipleStatus.data = this.geDateAll(before, after); |
||||
this._getWeek(after) |
||||
} else { |
||||
this.multipleStatus.data = this.geDateAll(after, before); |
||||
this._getWeek(before) |
||||
} |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 获取每周数据 |
||||
* @param {Object} dateData |
||||
*/ |
||||
_getWeek(dateData) { |
||||
const { |
||||
fullDate, |
||||
year, |
||||
month, |
||||
date, |
||||
day |
||||
} = this.getDate(dateData) |
||||
let firstDay = new Date(year, month - 1, 1).getDay() |
||||
let currentDay = new Date(year, month, 0).getDate() |
||||
let dates = { |
||||
lastMonthDays: this._getLastMonthDays(firstDay, this.getDate(dateData)), // 上个月末尾几天
|
||||
currentMonthDys: this._currentMonthDys(currentDay, this.getDate(dateData)), // 本月天数
|
||||
nextMonthDays: [], // 下个月开始几天
|
||||
weeks: [] |
||||
} |
||||
let canlender = [] |
||||
const surplus = 42 - (dates.lastMonthDays.length + dates.currentMonthDys.length) |
||||
dates.nextMonthDays = this._getNextMonthDays(surplus, this.getDate(dateData)) |
||||
canlender = canlender.concat(dates.lastMonthDays, dates.currentMonthDys, dates.nextMonthDays) |
||||
let weeks = {} |
||||
// 拼接数组 上个月开始几天 + 本月天数+ 下个月开始几天
|
||||
for (let i = 0; i < canlender.length; i++) { |
||||
if (i % 7 === 0) { |
||||
weeks[parseInt(i / 7)] = new Array(7) |
||||
} |
||||
weeks[parseInt(i / 7)][i % 7] = canlender[i] |
||||
} |
||||
this.canlender = canlender |
||||
this.weeks = weeks |
||||
} |
||||
|
||||
//静态方法
|
||||
// static init(date) {
|
||||
// if (!this.instance) {
|
||||
// this.instance = new Calendar(date);
|
||||
// }
|
||||
// return this.instance;
|
||||
// }
|
||||
} |
||||
|
||||
|
||||
export default Calendar |
@ -0,0 +1,87 @@ |
||||
{ |
||||
"id": "uni-datetime-picker", |
||||
"displayName": "uni-datetime-picker 日期选择器", |
||||
"version": "2.2.11", |
||||
"description": "uni-datetime-picker 日期时间选择器,支持日历,支持范围选择", |
||||
"keywords": [ |
||||
"uni-datetime-picker", |
||||
"uni-ui", |
||||
"uniui", |
||||
"日期时间选择器", |
||||
"日期时间" |
||||
], |
||||
"repository": "https://github.com/dcloudio/uni-ui", |
||||
"engines": { |
||||
"HBuilderX": "" |
||||
}, |
||||
"directories": { |
||||
"example": "../../temps/example_temps" |
||||
}, |
||||
"dcloudext": { |
||||
"sale": { |
||||
"regular": { |
||||
"price": "0.00" |
||||
}, |
||||
"sourcecode": { |
||||
"price": "0.00" |
||||
} |
||||
}, |
||||
"contact": { |
||||
"qq": "" |
||||
}, |
||||
"declaration": { |
||||
"ads": "无", |
||||
"data": "无", |
||||
"permissions": "无" |
||||
}, |
||||
"npmurl": "https://www.npmjs.com/package/@dcloudio/uni-ui", |
||||
"type": "component-vue" |
||||
}, |
||||
"uni_modules": { |
||||
"dependencies": [ |
||||
"uni-scss", |
||||
"uni-icons" |
||||
], |
||||
"encrypt": [], |
||||
"platforms": { |
||||
"cloud": { |
||||
"tcb": "y", |
||||
"aliyun": "y" |
||||
}, |
||||
"client": { |
||||
"App": { |
||||
"app-vue": "y", |
||||
"app-nvue": "n" |
||||
}, |
||||
"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" |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,21 @@ |
||||
|
||||
|
||||
> `重要通知:组件升级更新 2.0.0 后,支持日期+时间范围选择,组件 ui 将使用日历选择日期,ui 变化较大,同时支持 PC 和 移动端。此版本不向后兼容,不再支持单独的时间选择(type=time)及相关的 hide-second 属性(时间选可使用内置组件 picker)。若仍需使用旧版本,可在插件市场下载*非uni_modules版本*,旧版本将不再维护` |
||||
|
||||
## DatetimePicker 时间选择器 |
||||
|
||||
> **组件名:uni-datetime-picker** |
||||
> 代码块: `uDatetimePicker` |
||||
|
||||
|
||||
该组件的优势是,支持**时间戳**输入和输出(起始时间、终止时间也支持时间戳),可**同时选择**日期和时间。 |
||||
|
||||
若只是需要单独选择日期和时间,不需要时间戳输入和输出,可使用原生的 picker 组件。 |
||||
|
||||
**_点击 picker 默认值规则:_** |
||||
|
||||
- 若设置初始值 value, 会显示在 picker 显示框中 |
||||
- 若无初始值 value,则初始值 value 为当前本地时间 Date.now(), 但不会显示在 picker 显示框中 |
||||
|
||||
### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-datetime-picker) |
||||
#### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839 |
@ -0,0 +1,13 @@ |
||||
## 1.2.1(2021-11-22) |
||||
- 修复 vue3中个别scss变量无法找到的问题 |
||||
## 1.2.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-drawer](https://uniapp.dcloud.io/component/uniui/uni-drawer) |
||||
## 1.1.1(2021-07-30) |
||||
- 优化 vue3下事件警告的问题 |
||||
## 1.1.0(2021-07-13) |
||||
- 组件兼容 vue3,如何创建vue3项目,详见 [uni-app 项目支持 vue3 介绍](https://ask.dcloud.net.cn/article/37834) |
||||
## 1.0.7(2021-05-12) |
||||
- 新增 组件示例地址 |
||||
## 1.0.6(2021-02-04) |
||||
- 调整为uni_modules目录规范 |
@ -0,0 +1,45 @@ |
||||
// #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
|
@ -0,0 +1,183 @@ |
||||
<template> |
||||
<view v-if="visibleSync" :class="{ 'uni-drawer--visible': showDrawer }" class="uni-drawer" @touchmove.stop.prevent="clear"> |
||||
<view class="uni-drawer__mask" :class="{ 'uni-drawer__mask--visible': showDrawer && mask }" @tap="close('mask')" /> |
||||
<view class="uni-drawer__content" :class="{'uni-drawer--right': rightMode,'uni-drawer--left': !rightMode, 'uni-drawer__content--visible': showDrawer}" :style="{width:drawerWidth+'px'}"> |
||||
<slot /> |
||||
</view> |
||||
<!-- #ifdef H5 --> |
||||
<keypress @esc="close('mask')" /> |
||||
<!-- #endif --> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
// #ifdef H5 |
||||
import keypress from './keypress.js' |
||||
// #endif |
||||
/** |
||||
* Drawer 抽屉 |
||||
* @description 抽屉侧滑菜单 |
||||
* @tutorial https://ext.dcloud.net.cn/plugin?id=26 |
||||
* @property {Boolean} mask = [true | false] 是否显示遮罩 |
||||
* @property {Boolean} maskClick = [true | false] 点击遮罩是否关闭 |
||||
* @property {Boolean} mode = [left | right] Drawer 滑出位置 |
||||
* @value left 从左侧滑出 |
||||
* @value right 从右侧侧滑出 |
||||
* @property {Number} width 抽屉的宽度 ,仅 vue 页面生效 |
||||
* @event {Function} close 组件关闭时触发事件 |
||||
*/ |
||||
export default { |
||||
name: 'UniDrawer', |
||||
components: { |
||||
// #ifdef H5 |
||||
keypress |
||||
// #endif |
||||
}, |
||||
emits:['change'], |
||||
props: { |
||||
/** |
||||
* 显示模式(左、右),只在初始化生效 |
||||
*/ |
||||
mode: { |
||||
type: String, |
||||
default: '' |
||||
}, |
||||
/** |
||||
* 蒙层显示状态 |
||||
*/ |
||||
mask: { |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
/** |
||||
* 遮罩是否可点击关闭 |
||||
*/ |
||||
maskClick:{ |
||||
type: Boolean, |
||||
default: true |
||||
}, |
||||
/** |
||||
* 抽屉宽度 |
||||
*/ |
||||
width: { |
||||
type: Number, |
||||
default: 220 |
||||
} |
||||
}, |
||||
data() { |
||||
return { |
||||
visibleSync: false, |
||||
showDrawer: false, |
||||
rightMode: false, |
||||
watchTimer: null, |
||||
drawerWidth: 220 |
||||
} |
||||
}, |
||||
created() { |
||||
// #ifndef APP-NVUE |
||||
this.drawerWidth = this.width |
||||
// #endif |
||||
this.rightMode = this.mode === 'right' |
||||
}, |
||||
methods: { |
||||
clear(){}, |
||||
close(type) { |
||||
// fixed by mehaotian 抽屉尚未完全关闭或遮罩禁止点击时不触发以下逻辑 |
||||
if((type === 'mask' && !this.maskClick) || !this.visibleSync) return |
||||
this._change('showDrawer', 'visibleSync', false) |
||||
}, |
||||
open() { |
||||
// fixed by mehaotian 处理重复点击打开的事件 |
||||
if(this.visibleSync) return |
||||
this._change('visibleSync', 'showDrawer', true) |
||||
}, |
||||
_change(param1, param2, status) { |
||||
this[param1] = status |
||||
if (this.watchTimer) { |
||||
clearTimeout(this.watchTimer) |
||||
} |
||||
this.watchTimer = setTimeout(() => { |
||||
this[param2] = status |
||||
this.$emit('change',status) |
||||
}, status ? 50 : 300) |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss" scoped> |
||||
$uni-mask: rgba($color: #000000, $alpha: 0.4) ; |
||||
// 抽屉宽度 |
||||
$drawer-width: 220px; |
||||
|
||||
.uni-drawer { |
||||
/* #ifndef APP-NVUE */ |
||||
display: block; |
||||
/* #endif */ |
||||
position: fixed; |
||||
top: 0; |
||||
left: 0; |
||||
right: 0; |
||||
bottom: 0; |
||||
overflow: hidden; |
||||
z-index: 999; |
||||
} |
||||
|
||||
.uni-drawer__content { |
||||
/* #ifndef APP-NVUE */ |
||||
display: block; |
||||
/* #endif */ |
||||
position: absolute; |
||||
top: 0; |
||||
width: $drawer-width; |
||||
bottom: 0; |
||||
background-color: $uni-bg-color; |
||||
transition: transform 0.3s ease; |
||||
} |
||||
|
||||
.uni-drawer--left { |
||||
left: 0; |
||||
/* #ifdef APP-NVUE */ |
||||
transform: translateX(-$drawer-width); |
||||
/* #endif */ |
||||
/* #ifndef APP-NVUE */ |
||||
transform: translateX(-100%); |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-drawer--right { |
||||
right: 0; |
||||
/* #ifdef APP-NVUE */ |
||||
transform: translateX($drawer-width); |
||||
/* #endif */ |
||||
/* #ifndef APP-NVUE */ |
||||
transform: translateX(100%); |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-drawer__content--visible { |
||||
transform: translateX(0px); |
||||
} |
||||
|
||||
|
||||
.uni-drawer__mask { |
||||
/* #ifndef APP-NVUE */ |
||||
display: block; |
||||
/* #endif */ |
||||
opacity: 0; |
||||
position: absolute; |
||||
top: 0; |
||||
left: 0; |
||||
bottom: 0; |
||||
right: 0; |
||||
background-color: $uni-mask; |
||||
transition: opacity 0.3s; |
||||
} |
||||
|
||||
.uni-drawer__mask--visible { |
||||
/* #ifndef APP-NVUE */ |
||||
display: block; |
||||
/* #endif */ |
||||
opacity: 1; |
||||
} |
||||
</style> |
@ -0,0 +1,87 @@ |
||||
{ |
||||
"id": "uni-drawer", |
||||
"displayName": "uni-drawer 抽屉", |
||||
"version": "1.2.1", |
||||
"description": "抽屉式导航,用于展示侧滑菜单,侧滑导航。", |
||||
"keywords": [ |
||||
"uni-ui", |
||||
"uniui", |
||||
"drawer", |
||||
"抽屉", |
||||
"侧滑导航" |
||||
], |
||||
"repository": "https://github.com/dcloudio/uni-ui", |
||||
"engines": { |
||||
"HBuilderX": "" |
||||
}, |
||||
"directories": { |
||||
"example": "../../temps/example_temps" |
||||
}, |
||||
"dcloudext": { |
||||
"category": [ |
||||
"前端组件", |
||||
"通用组件" |
||||
], |
||||
"sale": { |
||||
"regular": { |
||||
"price": "0.00" |
||||
}, |
||||
"sourcecode": { |
||||
"price": "0.00" |
||||
} |
||||
}, |
||||
"contact": { |
||||
"qq": "" |
||||
}, |
||||
"declaration": { |
||||
"ads": "无", |
||||
"data": "无", |
||||
"permissions": "无" |
||||
}, |
||||
"npmurl": "https://www.npmjs.com/package/@dcloudio/uni-ui" |
||||
}, |
||||
"uni_modules": { |
||||
"dependencies": ["uni-scss"], |
||||
"encrypt": [], |
||||
"platforms": { |
||||
"cloud": { |
||||
"tcb": "y", |
||||
"aliyun": "y" |
||||
}, |
||||
"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" |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,10 @@ |
||||
|
||||
|
||||
## Drawer 抽屉 |
||||
> **组件名:uni-drawer** |
||||
> 代码块: `uDrawer` |
||||
|
||||
抽屉侧滑菜单。 |
||||
|
||||
### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-drawer) |
||||
#### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839 |
@ -0,0 +1,53 @@ |
||||
## 1.1.3(2022-09-22) |
||||
- 修复,引入 uni.scss 引入默认主题色报错的问题 |
||||
## 1.1.2(2022-09-22) |
||||
- 增加主题色 primaryColor 配置选项 |
||||
## 1.1.1(2022-09-19) |
||||
- 修复,输入后回车,change 事件触发两次,[详情](https://ask.dcloud.net.cn/question/152149) |
||||
## 1.1.0(2022-06-30) |
||||
- 新增 在 uni-forms 1.4.0 中使用可以在 blur 时校验内容 |
||||
- 新增 clear 事件,点击右侧叉号图标触发 |
||||
- 新增 change 事件 ,仅在输入框失去焦点或用户按下回车时触发 |
||||
- 优化 组件样式,组件获取焦点时高亮显示,图标颜色调整等 |
||||
- |
||||
## 1.0.5(2022-06-07) |
||||
- 优化 clearable 显示策略 |
||||
## 1.0.4(2022-06-07) |
||||
- 优化 clearable 显示策略 |
||||
## 1.0.3(2022-05-20) |
||||
- 修复 关闭图标某些情况下无法取消的bug |
||||
## 1.0.2(2022-04-12) |
||||
- 修复 默认值不生效的bug |
||||
## 1.0.1(2022-04-02) |
||||
- 修复 value不能为0的bug |
||||
## 1.0.0(2021-11-19) |
||||
- 优化 组件UI,并提供设计资源,详见:[https://uniapp.dcloud.io/component/uniui/resource](https://uniapp.dcloud.io/component/uniui/resource) |
||||
- 文档迁移,详见:[https://uniapp.dcloud.io/component/uniui/uni-easyinput](https://uniapp.dcloud.io/component/uniui/uni-easyinput) |
||||
## 0.1.4(2021-08-20) |
||||
- 修复 在 uni-forms 的动态表单中默认值校验不通过的 bug |
||||
## 0.1.3(2021-08-11) |
||||
- 修复 在 uni-forms 中重置表单,错误信息无法清除的问题 |
||||
## 0.1.2(2021-07-30) |
||||
- 优化 vue3下事件警告的问题 |
||||
## 0.1.1 |
||||
- 优化 errorMessage 属性支持 Boolean 类型 |
||||
## 0.1.0(2021-07-13) |
||||
- 组件兼容 vue3,如何创建vue3项目,详见 [uni-app 项目支持 vue3 介绍](https://ask.dcloud.net.cn/article/37834) |
||||
## 0.0.16(2021-06-29) |
||||
- 修复 confirmType 属性(仅 type="text" 生效)导致多行文本框无法换行的 bug |
||||
## 0.0.15(2021-06-21) |
||||
- 修复 passwordIcon 属性拼写错误的 bug |
||||
## 0.0.14(2021-06-18) |
||||
- 新增 passwordIcon 属性,当type=password时是否显示小眼睛图标 |
||||
- 修复 confirmType 属性不生效的问题 |
||||
## 0.0.13(2021-06-04) |
||||
- 修复 disabled 状态可清出内容的 bug |
||||
## 0.0.12(2021-05-12) |
||||
- 新增 组件示例地址 |
||||
## 0.0.11(2021-05-07) |
||||
- 修复 input-border 属性不生效的问题 |
||||
## 0.0.10(2021-04-30) |
||||
- 修复 ios 遮挡文字、显示一半的问题 |
||||
## 0.0.9(2021-02-05) |
||||
- 调整为uni_modules目录规范 |
||||
- 优化 兼容 nvue 页面 |
@ -0,0 +1,56 @@ |
||||
/** |
||||
* @desc 函数防抖 |
||||
* @param func 目标函数 |
||||
* @param wait 延迟执行毫秒数 |
||||
* @param immediate true - 立即执行, false - 延迟执行 |
||||
*/ |
||||
export const debounce = function(func, wait = 1000, immediate = true) { |
||||
let timer; |
||||
console.log(1); |
||||
return function() { |
||||
console.log(123); |
||||
let context = this, |
||||
args = arguments; |
||||
if (timer) clearTimeout(timer); |
||||
if (immediate) { |
||||
let callNow = !timer; |
||||
timer = setTimeout(() => { |
||||
timer = null; |
||||
}, wait); |
||||
if (callNow) func.apply(context, args); |
||||
} else { |
||||
timer = setTimeout(() => { |
||||
func.apply(context, args); |
||||
}, wait) |
||||
} |
||||
} |
||||
} |
||||
/** |
||||
* @desc 函数节流 |
||||
* @param func 函数 |
||||
* @param wait 延迟执行毫秒数 |
||||
* @param type 1 使用表时间戳,在时间段开始的时候触发 2 使用表定时器,在时间段结束的时候触发 |
||||
*/ |
||||
export const throttle = (func, wait = 1000, type = 1) => { |
||||
let previous = 0; |
||||
let timeout; |
||||
return function() { |
||||
let context = this; |
||||
let args = arguments; |
||||
if (type === 1) { |
||||
let now = Date.now(); |
||||
|
||||
if (now - previous > wait) { |
||||
func.apply(context, args); |
||||
previous = now; |
||||
} |
||||
} else if (type === 2) { |
||||
if (!timeout) { |
||||
timeout = setTimeout(() => { |
||||
timeout = null; |
||||
func.apply(context, args) |
||||
}, wait) |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,87 @@ |
||||
{ |
||||
"id": "uni-easyinput", |
||||
"displayName": "uni-easyinput 增强输入框", |
||||
"version": "1.1.3", |
||||
"description": "Easyinput 组件是对原生input组件的增强", |
||||
"keywords": [ |
||||
"uni-ui", |
||||
"uniui", |
||||
"input", |
||||
"uni-easyinput", |
||||
"输入框" |
||||
], |
||||
"repository": "https://github.com/dcloudio/uni-ui", |
||||
"engines": { |
||||
"HBuilderX": "" |
||||
}, |
||||
"directories": { |
||||
"example": "../../temps/example_temps" |
||||
}, |
||||
"dcloudext": { |
||||
"sale": { |
||||
"regular": { |
||||
"price": "0.00" |
||||
}, |
||||
"sourcecode": { |
||||
"price": "0.00" |
||||
} |
||||
}, |
||||
"contact": { |
||||
"qq": "" |
||||
}, |
||||
"declaration": { |
||||
"ads": "无", |
||||
"data": "无", |
||||
"permissions": "无" |
||||
}, |
||||
"npmurl": "https://www.npmjs.com/package/@dcloudio/uni-ui", |
||||
"type": "component-vue" |
||||
}, |
||||
"uni_modules": { |
||||
"dependencies": [ |
||||
"uni-scss", |
||||
"uni-icons" |
||||
], |
||||
"encrypt": [], |
||||
"platforms": { |
||||
"cloud": { |
||||
"tcb": "y", |
||||
"aliyun": "y" |
||||
}, |
||||
"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" |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,11 @@ |
||||
|
||||
|
||||
### Easyinput 增强输入框 |
||||
> **组件名:uni-easyinput** |
||||
> 代码块: `uEasyinput` |
||||
|
||||
|
||||
easyinput 组件是对原生input组件的增强 ,是专门为配合表单组件[uni-forms](https://ext.dcloud.net.cn/plugin?id=2773)而设计的,easyinput 内置了边框,图标等,同时包含 input 所有功能 |
||||
|
||||
### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-easyinput) |
||||
#### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839 |
@ -0,0 +1,21 @@ |
||||
## 1.2.4(2022-09-07) |
||||
小程序端由于 style 使用了对象导致报错,[详情](https://ask.dcloud.net.cn/question/152790?item_id=211778&rf=false) |
||||
## 1.2.3(2022-09-05) |
||||
- 修复 nvue 环境下,具有 tabBar 时,fab 组件下部位置无法正常获取 --window-bottom 的bug,详见:[https://ask.dcloud.net.cn/question/110638?notification_id=826310](https://ask.dcloud.net.cn/question/110638?notification_id=826310) |
||||
## 1.2.2(2021-12-29) |
||||
- 更新 组件依赖 |
||||
## 1.2.1(2021-11-19) |
||||
- 修复 阴影颜色不正确的bug |
||||
## 1.2.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-fab](https://uniapp.dcloud.io/component/uniui/uni-fab) |
||||
## 1.1.1(2021-11-09) |
||||
- 新增 提供组件设计资源,组件样式调整 |
||||
## 1.1.0(2021-07-30) |
||||
- 组件兼容 vue3,如何创建vue3项目,详见 [uni-app 项目支持 vue3 介绍](https://ask.dcloud.net.cn/article/37834) |
||||
## 1.0.7(2021-05-12) |
||||
- 新增 组件示例地址 |
||||
## 1.0.6(2021-02-05) |
||||
- 调整为uni_modules目录规范 |
||||
- 优化 按钮背景色调整 |
||||
- 优化 兼容pc端 |
@ -0,0 +1,487 @@ |
||||
<template> |
||||
<view class="uni-cursor-point"> |
||||
<view v-if="popMenu && (leftBottom||rightBottom||leftTop||rightTop) && content.length > 0" :class="{ |
||||
'uni-fab--leftBottom': leftBottom, |
||||
'uni-fab--rightBottom': rightBottom, |
||||
'uni-fab--leftTop': leftTop, |
||||
'uni-fab--rightTop': rightTop |
||||
}" class="uni-fab" |
||||
:style="nvueBottom" |
||||
> |
||||
<view :class="{ |
||||
'uni-fab__content--left': horizontal === 'left', |
||||
'uni-fab__content--right': horizontal === 'right', |
||||
'uni-fab__content--flexDirection': direction === 'vertical', |
||||
'uni-fab__content--flexDirectionStart': flexDirectionStart, |
||||
'uni-fab__content--flexDirectionEnd': flexDirectionEnd, |
||||
'uni-fab__content--other-platform': !isAndroidNvue |
||||
}" :style="{ width: boxWidth, height: boxHeight, backgroundColor: styles.backgroundColor }" |
||||
class="uni-fab__content" elevation="5"> |
||||
<view v-if="flexDirectionStart || horizontalLeft" class="uni-fab__item uni-fab__item--first" /> |
||||
<view v-for="(item, index) in content" :key="index" :class="{ 'uni-fab__item--active': isShow }" |
||||
class="uni-fab__item" @click="_onItemClick(index, item)"> |
||||
<image :src="item.active ? item.selectedIconPath : item.iconPath" class="uni-fab__item-image" |
||||
mode="aspectFit" /> |
||||
<text class="uni-fab__item-text" |
||||
:style="{ color: item.active ? styles.selectedColor : styles.color }">{{ item.text }}</text> |
||||
</view> |
||||
<view v-if="flexDirectionEnd || horizontalRight" class="uni-fab__item uni-fab__item--first" /> |
||||
</view> |
||||
</view> |
||||
<view :class="{ |
||||
'uni-fab__circle--leftBottom': leftBottom, |
||||
'uni-fab__circle--rightBottom': rightBottom, |
||||
'uni-fab__circle--leftTop': leftTop, |
||||
'uni-fab__circle--rightTop': rightTop, |
||||
'uni-fab__content--other-platform': !isAndroidNvue |
||||
}" class="uni-fab__circle uni-fab__plus" :style="{ 'background-color': styles.buttonColor, 'bottom': nvueBottom }" @click="_onClick"> |
||||
<uni-icons class="fab-circle-icon" type="plusempty" :color="styles.iconColor" size="32" |
||||
:class="{'uni-fab__plus--active': isShow && content.length > 0}"></uni-icons> |
||||
<!-- <view class="fab-circle-v" :class="{'uni-fab__plus--active': isShow && content.length > 0}"></view> |
||||
<view class="fab-circle-h" :class="{'uni-fab__plus--active': isShow && content.length > 0}"></view> --> |
||||
</view> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
let platform = 'other' |
||||
// #ifdef APP-NVUE |
||||
platform = uni.getSystemInfoSync().platform |
||||
// #endif |
||||
|
||||
/** |
||||
* Fab 悬浮按钮 |
||||
* @description 点击可展开一个图形按钮菜单 |
||||
* @tutorial https://ext.dcloud.net.cn/plugin?id=144 |
||||
* @property {Object} pattern 可选样式配置项 |
||||
* @property {Object} horizontal = [left | right] 水平对齐方式 |
||||
* @value left 左对齐 |
||||
* @value right 右对齐 |
||||
* @property {Object} vertical = [bottom | top] 垂直对齐方式 |
||||
* @value bottom 下对齐 |
||||
* @value top 上对齐 |
||||
* @property {Object} direction = [horizontal | vertical] 展开菜单显示方式 |
||||
* @value horizontal 水平显示 |
||||
* @value vertical 垂直显示 |
||||
* @property {Array} content 展开菜单内容配置项 |
||||
* @property {Boolean} popMenu 是否使用弹出菜单 |
||||
* @event {Function} trigger 展开菜单点击事件,返回点击信息 |
||||
* @event {Function} fabClick 悬浮按钮点击事件 |
||||
*/ |
||||
export default { |
||||
name: 'UniFab', |
||||
emits: ['fabClick', 'trigger'], |
||||
props: { |
||||
pattern: { |
||||
type: Object, |
||||
default () { |
||||
return {} |
||||
} |
||||
}, |
||||
horizontal: { |
||||
type: String, |
||||
default: 'left' |
||||
}, |
||||
vertical: { |
||||
type: String, |
||||
default: 'bottom' |
||||
}, |
||||
direction: { |
||||
type: String, |
||||
default: 'horizontal' |
||||
}, |
||||
content: { |
||||
type: Array, |
||||
default () { |
||||
return [] |
||||
} |
||||
}, |
||||
show: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
popMenu: { |
||||
type: Boolean, |
||||
default: true |
||||
} |
||||
}, |
||||
data() { |
||||
return { |
||||
fabShow: false, |
||||
isShow: false, |
||||
isAndroidNvue: platform === 'android', |
||||
styles: { |
||||
color: '#3c3e49', |
||||
selectedColor: '#007AFF', |
||||
backgroundColor: '#fff', |
||||
buttonColor: '#007AFF', |
||||
iconColor: '#fff' |
||||
} |
||||
} |
||||
}, |
||||
computed: { |
||||
contentWidth(e) { |
||||
return (this.content.length + 1) * 55 + 15 + 'px' |
||||
}, |
||||
contentWidthMin() { |
||||
return '55px' |
||||
}, |
||||
// 动态计算宽度 |
||||
boxWidth() { |
||||
return this.getPosition(3, 'horizontal') |
||||
}, |
||||
// 动态计算高度 |
||||
boxHeight() { |
||||
return this.getPosition(3, 'vertical') |
||||
}, |
||||
// 计算左下位置 |
||||
leftBottom() { |
||||
return this.getPosition(0, 'left', 'bottom') |
||||
}, |
||||
// 计算右下位置 |
||||
rightBottom() { |
||||
return this.getPosition(0, 'right', 'bottom') |
||||
}, |
||||
// 计算左上位置 |
||||
leftTop() { |
||||
return this.getPosition(0, 'left', 'top') |
||||
}, |
||||
rightTop() { |
||||
return this.getPosition(0, 'right', 'top') |
||||
}, |
||||
flexDirectionStart() { |
||||
return this.getPosition(1, 'vertical', 'top') |
||||
}, |
||||
flexDirectionEnd() { |
||||
return this.getPosition(1, 'vertical', 'bottom') |
||||
}, |
||||
horizontalLeft() { |
||||
return this.getPosition(2, 'horizontal', 'left') |
||||
}, |
||||
horizontalRight() { |
||||
return this.getPosition(2, 'horizontal', 'right') |
||||
}, |
||||
// 计算 nvue bottom |
||||
nvueBottom() { |
||||
const safeBottom = uni.getSystemInfoSync().windowBottom; |
||||
// #ifdef APP-NVUE |
||||
return 30 + safeBottom |
||||
// #endif |
||||
// #ifndef APP-NVUE |
||||
return 30 |
||||
// #endif |
||||
} |
||||
}, |
||||
watch: { |
||||
pattern: { |
||||
handler(val, oldVal) { |
||||
this.styles = Object.assign({}, this.styles, val) |
||||
}, |
||||
deep: true |
||||
} |
||||
}, |
||||
created() { |
||||
this.isShow = this.show |
||||
if (this.top === 0) { |
||||
this.fabShow = true |
||||
} |
||||
// 初始化样式 |
||||
this.styles = Object.assign({}, this.styles, this.pattern) |
||||
}, |
||||
methods: { |
||||
_onClick() { |
||||
this.$emit('fabClick') |
||||
if (!this.popMenu) { |
||||
return |
||||
} |
||||
this.isShow = !this.isShow |
||||
}, |
||||
open() { |
||||
this.isShow = true |
||||
}, |
||||
close() { |
||||
this.isShow = false |
||||
}, |
||||
/** |
||||
* 按钮点击事件 |
||||
*/ |
||||
_onItemClick(index, item) { |
||||
this.$emit('trigger', { |
||||
index, |
||||
item |
||||
}) |
||||
}, |
||||
/** |
||||
* 获取 位置信息 |
||||
*/ |
||||
getPosition(types, paramA, paramB) { |
||||
if (types === 0) { |
||||
return this.horizontal === paramA && this.vertical === paramB |
||||
} else if (types === 1) { |
||||
return this.direction === paramA && this.vertical === paramB |
||||
} else if (types === 2) { |
||||
return this.direction === paramA && this.horizontal === paramB |
||||
} else { |
||||
return this.isShow && this.direction === paramA ? this.contentWidth : this.contentWidthMin |
||||
} |
||||
} |
||||
} |
||||
} |
||||
</script> |
||||
|
||||
<style lang="scss" > |
||||
$uni-shadow-base:0 1px 5px 2px rgba($color: #000000, $alpha: 0.3) !default; |
||||
|
||||
.uni-fab { |
||||
position: fixed; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
justify-content: center; |
||||
align-items: center; |
||||
z-index: 10; |
||||
border-radius: 45px; |
||||
box-shadow: $uni-shadow-base; |
||||
} |
||||
|
||||
.uni-cursor-point { |
||||
/* #ifdef H5 */ |
||||
cursor: pointer; |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-fab--active { |
||||
opacity: 1; |
||||
} |
||||
|
||||
.uni-fab--leftBottom { |
||||
left: 15px; |
||||
bottom: 30px; |
||||
/* #ifdef H5 */ |
||||
left: calc(15px + var(--window-left)); |
||||
bottom: calc(30px + var(--window-bottom)); |
||||
/* #endif */ |
||||
// padding: 10px; |
||||
} |
||||
|
||||
.uni-fab--leftTop { |
||||
left: 15px; |
||||
top: 30px; |
||||
/* #ifdef H5 */ |
||||
left: calc(15px + var(--window-left)); |
||||
top: calc(30px + var(--window-top)); |
||||
/* #endif */ |
||||
// padding: 10px; |
||||
} |
||||
|
||||
.uni-fab--rightBottom { |
||||
right: 15px; |
||||
bottom: 30px; |
||||
/* #ifdef H5 */ |
||||
right: calc(15px + var(--window-right)); |
||||
bottom: calc(30px + var(--window-bottom)); |
||||
/* #endif */ |
||||
// padding: 10px; |
||||
} |
||||
|
||||
.uni-fab--rightTop { |
||||
right: 15px; |
||||
top: 30px; |
||||
/* #ifdef H5 */ |
||||
right: calc(15px + var(--window-right)); |
||||
top: calc(30px + var(--window-top)); |
||||
/* #endif */ |
||||
// padding: 10px; |
||||
} |
||||
|
||||
.uni-fab__circle { |
||||
position: fixed; |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
justify-content: center; |
||||
align-items: center; |
||||
width: 55px; |
||||
height: 55px; |
||||
background-color: #3c3e49; |
||||
border-radius: 45px; |
||||
z-index: 11; |
||||
// box-shadow: $uni-shadow-base; |
||||
} |
||||
|
||||
.uni-fab__circle--leftBottom { |
||||
left: 15px; |
||||
bottom: 30px; |
||||
/* #ifdef H5 */ |
||||
left: calc(15px + var(--window-left)); |
||||
bottom: calc(30px + var(--window-bottom)); |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-fab__circle--leftTop { |
||||
left: 15px; |
||||
top: 30px; |
||||
/* #ifdef H5 */ |
||||
left: calc(15px + var(--window-left)); |
||||
top: calc(30px + var(--window-top)); |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-fab__circle--rightBottom { |
||||
right: 15px; |
||||
bottom: 30px; |
||||
/* #ifdef H5 */ |
||||
right: calc(15px + var(--window-right)); |
||||
bottom: calc(30px + var(--window-bottom)); |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-fab__circle--rightTop { |
||||
right: 15px; |
||||
top: 30px; |
||||
/* #ifdef H5 */ |
||||
right: calc(15px + var(--window-right)); |
||||
top: calc(30px + var(--window-top)); |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-fab__circle--left { |
||||
left: 0; |
||||
} |
||||
|
||||
.uni-fab__circle--right { |
||||
right: 0; |
||||
} |
||||
|
||||
.uni-fab__circle--top { |
||||
top: 0; |
||||
} |
||||
|
||||
.uni-fab__circle--bottom { |
||||
bottom: 0; |
||||
} |
||||
|
||||
.uni-fab__plus { |
||||
font-weight: bold; |
||||
} |
||||
|
||||
// .fab-circle-v { |
||||
// position: absolute; |
||||
// width: 2px; |
||||
// height: 24px; |
||||
// left: 0; |
||||
// top: 0; |
||||
// right: 0; |
||||
// bottom: 0; |
||||
// /* #ifndef APP-NVUE */ |
||||
// margin: auto; |
||||
// /* #endif */ |
||||
// background-color: white; |
||||
// transform: rotate(0deg); |
||||
// transition: transform 0.3s; |
||||
// } |
||||
|
||||
// .fab-circle-h { |
||||
// position: absolute; |
||||
// width: 24px; |
||||
// height: 2px; |
||||
// left: 0; |
||||
// top: 0; |
||||
// right: 0; |
||||
// bottom: 0; |
||||
// /* #ifndef APP-NVUE */ |
||||
// margin: auto; |
||||
// /* #endif */ |
||||
// background-color: white; |
||||
// transform: rotate(0deg); |
||||
// transition: transform 0.3s; |
||||
// } |
||||
|
||||
.fab-circle-icon { |
||||
transform: rotate(0deg); |
||||
transition: transform 0.3s; |
||||
font-weight: 200; |
||||
} |
||||
|
||||
.uni-fab__plus--active { |
||||
transform: rotate(135deg); |
||||
} |
||||
|
||||
.uni-fab__content { |
||||
/* #ifndef APP-NVUE */ |
||||
box-sizing: border-box; |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
border-radius: 55px; |
||||
overflow: hidden; |
||||
transition-property: width, height; |
||||
transition-duration: 0.2s; |
||||
width: 55px; |
||||
border-color: #DDDDDD; |
||||
border-width: 1rpx; |
||||
border-style: solid; |
||||
} |
||||
|
||||
.uni-fab__content--other-platform { |
||||
border-width: 0px; |
||||
box-shadow: $uni-shadow-base; |
||||
} |
||||
|
||||
.uni-fab__content--left { |
||||
justify-content: flex-start; |
||||
} |
||||
|
||||
.uni-fab__content--right { |
||||
justify-content: flex-end; |
||||
} |
||||
|
||||
.uni-fab__content--flexDirection { |
||||
flex-direction: column; |
||||
justify-content: flex-end; |
||||
} |
||||
|
||||
.uni-fab__content--flexDirectionStart { |
||||
flex-direction: column; |
||||
justify-content: flex-start; |
||||
} |
||||
|
||||
.uni-fab__content--flexDirectionEnd { |
||||
flex-direction: column; |
||||
justify-content: flex-end; |
||||
} |
||||
|
||||
.uni-fab__item { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: column; |
||||
justify-content: center; |
||||
align-items: center; |
||||
width: 55px; |
||||
height: 55px; |
||||
opacity: 0; |
||||
transition: opacity 0.2s; |
||||
} |
||||
|
||||
.uni-fab__item--active { |
||||
opacity: 1; |
||||
} |
||||
|
||||
.uni-fab__item-image { |
||||
width: 20px; |
||||
height: 20px; |
||||
margin-bottom: 4px; |
||||
} |
||||
|
||||
.uni-fab__item-text { |
||||
color: #FFFFFF; |
||||
font-size: 12px; |
||||
line-height: 12px; |
||||
margin-top: 2px; |
||||
} |
||||
|
||||
.uni-fab__item--first { |
||||
width: 55px; |
||||
} |
||||
</style> |
@ -0,0 +1,84 @@ |
||||
{ |
||||
"id": "uni-fab", |
||||
"displayName": "uni-fab 悬浮按钮", |
||||
"version": "1.2.4", |
||||
"description": "悬浮按钮 fab button ,点击可展开一个图标按钮菜单。", |
||||
"keywords": [ |
||||
"uni-ui", |
||||
"uniui", |
||||
"按钮", |
||||
"悬浮按钮", |
||||
"fab" |
||||
], |
||||
"repository": "https://github.com/dcloudio/uni-ui", |
||||
"engines": { |
||||
"HBuilderX": "" |
||||
}, |
||||
"directories": { |
||||
"example": "../../temps/example_temps" |
||||
}, |
||||
"dcloudext": { |
||||
"sale": { |
||||
"regular": { |
||||
"price": "0.00" |
||||
}, |
||||
"sourcecode": { |
||||
"price": "0.00" |
||||
} |
||||
}, |
||||
"contact": { |
||||
"qq": "" |
||||
}, |
||||
"declaration": { |
||||
"ads": "无", |
||||
"data": "无", |
||||
"permissions": "无" |
||||
}, |
||||
"npmurl": "https://www.npmjs.com/package/@dcloudio/uni-ui", |
||||
"type": "component-vue" |
||||
}, |
||||
"uni_modules": { |
||||
"dependencies": ["uni-scss","uni-icons"], |
||||
"encrypt": [], |
||||
"platforms": { |
||||
"cloud": { |
||||
"tcb": "y", |
||||
"aliyun": "y" |
||||
}, |
||||
"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" |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,9 @@ |
||||
## Fab 悬浮按钮 |
||||
> **组件名:uni-fab** |
||||
> 代码块: `uFab` |
||||
|
||||
|
||||
点击可展开一个图形按钮菜单 |
||||
|
||||
### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-fab) |
||||
#### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839 |
@ -0,0 +1,19 @@ |
||||
## 1.2.1(2022-05-30) |
||||
- 新增 stat 属性 ,是否开启uni统计功能 |
||||
## 1.2.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-fav](https://uniapp.dcloud.io/component/uniui/uni-fav) |
||||
## 1.1.1(2021-08-24) |
||||
- 新增 支持国际化 |
||||
## 1.1.0(2021-07-13) |
||||
- 组件兼容 vue3,如何创建vue3项目,详见 [uni-app 项目支持 vue3 介绍](https://ask.dcloud.net.cn/article/37834) |
||||
## 1.0.6(2021-05-12) |
||||
- 新增 组件示例地址 |
||||
## 1.0.5(2021-04-21) |
||||
- 优化 添加依赖 uni-icons, 导入后自动下载依赖 |
||||
## 1.0.4(2021-02-05) |
||||
- 优化 组件引用关系,通过uni_modules引用组件 |
||||
## 1.0.3(2021-02-05) |
||||
- 优化 组件引用关系,通过uni_modules引用组件 |
||||
## 1.0.2(2021-02-05) |
||||
- 调整为uni_modules目录规范 |
@ -0,0 +1,4 @@ |
||||
{ |
||||
"uni-fav.collect": "collect", |
||||
"uni-fav.collected": "collected" |
||||
} |
@ -0,0 +1,8 @@ |
||||
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 |
||||
} |
@ -0,0 +1,4 @@ |
||||
{ |
||||
"uni-fav.collect": "收藏", |
||||
"uni-fav.collected": "已收藏" |
||||
} |
@ -0,0 +1,4 @@ |
||||
{ |
||||
"uni-fav.collect": "收藏", |
||||
"uni-fav.collected": "已收藏" |
||||
} |
@ -0,0 +1,161 @@ |
||||
<template> |
||||
<view :class="[circle === true || circle === 'true' ? 'uni-fav--circle' : '']" :style="[{ backgroundColor: checked ? bgColorChecked : bgColor }]" |
||||
@click="onClick" class="uni-fav"> |
||||
<!-- #ifdef MP-ALIPAY --> |
||||
<view class="uni-fav-star" v-if="!checked && (star === true || star === 'true')"> |
||||
<uni-icons :color="fgColor" :style="{color: checked ? fgColorChecked : fgColor}" size="14" type="star-filled" /> |
||||
</view> |
||||
<!-- #endif --> |
||||
<!-- #ifndef MP-ALIPAY --> |
||||
<uni-icons :color="fgColor" :style="{color: checked ? fgColorChecked : fgColor}" class="uni-fav-star" size="14" type="star-filled" |
||||
v-if="!checked && (star === true || star === 'true')" /> |
||||
<!-- #endif --> |
||||
<text :style="{color: checked ? fgColorChecked : fgColor}" class="uni-fav-text">{{ checked ? contentFav : contentDefault }}</text> |
||||
</view> |
||||
</template> |
||||
|
||||
<script> |
||||
|
||||
/** |
||||
* Fav 收藏按钮 |
||||
* @description 用于收藏功能,可点击切换选中、不选中的状态 |
||||
* @tutorial https://ext.dcloud.net.cn/plugin?id=864 |
||||
* @property {Boolean} star = [true|false] 按钮是否带星星 |
||||
* @property {String} bgColor 未收藏时的背景色 |
||||
* @property {String} bgColorChecked 已收藏时的背景色 |
||||
* @property {String} fgColor 未收藏时的文字颜色 |
||||
* @property {String} fgColorChecked 已收藏时的文字颜色 |
||||
* @property {Boolean} circle = [true|false] 是否为圆角 |
||||
* @property {Boolean} checked = [true|false] 是否为已收藏 |
||||
* @property {Object} contentText = [true|false] 收藏按钮文字 |
||||
* @property {Boolean} stat 是否开启统计功能 |
||||
* @event {Function} click 点击 fav按钮触发事件 |
||||
* @example <uni-fav :checked="true"/> |
||||
*/ |
||||
|
||||
import { |
||||
initVueI18n |
||||
} from '@dcloudio/uni-i18n' |
||||
import messages from './i18n/index.js' |
||||
const { t } = initVueI18n(messages) |
||||
|
||||
export default { |
||||
name: "UniFav", |
||||
// TODO 兼容 vue3,需要注册事件 |
||||
emits: ['click'], |
||||
props: { |
||||
star: { |
||||
type: [Boolean, String], |
||||
default: true |
||||
}, |
||||
bgColor: { |
||||
type: String, |
||||
default: "#eeeeee" |
||||
}, |
||||
fgColor: { |
||||
type: String, |
||||
default: "#666666" |
||||
}, |
||||
bgColorChecked: { |
||||
type: String, |
||||
default: "#007aff" |
||||
}, |
||||
fgColorChecked: { |
||||
type: String, |
||||
default: "#FFFFFF" |
||||
}, |
||||
circle: { |
||||
type: [Boolean, String], |
||||
default: false |
||||
}, |
||||
checked: { |
||||
type: Boolean, |
||||
default: false |
||||
}, |
||||
contentText: { |
||||
type: Object, |
||||
default () { |
||||
return { |
||||
contentDefault: "", |
||||
contentFav: "" |
||||
}; |
||||
} |
||||
}, |
||||
stat:{ |
||||
type: Boolean, |
||||
default: false |
||||
} |
||||
}, |
||||
computed: { |
||||
contentDefault() { |
||||
return this.contentText.contentDefault || t("uni-fav.collect") |
||||
}, |
||||
contentFav() { |
||||
return this.contentText.contentFav || t("uni-fav.collected") |
||||
}, |
||||
}, |
||||
watch: { |
||||
checked() { |
||||
if (uni.report && this.stat) { |
||||
if (this.checked) { |
||||
uni.report("收藏", "收藏"); |
||||
} else { |
||||
uni.report("取消收藏", "取消收藏"); |
||||
} |
||||
} |
||||
} |
||||
}, |
||||
methods: { |
||||
onClick() { |
||||
this.$emit("click"); |
||||
} |
||||
} |
||||
}; |
||||
</script> |
||||
|
||||
<style lang="scss" > |
||||
$fav-height: 25px; |
||||
|
||||
.uni-fav { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
flex-direction: row; |
||||
align-items: center; |
||||
justify-content: center; |
||||
width: 60px; |
||||
height: $fav-height; |
||||
line-height: $fav-height; |
||||
text-align: center; |
||||
border-radius: 3px; |
||||
/* #ifdef H5 */ |
||||
cursor: pointer; |
||||
/* #endif */ |
||||
} |
||||
|
||||
.uni-fav--circle { |
||||
border-radius: 30px; |
||||
} |
||||
|
||||
.uni-fav-star { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
height: $fav-height; |
||||
line-height: 24px; |
||||
margin-right: 3px; |
||||
align-items: center; |
||||
justify-content: center; |
||||
} |
||||
|
||||
.uni-fav-text { |
||||
/* #ifndef APP-NVUE */ |
||||
display: flex; |
||||
/* #endif */ |
||||
height: $fav-height; |
||||
line-height: $fav-height; |
||||
align-items: center; |
||||
justify-content: center; |
||||
font-size: 12px; |
||||
} |
||||
</style> |
@ -0,0 +1,89 @@ |
||||
{ |
||||
"id": "uni-fav", |
||||
"displayName": "uni-fav 收藏按钮", |
||||
"version": "1.2.1", |
||||
"description": " Fav 收藏组件,可自定义颜色、大小。", |
||||
"keywords": [ |
||||
"fav", |
||||
"uni-ui", |
||||
"uniui", |
||||
"收藏" |
||||
], |
||||
"repository": "https://github.com/dcloudio/uni-ui", |
||||
"engines": { |
||||
"HBuilderX": "" |
||||
}, |
||||
"directories": { |
||||
"example": "../../temps/example_temps" |
||||
}, |
||||
"dcloudext": { |
||||
"category": [ |
||||
"前端组件", |
||||
"通用组件" |
||||
], |
||||
"sale": { |
||||
"regular": { |
||||
"price": "0.00" |
||||
}, |
||||
"sourcecode": { |
||||
"price": "0.00" |
||||
} |
||||
}, |
||||
"contact": { |
||||
"qq": "" |
||||
}, |
||||
"declaration": { |
||||
"ads": "无", |
||||
"data": "无", |
||||
"permissions": "无" |
||||
}, |
||||
"npmurl": "https://www.npmjs.com/package/@dcloudio/uni-ui" |
||||
}, |
||||
"uni_modules": { |
||||
"dependencies": [ |
||||
"uni-scss", |
||||
"uni-icons" |
||||
], |
||||
"encrypt": [], |
||||
"platforms": { |
||||
"cloud": { |
||||
"tcb": "y", |
||||
"aliyun": "y" |
||||
}, |
||||
"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" |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in new issue