jquery-datatables-rails 1.10.0 → 1.11.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/Readme.md +35 -22
- data/lib/jquery/datatables/rails/version.rb +1 -1
- data/vendor/assets/javascripts/dataTables/extras/AutoFill.js +820 -0
- data/vendor/assets/javascripts/dataTables/extras/ColReorder.js +956 -0
- data/vendor/assets/javascripts/dataTables/extras/ColVis.js +1013 -0
- data/vendor/assets/javascripts/dataTables/extras/FixedColumns.js +144 -130
- data/vendor/assets/javascripts/dataTables/extras/KeyTable.js +1111 -0
- data/vendor/assets/javascripts/dataTables/extras/Scroller.js +904 -0
- data/vendor/assets/javascripts/dataTables/extras/TableTools.js +2406 -0
- data/vendor/assets/javascripts/dataTables/extras/ZeroClipboard.js +367 -0
- data/vendor/assets/javascripts/dataTables/jquery.dataTables.api.fnFilterOnReturn.js +17 -0
- data/vendor/assets/javascripts/dataTables/jquery.dataTables.js +1711 -1686
- metadata +11 -2
@@ -0,0 +1,1111 @@
|
|
1
|
+
/*
|
2
|
+
* File: KeyTable.js
|
3
|
+
* Version: 1.1.7
|
4
|
+
* CVS: $Idj$
|
5
|
+
* Description: Keyboard navigation for HTML tables
|
6
|
+
* Author: Allan Jardine (www.sprymedia.co.uk)
|
7
|
+
* Created: Fri Mar 13 21:24:02 GMT 2009
|
8
|
+
* Modified: $Date$ by $Author$
|
9
|
+
* Language: Javascript
|
10
|
+
* License: GPL v2 or BSD 3 point style
|
11
|
+
* Project: Just a little bit of fun :-)
|
12
|
+
* Contact: www.sprymedia.co.uk/contact
|
13
|
+
*
|
14
|
+
* Copyright 2009-2011 Allan Jardine, all rights reserved.
|
15
|
+
*
|
16
|
+
* This source file is free software, under either the GPL v2 license or a
|
17
|
+
* BSD style license, available at:
|
18
|
+
* http://datatables.net/license_gpl2
|
19
|
+
* http://datatables.net/license_bsd
|
20
|
+
*/
|
21
|
+
|
22
|
+
|
23
|
+
function KeyTable ( oInit )
|
24
|
+
{
|
25
|
+
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
26
|
+
* API parameters
|
27
|
+
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
28
|
+
|
29
|
+
/*
|
30
|
+
* Variable: block
|
31
|
+
* Purpose: Flag whether or not KeyTable events should be processed
|
32
|
+
* Scope: KeyTable - public
|
33
|
+
*/
|
34
|
+
this.block = false;
|
35
|
+
|
36
|
+
/*
|
37
|
+
* Variable: event
|
38
|
+
* Purpose: Container for all event application methods
|
39
|
+
* Scope: KeyTable - public
|
40
|
+
* Notes: This object contains all the public methods for adding and removing events - these
|
41
|
+
* are dynamically added later on
|
42
|
+
*/
|
43
|
+
this.event = {
|
44
|
+
"remove": {}
|
45
|
+
};
|
46
|
+
|
47
|
+
|
48
|
+
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
49
|
+
* API methods
|
50
|
+
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
51
|
+
|
52
|
+
/*
|
53
|
+
* Function: fnGetCurrentPosition
|
54
|
+
* Purpose: Get the currently focused cell's position
|
55
|
+
* Returns: array int: [ x, y ]
|
56
|
+
* Inputs: void
|
57
|
+
*/
|
58
|
+
this.fnGetCurrentPosition = function ()
|
59
|
+
{
|
60
|
+
return [ _iOldX, _iOldY ];
|
61
|
+
};
|
62
|
+
|
63
|
+
|
64
|
+
/*
|
65
|
+
* Function: fnGetCurrentData
|
66
|
+
* Purpose: Get the currently focused cell's data (innerHTML)
|
67
|
+
* Returns: string: - data requested
|
68
|
+
* Inputs: void
|
69
|
+
*/
|
70
|
+
this.fnGetCurrentData = function ()
|
71
|
+
{
|
72
|
+
return _nOldFocus.innerHTML;
|
73
|
+
};
|
74
|
+
|
75
|
+
|
76
|
+
/*
|
77
|
+
* Function: fnGetCurrentTD
|
78
|
+
* Purpose: Get the currently focused cell
|
79
|
+
* Returns: node: - focused element
|
80
|
+
* Inputs: void
|
81
|
+
*/
|
82
|
+
this.fnGetCurrentTD = function ()
|
83
|
+
{
|
84
|
+
return _nOldFocus;
|
85
|
+
};
|
86
|
+
|
87
|
+
|
88
|
+
/*
|
89
|
+
* Function: fnSetPosition
|
90
|
+
* Purpose: Set the position of the focused cell
|
91
|
+
* Returns: -
|
92
|
+
* Inputs: int:x - x coordinate
|
93
|
+
* int:y - y coordinate
|
94
|
+
* Notes: Thanks to Rohan Daxini for the basis of this function
|
95
|
+
*/
|
96
|
+
this.fnSetPosition = function( x, y )
|
97
|
+
{
|
98
|
+
if ( typeof x == 'object' && x.nodeName )
|
99
|
+
{
|
100
|
+
_fnSetFocus( x );
|
101
|
+
}
|
102
|
+
else
|
103
|
+
{
|
104
|
+
_fnSetFocus( _fnCellFromCoords(x, y) );
|
105
|
+
}
|
106
|
+
};
|
107
|
+
|
108
|
+
|
109
|
+
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
110
|
+
* Private parameters
|
111
|
+
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
112
|
+
|
113
|
+
/*
|
114
|
+
* Variable: _nBody
|
115
|
+
* Purpose: Body node of the table - cached for renference
|
116
|
+
* Scope: KeyTable - private
|
117
|
+
*/
|
118
|
+
var _nBody = null;
|
119
|
+
|
120
|
+
/*
|
121
|
+
* Variable:
|
122
|
+
* Purpose:
|
123
|
+
* Scope: KeyTable - private
|
124
|
+
*/
|
125
|
+
var _nOldFocus = null;
|
126
|
+
|
127
|
+
/*
|
128
|
+
* Variable: _iOldX and _iOldY
|
129
|
+
* Purpose: X and Y coords of the old elemet that was focused on
|
130
|
+
* Scope: KeyTable - private
|
131
|
+
*/
|
132
|
+
var _iOldX = null;
|
133
|
+
var _iOldY = null;
|
134
|
+
|
135
|
+
/*
|
136
|
+
* Variable: _that
|
137
|
+
* Purpose: Scope saving for 'this' after a jQuery event
|
138
|
+
* Scope: KeyTable - private
|
139
|
+
*/
|
140
|
+
var _that = null;
|
141
|
+
|
142
|
+
/*
|
143
|
+
* Variable: sFocusClass
|
144
|
+
* Purpose: Class that should be used for focusing on a cell
|
145
|
+
* Scope: KeyTable - private
|
146
|
+
*/
|
147
|
+
var _sFocusClass = "focus";
|
148
|
+
|
149
|
+
/*
|
150
|
+
* Variable: _bKeyCapture
|
151
|
+
* Purpose: Flag for should KeyTable capture key events or not
|
152
|
+
* Scope: KeyTable - private
|
153
|
+
*/
|
154
|
+
var _bKeyCapture = false;
|
155
|
+
|
156
|
+
/*
|
157
|
+
* Variable: _oaoEvents
|
158
|
+
* Purpose: Event cache object, one array for each supported event for speed of searching
|
159
|
+
* Scope: KeyTable - private
|
160
|
+
*/
|
161
|
+
var _oaoEvents = {
|
162
|
+
"action": [],
|
163
|
+
"esc": [],
|
164
|
+
"focus": [],
|
165
|
+
"blur": []
|
166
|
+
};
|
167
|
+
|
168
|
+
/*
|
169
|
+
* Variable: _oDatatable
|
170
|
+
* Purpose: DataTables object for if we are actually using a DataTables table
|
171
|
+
* Scope: KeyTable - private
|
172
|
+
*/
|
173
|
+
var _oDatatable = null;
|
174
|
+
|
175
|
+
var _bForm;
|
176
|
+
var _nInput;
|
177
|
+
var _bInputFocused = false;
|
178
|
+
|
179
|
+
|
180
|
+
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
181
|
+
* Private methods
|
182
|
+
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
183
|
+
|
184
|
+
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
185
|
+
* Key table events
|
186
|
+
*/
|
187
|
+
|
188
|
+
/*
|
189
|
+
* Function: _fnEventAddTemplate
|
190
|
+
* Purpose: Create a function (with closure for sKey) event addition API
|
191
|
+
* Returns: function: - template function
|
192
|
+
* Inputs: string:sKey - type of event to detect
|
193
|
+
*/
|
194
|
+
function _fnEventAddTemplate( sKey )
|
195
|
+
{
|
196
|
+
/*
|
197
|
+
* Function: -
|
198
|
+
* Purpose: API function for adding event to cache
|
199
|
+
* Returns: -
|
200
|
+
* Inputs: 1. node:x - target node to add event for
|
201
|
+
* 2. function:y - callback function to apply
|
202
|
+
* or
|
203
|
+
* 1. int:x - x coord. of target cell (can be null for live events)
|
204
|
+
* 2. int:y - y coord. of target cell (can be null for live events)
|
205
|
+
* 3. function:z - callback function to apply
|
206
|
+
* Notes: This function is (interally) overloaded (in as much as javascript allows for
|
207
|
+
* that) - the target cell can be given by either node or coords.
|
208
|
+
*/
|
209
|
+
return function ( x, y, z ) {
|
210
|
+
if ( (x===null || typeof x == "number") &&
|
211
|
+
(y===null || typeof y == "number") &&
|
212
|
+
typeof z == "function" )
|
213
|
+
{
|
214
|
+
_fnEventAdd( sKey, x, y, z );
|
215
|
+
}
|
216
|
+
else if ( typeof x == "object" && typeof y == "function" )
|
217
|
+
{
|
218
|
+
var aCoords = _fnCoordsFromCell( x );
|
219
|
+
_fnEventAdd( sKey, aCoords[0], aCoords[1], y );
|
220
|
+
}
|
221
|
+
else
|
222
|
+
{
|
223
|
+
alert( "Unhandable event type was added: x" +x+ " y:" +y+ " z:" +z );
|
224
|
+
}
|
225
|
+
};
|
226
|
+
}
|
227
|
+
|
228
|
+
|
229
|
+
/*
|
230
|
+
* Function: _fnEventRemoveTemplate
|
231
|
+
* Purpose: Create a function (with closure for sKey) event removal API
|
232
|
+
* Returns: function: - template function
|
233
|
+
* Inputs: string:sKey - type of event to detect
|
234
|
+
*/
|
235
|
+
function _fnEventRemoveTemplate( sKey )
|
236
|
+
{
|
237
|
+
/*
|
238
|
+
* Function: -
|
239
|
+
* Purpose: API function for removing event from cache
|
240
|
+
* Returns: int: - number of events removed
|
241
|
+
* Inputs: 1. node:x - target node to remove event from
|
242
|
+
* 2. function:y - callback function to apply
|
243
|
+
* or
|
244
|
+
* 1. int:x - x coord. of target cell (can be null for live events)
|
245
|
+
* 2. int:y - y coord. of target cell (can be null for live events)
|
246
|
+
* 3. function:z - callback function to remove - optional
|
247
|
+
* Notes: This function is (interally) overloaded (in as much as javascript allows for
|
248
|
+
* that) - the target cell can be given by either node or coords and the function
|
249
|
+
* to remove is optional
|
250
|
+
*/
|
251
|
+
return function ( x, y, z ) {
|
252
|
+
if ( (x===null || typeof arguments[0] == "number") &&
|
253
|
+
(y===null || typeof arguments[1] == "number" ) )
|
254
|
+
{
|
255
|
+
if ( typeof arguments[2] == "function" )
|
256
|
+
{
|
257
|
+
_fnEventRemove( sKey, x, y, z );
|
258
|
+
}
|
259
|
+
else
|
260
|
+
{
|
261
|
+
_fnEventRemove( sKey, x, y );
|
262
|
+
}
|
263
|
+
}
|
264
|
+
else if ( typeof arguments[0] == "object" )
|
265
|
+
{
|
266
|
+
var aCoords = _fnCoordsFromCell( x );
|
267
|
+
if ( typeof arguments[1] == "function" )
|
268
|
+
{
|
269
|
+
_fnEventRemove( sKey, aCoords[0], aCoords[1], y );
|
270
|
+
}
|
271
|
+
else
|
272
|
+
{
|
273
|
+
_fnEventRemove( sKey, aCoords[0], aCoords[1] );
|
274
|
+
}
|
275
|
+
}
|
276
|
+
else
|
277
|
+
{
|
278
|
+
alert( "Unhandable event type was removed: x" +x+ " y:" +y+ " z:" +z );
|
279
|
+
}
|
280
|
+
};
|
281
|
+
}
|
282
|
+
|
283
|
+
/* Use the template functions to add the event API functions */
|
284
|
+
for ( var sKey in _oaoEvents )
|
285
|
+
{
|
286
|
+
if ( sKey )
|
287
|
+
{
|
288
|
+
this.event[sKey] = _fnEventAddTemplate( sKey );
|
289
|
+
this.event.remove[sKey] = _fnEventRemoveTemplate( sKey );
|
290
|
+
}
|
291
|
+
}
|
292
|
+
|
293
|
+
|
294
|
+
/*
|
295
|
+
* Function: _fnEventAdd
|
296
|
+
* Purpose: Add an event to the internal cache
|
297
|
+
* Returns: -
|
298
|
+
* Inputs: string:sType - type of event to add, given by the available elements in _oaoEvents
|
299
|
+
* int:x - x-coords to add event to - can be null for "blanket" event
|
300
|
+
* int:y - y-coords to add event to - can be null for "blanket" event
|
301
|
+
* function:fn - callback function for when triggered
|
302
|
+
*/
|
303
|
+
function _fnEventAdd( sType, x, y, fn )
|
304
|
+
{
|
305
|
+
_oaoEvents[sType].push( {
|
306
|
+
"x": x,
|
307
|
+
"y": y,
|
308
|
+
"fn": fn
|
309
|
+
} );
|
310
|
+
}
|
311
|
+
|
312
|
+
|
313
|
+
/*
|
314
|
+
* Function: _fnEventRemove
|
315
|
+
* Purpose: Remove an event from the event cache
|
316
|
+
* Returns: int: - number of matching events removed
|
317
|
+
* Inputs: string:sType - type of event to look for
|
318
|
+
* node:nTarget - target table cell
|
319
|
+
* function:fn - optional - remove this function. If not given all handlers of this
|
320
|
+
* type will be removed
|
321
|
+
*/
|
322
|
+
function _fnEventRemove( sType, x, y, fn )
|
323
|
+
{
|
324
|
+
var iCorrector = 0;
|
325
|
+
|
326
|
+
for ( var i=0, iLen=_oaoEvents[sType].length ; i<iLen-iCorrector ; i++ )
|
327
|
+
{
|
328
|
+
if ( typeof fn != 'undefined' )
|
329
|
+
{
|
330
|
+
if ( _oaoEvents[sType][i-iCorrector].x == x &&
|
331
|
+
_oaoEvents[sType][i-iCorrector].y == y &&
|
332
|
+
_oaoEvents[sType][i-iCorrector].fn == fn )
|
333
|
+
{
|
334
|
+
_oaoEvents[sType].splice( i-iCorrector, 1 );
|
335
|
+
iCorrector++;
|
336
|
+
}
|
337
|
+
}
|
338
|
+
else
|
339
|
+
{
|
340
|
+
if ( _oaoEvents[sType][i-iCorrector].x == x &&
|
341
|
+
_oaoEvents[sType][i-iCorrector].y == y )
|
342
|
+
{
|
343
|
+
_oaoEvents[sType].splice( i, 1 );
|
344
|
+
return 1;
|
345
|
+
}
|
346
|
+
}
|
347
|
+
}
|
348
|
+
return iCorrector;
|
349
|
+
}
|
350
|
+
|
351
|
+
|
352
|
+
/*
|
353
|
+
* Function: _fnEventFire
|
354
|
+
* Purpose: Look thought the events cache and fire off the event of interest
|
355
|
+
* Returns: int:iFired - number of events fired
|
356
|
+
* Inputs: string:sType - type of event to look for
|
357
|
+
* int:x - x coord of cell
|
358
|
+
* int:y - y coord of ell
|
359
|
+
* Notes: It might be more efficient to return after the first event has been tirggered,
|
360
|
+
* but that would mean that only one function of a particular type can be
|
361
|
+
* subscribed to a particular node.
|
362
|
+
*/
|
363
|
+
function _fnEventFire ( sType, x, y )
|
364
|
+
{
|
365
|
+
var iFired = 0;
|
366
|
+
var aEvents = _oaoEvents[sType];
|
367
|
+
for ( var i=0 ; i<aEvents.length ; i++ )
|
368
|
+
{
|
369
|
+
if ( (aEvents[i].x == x && aEvents[i].y == y ) ||
|
370
|
+
(aEvents[i].x === null && aEvents[i].y == y ) ||
|
371
|
+
(aEvents[i].x == x && aEvents[i].y === null ) ||
|
372
|
+
(aEvents[i].x === null && aEvents[i].y === null )
|
373
|
+
)
|
374
|
+
{
|
375
|
+
aEvents[i].fn( _fnCellFromCoords(x,y), x, y );
|
376
|
+
iFired++;
|
377
|
+
}
|
378
|
+
}
|
379
|
+
return iFired;
|
380
|
+
}
|
381
|
+
|
382
|
+
|
383
|
+
|
384
|
+
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
385
|
+
* Focus functions
|
386
|
+
*/
|
387
|
+
|
388
|
+
/*
|
389
|
+
* Function: _fnSetFocus
|
390
|
+
* Purpose: Set focus on a node, and remove from an old node if needed
|
391
|
+
* Returns: -
|
392
|
+
* Inputs: node:nTarget - node we want to focus on
|
393
|
+
* bool:bAutoScroll - optional - should we scroll the view port to the display
|
394
|
+
*/
|
395
|
+
function _fnSetFocus( nTarget, bAutoScroll )
|
396
|
+
{
|
397
|
+
/* If node already has focus, just ignore this call */
|
398
|
+
if ( _nOldFocus == nTarget )
|
399
|
+
{
|
400
|
+
return;
|
401
|
+
}
|
402
|
+
|
403
|
+
if ( typeof bAutoScroll == 'undefined' )
|
404
|
+
{
|
405
|
+
bAutoScroll = true;
|
406
|
+
}
|
407
|
+
|
408
|
+
/* Remove old focus (with blur event if needed) */
|
409
|
+
if ( _nOldFocus !== null )
|
410
|
+
{
|
411
|
+
_fnRemoveFocus( _nOldFocus );
|
412
|
+
}
|
413
|
+
|
414
|
+
/* Add the new class to highlight the focused cell */
|
415
|
+
jQuery(nTarget).addClass( _sFocusClass );
|
416
|
+
jQuery(nTarget).parent().addClass( _sFocusClass );
|
417
|
+
|
418
|
+
/* If it's a DataTable then we need to jump the paging to the relevant page */
|
419
|
+
var oSettings;
|
420
|
+
if ( _oDatatable )
|
421
|
+
{
|
422
|
+
oSettings = _oDatatable.fnSettings();
|
423
|
+
var iRow = _fnFindDtCell( nTarget )[1];
|
424
|
+
var bKeyCaptureCache = _bKeyCapture;
|
425
|
+
|
426
|
+
/* Page forwards */
|
427
|
+
while ( iRow >= oSettings.fnDisplayEnd() )
|
428
|
+
{
|
429
|
+
if ( oSettings._iDisplayLength >= 0 )
|
430
|
+
{
|
431
|
+
/* Make sure we are not over running the display array */
|
432
|
+
if ( oSettings._iDisplayStart + oSettings._iDisplayLength < oSettings.fnRecordsDisplay() )
|
433
|
+
{
|
434
|
+
oSettings._iDisplayStart += oSettings._iDisplayLength;
|
435
|
+
}
|
436
|
+
}
|
437
|
+
else
|
438
|
+
{
|
439
|
+
oSettings._iDisplayStart = 0;
|
440
|
+
}
|
441
|
+
_oDatatable.oApi._fnCalculateEnd( oSettings );
|
442
|
+
}
|
443
|
+
|
444
|
+
/* Page backwards */
|
445
|
+
while ( iRow < oSettings._iDisplayStart )
|
446
|
+
{
|
447
|
+
oSettings._iDisplayStart = oSettings._iDisplayLength>=0 ?
|
448
|
+
oSettings._iDisplayStart - oSettings._iDisplayLength :
|
449
|
+
0;
|
450
|
+
|
451
|
+
if ( oSettings._iDisplayStart < 0 )
|
452
|
+
{
|
453
|
+
oSettings._iDisplayStart = 0;
|
454
|
+
}
|
455
|
+
_oDatatable.oApi._fnCalculateEnd( oSettings );
|
456
|
+
}
|
457
|
+
|
458
|
+
/* Re-draw the table */
|
459
|
+
_oDatatable.oApi._fnDraw( oSettings );
|
460
|
+
|
461
|
+
/* Restore the key capture */
|
462
|
+
_bKeyCapture = bKeyCaptureCache;
|
463
|
+
}
|
464
|
+
|
465
|
+
/* Cache the information that we are interested in */
|
466
|
+
var aNewPos = _fnCoordsFromCell( nTarget );
|
467
|
+
_nOldFocus = nTarget;
|
468
|
+
_iOldX = aNewPos[0];
|
469
|
+
_iOldY = aNewPos[1];
|
470
|
+
|
471
|
+
var iViewportHeight, iViewportWidth, iScrollTop, iScrollLeft, iHeight, iWidth, aiPos;
|
472
|
+
if ( bAutoScroll )
|
473
|
+
{
|
474
|
+
/* Scroll the viewport such that the new cell is fully visible in the rendered window */
|
475
|
+
iViewportHeight = document.documentElement.clientHeight;
|
476
|
+
iViewportWidth = document.documentElement.clientWidth;
|
477
|
+
iScrollTop = document.body.scrollTop || document.documentElement.scrollTop;
|
478
|
+
iScrollLeft = document.body.scrollLeft || document.documentElement.scrollLeft;
|
479
|
+
iHeight = nTarget.offsetHeight;
|
480
|
+
iWidth = nTarget.offsetWidth;
|
481
|
+
aiPos = _fnGetPos( nTarget );
|
482
|
+
|
483
|
+
/* Take account of scrolling in DataTables 1.7 - remove scrolling since that would add to
|
484
|
+
* the positioning calculation
|
485
|
+
*/
|
486
|
+
if ( _oDatatable && typeof oSettings.oScroll != 'undefined' &&
|
487
|
+
(oSettings.oScroll.sX !== "" || oSettings.oScroll.sY !== "") )
|
488
|
+
{
|
489
|
+
aiPos[1] -= $(oSettings.nTable.parentNode).scrollTop();
|
490
|
+
aiPos[0] -= $(oSettings.nTable.parentNode).scrollLeft();
|
491
|
+
}
|
492
|
+
|
493
|
+
/* Correct viewport positioning for vertical scrolling */
|
494
|
+
if ( aiPos[1]+iHeight > iScrollTop+iViewportHeight )
|
495
|
+
{
|
496
|
+
/* Displayed element if off the bottom of the viewport */
|
497
|
+
_fnSetScrollTop( aiPos[1]+iHeight - iViewportHeight );
|
498
|
+
}
|
499
|
+
else if ( aiPos[1] < iScrollTop )
|
500
|
+
{
|
501
|
+
/* Displayed element if off the top of the viewport */
|
502
|
+
_fnSetScrollTop( aiPos[1] );
|
503
|
+
}
|
504
|
+
|
505
|
+
/* Correct viewport positioning for horizontal scrolling */
|
506
|
+
if ( aiPos[0]+iWidth > iScrollLeft+iViewportWidth )
|
507
|
+
{
|
508
|
+
/* Displayed element is off the bottom of the viewport */
|
509
|
+
_fnSetScrollLeft( aiPos[0]+iWidth - iViewportWidth );
|
510
|
+
}
|
511
|
+
else if ( aiPos[0] < iScrollLeft )
|
512
|
+
{
|
513
|
+
/* Displayed element if off the Left of the viewport */
|
514
|
+
_fnSetScrollLeft( aiPos[0] );
|
515
|
+
}
|
516
|
+
}
|
517
|
+
|
518
|
+
/* Take account of scrolling in DataTables 1.7 */
|
519
|
+
if ( _oDatatable && typeof oSettings.oScroll != 'undefined' &&
|
520
|
+
(oSettings.oScroll.sX !== "" || oSettings.oScroll.sY !== "") )
|
521
|
+
{
|
522
|
+
var dtScrollBody = oSettings.nTable.parentNode;
|
523
|
+
iViewportHeight = dtScrollBody.clientHeight;
|
524
|
+
iViewportWidth = dtScrollBody.clientWidth;
|
525
|
+
iScrollTop = dtScrollBody.scrollTop;
|
526
|
+
iScrollLeft = dtScrollBody.scrollLeft;
|
527
|
+
iHeight = nTarget.offsetHeight;
|
528
|
+
iWidth = nTarget.offsetWidth;
|
529
|
+
|
530
|
+
/* Correct for vertical scrolling */
|
531
|
+
if ( nTarget.offsetTop + iHeight > iViewportHeight+iScrollTop )
|
532
|
+
{
|
533
|
+
dtScrollBody.scrollTop = (nTarget.offsetTop + iHeight) - iViewportHeight;
|
534
|
+
}
|
535
|
+
else if ( nTarget.offsetTop < iScrollTop )
|
536
|
+
{
|
537
|
+
dtScrollBody.scrollTop = nTarget.offsetTop;
|
538
|
+
}
|
539
|
+
|
540
|
+
/* Correct for horizontal scrolling */
|
541
|
+
if ( nTarget.offsetLeft + iWidth > iViewportWidth+iScrollLeft )
|
542
|
+
{
|
543
|
+
dtScrollBody.scrollLeft = (nTarget.offsetLeft + iWidth) - iViewportWidth;
|
544
|
+
}
|
545
|
+
else if ( nTarget.offsetLeft < iScrollLeft )
|
546
|
+
{
|
547
|
+
dtScrollBody.scrollLeft = nTarget.offsetLeft;
|
548
|
+
}
|
549
|
+
}
|
550
|
+
|
551
|
+
/* Focused - so we want to capture the keys */
|
552
|
+
_fnCaptureKeys();
|
553
|
+
|
554
|
+
/* Fire of the focus event if there is one */
|
555
|
+
_fnEventFire( "focus", _iOldX, _iOldY );
|
556
|
+
}
|
557
|
+
|
558
|
+
|
559
|
+
/*
|
560
|
+
* Function: _fnBlur
|
561
|
+
* Purpose: Blur focus from the whole table
|
562
|
+
* Returns: -
|
563
|
+
* Inputs: -
|
564
|
+
*/
|
565
|
+
function _fnBlur()
|
566
|
+
{
|
567
|
+
_fnRemoveFocus( _nOldFocus );
|
568
|
+
_iOldX = null;
|
569
|
+
_iOldY = null;
|
570
|
+
_nOldFocus = null;
|
571
|
+
_fnReleaseKeys();
|
572
|
+
}
|
573
|
+
|
574
|
+
|
575
|
+
/*
|
576
|
+
* Function: _fnRemoveFocus
|
577
|
+
* Purpose: Remove focus from a cell and fire any blur events which are attached
|
578
|
+
* Returns: -
|
579
|
+
* Inputs: node:nTarget - cell of interest
|
580
|
+
*/
|
581
|
+
function _fnRemoveFocus( nTarget )
|
582
|
+
{
|
583
|
+
jQuery(nTarget).removeClass( _sFocusClass );
|
584
|
+
jQuery(nTarget).parent().removeClass( _sFocusClass );
|
585
|
+
_fnEventFire( "blur", _iOldX, _iOldY );
|
586
|
+
}
|
587
|
+
|
588
|
+
|
589
|
+
/*
|
590
|
+
* Function: _fnClick
|
591
|
+
* Purpose: Focus on the element that has been clicked on by the user
|
592
|
+
* Returns: -
|
593
|
+
* Inputs: event:e - click event
|
594
|
+
*/
|
595
|
+
function _fnClick ( e )
|
596
|
+
{
|
597
|
+
var nTarget = this;
|
598
|
+
while ( nTarget.nodeName != "TD" )
|
599
|
+
{
|
600
|
+
nTarget = nTarget.parentNode;
|
601
|
+
}
|
602
|
+
|
603
|
+
_fnSetFocus( nTarget );
|
604
|
+
_fnCaptureKeys();
|
605
|
+
}
|
606
|
+
|
607
|
+
|
608
|
+
|
609
|
+
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
610
|
+
* Key events
|
611
|
+
*/
|
612
|
+
|
613
|
+
/*
|
614
|
+
* Function: _fnKey
|
615
|
+
* Purpose: Deal with a key events, be it moving the focus or return etc.
|
616
|
+
* Returns: bool: - allow browser default action
|
617
|
+
* Inputs: event:e - key event
|
618
|
+
*/
|
619
|
+
function _fnKey ( e )
|
620
|
+
{
|
621
|
+
/* If user or system has blocked KeyTable from doing anything, just ignore this event */
|
622
|
+
if ( _that.block || !_bKeyCapture )
|
623
|
+
{
|
624
|
+
return true;
|
625
|
+
}
|
626
|
+
|
627
|
+
/* If a modifier key is pressed (exapct shift), ignore the event */
|
628
|
+
if ( e.metaKey || e.altKey || e.ctrlKey )
|
629
|
+
{
|
630
|
+
return true;
|
631
|
+
}
|
632
|
+
var
|
633
|
+
x, y,
|
634
|
+
iTableWidth = _nBody.getElementsByTagName('tr')[0].getElementsByTagName('td').length,
|
635
|
+
iTableHeight;
|
636
|
+
|
637
|
+
/* Get table height and width - done here so as to be dynamic (if table is updated) */
|
638
|
+
if ( _oDatatable )
|
639
|
+
{
|
640
|
+
/*
|
641
|
+
* Locate the current node in the DataTable overriding the old positions - the reason for
|
642
|
+
* is is that there might have been some DataTables interaction between the last focus and
|
643
|
+
* now
|
644
|
+
*/
|
645
|
+
var oSettings = _oDatatable.fnSettings();
|
646
|
+
iTableHeight = oSettings.aiDisplay.length;
|
647
|
+
|
648
|
+
var aDtPos = _fnFindDtCell( _nOldFocus );
|
649
|
+
if ( aDtPos === null )
|
650
|
+
{
|
651
|
+
/* If the table has been updated such that the focused cell can't be seen - do nothing */
|
652
|
+
return;
|
653
|
+
}
|
654
|
+
_iOldX = aDtPos[ 0 ];
|
655
|
+
_iOldY = aDtPos[ 1 ];
|
656
|
+
}
|
657
|
+
else
|
658
|
+
{
|
659
|
+
iTableHeight = _nBody.getElementsByTagName('tr').length;
|
660
|
+
}
|
661
|
+
|
662
|
+
/* Capture shift+tab to match the left arrow key */
|
663
|
+
var iKey = (e.keyCode == 9 && e.shiftKey) ? -1 : e.keyCode;
|
664
|
+
|
665
|
+
switch( iKey )
|
666
|
+
{
|
667
|
+
case 13: /* return */
|
668
|
+
e.preventDefault();
|
669
|
+
e.stopPropagation();
|
670
|
+
_fnEventFire( "action", _iOldX, _iOldY );
|
671
|
+
return true;
|
672
|
+
|
673
|
+
case 27: /* esc */
|
674
|
+
if ( !_fnEventFire( "esc", _iOldX, _iOldY ) )
|
675
|
+
{
|
676
|
+
/* Only lose focus if there isn't an escape handler on the cell */
|
677
|
+
_fnBlur();
|
678
|
+
return;
|
679
|
+
}
|
680
|
+
x = _iOldX;
|
681
|
+
y = _iOldY;
|
682
|
+
break;
|
683
|
+
|
684
|
+
case -1:
|
685
|
+
case 37: /* left arrow */
|
686
|
+
if ( _iOldX > 0 ) {
|
687
|
+
x = _iOldX - 1;
|
688
|
+
y = _iOldY;
|
689
|
+
} else if ( _iOldY > 0 ) {
|
690
|
+
x = iTableWidth-1;
|
691
|
+
y = _iOldY - 1;
|
692
|
+
} else {
|
693
|
+
/* at start of table */
|
694
|
+
if ( iKey == -1 && _bForm )
|
695
|
+
{
|
696
|
+
/* If we are in a form, return focus to the 'input' element such that tabbing will
|
697
|
+
* follow correctly in the browser
|
698
|
+
*/
|
699
|
+
_bInputFocused = true;
|
700
|
+
_nInput.focus();
|
701
|
+
|
702
|
+
/* This timeout is a little nasty - but IE appears to have some asyhnc behaviour for
|
703
|
+
* focus
|
704
|
+
*/
|
705
|
+
setTimeout( function(){ _bInputFocused = false; }, 0 );
|
706
|
+
_bKeyCapture = false;
|
707
|
+
_fnBlur();
|
708
|
+
return true;
|
709
|
+
}
|
710
|
+
else
|
711
|
+
{
|
712
|
+
return false;
|
713
|
+
}
|
714
|
+
}
|
715
|
+
break;
|
716
|
+
|
717
|
+
case 38: /* up arrow */
|
718
|
+
if ( _iOldY > 0 ) {
|
719
|
+
x = _iOldX;
|
720
|
+
y = _iOldY - 1;
|
721
|
+
} else {
|
722
|
+
return false;
|
723
|
+
}
|
724
|
+
break;
|
725
|
+
|
726
|
+
case 9: /* tab */
|
727
|
+
case 39: /* right arrow */
|
728
|
+
if ( _iOldX < iTableWidth-1 ) {
|
729
|
+
x = _iOldX + 1;
|
730
|
+
y = _iOldY;
|
731
|
+
} else if ( _iOldY < iTableHeight-1 ) {
|
732
|
+
x = 0;
|
733
|
+
y = _iOldY + 1;
|
734
|
+
} else {
|
735
|
+
/* at end of table */
|
736
|
+
if ( iKey == 9 && _bForm )
|
737
|
+
{
|
738
|
+
/* If we are in a form, return focus to the 'input' element such that tabbing will
|
739
|
+
* follow correctly in the browser
|
740
|
+
*/
|
741
|
+
_bInputFocused = true;
|
742
|
+
_nInput.focus();
|
743
|
+
|
744
|
+
/* This timeout is a little nasty - but IE appears to have some asyhnc behaviour for
|
745
|
+
* focus
|
746
|
+
*/
|
747
|
+
setTimeout( function(){ _bInputFocused = false; }, 0 );
|
748
|
+
_bKeyCapture = false;
|
749
|
+
_fnBlur();
|
750
|
+
return true;
|
751
|
+
}
|
752
|
+
else
|
753
|
+
{
|
754
|
+
return false;
|
755
|
+
}
|
756
|
+
}
|
757
|
+
break;
|
758
|
+
|
759
|
+
case 40: /* down arrow */
|
760
|
+
if ( _iOldY < iTableHeight-1 ) {
|
761
|
+
x = _iOldX;
|
762
|
+
y = _iOldY + 1;
|
763
|
+
} else {
|
764
|
+
return false;
|
765
|
+
}
|
766
|
+
break;
|
767
|
+
|
768
|
+
default: /* Nothing we are interested in */
|
769
|
+
return true;
|
770
|
+
}
|
771
|
+
|
772
|
+
_fnSetFocus( _fnCellFromCoords(x, y) );
|
773
|
+
return false;
|
774
|
+
}
|
775
|
+
|
776
|
+
|
777
|
+
/*
|
778
|
+
* Function: _fnCaptureKeys
|
779
|
+
* Purpose: Start capturing key events for this table
|
780
|
+
* Returns: -
|
781
|
+
* Inputs: -
|
782
|
+
*/
|
783
|
+
function _fnCaptureKeys( )
|
784
|
+
{
|
785
|
+
if ( !_bKeyCapture )
|
786
|
+
{
|
787
|
+
_bKeyCapture = true;
|
788
|
+
}
|
789
|
+
}
|
790
|
+
|
791
|
+
|
792
|
+
/*
|
793
|
+
* Function: _fnReleaseKeys
|
794
|
+
* Purpose: Stop capturing key events for this table
|
795
|
+
* Returns: -
|
796
|
+
* Inputs: -
|
797
|
+
*/
|
798
|
+
function _fnReleaseKeys( )
|
799
|
+
{
|
800
|
+
_bKeyCapture = false;
|
801
|
+
}
|
802
|
+
|
803
|
+
|
804
|
+
|
805
|
+
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
806
|
+
* Support functions
|
807
|
+
*/
|
808
|
+
|
809
|
+
/*
|
810
|
+
* Function: _fnCellFromCoords
|
811
|
+
* Purpose: Calulate the target TD cell from x and y coordinates
|
812
|
+
* Returns: node: - TD target
|
813
|
+
* Inputs: int:x - x coordinate
|
814
|
+
* int:y - y coordinate
|
815
|
+
*/
|
816
|
+
function _fnCellFromCoords( x, y )
|
817
|
+
{
|
818
|
+
if ( _oDatatable )
|
819
|
+
{
|
820
|
+
var oSettings = _oDatatable.fnSettings();
|
821
|
+
if ( typeof oSettings.aoData[ oSettings.aiDisplay[ y ] ] != 'undefined' )
|
822
|
+
{
|
823
|
+
return oSettings.aoData[ oSettings.aiDisplay[ y ] ].nTr.getElementsByTagName('td')[x];
|
824
|
+
}
|
825
|
+
else
|
826
|
+
{
|
827
|
+
return null;
|
828
|
+
}
|
829
|
+
}
|
830
|
+
else
|
831
|
+
{
|
832
|
+
return jQuery('tr:eq('+y+')>td:eq('+x+')', _nBody )[0];
|
833
|
+
}
|
834
|
+
}
|
835
|
+
|
836
|
+
|
837
|
+
/*
|
838
|
+
* Function: _fnCoordsFromCell
|
839
|
+
* Purpose: Calculate the x and y position in a table from a TD cell
|
840
|
+
* Returns: array[2] int: [x, y]
|
841
|
+
* Inputs: node:n - TD cell of interest
|
842
|
+
* Notes: Not actually interested in this for DataTables since it might go out of date
|
843
|
+
*/
|
844
|
+
function _fnCoordsFromCell( n )
|
845
|
+
{
|
846
|
+
if ( _oDatatable )
|
847
|
+
{
|
848
|
+
var oSettings = _oDatatable.fnSettings();
|
849
|
+
return [
|
850
|
+
jQuery('td', n.parentNode).index(n),
|
851
|
+
jQuery('tr', n.parentNode.parentNode).index(n.parentNode) + oSettings._iDisplayStart
|
852
|
+
];
|
853
|
+
}
|
854
|
+
else
|
855
|
+
{
|
856
|
+
return [
|
857
|
+
jQuery('td', n.parentNode).index(n),
|
858
|
+
jQuery('tr', n.parentNode.parentNode).index(n.parentNode)
|
859
|
+
];
|
860
|
+
}
|
861
|
+
}
|
862
|
+
|
863
|
+
|
864
|
+
/*
|
865
|
+
* Function: _fnSetScrollTop
|
866
|
+
* Purpose: Set the vertical scrolling position
|
867
|
+
* Returns: -
|
868
|
+
* Inputs: int:iPos - scrolltop
|
869
|
+
* Notes: This is so nasty, but without browser detection you can't tell which you should set
|
870
|
+
* So on browsers that support both, the scroll top will be set twice. I can live with
|
871
|
+
* that :-)
|
872
|
+
*/
|
873
|
+
function _fnSetScrollTop( iPos )
|
874
|
+
{
|
875
|
+
document.documentElement.scrollTop = iPos;
|
876
|
+
document.body.scrollTop = iPos;
|
877
|
+
}
|
878
|
+
|
879
|
+
|
880
|
+
/*
|
881
|
+
* Function: _fnSetScrollLeft
|
882
|
+
* Purpose: Set the horizontal scrolling position
|
883
|
+
* Returns: -
|
884
|
+
* Inputs: int:iPos - scrollleft
|
885
|
+
*/
|
886
|
+
function _fnSetScrollLeft( iPos )
|
887
|
+
{
|
888
|
+
document.documentElement.scrollLeft = iPos;
|
889
|
+
document.body.scrollLeft = iPos;
|
890
|
+
}
|
891
|
+
|
892
|
+
|
893
|
+
/*
|
894
|
+
* Function: _fnGetPos
|
895
|
+
* Purpose: Get the position of an object on the rendered page
|
896
|
+
* Returns: array[2] int: [left, right]
|
897
|
+
* Inputs: node:obj - element of interest
|
898
|
+
*/
|
899
|
+
function _fnGetPos ( obj )
|
900
|
+
{
|
901
|
+
var iLeft = 0;
|
902
|
+
var iTop = 0;
|
903
|
+
|
904
|
+
if (obj.offsetParent)
|
905
|
+
{
|
906
|
+
iLeft = obj.offsetLeft;
|
907
|
+
iTop = obj.offsetTop;
|
908
|
+
obj = obj.offsetParent;
|
909
|
+
while (obj)
|
910
|
+
{
|
911
|
+
iLeft += obj.offsetLeft;
|
912
|
+
iTop += obj.offsetTop;
|
913
|
+
obj = obj.offsetParent;
|
914
|
+
}
|
915
|
+
}
|
916
|
+
return [iLeft,iTop];
|
917
|
+
}
|
918
|
+
|
919
|
+
|
920
|
+
/*
|
921
|
+
* Function: _fnFindDtCell
|
922
|
+
* Purpose: Get the coords. of a cell from the DataTables internal information
|
923
|
+
* Returns: array[2] int: [x, y] coords. or null if not found
|
924
|
+
* Inputs: node:nTarget - the node of interest
|
925
|
+
*/
|
926
|
+
function _fnFindDtCell( nTarget )
|
927
|
+
{
|
928
|
+
var oSettings = _oDatatable.fnSettings();
|
929
|
+
for ( var i=0, iLen=oSettings.aiDisplay.length ; i<iLen ; i++ )
|
930
|
+
{
|
931
|
+
var nTr = oSettings.aoData[ oSettings.aiDisplay[i] ].nTr;
|
932
|
+
var nTds = nTr.getElementsByTagName('td');
|
933
|
+
for ( var j=0, jLen=nTds.length ; j<jLen ; j++ )
|
934
|
+
{
|
935
|
+
if ( nTds[j] == nTarget )
|
936
|
+
{
|
937
|
+
return [ j, i ];
|
938
|
+
}
|
939
|
+
}
|
940
|
+
}
|
941
|
+
return null;
|
942
|
+
}
|
943
|
+
|
944
|
+
|
945
|
+
|
946
|
+
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
947
|
+
* Initialisation
|
948
|
+
*/
|
949
|
+
|
950
|
+
/*
|
951
|
+
* Function: _fnInit
|
952
|
+
* Purpose: Initialise the KeyTable
|
953
|
+
* Returns: -
|
954
|
+
* Inputs: object:oInit - optional - Initalisation object with the following parameters:
|
955
|
+
* array[2] int:focus - x and y coordinates of the initial target
|
956
|
+
* or
|
957
|
+
* node:focus - the node to set initial focus on
|
958
|
+
* node:table - the table to use, if not given, first table with class 'KeyTable' will be used
|
959
|
+
* string:focusClass - focusing class to give to table elements
|
960
|
+
* object:that - focus
|
961
|
+
* bool:initScroll - scroll the view port on load, default true
|
962
|
+
* int:tabIndex - the tab index to give the hidden input element
|
963
|
+
*/
|
964
|
+
function _fnInit( oInit, that )
|
965
|
+
{
|
966
|
+
/* Save scope */
|
967
|
+
_that = that;
|
968
|
+
|
969
|
+
/* Capture undefined initialisation and apply the defaults */
|
970
|
+
if ( typeof oInit == 'undefined' ) {
|
971
|
+
oInit = {};
|
972
|
+
}
|
973
|
+
|
974
|
+
if ( typeof oInit.focus == 'undefined' ) {
|
975
|
+
oInit.focus = [0,0];
|
976
|
+
}
|
977
|
+
|
978
|
+
if ( typeof oInit.table == 'undefined' ) {
|
979
|
+
oInit.table = jQuery('table.KeyTable')[0];
|
980
|
+
} else {
|
981
|
+
$(oInit.table).addClass('KeyTable');
|
982
|
+
}
|
983
|
+
|
984
|
+
if ( typeof oInit.focusClass != 'undefined' ) {
|
985
|
+
_sFocusClass = oInit.focusClass;
|
986
|
+
}
|
987
|
+
|
988
|
+
if ( typeof oInit.datatable != 'undefined' ) {
|
989
|
+
_oDatatable = oInit.datatable;
|
990
|
+
}
|
991
|
+
|
992
|
+
if ( typeof oInit.initScroll == 'undefined' ) {
|
993
|
+
oInit.initScroll = true;
|
994
|
+
}
|
995
|
+
|
996
|
+
if ( typeof oInit.form == 'undefined' ) {
|
997
|
+
oInit.form = false;
|
998
|
+
}
|
999
|
+
_bForm = oInit.form;
|
1000
|
+
|
1001
|
+
/* Cache the tbody node of interest */
|
1002
|
+
_nBody = oInit.table.getElementsByTagName('tbody')[0];
|
1003
|
+
|
1004
|
+
/* If the table is inside a form, then we need a hidden input box which can be used by the
|
1005
|
+
* browser to catch the browser tabbing for our table
|
1006
|
+
*/
|
1007
|
+
if ( _bForm )
|
1008
|
+
{
|
1009
|
+
var nDiv = document.createElement('div');
|
1010
|
+
_nInput = document.createElement('input');
|
1011
|
+
nDiv.style.height = "1px"; /* Opera requires a little something */
|
1012
|
+
nDiv.style.width = "0px";
|
1013
|
+
nDiv.style.overflow = "hidden";
|
1014
|
+
if ( typeof oInit.tabIndex != 'undefined' )
|
1015
|
+
{
|
1016
|
+
_nInput.tabIndex = oInit.tabIndex;
|
1017
|
+
}
|
1018
|
+
nDiv.appendChild(_nInput);
|
1019
|
+
oInit.table.parentNode.insertBefore( nDiv, oInit.table.nextSibling );
|
1020
|
+
|
1021
|
+
jQuery(_nInput).focus( function () {
|
1022
|
+
/* See if we want to 'tab into' the table or out */
|
1023
|
+
if ( !_bInputFocused )
|
1024
|
+
{
|
1025
|
+
_bKeyCapture = true;
|
1026
|
+
_bInputFocused = false;
|
1027
|
+
if ( typeof oInit.focus.nodeName != "undefined" )
|
1028
|
+
{
|
1029
|
+
_fnSetFocus( oInit.focus, oInit.initScroll );
|
1030
|
+
}
|
1031
|
+
else
|
1032
|
+
{
|
1033
|
+
_fnSetFocus( _fnCellFromCoords( oInit.focus[0], oInit.focus[1]), oInit.initScroll );
|
1034
|
+
}
|
1035
|
+
|
1036
|
+
/* Need to interup the thread for this to work */
|
1037
|
+
setTimeout( function() { _nInput.blur(); }, 0 );
|
1038
|
+
}
|
1039
|
+
} );
|
1040
|
+
_bKeyCapture = false;
|
1041
|
+
}
|
1042
|
+
else
|
1043
|
+
{
|
1044
|
+
/* Set the initial focus on the table */
|
1045
|
+
if ( typeof oInit.focus.nodeName != "undefined" )
|
1046
|
+
{
|
1047
|
+
_fnSetFocus( oInit.focus, oInit.initScroll );
|
1048
|
+
}
|
1049
|
+
else
|
1050
|
+
{
|
1051
|
+
_fnSetFocus( _fnCellFromCoords( oInit.focus[0], oInit.focus[1]), oInit.initScroll );
|
1052
|
+
}
|
1053
|
+
_fnCaptureKeys();
|
1054
|
+
}
|
1055
|
+
|
1056
|
+
/*
|
1057
|
+
* Add event listeners
|
1058
|
+
* Well - I hate myself for doing this, but it would appear that key events in browsers are
|
1059
|
+
* a complete mess, particulay when you consider arrow keys, which of course are one of the
|
1060
|
+
* main areas of interest here. So basically for arrow keys, there is no keypress event in
|
1061
|
+
* Safari and IE, while there is in Firefox and Opera. But Firefox and Opera don't repeat the
|
1062
|
+
* keydown event for an arrow key. OUCH. See the following two articles for more:
|
1063
|
+
* http://www.quirksmode.org/dom/events/keys.html
|
1064
|
+
* https://lists.webkit.org/pipermail/webkit-dev/2007-December/002992.html
|
1065
|
+
* http://unixpapa.com/js/key.html
|
1066
|
+
* PPK considers the IE / Safari method correct (good enough for me!) so we (urgh) detect
|
1067
|
+
* Mozilla and Opera and apply keypress for them, while everything else gets keydown. If
|
1068
|
+
* Mozilla or Opera change their implemention in future, this will need to be updated...
|
1069
|
+
* although at the time of writing (14th March 2009) Minefield still uses the 3.0 behaviour.
|
1070
|
+
*/
|
1071
|
+
if ( jQuery.browser.mozilla || jQuery.browser.opera )
|
1072
|
+
{
|
1073
|
+
jQuery(document).bind( "keypress", _fnKey );
|
1074
|
+
}
|
1075
|
+
else
|
1076
|
+
{
|
1077
|
+
jQuery(document).bind( "keydown", _fnKey );
|
1078
|
+
}
|
1079
|
+
|
1080
|
+
if ( _oDatatable )
|
1081
|
+
{
|
1082
|
+
jQuery('tbody td', _oDatatable.fnSettings().nTable).live( 'click', _fnClick );
|
1083
|
+
}
|
1084
|
+
else
|
1085
|
+
{
|
1086
|
+
jQuery('td', _nBody).live( 'click', _fnClick );
|
1087
|
+
}
|
1088
|
+
|
1089
|
+
/* Loose table focus when click outside the table */
|
1090
|
+
jQuery(document).click( function(e) {
|
1091
|
+
var nTarget = e.target;
|
1092
|
+
var bTableClick = false;
|
1093
|
+
while ( nTarget )
|
1094
|
+
{
|
1095
|
+
if ( nTarget == oInit.table )
|
1096
|
+
{
|
1097
|
+
bTableClick = true;
|
1098
|
+
break;
|
1099
|
+
}
|
1100
|
+
nTarget = nTarget.parentNode;
|
1101
|
+
}
|
1102
|
+
if ( !bTableClick )
|
1103
|
+
{
|
1104
|
+
_fnBlur();
|
1105
|
+
}
|
1106
|
+
} );
|
1107
|
+
}
|
1108
|
+
|
1109
|
+
/* Initialise our new object */
|
1110
|
+
_fnInit( oInit, this );
|
1111
|
+
}
|