OSDN Git Service

Version 0.6.78, bugfix & starting jsdoc.
[pettanr/clientJs.git] / 0.6.x / js / 00_core / 05_XTimer.js
1 /**\r
2  * use X.Callback\r
3  */\r
4 \r
5 var _enterFrame =\r
6                 window.requestAnimationFrame ||\r
7                 window.webkitRequestAnimationFrame ||\r
8                 window.mozRequestAnimationFrame ||\r
9                 window.oRequestAnimationFrame ||\r
10                 window.msRequestAnimationFrame ||\r
11                 false,\r
12         _cancelFrame =\r
13                 window.cancelRequestAnimationFrame ||\r
14                 window.webkitCancelAnimationFrame ||\r
15                 window.webkitCancelRequestAnimationFrame ||\r
16                 window.mozCancelRequestAnimationFrame ||\r
17                 window.oCancelRequestAnimationFrame ||\r
18                 window.msCancelRequestAnimationFrame ||\r
19                 false;\r
20 \r
21 /*\r
22  * \r
23  * http://please-sleep.cou929.nu/script-yielding-with-setimmediate.html\r
24  * setImmediate での script yielding\r
25  * \r
26  * http://ie.microsoft.com/testdrive/Performance/setImmediateSorting/Default.html\r
27  * setImmediate API\r
28  * \r
29  * if( timer < 4ms ) useSetImmediate\r
30  * \r
31  *         if (window.msSetImmediate)\r
32         {\r
33             this.timer = msSetImmediate(function () { t.stepper(); });        \r
34         }\r
35         else if (window.MozSetImmediate)\r
36         {\r
37             this.timer = MozSetImmediate(function () { t.stepper(); });        \r
38         }\r
39         else if (window.WebkitSetImmediate) {\r
40             this.timer = WebkitSetImmediate(function () { t.stepper(); });\r
41         }\r
42         else if (window.OSetImmediate)\r
43         {\r
44             this.timer = OSetImmediate(function () { t.stepper(); });        \r
45         }\r
46  */\r
47 \r
48 X.Timer = {\r
49         INTERVAL_TIME  : 16,\r
50         TICKET_LIST    : [],\r
51         uid            : 0,\r
52         timerId        : 0,\r
53         endTime        : 0, // iOS\r
54         next           : 0,\r
55         busy           : false, // for Opera7\r
56         \r
57         REQ_FRAME_LIST : [],\r
58         requestID      : 0,\r
59         \r
60         _loop : function(){\r
61                 var next  = X.Timer.next,\r
62                         list  = X.Timer.TICKET_LIST,\r
63                         i     = 0,\r
64                         l     = list.length,\r
65                         start = X.getTime(),\r
66                         q, f, c, r;\r
67                 \r
68                 if( X.Timer.busy ){\r
69                         alert( 'busy!' );\r
70                 };\r
71                 \r
72                 X.Timer.busy = true;\r
73                 \r
74             for( ; i < l; ++i ){\r
75                 q = list[ i ];\r
76                         if( 0 < ( q.last -= next ) ) continue;\r
77                         c = q.count;\r
78                         \r
79                         if( q.k ){\r
80                                 q.a = [];\r
81                                 r = X.Callback._proxyCallback( q );\r
82                         } else {\r
83                                 r = q.f();\r
84                         };\r
85                         \r
86                         if( r & X.Callback.UN_LISTEN || c === 1 ){\r
87                                 list.splice( i, 1 );\r
88                                 --i;\r
89                                 --l;\r
90                                 continue;\r
91                         } else\r
92                         if( 1 < c ) --q.count;\r
93                         q.last = q.time;\r
94                         \r
95                         if( start < X.getTime() ){\r
96                                 console.log( '@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@' );\r
97                                 break;\r
98                         };\r
99             };\r
100             X.Timer.timerId = 0;\r
101             X.Timer.busy = false;\r
102             X.Timer._update();\r
103         },\r
104         _update : function(){\r
105                 var list = X.Timer.TICKET_LIST,\r
106                         i    = list.length,\r
107                         n;\r
108                 if( i === 0 ){\r
109                         X.Timer.timerId && window.clearTimeout( X.Timer.timerId );\r
110                         X.Timer.timerId = 0;\r
111                         return;\r
112                 };\r
113                 \r
114                 1 < i && list.sort( x_timer_compareQueue );\r
115                 \r
116             n = list[ i - 1 ].last;\r
117             \r
118             if( n < X.Timer.next || X.Timer.timerId === 0 ){\r
119                 X.Timer.timerId && window.clearTimeout( X.Timer.timerId );\r
120                 X.Timer.timerId = window.setTimeout( X.Timer._loop, X.Timer.INTERVAL_TIME * n );\r
121                 X.Timer.endTime = X.getTime() + X.Timer.INTERVAL_TIME * n; // iOS\r
122                 X.Timer.next = n;\r
123             };\r
124         },\r
125         \r
126         // ページを読み込んでからの時間\r
127         _onEnterFrame : function ( time ){\r
128                 var list = X.Timer.REQ_FRAME_LIST,\r
129                         l    = list.length,\r
130                         i    = 0, q;\r
131 \r
132                 time = time || X.getTime();\r
133                 // console.log( X.getTime() + ' , ' + time );\r
134             for( ; i < l; ++i ){\r
135                 q = list[ i ];\r
136                 \r
137                         if( q.k ){\r
138                                 q.a = [ time ];\r
139                                 X.Callback._proxyCallback( q );\r
140                         } else {\r
141                                 q( time );\r
142                         };\r
143             };\r
144 \r
145             list.splice( 0, l );\r
146             if( list.length ){\r
147                 X.Timer.requestID = _enterFrame ? _enterFrame( X.Timer._onEnterFrame ) : X.Timer.add( 0, 1, X.Timer._onEnterFrame );\r
148             };\r
149         },\r
150         \r
151         add : function( time, opt_count, args1, args2, args3 ){\r
152                 var list = X.Timer.TICKET_LIST,\r
153                         hash, obj;\r
154                 time = time < X.Timer.INTERVAL_TIME ? 1 : time / X.Timer.INTERVAL_TIME | 0; // 正の数で使える「Math.floor(x)」を「(x | 0)」に;\r
155                 \r
156                 if( !X.Type.isNumber( opt_count ) ){\r
157                         args3 = args2;\r
158                         args2 = args1;\r
159                         args1 = opt_count;\r
160                         opt_count = 0;\r
161                 };\r
162                 \r
163                 hash = X.Callback._classifyCallbackArgs( args1, args2, args3 );\r
164                 if( !hash ) return -1; // dev only\r
165                 \r
166                 if( !hash.k ) hash = { f : hash };\r
167                 hash.time  = time;\r
168                 hash.last  = time;\r
169                 hash.count = opt_count;\r
170                 hash.uid   = ++X.Timer.uid;\r
171                 list[ list.length ] = hash;\r
172                 \r
173             !X.Timer.busy && X.Timer._update();\r
174             return X.Timer.uid;\r
175         },\r
176         once : function( time, args1, args2, args3 ){\r
177                 return X.Timer.add( time, 1, args1, args2, args3 );\r
178         },\r
179         remove : function( uid ){\r
180                 var list = X.Timer.TICKET_LIST,\r
181                         i    = list.length,\r
182                         l    = i,\r
183                         f, q;\r
184                 for( ; i; ){\r
185                         // TODO\r
186                         // fire 中の cancel\r
187                         if( ( q = list[ --i ] ).uid === uid ){\r
188                                 list.splice( i, 1 );\r
189                                 !X.Timer.busy && ( /* q[ INDEX_COUNT ] <= next  || */ l === 1 ) && X.Timer._update();\r
190                                 break;\r
191                         };\r
192                 };\r
193         },\r
194         \r
195         requestFrame : _enterFrame ?\r
196                 (function( args1, args2, args3 ){\r
197                         var i = X.Timer.REQ_FRAME_LIST.length,\r
198                                 f;\r
199                         i === 0 && ( X.Timer.requestID = _enterFrame( X.Timer._onEnterFrame ) );\r
200                         f = X.Timer.REQ_FRAME_LIST[ i ] = X.Callback._classifyCallbackArgs( args1, args2, args3 );\r
201                         return f.uid = ++X.Timer.uid;\r
202                 }) :\r
203                 (function( args1, args2, args3 ){\r
204                         var i = X.Timer.REQ_FRAME_LIST.length,\r
205                                 f;\r
206                         i === 0 && ( X.Timer.requestID = X.Timer.add( 0, 1, X.Timer._onEnterFrame ) );\r
207                         f = X.Timer.REQ_FRAME_LIST[ i ] = X.Callback._classifyCallbackArgs( args1, args2, args3 );\r
208                         return f.uid = ++X.Timer.uid;\r
209                 }),\r
210         \r
211         cancelFrame : _cancelFrame ?\r
212                 (function( uid ){\r
213                         var list = X.Timer.REQ_FRAME_LIST,\r
214                                 l    = list.length,\r
215                                 i    = l,\r
216                                 f;\r
217                         for( ; i; ){\r
218                                 if( ( f = list[ --i ] ).uid < uid ) break;\r
219                                 if( f.uid === uid ){\r
220                                         // TODO\r
221                                         // fire 中の cancel\r
222                                         list.splice( i, 1 );\r
223                                         l === 1 && _cancelFrame( X.Timer.requestID );\r
224                                         break;\r
225                                 };\r
226                         };\r
227                 }) :\r
228                 (function( uid ){\r
229                         var list = X.Timer.REQ_FRAME_LIST,\r
230                                 l    = list.length,\r
231                                 i    = l,\r
232                                 f;\r
233                         for( ; i; ){\r
234                                 if( ( f = list[ --i ] ).uid < uid ) break;\r
235                                 if( f.uid === uid ){\r
236                                         list.splice( i, 1 );\r
237                                         l === 1 && X.Timer.remove( X.Timer.requestID );\r
238                                         break;\r
239                                 };\r
240                         };\r
241                 })\r
242         \r
243 };\r
244 \r
245 // 大きい -> 小さい\r
246 function x_timer_compareQueue( a, b ){\r
247         return a.last < b.last ? 1 : a.last === b.last ? 0 : -1;\r
248 };\r
249 \r
250 \r
251 // http://havelog.ayumusato.com/develop/javascript/e528-ios6_scrolling_timer_notcall.html\r
252 // iOS6 スクロール中のタイマー発火絡みのバグ備忘\r
253 if( X.UA.iOS ){\r
254         window.addEventListener( 'scroll', function(){\r
255                 if( X.Timer.timerId ){\r
256                         window.clearTimeout( X.Timer.timerId );\r
257                         X.Timer.timerId = window.setTimeout( X.Timer._loop, Math.max( 0, X.Timer.endTime - X.getTime() ) );\r
258                 };\r
259         });\r
260 };\r
261 \r
262 if( X.UA.IE < 5 || X.UA.MacIE ){\r
263         X.Timer[ '_ie_loop' ] = X.Timer._loop;\r
264         X.Timer._loop = 'X.Timer._ie_loop()';\r
265 };\r
266 \r
267 console.log( 'X.Core.Timer' );\r