diff --git a/src/components/decorators/has-tutorial-tip.es6 b/src/components/decorators/has-tutorial-tip.es6 index 8fa963493..28a31d4db 100644 --- a/src/components/decorators/has-tutorial-tip.es6 +++ b/src/components/decorators/has-tutorial-tip.es6 @@ -2,7 +2,7 @@ import React from 'react'; import ReactDOM from 'react-dom'; import _ from 'underscore'; -import {Actions} from 'nylas-exports'; +import {Actions, WorkspaceStore} from 'nylas-exports'; import NylasStore from 'nylas-store'; const TipsBackgroundEl = document.createElement('tutorial-tip-background'); @@ -104,13 +104,19 @@ export default function HasTutorialTip(ComposedComponent, TipConfig) { constructor(props) { super(props); + this._unlisteners = []; this.state = {visible: false}; } componentDidMount() { TipsStore.mountedTip(TipKey); - this._unlisten = TipsStore.listen(this._onTooltipStateChanged); + this._unlisteners = [ + TipsStore.listen(this._onTooltipStateChanged), + WorkspaceStore.listen(() => { + this._workspaceTimer = setTimeout(this._onTooltipStateChanged, 0); + }), + ] window.addEventListener('resize', this._onRecomputeTooltipPosition); // unfortunately, we can't render() a container around ComposedComponent @@ -134,16 +140,28 @@ export default function HasTutorialTip(ComposedComponent, TipConfig) { } componentWillUnmount() { - this._unlisten(); + for (const unlisten of this._unlisteners) { + unlisten(); + } window.removeEventListener('resize', this._onRecomputeTooltipPosition); document.body.removeChild(this.tipNode); + clearTimeout(this._workspaceTimer); TipsStore.unmountedTip(TipKey); } + _containingSheetIsVisible = () => { + const el = ReactDOM.findDOMNode(this); + const sheetEl = el.closest('.sheet') || el.closest('.sheet-toolbar-container'); + if (!sheetEl) { + return true; + } + return (sheetEl.dataset.id === WorkspaceStore.topSheet().id); + } + _onTooltipStateChanged = () => { - const visible = TipsStore.isTipVisible(TipKey); + const visible = TipsStore.isTipVisible(TipKey) && this._containingSheetIsVisible(); if (this.state.visible !== visible) { this.setState({visible}); @@ -165,15 +183,12 @@ export default function HasTutorialTip(ComposedComponent, TipConfig) { el.removeEventListener('mouseover', this._onMouseOver); const tipRect = this.tipNode.getBoundingClientRect(); + const tipFocusCircleRadius = 64; const rect = ReactDOM.findDOMNode(this).getBoundingClientRect(); - const rectCX = rect.left + rect.width / 2; - const rectCY = rect.top + rect.height / 2; - TipsBackgroundEl.style.background = ` - -webkit-radial-gradient( - ${Math.round(rectCX / window.innerWidth * 100)}% - ${Math.round(rectCY / window.innerHeight * 100)}%, - circle, rgba(0, 0, 0, 0) 0%, rgba(0, 0, 0, 0) 3%, rgba(0, 0, 0, 0.2) 5%) - `; + const rectCX = Math.round(rect.left + rect.width / 2 - tipFocusCircleRadius); + const rectCY = Math.round(rect.top + rect.height / 2 - tipFocusCircleRadius); + TipsBackgroundEl.style.webkitMaskPosition = `0 0, ${rectCX}px ${rectCY}px`; + Actions.openPopover(( -
+
{toolbars}
diff --git a/static/components/tutorial-overlay.less b/static/components/tutorial-overlay.less index a3c68b6e3..653e7b998 100644 --- a/static/components/tutorial-overlay.less +++ b/static/components/tutorial-overlay.less @@ -4,11 +4,15 @@ tutorial-tip-background { left: 0; right: 0; bottom: 0; - background: -webkit-radial-gradient(50% 50%, circle, rgba(0,0,0,0) 0%, rgba(0,0,0,0) 4%, rgba(0,0,0,0.2) 7%); z-index: 20; pointer-events: none; opacity: 0; transition: opacity ease-in-out 250ms; + background-color: rgba(0,0,0,0.25); + -webkit-mask-image: linear-gradient(to top, #fff 0%, #aaa 100%), url(""); + -webkit-mask-composite: xor; + -webkit-mask-repeat: no-repeat; + -webkit-mask-size: 100% 100%, 128px 128px; } tutorial-tip-background.visible { @@ -30,6 +34,7 @@ tutorial-tip-background.visible { background-image: -webkit-gradient(linear, 0% 0%, 0% 100%, from(rgb(241, 170, 211)), to(rgb(185, 59, 255))); opacity: 0; transition: opacity ease-out 100ms; + box-shadow: 0 1px 2px rgba(0,0,0,0.2); } .tutorial-tip.visible { @@ -47,32 +52,32 @@ body { height: 100%; border-radius: 50%; content: ''; - -webkit-box-sizing: content-box; - -moz-box-sizing: content-box; - box-sizing: content-box; - box-shadow: 0 0 0 2px rgb(241, 170, 211); + border: 1px solid rgb(185, 59, 255); + transform-origin: 0.4 0.4; - animation: sonarEffect 2s ease-out 75ms; + animation: sonarEffect 3s ease-out 75ms; animation-iteration-count: infinite; + animation-delay: 1s; } @-webkit-keyframes sonarEffect { 0% { - opacity: 0.3; - } - 40% { - opacity: 0.7; - box-shadow: 0 0 0 2px rgba(255,255,255,0.1), 0 0 10px 10px #fff, 0 0 0 10px rgba(255,255,255,0.5); - } - 80% { - box-shadow: 0 0 0 2px rgba(255,255,255,0.1), 0 0 10px 10px #fff, 0 0 0 10px rgba(255,255,255,0.5); - -webkit-transform: scale(1.4); - opacity: 0.5; - } - 100% { - box-shadow: 0 0 0 2px rgba(255,255,255,0.1), 0 0 10px 10px #fff, 0 0 0 10px rgba(255,255,255,0.5); - -webkit-transform: scale(1.5); opacity: 0; } + 60% { + opacity: 0; + } + 65% { + opacity: 0.5; + transform: scale3d(1, 1, 1); + } + 80% { + opacity: 0.4; + transform: scale3d(2.6, 2.6, 2.6); + } + 100% { + opacity: 0; + transform: scale3d(3, 3, 3); + } }