홈으로 | 무료회원가입 | 아이디/비번찾기
추천음악방송
HTML5 Video - Audio CurrentTime - Ended 문제점
7년 전
최근 개발작업 중 Samsung Galaxy S4(Android 4.4.2)의 WebView 및 기본 Web Browser에서 아래와 같은 문제점으로 인한 버그가 존재하였고, 그에 따른 해결법을 소개합니다.

HTML5의 Audio, Video 요소를 사용하다 보면 Ended Event 및 Loop Atturibute가 작동하지 않는 현상을 겪게 됩니다. 그 이유는 아래와 같습니다.
Element.currentTime의 값이 milliseconds(1/1000초)가 아닌 picoseconds(1/1000000000000초) 단위로 나오는 현상이 발생 함
Element.duration 값의 차이 발생으로 인해 Loop, Ended 등의 기능 오류

해당 문제를 해결 하기 위해서는 picoseconds(이하 ps)를 milliseconds(이하 ms)로 변경 해주는 작업이 필요합니다.

ps를 ms로 변환하는 방법은 매우 간단합니다. JavaScript에서 제공하는 Number.prototype.toFixed 메소드를 이용하여 소수점 3자리만 표시해주면 됩니다.
var ps = 1.000000000000
var ms = ps.toFixed(3); // 1.000

또한,시스템에서 발생 시켜야할 Ended Event가 작동하지 않기 때문에 이 부분을 동적으로 확장 및 Custom Event 호출이 필요합니다.
Cutsom Event를 호출하는 이유는 대부분의 브라우저의 경우 Ended Event가 정상작동하기 때문에 Event가 중복으로 호출될 가능성이 있기 때문입니다.

해당 Cross-Browsing Trick 에서는 Ended Event 대신 TimeUpdate Event를 이용해 Finish Event를 호출하도록 하겠습니다.

위 내용을 토대로 하여 작성된 JavaScript 소스코드는 아래와 같습니다. “html5.media.cross.browse.js” 로 저장해주세요.
/* HTML5 Media Element Time Events Fix Kits */
(function(window, undefined){
  // Media Tag Names
  var media_tags = ['audio', 'video'];
  for(var type, type_id = 0; type = media_tags[type_id]; type_id++){
    // Get Media Element
    var tags = document.getElementsByTagName(type);
    // Each Media Element
    for(var media, index = 0; media = tags[index]; index++){
      // Binding Finish Event for Loop
      media.addEventListener('finish', function(e){
        var loop = !((this.loop === undefined) || (this.loop === false) || (this.loop === 'null'));
        if(loop) this.play();
      }, false);
      // Binding Loaded Data Event for AutoPlay
      media.addEventListener('loadeddata', function(e){
        var auto = !((this.autoplay === undefined) || (this.autoplay === false) || (this.autoplay === 'null'));
        if(auto) this.play();
      }, false);
      // Fix Media PicoSeconds Time Problems
      media.addEventListener('timeupdate', function(e){
        // Self - this function
        var callee = arguments.callee;
        // Cross-Browsing Timeout Callback
        callee.timeoutCB = function(){
          // Trigger Finish Event
          var eventCB = document.createEvent("HTMLEvents");
              eventCB.initEvent('finish', true, false);
          this.dispatchEvent(eventCB);
          // Trigger Ended Event - system base event function calling
          var eventOB = document.createEvent("HTMLEvents");
              eventOB.initEvent('ended', true, false);
          this.dispatchEvent(eventOB);
          // Reset Current Duration Time (once calling)
          this.addEventListener('play', function(){
            this.setCurrentTime(this.currentTime = 0);
            this.removeEventListener('play', arguments.callee);
          }, false);
        };
        // Save CurrentTime (ps to ms)
        callee.currentTime = Math.max(callee.currentTime || 0, this.currentTime).toFixed(3);
        // Calc RemainTime (Total Duration - Current Duration)
        if((callee.remainTime = parseFloat((this.duration - callee.currentTime).toFixed(3))) === 0){
          // Declare Methd : setCurrentTime(time)
          this.setCurrentTime = this.setCurrentTime || function(time){ this.currentTime = time; }
          // Reset Current Duration Time and Player State
          this.pause(); this.setCurrentTime(callee.currentTime = this.currentTime = 0);
          // ***JS Tricks, Blocking Duplicate Event Callback Function
          clearTimeout(callee.timeoutHandle);
          callee.timeoutHandle = setTimeout(callee.timeoutCB.bind(this), 100);
        }
      }, false);
    }
  }
})(window);

위 소스코드와 같은 경로에 index.html를 아래와 같이 만들고 오류가 발생한 단말기에서 열어보시기 바랍니다.
<!DOCTYPE HTML>
<html>
  <head>
    <title>HTML5 Media CurrentTime Problem ::: Ultimate;D ::: Cross-Browsing</title>
  </head>
  <body>
    <!-- Video 속성이 정상작동하는지 확인하기 위한 요소. -->
    <video id="video" src="http://www.w3schools.com/html/mov_bbb.mp4" controls="controls"></video>
    <script type="text/javascript">
      <!-- Finish 이벤트가 호출 되는지 확인하기 위한 요소. -->
      document.getElementById("video").addEventListener('finish', function(){
        alert('finish');
      }, false);
      <!-- Ended 이벤트가 호출 되는지 확인하기 위한 요소. -->
      document.getElementById("video").addEventListener('ended', function(){
        alert('ended');
      }, false);
    </script>
    <script type="text/javascript" src="html5.media.cross.browse.js"></script>
  </body>
</html>
추천추천 : 1136 추천 목록
번호 제목
3,033
 PHP 정규식 정리
3,032
 PHP 정규식을 활용한 태그 및 특정 문자열 제거 및 추출 방법
3,031
 php 크롤링 또는 파싱 함수, 정규식 모음
3,030
 제이쿼리 기본 명령어
3,029
 웹페이지 가로 모드세로 모드 인식하기
3,028
 모바일 웹 화면 강제 회전(가로모드 고정)
3,027
 [HTML5]에서 frameset 대체 방법과 iframe 속성
3,026
 HTML <Audio> 사용법
3,025
 윈도우10 시스템파일 손상 (초간단 오류 복구방법!!)
3,024
 PHP 파일 존재 여부 파악하기(로컬 파일 존재 및 원격지 파일 존재)
3,023
 [CSS] 박스 세로 가운데 중앙 정렬 6가지
3,022
 CSS Layout 수평 & 수직 정렬
3,021
 여러 도메인들 간 쿠키 공유하기
3,020
 태그 사이에 있는 텍스트를 추출
3,019
 [JQuery] textbox focus on off일때 숫자 콤마 보여주기
3,018
 쿠키 생성,가져오기,삭제
3,017
 사용자 함수 모음
3,016
 마우스,키보드 제한 ( 오른쪽클릭,드래그,영역선택등..)
3,015
 [HTML5] <video> - DOM으로 제어하기
3,014
 HTML5 video 태그에서 영상 좌우반전
3,013
 PHP - 특정 태그 및 문자열 추출, 제거
3,012
 [PHP] define과 defined의 차이
3,011
 우클릭 완벽차단 스크립트
3,010
 iframe 높이 100% 맞추기
3,009
 curl 함수를 이용한 HTTP REFERER 변경
목록
뮤직트로트 부산광역시 부산진구 가야동 ㅣ 개인정보취급방침
Copyright (C) musictrot All rights reserved.