/**
* @license
* Copyright The Closure Library Authors.
* SPDX-License-Identifier: Apache-2.0
*/
/**
* @fileoverview Global renderer and decorator registry.
*/
goog.provide('goog.ui.registry');
goog.require('goog.asserts');
goog.require('goog.dom.classlist');
goog.require('goog.object');
goog.requireType('goog.ui.Component');
goog.requireType('goog.ui.ControlRenderer');
/**
* Given a {@link goog.ui.Component} constructor, returns an instance of its
* default renderer. If the default renderer is a singleton, returns the
* singleton instance; otherwise returns a new instance of the renderer class.
* @param {!Function} componentCtor Component constructor function (for example
* `goog.ui.Button`).
* @return {?goog.ui.ControlRenderer} Renderer instance (for example the
* singleton instance of `goog.ui.ButtonRenderer`), or null if
* no default renderer was found.
*/
goog.ui.registry.getDefaultRenderer = function(componentCtor) {
'use strict';
// TODO(user): This should probably be implemented with a `WeakMap`.
// Locate the default renderer based on the constructor's unique ID. If no
// renderer is registered for this class, walk up the superClass_ chain.
var key;
var /** ?Function|undefined */ ctor = componentCtor;
var /** ?Function|undefined */ rendererCtor;
while (ctor) {
key = goog.getUid(ctor);
if ((rendererCtor = goog.ui.registry.defaultRenderers_[key])) break;
ctor = /** @type {?Function|undefined} */ (goog.object.getSuperClass(ctor));
}
// If the renderer has a static getInstance method, return the singleton
// instance; otherwise create and return a new instance.
if (rendererCtor) {
return typeof rendererCtor.getInstance === 'function' ?
rendererCtor.getInstance() :
new rendererCtor();
}
return null;
};
/**
* Sets the default renderer for the given {@link goog.ui.Component}
* constructor.
* @param {Function} componentCtor Component constructor function (for example
* `goog.ui.Button`).
* @param {Function} rendererCtor Renderer constructor function (for example
* `goog.ui.ButtonRenderer`).
* @throws {Error} If the arguments aren't functions.
*/
goog.ui.registry.setDefaultRenderer = function(componentCtor, rendererCtor) {
'use strict';
// In this case, explicit validation has negligible overhead (since each
// renderer is only registered once), and helps catch subtle bugs.
if (typeof componentCtor !== 'function') {
throw new Error('Invalid component class ' + componentCtor);
}
if (typeof rendererCtor !== 'function') {
throw new Error('Invalid renderer class ' + rendererCtor);
}
// Map the component constructor's unique ID to the renderer constructor.
var key = goog.getUid(componentCtor);
goog.ui.registry.defaultRenderers_[key] = rendererCtor;
};
/**
* Returns the {@link goog.ui.Component} instance created by the decorator
* factory function registered for the given CSS class name, or null if no
* decorator factory function was found.
* @param {string} className CSS class name.
* @return {goog.ui.Component?} Component instance.
*/
goog.ui.registry.getDecoratorByClassName = function(className) {
'use strict';
return className in goog.ui.registry.decoratorFunctions_ ?
goog.ui.registry.decoratorFunctions_[className]() :
null;
};
/**
* Maps a CSS class name to a function that returns a new instance of
* {@link goog.ui.Component} or a subclass, suitable to decorate an element
* that has the specified CSS class.
* @param {string} className CSS class name.
* @param {Function} decoratorFn No-argument function that returns a new
* instance of a {@link goog.ui.Component} to decorate an element.
* @throws {Error} If the class name or the decorator function is invalid.
*/
goog.ui.registry.setDecoratorByClassName = function(className, decoratorFn) {
'use strict';
// In this case, explicit validation has negligible overhead (since each
// decorator is only registered once), and helps catch subtle bugs.
if (!className) {
throw new Error('Invalid class name ' + className);
}
if (typeof decoratorFn !== 'function') {
throw new Error('Invalid decorator function ' + decoratorFn);
}
goog.ui.registry.decoratorFunctions_[className] = decoratorFn;
};
/**
* Returns an instance of {@link goog.ui.Component} or a subclass suitable to
* decorate the given element, based on its CSS class.
*
* TODO(nnaze): Type of element should be {!Element}.
*
* @param {Element} element Element to decorate.
* @return {goog.ui.Component?} Component to decorate the element (null if
* none).
*/
goog.ui.registry.getDecorator = function(element) {
'use strict';
var decorator;
goog.asserts.assert(element);
var classNames = goog.dom.classlist.get(element);
for (var i = 0, len = classNames.length; i < len; i++) {
if ((decorator = goog.ui.registry.getDecoratorByClassName(classNames[i]))) {
return decorator;
}
}
return null;
};
/**
* Resets the global renderer and decorator registry.
*/
goog.ui.registry.reset = function() {
'use strict';
goog.ui.registry.defaultRenderers_ = {};
goog.ui.registry.decoratorFunctions_ = {};
};
/**
* Map of {@link goog.ui.Component} constructor unique IDs to the constructors
* of their default {@link goog.ui.Renderer}s.
* @type {Object}
* @private
*/
goog.ui.registry.defaultRenderers_ = {};
/**
* Map of CSS class names to registry factory functions. The keys are
* class names. The values are function objects that return new instances
* of {@link goog.ui.registry} or one of its subclasses, suitable to
* decorate elements marked with the corresponding CSS class. Used by
* containers while decorating their children.
* @type {Object}
* @private
*/
goog.ui.registry.decoratorFunctions_ = {};