首页 > 其他分享 >h5移动端使用video实现拍照、上传文件对象、选择相册,做手机兼容。

h5移动端使用video实现拍照、上传文件对象、选择相册,做手机兼容。

时间:2023-12-05 10:11:44浏览次数:36  
标签:const 相册 h5 srcList video getUserMedia input type

html部分

<template>
  <div class="views">
    <video style="width: 100vw; height: calc(100vh - 18vh)" object-fit="fill"></video>
    <!-- <img style="width: 100vw; height: calc(100vh - 18vh)" :src="str" alt="" srcset=""> -->
    <div class="picture" @click="pictureClick">
      <i class="iconfont icon-picture"></i>
    </div>
    <div @click="handlePhotographClick" class="action"></div>
    <div class="folder" @click="folderClick">
      <i class="iconfont icon-folder"></i>
    </div>
    <div class="bac">
      <div>
        <div class="img_box" v-for="(img, index) in srcList" :key="img.src + index">
          <i class="iconfont icon-guanbi" @click="delImg(img.name, index)"></i>
          <img src="../../../../public/img/files1.png" v-if="img.name === 'files'" />
          <img :src="img.src" v-else />
          <span>{{ index + 1 }}</span>
        </div>
      </div>
      <div>
        <button class="btn" @click="upload">确认</button>
      </div>
    </div>
  </div>
</template>

js部分

<script>
export default {
  data() {
    return {
      imageUrl: '',
      // 媒体流,用于关闭摄像头
      mediaStreamTrack: null,
      fileName: '', // 上传文件名
      fileList: [], // 上传文件列表
      isCamera: true, // 是否是摄像头
      imgBase64: '', // 图片base64
      photoList: [], // 图片列表
      localHeight: 0, // 本地视频高度
      srcList: [], // 图片路径列表
    }
  },
  mounted() {
    this.invokingCamera()
  },
  destroyed() {
    this.handlePhotographCloseClick()
  },
  methods: {
    // 调用摄像头
    invokingCamera() {
      const self = this
      // 注意本例需要在HTTPS协议网站中运行,新版本Chrome中getUserMedia接口在http下不再支持。
      // 老的浏览器可能根本没有实现 mediaDevices,所以我们可以先设置一个空的对象
      if (navigator.mediaDevices === undefined) {
        navigator.mediaDevices = {}
      }
      // 一些浏览器部分支持 mediaDevices。我们不能直接给对象设置 getUserMedia
      // 因为这样可能会覆盖已有的属性。这里我们只会在没有getUserMedia属性的时候添加它。
      if (navigator.mediaDevices.getUserMedia === undefined) {
        navigator.mediaDevices.getUserMedia = function (constraints) {
          // 首先,如果有getUserMedia的话,就获得它
          const getUserMedia = navigator.getUserMedia || navigator.webkitGetUserMedia || navigator.mozGetUserMedia
          // 一些浏览器根本没实现它 - 那么就返回一个error到promise的reject来保持一个统一的接口
          if (!getUserMedia) {
            return Promise.reject(new Error('getUserMedia is not implemented in this browser'))
          }
          // 否则,为老的navigator.getUserMedia方法包裹一个Promise
          return new Promise(function (resolve, reject) {
            getUserMedia.call(navigator, constraints, resolve, reject)
          })
        }
      }
      const constraints = {
        audio: false,
        video: {
          // 前置摄像头
          facingMode: { exact: 'environment' },
          // 手机端相当于高
          width: Math.max(window.innerWidth, window.innerHeight),
          // 手机端相当于宽
          height: Math.min(window.innerWidth, window.innerHeight),
        },
      }
      navigator.mediaDevices
        .getUserMedia(constraints)
        .then(function (stream) {
          self.mediaStreamTrack = stream
          const video = document.querySelector('video')
          // 旧的浏览器可能没有srcObject
          if ('srcObject' in video) {
            video.srcObject = stream
          } else {
            // 防止在新的浏览器里使用它,应为它已经不再支持了
            video.src = window.URL.createObjectURL(stream)
          }
          video.onloadedmetadata = function (e) {
            video.play()
          }
        })
        .catch(function (err) {
          console.log(err.name + ': ' + err.message)
        })
    },
    // 关闭摄像头
    handlePhotographCloseClick() {
      if (this.mediaStreamTrack) {
        // 关闭摄像头
        this.mediaStreamTrack.getTracks().forEach(function (track) {
          track.stop()
        })
        this.mediaStreamTrack = null
      }
    },
    // 拍照
    handlePhotographClick() {
      const canvas = document.createElement('canvas')
      const ctx = canvas.getContext('2d')
      const video = document.querySelector('video')
      canvas.width = Math.min(video.videoWidth, video.videoHeight)
      canvas.height = Math.max(video.videoWidth, video.videoHeight)
      ctx.drawImage(video, 0, 0, canvas.width, canvas.height)
      // 将图片转为base64
      const base64Image = canvas.toDataURL('image/png')
      // const str = base64Image.replace('data:image/png;base64,', '')
      // // 文件对象数组
      // this.photoList.push(this.convertBlobToFile(this.convertBase64ToBlob(str), new Date().getTime()))
      // 图片路径数组
      this.srcList.push({
        src: base64Image,
        // src: this.str,
        name: new Date().getTime() + '.png',
      })
    },
    folderClick() {
      // 选择文件
      var input = document.createElement('input')
      input.type = 'file'
      input.accept = '.doc,.docx,.txt,application/msword,application/vnd.openxmlformats-officedocument.wordprocessingml.document,.pdf'
      input.addEventListener('change', (e) => {
        // console.log(e.target.files, 'eeeeee')
        // 读取选择的文件
        this.fileList.push(e.target.files)
        this.srcList.push({
          src: null,
          name: 'files',
        })
      })
      // 触发点击事件,打开文件
      input.click()
    },
    pictureClick() {
      const self = this
      // 选择图片
      var input = document.createElement('input')
      input.type = 'file'
      input.accept = 'image/*'
      input.multiple = true
      // 点击事件处理函数
      input.addEventListener('change', function () {
        if (this.files && this.files[0]) {
          // 读取选择的图片文件
          var reader = new FileReader()
          reader.onload = (e) => {
            // 图片加载完成后,将图片URL赋值给input的src属性,即可显示图片
            // console.log(e.target.result, 'e.target.result')
            const type = this.files[0].type.split('/')[1]
            self.srcList.push({
              src: e.target.result,
              name: new Date().getTime() + '.' + type,
            })
          }
          reader.readAsDataURL(this.files[0])
        }
      })
      // 触发点击事件,打开图库
      input.click()
    },
    // 转换为blob格式
    convertBase64ToBlob(base64Str) {
      // 将base64字符串转为二进制数据
      const byteCharacters = atob(base64Str)
      // 创建Blob对象
      const blob = new Blob([byteCharacters], { type: 'application/octet-stream' })
      return blob
    },
    // 将blob转为file对象
    convertBlobToFile(blob, fileName) {
      // 创建File对象
      const type = fileName.split('.')[1]
      const file = new File([blob], fileName, { type: `image/${type}` })
      return file
    },
    delImg(name, index) {
      // 删除图片
      if (name === 'files') {
        this.fileList.splice(index, 1)
      }
      this.srcList.splice(index, 1)
    },
    upload() {
      // 上传按钮
      if (this.srcList.length > 0) {
        for (let i = 0; i < this.srcList.length; i++) {
          // 文件对象数组
          if (this.srcList[i].src === null) {
            continue
          }
          const type = this.srcList[i].name.split('.')[1]
          this.photoList.push(this.convertBlobToFile(this.convertBase64ToBlob(this.srcList[i].src.replace(`data:image/${type};base64,`, '')), new Date().getTime() + `.${type}`))
        }
      }
      const fileObj = []
      for (let i = 0; i < this.fileList.length; i++) {
        fileObj.push(this.fileList[i][0])
      }
      for (let i = 0; i < this.photoList.length; i++) {
        fileObj.push(this.photoList[i])
      }
      console.log(fileObj, 'fileObj')
    },
  },
}
</script>

css部分

<style scoped lang="stylus">
.views{
  width: 100vw;
  height: 100vh;
  background-color: #ccc;
  position: relative

  .picture{
    position: absolute;
    left: 1rem;
    z-index: 9;
    bottom: 4rem;
    .icon-picture{
    font-size: 1rem;
    font-weight: bold;
    }
  }
  .folder{
    position: absolute;
    right: 1rem;
    z-index: 9;
    bottom: 4rem;
   .icon-folder{
      font-size: 1rem;
      font-weight: bold;
    }
  }
  .action{
  position: absolute;
  bottom: 20vh;
  left: 50%;
  margin-left: -35px;
  border-radius: 50px;
  border: 10px solid #ccc;
  box-shadow: 0 0 10px black;
  background-color: #fff;
  width: 70px;
  height: 70px;
  display: flex;
  justify-content: center;
  z-index :99
  }
.bac {
    overflow-x: scroll;
    white-space: nowrap; /* 横向内容不换行 */
    align-items: flex-end; /* 图片索引在左下角 */
  }

  .img_box {
    margin-top: 10px;
    position: relative;
    display: inline-block; /* img_box横向排列 */
    margin-left: 8px;
    margin-bottom: 8px;
  }

  .icon-guanbi {
    position: absolute;
    top: -10px;
    right: -10px;
    width: 20px;
    height: 20px;
    font-size : 20px;
    color: #fff;
    color:red;
    border-radius: 50%;
    display: flex;
    align-items: center;
    justify-content: center;
    cursor: pointer;
  }

  img {
    // width: 100px;
    height:15vh;
    margin-left: 8px
  }

  span {
    position: absolute;
    bottom: 0;
    left: 0;
    margin-left: 4px;
    margin-bottom: 4px;
    color: #fff;
    background-color: #64e8ff;
    padding: 4px;
    font-size: 20px;
    border-radius: 2px 2px 2px 6px;
  }
  .btn{
    color: #fff;
    border: none;
    background: #029afc;
    height: 1rem;
    width: 1.5rem;
    position: absolute;
    right: 0;
    bottom: 1rem;
    z-index: 99;
    border-radius: 10px;
    line-height: 1rem;
  }
}
</style>

以上代码可以直接复制使用,留个关注吧。

标签:const,相册,h5,srcList,video,getUserMedia,input,type
From: https://www.cnblogs.com/axing6688/p/17876598.html

相关文章

  • [Mac软件]HitPaw Video Converter 功能强大的视频格式转换编辑软件激活版
    软件介绍:以令人难以置信的速度将无损视频和音乐转换为1000多种格式:MP4、MOV、AVI、VOB、MKV等。不仅适用于普通编解码器,也适用于高级VP9、ProRes和Opus编码器。这解决了您不支持格式的所有问题,并允许您在任何平台和设备上播放视频。从10,000多个网站下载和保存视频,包括YouTube、Bil......
  • 知识付费在线教育云课堂公众号微信抖音小程序h5开发
    知识付费在线教育云课堂公众号微信抖音小程序h5开发知识付费在线教育云课堂公众号、微信、抖音、小程序、H5开发功能介绍如下:在线直播:提供师生互动、实时答疑、在线讲解等功能,让学生能够通过云课堂平台实时参与课程,提高学习效果。视频课程:提供高质量的视频课程,学生可以随时随地观......
  • h5新增标签demo【+清空浮动】
    直接上代码<!DOCTYPEhtml><htmllang="en"><head><metacharset="UTF-8"><metaname="viewport"content="width=device-width,initial-scale=1.0"><title>Document</title&......
  • HTML5 Video视频组件支持的视频编码格式
    一、HTML5Video视频格式与浏览器的支持情况当前,<video>元素支持三种视频格式:MP4,WebM,和Ogg:浏览器MP4WebMOggInternetExplorerYESNONOChromeYESYESYESFirefoxYESYESYESSafariYESNONOOperaYES(从Opera25起)YESYESMP4=带有......
  • 北京大学 | Video-LLaVA视觉语言大模型:统一输入,同时处理图片和视频数据
    前言 北京大学和其他机构的研究人员近期提出了一种名为Video-LLaVA的视觉语言大模型。该模型的创新之处在于能够同时处理图片和视频作为输入。在处理图片的任务中,该模型展现出了出色的性能,在多个评估榜单中名列前茅,尤其在视频方面取得了令人瞩目的成绩。这项研究的关键点在于关注......
  • H50088:js动态生成html代码的三种形式
    1,//方案一O.TEMPLATE='<divclass="viewer-container">\<divclass="viewer-canvas"></div>\<divclass="viewer-footer">\<divclass="viewer-title&quo......
  • 微信人证核身 - 浮层 H5 配置流程
    本文档以接入微信浮层H5的操作流程为例,描述通过微信HTML5方式接入人脸核身的完整操作流程。引用URL:https://cloud.tencent.com/document/product/1007/78124一、前提条件1.已注册腾讯云账号,并完成实名认证。2.已申请通过腾讯云人脸核身。如果还未完成以上操作,可参考流程指引......
  • uni-app打包成H5,与PC不适配的问题
    既然是写专门的H5站,那说明希望在pc打开,也是H5的排版,比如一体机上,它是网页打开,但是尺寸是1080*1920,在pages.json配置:这里我配置了1920,是因为网页端还有1920的尺寸最大宽度是配置了1920,超出两边留白,这个我测了一下,似乎有点变形,但是我这边目前只需要适配1080的宽度,所以这一点留给大......
  • 截取视频video首帧为图片
     this.url="https://xxx.com/video/demo.mp4"1.把视频放进canvas,在通过canvas拿到视频的第一帧。varcut=function(){letcanvas=document.createElement("canvas");//创建画布canvas.width=video.videoWidth*scale;canvas.height=video.vid......
  • 好用的视频修复软件DVR(Digital Video Repair)
    使用EV录屏时进程中止导致已录的视频也打不开,可以试试有录好的小视频可以作为辅助信息提高修复成功率。https://www.risingresearch.com/en/dvr/......