效果图

多图上传,到自己的腾讯云服务器

uniapp vue3 ref iOS失效 uniapp info.plist_thinkphp

主要步骤

  • 组件uni-file-picker,方便获取图片列表的本地路径
  • 循环使用uni.uploadFile,逐张上传图片到服务器
  • 服务器端tp6使用request()->file('image')和Filesystem::putFile接收图片,返回图片URL

踩过的坑

  • 组件uni-file-picker默认会自动上传图片到项目绑定的云服务器,不符合需求,需要给组件设置
    :auto-upload="false"
    来禁止自动上传
  • 组件uni-file-picker绑定变量v-model=“list”一点用都没有,图片列表需要手动通过组件绑定事件
    @select="select" @delete=“delete”
    来增减算出 真实的图片列表
  • @select="select"事件每次传值,都只是本次多选中的图片,还要手动去合并原有列表
  • 手册....坑多
  • uniapp上传接口uni.uploadFile官方手册里,没有写头部信息,导致PHP无法获取
header: {
	'Content-Type': 'multipart/form-data' 
},
  • thinkphp6文件上传还好

详细步骤

看下面的完整代码吧,心累了

推荐

手机上无广告的百度绿色版首页 baidu.rudon.cn

完整代码

uniapp页面

<template>
	<view style="background-color: white;">
		
		<view>
			<view>
				<view>
					名称:
				</view>
				<view>
					<uni-easyinput  v-model="formData.name" placeholder="输入名称" :inputBorder="false"/>
				</view>
			</view>
			
			<view>
				<view>
					<uni-file-picker file-mediatype="image" :auto-upload="false" limit="9" @select="whenImagesSelected" @delete="whenImagesDeleted"></uni-file-picker>
				</view>
			</view>
			
			
			
		</view>
		
		<view @click="clickBtn">
			<view>
				打卡
			</view>
		</view>
		
		<view>
			{{noticeMsg}}
		</view>
		
		
		
	</view>
</template>

<script>
	export default {
		data() {
			return {
				
				// 表数据
				formData: {
					// 名称
					name: ''
				},
				
				// 提示文字
				noticeMsg: '',
				
				
				// 图片列表
				imgList: [],
				
				// 图片URL 0 => 'https://xxxx/xx.jpg', 1 => '', ...
				imgListUrl: [],
				
				// 打卡记录ID
				record_id: 0
				
			}
		},
		methods: {
			
			
			// 图片选中时
			whenImagesSelected (e) {
				// 【变化+】 e.tempFilePaths,数组,多张图片
				// JS(ES6)合并数组 
				// https://wenku.baidu.com/view/e329dcda0142a8956bec0975f46527d3240ca679.html
				this.imgList = this.imgList.concat( e.tempFilePaths )
				console.log('whenImagesSelected()', this.imgList)
			},
			
			// 图片删除时
			whenImagesDeleted (e) {
				// 【变化-】 e.tempFilePath,字符串,单张图片
				this.imgList.splice( this.imgList.indexOf(e.tempFilePath), 1 )
				console.log('whenImagesDeleted()', this.imgList)
			},
			
			
			
			
			
			// 按钮点击事件
			clickBtn () {
				
				
				// 检查图片是否已经上传完毕
				if (this.record_id !== 0) {
					// 接着上传图片
					this.upload_image()
					return
				}
				
				
				
				// 检查图片数量
				if (this.imgList.length == 0) {
					this.showError( '请上传图片' )
					return
				}
				
				
				// 初始化图片上传后的路径
				this.imgListUrl = []
				for (var k in this.imgList) {
					this.imgListUrl[k] = ''
				}
				
				
				// 上传资料,获取打卡ID
				uni.request({
					url: 'https://xxxxxxxxxxxxxxx/api/add_record/',
					method: 'POST',
					dataType:'JSON',
					data:{
						name: this.formData.name
					},
					success: (res) => {
						console.log('Success', res)
						
						// 成功
						console.log('[记录添加] 成功', res.data.data.id)
						this.record_id = res.data.data.id
						this.showNotice('打卡记录已保存,准备上传图片')
						
						// 逐个上传图片 (根据打卡ID)
						this.upload_image()
						
					},
					fail: (e) => {
						console.log('Fail', e)
						this.showError('系统出错,请稍后再试')
						return
					}
				})
			},
			
			
			
			
			
			/**
			 * 
			 * 准备逐个上传图片
			 * 参考 
			 * 
			 */ 
			upload_image () {
				// 判断 - 全部上传完毕
				if (this.index_next_upload() === false) {
					uni.showToast({
						icon: "success",
						title: "打卡成功",
						success: () => {
							setTimeout(()=>{
								uni.navigateBack({})
							}, 2000)
						}
					})
					return
				}
				
				// 获取下一个需要上传的图片下标
				let nextIndex = this.index_next_upload()
				let numberNature = nextIndex + 1
				let imgPath = this.imgList[ nextIndex ]
				console.log('Uploading img #'+nextIndex)
				console.log('Local Path '+imgPath)
				
				
				
				// 开始上传
				uni.uploadFile({
					// 服务器地址
					url: 'https://xxxxxxxxxxxxxxx/api/upload/',
					
					// PHP服务器需要正确的头部信息
					header: {
						'Content-Type': 'multipart/form-data' 
					},
					
					// 传递的参数
					formData: {
						// 目标储存位置
						path: '/images/'+ numberNature +'.jpg'
					},
					
					// 本地图片路径
					filePath: imgPath,
					
					// 服务器端接收下标 $_FILES['file']
					name: 'file', 
					
					success: (res) => {
						
						console.log('[图片上传] ', res)
						
						// 因为uni.uploadFile不能指定dataType="JSON",服务器响应内容需要自行解析
						res.data = JSON.parse(res.data)
						
						// 成功
						this.imgListUrl[ nextIndex ] = res.data.data.url
						let msg = '图片上传成功 (' + numberNature + '/' + this.imgList.length + ')'
						console.log(msg)
						this.showNotice(msg)
						
						console.log('【图片URL】', this.imgListUrl)
						
						// 继续上传,直到全部上传完毕
						this.upload_image()
						
					},
					fail: (e) => {
						console.log('上传失败', e)
						this.showError('上传失败')
					}
				})
			},
			
			
			// 下一个要上传的图片的下标: false 或者 数字下标
			index_next_upload () {
				let res = false
				for (var i in this.imgListUrl) {
					if (res === false && this.imgListUrl[i] == '') {
						res = parseInt(i)
					}
				}
				console.log('[index_next_upload]', res)
				return res
			},
			
			
			// 显示错误
			showError (msg) {
				uni.showToast({
					icon:'error',
					title: msg
				})
			},
			
			// 显示提示
			showNotice (msg) {
				this.noticeMsg = msg
				setTimeout(()=>{
					this.noticeMsg = ''
				}, 1500)
			}

			
		}
	}
</script>

<style>
	
</style>

thinkphp6文件上传接收API

/**
     * 提交-图片上传
     * 
     */
    public function upload () {
        // 默认返回具体数据
        $res = array();

        // 默认参数
        $fileupload_key = 'file'; // 在uniapp中的 uni.uploadFile({name:'file'}) 设置
        $folder_name_tmp = 'topic'; // 随意、非空,作为接收文件夹名字
        $folder_root = dirname(dirname(dirname(__FILE__))).'/';
        $folder_path_tmp_storage = $folder_root . 'runtime/storage/';
        $folder_public = $folder_root . 'public/';
        $url_root = 'https://xxxxxxxxxxxxxxx/';

        
        // 获取参数
        $targetPath_key = 'path'; // 在uniapp中的 uni.uploadFile({ formData:{path: '/path/to/image/in/server/'} }) 设置
        if (!input('?post.'.$targetPath_key)) {
            return json('缺少参数'.$targetPath_key);
        }
        $subPath = input('post.'.$targetPath_key.'/s');
        $subPath = ltrim($subPath, '/\\');
        
        
        // 获取目标存放路径 - 原始
        // https://www.kancloud.cn/manual/thinkphp6_0/1037639 TP6手册 - 文件上传
        $tmp_file = request()->file( $fileupload_key ); 
        $tmp_savename = Filesystem::putFile( $folder_name_tmp, $tmp_file);

        
        // 获取目标存放路径 - 绝对路径
        $tmp_file_path = realpath($folder_path_tmp_storage . $tmp_savename);
        
        

        // 移动到最终路径
        $target_path = $folder_public . $subPath;
        $target_path_dir = dirname($target_path);
        if (!is_dir( $target_path_dir )) {
            mkdir( $target_path_dir, 0755, true );
        }
        rename($tmp_file_path, $target_path);
        $url = $url_root . $subPath;


        
        // 返回成功信息
        $res['url'] = $url;
        return json($res);
    }

封面

 

uniapp vue3 ref iOS失效 uniapp info.plist_thinkphp_02