Add Customizer docs.

Props ericlewis.
See #33503.


git-svn-id: https://develop.svn.wordpress.org/trunk@33911 602fd350-edb4-49c9-b593-d223f7449a82
This commit is contained in:
Scott Taylor 2015-09-05 19:52:17 +00:00
parent 26dd8b5b08
commit 38038ccba3
5 changed files with 169 additions and 27 deletions

View File

@ -3,13 +3,22 @@
var Container, focus, api = wp.customize; var Container, focus, api = wp.customize;
/** /**
* A Customizer Setting.
*
* A setting is WordPress data (theme mod, option, menu, etc.) that the user can
* draft changes to in the Customizer.
*
* @see PHP class WP_Customize_Setting.
*
* @class * @class
* @augments wp.customize.Value * @augments wp.customize.Value
* @augments wp.customize.Class * @augments wp.customize.Class
* *
* @param options * @param {object} id The Setting ID.
* - previewer - The Previewer instance to sync with. * @param {object} value The initial value of the setting.
* - transport - The transport to use for previewing. Supports 'refresh' and 'postMessage'. * @param {object} options.previewer The Previewer instance to sync with.
* @param {object} options.transport The transport to use for previewing. Supports 'refresh' and 'postMessage'.
* @param {object} options.dirty
*/ */
api.Setting = api.Value.extend({ api.Setting = api.Value.extend({
initialize: function( id, value, options ) { initialize: function( id, value, options ) {
@ -19,8 +28,13 @@
this.transport = this.transport || 'refresh'; this.transport = this.transport || 'refresh';
this._dirty = options.dirty || false; this._dirty = options.dirty || false;
// Whenever the setting's value changes, refresh the preview.
this.bind( this.preview ); this.bind( this.preview );
}, },
/**
* Refresh the preview, respective of the setting's refresh policy.
*/
preview: function() { preview: function() {
switch ( this.transport ) { switch ( this.transport ) {
case 'refresh': case 'refresh':
@ -270,10 +284,9 @@
}, },
/** /**
* Handle changes to the active state. * Active state change handler.
* *
* This does not change the active state, it merely handles the behavior * Shows the container if it is active, hides it if not.
* for when it does change.
* *
* To override by subclass, update the container's UI to reflect the provided active state. * To override by subclass, update the container's UI to reflect the provided active state.
* *
@ -1354,7 +1367,9 @@
* @param {string} options.params.content The HTML content for the control. * @param {string} options.params.content The HTML content for the control.
* @param {string} options.params.priority Order of priority to show the control within the section. * @param {string} options.params.priority Order of priority to show the control within the section.
* @param {string} options.params.active * @param {string} options.params.active
* @param {string} options.params.section * @param {string} options.params.section The ID of the section the control belongs to.
* @param {string} options.params.settings.default The ID of the setting the control relates to.
* @param {string} options.params.settings.data
* @param {string} options.params.label * @param {string} options.params.label
* @param {string} options.params.description * @param {string} options.params.description
* @param {string} options.params.instanceNumber Order in which this instance was created in relation to other instances. * @param {string} options.params.instanceNumber Order in which this instance was created in relation to other instances.
@ -1420,7 +1435,10 @@
api.utils.bubbleChildValueChanges( control, [ 'section', 'priority', 'active' ] ); api.utils.bubbleChildValueChanges( control, [ 'section', 'priority', 'active' ] );
// Associate this control with its settings when they are created /*
* After all settings related to the control are available,
* make them available on the control and embed the control into the page.
*/
settings = $.map( control.params.settings, function( value ) { settings = $.map( control.params.settings, function( value ) {
return value; return value;
}); });
@ -1437,6 +1455,7 @@
control.embed(); control.embed();
}) ); }) );
// After the control is embedded on the page, invoke the "ready" method.
control.deferred.embedded.done( function () { control.deferred.embedded.done( function () {
control.ready(); control.ready();
}); });
@ -2573,6 +2592,9 @@
api.panel = new api.Values({ defaultConstructor: api.Panel }); api.panel = new api.Values({ defaultConstructor: api.Panel });
/** /**
* An object that fetches a preview in the background of the document, which
* allows for seamless replacement of an existing preview.
*
* @class * @class
* @augments wp.customize.Messenger * @augments wp.customize.Messenger
* @augments wp.customize.Class * @augments wp.customize.Class
@ -2581,10 +2603,22 @@
api.PreviewFrame = api.Messenger.extend({ api.PreviewFrame = api.Messenger.extend({
sensitivity: 2000, sensitivity: 2000,
/**
* Initialize the PreviewFrame.
*
* @param {object} params.container
* @param {object} params.signature
* @param {object} params.previewUrl
* @param {object} params.query
* @param {object} options
*/
initialize: function( params, options ) { initialize: function( params, options ) {
var deferred = $.Deferred(); var deferred = $.Deferred();
// This is the promise object. /*
* Make the instance of the PreviewFrame the promise object
* so other objects can easily interact with it.
*/
deferred.promise( this ); deferred.promise( this );
this.container = params.container; this.container = params.container;
@ -2601,6 +2635,12 @@
this.run( deferred ); this.run( deferred );
}, },
/**
* Run the preview request.
*
* @param {object} deferred jQuery Deferred object to be resolved with
* the request.
*/
run: function( deferred ) { run: function( deferred ) {
var self = this, var self = this,
loaded = false, loaded = false,
@ -2804,9 +2844,13 @@
refreshBuffer: 250, refreshBuffer: 250,
/** /**
* Requires params: * @param {array} params.allowedUrls
* - container - a selector or jQuery element * @param {string} params.container A selector or jQuery element for the preview
* - previewUrl - the URL of preview frame * frame to be placed.
* @param {string} params.form
* @param {string} params.previewUrl The URL to preview.
* @param {string} params.signature
* @param {object} options
*/ */
initialize: function( params, options ) { initialize: function( params, options ) {
var self = this, var self = this,
@ -2919,6 +2963,11 @@
} ); } );
}, },
/**
* Query string data sent with each preview request.
*
* @abstract
*/
query: function() {}, query: function() {},
abort: function() { abort: function() {
@ -2928,6 +2977,9 @@
} }
}, },
/**
* Refresh the preview.
*/
refresh: function() { refresh: function() {
var self = this; var self = this;
@ -3140,6 +3192,11 @@
nonce: api.settings.nonce, nonce: api.settings.nonce,
/**
* Build the query to send along with the Preview request.
*
* @return {object}
*/
query: function() { query: function() {
var dirtyCustomized = {}; var dirtyCustomized = {};
api.each( function ( value, key ) { api.each( function ( value, key ) {
@ -3467,14 +3524,21 @@
} ); } );
} }
// Create a potential postMessage connection with the parent frame. /*
* Create a postMessage connection with a parent frame,
* in case the Customizer frame was opened with the Customize loader.
*
* @see wp.customize.Loader
*/
parent = new api.Messenger({ parent = new api.Messenger({
url: api.settings.url.parent, url: api.settings.url.parent,
channel: 'loader' channel: 'loader'
}); });
// If we receive a 'back' event, we're inside an iframe. /*
// Send any clicks to the 'Return' link to the parent page. * If we receive a 'back' event, we're inside an iframe.
* Send any clicks to the 'Return' link to the parent page.
*/
parent.bind( 'back', function() { parent.bind( 'back', function() {
closeBtn.on( 'click.customize-controls-close', function( event ) { closeBtn.on( 'click.customize-controls-close', function( event ) {
event.preventDefault(); event.preventDefault();
@ -3499,8 +3563,10 @@
}); });
} ); } );
// When activated, let the loader handle redirecting the page. /*
// If no loader exists, redirect the page ourselves (if a url exists). * When activated, let the loader handle redirecting the page.
* If no loader exists, redirect the page ourselves (if a url exists).
*/
api.bind( 'activated', function() { api.bind( 'activated', function() {
if ( parent.targetWindow() ) if ( parent.targetWindow() )
parent.send( 'activated', api.settings.url.activated ); parent.send( 'activated', api.settings.url.activated );

View File

@ -24,6 +24,8 @@ class WP_Customize_Setting {
public $manager; public $manager;
/** /**
* Unique string identifier for the setting.
*
* @access public * @access public
* @var string * @var string
*/ */
@ -74,6 +76,9 @@ class WP_Customize_Setting {
*/ */
public $dirty = false; public $dirty = false;
/**
* @var array
*/
protected $id_data = array(); protected $id_data = array();
/** /**
@ -148,7 +153,8 @@ class WP_Customize_Setting {
protected $_original_value; protected $_original_value;
/** /**
* Handle previewing the setting. * Set up filters for the setting so that the preview request
* will render the drafted changes.
* *
* @since 3.4.0 * @since 3.4.0
*/ */

View File

@ -78,7 +78,9 @@ window.wp = window.wp || {};
/* /*
* If the class has a method called "instance", * If the class has a method called "instance",
* the return value from the class' constructor will be a function that * the return value from the class' constructor will be a function that
* calls invoked, along with all the object properties of the class. * calls the "instance" method.
*
* It is also an object that has properties and methods inside it.
*/ */
if ( this.instance ) { if ( this.instance ) {
magic = function() { magic = function() {
@ -166,6 +168,10 @@ window.wp = window.wp || {};
* @constuctor * @constuctor
*/ */
api.Value = api.Class.extend({ api.Value = api.Class.extend({
/**
* @param {mixed} initial The initial value.
* @param {object} options
*/
initialize: function( initial, options ) { initialize: function( initial, options ) {
this._value = initial; // @todo: potentially change this to a this.set() call. this._value = initial; // @todo: potentially change this to a this.set() call.
this.callbacks = $.Callbacks(); this.callbacks = $.Callbacks();
@ -184,10 +190,20 @@ window.wp = window.wp || {};
return arguments.length ? this.set.apply( this, arguments ) : this.get(); return arguments.length ? this.set.apply( this, arguments ) : this.get();
}, },
/**
* Get the value.
*
* @return {mixed}
*/
get: function() { get: function() {
return this._value; return this._value;
}, },
/**
* Set the value and trigger all bound callbacks.
*
* @param {object} to New value.
*/
set: function( to ) { set: function( to ) {
var from = this._value; var from = this._value;
@ -230,11 +246,21 @@ window.wp = window.wp || {};
return value; return value;
}, },
/**
* Bind a function to be invoked whenever the value changes.
*
* @param {...Function} A function, or multiple functions, to add to the callback stack.
*/
bind: function() { bind: function() {
this.callbacks.add.apply( this.callbacks, arguments ); this.callbacks.add.apply( this.callbacks, arguments );
return this; return this;
}, },
/**
* Unbind a previously bound function.
*
* @param {...Function} A function, or multiple functions, to remove from the callback stack.
*/
unbind: function() { unbind: function() {
this.callbacks.remove.apply( this.callbacks, arguments ); this.callbacks.remove.apply( this.callbacks, arguments );
return this; return this;
@ -283,6 +309,12 @@ window.wp = window.wp || {};
* @mixes wp.customize.Events * @mixes wp.customize.Events
*/ */
api.Values = api.Class.extend({ api.Values = api.Class.extend({
/**
* The default constructor for items of the collection.
*
* @type {object}
*/
defaultConstructor: api.Value, defaultConstructor: api.Value,
initialize: function( options ) { initialize: function( options ) {
@ -347,6 +379,8 @@ window.wp = window.wp || {};
this._value[ id ] = value; this._value[ id ] = value;
value.parent = this; value.parent = this;
// Propagate a 'change' event on an item up to the collection.
if ( value.extended( api.Value ) ) if ( value.extended( api.Value ) )
value.bind( this._change ); value.bind( this._change );
@ -372,6 +406,12 @@ window.wp = window.wp || {};
return this.add( id, new this.defaultConstructor( api.Class.applicator, slice.call( arguments, 1 ) ) ); return this.add( id, new this.defaultConstructor( api.Class.applicator, slice.call( arguments, 1 ) ) );
}, },
/**
* Iterate over all items in the collection invoking the provided callback.
*
* @param {Function} callback Function to invoke.
* @param {object} context Object context to invoke the function with. Optional.
*/
each: function( callback, context ) { each: function( callback, context ) {
context = typeof context === 'undefined' ? this : context; context = typeof context === 'undefined' ? this : context;
@ -453,11 +493,16 @@ window.wp = window.wp || {};
return dfd.promise(); return dfd.promise();
}, },
/**
* A helper function to propagate a 'change' event from an item
* to the collection itself.
*/
_change: function() { _change: function() {
this.parent.trigger( 'change', this ); this.parent.trigger( 'change', this );
} }
}); });
// Create a global events bus on the Customizer.
$.extend( api.Values.prototype, api.Events ); $.extend( api.Values.prototype, api.Events );
@ -570,7 +615,7 @@ window.wp = window.wp || {};
$.support.postMessage = !! window.postMessage; $.support.postMessage = !! window.postMessage;
/** /**
* Messenger for postMessage. * A communicator for sending data from one window to another over postMessage.
* *
* @constuctor * @constuctor
* @augments wp.customize.Class * @augments wp.customize.Class
@ -649,6 +694,11 @@ window.wp = window.wp || {};
$( window ).off( 'message', this.receive ); $( window ).off( 'message', this.receive );
}, },
/**
* Receive data from the other window.
*
* @param {jQuery.Event} event Event with embedded data.
*/
receive: function( event ) { receive: function( event ) {
var message; var message;
@ -679,6 +729,12 @@ window.wp = window.wp || {};
this.trigger( message.id, message.data ); this.trigger( message.id, message.data );
}, },
/**
* Send data to the other window.
*
* @param {string} id The event name.
* @param {object} data Data.
*/
send: function( id, data ) { send: function( id, data ) {
var message; var message;
@ -698,8 +754,14 @@ window.wp = window.wp || {};
// Add the Events mixin to api.Messenger. // Add the Events mixin to api.Messenger.
$.extend( api.Messenger.prototype, api.Events ); $.extend( api.Messenger.prototype, api.Events );
// Core customize object. // The main API object is also a collection of all customizer settings.
api = $.extend( new api.Values(), api ); api = $.extend( new api.Values(), api );
/**
* Get all customize settings.
*
* @return {object}
*/
api.get = function() { api.get = function() {
var result = {}; var result = {};

View File

@ -115,7 +115,11 @@ window.wp = window.wp || {};
this.active = true; this.active = true;
this.body.addClass('customize-loading'); this.body.addClass('customize-loading');
// Dirty state of Customizer in iframe /*
* Track the dirtiness state (whether the drafted changes have been published)
* of the Customizer in the iframe. This is used to decide whether to display
* an AYS alert if the user tries to close the window before saving changes.
*/
this.saved = new api.Value( true ); this.saved = new api.Value( true );
this.iframe = $( '<iframe />', { 'src': src, 'title': Loader.settings.l10n.mainIframeTitle } ).appendTo( this.element ); this.iframe = $( '<iframe />', { 'src': src, 'title': Loader.settings.l10n.mainIframeTitle } ).appendTo( this.element );

View File

@ -111,6 +111,10 @@
api.preview.send( 'documentTitle', document.title ); api.preview.send( 'documentTitle', document.title );
}); });
/*
* Send a message to the parent customize frame with a list of which
* containers and controls are active.
*/
api.preview.send( 'ready', { api.preview.send( 'ready', {
activePanels: api.settings.activePanels, activePanels: api.settings.activePanels,
activeSections: api.settings.activeSections, activeSections: api.settings.activeSections,