Mailspring/src/component-registry.coffee
Juan Tejada e83bf2bbec feat(selection): Add new display for selection count + update toolbar
Summary:
- New behavior is that the in split mode, you will perform actions on
  the selection via the MessageListToolbar (the toolbar positioned above
  the message list)
- Refactored and moved around a bunch of code to achieve this:
  - Mostly renaming stuff and moving stuff around and removing some
    duplication
  - Update naming of toolbar role to a single role, and update relevant code
  - Converted and refactored a bunch of code into ES6, specifically to reuse the code for the ThreadActionsToolbar at the 2 locations
  - Deprecated MultiselectActionBar in favor of MultiselectToolbar
  - Deprecated old roles
- Punted the animation for the stackable cards in the selection display for now.
- #370

Test Plan: - Manual and unit tests

Reviewers: evan, drew, bengotow

Reviewed By: bengotow

Differential Revision: https://phab.nylas.com/D2756
2016-03-21 12:20:11 -07:00

196 lines
6.7 KiB
CoffeeScript

_ = require 'underscore'
{Listener, Publisher} = require './flux/modules/reflux-coffee'
CoffeeHelpers = require './flux/coffee-helpers'
DeprecatedRoles = {
'thread:BulkAction': 'ThreadActionsToolbarButton',
'draft:BulkAction': 'DraftActionsToolbarButton',
'message:Toolbar': 'ThreadActionsToolbarButton',
'thread:Toolbar': 'ThreadActionsToolbarButton',
}
###
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}")
roles = @_removeDeprecatedRoles(component.displayName, roles) if roles
@_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)
_removeDeprecatedRoles: (displayName, roles) ->
newRoles = _.clone(roles)
roles.forEach (role, idx) ->
if role of DeprecatedRoles
instead = DeprecatedRoles[role]
console.warn("Deprecation warning! The role `#{role}` has been deprecated.
Register `#{displayName}` for the role `#{instead}` instead.")
newRoles.splice(idx, 1, instead)
return newRoles
_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()