]> scripts.mit.edu Git - autoinstalls/wordpress.git/blob - wp-admin/js/customize-controls.js
WordPress 3.7.1
[autoinstalls/wordpress.git] / wp-admin / js / customize-controls.js
1 (function( exports, $ ){
2         var api = wp.customize;
3
4         /*
5          * @param options
6          * - previewer - The Previewer instance to sync with.
7          * - transport - The transport to use for previewing. Supports 'refresh' and 'postMessage'.
8          */
9         api.Setting = api.Value.extend({
10                 initialize: function( id, value, options ) {
11                         var element;
12
13                         api.Value.prototype.initialize.call( this, value, options );
14
15                         this.id = id;
16                         this.transport = this.transport || 'refresh';
17
18                         this.bind( this.preview );
19                 },
20                 preview: function() {
21                         switch ( this.transport ) {
22                                 case 'refresh':
23                                         return this.previewer.refresh();
24                                 case 'postMessage':
25                                         return this.previewer.send( 'setting', [ this.id, this() ] );
26                         }
27                 }
28         });
29
30         api.Control = api.Class.extend({
31                 initialize: function( id, options ) {
32                         var control = this,
33                                 nodes, radios, settings;
34
35                         this.params = {};
36                         $.extend( this, options || {} );
37
38                         this.id = id;
39                         this.selector = '#customize-control-' + id.replace( /\]/g, '' ).replace( /\[/g, '-' );
40                         this.container = $( this.selector );
41
42                         settings = $.map( this.params.settings, function( value ) {
43                                 return value;
44                         });
45
46                         api.apply( api, settings.concat( function() {
47                                 var key;
48
49                                 control.settings = {};
50                                 for ( key in control.params.settings ) {
51                                         control.settings[ key ] = api( control.params.settings[ key ] );
52                                 }
53
54                                 control.setting = control.settings['default'] || null;
55                                 control.ready();
56                         }) );
57
58                         control.elements = [];
59
60                         nodes  = this.container.find('[data-customize-setting-link]');
61                         radios = {};
62
63                         nodes.each( function() {
64                                 var node = $(this),
65                                         name;
66
67                                 if ( node.is(':radio') ) {
68                                         name = node.prop('name');
69                                         if ( radios[ name ] )
70                                                 return;
71
72                                         radios[ name ] = true;
73                                         node = nodes.filter( '[name="' + name + '"]' );
74                                 }
75
76                                 api( node.data('customizeSettingLink'), function( setting ) {
77                                         var element = new api.Element( node );
78                                         control.elements.push( element );
79                                         element.sync( setting );
80                                         element.set( setting() );
81                                 });
82                         });
83                 },
84
85                 ready: function() {},
86
87                 dropdownInit: function() {
88                         var control  = this,
89                                 statuses = this.container.find('.dropdown-status'),
90                                 params   = this.params,
91                                 update   = function( to ) {
92                                         if ( typeof     to === 'string' && params.statuses && params.statuses[ to ] )
93                                                 statuses.html( params.statuses[ to ] ).show();
94                                         else
95                                                 statuses.hide();
96                                 };
97
98                         var toggleFreeze = false;
99
100                         // Support the .dropdown class to open/close complex elements
101                         this.container.on( 'click keydown', '.dropdown', function( event ) {
102                                 if ( event.type === 'keydown' &&  13 !== event.which ) // enter
103                                         return;
104
105                                 event.preventDefault();
106
107                                 if (!toggleFreeze)
108                                         control.container.toggleClass('open');
109
110                                 if ( control.container.hasClass('open') )
111                                         control.container.parent().parent().find('li.library-selected').focus();
112
113                                 // Don't want to fire focus and click at same time
114                                 toggleFreeze = true;
115                                 setTimeout(function () {
116                                         toggleFreeze = false;
117                                 }, 400);
118                         });
119
120                         this.setting.bind( update );
121                         update( this.setting() );
122                 }
123         });
124
125         api.ColorControl = api.Control.extend({
126                 ready: function() {
127                         var control = this,
128                                 picker = this.container.find('.color-picker-hex');
129
130                         picker.val( control.setting() ).wpColorPicker({
131                                 change: function( event, options ) {
132                                         control.setting.set( picker.wpColorPicker('color') );
133                                 },
134                                 clear: function() {
135                                         control.setting.set( false );
136                                 }
137                         });
138                 }
139         });
140
141         api.UploadControl = api.Control.extend({
142                 ready: function() {
143                         var control = this;
144
145                         this.params.removed = this.params.removed || '';
146
147                         this.success = $.proxy( this.success, this );
148
149                         this.uploader = $.extend({
150                                 container: this.container,
151                                 browser:   this.container.find('.upload'),
152                                 dropzone:  this.container.find('.upload-dropzone'),
153                                 success:   this.success,
154                                 plupload:  {},
155                                 params:    {}
156                         }, this.uploader || {} );
157
158                         if ( control.params.extensions ) {
159                                 control.uploader.plupload.filters = [{
160                                         title:      api.l10n.allowedFiles,
161                                         extensions: control.params.extensions
162                                 }];
163                         }
164
165                         if ( control.params.context )
166                                 control.uploader.params['post_data[context]'] = this.params.context;
167
168                         if ( api.settings.theme.stylesheet )
169                                 control.uploader.params['post_data[theme]'] = api.settings.theme.stylesheet;
170
171                         this.uploader = new wp.Uploader( this.uploader );
172
173                         this.remover = this.container.find('.remove');
174                         this.remover.on( 'click keydown', function( event ) {
175                                 if ( event.type === 'keydown' &&  13 !== event.which ) // enter
176                                         return;
177
178                                 control.setting.set( control.params.removed );
179                                 event.preventDefault();
180                         });
181
182                         this.removerVisibility = $.proxy( this.removerVisibility, this );
183                         this.setting.bind( this.removerVisibility );
184                         this.removerVisibility( this.setting.get() );
185                 },
186                 success: function( attachment ) {
187                         this.setting.set( attachment.get('url') );
188                 },
189                 removerVisibility: function( to ) {
190                         this.remover.toggle( to != this.params.removed );
191                 }
192         });
193
194         api.ImageControl = api.UploadControl.extend({
195                 ready: function() {
196                         var control = this,
197                                 panels;
198
199                         this.uploader = {
200                                 init: function( up ) {
201                                         var fallback, button;
202
203                                         if ( this.supports.dragdrop )
204                                                 return;
205
206                                         // Maintain references while wrapping the fallback button.
207                                         fallback = control.container.find( '.upload-fallback' );
208                                         button   = fallback.children().detach();
209
210                                         this.browser.detach().empty().append( button );
211                                         fallback.append( this.browser ).show();
212                                 }
213                         };
214
215                         api.UploadControl.prototype.ready.call( this );
216
217                         this.thumbnail    = this.container.find('.preview-thumbnail img');
218                         this.thumbnailSrc = $.proxy( this.thumbnailSrc, this );
219                         this.setting.bind( this.thumbnailSrc );
220
221                         this.library = this.container.find('.library');
222
223                         // Generate tab objects
224                         this.tabs = {};
225                         panels    = this.library.find('.library-content');
226
227                         this.library.children('ul').children('li').each( function() {
228                                 var link  = $(this),
229                                         id    = link.data('customizeTab'),
230                                         panel = panels.filter('[data-customize-tab="' + id + '"]');
231
232                                 control.tabs[ id ] = {
233                                         both:  link.add( panel ),
234                                         link:  link,
235                                         panel: panel
236                                 };
237                         });
238
239                         // Bind tab switch events
240                         this.library.children('ul').on( 'click keydown', 'li', function( event ) {
241                                 if ( event.type === 'keydown' &&  13 !== event.which ) // enter
242                                         return;
243
244                                 var id  = $(this).data('customizeTab'),
245                                         tab = control.tabs[ id ];
246
247                                 event.preventDefault();
248
249                                 if ( tab.link.hasClass('library-selected') )
250                                         return;
251
252                                 control.selected.both.removeClass('library-selected');
253                                 control.selected = tab;
254                                 control.selected.both.addClass('library-selected');
255                         });
256
257                         // Bind events to switch image urls.
258                         this.library.on( 'click keydown', 'a', function( event ) {
259                                 if ( event.type === 'keydown' && 13 !== event.which ) // enter
260                                         return;
261
262                                 var value = $(this).data('customizeImageValue');
263
264                                 if ( value ) {
265                                         control.setting.set( value );
266                                         event.preventDefault();
267                                 }
268                         });
269
270                         if ( this.tabs.uploaded ) {
271                                 this.tabs.uploaded.target = this.library.find('.uploaded-target');
272                                 if ( ! this.tabs.uploaded.panel.find('.thumbnail').length )
273                                         this.tabs.uploaded.both.addClass('hidden');
274                         }
275
276                         // Select a tab
277                         panels.each( function() {
278                                 var tab = control.tabs[ $(this).data('customizeTab') ];
279
280                                 // Select the first visible tab.
281                                 if ( ! tab.link.hasClass('hidden') ) {
282                                         control.selected = tab;
283                                         tab.both.addClass('library-selected');
284                                         return false;
285                                 }
286                         });
287
288                         this.dropdownInit();
289                 },
290                 success: function( attachment ) {
291                         api.UploadControl.prototype.success.call( this, attachment );
292
293                         // Add the uploaded image to the uploaded tab.
294                         if ( this.tabs.uploaded && this.tabs.uploaded.target.length ) {
295                                 this.tabs.uploaded.both.removeClass('hidden');
296
297                                 // @todo: Do NOT store this on the attachment model. That is bad.
298                                 attachment.element = $( '<a href="#" class="thumbnail"></a>' )
299                                         .data( 'customizeImageValue', attachment.get('url') )
300                                         .append( '<img src="' +  attachment.get('url')+ '" />' )
301                                         .appendTo( this.tabs.uploaded.target );
302                         }
303                 },
304                 thumbnailSrc: function( to ) {
305                         if ( /^(https?:)?\/\//.test( to ) )
306                                 this.thumbnail.prop( 'src', to ).show();
307                         else
308                                 this.thumbnail.hide();
309                 }
310         });
311
312         // Change objects contained within the main customize object to Settings.
313         api.defaultConstructor = api.Setting;
314
315         // Create the collection of Control objects.
316         api.control = new api.Values({ defaultConstructor: api.Control });
317
318         api.PreviewFrame = api.Messenger.extend({
319                 sensitivity: 2000,
320
321                 initialize: function( params, options ) {
322                         var deferred = $.Deferred(),
323                                 self     = this;
324
325                         // This is the promise object.
326                         deferred.promise( this );
327
328                         this.container = params.container;
329                         this.signature = params.signature;
330
331                         $.extend( params, { channel: api.PreviewFrame.uuid() });
332
333                         api.Messenger.prototype.initialize.call( this, params, options );
334
335                         this.add( 'previewUrl', params.previewUrl );
336
337                         this.query = $.extend( params.query || {}, { customize_messenger_channel: this.channel() });
338
339                         this.run( deferred );
340                 },
341
342                 run: function( deferred ) {
343                         var self   = this,
344                                 loaded = false,
345                                 ready  = false;
346
347                         if ( this._ready )
348                                 this.unbind( 'ready', this._ready );
349
350                         this._ready = function() {
351                                 ready = true;
352
353                                 if ( loaded )
354                                         deferred.resolveWith( self );
355                         };
356
357                         this.bind( 'ready', this._ready );
358
359                         this.request = $.ajax( this.previewUrl(), {
360                                 type: 'POST',
361                                 data: this.query,
362                                 xhrFields: {
363                                         withCredentials: true
364                                 }
365                         } );
366
367                         this.request.fail( function() {
368                                 deferred.rejectWith( self, [ 'request failure' ] );
369                         });
370
371                         this.request.done( function( response ) {
372                                 var location = self.request.getResponseHeader('Location'),
373                                         signature = self.signature,
374                                         index;
375
376                                 // Check if the location response header differs from the current URL.
377                                 // If so, the request was redirected; try loading the requested page.
378                                 if ( location && location != self.previewUrl() ) {
379                                         deferred.rejectWith( self, [ 'redirect', location ] );
380                                         return;
381                                 }
382
383                                 // Check if the user is not logged in.
384                                 if ( '0' === response ) {
385                                         self.login( deferred );
386                                         return;
387                                 }
388
389                                 // Check for cheaters.
390                                 if ( '-1' === response ) {
391                                         deferred.rejectWith( self, [ 'cheatin' ] );
392                                         return;
393                                 }
394
395                                 // Check for a signature in the request.
396                                 index = response.lastIndexOf( signature );
397                                 if ( -1 === index || index < response.lastIndexOf('</html>') ) {
398                                         deferred.rejectWith( self, [ 'unsigned' ] );
399                                         return;
400                                 }
401
402                                 // Strip the signature from the request.
403                                 response = response.slice( 0, index ) + response.slice( index + signature.length );
404
405                                 // Create the iframe and inject the html content.
406                                 self.iframe = $('<iframe />').appendTo( self.container );
407
408                                 // Bind load event after the iframe has been added to the page;
409                                 // otherwise it will fire when injected into the DOM.
410                                 self.iframe.one( 'load', function() {
411                                         loaded = true;
412
413                                         if ( ready ) {
414                                                 deferred.resolveWith( self );
415                                         } else {
416                                                 setTimeout( function() {
417                                                         deferred.rejectWith( self, [ 'ready timeout' ] );
418                                                 }, self.sensitivity );
419                                         }
420                                 });
421
422                                 self.targetWindow( self.iframe[0].contentWindow );
423
424                                 self.targetWindow().document.open();
425                                 self.targetWindow().document.write( response );
426                                 self.targetWindow().document.close();
427                         });
428                 },
429
430                 login: function( deferred ) {
431                         var self = this,
432                                 reject;
433
434                         reject = function() {
435                                 deferred.rejectWith( self, [ 'logged out' ] );
436                         };
437
438                         if ( this.triedLogin )
439                                 return reject();
440
441                         // Check if we have an admin cookie.
442                         $.get( api.settings.url.ajax, {
443                                 action: 'logged-in'
444                         }).fail( reject ).done( function( response ) {
445                                 var iframe;
446
447                                 if ( '1' !== response )
448                                         reject();
449
450                                 iframe = $('<iframe src="' + self.previewUrl() + '" />').hide();
451                                 iframe.appendTo( self.container );
452                                 iframe.load( function() {
453                                         self.triedLogin = true;
454
455                                         iframe.remove();
456                                         self.run( deferred );
457                                 });
458                         });
459                 },
460
461                 destroy: function() {
462                         api.Messenger.prototype.destroy.call( this );
463                         this.request.abort();
464
465                         if ( this.iframe )
466                                 this.iframe.remove();
467
468                         delete this.request;
469                         delete this.iframe;
470                         delete this.targetWindow;
471                 }
472         });
473
474         (function(){
475                 var uuid = 0;
476                 api.PreviewFrame.uuid = function() {
477                         return 'preview-' + uuid++;
478                 };
479         }());
480
481         api.Previewer = api.Messenger.extend({
482                 refreshBuffer: 250,
483
484                 /**
485                  * Requires params:
486                  *  - container  - a selector or jQuery element
487                  *  - previewUrl - the URL of preview frame
488                  */
489                 initialize: function( params, options ) {
490                         var self = this,
491                                 rscheme = /^https?/,
492                                 url;
493
494                         $.extend( this, options || {} );
495
496                         /*
497                          * Wrap this.refresh to prevent it from hammering the servers:
498                          *
499                          * If refresh is called once and no other refresh requests are
500                          * loading, trigger the request immediately.
501                          *
502                          * If refresh is called while another refresh request is loading,
503                          * debounce the refresh requests:
504                          * 1. Stop the loading request (as it is instantly outdated).
505                          * 2. Trigger the new request once refresh hasn't been called for
506                          *    self.refreshBuffer milliseconds.
507                          */
508                         this.refresh = (function( self ) {
509                                 var refresh  = self.refresh,
510                                         callback = function() {
511                                                 timeout = null;
512                                                 refresh.call( self );
513                                         },
514                                         timeout;
515
516                                 return function() {
517                                         if ( typeof timeout !== 'number' ) {
518                                                 if ( self.loading ) {
519                                                         self.abort();
520                                                 } else {
521                                                         return callback();
522                                                 }
523                                         }
524
525                                         clearTimeout( timeout );
526                                         timeout = setTimeout( callback, self.refreshBuffer );
527                                 };
528                         })( this );
529
530                         this.container   = api.ensure( params.container );
531                         this.allowedUrls = params.allowedUrls;
532                         this.signature   = params.signature;
533
534                         params.url = window.location.href;
535
536                         api.Messenger.prototype.initialize.call( this, params );
537
538                         this.add( 'scheme', this.origin() ).link( this.origin ).setter( function( to ) {
539                                 var match = to.match( rscheme );
540                                 return match ? match[0] : '';
541                         });
542
543                         // Limit the URL to internal, front-end links.
544                         //
545                         // If the frontend and the admin are served from the same domain, load the
546                         // preview over ssl if the customizer is being loaded over ssl. This avoids
547                         // insecure content warnings. This is not attempted if the admin and frontend
548                         // are on different domains to avoid the case where the frontend doesn't have
549                         // ssl certs.
550
551                         this.add( 'previewUrl', params.previewUrl ).setter( function( to ) {
552                                 var result;
553
554                                 // Check for URLs that include "/wp-admin/" or end in "/wp-admin".
555                                 // Strip hashes and query strings before testing.
556                                 if ( /\/wp-admin(\/|$)/.test( to.replace( /[#?].*$/, '' ) ) )
557                                         return null;
558
559                                 // Attempt to match the URL to the control frame's scheme
560                                 // and check if it's allowed. If not, try the original URL.
561                                 $.each([ to.replace( rscheme, self.scheme() ), to ], function( i, url ) {
562                                         $.each( self.allowedUrls, function( i, allowed ) {
563                                                 var path;
564
565                                                 allowed = allowed.replace( /\/+$/, '' );
566                                                 path = url.replace( allowed, '' );
567
568                                                 if ( 0 === url.indexOf( allowed ) && /^([/#?]|$)/.test( path ) ) {
569                                                         result = url;
570                                                         return false;
571                                                 }
572                                         });
573                                         if ( result )
574                                                 return false;
575                                 });
576
577                                 // If we found a matching result, return it. If not, bail.
578                                 return result ? result : null;
579                         });
580
581                         // Refresh the preview when the URL is changed (but not yet).
582                         this.previewUrl.bind( this.refresh );
583
584                         this.scroll = 0;
585                         this.bind( 'scroll', function( distance ) {
586                                 this.scroll = distance;
587                         });
588
589                         // Update the URL when the iframe sends a URL message.
590                         this.bind( 'url', this.previewUrl );
591                 },
592
593                 query: function() {},
594
595                 abort: function() {
596                         if ( this.loading ) {
597                                 this.loading.destroy();
598                                 delete this.loading;
599                         }
600                 },
601
602                 refresh: function() {
603                         var self = this;
604
605                         this.abort();
606
607                         this.loading = new api.PreviewFrame({
608                                 url:        this.url(),
609                                 previewUrl: this.previewUrl(),
610                                 query:      this.query() || {},
611                                 container:  this.container,
612                                 signature:  this.signature
613                         });
614
615                         this.loading.done( function() {
616                                 // 'this' is the loading frame
617                                 this.bind( 'synced', function() {
618                                         if ( self.preview )
619                                                 self.preview.destroy();
620                                         self.preview = this;
621                                         delete self.loading;
622
623                                         self.targetWindow( this.targetWindow() );
624                                         self.channel( this.channel() );
625
626                                         self.send( 'active' );
627                                 });
628
629                                 this.send( 'sync', {
630                                         scroll:   self.scroll,
631                                         settings: api.get()
632                                 });
633                         });
634
635                         this.loading.fail( function( reason, location ) {
636                                 if ( 'redirect' === reason && location )
637                                         self.previewUrl( location );
638
639                                 if ( 'logged out' === reason ) {
640                                         if ( self.preview ) {
641                                                 self.preview.destroy();
642                                                 delete self.preview;
643                                         }
644
645                                         self.login().done( self.refresh );
646                                 }
647
648                                 if ( 'cheatin' === reason )
649                                         self.cheatin();
650                         });
651                 },
652
653                 login: function() {
654                         var previewer = this,
655                                 deferred, messenger, iframe;
656
657                         if ( this._login )
658                                 return this._login;
659
660                         deferred = $.Deferred();
661                         this._login = deferred.promise();
662
663                         messenger = new api.Messenger({
664                                 channel: 'login',
665                                 url:     api.settings.url.login
666                         });
667
668                         iframe = $('<iframe src="' + api.settings.url.login + '" />').appendTo( this.container );
669
670                         messenger.targetWindow( iframe[0].contentWindow );
671
672                         messenger.bind( 'login', function() {
673                                 iframe.remove();
674                                 messenger.destroy();
675                                 delete previewer._login;
676                                 deferred.resolve();
677                         });
678
679                         return this._login;
680                 },
681
682                 cheatin: function() {
683                         $( document.body ).empty().addClass('cheatin').append( '<p>' + api.l10n.cheatin + '</p>' );
684                 }
685         });
686
687         /* =====================================================================
688          * Ready.
689          * ===================================================================== */
690
691         api.controlConstructor = {
692                 color:  api.ColorControl,
693                 upload: api.UploadControl,
694                 image:  api.ImageControl
695         };
696
697         $( function() {
698                 api.settings = window._wpCustomizeSettings;
699                 api.l10n = window._wpCustomizeControlsL10n;
700
701                 // Check if we can run the customizer.
702                 if ( ! api.settings )
703                         return;
704
705                 // Redirect to the fallback preview if any incompatibilities are found.
706                 if ( ! $.support.postMessage || ( ! $.support.cors && api.settings.isCrossDomain ) )
707                         return window.location = api.settings.url.fallback;
708
709                 var body = $( document.body ),
710                         overlay = body.children('.wp-full-overlay'),
711                         query, previewer, parent;
712
713                 // Prevent the form from saving when enter is pressed.
714                 $('#customize-controls').on( 'keydown', function( e ) {
715                         if ( $( e.target ).is('textarea') )
716                                 return;
717
718                         if ( 13 === e.which ) // Enter
719                                 e.preventDefault();
720                 });
721
722                 // Initialize Previewer
723                 previewer = new api.Previewer({
724                         container:   '#customize-preview',
725                         form:        '#customize-controls',
726                         previewUrl:  api.settings.url.preview,
727                         allowedUrls: api.settings.url.allowed,
728                         signature:   'WP_CUSTOMIZER_SIGNATURE'
729                 }, {
730
731                         nonce: api.settings.nonce,
732
733                         query: function() {
734                                 return {
735                                         wp_customize: 'on',
736                                         theme:        api.settings.theme.stylesheet,
737                                         customized:   JSON.stringify( api.get() ),
738                                         nonce:        this.nonce.preview
739                                 };
740                         },
741
742                         save: function() {
743                                 var self  = this,
744                                         query = $.extend( this.query(), {
745                                                 action: 'customize_save',
746                                                 nonce:  this.nonce.save
747                                         }),
748                                         request = $.post( api.settings.url.ajax, query );
749
750                                 api.trigger( 'save', request );
751
752                                 body.addClass('saving');
753
754                                 request.always( function() {
755                                         body.removeClass('saving');
756                                 });
757
758                                 request.done( function( response ) {
759                                         // Check if the user is logged out.
760                                         if ( '0' === response ) {
761                                                 self.preview.iframe.hide();
762                                                 self.login().done( function() {
763                                                         self.save();
764                                                         self.preview.iframe.show();
765                                                 });
766                                                 return;
767                                         }
768
769                                         // Check for cheaters.
770                                         if ( '-1' === response ) {
771                                                 self.cheatin();
772                                                 return;
773                                         }
774
775                                         api.trigger( 'saved' );
776                                 });
777                         }
778                 });
779
780                 // Refresh the nonces if the preview sends updated nonces over.
781                 previewer.bind( 'nonce', function( nonce ) {
782                         $.extend( this.nonce, nonce );
783                 });
784
785                 $.each( api.settings.settings, function( id, data ) {
786                         api.create( id, id, data.value, {
787                                 transport: data.transport,
788                                 previewer: previewer
789                         } );
790                 });
791
792                 $.each( api.settings.controls, function( id, data ) {
793                         var constructor = api.controlConstructor[ data.type ] || api.Control,
794                                 control;
795
796                         control = api.control.add( id, new constructor( id, {
797                                 params: data,
798                                 previewer: previewer
799                         } ) );
800                 });
801
802                 // Check if preview url is valid and load the preview frame.
803                 if ( previewer.previewUrl() )
804                         previewer.refresh();
805                 else
806                         previewer.previewUrl( api.settings.url.home );
807
808                 // Save and activated states
809                 (function() {
810                         var state = new api.Values(),
811                                 saved = state.create('saved'),
812                                 activated = state.create('activated');
813
814                         state.bind( 'change', function() {
815                                 var save = $('#save'),
816                                         back = $('.back');
817
818                                 if ( ! activated() ) {
819                                         save.val( api.l10n.activate ).prop( 'disabled', false );
820                                         back.text( api.l10n.cancel );
821
822                                 } else if ( saved() ) {
823                                         save.val( api.l10n.saved ).prop( 'disabled', true );
824                                         back.text( api.l10n.close );
825
826                                 } else {
827                                         save.val( api.l10n.save ).prop( 'disabled', false );
828                                         back.text( api.l10n.cancel );
829                                 }
830                         });
831
832                         // Set default states.
833                         saved( true );
834                         activated( api.settings.theme.active );
835
836                         api.bind( 'change', function() {
837                                 state('saved').set( false );
838                         });
839
840                         api.bind( 'saved', function() {
841                                 state('saved').set( true );
842                                 state('activated').set( true );
843                         });
844
845                         activated.bind( function( to ) {
846                                 if ( to )
847                                         api.trigger( 'activated' );
848                         });
849
850                         // Expose states to the API.
851                         api.state = state;
852                 }());
853
854                 // Button bindings.
855                 $('#save').click( function( event ) {
856                         previewer.save();
857                         event.preventDefault();
858                 }).keydown( function( event ) {
859                         if ( 9 === event.which ) // tab
860                                 return;
861                         if ( 13 === event.which ) // enter
862                                 previewer.save();
863                         event.preventDefault();
864                 });
865
866                 $('.back').keydown( function( event ) {
867                         if ( 9 === event.which ) // tab
868                                 return;
869                         if ( 13 === event.which ) // enter
870                                 this.click();
871                         event.preventDefault();
872                 });
873
874                 $('.upload-dropzone a.upload').keydown( function( event ) {
875                         if ( 13 === event.which ) // enter
876                                 this.click();
877                 });
878
879                 $('.collapse-sidebar').on( 'click keydown', function( event ) {
880                         if ( event.type === 'keydown' &&  13 !== event.which ) // enter
881                                 return;
882
883                         overlay.toggleClass( 'collapsed' ).toggleClass( 'expanded' );
884                         event.preventDefault();
885                 });
886
887                 // Create a potential postMessage connection with the parent frame.
888                 parent = new api.Messenger({
889                         url: api.settings.url.parent,
890                         channel: 'loader'
891                 });
892
893                 // If we receive a 'back' event, we're inside an iframe.
894                 // Send any clicks to the 'Return' link to the parent page.
895                 parent.bind( 'back', function() {
896                         $('.back').on( 'click.back', function( event ) {
897                                 event.preventDefault();
898                                 parent.send( 'close' );
899                         });
900                 });
901
902                 // Pass events through to the parent.
903                 api.bind( 'saved', function() {
904                         parent.send( 'saved' );
905                 });
906
907                 // When activated, let the loader handle redirecting the page.
908                 // If no loader exists, redirect the page ourselves (if a url exists).
909                 api.bind( 'activated', function() {
910                         if ( parent.targetWindow() )
911                                 parent.send( 'activated', api.settings.url.activated );
912                         else if ( api.settings.url.activated )
913                                 window.location = api.settings.url.activated;
914                 });
915
916                 // Initialize the connection with the parent frame.
917                 parent.send( 'ready' );
918
919                 // Control visibility for default controls
920                 $.each({
921                         'background_image': {
922                                 controls: [ 'background_repeat', 'background_position_x', 'background_attachment' ],
923                                 callback: function( to ) { return !! to }
924                         },
925                         'show_on_front': {
926                                 controls: [ 'page_on_front', 'page_for_posts' ],
927                                 callback: function( to ) { return 'page' === to }
928                         },
929                         'header_textcolor': {
930                                 controls: [ 'header_textcolor' ],
931                                 callback: function( to ) { return 'blank' !== to }
932                         }
933                 }, function( settingId, o ) {
934                         api( settingId, function( setting ) {
935                                 $.each( o.controls, function( i, controlId ) {
936                                         api.control( controlId, function( control ) {
937                                                 var visibility = function( to ) {
938                                                         control.container.toggle( o.callback( to ) );
939                                                 };
940
941                                                 visibility( setting.get() );
942                                                 setting.bind( visibility );
943                                         });
944                                 });
945                         });
946                 });
947
948                 // Juggle the two controls that use header_textcolor
949                 api.control( 'display_header_text', function( control ) {
950                         var last = '';
951
952                         control.elements[0].unsync( api( 'header_textcolor' ) );
953
954                         control.element = new api.Element( control.container.find('input') );
955                         control.element.set( 'blank' !== control.setting() );
956
957                         control.element.bind( function( to ) {
958                                 if ( ! to )
959                                         last = api( 'header_textcolor' ).get();
960
961                                 control.setting.set( to ? last : 'blank' );
962                         });
963
964                         control.setting.bind( function( to ) {
965                                 control.element.set( 'blank' !== to );
966                         });
967                 });
968
969                 // Handle header image data
970                 api.control( 'header_image', function( control ) {
971                         control.setting.bind( function( to ) {
972                                 if ( to === control.params.removed )
973                                         control.settings.data.set( false );
974                         });
975
976                         control.library.on( 'click', 'a', function( event ) {
977                                 control.settings.data.set( $(this).data('customizeHeaderImageData') );
978                         });
979
980                         control.uploader.success = function( attachment ) {
981                                 var data;
982
983                                 api.ImageControl.prototype.success.call( control, attachment );
984
985                                 data = {
986                                         attachment_id: attachment.get('id'),
987                                         url:           attachment.get('url'),
988                                         thumbnail_url: attachment.get('url'),
989                                         height:        attachment.get('height'),
990                                         width:         attachment.get('width')
991                                 };
992
993                                 attachment.element.data( 'customizeHeaderImageData', data );
994                                 control.settings.data.set( data );
995                         };
996                 });
997
998                 api.trigger( 'ready' );
999
1000                 // Make sure left column gets focus
1001                 var topFocus = $('.back');
1002                 topFocus.focus();
1003                 setTimeout(function () {
1004                         topFocus.focus();
1005                 }, 200);
1006
1007         });
1008
1009 })( wp, jQuery );