From a317a727c80ed9dd1eb55d9cef95d794822e2a93 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jaros=C5=82aw=20Sadzi=C5=84ski?= Date: Tue, 28 Mar 2023 18:11:40 +0200 Subject: [PATCH] (core) Adding tutorial card Summary: For now only html stub and docList adjustement for showing a tutorial card. It will be used in future diffs after tutorial implementation. Test Plan: Manual Reviewers: georgegevoian Reviewed By: georgegevoian Differential Revision: https://phab.getgrist.com/D3750 --- app/client/components/GristDoc.ts | 6 - app/client/models/AppModel.ts | 20 ++- app/client/models/UserPrefs.ts | 14 ++ app/client/ui/DocMenu.ts | 10 +- app/client/ui/DocMenuCss.ts | 15 +- app/client/ui/HomeLeftPane.ts | 13 +- app/client/ui/RightPanel.ts | 2 +- app/client/ui/TutorialCard.ts | 219 ++++++++++++++++++++++++++++++ app/client/ui2018/IconList.ts | 8 ++ app/common/Prefs.ts | 5 +- app/common/gristUrls.ts | 2 + static/icons/icons.css | 4 + static/locales/en.client.json | 2 +- static/ui-icons/UI/Board.svg | 1 + static/ui-icons/UI/Bookmark.svg | 1 + static/ui-icons/UI/GreenArrow.svg | 3 + static/ui-icons/UI/RemoveBig.svg | 26 ++++ test/nbrowser/DocTutorial.ts | 41 ++++++ test/nbrowser/HomeIntro.ts | 2 +- test/nbrowser/gristUtils.ts | 60 ++++++-- 20 files changed, 424 insertions(+), 30 deletions(-) create mode 100644 app/client/ui/TutorialCard.ts create mode 100644 static/ui-icons/UI/Board.svg create mode 100644 static/ui-icons/UI/Bookmark.svg create mode 100644 static/ui-icons/UI/GreenArrow.svg create mode 100644 static/ui-icons/UI/RemoveBig.svg diff --git a/app/client/components/GristDoc.ts b/app/client/components/GristDoc.ts index ceb7e243..3527ae6e 100644 --- a/app/client/components/GristDoc.ts +++ b/app/client/components/GristDoc.ts @@ -66,7 +66,6 @@ import {isList, isListType, isRefListType, RecalcWhen} from 'app/common/gristTyp import {HashLink, IDocPage, isViewDocPage, SpecialDocPage, ViewDocPage} from 'app/common/gristUrls'; import {undef, waitObs} from 'app/common/gutil'; import {LocalPlugin} from "app/common/plugin"; -import {DismissedPopup} from 'app/common/Prefs'; import {StringUnion} from 'app/common/StringUnion'; import {TableData} from 'app/common/TableData'; import {DocStateComparison} from 'app/common/UserAPI'; @@ -490,11 +489,6 @@ export class GristDoc extends DisposableWithEvents { this.draftMonitor = Drafts.create(this, this); this.cursorMonitor = CursorMonitor.create(this, this); this.editorMonitor = EditorMonitor.create(this, this); - - - G.window.resetSeenPopups = (seen = false) => { - this.docPageModel.appModel.dismissedPopups.set(seen ? DismissedPopup.values : []); - }; } /** diff --git a/app/client/models/AppModel.ts b/app/client/models/AppModel.ts index c8b75344..0841dc93 100644 --- a/app/client/models/AppModel.ts +++ b/app/client/models/AppModel.ts @@ -21,7 +21,7 @@ import {getDefaultThemePrefs, Theme, ThemeAppearance, ThemeColors, ThemePrefs, import {getThemeColors} from 'app/common/Themes'; import {getGristConfig} from 'app/common/urlUtils'; import {getOrgName, Organization, OrgError, UserAPI, UserAPIImpl} from 'app/common/UserAPI'; -import {getUserPrefObs, getUserPrefsObs} from 'app/client/models/UserPrefs'; +import {getUserPrefObs, getUserPrefsObs, markAsSeen, markAsUnSeen} from 'app/client/models/UserPrefs'; import {bundleChanges, Computed, Disposable, Observable, subscribe} from 'grainjs'; const t = makeT('AppModel'); @@ -111,6 +111,8 @@ export interface AppModel { isSupport(): boolean; // If user is a Support user isOwner(): boolean; // If user is an owner of this org isOwnerOrEditor(): boolean; // If user is an owner or editor of this org + /** Creates an computed observable to dismiss a popup or check if it was dismissed */ + dismissedPopup(name: DismissedPopup): Observable; } export class TopAppModelImpl extends Disposable implements TopAppModel { @@ -277,6 +279,10 @@ export class AppModelImpl extends Disposable implements AppModel { urlState().pushUrl({createTeam: false, params: {}}, {avoidReload: true, replace: true}).catch(() => {}); this.showNewSiteModal(state.params?.planType); } + + G.window.resetSeenPopups = (seen = false) => { + this.dismissedPopups.set(seen ? DismissedPopup.values : []); + }; } public get planName() { @@ -337,6 +343,18 @@ export class AppModelImpl extends Disposable implements AppModel { } } + public dismissedPopup(name: DismissedPopup): Computed { + const computed = Computed.create(null, use => use(this.dismissedPopups).includes(name)); + computed.onWrite(value => { + if (value) { + markAsSeen(this.dismissedPopups, name); + } else { + markAsUnSeen(this.dismissedPopups, name); + } + }); + return computed; + } + /** * If the current user is a new user, record a sign-up event via Google Tag Manager. */ diff --git a/app/client/models/UserPrefs.ts b/app/client/models/UserPrefs.ts index ad29cf76..94a0f3a3 100644 --- a/app/client/models/UserPrefs.ts +++ b/app/client/models/UserPrefs.ts @@ -98,3 +98,17 @@ export function markAsSeen(seenIdsObs: Observable, itemId: T console.warn("Failed to save preference in markAsSeen", e); } } + +export function markAsUnSeen(seenIdsObs: Observable, itemId: T) { + const seenIds = seenIdsObs.get() || []; + try { + if (seenIds.includes(itemId)) { + const seen = new Set(seenIds); + seen.delete(itemId); + seenIdsObs.set([...seen].sort()); + } + } catch (e) { + // tslint:disable-next-line:no-console + console.warn("Failed to save preference in markAsUnSeen", e); + } +} diff --git a/app/client/ui/DocMenu.ts b/app/client/ui/DocMenu.ts index 38323bbe..847ba9cc 100644 --- a/app/client/ui/DocMenu.ts +++ b/app/client/ui/DocMenu.ts @@ -12,6 +12,7 @@ import {attachAddNewTip} from 'app/client/ui/AddNewTip'; import * as css from 'app/client/ui/DocMenuCss'; import {buildHomeIntro, buildWorkspaceIntro} from 'app/client/ui/HomeIntro'; import {buildUpgradeButton} from 'app/client/ui/ProductUpgrades'; +import {buildTutorialCard} from 'app/client/ui/TutorialCard'; import {buildPinnedDoc, createPinnedDocs} from 'app/client/ui/PinnedDocs'; import {shadowScroll} from 'app/client/ui/shadowScroll'; import {transition} from 'app/client/ui/transitions'; @@ -71,7 +72,12 @@ function attachWelcomePopups(home: HomeModel): (el: Element) => void { function createLoadedDocMenu(owner: IDisposableOwner, home: HomeModel) { const flashDocId = observable(null); const upgradeButton = buildUpgradeButton(owner, home.app); - return css.docList( + return css.docList( /* vbox */ + /* first line */ + dom.create(buildTutorialCard, { app: home.app }), + /* hbox */ + css.docListContent( + /* left column - grow 1 */ css.docMenu( attachAddNewTip(home), @@ -169,7 +175,7 @@ function createLoadedDocMenu(owner: IDisposableOwner, home: HomeModel) { ), dom.maybe(use => !use(isNarrowScreenObs()) && ['all', 'workspace'].includes(use(home.currentPage)), () => upgradeButton.showUpgradeCard(css.upgradeCard.cls(''))), - ); + )); } function buildAllDocsBlock( diff --git a/app/client/ui/DocMenuCss.ts b/app/client/ui/DocMenuCss.ts index aa10acc2..c6034fae 100644 --- a/app/client/ui/DocMenuCss.ts +++ b/app/client/ui/DocMenuCss.ts @@ -8,11 +8,6 @@ import {bigBasicButton} from 'app/client/ui2018/buttons'; // styles, which gives it priority. import 'popweasel'; -export const docMenu = styled('div', ` - flex-grow: 1; - max-width: 100%; -`); - // The "&:after" clause forces some padding below all docs. export const docList = styled('div', ` height: 100%; @@ -20,6 +15,7 @@ export const docList = styled('div', ` overflow-y: auto; position: relative; display: flex; + flex-direction: column; &:after { content: ""; @@ -38,6 +34,15 @@ export const docList = styled('div', ` } `); +export const docListContent = styled('div', ` + display: flex; +`); + +export const docMenu = styled('div', ` + flex-grow: 1; + max-width: 100%; +`); + const listHeader = styled('div', ` min-height: 32px; line-height: 32px; diff --git a/app/client/ui/HomeLeftPane.ts b/app/client/ui/HomeLeftPane.ts index 7e378ed8..96104e63 100644 --- a/app/client/ui/HomeLeftPane.ts +++ b/app/client/ui/HomeLeftPane.ts @@ -16,7 +16,7 @@ import {testId, theme} from 'app/client/ui2018/cssVars'; import {icon} from 'app/client/ui2018/icons'; import {menu, menuIcon, menuItem, upgradableMenuItem, upgradeText} from 'app/client/ui2018/menus'; import {confirmModal} from 'app/client/ui2018/modals'; -import {shouldHideUiElement} from 'app/common/gristUrls'; +import {commonUrls, shouldHideUiElement} from 'app/common/gristUrls'; import * as roles from 'app/common/roles'; import {Workspace} from 'app/common/UserAPI'; import {computed, dom, domComputed, DomElementArg, observable, Observable, styled} from 'grainjs'; @@ -111,19 +111,26 @@ export function createHomeLeftPane(leftPanelOpen: Observable, home: Hom cssPageEntry( dom.hide(shouldHideUiElement("templates")), cssPageEntry.cls('-selected', (use) => use(home.currentPage) === "templates"), - cssPageLink(cssPageIcon('FieldTable'), cssLinkText(t("Examples & Templates")), + cssPageLink(cssPageIcon('Board'), cssLinkText(t("Examples & Templates")), urlState().setLinkUrl({homePage: "templates"}), testId('dm-templates-page'), ), ), cssPageEntry( cssPageEntry.cls('-selected', (use) => use(home.currentPage) === "trash"), - cssPageLink(cssPageIcon('Remove'), cssLinkText(t("Trash")), + cssPageLink(cssPageIcon('RemoveBig'), cssLinkText(t("Trash")), urlState().setLinkUrl({homePage: "trash"}), testId('dm-trash'), ), ), cssSpacer(), + cssPageEntry( + dom.hide(shouldHideUiElement("templates")), + cssPageLink(cssPageIcon('Bookmark'), cssLinkText(t("Tutorial")), + { href: commonUrls.basicTutorial, target: '_blank' }, + testId('dm-basic-tutorial'), + ), + ), createVideoTourToolsButton(), createHelpTools(home.app), ) diff --git a/app/client/ui/RightPanel.ts b/app/client/ui/RightPanel.ts index c3883f1a..676d27f1 100644 --- a/app/client/ui/RightPanel.ts +++ b/app/client/ui/RightPanel.ts @@ -758,7 +758,7 @@ const cssSeparator = styled('div', ` margin-top: 16px; `); -const cssConfigContainer = styled('div', ` +const cssConfigContainer = styled('div.test-config-container', ` overflow: auto; --color-list-item: none; --color-list-item-hover: none; diff --git a/app/client/ui/TutorialCard.ts b/app/client/ui/TutorialCard.ts new file mode 100644 index 00000000..81d72b5a --- /dev/null +++ b/app/client/ui/TutorialCard.ts @@ -0,0 +1,219 @@ +import {AppModel} from 'app/client/models/AppModel'; +import {bigPrimaryButton} from 'app/client/ui2018/buttons'; +import {isNarrowScreenObs, theme} from 'app/client/ui2018/cssVars'; +import {icon} from 'app/client/ui2018/icons'; +import {commonUrls, shouldHideUiElement} from 'app/common/gristUrls'; +import {Computed, dom, IDisposableOwner, makeTestId, styled} from 'grainjs'; + +const testId = makeTestId('test-tutorial-card-'); + +interface Options { + app: AppModel, + onStart?: () => void, +} + +export function buildTutorialCard(owner: IDisposableOwner, options: Options) { + const {app, onStart} = options; + const dismissed = app.dismissedPopup('tutorialFirstCard'); + owner.autoDispose(dismissed); + function onClose() { + dismissed.set(true); + } + const visible = Computed.create(owner, (use) => + !use(dismissed) + && !use(isNarrowScreenObs()) + && !shouldHideUiElement("templates") + ); + return dom.maybe(visible, () => { + return cssCard( + cssCaption( + dom('div', cssNewToGrist("New to Grist?")), + cssRelative( + cssStartHere("Start here."), + cssArrow() + ), + ), + cssContent( + testId('content'), + cssImage({src: commonUrls.basicTutorialImage}), + cssCardText( + cssLine(cssTitle("Grist Basics Tutorial")), + cssLine("Learn the basics of reference columns, linked widgets, column types, & cards."), + cssLine(cssSub('Beginner - 10 mins')), + cssButtonWrapper( + cssButtonWrapper.cls('-small'), + cssHeroButton("Start Tutorial"), + dom.on('click', () => onStart?.()) + ), + ), + ), + cssButtonWrapper( + cssButtonWrapper.cls('-big'), + cssHeroButton("Start Tutorial"), + {href: commonUrls.basicTutorial, target: '_blank'}, + ), + cssCloseButton(icon('CrossBig'), dom.on('click', () => onClose?.()), testId('close')), + ); + }); +} + +const cssContent = styled('div', ` + position: relative; + display: flex; + align-items: flex-start; + padding-top: 24px; + padding-bottom: 20px; + padding-right: 20px; + max-width: 460px; +`); + +const cssCardText = styled('div', ` + display: flex; + flex-direction: column; + justify-content: center; + align-self: stretch; + margin-left: 12px; +`); + +const cssRelative = styled('div', ` + position: relative; +`); + +const cssNewToGrist = styled('span', ` + font-style: normal; + font-weight: 400; + font-size: 24px; + line-height: 16px; + letter-spacing: 0.2px; + white-space: nowrap; +`); + +const cssStartHere = styled('span', ` + font-style: normal; + font-weight: 700; + font-size: 24px; + line-height: 16px; + letter-spacing: 0.2px; + white-space: nowrap; +`); + +const cssCaption = styled('div', ` + display: flex; + flex-direction: column; + gap: 12px; + margin-left: 32px; + margin-top: 42px; + margin-right: 64px; +`); + +const cssTitle = styled('span', ` + font-weight: 600; + font-size: 20px; +`); + +const cssSub = styled('span', ` + font-size: 12px; + color: ${theme.lightText}; +`); + +const cssLine = styled('div', ` + margin-bottom: 6px; +`); + +const cssHeroButton = styled(bigPrimaryButton, ` +`); + +const cssButtonWrapper = styled('a', ` + flex-grow: 1; + display: flex; + justify-content: flex-end; + margin-right: 60px; + align-items: center; + text-decoration: none; + &:hover { + text-decoration: none; + } + &-big .${cssHeroButton.className} { + padding: 16px 28px; + font-weight: 600; + font-size: 20px; + line-height: 1em; + } +`); + +const cssCloseButton = styled('div', ` + flex-shrink: 0; + align-self: flex-end; + cursor: pointer; + --icon-color: ${theme.controlSecondaryFg}; + margin: 8px 8px 4px 0px; + padding: 2px; + border-radius: 4px; + position: absolute; + top: 0; + right: 0; + &:hover { + background-color: ${theme.lightHover}; + } + &:active { + background-color: ${theme.hover}; + } +`); + +const cssImage = styled('img', ` + width: 187px; + height: 145px; + flex: none; +`); + +const cssArrow = styled('div', ` + position: absolute; + background-image: var(--icon-GreenArrow); + width: 94px; + height: 12px; + top: calc(50% - 6px); + left: calc(100% - 12px); + z-index: 1; +`); + + +const cssCard = styled('div', ` + display: flex; + position: relative; + color: ${theme.text}; + border-radius: 3px; + margin-bottom: 24px; + max-width: 1000px; + box-shadow: 0 2px 18px 0 ${theme.modalInnerShadow}, 0 0 1px 0 ${theme.modalOuterShadow}; + & .${cssButtonWrapper.className}-small { + display: none; + } + @media (max-width: 1320px) { + & .${cssButtonWrapper.className}-small { + flex-direction: column; + display: flex; + margin-top: 14px; + align-self: flex-start; + } + & .${cssButtonWrapper.className}-big { + display: none; + } + } + @media (max-width: 1000px) { + & .${cssArrow.className} { + display: none; + } + & .${cssCaption.className} { + flex-direction: row; + margin-bottom: 24px; + } + & { + flex-direction: column; + } + & .${cssContent.className} { + padding: 12px; + max-width: 100%; + margin-bottom: 28px; + } + } +`); diff --git a/app/client/ui2018/IconList.ts b/app/client/ui2018/IconList.ts index f51bfb71..b5be98da 100644 --- a/app/client/ui2018/IconList.ts +++ b/app/client/ui2018/IconList.ts @@ -39,6 +39,8 @@ export type IconName = "ChartArea" | "AddUser" | "BarcodeQR" | "BarcodeQR2" | + "Board" | + "Bookmark" | "CenterAlign" | "Chat" | "Code" | @@ -68,6 +70,7 @@ export type IconName = "ChartArea" | "FontStrikethrough" | "FontUnderline" | "FunctionResult" | + "GreenArrow" | "Grow" | "Help" | "Home" | @@ -105,6 +108,7 @@ export type IconName = "ChartArea" | "PublicFilled" | "Redo" | "Remove" | + "RemoveBig" | "Repl" | "ResizePanel" | "Revert" | @@ -176,6 +180,8 @@ export const IconList: IconName[] = ["ChartArea", "AddUser", "BarcodeQR", "BarcodeQR2", + "Board", + "Bookmark", "CenterAlign", "Chat", "Code", @@ -205,6 +211,7 @@ export const IconList: IconName[] = ["ChartArea", "FontStrikethrough", "FontUnderline", "FunctionResult", + "GreenArrow", "Grow", "Help", "Home", @@ -242,6 +249,7 @@ export const IconList: IconName[] = ["ChartArea", "PublicFilled", "Redo", "Remove", + "RemoveBig", "Repl", "ResizePanel", "Revert", diff --git a/app/common/Prefs.ts b/app/common/Prefs.ts index 1a676782..41abddff 100644 --- a/app/common/Prefs.ts +++ b/app/common/Prefs.ts @@ -100,8 +100,9 @@ export interface BehavioralPromptPrefs { * List of all popups that user can see and dismiss */ export const DismissedPopup = StringUnion( - 'deleteRecords', // confirmation for deleting records keyboard shortcut - 'deleteFields', // confirmation for deleting columns keyboard shortcut + 'deleteRecords', // confirmation for deleting records keyboard shortcut, + 'deleteFields', // confirmation for deleting columns keyboard shortcut, + 'tutorialFirstCard', // first card of the tutorial, ); export type DismissedPopup = typeof DismissedPopup.type; diff --git a/app/common/gristUrls.ts b/app/common/gristUrls.ts index 8aae2399..15dcd9db 100644 --- a/app/common/gristUrls.ts +++ b/app/common/gristUrls.ts @@ -77,6 +77,8 @@ export const commonUrls = { efcrConnect: 'https://efc-r.com/connect', efcrHelp: 'https://www.nioxus.info/eFCR-Help', + basicTutorial: 'https://templates.getgrist.com/woXtXUBmiN5T/Grist-Basics', + basicTutorialImage: 'https://www.getgrist.com/wp-content/uploads/2021/08/lightweight-crm.png', }; /** diff --git a/static/icons/icons.css b/static/icons/icons.css index bfc303bf..8f98ccf0 100644 --- a/static/icons/icons.css +++ b/static/icons/icons.css @@ -40,6 +40,8 @@ --icon-AddUser: url(''); --icon-BarcodeQR: url(''); --icon-BarcodeQR2: url(''); + --icon-Board: url(''); + --icon-Bookmark: url(''); --icon-CenterAlign: url(''); --icon-Chat: url(''); --icon-Code: url(''); @@ -69,6 +71,7 @@ --icon-FontStrikethrough: url(''); --icon-FontUnderline: url(''); --icon-FunctionResult: url(''); + --icon-GreenArrow: url(''); --icon-Grow: url(''); --icon-Help: url(''); --icon-Home: url(''); @@ -106,6 +109,7 @@ --icon-PublicFilled: url(''); --icon-Redo: url(''); --icon-Remove: url(''); + --icon-RemoveBig: url(''); --icon-Repl: url(''); --icon-ResizePanel: url(''); --icon-Revert: url(''); diff --git a/static/locales/en.client.json b/static/locales/en.client.json index 8d1117a4..755e0d9c 100644 --- a/static/locales/en.client.json +++ b/static/locales/en.client.json @@ -416,7 +416,7 @@ "Create Workspace": "Create Workspace", "Delete": "Delete", "Delete {{workspace}} and all included documents?": "Delete {{workspace}} and all included documents?", - "Examples & Templates": "Examples & Templates", + "Examples & Templates": "Templates", "Import Document": "Import Document", "Manage Users": "Manage Users", "Rename": "Rename", diff --git a/static/ui-icons/UI/Board.svg b/static/ui-icons/UI/Board.svg new file mode 100644 index 00000000..5aa1a46d --- /dev/null +++ b/static/ui-icons/UI/Board.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/static/ui-icons/UI/Bookmark.svg b/static/ui-icons/UI/Bookmark.svg new file mode 100644 index 00000000..d87a94e2 --- /dev/null +++ b/static/ui-icons/UI/Bookmark.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/static/ui-icons/UI/GreenArrow.svg b/static/ui-icons/UI/GreenArrow.svg new file mode 100644 index 00000000..b9730e95 --- /dev/null +++ b/static/ui-icons/UI/GreenArrow.svg @@ -0,0 +1,3 @@ + + + diff --git a/static/ui-icons/UI/RemoveBig.svg b/static/ui-icons/UI/RemoveBig.svg new file mode 100644 index 00000000..e11178dd --- /dev/null +++ b/static/ui-icons/UI/RemoveBig.svg @@ -0,0 +1,26 @@ + + + + Icons / UI / Remove + Created with Sketch. + + + + diff --git a/test/nbrowser/DocTutorial.ts b/test/nbrowser/DocTutorial.ts index 8d3ba803..2126d4e8 100644 --- a/test/nbrowser/DocTutorial.ts +++ b/test/nbrowser/DocTutorial.ts @@ -30,6 +30,19 @@ describe('DocTutorial', function () { session = await gu.session().anon.login(); }); + it('shows a tutorial card', async function() { + await session.loadRelPath('/'); + await gu.waitForDocMenuToLoad(); + await gu.skipWelcomeQuestions(); + + assert.isTrue(await driver.find('.test-tutorial-card-content').isDisplayed()); + // Can dismiss it. + await driver.find('.test-tutorial-card-close').click(); + assert.isFalse((await driver.findAll('.test-tutorial-card-content')).length > 0); + // When dismissed, we can see link in the menu. + assert.isTrue(await driver.find('.test-dm-basic-tutorial').isDisplayed()); + }); + it('redirects user to log in', async function() { await session.loadDoc(`/doc/${doc.id}`, false); await gu.checkLoginPage(); @@ -45,6 +58,34 @@ describe('DocTutorial', function () { afterEach(() => gu.checkForErrors()); + it('shows a tutorial card', async function() { + await session.loadRelPath('/'); + await gu.waitForDocMenuToLoad(); + await gu.skipWelcomeQuestions(); + + // Make sure we have clean start. + await driver.executeScript('resetSeenPopups();'); + await gu.waitForServer(); + await driver.navigate().refresh(); + await gu.waitForDocMenuToLoad(); + + // Make sure we see the card. + assert.isTrue(await driver.find('.test-tutorial-card-content').isDisplayed()); + + // And can dismiss it. + await driver.find('.test-tutorial-card-close').click(); + assert.isFalse((await driver.findAll('.test-tutorial-card-content')).length > 0); + + // When dismissed, we can see link in the menu. + assert.isTrue(await driver.find('.test-dm-basic-tutorial').isDisplayed()); + + // Prefs are preserved after reload. + await driver.navigate().refresh(); + await gu.waitForDocMenuToLoad(); + assert.isFalse((await driver.findAll('.test-tutorial-card-content')).length > 0); + assert.isTrue(await driver.find('.test-dm-basic-tutorial').isDisplayed()); + }); + it('creates a fork the first time the document is opened', async function() { await session.loadDoc(`/doc/${doc.id}`); await driver.wait(async () => { diff --git a/test/nbrowser/HomeIntro.ts b/test/nbrowser/HomeIntro.ts index 95f2084b..21733a99 100644 --- a/test/nbrowser/HomeIntro.ts +++ b/test/nbrowser/HomeIntro.ts @@ -248,7 +248,7 @@ describe('HomeIntro', function() { async function testSelectedExamplesPage() { // Click Examples & Templates in left panel. - await driver.findContent('.test-dm-templates-page', /Examples & Templates/).click(); + await driver.find('.test-dm-templates-page').click(); await gu.waitForDocMenuToLoad(); // Check Featured Templates are shown at the top of the page. diff --git a/test/nbrowser/gristUtils.ts b/test/nbrowser/gristUtils.ts index d7a80764..695aba3d 100644 --- a/test/nbrowser/gristUtils.ts +++ b/test/nbrowser/gristUtils.ts @@ -1977,6 +1977,7 @@ export class Session { freshAccount?: boolean, isFirstLogin?: boolean, showTips?: boolean, + skipTutorial?: boolean, // By default true retainExistingLogin?: boolean}) { // Optimize testing a little bit, so if we are already logged in as the expected // user on the expected org, and there are no options set, we can just continue. @@ -1994,6 +1995,11 @@ export class Session { } await server.simulateLogin(this.settings.name, this.settings.email, this.settings.orgDomain, {isFirstLogin: false, cacheCredentials: true, ...options}); + + if (options?.skipTutorial ?? true) { + await dismissTutorialCard(); + } + return this; } @@ -2033,10 +2039,7 @@ export class Session { if (wait === 'skipWelcomeQuestions') { // When waitForDocMenuToLoad() returns, welcome questions should also render, so that we // don't need to wait extra for them. - if (await driver.find('.test-welcome-questions').isPresent()) { - await driver.sendKeys(Key.ESCAPE); - assert.equal(await driver.find('.test-welcome-questions').isPresent(), false); - } + await skipWelcomeQuestions(); } } @@ -2802,13 +2805,35 @@ export async function dismissBehavioralPrompts() { } } +/** + * Dismisses any tutorial card that might be active. + */ +export async function dismissTutorialCard() { + // If there is something in our way, we can't do it. + if (await driver.find('.test-welcome-questions').isPresent()) { + return; + } + if (await driver.find('.test-tutorial-card-close').isPresent()) { + if (await driver.find('.test-tutorial-card-close').isDisplayed()) { + await driver.find('.test-tutorial-card-close').click(); + } + } +} + +/** + * Dismisses coaching call if needed. + */ +export async function dismissCoachingCall() { + const selector = '.test-coaching-call .test-popup-close-button'; + if ((await driver.findAll(selector)).length) { + await driver.find(selector).click(); + } +} + /** * Dismisses all card popups that are present. - * - * Optionally takes a `waitForServerTimeoutMs`, which may be null to skip waiting - * after closing each popup. */ - export async function dismissCardPopups(waitForServerTimeoutMs: number | null = 2000) { +export async function dismissCardPopups(waitForServerTimeoutMs: number | null = 2000) { let i = 0; const max = 10; @@ -2820,6 +2845,7 @@ export async function dismissBehavioralPrompts() { } } + /** * Confirms that anchor link was used for navigation. */ @@ -2920,6 +2946,17 @@ export function withComments() { }); } +/** + * Helper to scroll creator panel top or bottom. By default bottom. + */ +export function scrollPanel(top = false): WebElementPromise { + return new WebElementPromise(driver, + driver.executeScript((top: number) => { + document.getElementsByClassName('test-config-container')[0].scrollTop = top ? 0 : 10000; + }, top) + ); +} + /** * Helper to revert ACL changes. It first saves the current ACL data, and * then removes everything and adds it back. @@ -2968,6 +3005,13 @@ export async function setRangeFilterBound(minMax: 'min'|'max', value: string|{re } } +export async function skipWelcomeQuestions() { + if (await driver.find('.test-welcome-questions').isPresent()) { + await driver.sendKeys(Key.ESCAPE); + assert.equal(await driver.find('.test-welcome-questions').isPresent(), false); + } +} + } // end of namespace gristUtils stackWrapOwnMethods(gristUtils);