mirror of
https://github.com/twisterarmy/twister-react.git
synced 2025-01-25 22:24:26 +00:00
184 lines
5.5 KiB
JavaScript
Executable File
184 lines
5.5 KiB
JavaScript
Executable File
"use strict";
|
||
|
||
var _interopRequire = function (obj) { return obj && obj.__esModule ? obj["default"] : obj; };
|
||
|
||
var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; };
|
||
|
||
var React = _interopRequire(require("react"));
|
||
|
||
var classNames = _interopRequire(require("classnames"));
|
||
|
||
var BootstrapMixin = _interopRequire(require("./BootstrapMixin"));
|
||
|
||
var FadeMixin = _interopRequire(require("./FadeMixin"));
|
||
|
||
var domUtils = _interopRequire(require("./utils/domUtils"));
|
||
|
||
var EventListener = _interopRequire(require("./utils/EventListener"));
|
||
|
||
// TODO:
|
||
// - aria-labelledby
|
||
// - Add `modal-body` div if only one child passed in that doesn't already have it
|
||
// - Tests
|
||
|
||
var Modal = React.createClass({
|
||
displayName: "Modal",
|
||
|
||
mixins: [BootstrapMixin, FadeMixin],
|
||
|
||
propTypes: {
|
||
title: React.PropTypes.node,
|
||
backdrop: React.PropTypes.oneOf(["static", true, false]),
|
||
keyboard: React.PropTypes.bool,
|
||
closeButton: React.PropTypes.bool,
|
||
animation: React.PropTypes.bool,
|
||
onRequestHide: React.PropTypes.func.isRequired
|
||
},
|
||
|
||
getDefaultProps: function getDefaultProps() {
|
||
return {
|
||
bsClass: "modal",
|
||
backdrop: true,
|
||
keyboard: true,
|
||
animation: true,
|
||
closeButton: true
|
||
};
|
||
},
|
||
|
||
render: function render() {
|
||
var modalStyle = { display: "block" };
|
||
var dialogClasses = this.getBsClassSet();
|
||
delete dialogClasses.modal;
|
||
dialogClasses["modal-dialog"] = true;
|
||
|
||
var classes = {
|
||
modal: true,
|
||
fade: this.props.animation,
|
||
"in": !this.props.animation || !document.querySelectorAll
|
||
};
|
||
|
||
var modal = React.createElement(
|
||
"div",
|
||
_extends({}, this.props, {
|
||
title: null,
|
||
tabIndex: "-1",
|
||
role: "dialog",
|
||
style: modalStyle,
|
||
className: classNames(this.props.className, classes),
|
||
onClick: this.props.backdrop === true ? this.handleBackdropClick : null,
|
||
ref: "modal" }),
|
||
React.createElement(
|
||
"div",
|
||
{ className: classNames(dialogClasses) },
|
||
React.createElement(
|
||
"div",
|
||
{ className: "modal-content", style: { overflow: "hidden" } },
|
||
this.props.title ? this.renderHeader() : null,
|
||
this.props.children
|
||
)
|
||
)
|
||
);
|
||
|
||
return this.props.backdrop ? this.renderBackdrop(modal) : modal;
|
||
},
|
||
|
||
renderBackdrop: function renderBackdrop(modal) {
|
||
var classes = {
|
||
"modal-backdrop": true,
|
||
fade: this.props.animation
|
||
};
|
||
|
||
classes["in"] = !this.props.animation || !document.querySelectorAll;
|
||
|
||
var onClick = this.props.backdrop === true ? this.handleBackdropClick : null;
|
||
|
||
return React.createElement(
|
||
"div",
|
||
null,
|
||
React.createElement("div", { className: classNames(classes), ref: "backdrop", onClick: onClick }),
|
||
modal
|
||
);
|
||
},
|
||
|
||
renderHeader: function renderHeader() {
|
||
var closeButton = undefined;
|
||
if (this.props.closeButton) {
|
||
closeButton = React.createElement(
|
||
"button",
|
||
{ type: "button", className: "close", "aria-hidden": "true", onClick: this.props.onRequestHide },
|
||
"×"
|
||
);
|
||
}
|
||
|
||
var style = this.props.bsStyle;
|
||
var classes = {
|
||
"modal-header": true
|
||
};
|
||
classes["bg-" + style] = style;
|
||
classes["text-" + style] = style;
|
||
|
||
var className = classNames(classes);
|
||
|
||
return React.createElement(
|
||
"div",
|
||
{ className: className },
|
||
closeButton,
|
||
this.renderTitle()
|
||
);
|
||
},
|
||
|
||
renderTitle: function renderTitle() {
|
||
return React.isValidElement(this.props.title) ? this.props.title : React.createElement(
|
||
"h4",
|
||
{ className: "modal-title" },
|
||
this.props.title
|
||
);
|
||
},
|
||
|
||
iosClickHack: function iosClickHack() {
|
||
// IOS only allows click events to be delegated to the document on elements
|
||
// it considers 'clickable' - anchors, buttons, etc. We fake a click handler on the
|
||
// DOM nodes themselves. Remove if handled by React: https://github.com/facebook/react/issues/1169
|
||
React.findDOMNode(this.refs.modal).onclick = function () {};
|
||
React.findDOMNode(this.refs.backdrop).onclick = function () {};
|
||
},
|
||
|
||
componentDidMount: function componentDidMount() {
|
||
this._onDocumentKeyupListener = EventListener.listen(domUtils.ownerDocument(this), "keyup", this.handleDocumentKeyUp);
|
||
|
||
var container = this.props.container && React.findDOMNode(this.props.container) || domUtils.ownerDocument(this).body;
|
||
container.className += container.className.length ? " modal-open" : "modal-open";
|
||
|
||
if (this.props.backdrop) {
|
||
this.iosClickHack();
|
||
}
|
||
},
|
||
|
||
componentDidUpdate: function componentDidUpdate(prevProps) {
|
||
if (this.props.backdrop && this.props.backdrop !== prevProps.backdrop) {
|
||
this.iosClickHack();
|
||
}
|
||
},
|
||
|
||
componentWillUnmount: function componentWillUnmount() {
|
||
this._onDocumentKeyupListener.remove();
|
||
var container = this.props.container && React.findDOMNode(this.props.container) || domUtils.ownerDocument(this).body;
|
||
container.className = container.className.replace(/ ?modal-open/, "");
|
||
},
|
||
|
||
handleBackdropClick: function handleBackdropClick(e) {
|
||
if (e.target !== e.currentTarget) {
|
||
return;
|
||
}
|
||
|
||
this.props.onRequestHide();
|
||
},
|
||
|
||
handleDocumentKeyUp: function handleDocumentKeyUp(e) {
|
||
if (this.props.keyboard && e.keyCode === 27) {
|
||
this.props.onRequestHide();
|
||
}
|
||
}
|
||
});
|
||
|
||
module.exports = Modal; |