X-Git-Url: http://git.osdn.jp/view?a=blobdiff_plain;f=0.6.x%2Fjs%2F07_audio%2F10_XAudioSprite.js;h=bb4272b1ebd559f5603ab4734f39839b9bee5e12;hb=512e08f4d38eab417f9651277e8a50c08535cb07;hp=74fb1c662cc4ee8943f898d5d27be35be3393857;hpb=3256f11c856314a1425b8459b9d000a88caf8258;p=pettanr%2FclientJs.git diff --git a/0.6.x/js/07_audio/10_XAudioSprite.js b/0.6.x/js/07_audio/10_XAudioSprite.js index 74fb1c6..bb4272b 100644 --- a/0.6.x/js/07_audio/10_XAudioSprite.js +++ b/0.6.x/js/07_audio/10_XAudioSprite.js @@ -4,64 +4,130 @@ * Mobile Opera11 は Audio をサポートするがイベントが取れない * iframe 内で生成して、Audio Sprite の preset で再生できないか? */ -var X_Audio_Sprite_shouldUse = window.HTMLAudioElement && ( X_UA.iOS || X_UA.AndroidBrowser || X_UA.OperaMobile || X_UA.OperaTablet ), - X_Audio_Sprite_needTouchFirst = !!X_UA.iOS, - X_Audio_Sprite_inTouchAction = false, - X_Audio_Sprite_enableMultiTrack = !( X_UA.iOS < 6 ), - X_Audio_Sprite_enableVolume = window.HTMLAudioElement && ( !X_UA.iOS && !X_UA.AndroidBrowser && !X_UA.OperaMobile && !X_UA.OperaTablet ), - X_Audio_Sprite_useVideoForMulti = 4 <= X_UA.AndroidBrowser, - X_Audio_Sprite_maxTracks = X_UA.iOS < 6 ? 1 : X_Audio_Sprite_useVideoForMulti ? 2 : 9, - X_Audio_Sprite_lengthSilence = 10000, // 一番最初の無音部分の長さ - X_Audio_Sprite_lengthDistance = 5000, // 音間の無音の長さ - X_Audio_Sprite_uid = 0, - X_Audio_Sprite_members = {}, - X_Audio_Sprite_TEMP = { +var X_AudioSprite_shouldUse = X_HTMLAudio && ( X_UA[ 'iOS' ] || X_UA[ 'AOSP' ] || X_UA[ 'OperaMobile' ] || X_UA[ 'OperaTablet' ] ), // Flash がない + X_AudioSprite_useVideoForMulti = //( 3.1 <= X_UA[ 'AOSP' ] < 4 ) || + //( ( 4.2 <= X_UA[ 'AOSP' ] ), + // ドスパラパッドはビデオのインライン再生が不可 + false, + X_AudioSprite_disableMultiTrack = !X_WebAudio && ( X_UA[ 'iOS' ] || 4 <= X_UA[ 'AOSP' ] || X_UA[ 'ChromeWV' ] || ( X_UA[ 'WinPhone' ] && X_UA[ 'IE9' ] ) ), + X_AudioSprite_enableVolume = X_HTMLAudio && ( !X_UA[ 'iOS' ] && !X_UA[ 'AOSP' ] && !X_UA[ 'OperaMobile' ] && !X_UA[ 'OperaTablet' ] ), // TODO fennec は 25以上 + // http://tukumemo.com/html5-audio-sp/ + // iOS6、Android4.1から同時再生が可能になりました。 + X_AudioSprite_maxTracks = X_AudioSprite_useVideoForMulti ? 2 : X_AudioSprite_disableMultiTrack ? 1 : 9, + X_AudioSprite_lengthSilence = 10000, // 一番最初の無音部分の長さ + X_AudioSprite_lengthDistance = 5000, // 音間の無音の長さ + X_AudioSprite_uid = 0, + X_AudioSprite_TEMP = { presets : {}, BGMs : {}, tracks : [], + pauseTracks : [], // X_EVENT_DEACTIVATE によって pause した再生中のトラックたち。 volume : 1, bgmTrack : null, bgmPosition : 0, bgmName : '', bgmLooped : false, - bgmPlaying : false + bgmPlaying : false, + event : null }, - X_Audio_Sprite_instance; + X_AudioSprite, + X_AudioSprite_numTracks, + X_AudioSprite_useVideo; -X.Audio.Sprite = { +/** + * { + * urls : [ 'xx.ogg', 'xx.mp3' ], + * numTracks : 3, + * useVideo : false, + * volume : 1, + * BGM_01 : [ '15.00', '45.500', true, '17.666', '50.999' ], + * BGM_02 : [ '56.00', '1:15.230', true ] + * } + * + * X_EVENT_BACKEND_READY + * X_EVENT_BACKEND_NONE + * + * X_EVENT_READY + * X_EVENT_MEDIA_LOOPED + * X_EVENT_MEDIA_ENDED + * + * @namespace X.AudioSprite + * @alias X.AudioSprite + */ +X[ 'AudioSprite' ] = function( setting ){ + var tracks = X_AudioSprite_TEMP.tracks, + bgms = X_AudioSprite_TEMP.BGMs, + presets = X_AudioSprite_TEMP.presets, + urls = setting[ 'urls' ], + video = setting[ 'useVideo' ], + n = video ? 1 : setting[ 'numTracks' ] || 1, + volume = setting[ 'volume' ], + k, i, v, track; - shouldUse : X_Audio_Sprite_shouldUse, + + if( X_AudioSprite ) X_AudioSprite[ 'kill' ](); + + X_AudioSprite = X_Class_override( X_EventDispatcher(), X_AudioSprite_members ); + X_ViewPort[ 'listen' ]( [ X_EVENT_VIEW_ACTIVATE, X_EVENT_VIEW_DEACTIVATE, X_EVENT_UNLOAD ], X_AudioSprite_handleEvent ); - needTouchFirst : X_Audio_Sprite_needTouchFirst, + n = n <= X_AudioSprite_maxTracks ? n : X_AudioSprite_maxTracks; - enableMultiTrack : X_Audio_Sprite_enableMultiTrack, + // TODO + // Android4.x標準ブラウザ(Chrome系)でブラウザが隠れた場合に音が鳴り続ける問題、ビデオで解決できる? + //if( X_AudioSprite_needTouchAndroid && n === 1 ){ + // video = true; + //}; - create : function( setting ){ - // close() - if( X_Audio_Sprite_instance ){ - X_Audio_Sprite_instance.close(); - } else { - X_Audio_Sprite_instance = X_Class_override( new X.EventDispatcher(), X_Audio_Sprite_members ); + for( k in setting ){ + v = setting[ k ]; + if( X_Type_isArray( v ) && v !== urls ){ + v = X_Array_copy( v ); + for( i = v.length; i; ){ + --i; + if( i !== 2 ) v[ i ] = X_Audio_timeStringToNumber( v[ i ] ); + }; + if( v[ 2 ] ) bgms[ k ] = v; + presets[ k ] = v; }; - X_Audio_Sprite_instance.setup( setting ); - return X_Audio_Sprite_instance; - - } + }; + + X_Audio_startDetectionBackend( + X_Audio_BACKENDS[ 0 ], + X_AudioSprite, + X_Array_copy( urls ), + { + 'volume' : 0 <= volume && volume <= 1 ? volume : 1, + 'autoplay' : true, + 'startTime' : 0, + 'endTime' : X_AudioSprite_lengthSilence, + 'loop' : true + }); + + X_AudioSprite[ 'listenOnce' ]( [ X_EVENT_BACKEND_READY, X_EVENT_BACKEND_NONE ], X_AudioSprite_backendHandler ); + X_AudioSprite[ 'listenOnce' ]( X_EVENT_KILL_INSTANCE, X_AudioSprite_handleEvent ); + + X_AudioSprite_useVideo = video; + X_AudioSprite_numTracks = X_AudioSprite[ 'numTracks' ] = n; + + return X_AudioSprite; }; +X[ 'AudioSprite' ][ 'shouldUse' ] = X_AudioSprite_shouldUse; +X[ 'AudioSprite' ][ 'enableMultiTrack' ] = !X_AudioSprite_disableMultiTrack; + // 再生が終わっているもの、終わりかけのものを探す -// TODO 終わりかけのもの、と一番古いもの、どちらを再利用するか?または、再利用を待つ。 -function X_Audio_Sprite_getTrackEnded(){ - var tracks = X_Audio_Sprite_TEMP.tracks, - l = tracks.length, +// TODO 終わりかけのもの、と一番古いもの、どちらを再利用するか?これ以上に細かい実装を望む場合は X.AudioSprite は使わず自力で実装 +function X_AudioSprite_getTrackEnded(){ + var tracks = X_AudioSprite_TEMP.tracks, + l = X_AudioSprite_numTracks, i = 0, track, state, last = 1 / 0, _last, index; for( ; i < l; ++i ){ track = tracks[ i ]; - state = track.state(); + state = track.getState(); if( !state.playing ) return track; - if( track === X_Audio_Sprite_TEMP.bgmTrack ) continue; - if( state.currentTime <= X_Audio_Sprite_lengthSilence + X_Audio_Sprite_lengthDistance ) return track; + if( track === X_AudioSprite_TEMP.bgmTrack ) continue; + if( state.currentTime <= X_AudioSprite_lengthSilence + X_AudioSprite_lengthDistance ) return track; _last = state.endTime - state.currentTime; if( _last < last ){ last = _last; @@ -71,175 +137,91 @@ function X_Audio_Sprite_getTrackEnded(){ return tracks[ index ]; }; -/* - * { - * urls : [ 'xx.ogg', 'xx.mp3' ], - * numTracks : 3, - * useVideo : false, - * volume : 1, - * BGM_01 : [ '15.00', '45.500', true, '17.666', '50.999' ], - * BGM_02 : [ '56.00', '1:15.230', true ] - * } - */ - -X_Audio_Sprite_members = { - - setup : function( setting ){ - - var tracks = X_Audio_Sprite_TEMP.tracks, - bgms = X_Audio_Sprite_TEMP.BGMs, - presets = X_Audio_Sprite_TEMP.presets, - urls = setting[ 'urls' ], - n = setting[ 'numTracks' ] || 1, - video = setting[ 'useVideo' ], - option = { - volume : setting[ 'volume' ] || 0.5, - autoplay : false, - startTime : 0, - endTime : X_Audio_Sprite_lengthSilence, - loop : true - }, - k, i, v, track; - - n = n <= X_Audio_Sprite_maxTracks ? n : X_Audio_Sprite_maxTracks; - - video = video || ( 1 < n && X_Audio_Sprite_useVideoForMulti ); - - for( k in setting ){ - v = setting[ k ]; - if( X.Type.isArray( v ) && v !== urls){ - v = X.Object.cloneArray( v ); - for( i = v.length; i; ){ - --i; - if( i !== 2 ) v[ i ] = X_AudioWrapper_timeStringToNumber( v[ i ] ); - }; - if( v[ 2 ] ) bgms[ k ] = v; - presets[ k ] = v; - }; - }; - - for( i = 0; i < n; ++i ){ - if( i === 1 && X_Audio_Sprite_useVideoForMulti ){ - // TODO use