weixin_33720078 2020-04-02 12:57 采纳率: 0%
浏览 83

快照视频作为预览

I have an issue with getting a snapshot from a video to use as a header image. Underneath I have the code I currently use, but the output I get is only a black image. In the console I don't get any issues. I do this after I have uploaded the video to the fileserver I use.

    let video = $(videoPath).find('#newVideo').get(0);
    let canvas = document.createElement('canvas');
    video.onloadedmetadata = function() {
        video.play();
        canvas.width = 640;
        canvas.height = 480;
        var ctx = canvas.getContext('2d');
        ctx.drawImage(video, 0, 0, canvas.width, canvas.height);
        var dataURI = canvas.toDataURL('image/jpeg');
        let $previewBox = $(".post-preview-box");
        let $previewCardImg = $previewBox.find(".card-img");

I have used How to take a snapshot of HTML5-JavaScript-based video player? as a base to get a snapshot and then added some other things in the hope it would work, but so far I have been unable to do.

Any help or input on what to try is welcome.

Edit 04-03-2020 (april third): To give more clarification I will add a console out of the dataURI and changes to code I made.



Added the currentTime of the video and set it to 3 (I also have tried it with 5).

    let video = $(videoPath).find('#newVideo').get(0);
    let canvas = document.createElement('canvas');
    video.onloadedmetadata = function() {
        video.currentTime = 3;
        video.play();
        var ctx = canvas.getContext('2d');
        ctx.drawImage(video, 0, 0, canvas.width, canvas.height);
        var dataURI = canvas.toDataURL('image/jpeg');
        let $previewBox = $(".post-preview-box");
        console.log(dataURI);
        let $previewCardImg = $previewBox.find(".card-img");
  • 写回答

1条回答 默认 最新

  • weixin_33716941 2020-04-03 13:12
    关注

    Without seeing all your code, here is an example which gets 10 screens.

    var screens = [];
    
    var video = document.getElementById('video');
    var video_preview = document.getElementById('video_preview');
    
    var loadingContainer = document.getElementById('loading-container')
    
    function loadVideo(event) {
      loadingContainer.style.display = 'block'
      screens = [];
    
      var reader = new FileReader()
      reader.onload = function(e) {
        video.src = video_preview.src = e.target.result;
        video.autoplay = video_preview.autoplay = true;
        video.hasLoaded = video_preview.hasLoaded = false;
    
        video_preview.addEventListener('canplay', function() {
          video_preview.hasLoaded = true;
          video.play();
        })
    
        video.addEventListener('canplay', function() {
    
          // first time
          if (!video.hasLoaded) {
            console.log('Loaded: video duration: ', this.duration, event.target.files[0].size)
    
            loadingContainer.innerText = 'Generating screens...'
    
            video.hasLoaded = true;
    
            var self = this;
    
            (function repeat(i) {
              setTimeout(function() {
                var timestamp = ((self.duration / 10) * i) / 1.1; // fudge abit so dont get start/end frames
    
                console.log('seeking to:', timestamp)
                self.currentTime = timestamp
    
                if (--i) {
                  // next
                  repeat(i);
                } else {
                  //
                  loadingContainer.style.display = 'none'
    
                  // all screens grabbed
                  var str = '<div style="position:relative;width:calc(100% + .25rem)">'
                  screens.reverse().forEach(function(screen) {
                    str += '<img src="' + screen + '" style="width:32%" class="img-thumbnail m-1" />';
                  });
                  str += '</div>';
                  document.getElementById("screens-container").innerHTML = str;
                }
              }, 500) // how fast to attempt to grab screens
            })(11); // iterations i.e how many screens
          }
        }, false);
    
        video.addEventListener('seeked', function() {
          console.log('grabbing screen for', this.currentTime)
          takeScreen();
        }, false);
      }
    
      reader.readAsDataURL(event.target.files[0]);
    }
    
    function takeScreen() {
      var filename = video.src;
      var w = video.videoWidth;
      var h = video.videoHeight;
      var canvas = document.createElement('canvas');
      canvas.width = w;
      canvas.height = h;
      var ctx = canvas.getContext('2d');
      ctx.drawImage(video, 0, 0, w, h);
      var data = canvas.toDataURL("image/jpg");
    
      screens.push(data)
    
      loadingContainer.innerText = 'Generated ' + screens.length + ' out of 10 screens...'
    }
    
    function failed(e) {
      switch (e.target.error.code) {
        case e.target.error.MEDIA_ERR_ABORTED:
          console.log('You aborted the video playback.');
          break;
        case e.target.error.MEDIA_ERR_NETWORK:
          console.log('A network error caused the video download to fail part-way.');
          break;
        case e.target.error.MEDIA_ERR_DECODE:
          console.log('The video playback was aborted due to a corruption problem or because the video used features your browser did not support.');
          break;
        case e.target.error.MEDIA_ERR_SRC_NOT_SUPPORTED:
          console.log('The video could not be loaded, either because the server or network failed or because the format is not supported.');
          break;
        default:
          console.log('An unknown error occurred.');
          break;
      }
    }
    body,
    html {
      margin: 0;
      padding: 0
    }
    
    li {
      display: inline-block;
      list-style-type: none
    }
    
    .jumbotron {
      margin-bottom: 1rem;
    }
    <div class="container">
    
      <div class="card">
        <div class="card-header" style="background-color: #e9ecef">
          <h5 class="card-title">Video Preview</h5>
          <video id="video_preview" onerror="failed(event)" controls="controls" preload="none" muted style="width: 100%"></video>
          <video id="video" onerror="failed(event)" controls="controls" preload="none" muted style="display:none"></video>
          <div class="card-body">
            <a href="javascript:void(0)" class="btn btn-sm btn-primary btn-block" onClick="$('[type=\'file\']').trigger('click')">
              <i class="fa fa-upload"></i> Load Video
            </a>
            <form id="uploadForm" ref="uploadForm" action='/upload' method='post' enctype="multipart/form-data">
              <input type="file" name="file" accept="video/*" style="display: none" onchange="loadVideo(event)">
            </form>
            <div id="loading-container" style="display:none">Loading...</div>
          </div>
    
          <div class="card-body">
            <h5 class="card-title">Screens</h5>
            <div id="screens-container"></div>
          </div>
        </div>
      </div>
    
      <link href="https://stackpath.bootstrapcdn.com/bootstrap/4.3.1/css/bootstrap.min.css" rel="stylesheet" integrity="sha384-ggOyR0iXCbMQv3Xipma34MD+dH/1fQ784/j6cY/iJTQUOhcWr7x9JvoRxT2MZw1T" crossorigin="anonymous">
    
      <script src="https://code.jquery.com/jquery-3.2.1.min.js"></script>
      <script src="https://maxcdn.bootstrapcdn.com/bootstrap/4.3.1/js/bootstrap.min.js"></script>

    For a more complete example see: https://codesandbox.io/s/clipseed-hfhee

    </div>
    
    评论

报告相同问题?

悬赏问题

  • ¥15 树莓派与pix飞控通信
  • ¥15 自动转发微信群信息到另外一个微信群
  • ¥15 outlook无法配置成功
  • ¥30 这是哪个作者做的宝宝起名网站
  • ¥60 版本过低apk如何修改可以兼容新的安卓系统
  • ¥25 由IPR导致的DRIVER_POWER_STATE_FAILURE蓝屏
  • ¥50 有数据,怎么建立模型求影响全要素生产率的因素
  • ¥50 有数据,怎么用matlab求全要素生产率
  • ¥15 TI的insta-spin例程
  • ¥15 完成下列问题完成下列问题