123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156 |
- /*!
- * use <https://github.com/jonschlinkert/use>
- *
- * Copyright (c) 2015-2017, Jon Schlinkert.
- * Released under the MIT License.
- */
- 'use strict';
- module.exports = function base(app, options) {
- if (!isObject(app) && typeof app !== 'function') {
- throw new TypeError('expected an object or function');
- }
- var opts = isObject(options) ? options : {};
- var prop = typeof opts.prop === 'string' ? opts.prop : 'fns';
- if (!Array.isArray(app[prop])) {
- define(app, prop, []);
- }
- /**
- * Define a plugin function to be passed to use. The only
- * parameter exposed to the plugin is `app`, the object or function.
- * passed to `use(app)`. `app` is also exposed as `this` in plugins.
- *
- * Additionally, **if a plugin returns a function, the function will
- * be pushed onto the `fns` array**, allowing the plugin to be
- * called at a later point by the `run` method.
- *
- * ```js
- * var use = require('use');
- *
- * // define a plugin
- * function foo(app) {
- * // do stuff
- * }
- *
- * var app = function(){};
- * use(app);
- *
- * // register plugins
- * app.use(foo);
- * app.use(bar);
- * app.use(baz);
- * ```
- * @name .use
- * @param {Function} `fn` plugin function to call
- * @api public
- */
- define(app, 'use', use);
- /**
- * Run all plugins on `fns`. Any plugin that returns a function
- * when called by `use` is pushed onto the `fns` array.
- *
- * ```js
- * var config = {};
- * app.run(config);
- * ```
- * @name .run
- * @param {Object} `value` Object to be modified by plugins.
- * @return {Object} Returns the object passed to `run`
- * @api public
- */
- define(app, 'run', function(val) {
- if (!isObject(val)) return;
- if (!val.use || !val.run) {
- define(val, prop, val[prop] || []);
- define(val, 'use', use);
- }
- if (!val[prop] || val[prop].indexOf(base) === -1) {
- val.use(base);
- }
- var self = this || app;
- var fns = self[prop];
- var len = fns.length;
- var idx = -1;
- while (++idx < len) {
- val.use(fns[idx]);
- }
- return val;
- });
- /**
- * Call plugin `fn`. If a function is returned push it into the
- * `fns` array to be called by the `run` method.
- */
- function use(type, fn, options) {
- var offset = 1;
- if (typeof type === 'string' || Array.isArray(type)) {
- fn = wrap(type, fn);
- offset++;
- } else {
- options = fn;
- fn = type;
- }
- if (typeof fn !== 'function') {
- throw new TypeError('expected a function');
- }
- var self = this || app;
- var fns = self[prop];
- var args = [].slice.call(arguments, offset);
- args.unshift(self);
- if (typeof opts.hook === 'function') {
- opts.hook.apply(self, args);
- }
- var val = fn.apply(self, args);
- if (typeof val === 'function' && fns.indexOf(val) === -1) {
- fns.push(val);
- }
- return self;
- }
- /**
- * Wrap a named plugin function so that it's only called on objects of the
- * given `type`
- *
- * @param {String} `type`
- * @param {Function} `fn` Plugin function
- * @return {Function}
- */
- function wrap(type, fn) {
- return function plugin() {
- return this.type === type ? fn.apply(this, arguments) : plugin;
- };
- }
- return app;
- };
- function isObject(val) {
- return val && typeof val === 'object' && !Array.isArray(val);
- }
- function define(obj, key, val) {
- Object.defineProperty(obj, key, {
- configurable: true,
- writable: true,
- value: val
- });
- }
|