Browse Source

增加商品表单

cr 4 years ago
parent
commit
ce488db5f7

+ 3 - 0
src/App.vue

@@ -68,4 +68,7 @@
68 68
     page {
69 69
         background-color: #f8f8f8;
70 70
     }
71
+    .page--iphoneX{
72
+        padding-bottom: px(30);
73
+    }
71 74
 </style>

+ 286 - 0
src/components/easy-upload/easy-upload.vue

@@ -0,0 +1,286 @@
1
+<template>
2
+	<view>
3
+		<view class="upload">
4
+			<block v-for="(upload,index) in uploads" :key="index">
5
+				<view class="uplode-file">
6
+					<image v-if="types == 'image'" class="uploade-img" :src="upload" :data-src="upload" @tap="previewImage"></image>
7
+					<image v-if="types == 'image'" class="clear-one-icon" :src="clearIcon" @tap="delImage(index)"></image>
8
+					<video v-if="types == 'video'" class="uploade-img" :src="upload" controls>
9
+						<cover-image v-if="types == 'video'" class="clear-one-icon" :src="clearIcon" @tap="delImage(index)"></cover-image>
10
+					</video>
11
+				</view>
12
+			</block>
13
+			<view v-if="uploads.length < uploadCount" :class="uploadIcon ? 'uploader-icon' : 'uploader-input-box'">
14
+				<view v-if="!uploadIcon" class="uploader-input" @tap="chooseUploads"></view>
15
+				<image v-else class="image-cion" :src="uploadIcon" @tap="chooseUploads"></image>
16
+			</view>
17
+		</view>	
18
+		<!-- <button type="primary" v-if="types == 'image'" @tap="upload">上传</button> -->
19
+	</view>
20
+</template>
21
+
22
+<script>
23
+	export default{
24
+		props: {
25
+			types: {
26
+				type: String,
27
+				default: 'image'
28
+			},
29
+			dataList: {
30
+				type: Array,
31
+				default: function() {
32
+					return []
33
+				}
34
+			},
35
+			clearIcon: {
36
+				type: String,
37
+				default: 'http://img1.imgtn.bdimg.com/it/u=451604666,2295832001&fm=26&gp=0.jpg'
38
+			},
39
+			uploadIcon: {
40
+				type: String,
41
+				default: ''
42
+			},
43
+			uploadUrl: {
44
+				type: String,
45
+				default: ''
46
+			},
47
+			deleteUrl: {
48
+				type: String,
49
+				default: ''
50
+			},
51
+			uploadCount: {
52
+				type: Number,
53
+				default: 1
54
+			},
55
+			//上传图片大小 默认3M
56
+			upload_max: {
57
+				type: Number,
58
+				default: 3
59
+			}
60
+		},
61
+		data(){
62
+			return {
63
+				//上传的图片地址
64
+				uploadImages: [],
65
+				//展示的图片地址
66
+				uploads: [],
67
+				// 超出限制数组
68
+				exceeded_list: [],
69
+			}
70
+		},
71
+		watch:{
72
+			dataList:{
73
+				handler(val){
74
+					this.uploads = val;
75
+				},
76
+				immediate: true
77
+			}
78
+		},
79
+		methods:{
80
+			previewImage (e) {
81
+				var current = e.target.dataset.src
82
+				uni.previewImage({
83
+					current: current,
84
+					urls: this.dataList
85
+				})
86
+			},
87
+			chooseUploads(){
88
+				switch (this.types){
89
+					case 'image': 
90
+						uni.chooseImage({
91
+							count: this.uploadCount - this.uploads.length, //默认9
92
+							sizeType: ['original', 'compressed'], //可以指定是原图还是压缩图,默认二者都有
93
+							sourceType: ['album', 'camera'], //从相册选择
94
+							success: (res) => {
95
+								for(let i = 0; i< res.tempFiles.length; i++){
96
+									if(Math.ceil(res.tempFiles[i].size / 1024) < this.upload_max * 1024){
97
+										this.uploads.push(res.tempFiles[i].path)
98
+										this.uploadImages.push(res.tempFiles[i].path)
99
+									}else {
100
+										this.exceeded_list.push(i === 0 ? 1 : i + 1);
101
+										uni.showModal({
102
+											title: '提示',
103
+											content: `第${[...new Set(this.exceeded_list)].join(',')}张图片超出限制${this.upload_max}MB,已过滤`
104
+										});
105
+									}
106
+								}
107
+								this.upload();
108
+							},
109
+							fail: (err) => {
110
+								uni.showModal({
111
+									content: JSON.stringify(err)
112
+								});
113
+							}
114
+						});
115
+					break;
116
+					case 'video' :
117
+						uni.chooseVideo({
118
+							sourceType: ['camera', 'album'],
119
+							success: (res) => {
120
+								if(Math.ceil(res.size / 1024) < this.upload_max * 1024){
121
+									this.uploads.push(res.tempFilePath)
122
+									uni.uploadFile({
123
+										url: this.uploadUrl, //仅为示例,非真实的接口地址
124
+										filePath: res.tempFilePath,
125
+										name: 'file',
126
+										//请求参数
127
+										formData: {
128
+											'user': 'test'
129
+										},
130
+										success: (uploadFileRes) => {
131
+											this.$emit('successVideo',uploadFileRes)
132
+										}
133
+									});
134
+								}else {
135
+									uni.showModal({
136
+										title: '提示',
137
+										content: `第${[...new Set(this.exceeded_list)].join(',')}张视频超出限制${this.upload_max}MB,已过滤`
138
+									});
139
+								}
140
+							},
141
+							fail: (err) => {
142
+								uni.showModal({
143
+									content: JSON.stringify(err)
144
+								});
145
+							}
146
+						});
147
+					break;
148
+				}
149
+			},
150
+			delImage(index){
151
+				//第一个是判断app或者h5的 第二个是判断小程序的
152
+				if(this.uploads[index].substring(0,4) !== 'http' || this.uploads[index].substring(0,11) == 'http://tmp/'){
153
+					this.uploads.splice(index,1)
154
+					return;
155
+				};
156
+				if(!this.deleteUrl) {
157
+					uni.showModal({
158
+						content: '请填写删除接口'
159
+					});
160
+					return;
161
+				};
162
+				uni.request({
163
+					url: this.deleteUrl,
164
+					method: 'DELETE',
165
+					data: {
166
+						image: this.dataList[index]
167
+					},
168
+					success: res => {
169
+						if(res.data.status == 1) {
170
+							uni.showToast({
171
+								title: '删除成功'
172
+							})
173
+							this.uploads.splice(index,1)
174
+						}
175
+					},
176
+				});
177
+			},
178
+			upload(){
179
+				console.log(this.uploadUrl);
180
+				if(!this.uploadUrl) {
181
+					uni.showModal({
182
+						content: '请填写上传接口'
183
+					});
184
+					return;
185
+				};
186
+				for (let i of this.uploadImages) {
187
+					uni.uploadFile({
188
+						url: this.uploadUrl, //仅为示例,非真实的接口地址
189
+						filePath: i,
190
+						name: 'file',
191
+						//请求参数
192
+						formData: {
193
+							'user': 'test'
194
+						},
195
+						success: (uploadFileRes) => {
196
+							this.$emit('successImage',uploadFileRes)
197
+						}
198
+					});
199
+				}
200
+			}
201
+		}
202
+	}
203
+</script>
204
+
205
+<style scoped>
206
+	.upload {
207
+		display: flex;
208
+		flex-direction: row;
209
+		flex-wrap: wrap;
210
+	}
211
+	.uplode-file {
212
+		margin: 10upx;
213
+		width: 210upx;
214
+		height: 210upx;
215
+		position: relative;
216
+	}
217
+	.uploade-img {
218
+		display: block;
219
+		width: 210upx;
220
+		height: 210upx;
221
+	}
222
+	.clear-one{
223
+		position: absolute;
224
+		top: -10rpx;
225
+		right: 0;
226
+	}
227
+	.clear-one-icon{
228
+		position: absolute;
229
+		width: 20px;
230
+		height: 20px;
231
+		top: 0;
232
+		right: 0;
233
+		z-index: 9;
234
+	}
235
+	.uploader-input-box {
236
+		position: relative;
237
+		margin:10upx;
238
+		width: 208upx;
239
+		height: 208upx;
240
+		border: 2upx solid #D9D9D9;
241
+	}
242
+	.uploader-input-box:before,
243
+	.uploader-input-box:after {
244
+		content: " ";
245
+		position: absolute;
246
+		top: 50%;
247
+		left: 50%;
248
+		-webkit-transform: translate(-50%, -50%);
249
+		transform: translate(-50%, -50%);
250
+		background-color: #D9D9D9;
251
+	}
252
+	.uploader-input-box:before {
253
+		width: 4upx;
254
+		height: 79upx;
255
+	}
256
+	.uploader-input-box:after {
257
+		width: 79upx;
258
+		height: 4upx;
259
+	}
260
+	.uploader-input-box:active {
261
+		border-color: #999999;
262
+	}
263
+	.uploader-input-box:active:before,
264
+	.uploader-input-box:active:after {
265
+		background-color: #999999;
266
+	}
267
+	.uploader-input {
268
+		position: absolute;
269
+		z-index: 1;
270
+		top: 0;
271
+		left: 0;
272
+		width: 100%;
273
+		height: 100%;
274
+		opacity: 0;
275
+	}
276
+	.uploader-icon{
277
+		position: relative;
278
+		margin:10upx;
279
+		width: 208upx;
280
+		height: 208upx;
281
+	}
282
+	.uploader-icon .image-cion{
283
+		width: 100%;
284
+		height: 100%;
285
+	}
286
+</style>

+ 42 - 0
src/components/easy-upload/readme.md

@@ -0,0 +1,42 @@
1
+### easy-upload 组件
2
+
3
+使用方法
4
+```js
5
+	<easy-upload
6
+	:dataList="imageList" uploadUrl="http://localhost:3000/upload" :types="category"
7
+	deleteUrl='http://localhost:3000/upload' :uploadCount="6"
8
+	@successImage="successImage" @successVideo="successvideo"
9
+	></easy-upload>
10
+
11
+	//使用 hbuilderX (easycom) 可以直接使用
12
+	//在data中绑定 
13
+	//imageList 对应上边绑定的imageList category对应上边的category
14
+	data() {
15
+		return {
16
+			imageList: [],
17
+			category: 'image'
18
+		}
19
+	},
20
+	
21
+```
22
+
23
+|  参数   | 类型  | 是否必填 | 参数描述
24
+|  ----  | ----  | ---- | ----
25
+| types  | String | 否 | 上传类型 image/video
26
+| dataList  | Array | 否 | 图片/视频数据展示
27
+| clearIcon  | String | 否 | 删除图标(可以换成自己的图片)
28
+| uploadIcon  | String | 否 | 上传图标(可以换成自己的图片)
29
+| uploadUrl  | String | 否 | 上传的接口
30
+| deleteUrl  | String | 否 | 删除的接口
31
+| uploadCount  | Number | 否 | 上传图片最大个数(默认为一张)
32
+| upload_max  | Number | 否 | 上传大小(默认为3M)
33
+| upload_max  | Number | 否 | 上传大小(默认为3M)
34
+| upload_max  | Number | 否 | 上传大小(默认为3M)
35
+
36
+|  事件  | 是否必填 | 参数描述
37
+|  ---- | ---- | ----
38
+| successImage  | 否 | 上传图片成功事件
39
+| successVideo  |  否 | 上传视频成功回调
40
+
41
+示例项目中有服务端代码 (node.js)
42
+

+ 235 - 14
src/pages/manage/add-good-form.vue

@@ -1,5 +1,5 @@
1 1
 <template>
2
-    <div class="page">
2
+    <div class="page" :class="{'page--iphoneX': iphoneX}">
3 3
         <div class="form-card">
4 4
             <section class="form-item">
5 5
                 <div class="label">
@@ -9,8 +9,13 @@
9 9
                 <div class="box">
10 10
                     <input type="text" />
11 11
                 </div>
12
-                <div class="tool">
13
-                    <span class="unit">元</span>
12
+                <div class="tool code-tool">
13
+                    <span class="generate">生成</span>
14
+                    <my-image
15
+                        class="scan-img"
16
+                        @click="scan"
17
+                        src="/static/icon/scan.png"
18
+                    ></my-image>
14 19
                 </div>
15 20
             </section>
16 21
             <section class="form-item">
@@ -28,7 +33,13 @@
28 33
                     分类:
29 34
                 </div>
30 35
                 <div class="box">
31
-                    <input type="text" />
36
+                    <picker :mode="'multiSelector'" :range="cateList">
37
+                        <input
38
+                            type="text"
39
+                            :disabled="true"
40
+                            placeholder="请选择"
41
+                        />
42
+                    </picker>
32 43
                 </div>
33 44
             </section>
34 45
             <section class="form-item">
@@ -37,7 +48,10 @@
37 48
                     重量:
38 49
                 </div>
39 50
                 <div class="box">
40
-                    <input type="text" />
51
+                    <input type="digit" />
52
+                </div>
53
+                <div class="tool">
54
+                    <span class="">kg</span>
41 55
                 </div>
42 56
             </section>
43 57
             <section class="form-item">
@@ -46,7 +60,10 @@
46 60
                     售价:
47 61
                 </div>
48 62
                 <div class="box">
49
-                    <input type="text" />
63
+                    <input type="digit" />
64
+                </div>
65
+                <div class="tool">
66
+                    <span class="">元</span>
50 67
                 </div>
51 68
             </section>
52 69
             <section class="form-item">
@@ -55,7 +72,10 @@
55 72
                     划线价:
56 73
                 </div>
57 74
                 <div class="box">
58
-                    <input type="text" />
75
+                    <input type="digit" />
76
+                </div>
77
+                <div class="tool">
78
+                    <span class="">元</span>
59 79
                 </div>
60 80
             </section>
61 81
             <section class="form-item">
@@ -64,28 +84,172 @@
64 84
                     库存:
65 85
                 </div>
66 86
                 <div class="box">
67
-                    <input type="text" />
87
+                    <input type="number" />
88
+                </div>
89
+            </section>
90
+        </div>
91
+
92
+        <div class="form-card">
93
+            <div class="head">
94
+                <div class="tit">其他信息</div>
95
+                <div class="switch">
96
+                    <switch :checked="showOther" @change="otherChange"></switch>
97
+                </div>
98
+            </div>
99
+            <block v-if="showOther">
100
+                <section class="form-item">
101
+                    <div class="label">
102
+                        <span class="required">*</span>
103
+                        模板详情:
104
+                    </div>
105
+                    <div class="box">
106
+                        <radio-group @change="tempChange">
107
+                            <label>
108
+                                <radio :value="true" />
109
+                                <text>是</text>
110
+                            </label>
111
+                            <label>
112
+                                <radio :value="false" />
113
+                                <text>否</text>
114
+                            </label>
115
+                        </radio-group>
116
+                    </div>
117
+                </section>
118
+                <section class="form-item">
119
+                    <div class="label">
120
+                        <span class="required">*</span>
121
+                        初始销量:
122
+                    </div>
123
+                    <div class="box">
124
+                        <input type="number" />
125
+                    </div>
126
+                </section>
127
+                <section class="form-item">
128
+                    <div class="label">
129
+                        <span class="required">*</span>
130
+                        出售方式:
131
+                    </div>
132
+                    <div class="box">
133
+                        <picker :mode="'selector'" :range="sellType">
134
+                            <input type="text" :disabled="true"  placeholder="请选择"/>
135
+                        </picker>
136
+                    </div>
137
+                </section>
138
+                <section class="form-item">
139
+                    <div class="label">
140
+                        <span class="required">*</span>
141
+                        限购数:
142
+                    </div>
143
+                    <div class="box">
144
+                        <input type="number" />
145
+                    </div>
146
+                </section>
147
+                <section class="form-item">
148
+                    <div class="label">
149
+                        <span class="required">*</span>
150
+                        供应商:
151
+                    </div>
152
+                    <div class="box">
153
+                        <picker :mode="'selector'" :range="supplierList">
154
+                            <input type="text" :disabled="true" placeholder="请选择"/>
155
+                        </picker>
156
+                    </div>
157
+                </section>
158
+            </block>
159
+        </div>
160
+
161
+        <div class="form-card img-card">
162
+            <section class="form-img-item">
163
+                <div class="tit">封面图</div>
164
+                <div class="list">
165
+                    <easy-upload :types="'image'" :dataList="coverImg" :uploadUrl="'http://localhost:3000/upload'" :uploadCount="1" @successImage="successImage"></easy-upload>
166
+                </div>
167
+            </section>
168
+            <section class="form-img-item">
169
+                <div class="tit">主图</div>
170
+                <div class="list">
171
+                    <easy-upload :types="'image'" :dataList="mainImg" :uploadUrl="'http://localhost:3000/upload'" :uploadCount="100" @successImage="successImage"></easy-upload>
172
+                </div>
173
+            </section>
174
+            <section class="form-img-item">
175
+                <div class="tit">详情图</div>
176
+                <div class="list">
177
+                    <easy-upload :types="'image'" :dataList="detailImg" :uploadUrl="'http://localhost:3000/upload'" :uploadCount="100" @successImage="successImage"></easy-upload>
68 178
                 </div>
69 179
             </section>
70 180
         </div>
181
+
182
+        <button type="button" class="btn">保存</button>
71 183
     </div>
72 184
 </template>
73 185
 
74 186
 <script>
75 187
 import MyImage from "../../components/image/index";
188
+import easyUpload from "../../components/easy-upload/easy-upload";
76 189
 
77 190
 export default {
78 191
     name: "",
79
-    components: { MyImage },
192
+    components: { MyImage,easyUpload },
80 193
 
81 194
     // 数据
82 195
     data() {
83
-        return {};
196
+        return {
197
+            cateList: [], // 分类列表
198
+            sellType:['立即开售','放入仓库'], // 出售方式
199
+            supplierList:['测试供应商'], // 供应商
200
+            showOther: true,
201
+            mainImg:[],
202
+            coverImg:[],
203
+            detailImg:[],
204
+        };
205
+    },
206
+
207
+    async onShow() {
208
+        this.cateList = [
209
+            [1, 2, 3, 4, 5],
210
+            ["一", "二", "三", "四", "五"],
211
+        ];
84 212
     },
85 213
 
86
-    async onShow() {},
87 214
     // 函数
88
-    methods: {},
215
+    methods: {
216
+        // 条码扫描
217
+        scan() {
218
+            let self = this;
219
+            wx.scanCode({
220
+                success(res) {
221
+                    console.log(res);
222
+                },
223
+                fail(res) {
224
+                    console.log("扫码失败:", res);
225
+                },
226
+            });
227
+        },
228
+
229
+        // 生成条码
230
+        generateBarCode() {
231
+            // 后台编码生成方式
232
+            const timestamp = Date.parse(new Date());
233
+        },
234
+
235
+        // 其他面板切换
236
+        otherChange(e) {
237
+            this.showOther = !this.showOther;
238
+        },
239
+
240
+        // 模板详情切换
241
+        tempChange(e) {
242
+            let val = e.target.value;
243
+            if (val === "true") {
244
+            } else {
245
+            }
246
+        },
247
+
248
+        // 上传成功
249
+        successImage(e){
250
+            console.log(e);
251
+        }
252
+    },
89 253
 
90 254
     // 数据计算
91 255
     computed: {},
@@ -99,13 +263,26 @@ export default {
99 263
 .form-card {
100 264
     background-color: #fff;
101 265
     padding: 0 px(30);
266
+    & ~ .form-card {
267
+        margin-top: px(25);
268
+    }
269
+    .head {
270
+        display: flex;
271
+        align-items: center;
272
+        justify-content: space-between;
273
+        height: px(140);
274
+        background-color: #fbfbfb;
275
+        border-bottom: 1px solid #f1f1f1;
276
+        border-top: 1px solid #f1f1f1;
277
+        font-size: px(48);
278
+    }
102 279
 }
103 280
 .form-item {
104 281
     display: flex;
105 282
     font-size: px(44);
106 283
     height: px(140);
107 284
     align-items: center;
108
-    &~.form-item {
285
+    & ~ .form-item {
109 286
         border-top: 1px solid #f1f1f1;
110 287
     }
111 288
     .label {
@@ -125,8 +302,52 @@ export default {
125 302
         color: #999;
126 303
         margin-left: px(10);
127 304
     }
128
-    .required{
305
+    .required {
129 306
         color: #f00;
130 307
     }
131 308
 }
309
+.code-tool {
310
+    display: flex;
311
+    justify-content: flex-start;
312
+    align-items: center;
313
+    .scan-img {
314
+        flex-shrink: 0;
315
+        width: px(80);
316
+        height: px(80);
317
+        /deep/ img {
318
+            width: px(80);
319
+            height: px(80);
320
+        }
321
+    }
322
+    .generate {
323
+        color: #0097d1;
324
+        margin-right: px(30);
325
+        flex-shrink: 0;
326
+    }
327
+}
328
+radio-group{
329
+    label~label{
330
+        margin-left: px(40);
331
+    }
332
+}
333
+.img-card{
334
+    padding-top: px(30);
335
+    padding-bottom: px(30);
336
+}
337
+.form-img-item{
338
+    .tit{
339
+        padding: px(30);
340
+        font-size: px(44);
341
+    }
342
+    .list{
343
+
344
+    }
345
+}
346
+.btn{
347
+    margin-top: px(60);
348
+    background-color: rgb(0, 188, 38);
349
+    color: #fff;
350
+    border-radius: 0;
351
+    border: none;
352
+}
132 353
 </style>

BIN
src/static/icon/scan.png