cloudinary 1.0.82 → 1.0.83

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.
Files changed (36) hide show
  1. checksums.yaml +4 -4
  2. data/.gitignore +58 -7
  3. data/CHANGELOG +13 -0
  4. data/Gemfile +1 -1
  5. data/cloudinary.gemspec +13 -3
  6. data/lib/cloudinary/active_support/core_ext/hash/keys.rb +166 -0
  7. data/lib/cloudinary/active_support/core_ext/hash/readme.md +1 -0
  8. data/lib/cloudinary/api.rb +2 -1
  9. data/lib/cloudinary/carrier_wave/preloaded.rb +1 -1
  10. data/lib/cloudinary/helper.rb +38 -21
  11. data/lib/cloudinary/missing.rb +11 -10
  12. data/lib/cloudinary/uploader.rb +18 -20
  13. data/lib/cloudinary/utils.rb +187 -75
  14. data/lib/cloudinary/version.rb +1 -1
  15. data/lib/cloudinary/video_helper.rb +126 -0
  16. data/spec/api_spec.rb +23 -22
  17. data/spec/cloudinary_helper_spec.rb +34 -20
  18. data/spec/spec_helper.rb +75 -5
  19. data/spec/uploader_spec.rb +32 -3
  20. data/spec/utils_methods_spec.rb +18 -0
  21. data/spec/utils_spec.rb +73 -78
  22. data/spec/video_tag_spec.rb +186 -0
  23. data/spec/video_url_spec.rb +169 -0
  24. data/vendor/assets/html/cloudinary_cors.html +47 -0
  25. data/vendor/assets/javascripts/cloudinary/canvas-to-blob.min.js +1 -0
  26. data/vendor/assets/javascripts/cloudinary/index.js +4 -0
  27. data/vendor/assets/javascripts/cloudinary/jquery.cloudinary.js +898 -0
  28. data/vendor/assets/javascripts/cloudinary/jquery.fileupload-image.js +315 -0
  29. data/vendor/assets/javascripts/cloudinary/jquery.fileupload-process.js +172 -0
  30. data/vendor/assets/javascripts/cloudinary/jquery.fileupload-validate.js +119 -0
  31. data/vendor/assets/javascripts/cloudinary/jquery.fileupload.js +1460 -0
  32. data/vendor/assets/javascripts/cloudinary/jquery.iframe-transport.js +214 -0
  33. data/vendor/assets/javascripts/cloudinary/jquery.ui.widget.js +558 -0
  34. data/vendor/assets/javascripts/cloudinary/load-image.min.js +1 -0
  35. data/vendor/assets/javascripts/cloudinary/processing.js +5 -0
  36. metadata +44 -7
@@ -0,0 +1,214 @@
1
+ /*
2
+ * jQuery Iframe Transport Plugin 1.8.2
3
+ * https://github.com/blueimp/jQuery-File-Upload
4
+ *
5
+ * Copyright 2011, Sebastian Tschan
6
+ * https://blueimp.net
7
+ *
8
+ * Licensed under the MIT license:
9
+ * http://www.opensource.org/licenses/MIT
10
+ */
11
+
12
+ /* global define, window, document */
13
+
14
+ (function (factory) {
15
+ 'use strict';
16
+ if (typeof define === 'function' && define.amd) {
17
+ // Register as an anonymous AMD module:
18
+ define(['jquery'], factory);
19
+ } else {
20
+ // Browser globals:
21
+ factory(window.jQuery);
22
+ }
23
+ }(function ($) {
24
+ 'use strict';
25
+
26
+ // Helper variable to create unique names for the transport iframes:
27
+ var counter = 0;
28
+
29
+ // The iframe transport accepts four additional options:
30
+ // options.fileInput: a jQuery collection of file input fields
31
+ // options.paramName: the parameter name for the file form data,
32
+ // overrides the name property of the file input field(s),
33
+ // can be a string or an array of strings.
34
+ // options.formData: an array of objects with name and value properties,
35
+ // equivalent to the return data of .serializeArray(), e.g.:
36
+ // [{name: 'a', value: 1}, {name: 'b', value: 2}]
37
+ // options.initialIframeSrc: the URL of the initial iframe src,
38
+ // by default set to "javascript:false;"
39
+ $.ajaxTransport('iframe', function (options) {
40
+ if (options.async) {
41
+ // javascript:false as initial iframe src
42
+ // prevents warning popups on HTTPS in IE6:
43
+ /*jshint scripturl: true */
44
+ var initialIframeSrc = options.initialIframeSrc || 'javascript:false;',
45
+ /*jshint scripturl: false */
46
+ form,
47
+ iframe,
48
+ addParamChar;
49
+ return {
50
+ send: function (_, completeCallback) {
51
+ form = $('<form style="display:none;"></form>');
52
+ form.attr('accept-charset', options.formAcceptCharset);
53
+ addParamChar = /\?/.test(options.url) ? '&' : '?';
54
+ // XDomainRequest only supports GET and POST:
55
+ if (options.type === 'DELETE') {
56
+ options.url = options.url + addParamChar + '_method=DELETE';
57
+ options.type = 'POST';
58
+ } else if (options.type === 'PUT') {
59
+ options.url = options.url + addParamChar + '_method=PUT';
60
+ options.type = 'POST';
61
+ } else if (options.type === 'PATCH') {
62
+ options.url = options.url + addParamChar + '_method=PATCH';
63
+ options.type = 'POST';
64
+ }
65
+ // IE versions below IE8 cannot set the name property of
66
+ // elements that have already been added to the DOM,
67
+ // so we set the name along with the iframe HTML markup:
68
+ counter += 1;
69
+ iframe = $(
70
+ '<iframe src="' + initialIframeSrc +
71
+ '" name="iframe-transport-' + counter + '"></iframe>'
72
+ ).bind('load', function () {
73
+ var fileInputClones,
74
+ paramNames = $.isArray(options.paramName) ?
75
+ options.paramName : [options.paramName];
76
+ iframe
77
+ .unbind('load')
78
+ .bind('load', function () {
79
+ var response;
80
+ // Wrap in a try/catch block to catch exceptions thrown
81
+ // when trying to access cross-domain iframe contents:
82
+ try {
83
+ response = iframe.contents();
84
+ // Google Chrome and Firefox do not throw an
85
+ // exception when calling iframe.contents() on
86
+ // cross-domain requests, so we unify the response:
87
+ if (!response.length || !response[0].firstChild) {
88
+ throw new Error();
89
+ }
90
+ } catch (e) {
91
+ response = undefined;
92
+ }
93
+ // The complete callback returns the
94
+ // iframe content document as response object:
95
+ completeCallback(
96
+ 200,
97
+ 'success',
98
+ {'iframe': response}
99
+ );
100
+ // Fix for IE endless progress bar activity bug
101
+ // (happens on form submits to iframe targets):
102
+ $('<iframe src="' + initialIframeSrc + '"></iframe>')
103
+ .appendTo(form);
104
+ window.setTimeout(function () {
105
+ // Removing the form in a setTimeout call
106
+ // allows Chrome's developer tools to display
107
+ // the response result
108
+ form.remove();
109
+ }, 0);
110
+ });
111
+ form
112
+ .prop('target', iframe.prop('name'))
113
+ .prop('action', options.url)
114
+ .prop('method', options.type);
115
+ if (options.formData) {
116
+ $.each(options.formData, function (index, field) {
117
+ $('<input type="hidden"/>')
118
+ .prop('name', field.name)
119
+ .val(field.value)
120
+ .appendTo(form);
121
+ });
122
+ }
123
+ if (options.fileInput && options.fileInput.length &&
124
+ options.type === 'POST') {
125
+ fileInputClones = options.fileInput.clone();
126
+ // Insert a clone for each file input field:
127
+ options.fileInput.after(function (index) {
128
+ return fileInputClones[index];
129
+ });
130
+ if (options.paramName) {
131
+ options.fileInput.each(function (index) {
132
+ $(this).prop(
133
+ 'name',
134
+ paramNames[index] || options.paramName
135
+ );
136
+ });
137
+ }
138
+ // Appending the file input fields to the hidden form
139
+ // removes them from their original location:
140
+ form
141
+ .append(options.fileInput)
142
+ .prop('enctype', 'multipart/form-data')
143
+ // enctype must be set as encoding for IE:
144
+ .prop('encoding', 'multipart/form-data');
145
+ // Remove the HTML5 form attribute from the input(s):
146
+ options.fileInput.removeAttr('form');
147
+ }
148
+ form.submit();
149
+ // Insert the file input fields at their original location
150
+ // by replacing the clones with the originals:
151
+ if (fileInputClones && fileInputClones.length) {
152
+ options.fileInput.each(function (index, input) {
153
+ var clone = $(fileInputClones[index]);
154
+ // Restore the original name and form properties:
155
+ $(input)
156
+ .prop('name', clone.prop('name'))
157
+ .attr('form', clone.attr('form'));
158
+ clone.replaceWith(input);
159
+ });
160
+ }
161
+ });
162
+ form.append(iframe).appendTo(document.body);
163
+ },
164
+ abort: function () {
165
+ if (iframe) {
166
+ // javascript:false as iframe src aborts the request
167
+ // and prevents warning popups on HTTPS in IE6.
168
+ // concat is used to avoid the "Script URL" JSLint error:
169
+ iframe
170
+ .unbind('load')
171
+ .prop('src', initialIframeSrc);
172
+ }
173
+ if (form) {
174
+ form.remove();
175
+ }
176
+ }
177
+ };
178
+ }
179
+ });
180
+
181
+ // The iframe transport returns the iframe content document as response.
182
+ // The following adds converters from iframe to text, json, html, xml
183
+ // and script.
184
+ // Please note that the Content-Type for JSON responses has to be text/plain
185
+ // or text/html, if the browser doesn't include application/json in the
186
+ // Accept header, else IE will show a download dialog.
187
+ // The Content-Type for XML responses on the other hand has to be always
188
+ // application/xml or text/xml, so IE properly parses the XML response.
189
+ // See also
190
+ // https://github.com/blueimp/jQuery-File-Upload/wiki/Setup#content-type-negotiation
191
+ $.ajaxSetup({
192
+ converters: {
193
+ 'iframe text': function (iframe) {
194
+ return iframe && $(iframe[0].body).text();
195
+ },
196
+ 'iframe json': function (iframe) {
197
+ return iframe && $.parseJSON($(iframe[0].body).text());
198
+ },
199
+ 'iframe html': function (iframe) {
200
+ return iframe && $(iframe[0].body).html();
201
+ },
202
+ 'iframe xml': function (iframe) {
203
+ var xmlDoc = iframe && iframe[0];
204
+ return xmlDoc && $.isXMLDoc(xmlDoc) ? xmlDoc :
205
+ $.parseXML((xmlDoc.XMLDocument && xmlDoc.XMLDocument.xml) ||
206
+ $(xmlDoc.body).html());
207
+ },
208
+ 'iframe script': function (iframe) {
209
+ return iframe && $.globalEval($(iframe[0].body).text());
210
+ }
211
+ }
212
+ });
213
+
214
+ }));
@@ -0,0 +1,558 @@
1
+ /*! jQuery UI - v1.11.1 - 2014-09-17
2
+ * http://jqueryui.com
3
+ * Includes: widget.js
4
+ * Copyright 2014 jQuery Foundation and other contributors; Licensed MIT */
5
+
6
+ (function( factory ) {
7
+ if ( typeof define === "function" && define.amd ) {
8
+
9
+ // AMD. Register as an anonymous module.
10
+ define([ "jquery" ], factory );
11
+ } else {
12
+
13
+ // Browser globals
14
+ factory( jQuery );
15
+ }
16
+ }(function( $ ) {
17
+ /*!
18
+ * jQuery UI Widget 1.11.1
19
+ * http://jqueryui.com
20
+ *
21
+ * Copyright 2014 jQuery Foundation and other contributors
22
+ * Released under the MIT license.
23
+ * http://jquery.org/license
24
+ *
25
+ * http://api.jqueryui.com/jQuery.widget/
26
+ */
27
+
28
+
29
+ var widget_uuid = 0,
30
+ widget_slice = Array.prototype.slice;
31
+
32
+ $.cleanData = (function( orig ) {
33
+ return function( elems ) {
34
+ var events, elem, i;
35
+ for ( i = 0; (elem = elems[i]) != null; i++ ) {
36
+ try {
37
+
38
+ // Only trigger remove when necessary to save time
39
+ events = $._data( elem, "events" );
40
+ if ( events && events.remove ) {
41
+ $( elem ).triggerHandler( "remove" );
42
+ }
43
+
44
+ // http://bugs.jquery.com/ticket/8235
45
+ } catch( e ) {}
46
+ }
47
+ orig( elems );
48
+ };
49
+ })( $.cleanData );
50
+
51
+ $.widget = function( name, base, prototype ) {
52
+ var fullName, existingConstructor, constructor, basePrototype,
53
+ // proxiedPrototype allows the provided prototype to remain unmodified
54
+ // so that it can be used as a mixin for multiple widgets (#8876)
55
+ proxiedPrototype = {},
56
+ namespace = name.split( "." )[ 0 ];
57
+
58
+ name = name.split( "." )[ 1 ];
59
+ fullName = namespace + "-" + name;
60
+
61
+ if ( !prototype ) {
62
+ prototype = base;
63
+ base = $.Widget;
64
+ }
65
+
66
+ // create selector for plugin
67
+ $.expr[ ":" ][ fullName.toLowerCase() ] = function( elem ) {
68
+ return !!$.data( elem, fullName );
69
+ };
70
+
71
+ $[ namespace ] = $[ namespace ] || {};
72
+ existingConstructor = $[ namespace ][ name ];
73
+ constructor = $[ namespace ][ name ] = function( options, element ) {
74
+ // allow instantiation without "new" keyword
75
+ if ( !this._createWidget ) {
76
+ return new constructor( options, element );
77
+ }
78
+
79
+ // allow instantiation without initializing for simple inheritance
80
+ // must use "new" keyword (the code above always passes args)
81
+ if ( arguments.length ) {
82
+ this._createWidget( options, element );
83
+ }
84
+ };
85
+ // extend with the existing constructor to carry over any static properties
86
+ $.extend( constructor, existingConstructor, {
87
+ version: prototype.version,
88
+ // copy the object used to create the prototype in case we need to
89
+ // redefine the widget later
90
+ _proto: $.extend( {}, prototype ),
91
+ // track widgets that inherit from this widget in case this widget is
92
+ // redefined after a widget inherits from it
93
+ _childConstructors: []
94
+ });
95
+
96
+ basePrototype = new base();
97
+ // we need to make the options hash a property directly on the new instance
98
+ // otherwise we'll modify the options hash on the prototype that we're
99
+ // inheriting from
100
+ basePrototype.options = $.widget.extend( {}, basePrototype.options );
101
+ $.each( prototype, function( prop, value ) {
102
+ if ( !$.isFunction( value ) ) {
103
+ proxiedPrototype[ prop ] = value;
104
+ return;
105
+ }
106
+ proxiedPrototype[ prop ] = (function() {
107
+ var _super = function() {
108
+ return base.prototype[ prop ].apply( this, arguments );
109
+ },
110
+ _superApply = function( args ) {
111
+ return base.prototype[ prop ].apply( this, args );
112
+ };
113
+ return function() {
114
+ var __super = this._super,
115
+ __superApply = this._superApply,
116
+ returnValue;
117
+
118
+ this._super = _super;
119
+ this._superApply = _superApply;
120
+
121
+ returnValue = value.apply( this, arguments );
122
+
123
+ this._super = __super;
124
+ this._superApply = __superApply;
125
+
126
+ return returnValue;
127
+ };
128
+ })();
129
+ });
130
+ constructor.prototype = $.widget.extend( basePrototype, {
131
+ // TODO: remove support for widgetEventPrefix
132
+ // always use the name + a colon as the prefix, e.g., draggable:start
133
+ // don't prefix for widgets that aren't DOM-based
134
+ widgetEventPrefix: existingConstructor ? (basePrototype.widgetEventPrefix || name) : name
135
+ }, proxiedPrototype, {
136
+ constructor: constructor,
137
+ namespace: namespace,
138
+ widgetName: name,
139
+ widgetFullName: fullName
140
+ });
141
+
142
+ // If this widget is being redefined then we need to find all widgets that
143
+ // are inheriting from it and redefine all of them so that they inherit from
144
+ // the new version of this widget. We're essentially trying to replace one
145
+ // level in the prototype chain.
146
+ if ( existingConstructor ) {
147
+ $.each( existingConstructor._childConstructors, function( i, child ) {
148
+ var childPrototype = child.prototype;
149
+
150
+ // redefine the child widget using the same prototype that was
151
+ // originally used, but inherit from the new version of the base
152
+ $.widget( childPrototype.namespace + "." + childPrototype.widgetName, constructor, child._proto );
153
+ });
154
+ // remove the list of existing child constructors from the old constructor
155
+ // so the old child constructors can be garbage collected
156
+ delete existingConstructor._childConstructors;
157
+ } else {
158
+ base._childConstructors.push( constructor );
159
+ }
160
+
161
+ $.widget.bridge( name, constructor );
162
+
163
+ return constructor;
164
+ };
165
+
166
+ $.widget.extend = function( target ) {
167
+ var input = widget_slice.call( arguments, 1 ),
168
+ inputIndex = 0,
169
+ inputLength = input.length,
170
+ key,
171
+ value;
172
+ for ( ; inputIndex < inputLength; inputIndex++ ) {
173
+ for ( key in input[ inputIndex ] ) {
174
+ value = input[ inputIndex ][ key ];
175
+ if ( input[ inputIndex ].hasOwnProperty( key ) && value !== undefined ) {
176
+ // Clone objects
177
+ if ( $.isPlainObject( value ) ) {
178
+ target[ key ] = $.isPlainObject( target[ key ] ) ?
179
+ $.widget.extend( {}, target[ key ], value ) :
180
+ // Don't extend strings, arrays, etc. with objects
181
+ $.widget.extend( {}, value );
182
+ // Copy everything else by reference
183
+ } else {
184
+ target[ key ] = value;
185
+ }
186
+ }
187
+ }
188
+ }
189
+ return target;
190
+ };
191
+
192
+ $.widget.bridge = function( name, object ) {
193
+ var fullName = object.prototype.widgetFullName || name;
194
+ $.fn[ name ] = function( options ) {
195
+ var isMethodCall = typeof options === "string",
196
+ args = widget_slice.call( arguments, 1 ),
197
+ returnValue = this;
198
+
199
+ // allow multiple hashes to be passed on init
200
+ options = !isMethodCall && args.length ?
201
+ $.widget.extend.apply( null, [ options ].concat(args) ) :
202
+ options;
203
+
204
+ if ( isMethodCall ) {
205
+ this.each(function() {
206
+ var methodValue,
207
+ instance = $.data( this, fullName );
208
+ if ( options === "instance" ) {
209
+ returnValue = instance;
210
+ return false;
211
+ }
212
+ if ( !instance ) {
213
+ return $.error( "cannot call methods on " + name + " prior to initialization; " +
214
+ "attempted to call method '" + options + "'" );
215
+ }
216
+ if ( !$.isFunction( instance[options] ) || options.charAt( 0 ) === "_" ) {
217
+ return $.error( "no such method '" + options + "' for " + name + " widget instance" );
218
+ }
219
+ methodValue = instance[ options ].apply( instance, args );
220
+ if ( methodValue !== instance && methodValue !== undefined ) {
221
+ returnValue = methodValue && methodValue.jquery ?
222
+ returnValue.pushStack( methodValue.get() ) :
223
+ methodValue;
224
+ return false;
225
+ }
226
+ });
227
+ } else {
228
+ this.each(function() {
229
+ var instance = $.data( this, fullName );
230
+ if ( instance ) {
231
+ instance.option( options || {} );
232
+ if ( instance._init ) {
233
+ instance._init();
234
+ }
235
+ } else {
236
+ $.data( this, fullName, new object( options, this ) );
237
+ }
238
+ });
239
+ }
240
+
241
+ return returnValue;
242
+ };
243
+ };
244
+
245
+ $.Widget = function( /* options, element */ ) {};
246
+ $.Widget._childConstructors = [];
247
+
248
+ $.Widget.prototype = {
249
+ widgetName: "widget",
250
+ widgetEventPrefix: "",
251
+ defaultElement: "<div>",
252
+ options: {
253
+ disabled: false,
254
+
255
+ // callbacks
256
+ create: null
257
+ },
258
+ _createWidget: function( options, element ) {
259
+ element = $( element || this.defaultElement || this )[ 0 ];
260
+ this.element = $( element );
261
+ this.uuid = widget_uuid++;
262
+ this.eventNamespace = "." + this.widgetName + this.uuid;
263
+ this.options = $.widget.extend( {},
264
+ this.options,
265
+ this._getCreateOptions(),
266
+ options );
267
+
268
+ this.bindings = $();
269
+ this.hoverable = $();
270
+ this.focusable = $();
271
+
272
+ if ( element !== this ) {
273
+ $.data( element, this.widgetFullName, this );
274
+ this._on( true, this.element, {
275
+ remove: function( event ) {
276
+ if ( event.target === element ) {
277
+ this.destroy();
278
+ }
279
+ }
280
+ });
281
+ this.document = $( element.style ?
282
+ // element within the document
283
+ element.ownerDocument :
284
+ // element is window or document
285
+ element.document || element );
286
+ this.window = $( this.document[0].defaultView || this.document[0].parentWindow );
287
+ }
288
+
289
+ this._create();
290
+ this._trigger( "create", null, this._getCreateEventData() );
291
+ this._init();
292
+ },
293
+ _getCreateOptions: $.noop,
294
+ _getCreateEventData: $.noop,
295
+ _create: $.noop,
296
+ _init: $.noop,
297
+
298
+ destroy: function() {
299
+ this._destroy();
300
+ // we can probably remove the unbind calls in 2.0
301
+ // all event bindings should go through this._on()
302
+ this.element
303
+ .unbind( this.eventNamespace )
304
+ .removeData( this.widgetFullName )
305
+ // support: jquery <1.6.3
306
+ // http://bugs.jquery.com/ticket/9413
307
+ .removeData( $.camelCase( this.widgetFullName ) );
308
+ this.widget()
309
+ .unbind( this.eventNamespace )
310
+ .removeAttr( "aria-disabled" )
311
+ .removeClass(
312
+ this.widgetFullName + "-disabled " +
313
+ "ui-state-disabled" );
314
+
315
+ // clean up events and states
316
+ this.bindings.unbind( this.eventNamespace );
317
+ this.hoverable.removeClass( "ui-state-hover" );
318
+ this.focusable.removeClass( "ui-state-focus" );
319
+ },
320
+ _destroy: $.noop,
321
+
322
+ widget: function() {
323
+ return this.element;
324
+ },
325
+
326
+ option: function( key, value ) {
327
+ var options = key,
328
+ parts,
329
+ curOption,
330
+ i;
331
+
332
+ if ( arguments.length === 0 ) {
333
+ // don't return a reference to the internal hash
334
+ return $.widget.extend( {}, this.options );
335
+ }
336
+
337
+ if ( typeof key === "string" ) {
338
+ // handle nested keys, e.g., "foo.bar" => { foo: { bar: ___ } }
339
+ options = {};
340
+ parts = key.split( "." );
341
+ key = parts.shift();
342
+ if ( parts.length ) {
343
+ curOption = options[ key ] = $.widget.extend( {}, this.options[ key ] );
344
+ for ( i = 0; i < parts.length - 1; i++ ) {
345
+ curOption[ parts[ i ] ] = curOption[ parts[ i ] ] || {};
346
+ curOption = curOption[ parts[ i ] ];
347
+ }
348
+ key = parts.pop();
349
+ if ( arguments.length === 1 ) {
350
+ return curOption[ key ] === undefined ? null : curOption[ key ];
351
+ }
352
+ curOption[ key ] = value;
353
+ } else {
354
+ if ( arguments.length === 1 ) {
355
+ return this.options[ key ] === undefined ? null : this.options[ key ];
356
+ }
357
+ options[ key ] = value;
358
+ }
359
+ }
360
+
361
+ this._setOptions( options );
362
+
363
+ return this;
364
+ },
365
+ _setOptions: function( options ) {
366
+ var key;
367
+
368
+ for ( key in options ) {
369
+ this._setOption( key, options[ key ] );
370
+ }
371
+
372
+ return this;
373
+ },
374
+ _setOption: function( key, value ) {
375
+ this.options[ key ] = value;
376
+
377
+ if ( key === "disabled" ) {
378
+ this.widget()
379
+ .toggleClass( this.widgetFullName + "-disabled", !!value );
380
+
381
+ // If the widget is becoming disabled, then nothing is interactive
382
+ if ( value ) {
383
+ this.hoverable.removeClass( "ui-state-hover" );
384
+ this.focusable.removeClass( "ui-state-focus" );
385
+ }
386
+ }
387
+
388
+ return this;
389
+ },
390
+
391
+ enable: function() {
392
+ return this._setOptions({ disabled: false });
393
+ },
394
+ disable: function() {
395
+ return this._setOptions({ disabled: true });
396
+ },
397
+
398
+ _on: function( suppressDisabledCheck, element, handlers ) {
399
+ var delegateElement,
400
+ instance = this;
401
+
402
+ // no suppressDisabledCheck flag, shuffle arguments
403
+ if ( typeof suppressDisabledCheck !== "boolean" ) {
404
+ handlers = element;
405
+ element = suppressDisabledCheck;
406
+ suppressDisabledCheck = false;
407
+ }
408
+
409
+ // no element argument, shuffle and use this.element
410
+ if ( !handlers ) {
411
+ handlers = element;
412
+ element = this.element;
413
+ delegateElement = this.widget();
414
+ } else {
415
+ element = delegateElement = $( element );
416
+ this.bindings = this.bindings.add( element );
417
+ }
418
+
419
+ $.each( handlers, function( event, handler ) {
420
+ function handlerProxy() {
421
+ // allow widgets to customize the disabled handling
422
+ // - disabled as an array instead of boolean
423
+ // - disabled class as method for disabling individual parts
424
+ if ( !suppressDisabledCheck &&
425
+ ( instance.options.disabled === true ||
426
+ $( this ).hasClass( "ui-state-disabled" ) ) ) {
427
+ return;
428
+ }
429
+ return ( typeof handler === "string" ? instance[ handler ] : handler )
430
+ .apply( instance, arguments );
431
+ }
432
+
433
+ // copy the guid so direct unbinding works
434
+ if ( typeof handler !== "string" ) {
435
+ handlerProxy.guid = handler.guid =
436
+ handler.guid || handlerProxy.guid || $.guid++;
437
+ }
438
+
439
+ var match = event.match( /^([\w:-]*)\s*(.*)$/ ),
440
+ eventName = match[1] + instance.eventNamespace,
441
+ selector = match[2];
442
+ if ( selector ) {
443
+ delegateElement.delegate( selector, eventName, handlerProxy );
444
+ } else {
445
+ element.bind( eventName, handlerProxy );
446
+ }
447
+ });
448
+ },
449
+
450
+ _off: function( element, eventName ) {
451
+ eventName = (eventName || "").split( " " ).join( this.eventNamespace + " " ) + this.eventNamespace;
452
+ element.unbind( eventName ).undelegate( eventName );
453
+ },
454
+
455
+ _delay: function( handler, delay ) {
456
+ function handlerProxy() {
457
+ return ( typeof handler === "string" ? instance[ handler ] : handler )
458
+ .apply( instance, arguments );
459
+ }
460
+ var instance = this;
461
+ return setTimeout( handlerProxy, delay || 0 );
462
+ },
463
+
464
+ _hoverable: function( element ) {
465
+ this.hoverable = this.hoverable.add( element );
466
+ this._on( element, {
467
+ mouseenter: function( event ) {
468
+ $( event.currentTarget ).addClass( "ui-state-hover" );
469
+ },
470
+ mouseleave: function( event ) {
471
+ $( event.currentTarget ).removeClass( "ui-state-hover" );
472
+ }
473
+ });
474
+ },
475
+
476
+ _focusable: function( element ) {
477
+ this.focusable = this.focusable.add( element );
478
+ this._on( element, {
479
+ focusin: function( event ) {
480
+ $( event.currentTarget ).addClass( "ui-state-focus" );
481
+ },
482
+ focusout: function( event ) {
483
+ $( event.currentTarget ).removeClass( "ui-state-focus" );
484
+ }
485
+ });
486
+ },
487
+
488
+ _trigger: function( type, event, data ) {
489
+ var prop, orig,
490
+ callback = this.options[ type ];
491
+
492
+ data = data || {};
493
+ event = $.Event( event );
494
+ event.type = ( type === this.widgetEventPrefix ?
495
+ type :
496
+ this.widgetEventPrefix + type ).toLowerCase();
497
+ // the original event may come from any element
498
+ // so we need to reset the target on the new event
499
+ event.target = this.element[ 0 ];
500
+
501
+ // copy original event properties over to the new event
502
+ orig = event.originalEvent;
503
+ if ( orig ) {
504
+ for ( prop in orig ) {
505
+ if ( !( prop in event ) ) {
506
+ event[ prop ] = orig[ prop ];
507
+ }
508
+ }
509
+ }
510
+
511
+ this.element.trigger( event, data );
512
+ return !( $.isFunction( callback ) &&
513
+ callback.apply( this.element[0], [ event ].concat( data ) ) === false ||
514
+ event.isDefaultPrevented() );
515
+ }
516
+ };
517
+
518
+ $.each( { show: "fadeIn", hide: "fadeOut" }, function( method, defaultEffect ) {
519
+ $.Widget.prototype[ "_" + method ] = function( element, options, callback ) {
520
+ if ( typeof options === "string" ) {
521
+ options = { effect: options };
522
+ }
523
+ var hasOptions,
524
+ effectName = !options ?
525
+ method :
526
+ options === true || typeof options === "number" ?
527
+ defaultEffect :
528
+ options.effect || defaultEffect;
529
+ options = options || {};
530
+ if ( typeof options === "number" ) {
531
+ options = { duration: options };
532
+ }
533
+ hasOptions = !$.isEmptyObject( options );
534
+ options.complete = callback;
535
+ if ( options.delay ) {
536
+ element.delay( options.delay );
537
+ }
538
+ if ( hasOptions && $.effects && $.effects.effect[ effectName ] ) {
539
+ element[ method ]( options );
540
+ } else if ( effectName !== method && element[ effectName ] ) {
541
+ element[ effectName ]( options.duration, options.easing, callback );
542
+ } else {
543
+ element.queue(function( next ) {
544
+ $( this )[ method ]();
545
+ if ( callback ) {
546
+ callback.call( element[ 0 ] );
547
+ }
548
+ next();
549
+ });
550
+ }
551
+ };
552
+ });
553
+
554
+ var widget = $.widget;
555
+
556
+
557
+
558
+ }));