mirror of
https://github.com/overleaf/overleaf.git
synced 2024-11-21 20:47:08 -05:00
Add Writefull promo banner to project list (#12843)
GitOrigin-RevId: a41746a29dfba867ff60401cfce1478812358644
This commit is contained in:
parent
a12c274195
commit
4e6749966e
5 changed files with 102 additions and 0 deletions
|
@ -21,6 +21,7 @@ const UserPrimaryEmailCheckHandler = require('../User/UserPrimaryEmailCheckHandl
|
||||||
const UserController = require('../User/UserController')
|
const UserController = require('../User/UserController')
|
||||||
const LimitationsManager = require('../Subscription/LimitationsManager')
|
const LimitationsManager = require('../Subscription/LimitationsManager')
|
||||||
const NotificationsBuilder = require('../Notifications/NotificationsBuilder')
|
const NotificationsBuilder = require('../Notifications/NotificationsBuilder')
|
||||||
|
const SplitTestHandler = require('../SplitTests/SplitTestHandler')
|
||||||
|
|
||||||
/** @typedef {import("./types").GetProjectsRequest} GetProjectsRequest */
|
/** @typedef {import("./types").GetProjectsRequest} GetProjectsRequest */
|
||||||
/** @typedef {import("./types").GetProjectsResponse} GetProjectsResponse */
|
/** @typedef {import("./types").GetProjectsResponse} GetProjectsResponse */
|
||||||
|
@ -319,6 +320,23 @@ async function projectListPage(req, res, next) {
|
||||||
showGroupsAndEnterpriseBanner &&
|
showGroupsAndEnterpriseBanner &&
|
||||||
_.sample(['did-you-know', 'on-premise', 'people', 'FOMO'])
|
_.sample(['did-you-know', 'on-premise', 'people', 'FOMO'])
|
||||||
|
|
||||||
|
let showWritefullPromoBanner = false
|
||||||
|
if (Features.hasFeature('saas') && !req.session.justRegistered) {
|
||||||
|
try {
|
||||||
|
const { variant } = await SplitTestHandler.promises.getAssignment(
|
||||||
|
req,
|
||||||
|
res,
|
||||||
|
'writefull-promo-banner'
|
||||||
|
)
|
||||||
|
showWritefullPromoBanner = variant === 'enabled'
|
||||||
|
} catch (error) {
|
||||||
|
logger.warn(
|
||||||
|
{ err: error },
|
||||||
|
'failed to get "writefull-promo-banner" split test assignment'
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
res.render('project/list-react', {
|
res.render('project/list-react', {
|
||||||
title: 'your_projects',
|
title: 'your_projects',
|
||||||
usersBestSubscription,
|
usersBestSubscription,
|
||||||
|
@ -335,6 +353,7 @@ async function projectListPage(req, res, next) {
|
||||||
prefetchedProjectsBlob,
|
prefetchedProjectsBlob,
|
||||||
showGroupsAndEnterpriseBanner,
|
showGroupsAndEnterpriseBanner,
|
||||||
groupsAndEnterpriseBannerVariant,
|
groupsAndEnterpriseBannerVariant,
|
||||||
|
showWritefullPromoBanner,
|
||||||
projectDashboardReact: true, // used in navbar
|
projectDashboardReact: true, // used in navbar
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,6 +26,7 @@ block append meta
|
||||||
}))
|
}))
|
||||||
meta(name="ol-currentUrl" data-type="string" content=currentUrl)
|
meta(name="ol-currentUrl" data-type="string" content=currentUrl)
|
||||||
meta(name="ol-showGroupsAndEnterpriseBanner" data-type="boolean" content=showGroupsAndEnterpriseBanner)
|
meta(name="ol-showGroupsAndEnterpriseBanner" data-type="boolean" content=showGroupsAndEnterpriseBanner)
|
||||||
|
meta(name="ol-showWritefullPromoBanner" data-type="boolean" content=showWritefullPromoBanner)
|
||||||
meta(name="ol-groupsAndEnterpriseBannerVariant" data-type="string" content=groupsAndEnterpriseBannerVariant)
|
meta(name="ol-groupsAndEnterpriseBannerVariant" data-type="string" content=groupsAndEnterpriseBannerVariant)
|
||||||
|
|
||||||
block content
|
block content
|
||||||
|
|
|
@ -3,6 +3,7 @@ import Institution from './groups/institution'
|
||||||
import ConfirmEmail from './groups/confirm-email'
|
import ConfirmEmail from './groups/confirm-email'
|
||||||
import ReconfirmationInfo from './groups/affiliation/reconfirmation-info'
|
import ReconfirmationInfo from './groups/affiliation/reconfirmation-info'
|
||||||
import GroupsAndEnterpriseBanner from './groups-and-enterprise-banner'
|
import GroupsAndEnterpriseBanner from './groups-and-enterprise-banner'
|
||||||
|
import WritefullPromoBanner from './writefull-promo-banner'
|
||||||
|
|
||||||
function UserNotifications() {
|
function UserNotifications() {
|
||||||
return (
|
return (
|
||||||
|
@ -13,6 +14,7 @@ function UserNotifications() {
|
||||||
<ConfirmEmail />
|
<ConfirmEmail />
|
||||||
<ReconfirmationInfo />
|
<ReconfirmationInfo />
|
||||||
<GroupsAndEnterpriseBanner />
|
<GroupsAndEnterpriseBanner />
|
||||||
|
<WritefullPromoBanner />
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
|
|
|
@ -0,0 +1,77 @@
|
||||||
|
import { memo, useCallback, useState } from 'react'
|
||||||
|
import Notification from './notification'
|
||||||
|
import { sendMB } from '../../../../infrastructure/event-tracking'
|
||||||
|
import getMeta from '../../../../utils/meta'
|
||||||
|
import customLocalStorage from '../../../../infrastructure/local-storage'
|
||||||
|
|
||||||
|
const STORAGE_KEY = 'has_dismissed_writefull_promo_banner'
|
||||||
|
|
||||||
|
const eventSegmentation = {
|
||||||
|
location: 'dashboard-banner',
|
||||||
|
page: '/project',
|
||||||
|
name: 'writefull',
|
||||||
|
}
|
||||||
|
|
||||||
|
function WritefullPromoBanner() {
|
||||||
|
const [show, setShow] = useState(() => {
|
||||||
|
const show =
|
||||||
|
getMeta('ol-showWritefullPromoBanner') &&
|
||||||
|
!customLocalStorage.getItem(STORAGE_KEY)
|
||||||
|
|
||||||
|
if (show) {
|
||||||
|
sendMB('promo-prompt', eventSegmentation)
|
||||||
|
}
|
||||||
|
|
||||||
|
return show
|
||||||
|
})
|
||||||
|
|
||||||
|
const handleOpenLink = useCallback(() => {
|
||||||
|
sendMB('promo-click', eventSegmentation)
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
const handleClose = useCallback(() => {
|
||||||
|
customLocalStorage.setItem(STORAGE_KEY, new Date())
|
||||||
|
setShow(false)
|
||||||
|
sendMB('promo-dismiss', eventSegmentation)
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
if (!show) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Notification
|
||||||
|
bsStyle="info"
|
||||||
|
onDismiss={handleClose}
|
||||||
|
className="centered-alert"
|
||||||
|
>
|
||||||
|
<Notification.Body>
|
||||||
|
<span>
|
||||||
|
Get <b>10% off</b> Writefull premium—AI-based language feedback and
|
||||||
|
TeXGPT to help you write great papers faster. Use code:{' '}
|
||||||
|
<b>OVERLEAF10</b>
|
||||||
|
</span>
|
||||||
|
</Notification.Body>
|
||||||
|
<Notification.Action>
|
||||||
|
<a
|
||||||
|
className="pull-right btn btn-info btn-sm"
|
||||||
|
href="https://my.writefull.com/overleaf-invite?code=OVERLEAF10"
|
||||||
|
target="_blank"
|
||||||
|
rel="noreferrer"
|
||||||
|
onClick={handleOpenLink}
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
alt="Writefull Logo"
|
||||||
|
src=""
|
||||||
|
height={16}
|
||||||
|
width={16}
|
||||||
|
style={{ marginRight: 4 }}
|
||||||
|
/>
|
||||||
|
<span>Get Writefull for Overleaf</span>
|
||||||
|
</a>
|
||||||
|
</Notification.Action>
|
||||||
|
</Notification>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default memo(WritefullPromoBanner)
|
|
@ -248,6 +248,9 @@ input.project-list-table-select-item[type='checkbox'] {
|
||||||
flex-wrap: nowrap;
|
flex-wrap: nowrap;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
&.centered-alert .alert {
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
.notification-body {
|
.notification-body {
|
||||||
|
|
Loading…
Reference in a new issue