|
import React, { PropTypes } from 'react'; |
|
import ReactDOM from 'react-dom'; |
|
|
|
import { has, mapValues, omitBy, keys, pick } from 'lodash'; |
|
|
|
|
|
export const shape = { |
|
$injector: PropTypes.object.isRequired, |
|
$scope: PropTypes.object.isRequired, |
|
$element: PropTypes.object.isRequired |
|
} |
|
|
|
|
|
class Bridge { |
|
static $inject = ['$injector', '$scope', '$element']; |
|
|
|
static propTypes = shape; |
|
|
|
constructor($injector, $scope, $element) { |
|
this.props = { $injector, $scope, $element }; |
|
} |
|
|
|
$postLink() { |
|
this.container = this.props.$element[0]; |
|
this.render(); |
|
} |
|
|
|
$onChanges(changes) { |
|
this.container && this.render(); |
|
} |
|
|
|
$onDestroy() { |
|
ReactDOM.unmountComponentAtNode(this.container) |
|
} |
|
|
|
createElement = (props) => |
|
<this.ComponentClass {...this.props} {...props}/> |
|
|
|
render() { |
|
this.instance = |
|
this.createElement(pick(this, keys(this.bindings))); |
|
|
|
try { |
|
ReactDOM.render(this.instance, this.container); |
|
} |
|
catch(e) { |
|
console.error('Unable to mount ReactBridge\n', e); |
|
} |
|
} |
|
} |
|
|
|
|
|
export function create(ComponentClass) { |
|
let customBindings = ComponentClass.bindings || {}; |
|
let derivedBindings = |
|
mapValues( |
|
// Remove injected values |
|
omitBy(ComponentClass.propTypes, (_, key) => |
|
/^[$](injector|scope|element)$/.test(key)), |
|
// Set one way data binding |
|
(propType) => |
|
// propTypes that are not required have an isRequired() method |
|
`<${ ! has(propType, 'isRequired') ? '' : '?' }` |
|
) |
|
; |
|
|
|
let bindings = { ...derivedBindings, ...customBindings } |
|
|
|
return { |
|
bindings, |
|
controller: class ComponentBridge extends Bridge { |
|
bindings = bindings |
|
ComponentClass = ComponentClass; |
|
} |
|
} |
|
} |
|
|
|
|
|
export default { |
|
create, |
|
shape |
|
} |