Xiaoqiang の HTML5 モバイル開発への道 (5) – 美しいビデオ プレーヤーの作成

黄舟
リリース: 2018-05-24 14:40:02
オリジナル
11421 人が閲覧しました

前回の記事では、HTML5 の特徴と習得する必要のある基礎知識を紹介しました。では、HTML5 の利点を実際に体験してみましょう。心配しないでください。理解した上で始めてください。動画ファイルの基礎知識についてお話します。

1. ビデオ形式

現在主流で一般的に使用されているビデオ形式は、avi、rmvb、wmv、mpeg4、ogg、webm です。これらのビデオは、ビデオ、オーディオ、エンコード形式の 3 つの部分で構成されます。 HTML5 には現在、ブラウザに応じてさまざまなエンコーダのセットが多数あります:

H.264 (個人的には楽観的ではありません): このエンコーダは Apple 携帯電話を含む Apple システムのエンコーダであり、特許取得済みのビデオ エンコーダを備えています。エンコードおよび送信プロセスのどの部分に対してもロイヤルティが請求される場合があります。したがって、Safari (Apple のブラウザ) と Internet Explorer はこのエンコーダをサポートしていますが、オープンソースが一般的な傾向になっている現在でも、ブラウザではまだ特許料が請求されています。

Theora: これは特許フリーのエンコード形式であり、あらゆるレベルのエンコード、送信、再生に対応する無料のビデオ エンコーダーです。 Chrome、Firefox、Opera はこのエンコーダをサポートしています。

VP8: このビデオ エンコーダーは Theora に似ていますが、その所有者は Google であり、オープンソースであるため、特許料は必要ありません。 Chrome、Firefox、Opera はこのエンコーダをサポートしています。

AAC: オーディオ コーデック、H.264 と同じ。このオーディオ コーデックには特許制限があり、Safari、Chrome、Internet Explorer はこのオーディオ コーデックをサポートしています。

MP3: これも特許取得済みのテクノロジーである Safari、Chrome、Internet Explorer がこのオーディオ エンコーダーをサポートしています。

PCM: オーディオ CD にデータを保存するための形式。アナログ - デジタル コンバーターによってエンコードされた完全なデータが保存されます。したがって、中国語のロスレス エンコーダのファイル サイズは通常 AAC の数倍であり、Safari、Firefox、Internet Explorer はこのオーディオ エンコーダをサポートしています。

Vorbis: ファイル拡張子は .ogg で、Ogg Vorbis とも呼ばれます。このオーディオ エンコーダーは特許によって保護されていないため、著作権フリーです。サポートされているブラウザには、Chrome、Firefox、Opera が含まれます

Xiaoqiang の HTML5 モバイル開発への道 (5) – 美しいビデオ プレーヤーの作成 2. HTML5 の または

<video src="move.mp4"></video>
ログイン後にコピー

例えば、controls属性ではコンソールの有無を制御できます。

<video src="move.mp4" controls="controls">  
    浏览器不支持HTML5的视频播放功能  
</video>
ログイン後にコピー

上記のビデオ形式から、ブラウザごとに異なるビデオ形式がサポートされていることがわかります。そのため、 タグを使用して複数の形式のビデオを指定できます。デフォルトでは、ブラウザはファイルのダウンロードを自動的に開始します。タイプ。

<video width="400" controls="controls">  
    <source src="move.mp4"  type="video/mp4" />  
    <source src="move.ogg"  type="video/ogg" />  
</video>
ログイン後にコピー

3. ビデオプレーヤーを作成します

index.html

<!DOCTYPE html>  
<html>  
<head>  
<title>Demo 1 | Custom HTML5 Video Controls with jQuery</title>  
<link rel="stylesheet" href="../vendorstyle.css" />  
<link rel="stylesheet" href="style.css" />  
<script src="http://ajax.googleapis.com/ajax/libs/jquery/1.7.1/jquery.min.js"></script>  
<!--[if lt IE 9]>  
<script src="http://html5shim.googlecode.com/svn/trunk/html5.js"></script>  
<![endif]-->  
<script src="../vendorscript.js"></script>  
<script src="video.js"></script>  
<!--[if lt IE 9]>  
<script>  
$(document).ready(function() {  
    $(&#39;.control&#39;).hide();  
    $(&#39;.loading&#39;).fadeOut(500);  
    $(&#39;.caption&#39;).fadeOut(500);  
});  
</script>  
<![endif]-->  
<link rel="shortcut icon" href="http://www.inwebson.com/wp-content/themes/inwebson/favicon.ico" />  
</head>  
  
<body>  
<!-- Header -->  
<header>  
    <h1>Custom HTML5 Video Controls with jQuery</h1>  
    <p id="backlinks">  
 <a href="http://www.inwebson.com/custom-html5-video-controls-with-jquery-and-css/">BACK TO ARTICLE »</a>  
        <a href="http://www.inwebson.com">Visit inWebson.com »</a>  
    </p>  
    <p class="clearfix"></p>  
</header>  
  
<!-- Content -->  
<section id="wrapper">  
  
    <!-- Title -->  
    <h2>Demo 1</h2>  
    <h3>Custom HTML5 Video Controls</h3>  
  
<p class="videoContainer">  
      
    <video id="myVideo" controls preload="auto" poster="poster.jpg" width="600" height="350" >  
      <source src="http://demo.inwebson.com/html5-video/iceage4.mp4" type="video/mp4" />  
      <source src="http://demo.inwebson.com/html5-video/iceage4.webm" type="video/webM" />  
      <source src="http://demo.inwebson.com/html5-video/iceage4.ogv" type="video/ogg" />  
      <p>Your browser does not support the video tag.</p>  
    </video>  
    <p class="caption">This is HTML5 video with custom controls</p>  
    <p class="control">  
  
        <p class="topControl">  
            <p class="progress">  
                <span class="bufferBar"></span>  
                <span class="timeBar"></span>  
            </p>  
            <p class="time">  
                <span class="current"></span> /   
                <span class="duration"></span>   
            </p>  
        </p>  
          
        <p class="btmControl">  
            <p class="btnPlay btn" title="Play/Pause video"></p>  
            <p class="btnStop btn" title="Stop video"></p>  
            <p class="spdText btn">Speed: </p>  
            <p class="btnx1 btn text selected" title="Normal speed">x1</p>  
            <p class="btnx3 btn text" title="Fast forward x3">x3</p>  
            <p class="btnFS btn" title="Switch to full screen"></p>  
            <p class="btnLight lighton btn" title="Turn on/off light"></p>  
            <p class="volume" title="Set volume">  
                <span class="volumeBar"></span>  
            </p>  
            <p class="sound sound2 btn" title="Mute/Unmute sound"></p>  
        </p>  
          
    </p>  
    <p class="loading"></p>  
</p>  
  
    <!-- Navigation -->  
    <nav id="navigation">  
        <ul>  
            <li class="currentbtn"><a href="#" title="Demo 1">DEMO 1</a></li>  
            <li><a href="../demo2/" title="Demo 2">DEMO 2</a></li>  
        </ul>  
        <p class="clearfix"></p>      
    </nav>  
</section>  
  
<!-- Footer -->  
<footer>  
    <span>© 2011 <a href="http://www.inwebson.com">inWebson.com</a>. 
    Design by <a href="http://www.inwebson.com/contactus">Kenny Ooi</a>. 
    Powered by <a href="http://www.inwebson.com/html5/">HTML5</a> and 
    <a href="http://www.inwebson.com/jquery/">jQuery</a>.</span>  
</footer>  
</body>  
</html>
ログイン後にコピー

style.css

/* video container */  
.videoContainer{  
    width:600px;  
    height:350px;  
    position:relative;  
    overflow:hidden;  
    background:#000;  
    color:#ccc;  
}  
  
/* video caption css */  
.caption{  
    display:none;  
    position:absolute;  
    top:0;  
    left:0;  
    width:100%;  
    padding:10px;  
    color:#ccc;  
    font-size:20px;  
    font-weight:bold;  
    box-sizing: border-box;  
    -ms-box-sizing: border-box;  
    -webkit-box-sizing: border-box;  
    -moz-box-sizing: border-box;  
    background: #1F1F1F; /* fallback */  
    background:-moz-linear-gradient(top,#242424 50%,#1F1F1F 50%,#171717 100%);  
    background:-webkit-linear-gradient(top,#242424 50%,#1F1F1F 50%,#171717 100%);  
    background:-o-linear-gradient(top,#242424 50%,#1F1F1F 50%,#171717 100%);  
}  
  
/*** VIDEO CONTROLS CSS ***/  
/* control holder */  
.control{  
    background:#333;  
    color:#ccc;  
    position:absolute;  
    bottom:0;  
    left:0;  
    width:100%;  
    z-index:5;  
    display:none;  
}  
/* control top part */  
.topControl{  
    height:11px;  
    border-bottom:1px solid #404040;  
    padding:1px 5px;  
    background:#1F1F1F; /* fallback */  
    background:-moz-linear-gradient(top,#242424 50%,#1F1F1F 50%,#171717 100%);  
    background:-webkit-linear-gradient(top,#242424 50%,#1F1F1F 50%,#171717 100%);  
    background:-o-linear-gradient(top,#242424 50%,#1F1F1F 50%,#171717 100%);  
}  
/* control bottom part */  
.btmControl{  
    clear:both;  
    background: #1F1F1F; /* fallback */  
    background:-moz-linear-gradient(top,#242424 50%,#1F1F1F 50%,#171717 100%);  
    background:-webkit-linear-gradient(top,#242424 50%,#1F1F1F 50%,#171717 100%);  
    background:-o-linear-gradient(top,#242424 50%,#1F1F1F 50%,#171717 100%);  
}  
.control p.btn {  
    float:left;  
    width:34px;  
    height:30px;  
    padding:0 5px;  
    border-right:1px solid #404040;  
    cursor:pointer;  
}  
.control p.text{  
    font-size:12px;  
    font-weight:bold;  
    line-height:30px;  
    text-align:center;  
    font-family:verdana;  
    width:20px;  
    border:none;  
    color:#777;  
}  
.control p.btnPlay{  
    background:url(control.png) no-repeat 0 0;  
    border-left:1px solid #404040;  
}  
.control p.paused{  
    background:url(control.png) no-repeat 0 -30px;  
}  
.control p.btnStop{  
    background:url(control.png) no-repeat 0 -60px;  
}  
.control p.spdText{  
    border:none;  
    font-size:14px;  
    line-height:30px;  
    font-style:italic;  
}  
.control p.selected{  
    font-size:15px;  
    color:#ccc;  
}  
.control p.sound{  
    background:url(control.png) no-repeat -88px -30px;  
    border:none;  
    float:right;  
}  
.control p.sound2{  
    background:url(control.png) no-repeat -88px -60px !important;  
}  
.control p.muted{  
    background:url(control.png) no-repeat -88px 0 !important;  
}  
.control p.btnFS{  
    background:url(control.png) no-repeat -44px 0;  
    float:right;  
}  
.control p.btnLight{  
    background:url(control.png) no-repeat -44px -60px;  
    border-left:1px solid #404040;  
    float:right;  
}  
.control p.lighton{  
    background:url(control.png) no-repeat -44px -30px !important;  
}  
  
/* PROGRESS BAR CSS */  
/* Progress bar */  
.progress {  
    width:85%;  
    height:10px;  
    position:relative;  
    float:left;  
    cursor:pointer;  
    background: #444; /* fallback */  
    background:-moz-linear-gradient(top,#666,#333);  
    background:-webkit-linear-gradient(top,#666,#333);  
    background:-o-linear-gradient(top,#666,#333);  
    box-shadow:0 2px 3px #333 inset;  
    -moz-box-shadow:0 2px 3px #333 inset;  
    -webkit-box-shadow:0 2px 3px #333 inset;  
    border-radius:10px;  
    -moz-border-radius:10px;  
    -webkit-border-radius:10px;  
}  
.progress span {  
    height:100%;  
    position:absolute;  
    top:0;  
    left:0;  
    display:block;  
    border-radius:10px;  
    -moz-border-radius:10px;  
    -webkit-border-radius:10px;  
}  
.timeBar{  
    z-index:10;  
    width:0;  
    background: #3FB7FC; /* fallback */  
    background:-moz-linear-gradient(top,#A0DCFF 50%,#3FB7FC 50%,#16A9FF 100%);  
    background:-webkit-linear-gradient(top,#A0DCFF 50%,#3FB7FC 50%,#16A9FF 100%);  
    background:-o-linear-gradient(top,#A0DCFF 50%,#3FB7FC 50%,#16A9FF 100%);  
    box-shadow:0 0 1px #fff;  
    -moz-box-shadow:0 0 1px #fff;  
    -webkit-box-shadow:0 0 1px #fff;  
}  
.bufferBar{  
    z-index:5;  
    width:0;  
    background: #777;  
    background:-moz-linear-gradient(top,#999,#666);  
    background:-webkit-linear-gradient(top,#999,#666);  
    background:-o-linear-gradient(top,#999,#666);  
    box-shadow:2px 0 5px #333;  
    -moz-box-shadow:2px 0 5px #333;  
    -webkit-box-shadow:2px 0 5px #333;  
}  
/* time and duration */  
.time{  
    width:15%;  
    float:right;  
    text-align:center;  
    font-size:11px;  
    line-height:12px;  
}  
  
/* VOLUME BAR CSS */  
/* volume bar */  
.volume{  
    position:relative;  
    cursor:pointer;  
    width:70px;  
    height:10px;  
    float:right;  
    margin-top:10px;  
    margin-right:10px;  
}  
.volumeBar{  
    display:block;  
    height:100%;  
    position:absolute;  
    top:0;  
    left:0;  
    background-color:#eee;  
    z-index:10;  
}  
  
/* OTHERS CSS */  
/* video screen cover */  
.loading, #init{  
    position:absolute;  
    top:0;  
    left:0;  
    width:100%;  
    height:100%;  
    background:url(loading.gif) no-repeat 50% 50%;  
    z-index:2;  
    display:none;  
}  
#init{  
    background:url(bigplay.png) no-repeat 50% 50% !important;  
    cursor:pointer;  
}
ログイン後にコピー

video.js

$(document).ready(function(){  
    //INITIALIZE  
    var video = $(&#39;#myVideo&#39;);  
      
    //remove default control when JS loaded  
    video[0].removeAttribute("controls");  
    $(&#39;.control&#39;).show().css({&#39;bottom&#39;:-45});  
    $(&#39;.loading&#39;).fadeIn(500);  
    $(&#39;.caption&#39;).fadeIn(500);  
   
    //before everything get started  
    video.on(&#39;loadedmetadata&#39;, function() {  
        $(&#39;.caption&#39;).animate({&#39;top&#39;:-45},300);  
              
        //set video properties  
        $(&#39;.current&#39;).text(timeFormat(0));  
        $(&#39;.duration&#39;).text(timeFormat(video[0].duration));  
        updateVolume(0, 0.7);  
              
        //start to get video buffering data   
        setTimeout(startBuffer, 150);  
              
        //bind video events  
        $(&#39;.videoContainer&#39;)  
        .append(&#39;<p id="init"></p>&#39;)  
        .hover(function() {  
            $(&#39;.control&#39;).stop().animate({&#39;bottom&#39;:0}, 500);  
            $(&#39;.caption&#39;).stop().animate({&#39;top&#39;:0}, 500);  
        }, function() {  
            if(!volumeDrag && !timeDrag){  
                $(&#39;.control&#39;).stop().animate({&#39;bottom&#39;:-45}, 500);  
                $(&#39;.caption&#39;).stop().animate({&#39;top&#39;:-45}, 500);  
            }  
        })  
        .on(&#39;click&#39;, function() {  
            $(&#39;#init&#39;).remove();  
            $(&#39;.btnPlay&#39;).addClass(&#39;paused&#39;);  
            $(this).unbind(&#39;click&#39;);  
            video[0].play();  
        });  
        $(&#39;#init&#39;).fadeIn(200);  
    });  
      
    //display video buffering bar  
    var startBuffer = function() {  
        var currentBuffer = video[0].buffered.end(0);  
        var maxduration = video[0].duration;  
        var perc = 100 * currentBuffer / maxduration;  
        $(&#39;.bufferBar&#39;).css(&#39;width&#39;,perc+&#39;%&#39;);  
              
        if(currentBuffer < maxduration) {  
            setTimeout(startBuffer, 500);  
        }  
    };    
      
    //display current video play time  
    video.on(&#39;timeupdate&#39;, function() {  
        var currentPos = video[0].currentTime;  
        var maxduration = video[0].duration;  
        var perc = 100 * currentPos / maxduration;  
        $(&#39;.timeBar&#39;).css(&#39;width&#39;,perc+&#39;%&#39;);      
        $(&#39;.current&#39;).text(timeFormat(currentPos));   
    });  
      
    //CONTROLS EVENTS  
    //video screen and play button clicked  
    video.on(&#39;click&#39;, function() { playpause(); } );  
    $(&#39;.btnPlay&#39;).on(&#39;click&#39;, function() { playpause(); } );  
    var playpause = function() {  
        if(video[0].paused || video[0].ended) {  
            $(&#39;.btnPlay&#39;).addClass(&#39;paused&#39;);  
            video[0].play();  
        }  
        else {  
            $(&#39;.btnPlay&#39;).removeClass(&#39;paused&#39;);  
            video[0].pause();  
        }  
    };  
      
    //speed text clicked  
    $(&#39;.btnx1&#39;).on(&#39;click&#39;, function() { fastfowrd(this, 1); });  
    $(&#39;.btnx3&#39;).on(&#39;click&#39;, function() { fastfowrd(this, 3); });  
    var fastfowrd = function(obj, spd) {  
        $(&#39;.text&#39;).removeClass(&#39;selected&#39;);  
        $(obj).addClass(&#39;selected&#39;);  
        video[0].playbackRate = spd;  
        video[0].play();  
    };  
      
    //stop button clicked  
    $(&#39;.btnStop&#39;).on(&#39;click&#39;, function() {  
        $(&#39;.btnPlay&#39;).removeClass(&#39;paused&#39;);  
        updatebar($(&#39;.progress&#39;).offset().left);  
        video[0].pause();  
    });  
      
    //fullscreen button clicked  
    $(&#39;.btnFS&#39;).on(&#39;click&#39;, function() {  
        if($.isFunction(video[0].webkitEnterFullscreen)) {  
            video[0].webkitEnterFullscreen();  
        }     
        else if ($.isFunction(video[0].mozRequestFullScreen)) {  
            video[0].mozRequestFullScreen();  
        }  
        else {  
            alert(&#39;Your browsers doesn\&#39;t support fullscreen&#39;);  
        }  
    });  
      
    //light bulb button clicked  
    $(&#39;.btnLight&#39;).click(function() {  
        $(this).toggleClass(&#39;lighton&#39;);  
          
        //if lightoff, create an overlay  
        if(!$(this).hasClass(&#39;lighton&#39;)) {  
            $(&#39;body&#39;).append(&#39;<p class="overlay"></p>&#39;);  
            $(&#39;.overlay&#39;).css({  
                &#39;position&#39;:&#39;absolute&#39;,  
                &#39;width&#39;:100+&#39;%&#39;,  
                &#39;height&#39;:$(document).height(),  
                &#39;background&#39;:&#39;#000&#39;,  
                &#39;opacity&#39;:0.9,  
                &#39;top&#39;:0,  
                &#39;left&#39;:0,  
                &#39;z-index&#39;:999  
            });  
            $(&#39;.videoContainer&#39;).css({  
                &#39;z-index&#39;:1000  
            });  
        }  
        //if lighton, remove overlay  
        else {  
            $(&#39;.overlay&#39;).remove();  
        }  
    });  
      
    //sound button clicked  
    $(&#39;.sound&#39;).click(function() {  
        video[0].muted = !video[0].muted;  
        $(this).toggleClass(&#39;muted&#39;);  
        if(video[0].muted) {  
            $(&#39;.volumeBar&#39;).css(&#39;width&#39;,0);  
        }  
        else{  
            $(&#39;.volumeBar&#39;).css(&#39;width&#39;, video[0].volume*100+&#39;%&#39;);  
        }  
    });  
      
    //VIDEO EVENTS  
    //video canplay event  
    video.on(&#39;canplay&#39;, function() {  
        $(&#39;.loading&#39;).fadeOut(100);  
    });  
      
    //video canplaythrough event  
    //solve Chrome cache issue  
    var completeloaded = false;  
    video.on(&#39;canplaythrough&#39;, function() {  
        completeloaded = true;  
    });  
      
    //video ended event  
    video.on(&#39;ended&#39;, function() {  
        $(&#39;.btnPlay&#39;).removeClass(&#39;paused&#39;);  
        video[0].pause();  
    });  
  
    //video seeking event  
    video.on(&#39;seeking&#39;, function() {  
        //if video fully loaded, ignore loading screen  
        if(!completeloaded) {   
            $(&#39;.loading&#39;).fadeIn(200);  
        }     
    });  
      
    //video seeked event  
    video.on(&#39;seeked&#39;, function() { });  
      
    //video waiting for more data event  
    video.on(&#39;waiting&#39;, function() {  
        $(&#39;.loading&#39;).fadeIn(200);  
    });  
      
    //VIDEO PROGRESS BAR  
    //when video timebar clicked  
    var timeDrag = false;   /* check for drag event */  
    $(&#39;.progress&#39;).on(&#39;mousedown&#39;, function(e) {  
        timeDrag = true;  
        updatebar(e.pageX);  
    });  
    $(document).on(&#39;mouseup&#39;, function(e) {  
        if(timeDrag) {  
            timeDrag = false;  
            updatebar(e.pageX);  
        }  
    });  
    $(document).on(&#39;mousemove&#39;, function(e) {  
        if(timeDrag) {  
            updatebar(e.pageX);  
        }  
    });  
    var updatebar = function(x) {  
        var progress = $(&#39;.progress&#39;);  
          
        //calculate drag position  
        //and update video currenttime  
        //as well as progress bar  
        var maxduration = video[0].duration;  
        var position = x - progress.offset().left;  
        var percentage = 100 * position / progress.width();  
        if(percentage > 100) {  
            percentage = 100;  
        }  
        if(percentage < 0) {  
            percentage = 0;  
        }  
        $(&#39;.timeBar&#39;).css(&#39;width&#39;,percentage+&#39;%&#39;);    
        video[0].currentTime = maxduration * percentage / 100;  
    };  
  
    //VOLUME BAR  
    //volume bar event  
    var volumeDrag = false;  
    $(&#39;.volume&#39;).on(&#39;mousedown&#39;, function(e) {  
        volumeDrag = true;  
        video[0].muted = false;  
        $(&#39;.sound&#39;).removeClass(&#39;muted&#39;);  
        updateVolume(e.pageX);  
    });  
    $(document).on(&#39;mouseup&#39;, function(e) {  
        if(volumeDrag) {  
            volumeDrag = false;  
            updateVolume(e.pageX);  
        }  
    });  
    $(document).on(&#39;mousemove&#39;, function(e) {  
        if(volumeDrag) {  
            updateVolume(e.pageX);  
        }  
    });  
    var updateVolume = function(x, vol) {  
        var volume = $(&#39;.volume&#39;);  
        var percentage;  
        //if only volume have specificed  
        //then direct update volume  
        if(vol) {  
            percentage = vol * 100;  
        }  
        else {  
            var position = x - volume.offset().left;  
            percentage = 100 * position / volume.width();  
        }  
          
        if(percentage > 100) {  
            percentage = 100;  
        }  
        if(percentage < 0) {  
            percentage = 0;  
        }  
          
        //update volume bar and video volume  
        $(&#39;.volumeBar&#39;).css(&#39;width&#39;,percentage+&#39;%&#39;);      
        video[0].volume = percentage / 100;  
          
        //change sound icon based on volume  
        if(video[0].volume == 0){  
            $(&#39;.sound&#39;).removeClass(&#39;sound2&#39;).addClass(&#39;muted&#39;);  
        }  
        else if(video[0].volume > 0.5){  
            $(&#39;.sound&#39;).removeClass(&#39;muted&#39;).addClass(&#39;sound2&#39;);  
        }  
        else{  
            $(&#39;.sound&#39;).removeClass(&#39;muted&#39;).removeClass(&#39;sound2&#39;);  
        }  
          
    };  
  
    //Time format converter - 00:00  
    var timeFormat = function(seconds){  
        var m = Math.floor(seconds/60)<10 ? "0"+Math.floor(seconds/60) : Math.floor(seconds/60);  
    var s = Math.floor(seconds-(m*60))<10 ? "0"+Math.floor(seconds-(m*60)) :Math.floor(seconds-(m*60));  
        return m+":"+s;  
    };  
});
ログイン後にコピー

上記は、Xiaoqiang の HTML5 モバイル開発の旅 (5) - 美しいビデオプレーヤーの作成です。 、PHP 中国語 Web サイト (www.php.cn) に注意してください。

ソース:php.cn
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート