ref(info): be explicit when opening the dialog with a timeout
Instead of assuming the initial info dialog open should auto close, explicitly call opening of the dialog with a flag for auto closing. This better facilitates the auto close timeout being set at any time. The changes led to refactoring out state in the InfoDialogButton in preference for always clearing the timeout instead of first checking for interaction before clearing.
This commit is contained in:
parent
f539240840
commit
887e1b6828
|
@ -24,14 +24,20 @@ export function openInviteDialog() {
|
||||||
* Opens the inline conference info dialog.
|
* Opens the inline conference info dialog.
|
||||||
*
|
*
|
||||||
* @param {boolean} visible - Whether or not the dialog should be displayed.
|
* @param {boolean} visible - Whether or not the dialog should be displayed.
|
||||||
|
* @param {boolean} autoClose - Whether or not the dialog should automatically
|
||||||
|
* close after a set period of time.
|
||||||
* @returns {{
|
* @returns {{
|
||||||
* type: SET_INFO_DIALOG_VISIBILITY,
|
* type: SET_INFO_DIALOG_VISIBILITY,
|
||||||
|
* autoClose: boolean,
|
||||||
* visible: boolean
|
* visible: boolean
|
||||||
* }}
|
* }}
|
||||||
*/
|
*/
|
||||||
export function setInfoDialogVisibility(visible: boolean) {
|
export function setInfoDialogVisibility(
|
||||||
|
visible: boolean,
|
||||||
|
autoClose: boolean = false) {
|
||||||
return {
|
return {
|
||||||
type: SET_INFO_DIALOG_VISIBILITY,
|
type: SET_INFO_DIALOG_VISIBILITY,
|
||||||
|
autoClose,
|
||||||
visible
|
visible
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -40,6 +40,12 @@ class InfoDialogButton extends Component {
|
||||||
* @static
|
* @static
|
||||||
*/
|
*/
|
||||||
static propTypes = {
|
static propTypes = {
|
||||||
|
/**
|
||||||
|
* Whether or not the {@code InfoDialog} should close by itself after a
|
||||||
|
* a timeout.
|
||||||
|
*/
|
||||||
|
_shouldAutoClose: PropTypes.bool,
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Whether or not {@code InfoDialog} should be displayed.
|
* Whether or not {@code InfoDialog} should be displayed.
|
||||||
*/
|
*/
|
||||||
|
@ -80,15 +86,6 @@ class InfoDialogButton extends Component {
|
||||||
*/
|
*/
|
||||||
this._autoHideDialogTimeout = null;
|
this._autoHideDialogTimeout = null;
|
||||||
|
|
||||||
this.state = {
|
|
||||||
/**
|
|
||||||
* Whether or not the dialog has been interacted with somehow, such
|
|
||||||
* as clicking or toggle display. A value of true will prevent the
|
|
||||||
* dialog from being automatically hidden.
|
|
||||||
*/
|
|
||||||
hasInteractedWithDialog: false
|
|
||||||
};
|
|
||||||
|
|
||||||
// Bind event handlers so they are only bound once for every instance.
|
// Bind event handlers so they are only bound once for every instance.
|
||||||
this._onDialogClose = this._onDialogClose.bind(this);
|
this._onDialogClose = this._onDialogClose.bind(this);
|
||||||
this._onDialogMouseOver = this._onDialogMouseOver.bind(this);
|
this._onDialogMouseOver = this._onDialogMouseOver.bind(this);
|
||||||
|
@ -101,23 +98,34 @@ class InfoDialogButton extends Component {
|
||||||
* @inheritdoc
|
* @inheritdoc
|
||||||
*/
|
*/
|
||||||
componentDidMount() {
|
componentDidMount() {
|
||||||
this._autoHideDialogTimeout = setTimeout(() => {
|
if (this.props._shouldAutoClose) {
|
||||||
this._maybeHideDialog();
|
this._setAutoCloseTimeout();
|
||||||
}, INITIAL_TOOLBAR_TIMEOUT);
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Update the state when the {@code InfoDialog} visibility has been updated.
|
* Set or clear the timeout to automatically hide the {@code InfoDialog}.
|
||||||
|
*
|
||||||
|
* @inheritdoc
|
||||||
|
*/
|
||||||
|
componentDidUpdate(prevProps) {
|
||||||
|
// If the _shouldAutoClose flag has been updated to be true then make
|
||||||
|
// sure to set _autoHideDialogTimeout.
|
||||||
|
if (this.props._shouldAutoClose && !prevProps._shouldAutoClose) {
|
||||||
|
this._setAutoCloseTimeout();
|
||||||
|
} else {
|
||||||
|
this._clearAutoCloseTimeout();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update the visibility of the {@code InfoDialog}.
|
||||||
*
|
*
|
||||||
* @inheritdoc
|
* @inheritdoc
|
||||||
*/
|
*/
|
||||||
componentWillReceiveProps(nextProps) {
|
componentWillReceiveProps(nextProps) {
|
||||||
if (!this.state.hasInteractedWithDialog
|
// Ensure the dialog is closed when the toolbox becomes hidden.
|
||||||
&& (nextProps._showDialog !== this.props._showDialog)) {
|
if (nextProps._showDialog && !nextProps._toolboxVisible) {
|
||||||
this.setState({ hasInteractedWithDialog: true });
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!nextProps._toolboxVisible && this.props._toolboxVisible) {
|
|
||||||
this._onDialogClose();
|
this._onDialogClose();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -128,7 +136,7 @@ class InfoDialogButton extends Component {
|
||||||
* @inheritdoc
|
* @inheritdoc
|
||||||
*/
|
*/
|
||||||
componentWillUnmount() {
|
componentWillUnmount() {
|
||||||
clearTimeout(this._autoHideDialogTimeout);
|
this._clearAutoCloseTimeout();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -154,7 +162,6 @@ class InfoDialogButton extends Component {
|
||||||
onMouseOver = { this._onDialogMouseOver } /> }
|
onMouseOver = { this._onDialogMouseOver } /> }
|
||||||
isOpen = { _toolboxVisible && _showDialog }
|
isOpen = { _toolboxVisible && _showDialog }
|
||||||
onClose = { this._onDialogClose }
|
onClose = { this._onDialogClose }
|
||||||
onContentClick = { this._onDialogInteract }
|
|
||||||
position = { TOOLTIP_TO_POPUP_POSITION[tooltipPosition] }>
|
position = { TOOLTIP_TO_POPUP_POSITION[tooltipPosition] }>
|
||||||
<ToolbarButton
|
<ToolbarButton
|
||||||
button = { buttonConfiguration }
|
button = { buttonConfiguration }
|
||||||
|
@ -165,17 +172,14 @@ class InfoDialogButton extends Component {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Callback invoked after a timeout to trigger hiding of the
|
* Cancels the timeout to automatically hide the {@code InfoDialog}.
|
||||||
* {@code InfoDialog} if there has been no interaction with the dialog
|
|
||||||
* and the dialog is currently showing.
|
|
||||||
*
|
*
|
||||||
* @private
|
* @private
|
||||||
* @returns {void}
|
* @returns {void}
|
||||||
*/
|
*/
|
||||||
_maybeHideDialog() {
|
_clearAutoCloseTimeout() {
|
||||||
if (!this.state.hasInteractedWithDialog && this.props._showDialog) {
|
clearTimeout(this._autoHideDialogTimeout);
|
||||||
this._onDialogToggle();
|
this._autoHideDialogTimeout = null;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -189,16 +193,13 @@ class InfoDialogButton extends Component {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Updates the internal state to mark the {@code InfoDialog} as having been
|
* Cancels the timeout to automatically hide the {@code InfoDialog}.
|
||||||
* interacted with.
|
|
||||||
*
|
*
|
||||||
* @private
|
* @private
|
||||||
* @returns {void}
|
* @returns {void}
|
||||||
*/
|
*/
|
||||||
_onDialogMouseOver() {
|
_onDialogMouseOver() {
|
||||||
if (!this.state.hasInteractedWithDialog) {
|
this._clearAutoCloseTimeout();
|
||||||
this.setState({ hasInteractedWithDialog: true });
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -210,6 +211,22 @@ class InfoDialogButton extends Component {
|
||||||
_onDialogToggle() {
|
_onDialogToggle() {
|
||||||
this.props.dispatch(setInfoDialogVisibility(!this.props._showDialog));
|
this.props.dispatch(setInfoDialogVisibility(!this.props._showDialog));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set a timeout to automatically hide the {@code InfoDialog}.
|
||||||
|
*
|
||||||
|
* @private
|
||||||
|
* @returns {void}
|
||||||
|
*/
|
||||||
|
_setAutoCloseTimeout() {
|
||||||
|
this._clearAutoCloseTimeout();
|
||||||
|
|
||||||
|
this._autoHideDialogTimeout = setTimeout(() => {
|
||||||
|
if (this.props._showDialog) {
|
||||||
|
this._onDialogClose();
|
||||||
|
}
|
||||||
|
}, INITIAL_TOOLBAR_TIMEOUT);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -219,13 +236,20 @@ class InfoDialogButton extends Component {
|
||||||
* @param {Object} state - The Redux state.
|
* @param {Object} state - The Redux state.
|
||||||
* @private
|
* @private
|
||||||
* @returns {{
|
* @returns {{
|
||||||
|
* _shouldAutoClose: boolean,
|
||||||
* _showDialog: boolean,
|
* _showDialog: boolean,
|
||||||
* _toolboxVisible: boolean
|
* _toolboxVisible: boolean
|
||||||
* }}
|
* }}
|
||||||
*/
|
*/
|
||||||
function _mapStateToProps(state) {
|
function _mapStateToProps(state) {
|
||||||
|
const {
|
||||||
|
infoDialogVisible,
|
||||||
|
infoDialogWillAutoClose
|
||||||
|
} = state['features/invite'];
|
||||||
|
|
||||||
return {
|
return {
|
||||||
_showDialog: state['features/invite'].infoDialogVisible,
|
_shouldAutoClose: infoDialogWillAutoClose,
|
||||||
|
_showDialog: infoDialogVisible,
|
||||||
_toolboxVisible: state['features/toolbox'].visible
|
_toolboxVisible: state['features/toolbox'].visible
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,7 +18,7 @@ MiddlewareRegistry.register(store => next => action => {
|
||||||
|
|
||||||
switch (action.type) {
|
switch (action.type) {
|
||||||
case CONFERENCE_JOINED:
|
case CONFERENCE_JOINED:
|
||||||
store.dispatch(setInfoDialogVisibility(true));
|
store.dispatch(setInfoDialogVisibility(true, true));
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case UPDATE_DIAL_IN_NUMBERS_FAILED:
|
case UPDATE_DIAL_IN_NUMBERS_FAILED:
|
||||||
|
|
|
@ -15,7 +15,8 @@ ReducerRegistry.register('features/invite', (state = DEFAULT_STATE, action) => {
|
||||||
case SET_INFO_DIALOG_VISIBILITY:
|
case SET_INFO_DIALOG_VISIBILITY:
|
||||||
return {
|
return {
|
||||||
...state,
|
...state,
|
||||||
infoDialogVisible: action.visible
|
infoDialogVisible: action.visible,
|
||||||
|
infoDialogWillAutoClose: action.autoClose
|
||||||
};
|
};
|
||||||
|
|
||||||
case UPDATE_DIAL_IN_NUMBERS_FAILED:
|
case UPDATE_DIAL_IN_NUMBERS_FAILED:
|
||||||
|
|
Loading…
Reference in New Issue