修改忽略文件
This commit is contained in:
		
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,2 +1,2 @@ | ||||
| unpackage/ | ||||
| uni_modules/ | ||||
| node_modules/ | ||||
| @@ -81,13 +81,13 @@ | ||||
|  | ||||
| <style> | ||||
| 	 | ||||
| 	.icon { | ||||
| 	/* .icon { | ||||
| 	      width: 80upx; | ||||
| 	      height: 80upx; | ||||
| 	      vertical-align: -0.15em; | ||||
| 	      fill: currentColor; | ||||
| 	      overflow: hidden; | ||||
| 	} | ||||
| 	} */ | ||||
| 	.home { | ||||
| 		height: 100vh; | ||||
| 		/* background-color: #060B26; */ | ||||
|   | ||||
							
								
								
									
										33
									
								
								uni_modules/uni-badge/changelog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								uni_modules/uni-badge/changelog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| ## 1.2.2(2023-01-28) | ||||
| - 修复 运行/打包 控制台警告问题 | ||||
| ## 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目录规范 | ||||
							
								
								
									
										268
									
								
								uni_modules/uni-badge/components/uni-badge/uni-badge.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										268
									
								
								uni_modules/uni-badge/components/uni-badge/uni-badge.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -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; | ||||
| 		font-feature-settings: "tnum"; | ||||
| 		min-width: 20px; | ||||
| 		/* #endif */ | ||||
| 		justify-content: center; | ||||
| 		flex-direction: row; | ||||
| 		height: 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-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> | ||||
							
								
								
									
										85
									
								
								uni_modules/uni-badge/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								uni_modules/uni-badge/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,85 @@ | ||||
| { | ||||
|   "id": "uni-badge", | ||||
|   "displayName": "uni-badge 数字角标", | ||||
|   "version": "1.2.2", | ||||
|   "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" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										10
									
								
								uni_modules/uni-badge/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								uni_modules/uni-badge/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| ## Badge 数字角标 | ||||
| > **组件名:uni-badge** | ||||
| > 代码块: `uBadge` | ||||
|  | ||||
| 数字角标一般和其它控件(列表、9宫格等)配合使用,用于进行数量提示,默认为实心灰色背景, | ||||
|  | ||||
| ### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-badge) | ||||
| #### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839  | ||||
|  | ||||
|  | ||||
							
								
								
									
										6
									
								
								uni_modules/uni-breadcrumb/changelog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								uni_modules/uni-breadcrumb/changelog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| ## 0.1.2(2022-06-08) | ||||
| - 修复 微信小程序 separator 不显示的Bug | ||||
| ## 0.1.1(2022-06-02) | ||||
| - 新增 支持 uni.scss 修改颜色 | ||||
| ## 0.1.0(2022-04-21) | ||||
| - 初始化 | ||||
| @@ -0,0 +1,121 @@ | ||||
| <template> | ||||
| 	<view class="uni-breadcrumb-item"> | ||||
| 		<view :class="{ | ||||
| 			'uni-breadcrumb-item--slot': true, | ||||
| 			'uni-breadcrumb-item--slot-link': to && currentPage !== to | ||||
| 			}" @click="navTo"> | ||||
| 			<slot /> | ||||
| 		</view> | ||||
| 		<i v-if="separatorClass" class="uni-breadcrumb-item--separator" :class="separatorClass" /> | ||||
| 		<text v-else class="uni-breadcrumb-item--separator">{{ separator }}</text> | ||||
| 	</view> | ||||
| </template> | ||||
| <script> | ||||
| 	/** | ||||
| 	 * BreadcrumbItem 面包屑导航子组件 | ||||
| 	 * @property {String/Object} to 路由跳转页面路径/对象 | ||||
| 	 * @property {Boolean} replace 在使用 to 进行路由跳转时,启用 replace 将不会向 history 添加新记录(仅 h5 支持) | ||||
| 	 */ | ||||
| 	export default { | ||||
| 		data() { | ||||
| 			return { | ||||
| 				currentPage: "" | ||||
| 			} | ||||
| 		}, | ||||
| 		options: { | ||||
| 			virtualHost: true | ||||
| 		}, | ||||
| 		props: { | ||||
| 			to: { | ||||
| 				type: String, | ||||
| 				default: '' | ||||
| 			}, | ||||
| 			replace:{ | ||||
| 				type: Boolean, | ||||
| 				default: false | ||||
| 			} | ||||
| 		}, | ||||
| 		inject: { | ||||
| 			uniBreadcrumb: { | ||||
| 				from: "uniBreadcrumb", | ||||
| 				default: null | ||||
| 			} | ||||
| 		}, | ||||
| 		created(){ | ||||
| 			const pages = getCurrentPages() | ||||
| 			const page = pages[pages.length-1] | ||||
|  | ||||
| 			if(page){ | ||||
| 				this.currentPage = `/${page.route}` | ||||
| 			} | ||||
| 		}, | ||||
| 		computed: { | ||||
| 			separator() { | ||||
| 				return this.uniBreadcrumb.separator | ||||
| 			}, | ||||
| 			separatorClass() { | ||||
| 				return this.uniBreadcrumb.separatorClass | ||||
| 			} | ||||
| 		}, | ||||
| 		methods: { | ||||
| 			navTo() { | ||||
| 				const { to } = this | ||||
|  | ||||
| 				if (!to || this.currentPage === to){ | ||||
| 					return | ||||
| 				} | ||||
|  | ||||
| 				if(this.replace){ | ||||
| 					uni.redirectTo({ | ||||
| 						url:to | ||||
| 					}) | ||||
| 				}else{ | ||||
| 					uni.navigateTo({ | ||||
| 						url:to | ||||
| 					}) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| </script> | ||||
| <style lang="scss"> | ||||
| 	$uni-primary: #2979ff !default; | ||||
| 	$uni-base-color: #6a6a6a !default; | ||||
| 	$uni-main-color: #3a3a3a !default; | ||||
| 	.uni-breadcrumb-item { | ||||
| 		display: flex; | ||||
| 		align-items: center; | ||||
| 		white-space: nowrap; | ||||
| 		font-size: 14px; | ||||
|  | ||||
| 		&--slot { | ||||
| 			color: $uni-base-color; | ||||
| 			padding: 0 10px; | ||||
|  | ||||
| 			&-link { | ||||
| 				color: $uni-main-color; | ||||
| 				font-weight: bold; | ||||
| 				/* #ifndef APP-NVUE */ | ||||
| 				cursor: pointer; | ||||
| 				/* #endif */ | ||||
|  | ||||
| 				&:hover { | ||||
| 					color: $uni-primary; | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		&--separator { | ||||
| 			font-size: 12px; | ||||
| 			color: $uni-base-color; | ||||
| 		} | ||||
|  | ||||
| 		&:first-child &--slot { | ||||
| 			padding-left: 0; | ||||
| 		} | ||||
| 		 | ||||
| 		&:last-child &--separator { | ||||
| 			display: none; | ||||
| 		} | ||||
| 	} | ||||
| </style> | ||||
| @@ -0,0 +1,41 @@ | ||||
| <template> | ||||
| 	<view class="uni-breadcrumb"> | ||||
| 		<slot /> | ||||
| 	</view> | ||||
| </template> | ||||
| <script> | ||||
| 	/** | ||||
| 	 * Breadcrumb 面包屑导航父组件 | ||||
| 	 * @description 显示当前页面的路径,快速返回之前的任意页面 | ||||
| 	 * @tutorial https://ext.dcloud.net.cn/plugin?id=xxx | ||||
| 	 * @property {String} separator 分隔符,默认为斜杠'/' | ||||
| 	 * @property {String} separatorClass 图标分隔符 class | ||||
| 	 */ | ||||
| 	export default { | ||||
| 		options: { | ||||
| 			virtualHost: true | ||||
| 		}, | ||||
| 		props: { | ||||
| 			separator: { | ||||
| 				type: String, | ||||
| 				default: '/' | ||||
| 			}, | ||||
| 			separatorClass: { | ||||
| 				type: String, | ||||
| 				default: '' | ||||
| 			} | ||||
| 		}, | ||||
|  | ||||
| 		provide() { | ||||
| 			return { | ||||
| 				uniBreadcrumb: this | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 	} | ||||
| </script> | ||||
| <style lang="scss"> | ||||
| 	.uni-breadcrumb { | ||||
| 		display: flex; | ||||
| 	} | ||||
| </style> | ||||
							
								
								
									
										88
									
								
								uni_modules/uni-breadcrumb/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										88
									
								
								uni_modules/uni-breadcrumb/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,88 @@ | ||||
| { | ||||
|   "id": "uni-breadcrumb", | ||||
|   "displayName": "uni-breadcrumb 面包屑", | ||||
|   "version": "0.1.2", | ||||
|   "description": "Breadcrumb  面包屑", | ||||
|   "keywords": [ | ||||
|     "uni-breadcrumb", | ||||
|     "breadcrumb", | ||||
|     "uni-ui", | ||||
|     "面包屑导航", | ||||
|     "面包屑" | ||||
| ], | ||||
|   "repository": "", | ||||
|   "engines": { | ||||
|     "HBuilderX": "^3.1.0" | ||||
|   }, | ||||
|   "directories": { | ||||
|     "example": "../../temps/example_temps" | ||||
|   }, | ||||
|   "dcloudext": { | ||||
|     "category": [ | ||||
|         "前端组件", | ||||
|         "通用组件" | ||||
|     ], | ||||
|     "sale": { | ||||
|       "regular": { | ||||
|         "price": "0.00" | ||||
|       }, | ||||
|       "sourcecode": { | ||||
|         "price": "0.00" | ||||
|       } | ||||
|     }, | ||||
|     "contact": { | ||||
|       "qq": "" | ||||
|     }, | ||||
|     "declaration": { | ||||
|       "ads": "无", | ||||
|       "data": "无", | ||||
|       "permissions": "无" | ||||
|     }, | ||||
|     "npmurl": "" | ||||
|   }, | ||||
|   "uni_modules": { | ||||
|     "dependencies": [], | ||||
|     "encrypt": [], | ||||
|     "platforms": { | ||||
|       "cloud": { | ||||
|         "tcb": "y", | ||||
|         "aliyun": "y" | ||||
|       }, | ||||
|       "client": { | ||||
|         "Vue": { | ||||
|           "vue2": "y", | ||||
|           "vue3": "y" | ||||
|         }, | ||||
|         "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", | ||||
|           "阿里": "u", | ||||
|           "百度": "u", | ||||
|           "字节跳动": "u", | ||||
|         "QQ": "u", | ||||
|         "京东": "u" | ||||
|         }, | ||||
|         "快应用": { | ||||
|           "华为": "u", | ||||
|           "联盟": "u" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										66
									
								
								uni_modules/uni-breadcrumb/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								uni_modules/uni-breadcrumb/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,66 @@ | ||||
|  | ||||
| ## breadcrumb 面包屑导航 | ||||
| > **组件名:uni-breadcrumb** | ||||
| > 代码块: `ubreadcrumb` | ||||
|  | ||||
| 显示当前页面的路径,快速返回之前的任意页面。 | ||||
|  | ||||
| ### 安装方式 | ||||
|  | ||||
| 本组件符合[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 | ||||
| <uni-breadcrumb separator="/"> | ||||
| 	<uni-breadcrumb-item v-for="(route,index) in routes" :key="index" :to="route.to">{{route.name}}</uni-breadcrumb-item> | ||||
| </uni-breadcrumb> | ||||
| ``` | ||||
|  | ||||
| ```js | ||||
| export default { | ||||
| 		name: "uni-stat-breadcrumb", | ||||
| 		data() { | ||||
| 			return { | ||||
| 				routes: [{ | ||||
| 					to: '/A', | ||||
| 					name: 'A页面' | ||||
| 				}, { | ||||
| 					to: '/B', | ||||
| 					name: 'B页面' | ||||
| 				}, { | ||||
| 					to: '/C', | ||||
| 					name: 'C页面' | ||||
| 				}] | ||||
| 			}; | ||||
| 		} | ||||
| 	} | ||||
| ``` | ||||
|  | ||||
|  | ||||
| ## API | ||||
|  | ||||
| ### Breadcrumb Props | ||||
|  | ||||
| |属性名			|类型	|默认值	|说明				| | ||||
| |:-:			|:-:	|:-:	|:-:				| | ||||
| |separator		|String	|斜杠'/' |分隔符				| | ||||
| |separatorClass	|String	|		|图标分隔符 class	    | | ||||
|  | ||||
| ### Breadcrumb Item Props | ||||
|  | ||||
| |属性名	|类型			|默认值	|说明																			| | ||||
| |:-:	|:-:			|:-:	|:-:																			| | ||||
| |to		|String     	|		|路由跳转页面路径           														| | ||||
| |replace|Boolean		|		|在使用 to 进行路由跳转时,启用 replace 将不会向 history 添加新记录(仅 h5 支持)         | | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ## 组件示例 | ||||
|  | ||||
| 点击查看:[https://hellouniapp.dcloud.net.cn/pages/extUI/breadcrumb/breadcrumb](https://hellouniapp.dcloud.net.cn/pages/extUI/breadcrumb/breadcrumb) | ||||
							
								
								
									
										26
									
								
								uni_modules/uni-calendar/changelog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								uni_modules/uni-calendar/changelog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| ## 1.4.10(2023-04-10) | ||||
| - 修复 某些情况 monthSwitch 未触发的Bug | ||||
| ## 1.4.9(2023-02-02) | ||||
| - 修复 某些情况切换月份错误的Bug | ||||
| ## 1.4.8(2023-01-30) | ||||
| - 修复 某些情况切换月份错误的Bug [详情](https://ask.dcloud.net.cn/question/161964) | ||||
| ## 1.4.7(2022-09-16) | ||||
| - 优化 支持使用 uni-scss 控制主题色 | ||||
| ## 1.4.6(2022-09-08) | ||||
| - 修复 表头年月切换,导致改变当前日期为选择月1号,且未触发change事件的Bug | ||||
| ## 1.4.5(2022-02-25) | ||||
| - 修复 条件编译 nvue 不支持的 css 样式的Bug | ||||
| ## 1.4.4(2022-02-25) | ||||
| - 修复 条件编译 nvue 不支持的 css 样式的Bug | ||||
| ## 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目录规范 | ||||
							
								
								
									
										546
									
								
								uni_modules/uni-calendar/components/uni-calendar/calendar.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										546
									
								
								uni_modules/uni-calendar/components/uni-calendar/calendar.js
									
									
									
									
									
										Normal file
									
								
							| @@ -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,187 @@ | ||||
| <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 i18nMessages from './i18n/index.js' | ||||
| 	const {	t	} = initVueI18n(i18nMessages) | ||||
|  | ||||
| 	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,566 @@ | ||||
| <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 i18nMessages from './i18n/index.js' | ||||
| 	const {	t	} = initVueI18n(i18nMessages) | ||||
|  | ||||
| 	/** | ||||
| 	 * 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({ | ||||
| 				selected: this.selected, | ||||
| 				startDate: this.startDate, | ||||
| 				endDate: this.endDate, | ||||
| 				range: this.range, | ||||
| 			}) | ||||
| 			this.init(this.date) | ||||
| 		}, | ||||
| 		methods: { | ||||
| 			// 取消穿透 | ||||
| 			clean() {}, | ||||
| 			bindDateChange(e) { | ||||
| 				const value = e.detail.value + '-1' | ||||
| 				this.setDate(value) | ||||
|  | ||||
| 				const { year,month } = this.cale.getDate(value) | ||||
|         this.$emit('monthSwitch', { | ||||
|             year, | ||||
|             month | ||||
|         }) | ||||
| 			}, | ||||
| 			/** | ||||
| 			 * 初始化日期显示 | ||||
| 			 * @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() { | ||||
| 				const nowYearMonth = `${this.nowDate.year}-${this.nowDate.month}` | ||||
| 				const date = this.cale.getDate(new Date()) | ||||
|         const todayYearMonth = `${date.year}-${date.month}` | ||||
|  | ||||
|         if(nowYearMonth !== todayYearMonth) { | ||||
|           this.monthSwitch() | ||||
|         } | ||||
|  | ||||
| 				this.init(date.fullDate) | ||||
| 				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; | ||||
| 	} | ||||
|  | ||||
| 	.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> | ||||
							
								
								
									
										360
									
								
								uni_modules/uni-calendar/components/uni-calendar/util.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										360
									
								
								uni_modules/uni-calendar/components/uni-calendar/util.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,360 @@ | ||||
| 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 && AddDayCount>0) { | ||||
| 					dd.setDate(dd.getDate() + AddDayCount) | ||||
| 				} else { | ||||
| 					const preMonth = dd.getMonth() | ||||
| 					dd.setMonth(preMonth + AddDayCount) // 获取AddDayCount天后的日期 | ||||
| 					const nextMonth = dd.getMonth() | ||||
| 					// 处理 pre 切换月份目标月份为2月没有当前日(30 31) 切换错误问题 | ||||
| 					if(AddDayCount<0 && preMonth!==0 && nextMonth-preMonth>AddDayCount){ | ||||
| 						dd.setMonth(nextMonth+(nextMonth-preMonth+AddDayCount)) | ||||
| 					} | ||||
| 					// 处理 next 切换月份目标月份为2月没有当前日(30 31) 切换错误问题 | ||||
| 					if(AddDayCount>0 && nextMonth-preMonth>AddDayCount){ | ||||
| 						dd.setMonth(nextMonth-(nextMonth-preMonth-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 | ||||
							
								
								
									
										85
									
								
								uni_modules/uni-calendar/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								uni_modules/uni-calendar/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,85 @@ | ||||
| { | ||||
|   "id": "uni-calendar", | ||||
|   "displayName": "uni-calendar 日历", | ||||
|   "version": "1.4.10", | ||||
|   "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" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										103
									
								
								uni_modules/uni-calendar/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										103
									
								
								uni_modules/uni-calendar/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -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) | ||||
							
								
								
									
										26
									
								
								uni_modules/uni-card/changelog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								uni_modules/uni-card/changelog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -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目录规范 | ||||
							
								
								
									
										272
									
								
								uni_modules/uni-card/components/uni-card/uni-card.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										272
									
								
								uni_modules/uni-card/components/uni-card/uni-card.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,272 @@ | ||||
| <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')"> | ||||
| 					<slot name="extra"> | ||||
| 						<text class="uni-card__header-extra-text">{{ extra }}</text> | ||||
| 					</slot> | ||||
| 				</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> | ||||
							
								
								
									
										90
									
								
								uni_modules/uni-card/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								uni_modules/uni-card/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -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" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										12
									
								
								uni_modules/uni-card/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								uni_modules/uni-card/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
|  | ||||
|  | ||||
| ## Card 卡片 | ||||
| > **组件名:uni-card** | ||||
| > 代码块: `uCard` | ||||
|  | ||||
| 卡片视图组件。 | ||||
|  | ||||
| ### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-card) | ||||
| #### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839  | ||||
|  | ||||
|  | ||||
							
								
								
									
										36
									
								
								uni_modules/uni-collapse/changelog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								uni_modules/uni-collapse/changelog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -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> | ||||
							
								
								
									
										89
									
								
								uni_modules/uni-collapse/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								uni_modules/uni-collapse/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -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" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										12
									
								
								uni_modules/uni-collapse/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								uni_modules/uni-collapse/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -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  | ||||
							
								
								
									
										15
									
								
								uni_modules/uni-combox/changelog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								uni_modules/uni-combox/changelog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -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目录规范 | ||||
							
								
								
									
										294
									
								
								uni_modules/uni-combox/components/uni-combox/uni-combox.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										294
									
								
								uni_modules/uni-combox/components/uni-combox/uni-combox.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,294 @@ | ||||
| <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" @scroll="onScroll"> | ||||
| 				<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 class="uni-combox__mask" v-show="showSelector" @click="showSelector = false"></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: '', | ||||
| 				blurTimer:null, | ||||
| 			} | ||||
| 		}, | ||||
| 		computed: { | ||||
| 			labelStyle() { | ||||
| 				if (this.labelWidth === 'auto') { | ||||
| 					return "" | ||||
| 				} | ||||
| 				return `width: ${this.labelWidth}` | ||||
| 			}, | ||||
| 			filterCandidates() { | ||||
| 				if (this.inputVal !== 0 && !this.inputVal) { | ||||
| 					return this.candidates | ||||
| 				} | ||||
| 				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() { | ||||
| 				this.blurTimer = setTimeout(() => { | ||||
| 					this.showSelector = false | ||||
| 				}, 153) | ||||
| 			}, | ||||
| 			onScroll(){ // 滚动时将blur的定时器关掉 | ||||
| 				if(this.blurTimer) { | ||||
| 					clearTimeout(this.blurTimer) | ||||
| 					this.blurTimer = null | ||||
| 				} | ||||
| 			}, | ||||
| 			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"> | ||||
| 	.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: 3; | ||||
| 		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; | ||||
| 	} | ||||
|  | ||||
| 	.uni-combox__mask { | ||||
| 		width:100%; | ||||
| 		height:100%; | ||||
| 		position: fixed; | ||||
| 		top: 0; | ||||
| 		left: 0; | ||||
|     z-index: 1; | ||||
| 	} | ||||
| </style> | ||||
							
								
								
									
										90
									
								
								uni_modules/uni-combox/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								uni_modules/uni-combox/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -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" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										11
									
								
								uni_modules/uni-combox/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								uni_modules/uni-combox/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
|  | ||||
|  | ||||
| ## Combox 组合框 | ||||
| > **组件名:uni-combox** | ||||
| > 代码块: `uCombox` | ||||
|  | ||||
|  | ||||
| 组合框组件。 | ||||
|  | ||||
| ### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-combox) | ||||
| #### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839  | ||||
							
								
								
									
										24
									
								
								uni_modules/uni-countdown/changelog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								uni_modules/uni-countdown/changelog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -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,267 @@ | ||||
| <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 | ||||
| 			}, | ||||
|       zeroPad: { | ||||
| 				type: Boolean, | ||||
| 				default: true | ||||
| 			} | ||||
| 		}, | ||||
| 		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() | ||||
| 				} | ||||
|         day = (day < 10 && this.zeroPad) ? `0${day}` : day | ||||
| 				hour = (hour < 10 && this.zeroPad) ? `0${hour}` : hour | ||||
| 				minute = (minute < 10 && this.zeroPad) ? `0${minute}` : minute | ||||
| 				second = (second < 10 && this.zeroPad) ? `0${second}` : 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> | ||||
							
								
								
									
										86
									
								
								uni_modules/uni-countdown/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										86
									
								
								uni_modules/uni-countdown/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -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" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										10
									
								
								uni_modules/uni-countdown/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								uni_modules/uni-countdown/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
|  | ||||
|  | ||||
| ## CountDown 倒计时 | ||||
| > **组件名:uni-countdown** | ||||
| > 代码块: `uCountDown` | ||||
|  | ||||
| 倒计时组件。 | ||||
|  | ||||
| ### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-countdown) | ||||
| #### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839  | ||||
							
								
								
									
										45
									
								
								uni_modules/uni-data-checkbox/changelog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								uni_modules/uni-data-checkbox/changelog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -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> | ||||
							
								
								
									
										84
									
								
								uni_modules/uni-data-checkbox/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								uni_modules/uni-data-checkbox/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -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" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										18
									
								
								uni_modules/uni-data-checkbox/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								uni_modules/uni-data-checkbox/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -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  | ||||
							
								
								
									
										75
									
								
								uni_modules/uni-data-picker/changelog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								uni_modules/uni-data-picker/changelog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,75 @@ | ||||
| ## 1.1.2(2023-04-11) | ||||
| - 修复 更改 modelValue 报错的 bug | ||||
| - 修复 v-for 未使用 key 值控制台 warning | ||||
| ## 1.1.1(2023-02-21) | ||||
| - 修复代码合并时引发 value 属性为空时不渲染数据的问题 | ||||
| ## 1.1.0(2023-02-15) | ||||
| - 修复 localdata 不支持动态更新的bug | ||||
| ## 1.0.9(2023-02-15) | ||||
| - 修复 localdata 不支持动态更新的bug | ||||
| ## 1.0.8(2022-09-16) | ||||
| - 可以使用 uni-scss 控制主题色 | ||||
| ## 1.0.7(2022-07-06) | ||||
| - 优化 pc端图标位置不正确的问题 | ||||
| ## 1.0.6(2022-07-05) | ||||
| - 优化 显示样式 | ||||
| ## 1.0.5(2022-07-04) | ||||
| - 修复 uni-data-picker 在 uni-forms-item 中宽度不正确的bug | ||||
| ## 1.0.4(2022-04-19) | ||||
| - 修复 字节小程序 本地数据无法选择下一级的Bug | ||||
| ## 1.0.3(2022-02-25) | ||||
| - 修复 nvue 不支持的 v-show 的 bug | ||||
| ## 1.0.2(2022-02-25) | ||||
| - 修复 条件编译 nvue 不支持的 css 样式 | ||||
| ## 1.0.1(2021-11-23) | ||||
| - 修复 由上个版本引发的map、v-model等属性不生效的bug | ||||
| ## 1.0.0(2021-11-19) | ||||
| - 优化 组件 UI,并提供设计资源,详见:[https://uniapp.dcloud.io/component/uniui/resource](https://uniapp.dcloud.io/component/uniui/resource) | ||||
| - 文档迁移,详见:[https://uniapp.dcloud.io/component/uniui/uni-data-picker](https://uniapp.dcloud.io/component/uniui/uni-data-picker) | ||||
| ## 0.4.9(2021-10-28) | ||||
| - 修复 VUE2 v-model 概率无效的 bug | ||||
| ## 0.4.8(2021-10-27) | ||||
| - 修复 v-model 概率无效的 bug | ||||
| ## 0.4.7(2021-10-25) | ||||
| - 新增 属性 spaceInfo 服务空间配置 HBuilderX 3.2.11+ | ||||
| - 修复 树型 uniCloud 数据类型为 int 时报错的 bug | ||||
| ## 0.4.6(2021-10-19) | ||||
| - 修复 非 VUE3 v-model 为 0 时无法选中的 bug | ||||
| ## 0.4.5(2021-09-26) | ||||
| - 新增 清除已选项的功能(通过 clearIcon 属性配置是否显示按钮),同时提供 clear 方法以供调用,二者等效 | ||||
| - 修复 readonly 为 true 时报错的 bug | ||||
| ## 0.4.4(2021-09-26) | ||||
| - 修复 上一版本造成的 map 属性失效的 bug | ||||
| - 新增 ellipsis 属性,支持配置 tab 选项长度过长时是否自动省略 | ||||
| ## 0.4.3(2021-09-24) | ||||
| - 修复 某些情况下级联未触发的 bug | ||||
| ## 0.4.2(2021-09-23) | ||||
| - 新增 提供 show 和 hide 方法,开发者可以通过 ref 调用 | ||||
| - 新增 选项内容过长自动添加省略号 | ||||
| ## 0.4.1(2021-09-15) | ||||
| - 新增 map 属性 字段映射,将 text/value 映射到数据中的其他字段 | ||||
| ## 0.4.0(2021-07-13) | ||||
| - 组件兼容 vue3,如何创建 vue3 项目,详见 [uni-app 项目支持 vue3 介绍](https://ask.dcloud.net.cn/article/37834) | ||||
| ## 0.3.5(2021-06-04) | ||||
| - 修复 无法加载云端数据的问题 | ||||
| ## 0.3.4(2021-05-28) | ||||
| - 修复 v-model 无效问题 | ||||
| - 修复 loaddata 为空数据组时加载时间过长问题 | ||||
| - 修复 上个版本引出的本地数据无法选择带有 children 的 2 级节点 | ||||
| ## 0.3.3(2021-05-12) | ||||
| - 新增 组件示例地址 | ||||
| ## 0.3.2(2021-04-22) | ||||
| - 修复 非树形数据有 where 属性查询报错的问题 | ||||
| ## 0.3.1(2021-04-15) | ||||
| - 修复 本地数据概率无法回显时问题 | ||||
| ## 0.3.0(2021-04-07) | ||||
| - 新增 支持云端非树形表结构数据 | ||||
| - 修复 根节点 parent_field 字段等于 null 时选择界面错乱问题 | ||||
| ## 0.2.0(2021-03-15) | ||||
| - 修复 nodeclick、popupopened、popupclosed 事件无法触发的问题 | ||||
| ## 0.1.9(2021-03-09) | ||||
| - 修复 微信小程序某些情况下无法选择的问题 | ||||
| ## 0.1.8(2021-02-05) | ||||
| - 优化 部分样式在 nvue 上的兼容表现 | ||||
| ## 0.1.7(2021-02-05) | ||||
| - 调整为 uni_modules 目录规范 | ||||
| @@ -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,551 @@ | ||||
| <template> | ||||
|   <view class="uni-data-tree"> | ||||
|     <view class="uni-data-tree-input" @click="handleInput"> | ||||
|       <slot :options="options" :data="inputSelected" :error="errorMessage"> | ||||
|         <view class="input-value" :class="{'input-value-border': border}"> | ||||
|           <text v-if="errorMessage" class="selected-area error-text">{{errorMessage}}</text> | ||||
|           <view v-else-if="loading && !isOpened" class="selected-area"> | ||||
|             <uni-load-more class="load-more" :contentText="loadMore" status="loading"></uni-load-more> | ||||
|           </view> | ||||
|           <scroll-view v-else-if="inputSelected.length" class="selected-area" scroll-x="true"> | ||||
|             <view class="selected-list"> | ||||
|               <view class="selected-item" v-for="(item,index) in inputSelected" :key="index"> | ||||
|                 <text class="text-color">{{item.text}}</text><text v-if="index<inputSelected.length-1" | ||||
|                   class="input-split-line">{{split}}</text> | ||||
|               </view> | ||||
|             </view> | ||||
|           </scroll-view> | ||||
|           <text v-else class="selected-area placeholder">{{placeholder}}</text> | ||||
|           <view v-if="clearIcon && !readonly && inputSelected.length" class="icon-clear" @click.stop="clear"> | ||||
|             <uni-icons type="clear" color="#c0c4cc" size="24"></uni-icons> | ||||
|           </view> | ||||
|           <view class="arrow-area" v-if="(!clearIcon || !inputSelected.length) && !readonly "> | ||||
|             <view class="input-arrow"></view> | ||||
|           </view> | ||||
|         </view> | ||||
|       </slot> | ||||
|     </view> | ||||
|     <view class="uni-data-tree-cover" v-if="isOpened" @click="handleClose"></view> | ||||
|     <view class="uni-data-tree-dialog" v-if="isOpened"> | ||||
|       <view class="uni-popper__arrow"></view> | ||||
|       <view class="dialog-caption"> | ||||
|         <view class="title-area"> | ||||
|           <text class="dialog-title">{{popupTitle}}</text> | ||||
|         </view> | ||||
|         <view class="dialog-close" @click="handleClose"> | ||||
|           <view class="dialog-close-plus" data-id="close"></view> | ||||
|           <view class="dialog-close-plus dialog-close-rotate" data-id="close"></view> | ||||
|         </view> | ||||
|       </view> | ||||
|       <data-picker-view class="picker-view" ref="pickerView" v-model="dataValue" :localdata="localdata" | ||||
|         :preload="preload" :collection="collection" :field="field" :orderby="orderby" :where="where" | ||||
|         :step-searh="stepSearh" :self-field="selfField" :parent-field="parentField" :managed-mode="true" :map="map" | ||||
|         :ellipsis="ellipsis" @change="onchange" @datachange="ondatachange" @nodeclick="onnodeclick"> | ||||
|       </data-picker-view> | ||||
|     </view> | ||||
|   </view> | ||||
| </template> | ||||
|  | ||||
| <script> | ||||
|   import dataPicker from "../uni-data-pickerview/uni-data-picker.js" | ||||
|   import DataPickerView from "../uni-data-pickerview/uni-data-pickerview.vue" | ||||
|  | ||||
|   /** | ||||
|    * DataPicker 级联选择 | ||||
|    * @description 支持单列、和多列级联选择。列数没有限制,如果屏幕显示不全,顶部tab区域会左右滚动。 | ||||
|    * @tutorial https://ext.dcloud.net.cn/plugin?id=3796 | ||||
|    * @property {String} popup-title 弹出窗口标题 | ||||
|    * @property {Array} localdata 本地数据,参考 | ||||
|    * @property {Boolean} border = [true|false] 是否有边框 | ||||
|    * @property {Boolean} readonly = [true|false] 是否仅读 | ||||
|    * @property {Boolean} preload = [true|false] 是否预加载数据 | ||||
|    * @value true 开启预加载数据,点击弹出窗口后显示已加载数据 | ||||
|    * @value false 关闭预加载数据,点击弹出窗口后开始加载数据 | ||||
|    * @property {Boolean} step-searh = [true|false] 是否分布查询 | ||||
|    * @value true 启用分布查询,仅查询当前选中节点 | ||||
|    * @value false 关闭分布查询,一次查询出所有数据 | ||||
|    * @property {String|DBFieldString} self-field 分布查询当前字段名称 | ||||
|    * @property {String|DBFieldString} parent-field 分布查询父字段名称 | ||||
|    * @property {String|DBCollectionString} collection 表名 | ||||
|    * @property {String|DBFieldString} field 查询字段,多个字段用 `,` 分割 | ||||
|    * @property {String} orderby 排序字段及正序倒叙设置 | ||||
|    * @property {String|JQLString} where 查询条件 | ||||
|    * @event {Function} popupshow 弹出的选择窗口打开时触发此事件 | ||||
|    * @event {Function} popuphide 弹出的选择窗口关闭时触发此事件 | ||||
|    */ | ||||
|   export default { | ||||
|     name: 'UniDataPicker', | ||||
|     emits: ['popupopened', 'popupclosed', 'nodeclick', 'input', 'change', 'update:modelValue','inputclick'], | ||||
|     mixins: [dataPicker], | ||||
|     components: { | ||||
|       DataPickerView | ||||
|     }, | ||||
|     props: { | ||||
|       options: { | ||||
|         type: [Object, Array], | ||||
|         default () { | ||||
|           return {} | ||||
|         } | ||||
|       }, | ||||
|       popupTitle: { | ||||
|         type: String, | ||||
|         default: '请选择' | ||||
|       }, | ||||
|       placeholder: { | ||||
|         type: String, | ||||
|         default: '请选择' | ||||
|       }, | ||||
|       heightMobile: { | ||||
|         type: String, | ||||
|         default: '' | ||||
|       }, | ||||
|       readonly: { | ||||
|         type: Boolean, | ||||
|         default: false | ||||
|       }, | ||||
|       clearIcon: { | ||||
|         type: Boolean, | ||||
|         default: true | ||||
|       }, | ||||
|       border: { | ||||
|         type: Boolean, | ||||
|         default: true | ||||
|       }, | ||||
|       split: { | ||||
|         type: String, | ||||
|         default: '/' | ||||
|       }, | ||||
|       ellipsis: { | ||||
|         type: Boolean, | ||||
|         default: true | ||||
|       } | ||||
|     }, | ||||
|     data() { | ||||
|       return { | ||||
|         isOpened: false, | ||||
|         inputSelected: [] | ||||
|       } | ||||
|     }, | ||||
|     created() { | ||||
|       this.$nextTick(() => { | ||||
|         this.load(); | ||||
|       }) | ||||
|     }, | ||||
|     watch: { | ||||
| 			localdata: { | ||||
| 				handler() { | ||||
| 					this.load() | ||||
| 				}, | ||||
|         deep: true | ||||
| 			}, | ||||
|     }, | ||||
|     methods: { | ||||
|       clear() { | ||||
|         this._dispatchEvent([]); | ||||
|       }, | ||||
|       onPropsChange() { | ||||
|         this._treeData = []; | ||||
|         this.selectedIndex = 0; | ||||
|  | ||||
|         this.load(); | ||||
|       }, | ||||
|       load() { | ||||
|         if (this.readonly) { | ||||
|           this._processReadonly(this.localdata, this.dataValue); | ||||
|           return; | ||||
|         } | ||||
|  | ||||
|         // 回显本地数据 | ||||
|         if (this.isLocalData) { | ||||
|           this.loadData(); | ||||
|           this.inputSelected = this.selected.slice(0); | ||||
|         } else if (this.isCloudDataList || this.isCloudDataTree) { // 回显 Cloud 数据 | ||||
|           this.loading = true; | ||||
|           this.getCloudDataValue().then((res) => { | ||||
|             this.loading = false; | ||||
|             this.inputSelected = res; | ||||
|           }).catch((err) => { | ||||
|             this.loading = false; | ||||
|             this.errorMessage = err; | ||||
|           }) | ||||
|         } | ||||
|       }, | ||||
|       show() { | ||||
|         this.isOpened = true | ||||
|         setTimeout(() => { | ||||
|           this.$refs.pickerView.updateData({ | ||||
|             treeData: this._treeData, | ||||
|             selected: this.selected, | ||||
|             selectedIndex: this.selectedIndex | ||||
|           }) | ||||
|         }, 200) | ||||
|         this.$emit('popupopened') | ||||
|       }, | ||||
|       hide() { | ||||
|         this.isOpened = false | ||||
|         this.$emit('popupclosed') | ||||
|       }, | ||||
|       handleInput() { | ||||
|         if (this.readonly) { | ||||
| 					this.$emit('inputclick') | ||||
|           return | ||||
|         } | ||||
|         this.show() | ||||
|       }, | ||||
|       handleClose(e) { | ||||
|         this.hide() | ||||
|       }, | ||||
|       onnodeclick(e) { | ||||
|         this.$emit('nodeclick', e) | ||||
|       }, | ||||
|       ondatachange(e) { | ||||
|         this._treeData = this.$refs.pickerView._treeData | ||||
|       }, | ||||
|       onchange(e) { | ||||
|         this.hide() | ||||
|         this.$nextTick(() => { | ||||
|           this.inputSelected = e; | ||||
|         }) | ||||
|         this._dispatchEvent(e) | ||||
|       }, | ||||
|       _processReadonly(dataList, value) { | ||||
|         var isTree = dataList.findIndex((item) => { | ||||
|           return item.children | ||||
|         }) | ||||
|         if (isTree > -1) { | ||||
|           let inputValue | ||||
|           if (Array.isArray(value)) { | ||||
|             inputValue = value[value.length - 1] | ||||
|             if (typeof inputValue === 'object' && inputValue.value) { | ||||
|               inputValue = inputValue.value | ||||
|             } | ||||
|           } else { | ||||
|             inputValue = value | ||||
|           } | ||||
|           this.inputSelected = this._findNodePath(inputValue, this.localdata) | ||||
|           return | ||||
|         } | ||||
|  | ||||
|         if (!this.hasValue) { | ||||
|           this.inputSelected = [] | ||||
|           return | ||||
|         } | ||||
|  | ||||
|         let result = [] | ||||
|         for (let i = 0; i < value.length; i++) { | ||||
|           var val = value[i] | ||||
|           var item = dataList.find((v) => { | ||||
|             return v.value == val | ||||
|           }) | ||||
|           if (item) { | ||||
|             result.push(item) | ||||
|           } | ||||
|         } | ||||
|         if (result.length) { | ||||
|           this.inputSelected = result | ||||
|         } | ||||
|       }, | ||||
|       _filterForArray(data, valueArray) { | ||||
|         var result = [] | ||||
|         for (let i = 0; i < valueArray.length; i++) { | ||||
|           var value = valueArray[i] | ||||
|           var found = data.find((item) => { | ||||
|             return item.value == value | ||||
|           }) | ||||
|           if (found) { | ||||
|             result.push(found) | ||||
|           } | ||||
|         } | ||||
|         return result | ||||
|       }, | ||||
|       _dispatchEvent(selected) { | ||||
|         let item = {} | ||||
|         if (selected.length) { | ||||
|           var value = new Array(selected.length) | ||||
|           for (var i = 0; i < selected.length; i++) { | ||||
|             value[i] = selected[i].value | ||||
|           } | ||||
|           item = selected[selected.length - 1] | ||||
|         } else { | ||||
|           item.value = '' | ||||
|         } | ||||
|         if (this.formItem) { | ||||
|           this.formItem.setValue(item.value) | ||||
|         } | ||||
|  | ||||
|         this.$emit('input', item.value) | ||||
|         this.$emit('update:modelValue', item.value) | ||||
|         this.$emit('change', { | ||||
|           detail: { | ||||
|             value: selected | ||||
|           } | ||||
|         }) | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| </script> | ||||
|  | ||||
| <style> | ||||
|   .uni-data-tree { | ||||
|     flex: 1; | ||||
|     position: relative; | ||||
|     font-size: 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; | ||||
|     /* #ifndef APP-NVUE */ | ||||
|     top: 20%; | ||||
|     /* #endif */ | ||||
|     /* #ifdef APP-NVUE */ | ||||
|     top: 200px; | ||||
|     /* #endif */ | ||||
|     right: 0; | ||||
|     bottom: 0; | ||||
|     background-color: #FFFFFF; | ||||
|     border-top-left-radius: 10px; | ||||
|     border-top-right-radius: 10px; | ||||
|     /* #ifndef APP-NVUE */ | ||||
|     display: flex; | ||||
|     /* #endif */ | ||||
|     flex-direction: column; | ||||
|     z-index: 102; | ||||
|     overflow: hidden; | ||||
|     /* #ifdef APP-NVUE */ | ||||
|     width: 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,622 @@ | ||||
| export default { | ||||
|   props: { | ||||
|     localdata: { | ||||
|       type: [Array, Object], | ||||
|       default () { | ||||
|         return [] | ||||
|       } | ||||
|     }, | ||||
|     spaceInfo: { | ||||
|       type: Object, | ||||
|       default () { | ||||
|         return {} | ||||
|       } | ||||
|     }, | ||||
|     collection: { | ||||
|       type: String, | ||||
|       default: '' | ||||
|     }, | ||||
|     action: { | ||||
|       type: String, | ||||
|       default: '' | ||||
|     }, | ||||
|     field: { | ||||
|       type: String, | ||||
|       default: '' | ||||
|     }, | ||||
|     orderby: { | ||||
|       type: String, | ||||
|       default: '' | ||||
|     }, | ||||
|     where: { | ||||
|       type: [String, Object], | ||||
|       default: '' | ||||
|     }, | ||||
|     pageData: { | ||||
|       type: String, | ||||
|       default: 'add' | ||||
|     }, | ||||
|     pageCurrent: { | ||||
|       type: Number, | ||||
|       default: 1 | ||||
|     }, | ||||
|     pageSize: { | ||||
|       type: Number, | ||||
|       default: 500 | ||||
|     }, | ||||
|     getcount: { | ||||
|       type: [Boolean, String], | ||||
|       default: false | ||||
|     }, | ||||
|     getone: { | ||||
|       type: [Boolean, String], | ||||
|       default: false | ||||
|     }, | ||||
|     gettree: { | ||||
|       type: [Boolean, String], | ||||
|       default: false | ||||
|     }, | ||||
|     manual: { | ||||
|       type: Boolean, | ||||
|       default: false | ||||
|     }, | ||||
|     value: { | ||||
|       type: [Array, String, Number], | ||||
|       default () { | ||||
|         return [] | ||||
|       } | ||||
|     }, | ||||
|     modelValue: { | ||||
|       type: [Array, String, Number], | ||||
|       default () { | ||||
|         return [] | ||||
|       } | ||||
|     }, | ||||
|     preload: { | ||||
|       type: Boolean, | ||||
|       default: false | ||||
|     }, | ||||
|     stepSearh: { | ||||
|       type: Boolean, | ||||
|       default: true | ||||
|     }, | ||||
|     selfField: { | ||||
|       type: String, | ||||
|       default: '' | ||||
|     }, | ||||
|     parentField: { | ||||
|       type: String, | ||||
|       default: '' | ||||
|     }, | ||||
|     multiple: { | ||||
|       type: Boolean, | ||||
|       default: false | ||||
|     }, | ||||
|     map: { | ||||
|       type: Object, | ||||
|       default () { | ||||
|         return { | ||||
|           text: "text", | ||||
|           value: "value" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   }, | ||||
|   data() { | ||||
|     return { | ||||
|       loading: false, | ||||
|       errorMessage: '', | ||||
|       loadMore: { | ||||
|         contentdown: '', | ||||
|         contentrefresh: '', | ||||
|         contentnomore: '' | ||||
|       }, | ||||
|       dataList: [], | ||||
|       selected: [], | ||||
|       selectedIndex: 0, | ||||
|       page: { | ||||
|         current: this.pageCurrent, | ||||
|         size: this.pageSize, | ||||
|         count: 0 | ||||
|       } | ||||
|     } | ||||
|   }, | ||||
|   computed: { | ||||
|     isLocalData() { | ||||
|       return !this.collection.length; | ||||
|     }, | ||||
|     isCloudData() { | ||||
|       return this.collection.length > 0; | ||||
|     }, | ||||
|     isCloudDataList() { | ||||
|       return (this.isCloudData && (!this.parentField && !this.selfField)); | ||||
|     }, | ||||
|     isCloudDataTree() { | ||||
|       return (this.isCloudData && this.parentField && this.selfField); | ||||
|     }, | ||||
|     dataValue() { | ||||
|       let isModelValue = Array.isArray(this.modelValue) ? (this.modelValue.length > 0) : (this.modelValue !== null || | ||||
|         this.modelValue !== undefined); | ||||
|       return isModelValue ? this.modelValue : this.value; | ||||
|     }, | ||||
|     hasValue() { | ||||
|       if (typeof this.dataValue === 'number') { | ||||
|         return true | ||||
|       } | ||||
|       return (this.dataValue != null) && (this.dataValue.length > 0) | ||||
|     } | ||||
|   }, | ||||
|   created() { | ||||
|     this.$watch(() => { | ||||
|       var al = []; | ||||
|       ['pageCurrent', | ||||
|         'pageSize', | ||||
|         'spaceInfo', | ||||
|         'value', | ||||
|         'modelValue', | ||||
|         'localdata', | ||||
|         'collection', | ||||
|         'action', | ||||
|         'field', | ||||
|         'orderby', | ||||
|         'where', | ||||
|         'getont', | ||||
|         'getcount', | ||||
|         'gettree' | ||||
|       ].forEach(key => { | ||||
|         al.push(this[key]) | ||||
|       }); | ||||
|       return al | ||||
|     }, (newValue, oldValue) => { | ||||
|       let needReset = false | ||||
|       for (let i = 2; i < newValue.length; i++) { | ||||
|         if (newValue[i] != oldValue[i]) { | ||||
|           needReset = true | ||||
|           break | ||||
|         } | ||||
|       } | ||||
|       if (newValue[0] != oldValue[0]) { | ||||
|         this.page.current = this.pageCurrent | ||||
|       } | ||||
|       this.page.size = this.pageSize | ||||
|  | ||||
|       this.onPropsChange() | ||||
|     }) | ||||
|     this._treeData = [] | ||||
|   }, | ||||
|   methods: { | ||||
|     onPropsChange() { | ||||
|       this._treeData = []; | ||||
|     }, | ||||
|  | ||||
|     // 填充 pickview 数据 | ||||
|     async loadData() { | ||||
|       if (this.isLocalData) { | ||||
|         this.loadLocalData(); | ||||
|       } else if (this.isCloudDataList) { | ||||
|         this.loadCloudDataList(); | ||||
|       } else if (this.isCloudDataTree) { | ||||
|         this.loadCloudDataTree(); | ||||
|       } | ||||
|     }, | ||||
|  | ||||
|     // 加载本地数据 | ||||
|     async loadLocalData() { | ||||
|       this._treeData = []; | ||||
|       this._extractTree(this.localdata, this._treeData); | ||||
|  | ||||
|       let inputValue = this.dataValue; | ||||
|       if (inputValue === undefined) { | ||||
|         return; | ||||
|       } | ||||
|  | ||||
|       if (Array.isArray(inputValue)) { | ||||
|         inputValue = inputValue[inputValue.length - 1]; | ||||
|         if (typeof inputValue === 'object' && inputValue[this.map.value]) { | ||||
|           inputValue = inputValue[this.map.value]; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       this.selected = this._findNodePath(inputValue, this.localdata); | ||||
|     }, | ||||
|  | ||||
|     // 加载 Cloud 数据 (单列) | ||||
|     async loadCloudDataList() { | ||||
|       if (this.loading) { | ||||
|         return; | ||||
|       } | ||||
|       this.loading = true; | ||||
|  | ||||
|       try { | ||||
|         let response = await this.getCommand(); | ||||
|         let responseData = response.result.data; | ||||
|  | ||||
|         this._treeData = responseData; | ||||
|  | ||||
|         this._updateBindData(); | ||||
|         this._updateSelected(); | ||||
|  | ||||
|         this.onDataChange(); | ||||
|       } catch (e) { | ||||
|         this.errorMessage = e; | ||||
|       } finally { | ||||
|         this.loading = false; | ||||
|       } | ||||
|     }, | ||||
|  | ||||
|     // 加载 Cloud 数据 (树形) | ||||
|     async loadCloudDataTree() { | ||||
|       if (this.loading) { | ||||
|         return; | ||||
|       } | ||||
|       this.loading = true; | ||||
|  | ||||
|       try { | ||||
|         let commandOptions = { | ||||
|           field: this._cloudDataPostField(), | ||||
|           where: this._cloudDataTreeWhere() | ||||
|         }; | ||||
|         if (this.gettree) { | ||||
|           commandOptions.startwith = `${this.selfField}=='${this.dataValue}'`; | ||||
|         } | ||||
|  | ||||
|         let response = await this.getCommand(commandOptions); | ||||
|         let responseData = response.result.data; | ||||
|  | ||||
|         this._treeData = responseData; | ||||
|         this._updateBindData(); | ||||
|         this._updateSelected(); | ||||
|  | ||||
|         this.onDataChange(); | ||||
|       } catch (e) { | ||||
|         this.errorMessage = e; | ||||
|       } finally { | ||||
|         this.loading = false; | ||||
|       } | ||||
|     }, | ||||
|  | ||||
|     // 加载 Cloud 数据 (节点) | ||||
|     async loadCloudDataNode(callback) { | ||||
|       if (this.loading) { | ||||
|         return; | ||||
|       } | ||||
|       this.loading = true; | ||||
|  | ||||
|       try { | ||||
|         let commandOptions = { | ||||
|           field: this._cloudDataPostField(), | ||||
|           where: this._cloudDataNodeWhere() | ||||
|         }; | ||||
|  | ||||
|         let response = await this.getCommand(commandOptions); | ||||
|         let responseData = response.result.data; | ||||
|  | ||||
|         callback(responseData); | ||||
|       } catch (e) { | ||||
|         this.errorMessage = e; | ||||
|       } finally { | ||||
|         this.loading = false; | ||||
|       } | ||||
|     }, | ||||
|  | ||||
|     // 回显 Cloud 数据 | ||||
|     getCloudDataValue() { | ||||
|       if (this.isCloudDataList) { | ||||
|         return this.getCloudDataListValue(); | ||||
|       } | ||||
|  | ||||
|       if (this.isCloudDataTree) { | ||||
|         return this.getCloudDataTreeValue(); | ||||
|       } | ||||
|     }, | ||||
|  | ||||
|     // 回显 Cloud 数据 (单列) | ||||
|     getCloudDataListValue() { | ||||
|       // 根据 field's as value标识匹配 where 条件 | ||||
|       let where = []; | ||||
|       let whereField = this._getForeignKeyByField(); | ||||
|       if (whereField) { | ||||
|         where.push(`${whereField} == '${this.dataValue}'`) | ||||
|       } | ||||
|  | ||||
|       where = where.join(' || '); | ||||
|  | ||||
|       if (this.where) { | ||||
|         where = `(${this.where}) && (${where})` | ||||
|       } | ||||
|  | ||||
|       return this.getCommand({ | ||||
|         field: this._cloudDataPostField(), | ||||
|         where | ||||
|       }).then((res) => { | ||||
|         this.selected = res.result.data; | ||||
|         return res.result.data; | ||||
|       }); | ||||
|     }, | ||||
|  | ||||
|     // 回显 Cloud 数据 (树形) | ||||
|     getCloudDataTreeValue() { | ||||
|       return this.getCommand({ | ||||
|         field: this._cloudDataPostField(), | ||||
|         getTreePath: { | ||||
|           startWith: `${this.selfField}=='${this.dataValue}'` | ||||
|         } | ||||
|       }).then((res) => { | ||||
|         let treePath = []; | ||||
|         this._extractTreePath(res.result.data, treePath); | ||||
|         this.selected = treePath; | ||||
|         return treePath; | ||||
|       }); | ||||
|     }, | ||||
|  | ||||
|     getCommand(options = {}) { | ||||
|       /* eslint-disable no-undef */ | ||||
|       let db = uniCloud.database(this.spaceInfo) | ||||
|  | ||||
|       const action = options.action || this.action | ||||
|       if (action) { | ||||
|         db = db.action(action) | ||||
|       } | ||||
|  | ||||
|       const collection = options.collection || this.collection | ||||
|       db = db.collection(collection) | ||||
|  | ||||
|       const where = options.where || this.where | ||||
|       if (!(!where || !Object.keys(where).length)) { | ||||
|         db = db.where(where) | ||||
|       } | ||||
|  | ||||
|       const field = options.field || this.field | ||||
|       if (field) { | ||||
|         db = db.field(field) | ||||
|       } | ||||
|  | ||||
|       const orderby = options.orderby || this.orderby | ||||
|       if (orderby) { | ||||
|         db = db.orderBy(orderby) | ||||
|       } | ||||
|  | ||||
|       const current = options.pageCurrent !== undefined ? options.pageCurrent : this.page.current | ||||
|       const size = options.pageSize !== undefined ? options.pageSize : this.page.size | ||||
|       const getCount = options.getcount !== undefined ? options.getcount : this.getcount | ||||
|       const getTree = options.gettree !== undefined ? options.gettree : this.gettree | ||||
|  | ||||
|       const getOptions = { | ||||
|         getCount, | ||||
|         getTree | ||||
|       } | ||||
|       if (options.getTreePath) { | ||||
|         getOptions.getTreePath = options.getTreePath | ||||
|       } | ||||
|  | ||||
|       db = db.skip(size * (current - 1)).limit(size).get(getOptions) | ||||
|  | ||||
|       return db | ||||
|     }, | ||||
|  | ||||
|     _cloudDataPostField() { | ||||
|       let fields = [this.field]; | ||||
|       if (this.parentField) { | ||||
|         fields.push(`${this.parentField} as parent_value`); | ||||
|       } | ||||
|       return fields.join(','); | ||||
|     }, | ||||
|  | ||||
|     _cloudDataTreeWhere() { | ||||
|       let result = [] | ||||
|       let selected = this.selected | ||||
|       let parentField = this.parentField | ||||
|       if (parentField) { | ||||
|         result.push(`${parentField} == null || ${parentField} == ""`) | ||||
|       } | ||||
|       if (selected.length) { | ||||
|         for (var i = 0; i < selected.length - 1; i++) { | ||||
|           result.push(`${parentField} == '${selected[i].value}'`) | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       let where = [] | ||||
|       if (this.where) { | ||||
|         where.push(`(${this.where})`) | ||||
|       } | ||||
|  | ||||
|       if (result.length) { | ||||
|         where.push(`(${result.join(' || ')})`) | ||||
|       } | ||||
|  | ||||
|       return where.join(' && ') | ||||
|     }, | ||||
|  | ||||
|     _cloudDataNodeWhere() { | ||||
|       let where = [] | ||||
|       let selected = this.selected; | ||||
|       if (selected.length) { | ||||
|         where.push(`${this.parentField} == '${selected[selected.length - 1].value}'`); | ||||
|       } | ||||
|  | ||||
|       where = where.join(' || '); | ||||
|  | ||||
|       if (this.where) { | ||||
|         return `(${this.where}) && (${where})` | ||||
|       } | ||||
|  | ||||
|       return where | ||||
|     }, | ||||
|  | ||||
|     _getWhereByForeignKey() { | ||||
|       let result = [] | ||||
|       let whereField = this._getForeignKeyByField(); | ||||
|       if (whereField) { | ||||
|         result.push(`${whereField} == '${this.dataValue}'`) | ||||
|       } | ||||
|  | ||||
|       if (this.where) { | ||||
|         return `(${this.where}) && (${result.join(' || ')})` | ||||
|       } | ||||
|  | ||||
|       return result.join(' || ') | ||||
|     }, | ||||
|  | ||||
|     _getForeignKeyByField() { | ||||
|       let fields = this.field.split(','); | ||||
|       let whereField = null; | ||||
|       for (let i = 0; i < fields.length; i++) { | ||||
|         const items = fields[i].split('as'); | ||||
|         if (items.length < 2) { | ||||
|           continue; | ||||
|         } | ||||
|         if (items[1].trim() === 'value') { | ||||
|           whereField = items[0].trim(); | ||||
|           break; | ||||
|         } | ||||
|       } | ||||
|       return whereField; | ||||
|     }, | ||||
|  | ||||
|     _updateBindData(node) { | ||||
|       const { | ||||
|         dataList, | ||||
|         hasNodes | ||||
|       } = this._filterData(this._treeData, this.selected) | ||||
|  | ||||
|       let isleaf = this._stepSearh === false && !hasNodes | ||||
|  | ||||
|       if (node) { | ||||
|         node.isleaf = isleaf | ||||
|       } | ||||
|  | ||||
|       this.dataList = dataList | ||||
|       this.selectedIndex = dataList.length - 1 | ||||
|  | ||||
|       if (!isleaf && this.selected.length < dataList.length) { | ||||
|         this.selected.push({ | ||||
|           value: null, | ||||
|           text: "请选择" | ||||
|         }) | ||||
|       } | ||||
|  | ||||
|       return { | ||||
|         isleaf, | ||||
|         hasNodes | ||||
|       } | ||||
|     }, | ||||
|  | ||||
|     _updateSelected() { | ||||
|       let dl = this.dataList | ||||
|       let sl = this.selected | ||||
|       let textField = this.map.text | ||||
|       let valueField = this.map.value | ||||
|       for (let i = 0; i < sl.length; i++) { | ||||
|         let value = sl[i].value | ||||
|         let dl2 = dl[i] | ||||
|         for (let j = 0; j < dl2.length; j++) { | ||||
|           let item2 = dl2[j] | ||||
|           if (item2[valueField] === value) { | ||||
|             sl[i].text = item2[textField] | ||||
|             break | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     }, | ||||
|  | ||||
|     _filterData(data, paths) { | ||||
|       let dataList = [] | ||||
|       let hasNodes = true | ||||
|  | ||||
|       dataList.push(data.filter((item) => { | ||||
|         return (item.parent_value === null || item.parent_value === undefined || item.parent_value === '') | ||||
|       })) | ||||
|       for (let i = 0; i < paths.length; i++) { | ||||
|         let value = paths[i].value | ||||
|         let nodes = data.filter((item) => { | ||||
|           return item.parent_value === value | ||||
|         }) | ||||
|  | ||||
|         if (nodes.length) { | ||||
|           dataList.push(nodes) | ||||
|         } else { | ||||
|           hasNodes = false | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       return { | ||||
|         dataList, | ||||
|         hasNodes | ||||
|       } | ||||
|     }, | ||||
|  | ||||
|     _extractTree(nodes, result, parent_value) { | ||||
|       let list = result || [] | ||||
|       let valueField = this.map.value | ||||
|       for (let i = 0; i < nodes.length; i++) { | ||||
|         let node = nodes[i] | ||||
|  | ||||
|         let child = {} | ||||
|         for (let key in node) { | ||||
|           if (key !== 'children') { | ||||
|             child[key] = node[key] | ||||
|           } | ||||
|         } | ||||
|         if (parent_value !== null && parent_value !== undefined && parent_value !== '') { | ||||
|           child.parent_value = parent_value | ||||
|         } | ||||
|         result.push(child) | ||||
|  | ||||
|         let children = node.children | ||||
|         if (children) { | ||||
|           this._extractTree(children, result, node[valueField]) | ||||
|         } | ||||
|       } | ||||
|     }, | ||||
|  | ||||
|     _extractTreePath(nodes, result) { | ||||
|       let list = result || [] | ||||
|       for (let i = 0; i < nodes.length; i++) { | ||||
|         let node = nodes[i] | ||||
|  | ||||
|         let child = {} | ||||
|         for (let key in node) { | ||||
|           if (key !== 'children') { | ||||
|             child[key] = node[key] | ||||
|           } | ||||
|         } | ||||
|         result.push(child) | ||||
|  | ||||
|         let children = node.children | ||||
|         if (children) { | ||||
|           this._extractTreePath(children, result) | ||||
|         } | ||||
|       } | ||||
|     }, | ||||
|  | ||||
|     _findNodePath(key, nodes, path = []) { | ||||
|       let textField = this.map.text | ||||
|       let valueField = this.map.value | ||||
|       for (let i = 0; i < nodes.length; i++) { | ||||
|         let node = nodes[i] | ||||
|         let children = node.children | ||||
|         let text = node[textField] | ||||
|         let value = node[valueField] | ||||
|  | ||||
|         path.push({ | ||||
|           value, | ||||
|           text | ||||
|         }) | ||||
|  | ||||
|         if (value === key) { | ||||
|           return path | ||||
|         } | ||||
|  | ||||
|         if (children) { | ||||
|           const p = this._findNodePath(key, children, path) | ||||
|           if (p.length) { | ||||
|             return p | ||||
|           } | ||||
|         } | ||||
|  | ||||
|         path.pop() | ||||
|       } | ||||
|       return [] | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -0,0 +1,323 @@ | ||||
| <template> | ||||
|   <view class="uni-data-pickerview"> | ||||
|     <scroll-view v-if="!isCloudDataList" class="selected-area" scroll-x="true"> | ||||
|       <view class="selected-list"> | ||||
|           <view  | ||||
|             class="selected-item" | ||||
|             v-for="(item,index) in selected" | ||||
|             :key="index" | ||||
|             :class="{ | ||||
|               'selected-item-active':index == selectedIndex | ||||
|             }" | ||||
|             @click="handleSelect(index)" | ||||
|           > | ||||
|             <text>{{item.text || ''}}</text> | ||||
|           </view> | ||||
|       </view> | ||||
|     </scroll-view> | ||||
|     <view class="tab-c"> | ||||
|       <scroll-view class="list" :scroll-y="true"> | ||||
|         <view class="item" :class="{'is-disabled': !!item.disable}" v-for="(item, j) in dataList[selectedIndex]" :key="j" | ||||
|           @click="handleNodeClick(item, selectedIndex, j)"> | ||||
|           <text class="item-text">{{item[map.text]}}</text> | ||||
|           <view class="check" v-if="selected.length > selectedIndex && item[map.value] == selected[selectedIndex].value"></view> | ||||
|         </view> | ||||
|       </scroll-view> | ||||
|  | ||||
|       <view class="loading-cover" v-if="loading"> | ||||
|         <uni-load-more class="load-more" :contentText="loadMore" status="loading"></uni-load-more> | ||||
|       </view> | ||||
|       <view class="error-message" v-if="errorMessage"> | ||||
|         <text class="error-text">{{errorMessage}}</text> | ||||
|       </view> | ||||
|     </view> | ||||
|   </view> | ||||
| </template> | ||||
|  | ||||
| <script> | ||||
|   import dataPicker from "./uni-data-picker.js" | ||||
|  | ||||
|   /** | ||||
|    * DataPickerview | ||||
|    * @description uni-data-pickerview | ||||
|    * @tutorial https://ext.dcloud.net.cn/plugin?id=3796 | ||||
|    * @property {Array} localdata 本地数据,参考 | ||||
|    * @property {Boolean} step-searh = [true|false] 是否分布查询 | ||||
|    * @value true 启用分布查询,仅查询当前选中节点 | ||||
|    * @value false 关闭分布查询,一次查询出所有数据 | ||||
|    * @property {String|DBFieldString} self-field 分布查询当前字段名称 | ||||
|    * @property {String|DBFieldString} parent-field 分布查询父字段名称 | ||||
|    * @property {String|DBCollectionString} collection 表名 | ||||
|    * @property {String|DBFieldString} field 查询字段,多个字段用 `,` 分割 | ||||
|    * @property {String} orderby 排序字段及正序倒叙设置 | ||||
|    * @property {String|JQLString} where 查询条件 | ||||
|    */ | ||||
|   export default { | ||||
|     name: 'UniDataPickerView', | ||||
|     emits: ['nodeclick', 'change', 'datachange', 'update:modelValue'], | ||||
|     mixins: [dataPicker], | ||||
|     props: { | ||||
|       managedMode: { | ||||
|         type: Boolean, | ||||
|         default: false | ||||
|       }, | ||||
|       ellipsis: { | ||||
|         type: Boolean, | ||||
|         default: true | ||||
|       } | ||||
|     }, | ||||
|     created() { | ||||
|       if (!this.managedMode) { | ||||
|         this.$nextTick(() => { | ||||
|           this.loadData(); | ||||
|         }) | ||||
|       } | ||||
|     }, | ||||
|     methods: { | ||||
|       onPropsChange() { | ||||
|         this._treeData = []; | ||||
|         this.selectedIndex = 0; | ||||
|         this.$nextTick(() => { | ||||
|           this.loadData(); | ||||
|         }) | ||||
|       }, | ||||
|       handleSelect(index) { | ||||
|         this.selectedIndex = index; | ||||
|       }, | ||||
|       handleNodeClick(item, i, j) { | ||||
|         if (item.disable) { | ||||
|           return; | ||||
|         } | ||||
|  | ||||
|         const node = this.dataList[i][j]; | ||||
|         const text = node[this.map.text]; | ||||
|         const value = node[this.map.value]; | ||||
|  | ||||
|         if (i < this.selected.length - 1) { | ||||
|           this.selected.splice(i, this.selected.length - i) | ||||
|           this.selected.push({ | ||||
|             text, | ||||
|             value | ||||
|           }) | ||||
|         } else if (i === this.selected.length - 1) { | ||||
|           this.selected.splice(i, 1, { | ||||
|             text, | ||||
|             value | ||||
|           }) | ||||
|         } | ||||
|  | ||||
|         if (node.isleaf) { | ||||
|           this.onSelectedChange(node, node.isleaf) | ||||
|           return | ||||
|         } | ||||
|  | ||||
|         const { | ||||
|           isleaf, | ||||
|           hasNodes | ||||
|         } = this._updateBindData() | ||||
|  | ||||
|         // 本地数据 | ||||
|         if (this.isLocalData) { | ||||
|           this.onSelectedChange(node, (!hasNodes || isleaf)) | ||||
|         } else if (this.isCloudDataList) { // Cloud 数据 (单列) | ||||
|           this.onSelectedChange(node, true) | ||||
|         } else if (this.isCloudDataTree) { // Cloud 数据 (树形) | ||||
|           if (isleaf) { | ||||
|             this.onSelectedChange(node, node.isleaf) | ||||
|           } else if (!hasNodes) { // 请求一次服务器以确定是否为叶子节点 | ||||
|             this.loadCloudDataNode((data) => { | ||||
|               if (!data.length) { | ||||
|                 node.isleaf = true | ||||
|               } else { | ||||
|                 this._treeData.push(...data) | ||||
|                 this._updateBindData(node) | ||||
|               } | ||||
|               this.onSelectedChange(node, node.isleaf) | ||||
|             }) | ||||
|           } | ||||
|         } | ||||
|       }, | ||||
|       updateData(data) { | ||||
|         this._treeData = data.treeData | ||||
|         this.selected = data.selected | ||||
|         if (!this._treeData.length) { | ||||
|           this.loadData() | ||||
|         } else { | ||||
|           //this.selected = data.selected | ||||
|           this._updateBindData() | ||||
|         } | ||||
|       }, | ||||
|       onDataChange() { | ||||
|         this.$emit('datachange'); | ||||
|       }, | ||||
|       onSelectedChange(node, isleaf) { | ||||
|         if (isleaf) { | ||||
|           this._dispatchEvent() | ||||
|         } | ||||
|  | ||||
|         if (node) { | ||||
|           this.$emit('nodeclick', node) | ||||
|         } | ||||
|       }, | ||||
|       _dispatchEvent() { | ||||
|         this.$emit('change', this.selected.slice(0)) | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| </script> | ||||
|  | ||||
| <style lang="scss"> | ||||
| 	$uni-primary: #007aff !default; | ||||
|  | ||||
| 	.uni-data-pickerview { | ||||
| 		flex: 1; | ||||
| 		/* #ifndef APP-NVUE */ | ||||
| 		display: flex; | ||||
| 		/* #endif */ | ||||
| 		flex-direction: column; | ||||
| 		overflow: hidden; | ||||
| 		height: 100%; | ||||
| 	} | ||||
|  | ||||
|   .error-text { | ||||
|     color: #DD524D; | ||||
|   } | ||||
|  | ||||
|   .loading-cover { | ||||
|     position: absolute; | ||||
|     left: 0; | ||||
|     top: 0; | ||||
|     right: 0; | ||||
|     bottom: 0; | ||||
|     background-color: rgba(255, 255, 255, .5); | ||||
|     /* #ifndef APP-NVUE */ | ||||
|     display: flex; | ||||
|     /* #endif */ | ||||
|     flex-direction: column; | ||||
|     align-items: center; | ||||
|     z-index: 1001; | ||||
|   } | ||||
|  | ||||
|   .load-more { | ||||
|     /* #ifndef APP-NVUE */ | ||||
|     margin: auto; | ||||
|     /* #endif */ | ||||
|   } | ||||
|  | ||||
|   .error-message { | ||||
|     background-color: #fff; | ||||
|     position: absolute; | ||||
|     left: 0; | ||||
|     top: 0; | ||||
|     right: 0; | ||||
|     bottom: 0; | ||||
|     padding: 15px; | ||||
|     opacity: .9; | ||||
|     z-index: 102; | ||||
|   } | ||||
|  | ||||
|   /* #ifdef APP-NVUE */ | ||||
|   .selected-area { | ||||
|     width: 750rpx; | ||||
|   } | ||||
|   /* #endif */ | ||||
|  | ||||
|   .selected-list { | ||||
|     /* #ifndef APP-NVUE */ | ||||
|     display: flex; | ||||
|     flex-wrap: nowrap; | ||||
|     /* #endif */ | ||||
|     flex-direction: row; | ||||
|     padding: 0 5px; | ||||
|     border-bottom: 1px solid #f8f8f8; | ||||
|   } | ||||
|  | ||||
|   .selected-item { | ||||
|     margin-left: 10px; | ||||
|     margin-right: 10px; | ||||
|     padding: 12px 0; | ||||
|     text-align: center; | ||||
|     /* #ifndef APP-NVUE */ | ||||
|     white-space: nowrap; | ||||
|     /* #endif */ | ||||
|   } | ||||
|  | ||||
|   .selected-item-text-overflow { | ||||
|     width: 168px; | ||||
|     /* fix nvue */ | ||||
|     overflow: hidden; | ||||
|     /* #ifndef APP-NVUE */ | ||||
|     width: 6em; | ||||
|     white-space: nowrap; | ||||
|     text-overflow: ellipsis; | ||||
|     -o-text-overflow: ellipsis; | ||||
|     /* #endif */ | ||||
|   } | ||||
|  | ||||
| 	.selected-item-active { | ||||
| 		border-bottom: 2px solid $uni-primary; | ||||
| 	} | ||||
|  | ||||
| 	.selected-item-text { | ||||
| 		color: $uni-primary; | ||||
| 	} | ||||
|  | ||||
|   .tab-c { | ||||
|     position: relative; | ||||
|     flex: 1; | ||||
|     /* #ifndef APP-NVUE */ | ||||
|     display: flex; | ||||
|     /* #endif */ | ||||
|     flex-direction: row; | ||||
|     overflow: hidden; | ||||
|   } | ||||
|  | ||||
|   .list { | ||||
|     flex: 1; | ||||
|   } | ||||
|  | ||||
|   .item { | ||||
|     padding: 12px 15px; | ||||
|     /* border-bottom: 1px solid #f0f0f0; */ | ||||
|     /* #ifndef APP-NVUE */ | ||||
|     display: flex; | ||||
|     /* #endif */ | ||||
|     flex-direction: row; | ||||
|     justify-content: space-between; | ||||
|   } | ||||
|  | ||||
|   .is-disabled { | ||||
|     opacity: .5; | ||||
|   } | ||||
|  | ||||
|   .item-text { | ||||
|     /* flex: 1; */ | ||||
|     color: #333333; | ||||
|   } | ||||
|  | ||||
|   .item-text-overflow { | ||||
|     width: 280px; | ||||
|     /* fix nvue */ | ||||
|     overflow: hidden; | ||||
|     /* #ifndef APP-NVUE */ | ||||
|     width: 20em; | ||||
|     white-space: nowrap; | ||||
|     text-overflow: ellipsis; | ||||
|     -o-text-overflow: ellipsis; | ||||
|     /* #endif */ | ||||
|   } | ||||
|  | ||||
| 	.check { | ||||
| 		margin-right: 5px; | ||||
| 		border: 2px solid $uni-primary; | ||||
| 		border-left: 0; | ||||
| 		border-top: 0; | ||||
| 		height: 12px; | ||||
| 		width: 6px; | ||||
| 		transform-origin: center; | ||||
| 		/* #ifndef APP-NVUE */ | ||||
| 		transition: all 0.3s; | ||||
| 		/* #endif */ | ||||
| 		transform: rotate(45deg); | ||||
| 	} | ||||
| </style> | ||||
							
								
								
									
										90
									
								
								uni_modules/uni-data-picker/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								uni_modules/uni-data-picker/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,90 @@ | ||||
| { | ||||
|   "id": "uni-data-picker", | ||||
|   "displayName": "uni-data-picker 数据驱动的picker选择器", | ||||
|   "version": "1.1.2", | ||||
|   "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" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										22
									
								
								uni_modules/uni-data-picker/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								uni_modules/uni-data-picker/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -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  | ||||
							
								
								
									
										35
									
								
								uni_modules/uni-data-select/changelog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								uni_modules/uni-data-select/changelog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| ## 1.0.6(2023-04-12) | ||||
| - 修复 微信小程序点击时会改变背景颜色的 bug | ||||
| ## 1.0.5(2023-02-03) | ||||
| - 修复 禁用时会显示清空按钮 | ||||
| ## 1.0.4(2023-02-02) | ||||
| - 优化 查询条件短期内多次变更只查询最后一次变更后的结果 | ||||
| - 调整 内部缓存键名调整为 uni-data-select-lastSelectedValue | ||||
| ## 1.0.3(2023-01-16) | ||||
| - 修复 不关联服务空间报错的问题 | ||||
| ## 1.0.2(2023-01-14) | ||||
| - 新增  属性 `format` 可用于格式化显示选项内容 | ||||
| ## 1.0.1(2022-12-06) | ||||
| - 修复  当where变化时,数据不会自动更新的问题 | ||||
| ## 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,517 @@ | ||||
| <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> | ||||
| 					<view v-if="current && clear && !disabled" @click.stop="clearVal" > | ||||
| 						<uni-icons type="clear" color="#c0c4cc" size="24"/> | ||||
| 					</view> | ||||
| 					<view v-else> | ||||
| 						<uni-icons :type="showSelector? 'top' : 'bottom'" size="14" color="#999" /> | ||||
| 					</view> | ||||
| 				</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-data-select", | ||||
| 		mixins: [uniCloud.mixinDatacom || {}], | ||||
| 		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 | ||||
| 			}, | ||||
| 			// 格式化输出 用法 field="_id as value, version as text, uni_platform as label" format="{label} - {text}" | ||||
| 			format: { | ||||
| 				type: String, | ||||
| 				default: '' | ||||
| 			}, | ||||
| 		}, | ||||
| 		data() { | ||||
| 			return { | ||||
| 				showSelector: false, | ||||
| 				current: '', | ||||
| 				mixinDatacomResData: [], | ||||
| 				apps: [], | ||||
| 				channels: [], | ||||
| 				cacheKey: "uni-data-select-lastSelectedValue", | ||||
| 			}; | ||||
| 		}, | ||||
| 		created() { | ||||
| 			this.debounceGet = this.debounce(() => { | ||||
| 				this.query(); | ||||
| 			}, 300); | ||||
| 			if (this.collection && !this.localdata.length) { | ||||
| 				this.debounceGet(); | ||||
| 			} | ||||
| 		}, | ||||
| 		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 | ||||
| 			}, | ||||
| 			valueCom(){ | ||||
| 				// #ifdef VUE3 | ||||
| 				return this.modelValue; | ||||
| 				// #endif | ||||
| 				// #ifndef VUE3 | ||||
| 				return this.value; | ||||
| 				// #endif | ||||
| 			} | ||||
| 		}, | ||||
| 		watch: { | ||||
| 			localdata: { | ||||
| 				immediate: true, | ||||
| 				handler(val, old) { | ||||
| 					if (Array.isArray(val) && old !== val) { | ||||
| 						this.mixinDatacomResData = val | ||||
| 					} | ||||
| 				} | ||||
| 			}, | ||||
| 			valueCom(val, old) { | ||||
| 				this.initDefVal() | ||||
| 			}, | ||||
| 			mixinDatacomResData: { | ||||
| 				immediate: true, | ||||
| 				handler(val) { | ||||
| 					if (val.length) { | ||||
| 						this.initDefVal() | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		}, | ||||
| 		methods: { | ||||
| 			debounce(fn, time = 100){ | ||||
| 				let timer = null | ||||
| 				return function(...args) { | ||||
| 					if (timer) clearTimeout(timer) | ||||
| 					timer = setTimeout(() => { | ||||
| 						fn.apply(this, args) | ||||
| 					}, time) | ||||
| 				} | ||||
| 			}, | ||||
| 			// 执行数据库查询 | ||||
| 			query(){ | ||||
| 				this.mixinDatacomEasyGet(); | ||||
| 			}, | ||||
| 			// 监听查询条件变更事件 | ||||
| 			onMixinDatacomPropsChange(){ | ||||
| 				if (this.collection) { | ||||
| 					this.debounceGet(); | ||||
| 				} | ||||
| 			}, | ||||
| 			initDefVal() { | ||||
| 				let defValue = '' | ||||
| 				if ((this.valueCom || this.valueCom === 0) && !this.isDisabled(this.valueCom)) { | ||||
| 					defValue = this.valueCom | ||||
| 				} else { | ||||
| 					let strogeValue | ||||
| 					if (this.collection) { | ||||
| 						strogeValue = this.getCache() | ||||
| 					} | ||||
| 					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 | ||||
| 					} | ||||
|           if (defValue || defValue === 0) { | ||||
| 					  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) { | ||||
| 					this.removeCache() | ||||
| 				} | ||||
| 			}, | ||||
| 			change(item) { | ||||
| 				if (!item.disable) { | ||||
| 					this.showSelector = false | ||||
| 					this.current = this.formatItemName(item) | ||||
| 					this.emit(item.value) | ||||
| 				} | ||||
| 			}, | ||||
| 			emit(val) { | ||||
| 				this.$emit('input', val) | ||||
| 				this.$emit('update:modelValue', val) | ||||
| 				this.$emit('change', val) | ||||
| 				if (this.collection) { | ||||
| 					this.setCache(val); | ||||
| 				} | ||||
| 			}, | ||||
| 			toggleSelector() { | ||||
| 				if (this.disabled) { | ||||
| 					return | ||||
| 				} | ||||
|  | ||||
| 				this.showSelector = !this.showSelector | ||||
| 			}, | ||||
| 			formatItemName(item) { | ||||
| 				let { | ||||
| 					text, | ||||
| 					value, | ||||
| 					channel_code | ||||
| 				} = item | ||||
| 				channel_code = channel_code ? `(${channel_code})` : '' | ||||
|  | ||||
| 				if (this.format) { | ||||
| 					// 格式化输出 | ||||
| 					let str = ""; | ||||
| 					str = this.format; | ||||
| 					for (let key in item) { | ||||
| 						str = str.replace(new RegExp(`{${key}}`,"g"),item[key]); | ||||
| 					} | ||||
| 					return str; | ||||
| 				} else { | ||||
| 					return this.collection.indexOf('app-list') > 0 ? | ||||
| 						`${text}(${value})` : | ||||
| 						( | ||||
| 							text ? | ||||
| 							text : | ||||
| 							`未命名${channel_code}` | ||||
| 						) | ||||
| 				} | ||||
| 			}, | ||||
| 			// 获取当前加载的数据 | ||||
| 			getLoadData(){ | ||||
| 				return this.mixinDatacomResData; | ||||
| 			}, | ||||
| 			// 获取当前缓存key | ||||
| 			getCurrentCacheKey(){ | ||||
| 				return this.collection; | ||||
| 			}, | ||||
| 			// 获取缓存 | ||||
| 			getCache(name=this.getCurrentCacheKey()){ | ||||
| 				let cacheData = uni.getStorageSync(this.cacheKey) || {}; | ||||
| 				return cacheData[name]; | ||||
| 			}, | ||||
| 			// 设置缓存 | ||||
| 			setCache(value, name=this.getCurrentCacheKey()){ | ||||
| 				let cacheData = uni.getStorageSync(this.cacheKey) || {}; | ||||
| 				cacheData[name] = value; | ||||
| 				uni.setStorageSync(this.cacheKey, cacheData); | ||||
| 			}, | ||||
| 			// 删除缓存 | ||||
| 			removeCache(name=this.getCurrentCacheKey()){ | ||||
| 				let cacheData = uni.getStorageSync(this.cacheKey) || {}; | ||||
| 				delete cacheData[name]; | ||||
| 				uni.setStorageSync(this.cacheKey, cacheData); | ||||
| 			}, | ||||
| 		} | ||||
| 	} | ||||
| </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; | ||||
| 		/* #ifdef H5 */ | ||||
| 		cursor: pointer; | ||||
| 		/* #endif */ | ||||
| 		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 */ | ||||
| 	} | ||||
|  | ||||
| 	/* #ifdef H5 */ | ||||
| 	@media (min-width: 768px) { | ||||
| 		.uni-select__selector-scroll { | ||||
| 			max-height: 600px; | ||||
| 		} | ||||
| 	} | ||||
| 	/* #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; | ||||
| 		z-index: 2; | ||||
| 	} | ||||
| </style> | ||||
							
								
								
									
										85
									
								
								uni_modules/uni-data-select/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								uni_modules/uni-data-select/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,85 @@ | ||||
| { | ||||
|   "id": "uni-data-select", | ||||
|   "displayName": "uni-data-select 下拉框选择器", | ||||
|   "version": "1.0.6", | ||||
|   "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" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										8
									
								
								uni_modules/uni-data-select/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								uni_modules/uni-data-select/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| ## DataSelect 下拉框选择器 | ||||
| > **组件名:uni-data-select** | ||||
| > 代码块: `uDataSelect` | ||||
|  | ||||
| 当选项过多时,使用下拉菜单展示并选择内容 | ||||
|  | ||||
| ### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-data-select) | ||||
| #### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839  | ||||
							
								
								
									
										10
									
								
								uni_modules/uni-dateformat/changelog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								uni_modules/uni-dateformat/changelog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -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> | ||||
							
								
								
									
										88
									
								
								uni_modules/uni-dateformat/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										88
									
								
								uni_modules/uni-dateformat/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -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" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										11
									
								
								uni_modules/uni-dateformat/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								uni_modules/uni-dateformat/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
|  | ||||
|  | ||||
| ### DateFormat 日期格式化 | ||||
| > **组件名:uni-dateformat** | ||||
| > 代码块: `uDateformat` | ||||
|  | ||||
|  | ||||
| 日期格式化组件。 | ||||
|  | ||||
| ### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-dateformat) | ||||
| #### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839  | ||||
							
								
								
									
										133
									
								
								uni_modules/uni-datetime-picker/changelog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										133
									
								
								uni_modules/uni-datetime-picker/changelog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,133 @@ | ||||
| ## 2.2.22(2023-03-30) | ||||
| - 修复 日历 picker 修改年月后,自动选中当月1日 [详情](https://ask.dcloud.net.cn/question/165937) | ||||
| - 修复 小程序端 低版本 ios NaN [详情](https://ask.dcloud.net.cn/question/162979) | ||||
| ## 2.2.21(2023-02-20) | ||||
| - 修复 firefox 浏览器显示区域点击无法拉起日历弹框的Bug [详情](https://ask.dcloud.net.cn/question/163362) | ||||
| ## 2.2.20(2023-02-17) | ||||
| - 优化 值为空依然选中当天问题 | ||||
| - 优化 提供 default-value 属性支持配置选择器打开时默认显示的时间 | ||||
| - 优化 非范围选择未选择日期时间,点击确认按钮选中当前日期时间 | ||||
| - 优化 字节小程序日期时间范围选择,底部日期换行问题 | ||||
| ## 2.2.19(2023-02-09) | ||||
| - 修复 2.2.18 引起范围选择配置 end 选择无效的Bug [详情](https://github.com/dcloudio/uni-ui/issues/686) | ||||
| ## 2.2.18(2023-02-08) | ||||
| - 修复 移动端范围选择change事件触发异常的Bug [详情](https://github.com/dcloudio/uni-ui/issues/684) | ||||
| - 优化 PC端输入日期格式错误时返回当前日期时间 | ||||
| - 优化 PC端输入日期时间超出 start、end 限制的Bug | ||||
| - 优化 移动端日期时间范围用法时间展示不完整问题 | ||||
| ## 2.2.17(2023-02-04) | ||||
| - 修复 小程序端绑定 Date 类型报错的Bug [详情](https://github.com/dcloudio/uni-ui/issues/679) | ||||
| - 修复 vue3 time-picker 无法显示绑定时分秒的Bug | ||||
| ## 2.2.16(2023-02-02) | ||||
| - 修复 字节小程序报错的Bug | ||||
| ## 2.2.15(2023-02-02) | ||||
| - 修复 某些情况切换月份错误的Bug | ||||
| ## 2.2.14(2023-01-30) | ||||
| - 修复 某些情况切换月份错误的Bug [详情](https://ask.dcloud.net.cn/question/162033) | ||||
| ## 2.2.13(2023-01-10) | ||||
| - 修复 多次加载组件造成内存占用的Bug | ||||
| ## 2.2.12(2022-12-01) | ||||
| - 修复 vue3 下 i18n 国际化初始值不正确的Bug | ||||
| ## 2.2.11(2022-09-19) | ||||
| - 修复 支付宝小程序样式错乱的Bug [详情](https://github.com/dcloudio/uni-app/issues/3861) | ||||
| ## 2.2.10(2022-09-19) | ||||
| - 修复 反向选择日期范围,日期显示异常的Bug [详情](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) | ||||
| ## 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/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,177 @@ | ||||
| <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--today': weeks.isToday}"></view> | ||||
| 	</view> | ||||
| </template> | ||||
|  | ||||
| <script> | ||||
| 	export default { | ||||
| 		props: { | ||||
| 			weeks: { | ||||
| 				type: Object, | ||||
| 				default () { | ||||
| 					return {} | ||||
| 				} | ||||
| 			}, | ||||
| 			calendar: { | ||||
| 				type: Object, | ||||
| 				default: () => { | ||||
| 					return {} | ||||
| 				} | ||||
| 			}, | ||||
| 			selected: { | ||||
| 				type: Array, | ||||
| 				default: () => { | ||||
| 					return [] | ||||
| 				} | ||||
| 			}, | ||||
| 			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-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 { | ||||
| 		cursor: default; | ||||
| 	} | ||||
|  | ||||
| 	.uni-calendar-item--disable .uni-calendar-item__weeks-box-text-disable { | ||||
| 		color: #D1D1D1; | ||||
| 	} | ||||
|  | ||||
| 	.uni-calendar-item--today { | ||||
| 		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,928 @@ | ||||
| <template> | ||||
| 	<view class="uni-calendar" @mouseleave="leaveCale"> | ||||
|  | ||||
| 		<view v-if="!insert && show" class="uni-calendar__mask" :class="{'uni-calendar--mask-show':aniMaskShow}" | ||||
| 			@click="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 class="uni-calendar__header-btn-box" @click.stop="changeMonth('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 class="uni-calendar__header-btn-box" @click.stop="changeMonth('next')"> | ||||
| 					<view class="uni-calendar__header-btn uni-calendar--right"></view> | ||||
| 				</view> | ||||
|  | ||||
| 				<view v-if="!insert" class="dialog-close" @click="close"> | ||||
| 					<view class="dialog-close-plus" data-id="close"></view> | ||||
| 					<view class="dialog-close-plus dialog-close-rotate" data-id="close"></view> | ||||
| 				</view> | ||||
| 			</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" :checkHover="range" @change="choiceDate" | ||||
| 							@handleMouse="handleMouse"> | ||||
| 						</calendar-item> | ||||
| 					</view> | ||||
| 				</view> | ||||
| 			</view> | ||||
|  | ||||
| 			<view v-if="!insert && !range && hasTime" 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="timepickerStartTime" :end="timepickerEndTime" v-model="time" | ||||
| 					:disabled="!tempSingleDate" :border="false" :hide-second="hideSecond" class="time-picker-style"> | ||||
| 				</time-picker> | ||||
| 			</view> | ||||
|  | ||||
| 			<view v-if="!insert && range && hasTime" 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="timepickerStartTime" 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="timepickerEndTime" 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-datetime-picker--btn" @click="confirm">{{confirmText}}</view> | ||||
| 			</view> | ||||
| 		</view> | ||||
| 	</view> | ||||
| </template> | ||||
|  | ||||
| <script> | ||||
| 	import { Calendar, getDate, getTime } from './util.js'; | ||||
| 	import calendarItem from './calendar-item.vue' | ||||
| 	import timePicker from './time-picker.vue' | ||||
|  | ||||
| 	import { initVueI18n } from '@dcloudio/uni-i18n' | ||||
| 	import i18nMessages from './i18n/index.js' | ||||
| 	const { t } = initVueI18n(i18nMessages) | ||||
|  | ||||
| 	/** | ||||
| 	 * Calendar 日历 | ||||
| 	 * @description 日历组件可以查看日期,选择任意范围内的日期,打点操作。常用场景如:酒店日期预订、火车机票选择购买日期、上下班打卡等 | ||||
| 	 * @tutorial https://ext.dcloud.net.cn/plugin?id=56 | ||||
| 	 * @property {String} date 自定义当前时间,默认为今天 | ||||
| 	 * @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 是否选择月份为背景 | ||||
| 	 * @property {[String} defaultValue 选择器打开时默认显示的时间 | ||||
| 	 * @event {Function} change 日期改变,`insert :ture` 时生效 | ||||
| 	 * @event {Function} confirm 确认选择`insert :false` 时生效 | ||||
| 	 * @event {Function} monthSwitch 切换月份时触发 | ||||
| 	 * @example <uni-calendar :insert="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 [] | ||||
| 				} | ||||
| 			}, | ||||
| 			startDate: { | ||||
| 				type: String, | ||||
| 				default: '' | ||||
| 			}, | ||||
| 			endDate: { | ||||
| 				type: String, | ||||
| 				default: '' | ||||
| 			}, | ||||
|       startPlaceholder: { | ||||
|         type: String, | ||||
| 				default: '' | ||||
| 			}, | ||||
| 			endPlaceholder: { | ||||
| 				type: String, | ||||
| 				default: '' | ||||
| 			}, | ||||
| 			range: { | ||||
| 				type: Boolean, | ||||
| 				default: false | ||||
| 			}, | ||||
| 			hasTime: { | ||||
| 				type: Boolean, | ||||
| 				default: false | ||||
| 			}, | ||||
| 			insert: { | ||||
| 				type: Boolean, | ||||
| 				default: true | ||||
| 			}, | ||||
| 			showMonth: { | ||||
| 				type: Boolean, | ||||
| 				default: true | ||||
| 			}, | ||||
| 			clearDate: { | ||||
| 				type: Boolean, | ||||
| 				default: true | ||||
| 			}, | ||||
| 			checkHover: { | ||||
| 				type: Boolean, | ||||
| 				default: true | ||||
| 			}, | ||||
| 			hideSecond: { | ||||
| 				type: [Boolean], | ||||
| 				default: false | ||||
| 			}, | ||||
| 			pleStatus: { | ||||
| 				type: Object, | ||||
| 				default () { | ||||
| 					return { | ||||
| 						before: '', | ||||
| 						after: '', | ||||
| 						data: [], | ||||
| 						fulldate: '' | ||||
| 					} | ||||
| 				} | ||||
| 			}, | ||||
|       defaultValue: { | ||||
|         type: [String, Object, Array], | ||||
|         default: '' | ||||
|       } | ||||
| 		}, | ||||
| 		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) { | ||||
| 					if (!this.range) { | ||||
| 						this.tempSingleDate = newVal | ||||
| 						setTimeout(() => { | ||||
| 							this.init(newVal) | ||||
| 						}, 100) | ||||
| 					} | ||||
| 				} | ||||
| 			}, | ||||
| 			defTime: { | ||||
| 				immediate: true, | ||||
| 				handler(newVal) { | ||||
| 					if (!this.range) { | ||||
| 						this.time = newVal | ||||
| 					} else { | ||||
| 						this.timeRange.startTime = newVal.start | ||||
| 						this.timeRange.endTime = newVal.end | ||||
| 					} | ||||
| 				} | ||||
| 			}, | ||||
| 			startDate(val) { | ||||
| 				// 字节小程序 watch 早于 created | ||||
| 				if(!this.cale){ | ||||
| 					return | ||||
| 				} | ||||
| 				this.cale.setStartDate(val) | ||||
| 				this.cale.setDate(this.nowDate.fullDate) | ||||
| 				this.weeks = this.cale.weeks | ||||
| 			}, | ||||
| 			endDate(val) { | ||||
| 				// 字节小程序 watch 早于 created | ||||
| 				if(!this.cale){ | ||||
| 					return | ||||
| 				} | ||||
| 				this.cale.setEndDate(val) | ||||
| 				this.cale.setDate(this.nowDate.fullDate) | ||||
| 				this.weeks = this.cale.weeks | ||||
| 			}, | ||||
| 			selected(newVal) { | ||||
| 				// 字节小程序 watch 早于 created | ||||
| 				if(!this.cale){ | ||||
| 					return | ||||
| 				} | ||||
| 				this.cale.setSelectInfo(this.nowDate.fullDate, newVal) | ||||
| 				this.weeks = this.cale.weeks | ||||
| 			}, | ||||
| 			pleStatus: { | ||||
| 				immediate: true, | ||||
| 				handler(newVal) { | ||||
| 					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 { | ||||
|               // 字节小程序 watch 早于 created | ||||
|               if(!this.cale){ | ||||
|                 return | ||||
|               } | ||||
|  | ||||
| 							this.cale.setDefaultMultiple(before, after) | ||||
| 							if (which === 'left' && before) { | ||||
| 								this.setDate(before) | ||||
| 								this.weeks = this.cale.weeks | ||||
| 							} else if(after) { | ||||
| 								this.setDate(after) | ||||
| 								this.weeks = this.cale.weeks | ||||
| 							} | ||||
| 							this.cale.lastHover = true | ||||
| 						} | ||||
| 					}, 16) | ||||
| 				} | ||||
| 			} | ||||
| 		}, | ||||
| 		computed: { | ||||
| 			timepickerStartTime() { | ||||
| 				const activeDate = this.range ? this.tempRange.before : this.calendar.fullDate | ||||
| 				return activeDate === this.startDate ? this.selectableTimes.start : '' | ||||
| 			}, | ||||
| 			timepickerEndTime() { | ||||
| 				const activeDate = this.range ? this.tempRange.after : this.calendar.fullDate | ||||
| 				return activeDate === this.endDate ? this.selectableTimes.end : '' | ||||
| 			}, | ||||
| 			/** | ||||
| 			 * 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({ | ||||
| 				selected: this.selected, | ||||
| 				startDate: this.startDate, | ||||
| 				endDate: this.endDate, | ||||
| 				range: this.range, | ||||
| 			}) | ||||
| 			// 选中某一天 | ||||
| 			this.init(this.date) | ||||
| 		}, | ||||
| 		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 | ||||
| 			}, | ||||
| 			// 蒙版点击事件 | ||||
| 			maskClick() { | ||||
|         this.close() | ||||
| 				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(new Date()) | ||||
| 			}, | ||||
|  | ||||
| 			bindDateChange(e) { | ||||
| 				const value = e.detail.value + '-1' | ||||
| 				this.setDate(value) | ||||
| 			}, | ||||
| 			/** | ||||
| 			 * 初始化日期显示 | ||||
| 			 * @param {Object} date | ||||
| 			 */ | ||||
| 			init(date) { | ||||
|         // 字节小程序 watch 早于 created | ||||
| 				if(!this.cale){ | ||||
| 					return | ||||
| 				} | ||||
| 				this.cale.setDate(date || new Date()) | ||||
| 				this.weeks = this.cale.weeks | ||||
| 				this.nowDate = this.cale.getInfo(date) | ||||
|         this.calendar = {...this.nowDate} | ||||
|         if(!date){ | ||||
|           // 优化date为空默认不选中今天 | ||||
|           this.calendar.fullDate = '' | ||||
|           if(this.defaultValue && !this.range){ | ||||
|             // 暂时只支持移动端非范围选择 | ||||
|             const defaultDate = new Date(this.defaultValue) | ||||
|             const fullDate = getDate(defaultDate) | ||||
|             const year = defaultDate.getFullYear() | ||||
|             const month = defaultDate.getMonth()+1 | ||||
|             const date = defaultDate.getDate() | ||||
|             const day = defaultDate.getDay() | ||||
|             this.calendar = { | ||||
|               fullDate, | ||||
|               year, | ||||
|               month, | ||||
|               date, | ||||
|               day | ||||
|             }, | ||||
|             this.tempSingleDate = fullDate | ||||
|             this.time = getTime(defaultDate, this.hideSecond) | ||||
|           } | ||||
|         } | ||||
| 			}, | ||||
| 			/** | ||||
| 			 * 打开日历弹窗 | ||||
| 			 */ | ||||
| 			open() { | ||||
| 				// 弹窗模式并且清理数据 | ||||
| 				if (this.clearDate && !this.insert) { | ||||
| 					this.cale.cleanMultipleStatus() | ||||
| 					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) { | ||||
|         if(!this.range){ | ||||
| 					if(!this.calendar.fullDate){ | ||||
| 					  this.calendar = this.cale.getInfo(new Date()) | ||||
| 					  this.tempSingleDate = this.calendar.fullDate | ||||
| 					} | ||||
| 					if(this.hasTime && !this.time) { | ||||
| 					  this.time = getTime(new Date(), this.hideSecond) | ||||
| 					} | ||||
| 				} | ||||
| 				let { | ||||
| 					year, | ||||
| 					month, | ||||
| 					date, | ||||
| 					fullDate, | ||||
| 					extraInfo | ||||
| 				} = this.calendar | ||||
| 				this.$emit(name, { | ||||
| 					range: this.cale.multipleStatus, | ||||
| 					year, | ||||
| 					month, | ||||
| 					date, | ||||
| 					time: this.time, | ||||
| 					timeRange: this.timeRange, | ||||
| 					fulldate: fullDate, | ||||
| 					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 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() | ||||
| 			}, | ||||
|       changeMonth(type) { | ||||
|         let newDate | ||||
|         if(type === 'pre') { | ||||
|           newDate = this.cale.getPreMonthObj(this.nowDate.fullDate).fullDate | ||||
|         } else if(type === 'next') { | ||||
|           newDate = this.cale.getNextMonthObj(this.nowDate.fullDate).fullDate | ||||
|         } | ||||
|  | ||||
|         this.setDate(newDate) | ||||
| 				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; | ||||
|     /* #ifdef MP-TOUTIAO */ | ||||
|     font-size: 16px; | ||||
|     /* #endif */ | ||||
| 		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 date and time", | ||||
| 	"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,934 @@ | ||||
| <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> | ||||
| 	</view> | ||||
| </template> | ||||
|  | ||||
| <script> | ||||
| 	import { initVueI18n } from '@dcloudio/uni-i18n' | ||||
| 	import i18nMessages from './i18n/index.js' | ||||
| 	const {	t	} = initVueI18n(i18nMessages) | ||||
|   import { fixIosDateFormat } from './util' | ||||
|  | ||||
| 	/** | ||||
| 	 * 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', | ||||
| 		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: { | ||||
| 			// #ifndef VUE3 | ||||
| 			value: { | ||||
| 				handler(newVal) { | ||||
|           if (newVal) { | ||||
|             this.parseValue(fixIosDateFormat(newVal)) | ||||
| 						this.initTime(false) | ||||
| 					} else { | ||||
|             this.time = '' | ||||
| 						this.parseValue(Date.now()) | ||||
| 					} | ||||
| 				}, | ||||
| 				immediate: true | ||||
| 			}, | ||||
| 			// #endif | ||||
| 			// #ifdef VUE3 | ||||
| 			modelValue: { | ||||
|         handler(newVal) { | ||||
|           if (newVal) { | ||||
| 						this.parseValue(fixIosDateFormat(newVal)) | ||||
| 						this.initTime(false) | ||||
| 					} else { | ||||
| 						this.time = '' | ||||
| 						this.parseValue(Date.now()) | ||||
| 					} | ||||
| 				}, | ||||
| 				immediate: true | ||||
| 			}, | ||||
| 			// #endif | ||||
| 			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(fixIosDateFormat(newVal), 'start') | ||||
| 				}, | ||||
| 				immediate: true | ||||
| 			}, | ||||
| 			end: { | ||||
| 				handler(newVal) { | ||||
| 					this.parseDatetimeRange(fixIosDateFormat(newVal), 'end') | ||||
| 				}, | ||||
| 				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)) { | ||||
| 					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)) { | ||||
| 						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 = fixIosDateFormat(this.time) | ||||
| 				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,403 @@ | ||||
| class Calendar { | ||||
| 	constructor({ | ||||
| 		selected, | ||||
| 		startDate, | ||||
| 		endDate, | ||||
| 		range, | ||||
| 	} = {}) { | ||||
| 		// 当前日期 | ||||
| 		this.date = this.getDateObj(new Date()) // 当前初入日期 | ||||
| 		// 打点信息 | ||||
| 		this.selected = selected || []; | ||||
| 		// 起始时间 | ||||
| 		this.startDate = startDate | ||||
| 		// 终止时间 | ||||
| 		this.endDate = endDate | ||||
|     // 是否范围选择 | ||||
| 		this.range = range | ||||
| 		// 多选状态 | ||||
| 		this.cleanMultipleStatus() | ||||
| 		// 每周日期 | ||||
| 		this.weeks = {} | ||||
| 		this.lastHover = false | ||||
| 	} | ||||
| 	/** | ||||
| 	 * 设置日期 | ||||
| 	 * @param {Object} date | ||||
| 	 */ | ||||
| 	setDate(date) { | ||||
| 		const selectDate = this.getDateObj(date) | ||||
| 		this.getWeeks(selectDate.fullDate) | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 清理多选状态 | ||||
| 	 */ | ||||
| 	cleanMultipleStatus() { | ||||
| 		this.multipleStatus = { | ||||
| 			before: '', | ||||
| 			after: '', | ||||
| 			data: [] | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	setStartDate(startDate) { | ||||
| 		this.startDate = startDate | ||||
| 	} | ||||
|  | ||||
| 	setEndDate(endDate) { | ||||
| 		this.endDate = endDate | ||||
| 	} | ||||
|  | ||||
|   getPreMonthObj(date){ | ||||
|     date = fixIosDateFormat(date) | ||||
|     date = new Date(date) | ||||
|  | ||||
|     const oldMonth = date.getMonth() | ||||
|     date.setMonth(oldMonth - 1) | ||||
|     const newMonth = date.getMonth() | ||||
|     if(oldMonth !== 0 && newMonth - oldMonth === 0){ | ||||
|       date.setMonth(newMonth - 1) | ||||
|     } | ||||
|     return this.getDateObj(date) | ||||
|   } | ||||
|   getNextMonthObj(date){ | ||||
|     date = fixIosDateFormat(date) | ||||
|     date = new Date(date) | ||||
|  | ||||
|     const oldMonth = date.getMonth() | ||||
|     date.setMonth(oldMonth + 1) | ||||
|     const newMonth = date.getMonth() | ||||
|     if(newMonth - oldMonth > 1){ | ||||
|       date.setMonth(newMonth - 1) | ||||
|     } | ||||
|     return this.getDateObj(date) | ||||
|   } | ||||
|  | ||||
| 	/** | ||||
| 	 * 获取指定格式Date对象 | ||||
| 	 */ | ||||
| 	getDateObj(date) { | ||||
|     date = fixIosDateFormat(date) | ||||
|     date = new Date(date) | ||||
|  | ||||
| 		return { | ||||
| 			fullDate: getDate(date), | ||||
|       year: date.getFullYear(), | ||||
|       month: addZero(date.getMonth() + 1), | ||||
|       date: addZero(date.getDate()), | ||||
|       day: date.getDay() | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 获取上一个月日期集合 | ||||
| 	 */ | ||||
| 	getPreMonthDays(amount, dateObj) { | ||||
| 		const result = [] | ||||
| 		for (let i = amount - 1; i >= 0; i--) { | ||||
|       const month = dateObj.month - 1 | ||||
| 			result.push({ | ||||
| 				date: new Date(dateObj.year, month, -i).getDate(), | ||||
| 				month, | ||||
| 				disable: true | ||||
| 			}) | ||||
| 		} | ||||
| 		return result | ||||
| 	} | ||||
| 	/** | ||||
| 	 * 获取本月日期集合 | ||||
| 	 */ | ||||
| 	getCurrentMonthDays(amount, dateObj) { | ||||
| 		const result = [] | ||||
| 		const fullDate = this.date.fullDate | ||||
| 		for (let i = 1; i <= amount; i++) { | ||||
| 			const currentDate = `${dateObj.year}-${dateObj.month}-${addZero(i)}` | ||||
| 			const isToday = fullDate === currentDate | ||||
| 			// 获取打点信息 | ||||
| 			const info = this.selected && this.selected.find((item) => { | ||||
| 				if (this.dateEqual(currentDate, item.date)) { | ||||
| 					return item | ||||
| 				} | ||||
| 			}) | ||||
|  | ||||
| 			// 日期禁用 | ||||
| 			let disableBefore = true | ||||
| 			let disableAfter = true | ||||
| 			if (this.startDate) { | ||||
| 				disableBefore = dateCompare(this.startDate, currentDate) | ||||
| 			} | ||||
|  | ||||
| 			if (this.endDate) { | ||||
| 				disableAfter = dateCompare(currentDate, this.endDate) | ||||
| 			} | ||||
|  | ||||
| 			let multiples = this.multipleStatus.data | ||||
| 			let multiplesStatus = -1 | ||||
| 			if (this.range && multiples) { | ||||
|         multiplesStatus = multiples.findIndex((item) => { | ||||
|           return this.dateEqual(item, currentDate) | ||||
|         }) | ||||
| 			} | ||||
|       const checked = multiplesStatus !== -1 | ||||
|  | ||||
| 			result.push({ | ||||
| 				fullDate: currentDate, | ||||
| 				year: dateObj.year, | ||||
| 				date: i, | ||||
| 				multiple: this.range ? checked : false, | ||||
| 				beforeMultiple: this.isLogicBefore(currentDate, this.multipleStatus.before, this.multipleStatus.after), | ||||
| 				afterMultiple: this.isLogicAfter(currentDate, this.multipleStatus.before, this.multipleStatus.after), | ||||
| 				month: dateObj.month, | ||||
| 				disable: (this.startDate && !dateCompare(this.startDate, currentDate)) || (this.endDate && !dateCompare(currentDate,this.endDate)), | ||||
| 				isToday, | ||||
| 				userChecked: false, | ||||
|         extraInfo: info | ||||
| 			}) | ||||
| 		} | ||||
| 		return result | ||||
| 	} | ||||
| 	/** | ||||
| 	 * 获取下一个月日期集合 | ||||
| 	 */ | ||||
| 	_getNextMonthDays(amount, dateObj) { | ||||
| 		const result = [] | ||||
|     const month = dateObj.month + 1 | ||||
| 		for (let i = 1; i <= amount; i++) { | ||||
| 			result.push({ | ||||
| 				date: i, | ||||
| 				month, | ||||
| 				disable: true | ||||
| 			}) | ||||
| 		} | ||||
| 		return result | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 获取当前日期详情 | ||||
| 	 * @param {Object} date | ||||
| 	 */ | ||||
| 	getInfo(date) { | ||||
| 		if (!date) { | ||||
| 			date = new Date() | ||||
| 		} | ||||
|  | ||||
| 		return this.calendar.find(item => item.fullDate === this.getDateObj(date).fullDate) | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 比较时间是否相等 | ||||
| 	 */ | ||||
| 	dateEqual(before, after) { | ||||
| 		before = new Date(fixIosDateFormat(before)) | ||||
| 		after = new Date(fixIosDateFormat(after)) | ||||
| 		return before.valueOf() === after.valueOf() | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 *  比较真实起始日期 | ||||
| 	 */ | ||||
|  | ||||
| 	isLogicBefore(currentDate, before, after) { | ||||
| 		let logicBefore = before | ||||
| 		if (before && after) { | ||||
| 			logicBefore = dateCompare(before, after) ? before : after | ||||
| 		} | ||||
| 		return this.dateEqual(logicBefore, currentDate) | ||||
| 	} | ||||
|  | ||||
| 	isLogicAfter(currentDate, before, after) { | ||||
| 		let logicAfter = after | ||||
| 		if (before && after) { | ||||
| 			logicAfter = dateCompare(before, after) ? after : before | ||||
| 		} | ||||
| 		return this.dateEqual(logicAfter, currentDate) | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 获取日期范围内所有日期 | ||||
| 	 * @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.getDateObj(new Date(parseInt(k))).fullDate) | ||||
| 		} | ||||
| 		return arr | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 *  获取多选状态 | ||||
| 	 */ | ||||
| 	setMultiple(fullDate) { | ||||
|     if (!this.range) return | ||||
|  | ||||
| 		let { | ||||
| 			before, | ||||
| 			after | ||||
| 		} = this.multipleStatus | ||||
| 		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 (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.getWeeks(fullDate) | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 *  鼠标 hover 更新多选状态 | ||||
| 	 */ | ||||
| 	setHoverMultiple(fullDate) { | ||||
|     if (!this.range || this.lastHover) return | ||||
|  | ||||
| 		const { before } = this.multipleStatus | ||||
|  | ||||
| 		if (!before) { | ||||
| 			this.multipleStatus.before = fullDate | ||||
| 		} else { | ||||
| 			this.multipleStatus.after = fullDate | ||||
| 			if (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.getWeeks(fullDate) | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 更新默认值多选状态 | ||||
| 	 */ | ||||
| 	setDefaultMultiple(before, after) { | ||||
| 		this.multipleStatus.before = before | ||||
| 		this.multipleStatus.after = after | ||||
| 		if (before && after) { | ||||
| 			if (dateCompare(before, after)) { | ||||
| 				this.multipleStatus.data = this.geDateAll(before, after); | ||||
| 				this.getWeeks(after) | ||||
| 			} else { | ||||
| 				this.multipleStatus.data = this.geDateAll(after, before); | ||||
| 				this.getWeeks(before) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * 获取每周数据 | ||||
| 	 * @param {Object} dateData | ||||
| 	 */ | ||||
| 	getWeeks(dateData) { | ||||
| 		const { | ||||
| 			year, | ||||
| 			month, | ||||
| 		} = this.getDateObj(dateData) | ||||
|  | ||||
| 		const preMonthDayAmount = new Date(year, month - 1, 1).getDay() | ||||
|     const preMonthDays = this.getPreMonthDays(preMonthDayAmount, this.getDateObj(dateData)) | ||||
|  | ||||
| 		const currentMonthDayAmount = new Date(year, month, 0).getDate() | ||||
|     const currentMonthDays = this.getCurrentMonthDays(currentMonthDayAmount, this.getDateObj(dateData)) | ||||
|  | ||||
|     const nextMonthDayAmount = 42 - preMonthDayAmount - currentMonthDayAmount | ||||
|     const nextMonthDays = this._getNextMonthDays(nextMonthDayAmount, this.getDateObj(dateData)) | ||||
|  | ||||
| 		const calendarDays = [...preMonthDays, ...currentMonthDays, ...nextMonthDays] | ||||
|  | ||||
| 		const weeks = new Array(6) | ||||
| 		for (let i = 0; i < calendarDays.length; i++) { | ||||
|       const index = Math.floor(i / 7) | ||||
|       if(!weeks[index]){ | ||||
|         weeks[index] = new Array(7) | ||||
|       } | ||||
| 			weeks[index][i % 7] = calendarDays[i] | ||||
| 		} | ||||
|  | ||||
| 		this.calendar = calendarDays | ||||
| 		this.weeks = weeks | ||||
| 	} | ||||
| } | ||||
|  | ||||
| function getDateTime(date, hideSecond){ | ||||
|   return `${getDate(date)} ${getTime(date, hideSecond)}` | ||||
| } | ||||
|  | ||||
| function getDate(date) { | ||||
|   date = fixIosDateFormat(date) | ||||
|   date = new Date(date) | ||||
|   const year = date.getFullYear() | ||||
|   const month = date.getMonth()+1 | ||||
|   const day = date.getDate() | ||||
|   return `${year}-${addZero(month)}-${addZero(day)}` | ||||
| } | ||||
|  | ||||
| function getTime(date, hideSecond){ | ||||
|   date = fixIosDateFormat(date) | ||||
|   date = new Date(date) | ||||
|   const hour = date.getHours() | ||||
|   const minute = date.getMinutes() | ||||
|   const second = date.getSeconds() | ||||
|   return hideSecond ? `${addZero(hour)}:${addZero(minute)}` : `${addZero(hour)}:${addZero(minute)}:${addZero(second)}` | ||||
| } | ||||
|  | ||||
| function addZero(num) { | ||||
|   if(num < 10){ | ||||
|     num = `0${num}` | ||||
|   } | ||||
|   return num | ||||
| } | ||||
|  | ||||
| function getDefaultSecond(hideSecond) { | ||||
|   return hideSecond ? '00:00' : '00:00:00' | ||||
| } | ||||
|  | ||||
| function dateCompare(startDate, endDate) { | ||||
|   startDate = new Date(fixIosDateFormat(startDate)) | ||||
|   endDate = new Date(fixIosDateFormat(endDate)) | ||||
|   return startDate <= endDate | ||||
| } | ||||
|  | ||||
| function checkDate(date){ | ||||
|   const dateReg = /((19|20)\d{2})(-|\/)\d{1,2}(-|\/)\d{1,2}/g | ||||
|   return date.match(dateReg) | ||||
| } | ||||
|  | ||||
| const dateTimeReg = /^\d{4}-(0[1-9]|1[012])-(0[1-9]|[12][0-9]|3[01])( [0-5][0-9]:[0-5][0-9]:[0-5][0-9])?$/ | ||||
| function fixIosDateFormat(value) { | ||||
|   if (typeof value === 'string' && dateTimeReg.test(value)) { | ||||
|     value = value.replace(/-/g, '/') | ||||
|   } | ||||
|   return value | ||||
| } | ||||
|  | ||||
| export {Calendar, getDateTime, getDate, getTime, addZero, getDefaultSecond, dateCompare, checkDate, fixIosDateFormat} | ||||
							
								
								
									
										87
									
								
								uni_modules/uni-datetime-picker/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								uni_modules/uni-datetime-picker/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | ||||
| { | ||||
|   "id": "uni-datetime-picker", | ||||
|   "displayName": "uni-datetime-picker 日期选择器", | ||||
|   "version": "2.2.22", | ||||
|   "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" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										21
									
								
								uni_modules/uni-datetime-picker/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								uni_modules/uni-datetime-picker/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -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  | ||||
							
								
								
									
										13
									
								
								uni_modules/uni-drawer/changelog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								uni_modules/uni-drawer/changelog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -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目录规范 | ||||
							
								
								
									
										45
									
								
								uni_modules/uni-drawer/components/uni-drawer/keypress.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								uni_modules/uni-drawer/components/uni-drawer/keypress.js
									
									
									
									
									
										Normal file
									
								
							| @@ -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 | ||||
							
								
								
									
										183
									
								
								uni_modules/uni-drawer/components/uni-drawer/uni-drawer.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										183
									
								
								uni_modules/uni-drawer/components/uni-drawer/uni-drawer.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -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" > | ||||
| 	$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> | ||||
							
								
								
									
										87
									
								
								uni_modules/uni-drawer/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								uni_modules/uni-drawer/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -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" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										10
									
								
								uni_modules/uni-drawer/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								uni_modules/uni-drawer/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
|  | ||||
|  | ||||
| ## Drawer 抽屉 | ||||
| > **组件名:uni-drawer** | ||||
| > 代码块: `uDrawer` | ||||
|  | ||||
| 抽屉侧滑菜单。 | ||||
|  | ||||
| ### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-drawer) | ||||
| #### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839  | ||||
							
								
								
									
										97
									
								
								uni_modules/uni-easyinput/changelog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										97
									
								
								uni_modules/uni-easyinput/changelog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,97 @@ | ||||
| ## 1.1.9(2023-04-11) | ||||
| - 修复 vue3 下 keyboardheightchange 事件报错的bug | ||||
| ## 1.1.8(2023-03-29) | ||||
| - 优化 trim 属性默认值 | ||||
| ## 1.1.7(2023-03-29) | ||||
| - 新增 cursor-spacing 属性 | ||||
| ## 1.1.6(2023-01-28) | ||||
| - 新增 keyboardheightchange 事件,可监听键盘高度变化 | ||||
| ## 1.1.5(2022-11-29) | ||||
| - 优化 主题样式 | ||||
| ## 1.1.4(2022-10-27) | ||||
| - 修复 props 中背景颜色无默认值的bug | ||||
| ## 1.1.0(2022-06-30) | ||||
|  | ||||
| - 新增 在 uni-forms 1.4.0 中使用可以在 blur 时校验内容 | ||||
| - 新增 clear 事件,点击右侧叉号图标触发 | ||||
| - 新增 change 事件 ,仅在输入框失去焦点或用户按下回车时触发 | ||||
| - 优化 组件样式,组件获取焦点时高亮显示,图标颜色调整等 | ||||
|  | ||||
| ## 1.0.5(2022-06-07) | ||||
|  | ||||
| - 优化 clearable 显示策略 | ||||
|  | ||||
| ## 1.0.4(2022-06-07) | ||||
|  | ||||
| - 优化 clearable 显示策略 | ||||
|  | ||||
| ## 1.0.3(2022-05-20) | ||||
|  | ||||
| - 修复 关闭图标某些情况下无法取消的 bug | ||||
|  | ||||
| ## 1.0.2(2022-04-12) | ||||
|  | ||||
| - 修复 默认值不生效的 bug | ||||
|  | ||||
| ## 1.0.1(2022-04-02) | ||||
|  | ||||
| - 修复 value 不能为 0 的 bug | ||||
|  | ||||
| ## 1.0.0(2021-11-19) | ||||
|  | ||||
| - 优化 组件 UI,并提供设计资源,详见:[https://uniapp.dcloud.io/component/uniui/resource](https://uniapp.dcloud.io/component/uniui/resource) | ||||
| - 文档迁移,详见:[https://uniapp.dcloud.io/component/uniui/uni-easyinput](https://uniapp.dcloud.io/component/uniui/uni-easyinput) | ||||
|  | ||||
| ## 0.1.4(2021-08-20) | ||||
|  | ||||
| - 修复 在 uni-forms 的动态表单中默认值校验不通过的 bug | ||||
|  | ||||
| ## 0.1.3(2021-08-11) | ||||
|  | ||||
| - 修复 在 uni-forms 中重置表单,错误信息无法清除的问题 | ||||
|  | ||||
| ## 0.1.2(2021-07-30) | ||||
|  | ||||
| - 优化 vue3 下事件警告的问题 | ||||
|  | ||||
| ## 0.1.1 | ||||
|  | ||||
| - 优化 errorMessage 属性支持 Boolean 类型 | ||||
|  | ||||
| ## 0.1.0(2021-07-13) | ||||
|  | ||||
| - 组件兼容 vue3,如何创建 vue3 项目,详见 [uni-app 项目支持 vue3 介绍](https://ask.dcloud.net.cn/article/37834) | ||||
|  | ||||
| ## 0.0.16(2021-06-29) | ||||
|  | ||||
| - 修复 confirmType 属性(仅 type="text" 生效)导致多行文本框无法换行的 bug | ||||
|  | ||||
| ## 0.0.15(2021-06-21) | ||||
|  | ||||
| - 修复 passwordIcon 属性拼写错误的 bug | ||||
|  | ||||
| ## 0.0.14(2021-06-18) | ||||
|  | ||||
| - 新增 passwordIcon 属性,当 type=password 时是否显示小眼睛图标 | ||||
| - 修复 confirmType 属性不生效的问题 | ||||
|  | ||||
| ## 0.0.13(2021-06-04) | ||||
|  | ||||
| - 修复 disabled 状态可清出内容的 bug | ||||
|  | ||||
| ## 0.0.12(2021-05-12) | ||||
|  | ||||
| - 新增 组件示例地址 | ||||
|  | ||||
| ## 0.0.11(2021-05-07) | ||||
|  | ||||
| - 修复 input-border 属性不生效的问题 | ||||
|  | ||||
| ## 0.0.10(2021-04-30) | ||||
|  | ||||
| - 修复 ios 遮挡文字、显示一半的问题 | ||||
|  | ||||
| ## 0.0.9(2021-02-05) | ||||
|  | ||||
| - 调整为 uni_modules 目录规范 | ||||
| - 优化 兼容 nvue 页面 | ||||
							
								
								
									
										56
									
								
								uni_modules/uni-easyinput/components/uni-easyinput/common.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								uni_modules/uni-easyinput/components/uni-easyinput/common.js
									
									
									
									
									
										Normal file
									
								
							| @@ -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,657 @@ | ||||
| <template> | ||||
| 	<view class="uni-easyinput" :class="{ 'uni-easyinput-error': msg }" :style="boxStyle"> | ||||
| 		<view class="uni-easyinput__content" :class="inputContentClass" :style="inputContentStyle"> | ||||
| 			<uni-icons v-if="prefixIcon" class="content-clear-icon" :type="prefixIcon" color="#c0c4cc" @click="onClickIcon('prefix')" size="22"></uni-icons> | ||||
| 			<textarea | ||||
| 				v-if="type === 'textarea'" | ||||
| 				class="uni-easyinput__content-textarea" | ||||
| 				:class="{ 'input-padding': inputBorder }" | ||||
| 				:name="name" | ||||
| 				:value="val" | ||||
| 				:placeholder="placeholder" | ||||
| 				:placeholderStyle="placeholderStyle" | ||||
| 				:disabled="disabled" | ||||
| 				placeholder-class="uni-easyinput__placeholder-class" | ||||
| 				:maxlength="inputMaxlength" | ||||
| 				:focus="focused" | ||||
| 				:autoHeight="autoHeight" | ||||
| 				:cursor-spacing="cursorSpacing" | ||||
| 				@input="onInput" | ||||
| 				@blur="_Blur" | ||||
| 				@focus="_Focus" | ||||
| 				@confirm="onConfirm" | ||||
|         @keyboardheightchange="onkeyboardheightchange" | ||||
| 			></textarea> | ||||
| 			<input | ||||
| 				v-else | ||||
| 				:type="type === 'password' ? 'text' : type" | ||||
| 				class="uni-easyinput__content-input" | ||||
| 				:style="inputStyle" | ||||
| 				:name="name" | ||||
| 				:value="val" | ||||
| 				:password="!showPassword && type === 'password'" | ||||
| 				:placeholder="placeholder" | ||||
| 				:placeholderStyle="placeholderStyle" | ||||
| 				placeholder-class="uni-easyinput__placeholder-class" | ||||
| 				:disabled="disabled" | ||||
| 				:maxlength="inputMaxlength" | ||||
| 				:focus="focused" | ||||
| 				:confirmType="confirmType" | ||||
| 				:cursor-spacing="cursorSpacing" | ||||
| 				@focus="_Focus" | ||||
| 				@blur="_Blur" | ||||
| 				@input="onInput" | ||||
| 				@confirm="onConfirm" | ||||
|         @keyboardheightchange="onkeyboardheightchange" | ||||
| 			/> | ||||
| 			<template v-if="type === 'password' && passwordIcon"> | ||||
| 				<!-- 开启密码时显示小眼睛 --> | ||||
| 				<uni-icons | ||||
| 					v-if="isVal" | ||||
| 					class="content-clear-icon" | ||||
| 					:class="{ 'is-textarea-icon': type === 'textarea' }" | ||||
| 					:type="showPassword ? 'eye-slash-filled' : 'eye-filled'" | ||||
| 					:size="22" | ||||
| 					:color="focusShow ? primaryColor : '#c0c4cc'" | ||||
| 					@click="onEyes" | ||||
| 				></uni-icons> | ||||
| 			</template> | ||||
| 			<template v-else-if="suffixIcon"> | ||||
| 				<uni-icons v-if="suffixIcon" class="content-clear-icon" :type="suffixIcon" color="#c0c4cc" @click="onClickIcon('suffix')" size="22"></uni-icons> | ||||
| 			</template> | ||||
| 			<template v-else> | ||||
| 				<uni-icons | ||||
| 					v-if="clearable && isVal && !disabled && type !== 'textarea'" | ||||
| 					class="content-clear-icon" | ||||
| 					:class="{ 'is-textarea-icon': type === 'textarea' }" | ||||
| 					type="clear" | ||||
| 					:size="clearSize" | ||||
| 					:color="msg ? '#dd524d' : focusShow ? primaryColor : '#c0c4cc'" | ||||
| 					@click="onClear" | ||||
| 				></uni-icons> | ||||
| 			</template> | ||||
| 			<slot name="right"></slot> | ||||
| 		</view> | ||||
| 	</view> | ||||
| </template> | ||||
|  | ||||
| <script> | ||||
| /** | ||||
|  * Easyinput 输入框 | ||||
|  * @description 此组件可以实现表单的输入与校验,包括 "text" 和 "textarea" 类型。 | ||||
|  * @tutorial https://ext.dcloud.net.cn/plugin?id=3455 | ||||
|  * @property {String}	value	输入内容 | ||||
|  * @property {String }	type	输入框的类型(默认text) password/text/textarea/.. | ||||
|  * 	@value text			文本输入键盘 | ||||
|  * 	@value textarea	多行文本输入键盘 | ||||
|  * 	@value password	密码输入键盘 | ||||
|  * 	@value number		数字输入键盘,注意iOS上app-vue弹出的数字键盘并非9宫格方式 | ||||
|  * 	@value idcard		身份证输入键盘,信、支付宝、百度、QQ小程序 | ||||
|  * 	@value digit		带小数点的数字键盘	,App的nvue页面、微信、支付宝、百度、头条、QQ小程序支持 | ||||
|  * @property {Boolean}	clearable	是否显示右侧清空内容的图标控件,点击可清空输入框内容(默认true) | ||||
|  * @property {Boolean}	autoHeight	是否自动增高输入区域,type为textarea时有效(默认true) | ||||
|  * @property {String }	placeholder	输入框的提示文字 | ||||
|  * @property {String }	placeholderStyle	placeholder的样式(内联样式,字符串),如"color: #ddd" | ||||
|  * @property {Boolean}	focus	是否自动获得焦点(默认false) | ||||
|  * @property {Boolean}	disabled	是否禁用(默认false) | ||||
|  * @property {Number }	maxlength	最大输入长度,设置为 -1 的时候不限制最大长度(默认140) | ||||
|  * @property {String }	confirmType	设置键盘右下角按钮的文字,仅在type="text"时生效(默认done) | ||||
|  * @property {Number }	clearSize	清除图标的大小,单位px(默认15) | ||||
|  * @property {String}	prefixIcon	输入框头部图标 | ||||
|  * @property {String}	suffixIcon	输入框尾部图标 | ||||
|  * @property {String}	primaryColor	设置主题色(默认#2979ff) | ||||
|  * @property {Boolean}	trim	是否自动去除两端的空格 | ||||
|  * @property {Boolean}	cursorSpacing	指定光标与键盘的距离,单位 px | ||||
|  * @value both	去除两端空格 | ||||
|  * @value left	去除左侧空格 | ||||
|  * @value right	去除右侧空格 | ||||
|  * @value start	去除左侧空格 | ||||
|  * @value end		去除右侧空格 | ||||
|  * @value all		去除全部空格 | ||||
|  * @value none	不去除空格 | ||||
|  * @property {Boolean}	inputBorder	是否显示input输入框的边框(默认true) | ||||
|  * @property {Boolean}	passwordIcon	type=password时是否显示小眼睛图标 | ||||
|  * @property {Object}	styles	自定义颜色 | ||||
|  * @event {Function}	input	输入框内容发生变化时触发 | ||||
|  * @event {Function}	focus	输入框获得焦点时触发 | ||||
|  * @event {Function}	blur	输入框失去焦点时触发 | ||||
|  * @event {Function}	confirm	点击完成按钮时触发 | ||||
|  * @event {Function}	iconClick	点击图标时触发 | ||||
|  * @example <uni-easyinput v-model="mobile"></uni-easyinput> | ||||
|  */ | ||||
| function obj2strClass(obj) { | ||||
| 	let classess = ''; | ||||
| 	for (let key in obj) { | ||||
| 		const val = obj[key]; | ||||
| 		if (val) { | ||||
| 			classess += `${key} `; | ||||
| 		} | ||||
| 	} | ||||
| 	return classess; | ||||
| } | ||||
|  | ||||
| function obj2strStyle(obj) { | ||||
| 	let style = ''; | ||||
| 	for (let key in obj) { | ||||
| 		const val = obj[key]; | ||||
| 		style += `${key}:${val};`; | ||||
| 	} | ||||
| 	return style; | ||||
| } | ||||
| export default { | ||||
| 	name: 'uni-easyinput', | ||||
| 	emits: ['click', 'iconClick', 'update:modelValue', 'input', 'focus', 'blur', 'confirm', 'clear', 'eyes', 'change', 'keyboardheightchange'], | ||||
| 	model: { | ||||
| 		prop: 'modelValue', | ||||
| 		event: 'update:modelValue' | ||||
| 	}, | ||||
| 	options: { | ||||
| 		virtualHost: true | ||||
| 	}, | ||||
| 	inject: { | ||||
| 		form: { | ||||
| 			from: 'uniForm', | ||||
| 			default: null | ||||
| 		}, | ||||
| 		formItem: { | ||||
| 			from: 'uniFormItem', | ||||
| 			default: null | ||||
| 		} | ||||
| 	}, | ||||
| 	props: { | ||||
| 		name: String, | ||||
| 		value: [Number, String], | ||||
| 		modelValue: [Number, String], | ||||
| 		type: { | ||||
| 			type: String, | ||||
| 			default: 'text' | ||||
| 		}, | ||||
| 		clearable: { | ||||
| 			type: Boolean, | ||||
| 			default: true | ||||
| 		}, | ||||
| 		autoHeight: { | ||||
| 			type: Boolean, | ||||
| 			default: false | ||||
| 		}, | ||||
| 		placeholder: { | ||||
| 			type: String, | ||||
| 			default: ' ' | ||||
| 		}, | ||||
| 		placeholderStyle: String, | ||||
| 		focus: { | ||||
| 			type: Boolean, | ||||
| 			default: false | ||||
| 		}, | ||||
| 		disabled: { | ||||
| 			type: Boolean, | ||||
| 			default: false | ||||
| 		}, | ||||
| 		maxlength: { | ||||
| 			type: [Number, String], | ||||
| 			default: 140 | ||||
| 		}, | ||||
| 		confirmType: { | ||||
| 			type: String, | ||||
| 			default: 'done' | ||||
| 		}, | ||||
| 		clearSize: { | ||||
| 			type: [Number, String], | ||||
| 			default: 24 | ||||
| 		}, | ||||
| 		inputBorder: { | ||||
| 			type: Boolean, | ||||
| 			default: true | ||||
| 		}, | ||||
| 		prefixIcon: { | ||||
| 			type: String, | ||||
| 			default: '' | ||||
| 		}, | ||||
| 		suffixIcon: { | ||||
| 			type: String, | ||||
| 			default: '' | ||||
| 		}, | ||||
| 		trim: { | ||||
| 			type: [Boolean, String], | ||||
| 			default: false | ||||
| 		}, | ||||
| 		cursorSpacing: { | ||||
| 			type: Number, | ||||
| 			default: 0 | ||||
| 		}, | ||||
| 		passwordIcon: { | ||||
| 			type: Boolean, | ||||
| 			default: true | ||||
| 		}, | ||||
| 		primaryColor: { | ||||
| 			type: String, | ||||
| 			default: '#2979ff' | ||||
| 		}, | ||||
| 		styles: { | ||||
| 			type: Object, | ||||
| 			default() { | ||||
| 				return { | ||||
| 					color: '#333', | ||||
| 					backgroundColor: '#fff', | ||||
| 					disableColor: '#F7F6F6', | ||||
| 					borderColor: '#e5e5e5' | ||||
| 				}; | ||||
| 			} | ||||
| 		}, | ||||
| 		errorMessage: { | ||||
| 			type: [String, Boolean], | ||||
| 			default: '' | ||||
| 		} | ||||
| 	}, | ||||
| 	data() { | ||||
| 		return { | ||||
| 			focused: false, | ||||
| 			val: '', | ||||
| 			showMsg: '', | ||||
| 			border: false, | ||||
| 			isFirstBorder: false, | ||||
| 			showClearIcon: false, | ||||
| 			showPassword: false, | ||||
| 			focusShow: false, | ||||
| 			localMsg: '', | ||||
| 			isEnter: false // 用于判断当前是否是使用回车操作 | ||||
| 		}; | ||||
| 	}, | ||||
| 	computed: { | ||||
| 		// 输入框内是否有值 | ||||
| 		isVal() { | ||||
| 			const val = this.val; | ||||
| 			// fixed by mehaotian 处理值为0的情况,字符串0不在处理范围 | ||||
| 			if (val || val === 0) { | ||||
| 				return true; | ||||
| 			} | ||||
| 			return false; | ||||
| 		}, | ||||
|  | ||||
| 		msg() { | ||||
| 			// console.log('computed', this.form, this.formItem); | ||||
| 			// if (this.form) { | ||||
| 			// 	return this.errorMessage || this.formItem.errMsg; | ||||
| 			// } | ||||
| 			// TODO 处理头条 formItem 中 errMsg 不更新的问题 | ||||
| 			return this.localMsg || this.errorMessage; | ||||
| 		}, | ||||
| 		// 因为uniapp的input组件的maxlength组件必须要数值,这里转为数值,用户可以传入字符串数值 | ||||
| 		inputMaxlength() { | ||||
| 			return Number(this.maxlength); | ||||
| 		}, | ||||
|  | ||||
| 		// 处理外层样式的style | ||||
| 		boxStyle() { | ||||
| 			return `color:${this.inputBorder && this.msg ? '#e43d33' : this.styles.color};`; | ||||
| 		}, | ||||
| 		// input 内容的类和样式处理 | ||||
| 		inputContentClass() { | ||||
| 			return obj2strClass({ | ||||
| 				'is-input-border': this.inputBorder, | ||||
| 				'is-input-error-border': this.inputBorder && this.msg, | ||||
| 				'is-textarea': this.type === 'textarea', | ||||
| 				'is-disabled': this.disabled, | ||||
| 				'is-focused': this.focusShow | ||||
| 			}); | ||||
| 		}, | ||||
| 		inputContentStyle() { | ||||
| 			const focusColor = this.focusShow ? this.primaryColor : this.styles.borderColor; | ||||
| 			const borderColor = this.inputBorder && this.msg ? '#dd524d' : focusColor; | ||||
| 			return obj2strStyle({ | ||||
| 				'border-color': borderColor || '#e5e5e5', | ||||
| 				'background-color': this.disabled ? this.styles.disableColor : this.styles.backgroundColor | ||||
| 			}); | ||||
| 		}, | ||||
| 		// input右侧样式 | ||||
| 		inputStyle() { | ||||
| 			const paddingRight = this.type === 'password' || this.clearable || this.prefixIcon ? '' : '10px'; | ||||
| 			return obj2strStyle({ | ||||
| 				'padding-right': paddingRight, | ||||
| 				'padding-left': this.prefixIcon ? '' : '10px' | ||||
| 			}); | ||||
| 		} | ||||
| 	}, | ||||
| 	watch: { | ||||
| 		value(newVal) { | ||||
| 			this.val = newVal; | ||||
| 		}, | ||||
| 		modelValue(newVal) { | ||||
| 			this.val = newVal; | ||||
| 		}, | ||||
| 		focus(newVal) { | ||||
| 			this.$nextTick(() => { | ||||
| 				this.focused = this.focus; | ||||
| 				this.focusShow = this.focus; | ||||
| 			}); | ||||
| 		} | ||||
| 	}, | ||||
| 	created() { | ||||
| 		this.init(); | ||||
| 		// TODO 处理头条vue3 computed 不监听 inject 更改的问题(formItem.errMsg) | ||||
| 		if (this.form && this.formItem) { | ||||
| 			this.$watch('formItem.errMsg', newVal => { | ||||
| 				this.localMsg = newVal; | ||||
| 			}); | ||||
| 		} | ||||
| 	}, | ||||
| 	mounted() { | ||||
| 		this.$nextTick(() => { | ||||
| 			this.focused = this.focus; | ||||
| 			this.focusShow = this.focus; | ||||
| 		}); | ||||
| 	}, | ||||
| 	methods: { | ||||
| 		/** | ||||
| 		 * 初始化变量值 | ||||
| 		 */ | ||||
| 		init() { | ||||
| 			if (this.value || this.value === 0) { | ||||
| 				this.val = this.value; | ||||
| 			} else if (this.modelValue || this.modelValue === 0 || this.modelValue === '') { | ||||
| 				this.val = this.modelValue; | ||||
| 			} else { | ||||
| 				this.val = null; | ||||
| 			} | ||||
| 		}, | ||||
|  | ||||
| 		/** | ||||
| 		 * 点击图标时触发 | ||||
| 		 * @param {Object} type | ||||
| 		 */ | ||||
| 		onClickIcon(type) { | ||||
| 			this.$emit('iconClick', type); | ||||
| 		}, | ||||
|  | ||||
| 		/** | ||||
| 		 * 显示隐藏内容,密码框时生效 | ||||
| 		 */ | ||||
| 		onEyes() { | ||||
| 			this.showPassword = !this.showPassword; | ||||
| 			this.$emit('eyes', this.showPassword); | ||||
| 		}, | ||||
|  | ||||
| 		/** | ||||
| 		 * 输入时触发 | ||||
| 		 * @param {Object} event | ||||
| 		 */ | ||||
| 		onInput(event) { | ||||
| 			let value = event.detail.value; | ||||
| 			// 判断是否去除空格 | ||||
| 			if (this.trim) { | ||||
| 				if (typeof this.trim === 'boolean' && this.trim) { | ||||
| 					value = this.trimStr(value); | ||||
| 				} | ||||
| 				if (typeof this.trim === 'string') { | ||||
| 					value = this.trimStr(value, this.trim); | ||||
| 				} | ||||
| 			} | ||||
| 			if (this.errMsg) this.errMsg = ''; | ||||
| 			this.val = value; | ||||
| 			// TODO 兼容 vue2 | ||||
| 			this.$emit('input', value); | ||||
| 			// TODO 兼容 vue3 | ||||
| 			this.$emit('update:modelValue', value); | ||||
| 		}, | ||||
|  | ||||
| 		/** | ||||
| 		 * 外部调用方法 | ||||
| 		 * 获取焦点时触发 | ||||
| 		 * @param {Object} event | ||||
| 		 */ | ||||
| 		onFocus() { | ||||
| 			this.$nextTick(() => { | ||||
| 				this.focused = true; | ||||
| 			}); | ||||
| 			this.$emit('focus', null); | ||||
| 		}, | ||||
|  | ||||
| 		_Focus(event) { | ||||
| 			this.focusShow = true; | ||||
| 			this.$emit('focus', event); | ||||
| 		}, | ||||
|  | ||||
| 		/** | ||||
| 		 * 外部调用方法 | ||||
| 		 * 失去焦点时触发 | ||||
| 		 * @param {Object} event | ||||
| 		 */ | ||||
| 		onBlur() { | ||||
| 			this.focused = false; | ||||
| 			this.$emit('focus', null); | ||||
| 		}, | ||||
| 		_Blur(event) { | ||||
| 			let value = event.detail.value; | ||||
| 			this.focusShow = false; | ||||
| 			this.$emit('blur', event); | ||||
| 			// 根据类型返回值,在event中获取的值理论上讲都是string | ||||
| 			if (this.isEnter === false) { | ||||
| 				this.$emit('change', this.val); | ||||
| 			} | ||||
| 			// 失去焦点时参与表单校验 | ||||
| 			if (this.form && this.formItem) { | ||||
| 				const { validateTrigger } = this.form; | ||||
| 				if (validateTrigger === 'blur') { | ||||
| 					this.formItem.onFieldChange(); | ||||
| 				} | ||||
| 			} | ||||
| 		}, | ||||
|  | ||||
| 		/** | ||||
| 		 * 按下键盘的发送键 | ||||
| 		 * @param {Object} e | ||||
| 		 */ | ||||
| 		onConfirm(e) { | ||||
| 			this.$emit('confirm', this.val); | ||||
| 			this.isEnter = true; | ||||
| 			this.$emit('change', this.val); | ||||
| 			this.$nextTick(() => { | ||||
| 				this.isEnter = false; | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		/** | ||||
| 		 * 清理内容 | ||||
| 		 * @param {Object} event | ||||
| 		 */ | ||||
| 		onClear(event) { | ||||
| 			this.val = ''; | ||||
| 			// TODO 兼容 vue2 | ||||
| 			this.$emit('input', ''); | ||||
| 			// TODO 兼容 vue2 | ||||
| 			// TODO 兼容 vue3 | ||||
| 			this.$emit('update:modelValue', ''); | ||||
| 			// 点击叉号触发 | ||||
| 			this.$emit('clear'); | ||||
| 		}, | ||||
|  | ||||
|     /** | ||||
|      * 键盘高度发生变化的时候触发此事件 | ||||
|      * 兼容性:微信小程序2.7.0+、App 3.1.0+ | ||||
|      * @param {Object} event | ||||
|      */ | ||||
|     onkeyboardheightchange(event) { | ||||
|       this.$emit("keyboardheightchange",event); | ||||
|     }, | ||||
|  | ||||
| 		/** | ||||
| 		 * 去除空格 | ||||
| 		 */ | ||||
| 		trimStr(str, pos = 'both') { | ||||
| 			if (pos === 'both') { | ||||
| 				return str.trim(); | ||||
| 			} else if (pos === 'left') { | ||||
| 				return str.trimLeft(); | ||||
| 			} else if (pos === 'right') { | ||||
| 				return str.trimRight(); | ||||
| 			} else if (pos === 'start') { | ||||
| 				return str.trimStart(); | ||||
| 			} else if (pos === 'end') { | ||||
| 				return str.trimEnd(); | ||||
| 			} else if (pos === 'all') { | ||||
| 				return str.replace(/\s+/g, ''); | ||||
| 			} else if (pos === 'none') { | ||||
| 				return str; | ||||
| 			} | ||||
| 			return str; | ||||
| 		} | ||||
| 	} | ||||
| }; | ||||
| </script> | ||||
|  | ||||
| <style lang="scss"> | ||||
| $uni-error: #e43d33; | ||||
| $uni-border-1: #dcdfe6 !default; | ||||
|  | ||||
| .uni-easyinput { | ||||
| 	/* #ifndef APP-NVUE */ | ||||
| 	width: 100%; | ||||
| 	/* #endif */ | ||||
| 	flex: 1; | ||||
| 	position: relative; | ||||
| 	text-align: left; | ||||
| 	color: #333; | ||||
| 	font-size: 14px; | ||||
| } | ||||
|  | ||||
| .uni-easyinput__content { | ||||
| 	flex: 1; | ||||
| 	/* #ifndef APP-NVUE */ | ||||
| 	width: 100%; | ||||
| 	display: flex; | ||||
| 	box-sizing: border-box; | ||||
| 	// min-height: 36px; | ||||
| 	/* #endif */ | ||||
| 	flex-direction: row; | ||||
| 	align-items: center; | ||||
| 	// 处理border动画刚开始显示黑色的问题 | ||||
| 	border-color: #fff; | ||||
| 	transition-property: border-color; | ||||
| 	transition-duration: 0.3s; | ||||
| } | ||||
|  | ||||
| .uni-easyinput__content-input { | ||||
| 	/* #ifndef APP-NVUE */ | ||||
| 	width: auto; | ||||
| 	/* #endif */ | ||||
| 	position: relative; | ||||
| 	overflow: hidden; | ||||
| 	flex: 1; | ||||
| 	line-height: 1; | ||||
| 	font-size: 14px; | ||||
| 	height: 35px; | ||||
| 	// min-height: 36px; | ||||
| } | ||||
|  | ||||
| .uni-easyinput__placeholder-class { | ||||
| 	color: #999; | ||||
| 	font-size: 12px; | ||||
| 	// font-weight: 200; | ||||
| } | ||||
|  | ||||
| .is-textarea { | ||||
| 	align-items: flex-start; | ||||
| } | ||||
|  | ||||
| .is-textarea-icon { | ||||
| 	margin-top: 5px; | ||||
| } | ||||
|  | ||||
| .uni-easyinput__content-textarea { | ||||
| 	position: relative; | ||||
| 	overflow: hidden; | ||||
| 	flex: 1; | ||||
| 	line-height: 1.5; | ||||
| 	font-size: 14px; | ||||
| 	margin: 6px; | ||||
| 	margin-left: 0; | ||||
| 	height: 80px; | ||||
| 	min-height: 80px; | ||||
| 	/* #ifndef APP-NVUE */ | ||||
| 	min-height: 80px; | ||||
| 	width: auto; | ||||
| 	/* #endif */ | ||||
| } | ||||
|  | ||||
| .input-padding { | ||||
| 	padding-left: 10px; | ||||
| } | ||||
|  | ||||
| .content-clear-icon { | ||||
| 	padding: 0 5px; | ||||
| } | ||||
|  | ||||
| .label-icon { | ||||
| 	margin-right: 5px; | ||||
| 	margin-top: -1px; | ||||
| } | ||||
|  | ||||
| // 显示边框 | ||||
| .is-input-border { | ||||
| 	/* #ifndef APP-NVUE */ | ||||
| 	display: flex; | ||||
| 	box-sizing: border-box; | ||||
| 	/* #endif */ | ||||
| 	flex-direction: row; | ||||
| 	align-items: center; | ||||
| 	border: 1px solid $uni-border-1; | ||||
| 	border-radius: 4px; | ||||
| 	/* #ifdef MP-ALIPAY */ | ||||
| 	overflow: hidden; | ||||
| 	/* #endif */ | ||||
| } | ||||
|  | ||||
| .uni-error-message { | ||||
| 	position: absolute; | ||||
| 	bottom: -17px; | ||||
| 	left: 0; | ||||
| 	line-height: 12px; | ||||
| 	color: $uni-error; | ||||
| 	font-size: 12px; | ||||
| 	text-align: left; | ||||
| } | ||||
|  | ||||
| .uni-error-msg--boeder { | ||||
| 	position: relative; | ||||
| 	bottom: 0; | ||||
| 	line-height: 22px; | ||||
| } | ||||
|  | ||||
| .is-input-error-border { | ||||
| 	border-color: $uni-error; | ||||
|  | ||||
| 	.uni-easyinput__placeholder-class { | ||||
| 		color: mix(#fff, $uni-error, 50%); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| .uni-easyinput--border { | ||||
| 	margin-bottom: 0; | ||||
| 	padding: 10px 15px; | ||||
| 	// padding-bottom: 0; | ||||
| 	border-top: 1px #eee solid; | ||||
| } | ||||
|  | ||||
| .uni-easyinput-error { | ||||
| 	padding-bottom: 0; | ||||
| } | ||||
|  | ||||
| .is-first-border { | ||||
| 	/* #ifndef APP-NVUE */ | ||||
| 	border: none; | ||||
| 	/* #endif */ | ||||
| 	/* #ifdef APP-NVUE */ | ||||
| 	border-width: 0; | ||||
| 	/* #endif */ | ||||
| } | ||||
|  | ||||
| .is-disabled { | ||||
| 	background-color: #f7f6f6; | ||||
| 	color: #d5d5d5; | ||||
|  | ||||
| 	.uni-easyinput__placeholder-class { | ||||
| 		color: #d5d5d5; | ||||
| 		font-size: 12px; | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
							
								
								
									
										87
									
								
								uni_modules/uni-easyinput/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								uni_modules/uni-easyinput/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | ||||
| { | ||||
|   "id": "uni-easyinput", | ||||
|   "displayName": "uni-easyinput 增强输入框", | ||||
|   "version": "1.1.9", | ||||
|   "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" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										11
									
								
								uni_modules/uni-easyinput/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								uni_modules/uni-easyinput/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -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  | ||||
							
								
								
									
										23
									
								
								uni_modules/uni-fab/changelog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								uni_modules/uni-fab/changelog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| ## 1.2.5(2023-03-29) | ||||
| - 新增 pattern.icon 属性,可自定义图标 | ||||
| ## 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端 | ||||
							
								
								
									
										491
									
								
								uni_modules/uni-fab/components/uni-fab/uni-fab.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										491
									
								
								uni_modules/uni-fab/components/uni-fab/uni-fab.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,491 @@ | ||||
| <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="styles.icon" :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', | ||||
| 					icon: 'plusempty' | ||||
| 				} | ||||
| 			} | ||||
| 		}, | ||||
| 		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) { | ||||
| 				if (!this.isShow) { | ||||
| 					return | ||||
| 				} | ||||
| 				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> | ||||
							
								
								
									
										84
									
								
								uni_modules/uni-fab/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								uni_modules/uni-fab/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,84 @@ | ||||
| { | ||||
|   "id": "uni-fab", | ||||
|   "displayName": "uni-fab 悬浮按钮", | ||||
|   "version": "1.2.5", | ||||
|   "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" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										9
									
								
								uni_modules/uni-fab/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								uni_modules/uni-fab/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| ## Fab 悬浮按钮 | ||||
| > **组件名:uni-fab** | ||||
| > 代码块: `uFab` | ||||
|  | ||||
|  | ||||
| 点击可展开一个图形按钮菜单 | ||||
|  | ||||
| ### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-fab) | ||||
| #### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839  | ||||
							
								
								
									
										19
									
								
								uni_modules/uni-fav/changelog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								uni_modules/uni-fav/changelog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -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目录规范 | ||||
							
								
								
									
										4
									
								
								uni_modules/uni-fav/components/uni-fav/i18n/en.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								uni_modules/uni-fav/components/uni-fav/i18n/en.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| { | ||||
| 	"uni-fav.collect": "collect", | ||||
| 	"uni-fav.collected": "collected" | ||||
| } | ||||
							
								
								
									
										8
									
								
								uni_modules/uni-fav/components/uni-fav/i18n/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								uni_modules/uni-fav/components/uni-fav/i18n/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -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 | ||||
| } | ||||
							
								
								
									
										4
									
								
								uni_modules/uni-fav/components/uni-fav/i18n/zh-Hans.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								uni_modules/uni-fav/components/uni-fav/i18n/zh-Hans.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| { | ||||
| 	"uni-fav.collect": "收藏", | ||||
| 	"uni-fav.collected": "已收藏" | ||||
| } | ||||
							
								
								
									
										4
									
								
								uni_modules/uni-fav/components/uni-fav/i18n/zh-Hant.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								uni_modules/uni-fav/components/uni-fav/i18n/zh-Hant.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| { | ||||
| 	"uni-fav.collect": "收藏", | ||||
| 	"uni-fav.collected": "已收藏" | ||||
| } | ||||
							
								
								
									
										161
									
								
								uni_modules/uni-fav/components/uni-fav/uni-fav.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										161
									
								
								uni_modules/uni-fav/components/uni-fav/uni-fav.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -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> | ||||
							
								
								
									
										89
									
								
								uni_modules/uni-fav/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								uni_modules/uni-fav/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -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" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										10
									
								
								uni_modules/uni-fav/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								uni_modules/uni-fav/readme.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
|  | ||||
|  | ||||
| ## Fav 收藏按钮 | ||||
| > **组件名:uni-fav** | ||||
| > 代码块: `uFav` | ||||
|  | ||||
| 用于收藏功能,可点击切换选中、不选中的状态。 | ||||
|  | ||||
| ### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-fav) | ||||
| #### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839  | ||||
							
								
								
									
										67
									
								
								uni_modules/uni-file-picker/changelog.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								uni_modules/uni-file-picker/changelog.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,67 @@ | ||||
| ## 1.0.4(2023-03-29) | ||||
| - 修复 手动上传删除一个文件后不能再上传的bug | ||||
| ## 1.0.3(2022-12-19) | ||||
| - 新增 sourceType 属性, 可以自定义图片和视频选择的来源 | ||||
| ## 1.0.2(2022-07-04) | ||||
| - 修复 在uni-forms下样式不生效的bug | ||||
| ## 1.0.1(2021-11-23) | ||||
| - 修复 参数为对象的情况下,url在某些情况显示错误的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-file-picker](https://uniapp.dcloud.io/component/uniui/uni-file-picker) | ||||
| ## 0.2.16(2021-11-08) | ||||
| - 修复 传入空对象 ,显示错误的Bug | ||||
| ## 0.2.15(2021-08-30) | ||||
| - 修复 return-type="object" 时且存在v-model时,无法删除文件的Bug | ||||
| ## 0.2.14(2021-08-23) | ||||
| - 新增 参数中返回 fileID 字段 | ||||
| ## 0.2.13(2021-08-23) | ||||
| - 修复 腾讯云传入fileID 不能回显的bug | ||||
| - 修复 选择图片后,不能放大的问题 | ||||
| ## 0.2.12(2021-08-17) | ||||
| - 修复 由于 0.2.11 版本引起的不能回显图片的Bug | ||||
| ## 0.2.11(2021-08-16) | ||||
| - 新增 clearFiles(index) 方法,可以手动删除指定文件 | ||||
| - 修复 v-model 值设为 null 报错的Bug | ||||
| ## 0.2.10(2021-08-13) | ||||
| - 修复 return-type="object" 时,无法删除文件的Bug | ||||
| ## 0.2.9(2021-08-03) | ||||
| - 修复 auto-upload 属性失效的Bug | ||||
| ## 0.2.8(2021-07-31) | ||||
| - 修复 fileExtname属性不指定值报错的Bug | ||||
| ## 0.2.7(2021-07-31) | ||||
| - 修复 在某种场景下图片不回显的Bug | ||||
| ## 0.2.6(2021-07-30) | ||||
| - 修复 return-type为object下,返回值不正确的Bug | ||||
| ## 0.2.5(2021-07-30) | ||||
| - 修复(重要) H5 平台下如果和uni-forms组件一同使用导致页面卡死的问题 | ||||
| ## 0.2.3(2021-07-28) | ||||
| - 优化 调整示例代码 | ||||
| ## 0.2.2(2021-07-27) | ||||
| - 修复 vue3 下赋值错误的Bug | ||||
| - 优化 h5平台下上传文件导致页面卡死的问题 | ||||
| ## 0.2.0(2021-07-13) | ||||
| - 组件兼容 vue3,如何创建vue3项目,详见 [uni-app 项目支持 vue3 介绍](https://ask.dcloud.net.cn/article/37834) | ||||
| ## 0.1.1(2021-07-02) | ||||
| - 修复 sourceType 缺少默认值导致 ios 无法选择文件 | ||||
| ## 0.1.0(2021-06-30) | ||||
| - 优化 解耦与uniCloud的强绑定关系 ,如不绑定服务空间,默认autoUpload为false且不可更改 | ||||
| ## 0.0.11(2021-06-30) | ||||
| - 修复 由 0.0.10 版本引发的 returnType 属性失效的问题 | ||||
| ## 0.0.10(2021-06-29) | ||||
| - 优化 文件上传后进度条消失时机 | ||||
| ## 0.0.9(2021-06-29) | ||||
| - 修复 在uni-forms 中,删除文件 ,获取的值不对的Bug | ||||
| ## 0.0.8(2021-06-15) | ||||
| - 修复 删除文件时无法触发 v-model 的Bug | ||||
| ## 0.0.7(2021-05-12) | ||||
| - 新增 组件示例地址 | ||||
| ## 0.0.6(2021-04-09) | ||||
| - 修复 选择的文件非 file-extname 字段指定的扩展名报错的Bug | ||||
| ## 0.0.5(2021-04-09) | ||||
| - 优化 更新组件示例 | ||||
| ## 0.0.4(2021-04-09) | ||||
| - 优化 file-extname 字段支持字符串写法,多个扩展名需要用逗号分隔 | ||||
| ## 0.0.3(2021-02-05) | ||||
| - 调整为uni_modules目录规范 | ||||
| - 修复 微信小程序不指定 fileExtname 属性选择失败的Bug | ||||
| @@ -0,0 +1,224 @@ | ||||
| 'use strict'; | ||||
|  | ||||
| const ERR_MSG_OK = 'chooseAndUploadFile:ok'; | ||||
| const ERR_MSG_FAIL = 'chooseAndUploadFile:fail'; | ||||
|  | ||||
| function chooseImage(opts) { | ||||
| 	const { | ||||
| 		count, | ||||
| 		sizeType = ['original', 'compressed'], | ||||
| 		sourceType, | ||||
| 		extension | ||||
| 	} = opts | ||||
| 	return new Promise((resolve, reject) => { | ||||
| 		uni.chooseImage({ | ||||
| 			count, | ||||
| 			sizeType, | ||||
| 			sourceType, | ||||
| 			extension, | ||||
| 			success(res) { | ||||
| 				resolve(normalizeChooseAndUploadFileRes(res, 'image')); | ||||
| 			}, | ||||
| 			fail(res) { | ||||
| 				reject({ | ||||
| 					errMsg: res.errMsg.replace('chooseImage:fail', ERR_MSG_FAIL), | ||||
| 				}); | ||||
| 			}, | ||||
| 		}); | ||||
| 	}); | ||||
| } | ||||
|  | ||||
| function chooseVideo(opts) { | ||||
| 	const { | ||||
| 		camera, | ||||
| 		compressed, | ||||
| 		maxDuration, | ||||
| 		sourceType, | ||||
| 		extension | ||||
| 	} = opts; | ||||
| 	return new Promise((resolve, reject) => { | ||||
| 		uni.chooseVideo({ | ||||
| 			camera, | ||||
| 			compressed, | ||||
| 			maxDuration, | ||||
| 			sourceType, | ||||
| 			extension, | ||||
| 			success(res) { | ||||
| 				const { | ||||
| 					tempFilePath, | ||||
| 					duration, | ||||
| 					size, | ||||
| 					height, | ||||
| 					width | ||||
| 				} = res; | ||||
| 				resolve(normalizeChooseAndUploadFileRes({ | ||||
| 					errMsg: 'chooseVideo:ok', | ||||
| 					tempFilePaths: [tempFilePath], | ||||
| 					tempFiles: [ | ||||
| 					{ | ||||
| 						name: (res.tempFile && res.tempFile.name) || '', | ||||
| 						path: tempFilePath, | ||||
| 						size, | ||||
| 						type: (res.tempFile && res.tempFile.type) || '', | ||||
| 						width, | ||||
| 						height, | ||||
| 						duration, | ||||
| 						fileType: 'video', | ||||
| 						cloudPath: '', | ||||
| 					}, ], | ||||
| 				}, 'video')); | ||||
| 			}, | ||||
| 			fail(res) { | ||||
| 				reject({ | ||||
| 					errMsg: res.errMsg.replace('chooseVideo:fail', ERR_MSG_FAIL), | ||||
| 				}); | ||||
| 			}, | ||||
| 		}); | ||||
| 	}); | ||||
| } | ||||
|  | ||||
| function chooseAll(opts) { | ||||
| 	const { | ||||
| 		count, | ||||
| 		extension | ||||
| 	} = opts; | ||||
| 	return new Promise((resolve, reject) => { | ||||
| 		let chooseFile = uni.chooseFile; | ||||
| 		if (typeof wx !== 'undefined' && | ||||
| 			typeof wx.chooseMessageFile === 'function') { | ||||
| 			chooseFile = wx.chooseMessageFile; | ||||
| 		} | ||||
| 		if (typeof chooseFile !== 'function') { | ||||
| 			return reject({ | ||||
| 				errMsg: ERR_MSG_FAIL + ' 请指定 type 类型,该平台仅支持选择 image 或 video。', | ||||
| 			}); | ||||
| 		} | ||||
| 		chooseFile({ | ||||
| 			type: 'all', | ||||
| 			count, | ||||
| 			extension, | ||||
| 			success(res) { | ||||
| 				resolve(normalizeChooseAndUploadFileRes(res)); | ||||
| 			}, | ||||
| 			fail(res) { | ||||
| 				reject({ | ||||
| 					errMsg: res.errMsg.replace('chooseFile:fail', ERR_MSG_FAIL), | ||||
| 				}); | ||||
| 			}, | ||||
| 		}); | ||||
| 	}); | ||||
| } | ||||
|  | ||||
| function normalizeChooseAndUploadFileRes(res, fileType) { | ||||
| 	res.tempFiles.forEach((item, index) => { | ||||
| 		if (!item.name) { | ||||
| 			item.name = item.path.substring(item.path.lastIndexOf('/') + 1); | ||||
| 		} | ||||
| 		if (fileType) { | ||||
| 			item.fileType = fileType; | ||||
| 		} | ||||
| 		item.cloudPath = | ||||
| 			Date.now() + '_' + index + item.name.substring(item.name.lastIndexOf('.')); | ||||
| 	}); | ||||
| 	if (!res.tempFilePaths) { | ||||
| 		res.tempFilePaths = res.tempFiles.map((file) => file.path); | ||||
| 	} | ||||
| 	return res; | ||||
| } | ||||
|  | ||||
| function uploadCloudFiles(files, max = 5, onUploadProgress) { | ||||
| 	files = JSON.parse(JSON.stringify(files)) | ||||
| 	const len = files.length | ||||
| 	let count = 0 | ||||
| 	let self = this | ||||
| 	return new Promise(resolve => { | ||||
| 		while (count < max) { | ||||
| 			next() | ||||
| 		} | ||||
|  | ||||
| 		function next() { | ||||
| 			let cur = count++ | ||||
| 			if (cur >= len) { | ||||
| 				!files.find(item => !item.url && !item.errMsg) && resolve(files) | ||||
| 				return | ||||
| 			} | ||||
| 			const fileItem = files[cur] | ||||
| 			const index = self.files.findIndex(v => v.uuid === fileItem.uuid) | ||||
| 			fileItem.url = '' | ||||
| 			delete fileItem.errMsg | ||||
|  | ||||
| 			uniCloud | ||||
| 				.uploadFile({ | ||||
| 					filePath: fileItem.path, | ||||
| 					cloudPath: fileItem.cloudPath, | ||||
| 					fileType: fileItem.fileType, | ||||
| 					onUploadProgress: res => { | ||||
| 						res.index = index | ||||
| 						onUploadProgress && onUploadProgress(res) | ||||
| 					} | ||||
| 				}) | ||||
| 				.then(res => { | ||||
| 					fileItem.url = res.fileID | ||||
| 					fileItem.index = index | ||||
| 					if (cur < len) { | ||||
| 						next() | ||||
| 					} | ||||
| 				}) | ||||
| 				.catch(res => { | ||||
| 					fileItem.errMsg = res.errMsg || res.message | ||||
| 					fileItem.index = index | ||||
| 					if (cur < len) { | ||||
| 						next() | ||||
| 					} | ||||
| 				}) | ||||
| 		} | ||||
| 	}) | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| function uploadFiles(choosePromise, { | ||||
| 	onChooseFile, | ||||
| 	onUploadProgress | ||||
| }) { | ||||
| 	return choosePromise | ||||
| 		.then((res) => { | ||||
| 			if (onChooseFile) { | ||||
| 				const customChooseRes = onChooseFile(res); | ||||
| 				if (typeof customChooseRes !== 'undefined') { | ||||
| 					return Promise.resolve(customChooseRes).then((chooseRes) => typeof chooseRes === 'undefined' ? | ||||
| 						res : chooseRes); | ||||
| 				} | ||||
| 			} | ||||
| 			return res; | ||||
| 		}) | ||||
| 		.then((res) => { | ||||
| 			if (res === false) { | ||||
| 				return { | ||||
| 					errMsg: ERR_MSG_OK, | ||||
| 					tempFilePaths: [], | ||||
| 					tempFiles: [], | ||||
| 				}; | ||||
| 			} | ||||
| 			return res | ||||
| 		}) | ||||
| } | ||||
|  | ||||
| function chooseAndUploadFile(opts = { | ||||
| 	type: 'all' | ||||
| }) { | ||||
| 	if (opts.type === 'image') { | ||||
| 		return uploadFiles(chooseImage(opts), opts); | ||||
| 	} | ||||
| 	else if (opts.type === 'video') { | ||||
| 		return uploadFiles(chooseVideo(opts), opts); | ||||
| 	} | ||||
| 	return uploadFiles(chooseAll(opts), opts); | ||||
| } | ||||
|  | ||||
| export { | ||||
| 	chooseAndUploadFile, | ||||
| 	uploadCloudFiles | ||||
| }; | ||||
| @@ -0,0 +1,667 @@ | ||||
| <template> | ||||
| 	<view class="uni-file-picker"> | ||||
| 		<view v-if="title" class="uni-file-picker__header"> | ||||
| 			<text class="file-title">{{ title }}</text> | ||||
| 			<text class="file-count">{{ filesList.length }}/{{ limitLength }}</text> | ||||
| 		</view> | ||||
| 		<upload-image v-if="fileMediatype === 'image' && showType === 'grid'" :readonly="readonly" | ||||
| 			:image-styles="imageStyles" :files-list="filesList" :limit="limitLength" :disablePreview="disablePreview" | ||||
| 			:delIcon="delIcon" @uploadFiles="uploadFiles" @choose="choose" @delFile="delFile"> | ||||
| 			<slot> | ||||
| 				<view class="is-add"> | ||||
| 					<view class="icon-add"></view> | ||||
| 					<view class="icon-add rotate"></view> | ||||
| 				</view> | ||||
| 			</slot> | ||||
| 		</upload-image> | ||||
| 		<upload-file v-if="fileMediatype !== 'image' || showType !== 'grid'" :readonly="readonly" | ||||
| 			:list-styles="listStyles" :files-list="filesList" :showType="showType" :delIcon="delIcon" | ||||
| 			@uploadFiles="uploadFiles" @choose="choose" @delFile="delFile"> | ||||
| 			<slot><button type="primary" size="mini">选择文件</button></slot> | ||||
| 		</upload-file> | ||||
| 	</view> | ||||
| </template> | ||||
|  | ||||
| <script> | ||||
| 	import { | ||||
| 		chooseAndUploadFile, | ||||
| 		uploadCloudFiles | ||||
| 	} from './choose-and-upload-file.js' | ||||
| 	import { | ||||
| 		get_file_ext, | ||||
| 		get_extname, | ||||
| 		get_files_and_is_max, | ||||
| 		get_file_info, | ||||
| 		get_file_data | ||||
| 	} from './utils.js' | ||||
| 	import uploadImage from './upload-image.vue' | ||||
| 	import uploadFile from './upload-file.vue' | ||||
| 	let fileInput = null | ||||
| 	/** | ||||
| 	 * FilePicker 文件选择上传 | ||||
| 	 * @description 文件选择上传组件,可以选择图片、视频等任意文件并上传到当前绑定的服务空间 | ||||
| 	 * @tutorial https://ext.dcloud.net.cn/plugin?id=4079 | ||||
| 	 * @property {Object|Array}	value	组件数据,通常用来回显 ,类型由return-type属性决定 | ||||
| 	 * @property {Boolean}	disabled = [true|false]	组件禁用 | ||||
| 	 * 	@value true 	禁用 | ||||
| 	 * 	@value false 	取消禁用 | ||||
| 	 * @property {Boolean}	readonly = [true|false]	组件只读,不可选择,不显示进度,不显示删除按钮 | ||||
| 	 * 	@value true 	只读 | ||||
| 	 * 	@value false 	取消只读 | ||||
| 	 * @property {String}	return-type = [array|object]	限制 value 格式,当为 object 时 ,组件只能单选,且会覆盖 | ||||
| 	 * 	@value array	规定 value 属性的类型为数组 | ||||
| 	 * 	@value object	规定 value 属性的类型为对象 | ||||
| 	 * @property {Boolean}	disable-preview = [true|false]	禁用图片预览,仅 mode:grid 时生效 | ||||
| 	 * 	@value true 	禁用图片预览 | ||||
| 	 * 	@value false 	取消禁用图片预览 | ||||
| 	 * @property {Boolean}	del-icon = [true|false]	是否显示删除按钮 | ||||
| 	 * 	@value true 	显示删除按钮 | ||||
| 	 * 	@value false 	不显示删除按钮 | ||||
| 	 * @property {Boolean}	auto-upload = [true|false]	是否自动上传,值为true则只触发@select,可自行上传 | ||||
| 	 * 	@value true 	自动上传 | ||||
| 	 * 	@value false 	取消自动上传 | ||||
| 	 * @property {Number|String}	limit	最大选择个数 ,h5 会自动忽略多选的部分 | ||||
| 	 * @property {String}	title	组件标题,右侧显示上传计数 | ||||
| 	 * @property {String}	mode = [list|grid]	选择文件后的文件列表样式 | ||||
| 	 * 	@value list 	列表显示 | ||||
| 	 * 	@value grid 	宫格显示 | ||||
| 	 * @property {String}	file-mediatype = [image|video|all]	选择文件类型 | ||||
| 	 * 	@value image	只选择图片 | ||||
| 	 * 	@value video	只选择视频 | ||||
| 	 * 	@value all		选择所有文件 | ||||
| 	 * @property {Array}	file-extname	选择文件后缀,根据 file-mediatype 属性而不同 | ||||
| 	 * @property {Object}	list-style	mode:list 时的样式 | ||||
| 	 * @property {Object}	image-styles	选择文件后缀,根据 file-mediatype 属性而不同 | ||||
| 	 * @event {Function} select 	选择文件后触发 | ||||
| 	 * @event {Function} progress 文件上传时触发 | ||||
| 	 * @event {Function} success 	上传成功触发 | ||||
| 	 * @event {Function} fail 		上传失败触发 | ||||
| 	 * @event {Function} delete 	文件从列表移除时触发 | ||||
| 	 */ | ||||
| 	export default { | ||||
| 		name: 'uniFilePicker', | ||||
| 		components: { | ||||
| 			uploadImage, | ||||
| 			uploadFile | ||||
| 		}, | ||||
| 		options: { | ||||
| 			virtualHost: true | ||||
| 		}, | ||||
| 		emits: ['select', 'success', 'fail', 'progress', 'delete', 'update:modelValue', 'input'], | ||||
| 		props: { | ||||
| 			// #ifdef VUE3 | ||||
| 			modelValue: { | ||||
| 				type: [Array, Object], | ||||
| 				default () { | ||||
| 					return [] | ||||
| 				} | ||||
| 			}, | ||||
| 			// #endif | ||||
|  | ||||
| 			// #ifndef VUE3 | ||||
| 			value: { | ||||
| 				type: [Array, Object], | ||||
| 				default () { | ||||
| 					return [] | ||||
| 				} | ||||
| 			}, | ||||
| 			// #endif | ||||
|  | ||||
| 			disabled: { | ||||
| 				type: Boolean, | ||||
| 				default: false | ||||
| 			}, | ||||
| 			disablePreview: { | ||||
| 				type: Boolean, | ||||
| 				default: false | ||||
| 			}, | ||||
| 			delIcon: { | ||||
| 				type: Boolean, | ||||
| 				default: true | ||||
| 			}, | ||||
| 			// 自动上传 | ||||
| 			autoUpload: { | ||||
| 				type: Boolean, | ||||
| 				default: true | ||||
| 			}, | ||||
| 			// 最大选择个数 ,h5只能限制单选或是多选 | ||||
| 			limit: { | ||||
| 				type: [Number, String], | ||||
| 				default: 9 | ||||
| 			}, | ||||
| 			// 列表样式 grid | list | list-card | ||||
| 			mode: { | ||||
| 				type: String, | ||||
| 				default: 'grid' | ||||
| 			}, | ||||
| 			// 选择文件类型  image/video/all | ||||
| 			fileMediatype: { | ||||
| 				type: String, | ||||
| 				default: 'image' | ||||
| 			}, | ||||
| 			// 文件类型筛选 | ||||
| 			fileExtname: { | ||||
| 				type: [Array, String], | ||||
| 				default () { | ||||
| 					return [] | ||||
| 				} | ||||
| 			}, | ||||
| 			title: { | ||||
| 				type: String, | ||||
| 				default: '' | ||||
| 			}, | ||||
| 			listStyles: { | ||||
| 				type: Object, | ||||
| 				default () { | ||||
| 					return { | ||||
| 						// 是否显示边框 | ||||
| 						border: true, | ||||
| 						// 是否显示分隔线 | ||||
| 						dividline: true, | ||||
| 						// 线条样式 | ||||
| 						borderStyle: {} | ||||
| 					} | ||||
| 				} | ||||
| 			}, | ||||
| 			imageStyles: { | ||||
| 				type: Object, | ||||
| 				default () { | ||||
| 					return { | ||||
| 						width: 'auto', | ||||
| 						height: 'auto' | ||||
| 					} | ||||
| 				} | ||||
| 			}, | ||||
| 			readonly: { | ||||
| 				type: Boolean, | ||||
| 				default: false | ||||
| 			}, | ||||
| 			returnType: { | ||||
| 				type: String, | ||||
| 				default: 'array' | ||||
| 			}, | ||||
| 			sizeType: { | ||||
| 				type: Array, | ||||
| 				default () { | ||||
| 					return ['original', 'compressed'] | ||||
| 				} | ||||
| 			}, | ||||
| 			sourceType: { | ||||
| 				type: Array, | ||||
| 				default () { | ||||
| 					return  ['album', 'camera'] | ||||
| 				} | ||||
| 			} | ||||
| 		}, | ||||
| 		data() { | ||||
| 			return { | ||||
| 				files: [], | ||||
| 				localValue: [] | ||||
| 			} | ||||
| 		}, | ||||
| 		watch: { | ||||
| 			// #ifndef VUE3 | ||||
| 			value: { | ||||
| 				handler(newVal, oldVal) { | ||||
| 					this.setValue(newVal, oldVal) | ||||
| 				}, | ||||
| 				immediate: true | ||||
| 			}, | ||||
| 			// #endif | ||||
| 			// #ifdef VUE3 | ||||
| 			modelValue: { | ||||
| 				handler(newVal, oldVal) { | ||||
| 					this.setValue(newVal, oldVal) | ||||
| 				}, | ||||
| 				immediate: true | ||||
| 			}, | ||||
| 			// #endif | ||||
| 		}, | ||||
| 		computed: { | ||||
| 			filesList() { | ||||
| 				let files = [] | ||||
| 				this.files.forEach(v => { | ||||
| 					files.push(v) | ||||
| 				}) | ||||
| 				return files | ||||
| 			}, | ||||
| 			showType() { | ||||
| 				if (this.fileMediatype === 'image') { | ||||
| 					return this.mode | ||||
| 				} | ||||
| 				return 'list' | ||||
| 			}, | ||||
| 			limitLength() { | ||||
| 				if (this.returnType === 'object') { | ||||
| 					return 1 | ||||
| 				} | ||||
| 				if (!this.limit) { | ||||
| 					return 1 | ||||
| 				} | ||||
| 				if (this.limit >= 9) { | ||||
| 					return 9 | ||||
| 				} | ||||
| 				return this.limit | ||||
| 			} | ||||
| 		}, | ||||
| 		created() { | ||||
| 			// TODO 兼容不开通服务空间的情况 | ||||
| 			if (!(uniCloud.config && uniCloud.config.provider)) { | ||||
| 				this.noSpace = true | ||||
| 				uniCloud.chooseAndUploadFile = chooseAndUploadFile | ||||
| 			} | ||||
| 			this.form = this.getForm('uniForms') | ||||
| 			this.formItem = this.getForm('uniFormsItem') | ||||
| 			if (this.form && this.formItem) { | ||||
| 				if (this.formItem.name) { | ||||
| 					this.rename = this.formItem.name | ||||
| 					this.form.inputChildrens.push(this) | ||||
| 				} | ||||
| 			} | ||||
| 		}, | ||||
| 		methods: { | ||||
| 			/** | ||||
| 			 * 公开用户使用,清空文件 | ||||
| 			 * @param {Object} index | ||||
| 			 */ | ||||
| 			clearFiles(index) { | ||||
| 				if (index !== 0 && !index) { | ||||
| 					this.files = [] | ||||
| 					this.$nextTick(() => { | ||||
| 						this.setEmit() | ||||
| 					}) | ||||
| 				} else { | ||||
| 					this.files.splice(index, 1) | ||||
| 				} | ||||
| 				this.$nextTick(() => { | ||||
| 					this.setEmit() | ||||
| 				}) | ||||
| 			}, | ||||
| 			/** | ||||
| 			 * 公开用户使用,继续上传 | ||||
| 			 */ | ||||
| 			upload() { | ||||
| 				let files = [] | ||||
| 				this.files.forEach((v, index) => { | ||||
| 					if (v.status === 'ready' || v.status === 'error') { | ||||
| 						files.push(Object.assign({}, v)) | ||||
| 					} | ||||
| 				}) | ||||
| 				return this.uploadFiles(files) | ||||
| 			}, | ||||
| 			async setValue(newVal, oldVal) { | ||||
| 				const newData =  async (v) => { | ||||
| 					const reg = /cloud:\/\/([\w.]+\/?)\S*/ | ||||
| 					let url = '' | ||||
| 					if(v.fileID){ | ||||
| 						url = v.fileID | ||||
| 					}else{ | ||||
| 						url = v.url | ||||
| 					} | ||||
| 					if (reg.test(url)) { | ||||
| 						v.fileID = url | ||||
| 						v.url = await this.getTempFileURL(url) | ||||
| 					} | ||||
| 					if(v.url) v.path = v.url | ||||
| 					return v | ||||
| 				} | ||||
| 				if (this.returnType === 'object') { | ||||
| 					if (newVal) { | ||||
| 						await newData(newVal) | ||||
| 					} else { | ||||
| 						newVal = {} | ||||
| 					} | ||||
| 				} else { | ||||
| 					if (!newVal) newVal = [] | ||||
| 					for(let i =0 ;i < newVal.length ;i++){ | ||||
| 						let v = newVal[i] | ||||
| 						await newData(v) | ||||
| 					} | ||||
| 				} | ||||
| 				this.localValue = newVal | ||||
| 				if (this.form && this.formItem &&!this.is_reset) { | ||||
| 					this.is_reset = false | ||||
| 					this.formItem.setValue(this.localValue) | ||||
| 				} | ||||
| 				let filesData = Object.keys(newVal).length > 0 ? newVal : []; | ||||
| 				this.files = [].concat(filesData) | ||||
| 			}, | ||||
|  | ||||
| 			/** | ||||
| 			 * 选择文件 | ||||
| 			 */ | ||||
| 			choose() { | ||||
|  | ||||
| 				if (this.disabled) return | ||||
| 				if (this.files.length >= Number(this.limitLength) && this.showType !== 'grid' && this.returnType === | ||||
| 					'array') { | ||||
| 					uni.showToast({ | ||||
| 						title: `您最多选择 ${this.limitLength} 个文件`, | ||||
| 						icon: 'none' | ||||
| 					}) | ||||
| 					return | ||||
| 				} | ||||
| 				this.chooseFiles() | ||||
| 			}, | ||||
|  | ||||
| 			/** | ||||
| 			 * 选择文件并上传 | ||||
| 			 */ | ||||
| 			chooseFiles() { | ||||
| 				const _extname = get_extname(this.fileExtname) | ||||
| 				// 获取后缀 | ||||
| 				uniCloud | ||||
| 					.chooseAndUploadFile({ | ||||
| 						type: this.fileMediatype, | ||||
| 						compressed: false, | ||||
| 						sizeType: this.sizeType, | ||||
| 						sourceType: this.sourceType, | ||||
| 						// TODO 如果为空,video 有问题 | ||||
| 						extension: _extname.length > 0 ? _extname : undefined, | ||||
| 						count: this.limitLength - this.files.length, //默认9 | ||||
| 						onChooseFile: this.chooseFileCallback, | ||||
| 						onUploadProgress: progressEvent => { | ||||
| 							this.setProgress(progressEvent, progressEvent.index) | ||||
| 						} | ||||
| 					}) | ||||
| 					.then(result => { | ||||
| 						this.setSuccessAndError(result.tempFiles) | ||||
| 					}) | ||||
| 					.catch(err => { | ||||
| 						console.log('选择失败', err) | ||||
| 					}) | ||||
| 			}, | ||||
|  | ||||
| 			/** | ||||
| 			 * 选择文件回调 | ||||
| 			 * @param {Object} res | ||||
| 			 */ | ||||
| 			async chooseFileCallback(res) { | ||||
| 				const _extname = get_extname(this.fileExtname) | ||||
| 				const is_one = (Number(this.limitLength) === 1 && | ||||
| 						this.disablePreview && | ||||
| 						!this.disabled) || | ||||
| 					this.returnType === 'object' | ||||
| 				// 如果这有一个文件 ,需要清空本地缓存数据 | ||||
| 				if (is_one) { | ||||
| 					this.files = [] | ||||
| 				} | ||||
|  | ||||
| 				let { | ||||
| 					filePaths, | ||||
| 					files | ||||
| 				} = get_files_and_is_max(res, _extname) | ||||
| 				if (!(_extname && _extname.length > 0)) { | ||||
| 					filePaths = res.tempFilePaths | ||||
| 					files = res.tempFiles | ||||
| 				} | ||||
|  | ||||
| 				let currentData = [] | ||||
| 				for (let i = 0; i < files.length; i++) { | ||||
| 					if (this.limitLength - this.files.length <= 0) break | ||||
| 					files[i].uuid = Date.now() | ||||
| 					let filedata = await get_file_data(files[i], this.fileMediatype) | ||||
| 					filedata.progress = 0 | ||||
| 					filedata.status = 'ready' | ||||
| 					this.files.push(filedata) | ||||
| 					currentData.push({ | ||||
| 						...filedata, | ||||
| 						file: files[i] | ||||
| 					}) | ||||
| 				} | ||||
| 				this.$emit('select', { | ||||
| 					tempFiles: currentData, | ||||
| 					tempFilePaths: filePaths | ||||
| 				}) | ||||
| 				res.tempFiles = files | ||||
| 				// 停止自动上传 | ||||
| 				if (!this.autoUpload || this.noSpace) { | ||||
| 					res.tempFiles = [] | ||||
| 				} | ||||
| 			}, | ||||
|  | ||||
| 			/** | ||||
| 			 * 批传 | ||||
| 			 * @param {Object} e | ||||
| 			 */ | ||||
| 			uploadFiles(files) { | ||||
| 				files = [].concat(files) | ||||
| 				return uploadCloudFiles.call(this, files, 5, res => { | ||||
| 						this.setProgress(res, res.index, true) | ||||
| 					}) | ||||
| 					.then(result => { | ||||
| 						this.setSuccessAndError(result) | ||||
| 						return result; | ||||
| 					}) | ||||
| 					.catch(err => { | ||||
| 						console.log(err) | ||||
| 					}) | ||||
| 			}, | ||||
|  | ||||
| 			/** | ||||
| 			 * 成功或失败 | ||||
| 			 */ | ||||
| 			async setSuccessAndError(res, fn) { | ||||
| 				let successData = [] | ||||
| 				let errorData = [] | ||||
| 				let tempFilePath = [] | ||||
| 				let errorTempFilePath = [] | ||||
| 				for (let i = 0; i < res.length; i++) { | ||||
| 					const item = res[i] | ||||
| 					const index = item.uuid ? this.files.findIndex(p => p.uuid === item.uuid) : item.index | ||||
|  | ||||
| 					if (index === -1 || !this.files) break | ||||
| 					if (item.errMsg === 'request:fail') { | ||||
| 						this.files[index].url = item.path | ||||
| 						this.files[index].status = 'error' | ||||
| 						this.files[index].errMsg = item.errMsg | ||||
| 						// this.files[index].progress = -1 | ||||
| 						errorData.push(this.files[index]) | ||||
| 						errorTempFilePath.push(this.files[index].url) | ||||
| 					} else { | ||||
| 						this.files[index].errMsg = '' | ||||
| 						this.files[index].fileID = item.url | ||||
| 						const reg = /cloud:\/\/([\w.]+\/?)\S*/ | ||||
| 						if (reg.test(item.url)) { | ||||
| 							this.files[index].url = await this.getTempFileURL(item.url) | ||||
| 						}else{ | ||||
| 							this.files[index].url = item.url | ||||
| 						} | ||||
|  | ||||
| 						this.files[index].status = 'success' | ||||
| 						this.files[index].progress += 1 | ||||
| 						successData.push(this.files[index]) | ||||
| 						tempFilePath.push(this.files[index].fileID) | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				if (successData.length > 0) { | ||||
| 					this.setEmit() | ||||
| 					// 状态改变返回 | ||||
| 					this.$emit('success', { | ||||
| 						tempFiles: this.backObject(successData), | ||||
| 						tempFilePaths: tempFilePath | ||||
| 					}) | ||||
| 				} | ||||
|  | ||||
| 				if (errorData.length > 0) { | ||||
| 					this.$emit('fail', { | ||||
| 						tempFiles: this.backObject(errorData), | ||||
| 						tempFilePaths: errorTempFilePath | ||||
| 					}) | ||||
| 				} | ||||
| 			}, | ||||
|  | ||||
| 			/** | ||||
| 			 * 获取进度 | ||||
| 			 * @param {Object} progressEvent | ||||
| 			 * @param {Object} index | ||||
| 			 * @param {Object} type | ||||
| 			 */ | ||||
| 			setProgress(progressEvent, index, type) { | ||||
| 				const fileLenth = this.files.length | ||||
| 				const percentNum = (index / fileLenth) * 100 | ||||
| 				const percentCompleted = Math.round((progressEvent.loaded * 100) / progressEvent.total) | ||||
| 				let idx = index | ||||
| 				if (!type) { | ||||
| 					idx = this.files.findIndex(p => p.uuid === progressEvent.tempFile.uuid) | ||||
| 				} | ||||
| 				if (idx === -1 || !this.files[idx]) return | ||||
| 				// fix by mehaotian 100 就会消失,-1 是为了让进度条消失 | ||||
| 				this.files[idx].progress = percentCompleted - 1 | ||||
| 				// 上传中 | ||||
| 				this.$emit('progress', { | ||||
| 					index: idx, | ||||
| 					progress: parseInt(percentCompleted), | ||||
| 					tempFile: this.files[idx] | ||||
| 				}) | ||||
| 			}, | ||||
|  | ||||
| 			/** | ||||
| 			 * 删除文件 | ||||
| 			 * @param {Object} index | ||||
| 			 */ | ||||
| 			delFile(index) { | ||||
| 				this.$emit('delete', { | ||||
| 					tempFile: this.files[index], | ||||
| 					tempFilePath: this.files[index].url | ||||
| 				}) | ||||
| 				this.files.splice(index, 1) | ||||
| 				this.$nextTick(() => { | ||||
| 					this.setEmit() | ||||
| 				}) | ||||
| 			}, | ||||
|  | ||||
| 			/** | ||||
| 			 * 获取文件名和后缀 | ||||
| 			 * @param {Object} name | ||||
| 			 */ | ||||
| 			getFileExt(name) { | ||||
| 				const last_len = name.lastIndexOf('.') | ||||
| 				const len = name.length | ||||
| 				return { | ||||
| 					name: name.substring(0, last_len), | ||||
| 					ext: name.substring(last_len + 1, len) | ||||
| 				} | ||||
| 			}, | ||||
|  | ||||
| 			/** | ||||
| 			 * 处理返回事件 | ||||
| 			 */ | ||||
| 			setEmit() { | ||||
| 				let data = [] | ||||
| 				if (this.returnType === 'object') { | ||||
| 					data = this.backObject(this.files)[0] | ||||
| 					this.localValue = data?data:null | ||||
| 				} else { | ||||
| 					data = this.backObject(this.files) | ||||
| 					if (!this.localValue) { | ||||
| 						this.localValue = [] | ||||
| 					} | ||||
| 					this.localValue = [...data] | ||||
| 				} | ||||
| 				// #ifdef VUE3 | ||||
| 				this.$emit('update:modelValue', this.localValue) | ||||
| 				// #endif | ||||
| 				// #ifndef VUE3 | ||||
| 				this.$emit('input', this.localValue) | ||||
| 				// #endif | ||||
| 			}, | ||||
|  | ||||
| 			/** | ||||
| 			 * 处理返回参数 | ||||
| 			 * @param {Object} files | ||||
| 			 */ | ||||
| 			backObject(files) { | ||||
| 				let newFilesData = [] | ||||
| 				files.forEach(v => { | ||||
| 					newFilesData.push({ | ||||
| 						extname: v.extname, | ||||
| 						fileType: v.fileType, | ||||
| 						image: v.image, | ||||
| 						name: v.name, | ||||
| 						path: v.path, | ||||
| 						size: v.size, | ||||
| 						fileID:v.fileID, | ||||
| 						url: v.url, | ||||
| 						// 修改删除一个文件后不能再上传的bug, #694 | ||||
|             uuid: v.uuid, | ||||
|             status: v.status, | ||||
|             cloudPath: v.cloudPath | ||||
| 					}) | ||||
| 				}) | ||||
| 				return newFilesData | ||||
| 			}, | ||||
| 			async getTempFileURL(fileList) { | ||||
| 				fileList = { | ||||
| 					fileList: [].concat(fileList) | ||||
| 				} | ||||
| 				const urls = await uniCloud.getTempFileURL(fileList) | ||||
| 				return urls.fileList[0].tempFileURL || '' | ||||
| 			}, | ||||
| 			/** | ||||
| 			 * 获取父元素实例 | ||||
| 			 */ | ||||
| 			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; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| </script> | ||||
|  | ||||
| <style> | ||||
| 	.uni-file-picker { | ||||
| 		/* #ifndef APP-NVUE */ | ||||
| 		box-sizing: border-box; | ||||
| 		overflow: hidden; | ||||
| 		width: 100%; | ||||
| 		/* #endif */ | ||||
| 		flex: 1; | ||||
| 	} | ||||
|  | ||||
| 	.uni-file-picker__header { | ||||
| 		padding-top: 5px; | ||||
| 		padding-bottom: 10px; | ||||
| 		/* #ifndef APP-NVUE */ | ||||
| 		display: flex; | ||||
| 		/* #endif */ | ||||
| 		justify-content: space-between; | ||||
| 	} | ||||
|  | ||||
| 	.file-title { | ||||
| 		font-size: 14px; | ||||
| 		color: #333; | ||||
| 	} | ||||
|  | ||||
| 	.file-count { | ||||
| 		font-size: 14px; | ||||
| 		color: #999; | ||||
| 	} | ||||
|  | ||||
| 	.is-add { | ||||
| 		/* #ifndef APP-NVUE */ | ||||
| 		display: flex; | ||||
| 		/* #endif */ | ||||
| 		align-items: center; | ||||
| 		justify-content: center; | ||||
| 	} | ||||
|  | ||||
| 	.icon-add { | ||||
| 		width: 50px; | ||||
| 		height: 5px; | ||||
| 		background-color: #f1f1f1; | ||||
| 		border-radius: 2px; | ||||
| 	} | ||||
|  | ||||
| 	.rotate { | ||||
| 		position: absolute; | ||||
| 		transform: rotate(90deg); | ||||
| 	} | ||||
| </style> | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user
	 ‘tuzi0421
					‘tuzi0421