mirror of
https://github.com/Foundry376/Mailspring.git
synced 2025-01-25 09:30:13 +08:00
178 lines
6.1 KiB
CoffeeScript
178 lines
6.1 KiB
CoffeeScript
_ = require 'underscore'
|
|
|
|
{Listener, Publisher} = require './flux/modules/reflux-coffee'
|
|
CoffeeHelpers = require './flux/coffee-helpers'
|
|
|
|
###
|
|
Public: The ComponentRegistry maintains an index of React components registered
|
|
by Nylas packages. Components can use {InjectedComponent} and {InjectedComponentSet}
|
|
to dynamically render components registered with the ComponentRegistry.
|
|
|
|
Section: Stores
|
|
|
|
###
|
|
class ComponentRegistry
|
|
@include: CoffeeHelpers.includeModule
|
|
|
|
@include Publisher
|
|
@include Listener
|
|
|
|
constructor: ->
|
|
@_registry = {}
|
|
@_cache = {}
|
|
@_showComponentRegions = false
|
|
|
|
|
|
# Public: Register a new component with the Component Registry.
|
|
# Typically, packages call this method from their main `activate` method
|
|
# to extend the Nylas user interface, and call the corresponding `unregister`
|
|
# method in `deactivate`.
|
|
#
|
|
# * `component` {Object} A React Component with a `displayName`
|
|
# * `options` {Object}:
|
|
#
|
|
# * `role`: (optional) {String} If you want to display your component in a location
|
|
# desigated by a role, pass the role identifier.
|
|
#
|
|
# * `modes`: (optional) {Array} If your component should only be displayed
|
|
# in particular Workspace Modes, pass an array of supported modes.
|
|
# ('list', 'split', etc.)
|
|
#
|
|
# * `location`: (optional) {Object} If your component should be displayed in a
|
|
# column or toolbar, pass the fully qualified location object, such as:
|
|
# `WorkspaceStore.Location.ThreadList`
|
|
#
|
|
# Note that for advanced use cases, you can also pass (`modes`, `roles`, `locations`)
|
|
# with arrays instead of single values.
|
|
#
|
|
# This method is chainable.
|
|
#
|
|
register: (component, options) =>
|
|
if component.view?
|
|
return console.warn("Ignoring component trying to register with old CommandRegistry.register syntax")
|
|
|
|
throw new Error("ComponentRegistry.register() requires `options` that describe the component") unless options
|
|
throw new Error("ComponentRegistry.register() requires `component`, a React component") unless component
|
|
throw new Error("ComponentRegistry.register() requires that your React Component defines a `displayName`") unless component.displayName
|
|
|
|
{locations, modes, roles} = @_pluralizeDescriptor(options)
|
|
|
|
throw new Error("ComponentRegistry.register() requires `role` or `location`") if not roles and not locations
|
|
|
|
if @_registry[component.displayName] and @_registry[component.displayName].component isnt component
|
|
throw new Error("ComponentRegistry.register(): A different component was already registered with the name #{component.displayName}")
|
|
|
|
@_cache = {}
|
|
@_registry[component.displayName] = {component, locations, modes, roles}
|
|
|
|
# Trigger listeners. It's very important the component registry is debounced.
|
|
# During app launch packages register tons of components and if we re-rendered
|
|
# the entire UI after each registration it takes forever to load the UI.
|
|
@triggerDebounced()
|
|
|
|
# Return `this` for chaining
|
|
@
|
|
|
|
unregister: (component) =>
|
|
if _.isString(component)
|
|
throw new Error("ComponentRegistry.unregister() must be called with a component.")
|
|
@_cache = {}
|
|
delete @_registry[component.displayName]
|
|
@triggerDebounced()
|
|
|
|
# Public: Retrieve the registry entry for a given name.
|
|
#
|
|
# - `name`: The {String} name of the registered component to retrieve.
|
|
#
|
|
# Returns a {React.Component}
|
|
#
|
|
findComponentByName: (name) =>
|
|
@_registry[name]?.component
|
|
|
|
###
|
|
Public: Retrieve all of the registry entries matching a given descriptor.
|
|
|
|
```coffee
|
|
ComponentRegistry.findComponentsMatching({
|
|
role: 'Composer:ActionButton'
|
|
})
|
|
|
|
ComponentRegistry.findComponentsMatching({
|
|
location: WorkspaceStore.Location.RootSidebar.Toolbar
|
|
})
|
|
```
|
|
|
|
- `descriptor`: An {Object} that specifies set of components using the
|
|
available keys below.
|
|
|
|
* `mode`: (optional) {String} Components that specifically list modes
|
|
will only be returned if they include this mode.
|
|
|
|
* `role`: (optional) {String} Only return components that have registered
|
|
for this role.
|
|
|
|
* `location`: (optional) {Object} Only return components that have registered
|
|
for this location.
|
|
|
|
Note that for advanced use cases, you can also pass (`modes`, `roles`, `locations`)
|
|
with arrays instead of single values.
|
|
|
|
Returns an {Array} of {React.Component} objects
|
|
###
|
|
findComponentsMatching: (descriptor) =>
|
|
if not descriptor?
|
|
throw new Error("ComponentRegistry.findComponentsMatching called without descriptor")
|
|
|
|
{locations, modes, roles} = @_pluralizeDescriptor(descriptor)
|
|
|
|
if not locations and not modes and not roles
|
|
throw new Error("ComponentRegistry.findComponentsMatching called with an empty descriptor")
|
|
|
|
cacheKey = JSON.stringify({locations, modes, roles})
|
|
return [].concat(@_cache[cacheKey]) if @_cache[cacheKey]
|
|
|
|
# Made into a convenience function because default
|
|
# values (`[]`) are necessary and it was getting messy.
|
|
overlaps = (entry = [], search = []) ->
|
|
_.intersection(entry, search).length > 0
|
|
|
|
entries = _.values @_registry
|
|
entries = _.filter entries, (entry) ->
|
|
if modes and entry.modes and not overlaps(modes, entry.modes)
|
|
return false
|
|
if locations and not overlaps(locations, entry.locations)
|
|
return false
|
|
if roles and not overlaps(roles, entry.roles)
|
|
return false
|
|
return true
|
|
|
|
results = _.map entries, (entry) -> entry.component
|
|
@_cache[cacheKey] = results
|
|
|
|
return [].concat(results)
|
|
|
|
triggerDebounced: _.debounce(( -> @trigger(@)), 1)
|
|
|
|
|
|
_pluralizeDescriptor: (descriptor) ->
|
|
{locations, modes, roles} = descriptor
|
|
modes = [descriptor.mode] if descriptor.mode
|
|
roles = [descriptor.role] if descriptor.role
|
|
locations = [descriptor.location] if descriptor.location
|
|
{locations, modes, roles}
|
|
|
|
_clear: =>
|
|
@_cache = {}
|
|
@_registry = {}
|
|
|
|
# Showing Component Regions
|
|
|
|
toggleComponentRegions: ->
|
|
@_showComponentRegions = !@_showComponentRegions
|
|
@trigger(@)
|
|
|
|
showComponentRegions: =>
|
|
@_showComponentRegions
|
|
|
|
|
|
module.exports = new ComponentRegistry()
|