2015-02-17 12:58:36 -08:00
|
|
|
/*global define*/
|
|
|
|
define(
|
|
|
|
[],
|
|
|
|
function () {
|
|
|
|
"use strict";
|
|
|
|
|
2015-02-17 13:47:14 -08:00
|
|
|
// Utility functions for reducing truth arrays
|
|
|
|
function and(a, b) { return a && b; }
|
|
|
|
function or(a, b) { return a || b; }
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Provides initial structure and state (as suitable for provision
|
|
|
|
* to the `mct-toolbar` directive) for a view's tool bar, based on
|
|
|
|
* that view's declaration of what belongs in its tool bar and on
|
|
|
|
* the current selection.
|
|
|
|
*
|
|
|
|
* @param structure toolbar structure, as provided by view definition
|
2015-02-18 20:12:55 -08:00
|
|
|
* @param {Function} commit callback to invoke after changes
|
2015-02-17 13:47:14 -08:00
|
|
|
* @constructor
|
|
|
|
*/
|
2015-02-23 15:57:45 -08:00
|
|
|
function EditToolbar(structure, commit) {
|
2015-02-17 13:47:14 -08:00
|
|
|
var toolbarStructure = Object.create(structure || {}),
|
|
|
|
toolbarState,
|
2015-02-23 15:57:45 -08:00
|
|
|
selection,
|
2015-02-17 13:47:14 -08:00
|
|
|
properties = [];
|
|
|
|
|
|
|
|
// Generate a new key for an item's property
|
|
|
|
function addKey(property) {
|
|
|
|
properties.push(property);
|
|
|
|
return properties.length - 1; // Return index of property
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update value for this property in all elements of the
|
|
|
|
// selection which have this property.
|
|
|
|
function updateProperties(property, value) {
|
2015-02-24 09:47:00 -08:00
|
|
|
var changed = false;
|
|
|
|
|
2015-02-17 13:47:14 -08:00
|
|
|
// Update property in a selected element
|
|
|
|
function updateProperty(selected) {
|
|
|
|
// Ignore selected elements which don't have this property
|
|
|
|
if (selected[property] !== undefined) {
|
|
|
|
// Check if this is a setter, or just assignable
|
|
|
|
if (typeof selected[property] === 'function') {
|
2015-02-24 09:47:00 -08:00
|
|
|
changed =
|
|
|
|
changed || (selected[property]() !== value);
|
2015-02-17 13:47:14 -08:00
|
|
|
selected[property](value);
|
|
|
|
} else {
|
2015-02-24 09:47:00 -08:00
|
|
|
changed =
|
|
|
|
changed || (selected[property] !== value);
|
2015-02-17 13:47:14 -08:00
|
|
|
selected[property] = value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update property in all selected elements
|
|
|
|
selection.forEach(updateProperty);
|
2015-02-24 09:47:00 -08:00
|
|
|
|
|
|
|
// Return whether or not anything changed
|
|
|
|
return changed;
|
2015-02-17 13:47:14 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Look up the current value associated with a property
|
|
|
|
// in selection i
|
|
|
|
function lookupState(property, selected) {
|
|
|
|
var value = selected[property];
|
|
|
|
return (typeof value === 'function') ? value() : value;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get initial value for a given property
|
|
|
|
function initializeState(property) {
|
|
|
|
var result;
|
|
|
|
// Look through all selections for this property;
|
|
|
|
// values should all match by the time we perform
|
|
|
|
// this lookup anyway.
|
|
|
|
selection.forEach(function (selected) {
|
|
|
|
result = (selected[property] !== undefined) ?
|
|
|
|
lookupState(property, selected) :
|
|
|
|
result;
|
|
|
|
});
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if all elements of the selection which have this
|
|
|
|
// property have the same value for this property.
|
|
|
|
function isConsistent(property) {
|
|
|
|
var consistent = true,
|
|
|
|
observed = false,
|
|
|
|
state;
|
|
|
|
|
|
|
|
// Check if a given element of the selection is consistent
|
|
|
|
// with previously-observed elements for this property.
|
|
|
|
function checkConsistency(selected) {
|
|
|
|
var next;
|
|
|
|
// Ignore selections which don't have this property
|
|
|
|
if (selected[property] !== undefined) {
|
|
|
|
// Look up state of this element in the selection
|
|
|
|
next = lookupState(property, selected);
|
|
|
|
// Detect inconsistency
|
|
|
|
if (observed) {
|
|
|
|
consistent = consistent && (next === state);
|
|
|
|
}
|
|
|
|
// Track state for next iteration
|
|
|
|
state = next;
|
|
|
|
observed = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iterate through selections
|
|
|
|
selection.forEach(checkConsistency);
|
|
|
|
|
|
|
|
return consistent;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Used to filter out items which are applicable (or not)
|
|
|
|
// to the current selection.
|
|
|
|
function isApplicable(item) {
|
|
|
|
var property = (item || {}).property,
|
2015-02-17 16:20:02 -08:00
|
|
|
method = (item || {}).method,
|
2015-02-23 19:47:56 -08:00
|
|
|
exclusive = !!(item || {}).exclusive;
|
2015-02-17 13:47:14 -08:00
|
|
|
|
|
|
|
// Check if a selected item defines this property
|
|
|
|
function hasProperty(selected) {
|
2015-02-17 16:20:02 -08:00
|
|
|
return (property && (selected[property] !== undefined)) ||
|
|
|
|
(method && (typeof selected[method] === 'function'));
|
2015-02-17 13:47:14 -08:00
|
|
|
}
|
|
|
|
|
2015-02-17 16:20:02 -08:00
|
|
|
return selection.map(hasProperty).reduce(
|
2015-02-17 13:47:14 -08:00
|
|
|
exclusive ? and : or,
|
|
|
|
exclusive
|
|
|
|
) && isConsistent(property);
|
|
|
|
}
|
|
|
|
|
2015-02-17 16:20:02 -08:00
|
|
|
// Invoke all functions in selections with the given name
|
|
|
|
function invoke(method, value) {
|
|
|
|
if (method) {
|
2015-02-18 20:12:55 -08:00
|
|
|
// Make the change in the selection
|
2015-02-17 16:20:02 -08:00
|
|
|
selection.forEach(function (selected) {
|
|
|
|
if (typeof selected[method] === 'function') {
|
|
|
|
selected[method](value);
|
|
|
|
}
|
|
|
|
});
|
2015-02-18 20:12:55 -08:00
|
|
|
// ...and commit!
|
|
|
|
commit();
|
2015-02-17 16:20:02 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-17 13:47:14 -08:00
|
|
|
// Prepare a toolbar item based on current selection
|
|
|
|
function convertItem(item) {
|
|
|
|
var converted = Object.create(item || {});
|
2015-02-17 16:20:02 -08:00
|
|
|
if (item.property) {
|
|
|
|
converted.key = addKey(item.property);
|
|
|
|
}
|
|
|
|
if (item.method) {
|
2015-02-18 20:12:55 -08:00
|
|
|
converted.click = function (v) {
|
|
|
|
invoke(item.method, v);
|
|
|
|
};
|
2015-02-17 16:20:02 -08:00
|
|
|
}
|
2015-02-17 13:47:14 -08:00
|
|
|
return converted;
|
|
|
|
}
|
|
|
|
|
2015-02-23 15:57:45 -08:00
|
|
|
// Prepare a toolbar section
|
2015-02-17 13:47:14 -08:00
|
|
|
function convertSection(section) {
|
|
|
|
var converted = Object.create(section || {});
|
|
|
|
converted.items =
|
|
|
|
((section || {}).items || [])
|
2015-02-23 15:57:45 -08:00
|
|
|
.map(convertItem);
|
2015-02-17 13:47:14 -08:00
|
|
|
return converted;
|
|
|
|
}
|
|
|
|
|
2015-02-23 15:57:45 -08:00
|
|
|
// Show/hide controls in this section per applicability
|
|
|
|
function refreshSectionApplicability(section) {
|
|
|
|
var count = 0;
|
|
|
|
// Show/hide each item
|
|
|
|
(section.items || []).forEach(function (item) {
|
|
|
|
item.hidden = !isApplicable(item);
|
|
|
|
count += item.hidden ? 0 : 1;
|
|
|
|
});
|
|
|
|
// Hide this section if there are no applicable items
|
|
|
|
section.hidden = !count;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Show/hide controls if they are applicable
|
|
|
|
function refreshApplicability() {
|
|
|
|
toolbarStructure.sections.forEach(refreshSectionApplicability);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Refresh toolbar state to match selection
|
|
|
|
function refreshState() {
|
|
|
|
toolbarState = properties.map(initializeState);
|
|
|
|
}
|
|
|
|
|
2015-02-17 13:47:14 -08:00
|
|
|
toolbarStructure.sections =
|
2015-02-23 19:47:56 -08:00
|
|
|
((structure || {}).sections || []).map(convertSection);
|
2015-02-17 13:47:14 -08:00
|
|
|
|
2015-02-23 15:57:45 -08:00
|
|
|
toolbarState = [];
|
2015-02-17 12:58:36 -08:00
|
|
|
|
|
|
|
return {
|
2015-02-17 13:47:14 -08:00
|
|
|
/**
|
2015-02-23 15:57:45 -08:00
|
|
|
* Set the current selection. Visisbility of sections
|
|
|
|
* and items in the toolbar will be updated to match this.
|
|
|
|
* @param {Array} s the new selection
|
|
|
|
*/
|
|
|
|
setSelection: function (s) {
|
|
|
|
selection = s;
|
|
|
|
refreshApplicability();
|
|
|
|
refreshState();
|
|
|
|
},
|
|
|
|
/**
|
|
|
|
* Get the structure of the toolbar, as appropriate to
|
|
|
|
* pass to `mct-toolbar`.
|
|
|
|
* @returns the toolbar structure
|
2015-02-17 13:47:14 -08:00
|
|
|
*/
|
2015-02-17 12:58:36 -08:00
|
|
|
getStructure: function () {
|
2015-02-17 13:47:14 -08:00
|
|
|
return toolbarStructure;
|
2015-02-17 12:58:36 -08:00
|
|
|
},
|
2015-02-23 15:57:45 -08:00
|
|
|
/**
|
|
|
|
* Get the current state of the toolbar, as appropriate
|
|
|
|
* to two-way bind to the state handled by `mct-toolbar`.
|
|
|
|
* @returns {Array} state of the toolbar
|
|
|
|
*/
|
2015-02-17 12:58:36 -08:00
|
|
|
getState: function () {
|
2015-02-17 13:47:14 -08:00
|
|
|
return toolbarState;
|
2015-02-17 12:58:36 -08:00
|
|
|
},
|
2015-02-23 15:57:45 -08:00
|
|
|
/**
|
|
|
|
* Update state within the current selection.
|
|
|
|
* @param {number} index the index of the corresponding
|
|
|
|
* element in the state array
|
|
|
|
* @param value the new value to convey to the selection
|
|
|
|
*/
|
|
|
|
updateState: function (index, value) {
|
2015-02-24 09:47:00 -08:00
|
|
|
return updateProperties(properties[index], value);
|
2015-02-17 12:58:36 -08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return EditToolbar;
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|