-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathjquery.list.js
447 lines (364 loc) · 14 KB
/
jquery.list.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
/**
* jQuery list plug-in 1.2.4
* Copyright 2012, Digital Fusion
* Licensed under the MIT license.
* http://teamdf.com/jquery-plugins/license/
*
* @author Sam Sehnert, Phil Taylor
*/
(function($){
"use strict";
// The name of your plugin. This is used to namespace your
// plugin methods, object data, and registerd events.
var plugin_name = 'list';
var plugin_version = '1.2.4';
var plugin_logging = true;
// Set up the plugin defaults.
// These will be stored in $this.data(plugin_name).settings,
// and can be overwritten by having 'options' passed through
// as the parameter to the init method.
var defaults = {
headerSelector : 'dt',
zIndex : 1,
fakeHeaderClass : 'ui-' + plugin_name + '-fakeheader'
};
var scrollTimeout = null;
// Any private methods that the plugin uses, such as event handlers,
// or small utility functions for parsing data etc that isn't useful
// outside the context of the plugin code itself.
var _private = {
log : function(){
// SAFELY call console.log without fear of errors
// also provides an override for turning off all plugin console output.
if( plugin_logging && window.console && window.console.log ){
try {
// IE 8 doesn't implement apply on the console methods.
window.console.log.apply(console,arguments);
} catch (e) {
// Because IE8 doesn't implement apply for console methods,
// we simply pass the arguments directly to the log.
window.console.log( arguments );
}
}
},
// Update the whole header object on the fly so all properties will be reserved. So setting
// different styles for specific titles, such as different font color for today or weekends,
// holidays, etc. becomes posible.
updateHeader: function(target, newHeader) {
var data = target.data(plugin_name);
if (data) {
data.fakeHeader = newHeader.clone().removeAttr('id').addClass(data.settings.fakeHeaderClass);
data.fakeHeader.css({
position: 'absolute',
top: 0,
width: data.headers.width(),
zIndex: data.settings.zIndex
});
target.data(plugin_name, data).children(0).eq(0).replaceWith(data.fakeHeader);
}
},
// Contains events for this plugin.
events : {
/**
* Window resize (should also be called when resizing the target element).
*/
resize : function(){
var $this = $(this),data = $this.data(plugin_name);
if( data ){
data.fakeHeader.width(data.headers.width());
data.wrapper.css('maxHeight',$this.css('maxHeight'));
}
},
/**
* List element scroll handler event. Called to animate and substitute heading blocks.
*/
scroll : function(){
var $this = $(this),data = $this.data(plugin_name);
if( data ){
var newHeader = null,
currentHeader = data.headers.eq( data.currentHeader ),
nextHeader = data.currentHeader >= data.headers.length-1 ? null : data.headers.eq( data.currentHeader+1 ),
prevHeader = data.currentHeader <= 0 ? null : data.headers.eq( data.currentHeader-1 ),
trigger = false;
// Make sure the container top position is fresh.
data.containerTop = $this.offset().top + parseInt($this.css('marginTop'),10) + parseInt($this.css('borderTopWidth'),10);
data.fakeHeader.css('top',0);
// Check the position of the current header rather than the previous header.
if( prevHeader !== null ){
var top = currentHeader.offset().top,
height = currentHeader.outerHeight();
if( top > data.containerTop ){
data.fakeHeader.css('top',(top-height)-data.containerTop);
_private.updateHeader($this, prevHeader);
data.currentHeader = data.currentHeader-1;
trigger = true;
}
if( (top-height) > data.containerTop ){
data.fakeHeader.css('top',0);
newHeader = data.currentHeader-1;
}
}
// Check the position of the next header element.
if( nextHeader !== null ){
var top = nextHeader.offset().top,
height = nextHeader.outerHeight();
if( (top-height) < data.containerTop ){
data.fakeHeader.css('top',(top-height)-data.containerTop);
}
if( top < data.containerTop ){
data.fakeHeader.css('top',0);
newHeader = data.currentHeader+1;
}
}
// Now assign the contents of the previous header.
if( newHeader !== null ){
var $header = data.headers.eq(newHeader);
data.currentHeader = newHeader;
_private.updateHeader($this, $header);
trigger = true;
}
var max = ( data.wrapper.scrollTop() >= data.wrapper.prop('scrollHeight') - data.wrapper.height() );
if( trigger || max || data.max && !max ){
// Trigger the headingChange event.
$this.trigger('headingChange',[data.currentHeader,data.headers.eq(data.currentHeader),max]);
}
data.max = max;
// Save the new data.
$this.data(plugin_name,data);
}
}
}
}
// The methods array will allow you to define public methods that
// can be called using the plugin function using the following syntax;
//
// $('.selector').plugin_name( 'my_method'/*, optional arguments */);
//
// The 'init' method is special, and will be called when the user calls;
//
// $('.selector').plugin_name(/*{ options object }*/);
var methods = {
/**
* Initialises the plugin.
*
* @param options object : An object containing any overrides to the default settings.
*
* @return collection : Returns the jQuery collection
*/
init : function( options ){
// Loop through each passed element.
return $(this).each(function(){
// Settings to the defaults.
var settings = $.extend({},defaults);
// If options exist, lets merge them
// with our default settings.
if( typeof options == 'object' ) $.extend( settings, options );
// Create shortcuts, and get any existing data.
var $this = $(this), data = $this.data(plugin_name);
// If the plugin hasn't been initialized yet
if ( ! data ) {
// Create the data object.
data = {
target : $this, // This element.
wrapper : $this.wrapInner('<div class="ui-'+plugin_name+'" />').find('.ui-'+plugin_name+''),
settings : settings, // The settings for this plugin.
headers : [],
containerTop : 0,
currentHeader : 0,
fakeHeader : null,
scrolllist : [],
original : {
position : '',
overflowX : '',
overflowY : ''
},
max : false
}
// Add the container class, and the base HTML structure
$this.addClass('-'+plugin_name+'-container').css({
position : $this.css('position') == 'absolute' ? 'absolute' : 'relative',
overflowY : 'hidden'
});
// Grab some variables to set up the list.
data.headers = $this.find(data.settings.headerSelector);
data.fakeHeader = data.headers.eq(0).clone().removeAttr('id').addClass(data.settings.fakeHeaderClass);
// bind a scroll event and change the text of the fake heading
data.wrapper.bind('scroll.'+plugin_name,$.proxy(_private.events.scroll,$this)).css({
height : '100%',
maxHeight : $this.css('maxHeight'),
overflowY : 'scroll',
position : 'relative'
});
// Set the fake headers
data.fakeHeader.css({
position : 'absolute',
top : 0,
width : data.headers.width(),
zIndex : data.settings.zIndex
});
// Bind the resize event to the window.
$(window).bind('resize.'+plugin_name,$.proxy(_private.events.resize,$this));
// Add the fake header before all other children, and set the HTML.
$this.data(plugin_name,data).prepend( data.fakeHeader );
}
});
},
/**
* Retrieves the current header index that the user is scrolled to.
*
* @return int : The index position of the header (relative to the headers collection).
*/
header : function(){
var $this = $(this),
data = $this.data(plugin_name);
// Only bother if we've set this up before.
if( data ){
return data.currentHeader;
}
},
/**
* Used to scroll to a new header element, or to retrieve the currently set header.
*
* @param int newHeader : The index position of the header (relative to the headers collection).
* @param mixed speed : The animation speed of scrolling to the new header.
* @param mixed easing : The animation easing to use.
* @param function completion : The animation completion function to call.
*
* @return collection : The collection that this was called with
*/
scrollTo : function( newHeader, speed, easing, completion ){
return this.each(function(){
var $this = $(this),
data = $this.data(plugin_name);
// Only bother if we've set this up before.
if( data ){
// If we've got the header, and its a number
if( newHeader !== undefined && !isNaN( newHeader ) && newHeader >= 0 && newHeader < data.headers.length ){
// Get the new header.
var $header = data.headers.eq(newHeader),
scrollTo = $header.position().top + data.wrapper.scrollTop() + parseInt($header.css('borderTopWidth'),10) + parseInt($header.css('borderBottomWidth'),10);
// If we're not animating, we need to set the element directly.
if( !speed ){
data.wrapper.stop().scrollTop( scrollTo );
// Set as the current header.
data.currentHeader = newHeader;
_private.updateHeader($this, $header);
// Trigger the headingChange event.
$this.trigger('headingChange',[newHeader,$header]);
// Store the new header data.
$this.data(plugin_name,data);
} else {
// If we are animating, the scroll event will fire.
data.wrapper.stop().animate({scrollTop:scrollTo},speed,easing,completion);
}
}
}
});
},
/**
* Used to modify settings after initialisation has taken place.
* an example switch construct has been written to show how you
* might fire off an update procedure for certain defaults.
* Should only be called on one element at a time.
*
* @param key string : The option name that you want to update.
* @param value mixed : (opt) The value that you want to set the option to.
*
* @return mixed : If no value is passed, will return the value for the passed key, otherwise, returns the jQuery collection.
*/
option : function( key, value ){
var $this = $(this),
data = $this.data(plugin_name);
// Only bother if we've set this up before.
if( data ){
// Return settings array if no key is provided.
if( typeof key == 'undefined' ) return data.settings;
// The key has to exist, otherwise its invalid.
if( !key in data.settings ) return false;
// Check if we're adding or updating.
if( typeof value == 'undefined' ){
return data.settings[key];
} else {
data.settings[key] = value;
return $this;
}
}
},
/**
* Get the current name and version number of this plugin.
*
* @param num bool : Whether to return the version number, or string.
*
* @return string | number : The version string or version number.
*/
version : function( num ){
// Returns the version string for this plugin.
if( num ){
// Calculate the version number.
var v = plugin_version.split('.'),
major = ( Number( v[0] ) || 1 )+'',
minor = ( Number( v[1] ) || 0 )+'',
bugfix = ( Number( v[2] ) || 0 )+'',
fill = '000';
// Return the version as a comparable number.
return Number( fill.slice(0,3-major.length)+major+fill.slice(0,3-minor.length)+minor+fill.slice(0,3-bugfix.length)+bugfix );
} else {
// Return the version string for this plugin.
return plugin_name+' v'+plugin_version;
}
},
/**
* Remove all data and events associated with the plugin, and restore
* the status of any manipulated elmenets to pre-plugin state.
*
* @return collection : Returns the jQuery collection.
*/
destroy: function(){
/* Remove the ui plugin from these elements that have it */
return this.each(function(){
var $this = $(this),
data = $this.data(plugin_name);
// Only bother if we've set this up before.
if( data ){
// Remove wrapper and fakeheader.
data.wrapper.children().unwrap();
data.fakeHeader.remove();
// Now, remove all data, etc, then
// reattach this element to its parent, then delete list div.
$this.css(data.original)
.removeData(plugin_name)
.removeClass('-'+plugin_name+'-container')
.unbind('.'+plugin_name);
}
});
}
};
/**
* Plugin method calling handler.
*
* @param method string : (opt) Calls the defined method (or init function if omitted).
* @param arguments : Any remaining arguments will be passed as arguments to the recieving method.
*
* @return mixed : Returns the result of the called method.
*/
$.fn[plugin_name] = function( method ){
// Method calling logic
if ( methods[method] ) {
return methods[ method ].apply( this, Array.prototype.slice.call( arguments, 1 ));
} else if ( typeof method === 'object' || ! method ) {
return methods.init.apply( this, arguments );
} else {
$.error( 'Method ' + method + ' does not exist on jQuery.' + plugin_name );
}
}
// Initialise $.plugin_name for methods below.
$[plugin_name] = {};
/**
* Allow console logging from this plugin?
*
* @param l bool : (opt) Turn logging on or off...
*
* @return bool : Whether this plugin will print to the log or not.
*/
$[plugin_name].log = function(l){ if(l!==undefined){ plugin_logging=l; } return plugin_logging; };
})(jQuery);