Skip to content

Commit

Permalink
Merge branch 'refactor/upgrade-modal' into rbac
Browse files Browse the repository at this point in the history
  • Loading branch information
benjackwhite committed Mar 18, 2024
2 parents 6583edf + 52da597 commit b4aac7a
Show file tree
Hide file tree
Showing 14 changed files with 137 additions and 114 deletions.
2 changes: 1 addition & 1 deletion frontend/src/layout/GlobalModals.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,14 +2,14 @@ import { LemonModal } from '@posthog/lemon-ui'
import { actions, kea, path, reducers, useActions, useValues } from 'kea'
import { FlaggedFeature } from 'lib/components/FlaggedFeature'
import { HedgehogBuddyWithLogic } from 'lib/components/HedgehogBuddy/HedgehogBuddyWithLogic'
import { UpgradeModal } from 'lib/components/UpgradeModal/UpgradeModal'
import { Prompt } from 'lib/logic/newPrompt/Prompt'
import { Setup2FA } from 'scenes/authentication/Setup2FA'
import { CreateOrganizationModal } from 'scenes/organization/CreateOrganizationModal'
import { membersLogic } from 'scenes/organization/membersLogic'
import { CreateProjectModal } from 'scenes/project/CreateProjectModal'
import { inviteLogic } from 'scenes/settings/organization/inviteLogic'
import { InviteModal } from 'scenes/settings/organization/InviteModal'
import { UpgradeModal } from 'scenes/UpgradeModal'
import { userLogic } from 'scenes/userLogic'

import type { globalModalsLogicType } from './GlobalModalsType'
Expand Down
7 changes: 3 additions & 4 deletions frontend/src/layout/navigation/OrganizationSwitcher.tsx
Original file line number Diff line number Diff line change
@@ -1,13 +1,13 @@
import { IconPlus } from '@posthog/icons'
import { useActions, useValues } from 'kea'
import { upgradeModalLogic } from 'lib/components/UpgradeModal/upgradeModalLogic'
import { LemonButton } from 'lib/lemon-ui/LemonButton'
import { LemonDivider } from 'lib/lemon-ui/LemonDivider'
import { LemonTag } from 'lib/lemon-ui/LemonTag/LemonTag'
import { Lettermark } from 'lib/lemon-ui/Lettermark'
import { membershipLevelToName } from 'lib/utils/permissioning'
import { organizationLogic } from 'scenes/organizationLogic'
import { preflightLogic } from 'scenes/PreflightCheck/preflightLogic'
import { sceneLogic } from 'scenes/sceneLogic'
import { userLogic } from 'scenes/userLogic'

import { AvailableFeature, OrganizationBasicType } from '~/types'
Expand Down Expand Up @@ -48,7 +48,7 @@ export function OtherOrganizationButton({
export function NewOrganizationButton(): JSX.Element {
const { closeAccountPopover } = useActions(navigationLogic)
const { showCreateOrganizationModal } = useActions(globalModalsLogic)
const { guardAvailableFeature } = useActions(sceneLogic)
const { guardAvailableFeature } = useValues(upgradeModalLogic)

return (
<LemonButton
Expand All @@ -61,8 +61,7 @@ export function NewOrganizationButton(): JSX.Element {
showCreateOrganizationModal()
},
{
cloud: false,
selfHosted: true,
guardOnCloud: false,
}
)
}
Expand Down
13 changes: 5 additions & 8 deletions frontend/src/layout/navigation/ProjectSwitcher.tsx
Original file line number Diff line number Diff line change
@@ -1,13 +1,13 @@
import { IconGear, IconPlus } from '@posthog/icons'
import { useActions, useValues } from 'kea'
import { router } from 'kea-router'
import { upgradeModalLogic } from 'lib/components/UpgradeModal/upgradeModalLogic'
import { LemonButton } from 'lib/lemon-ui/LemonButton'
import { LemonDivider } from 'lib/lemon-ui/LemonDivider'
import { LemonSnack } from 'lib/lemon-ui/LemonSnack/LemonSnack'
import { removeProjectIdIfPresent } from 'lib/utils/router-utils'
import { useMemo } from 'react'
import { organizationLogic } from 'scenes/organizationLogic'
import { sceneLogic } from 'scenes/sceneLogic'
import { isAuthenticatedTeam, teamLogic } from 'scenes/teamLogic'
import { urls } from 'scenes/urls'

Expand All @@ -27,7 +27,7 @@ export function ProjectName({ team }: { team: TeamBasicType }): JSX.Element {
export function ProjectSwitcherOverlay({ onClickInside }: { onClickInside?: () => void }): JSX.Element {
const { currentOrganization, projectCreationForbiddenReason } = useValues(organizationLogic)
const { currentTeam } = useValues(teamLogic)
const { guardAvailableFeature } = useActions(sceneLogic)
const { guardAvailableFeature } = useValues(upgradeModalLogic)
const { showCreateProjectModal } = useActions(globalModalsLogic)

return (
Expand All @@ -48,12 +48,9 @@ export function ProjectSwitcherOverlay({ onClickInside }: { onClickInside?: () =
tooltip={projectCreationForbiddenReason}
onClick={() => {
onClickInside?.()
guardAvailableFeature(
AvailableFeature.ORGANIZATIONS_PROJECTS,
showCreateProjectModal,
undefined,
currentOrganization?.teams?.length
)
guardAvailableFeature(AvailableFeature.ORGANIZATIONS_PROJECTS, showCreateProjectModal, {
currentUsage: currentOrganization?.teams?.length,
})
}}
>
New project
Expand Down
4 changes: 2 additions & 2 deletions frontend/src/lib/components/ObjectTags/ObjectTags.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -9,11 +9,11 @@ import { objectTagsLogic } from 'lib/components/ObjectTags/objectTagsLogic'
import { Spinner } from 'lib/lemon-ui/Spinner/Spinner'
import { colorForString } from 'lib/utils'
import { CSSProperties, useMemo } from 'react'
import { sceneLogic } from 'scenes/sceneLogic'

import { AvailableFeature } from '~/types'

import { SelectGradientOverflow } from '../SelectGradientOverflow'
import { upgradeModalLogic } from '../UpgradeModal/upgradeModalLogic'

interface ObjectTagsPropsBase {
tags: string[]
Expand Down Expand Up @@ -61,7 +61,7 @@ export function ObjectTags({
}: ObjectTagsProps): JSX.Element {
const objectTagId = useMemo(() => uniqueMemoizedIndex++, [])
const logic = objectTagsLogic({ id: objectTagId, onChange, tags })
const { guardAvailableFeature } = useActions(sceneLogic)
const { guardAvailableFeature } = useActions(upgradeModalLogic)
const { addingNewTag, cleanedNewTag, deletedTags } = useValues(logic)
const { setAddingNewTag, setNewTag, handleDelete, handleAdd } = useActions(logic)

Expand Down
4 changes: 2 additions & 2 deletions frontend/src/lib/components/PayGateMini/PayGateMini.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -10,10 +10,10 @@ import { useEffect } from 'react'
import { billingLogic } from 'scenes/billing/billingLogic'
import { preflightLogic } from 'scenes/PreflightCheck/preflightLogic'
import { getProductIcon } from 'scenes/products/Products'
import { sceneLogic } from 'scenes/sceneLogic'

import { AvailableFeature } from '~/types'

import { upgradeModalLogic } from '../UpgradeModal/upgradeModalLogic'
import { PayGateMiniButton } from './PayGateMiniButton'
import { payGateMiniLogic } from './payGateMiniLogic'

Expand Down Expand Up @@ -47,7 +47,7 @@ export function PayGateMini({
const { preflight } = useValues(preflightLogic)
const { billing, billingLoading } = useValues(billingLogic)
const { featureFlags } = useValues(featureFlagLogic)
const { hideUpgradeModal } = useActions(sceneLogic)
const { hideUpgradeModal } = useActions(upgradeModalLogic)

useEffect(() => {
if (gateVariant) {
Expand Down
4 changes: 2 additions & 2 deletions frontend/src/lib/components/Sharing/SharingModal.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -15,10 +15,10 @@ import { Tooltip } from 'lib/lemon-ui/Tooltip'
import { copyToClipboard } from 'lib/utils/copyToClipboard'
import { useEffect, useState } from 'react'
import { DashboardCollaboration } from 'scenes/dashboard/DashboardCollaborators'
import { sceneLogic } from 'scenes/sceneLogic'

import { AvailableFeature, InsightModel, InsightShortId, InsightType } from '~/types'

import { upgradeModalLogic } from '../UpgradeModal/upgradeModalLogic'
import { sharingLogic } from './sharingLogic'

export const SHARING_MODAL_WIDTH = 600
Expand Down Expand Up @@ -64,7 +64,7 @@ export function SharingModalContent({
shareLink,
} = useValues(sharingLogic(logicProps))
const { setIsEnabled, togglePreview } = useActions(sharingLogic(logicProps))
const { guardAvailableFeature } = useActions(sceneLogic)
const { guardAvailableFeature } = useActions(upgradeModalLogic)

const [iframeLoaded, setIframeLoaded] = useState(false)

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,11 +2,12 @@ import { LemonModal } from '@posthog/lemon-ui'
import { useActions, useValues } from 'kea'
import { PayGateMini } from 'lib/components/PayGateMini/PayGateMini'

import { sceneLogic } from './sceneLogic'
import { upgradeModalLogic } from './upgradeModalLogic'

export function UpgradeModal(): JSX.Element {
const { upgradeModalFeatureKey, upgradeModalFeatureUsage, upgradeModalIsGrandfathered } = useValues(sceneLogic)
const { hideUpgradeModal } = useActions(sceneLogic)
const { upgradeModalFeatureKey, upgradeModalFeatureUsage, upgradeModalIsGrandfathered } =
useValues(upgradeModalLogic)
const { hideUpgradeModal } = useActions(upgradeModalLogic)

return upgradeModalFeatureKey ? (
<LemonModal onClose={hideUpgradeModal} isOpen={!!upgradeModalFeatureKey}>
Expand Down
96 changes: 96 additions & 0 deletions frontend/src/lib/components/UpgradeModal/upgradeModalLogic.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,96 @@
import { actions, connect, kea, listeners, path, reducers, selectors } from 'kea'
import { featureFlagLogic } from 'lib/logic/featureFlagLogic'
import { eventUsageLogic } from 'lib/utils/eventUsageLogic'
import { preflightLogic } from 'scenes/PreflightCheck/preflightLogic'
import { userLogic } from 'scenes/userLogic'

import { AvailableFeature } from '~/types'

import type { upgradeModalLogicType } from './upgradeModalLogicType'

export type GuardAvailableFeatureFn = (
featureKey: AvailableFeature,
featureAvailableCallback?: () => void,
options?: {
guardOnCloud?: boolean
guardOnSelfHosted?: boolean
currentUsage?: number
isGrandfathered?: boolean
}
) => boolean

export const upgradeModalLogic = kea<upgradeModalLogicType>([
path(['lib', 'components', 'UpgradeModal', 'upgradeModalLogic']),
connect(() => ({
values: [preflightLogic, ['preflight'], featureFlagLogic, ['featureFlags'], userLogic, ['hasAvailableFeature']],
})),
actions({
showUpgradeModal: (featureKey: AvailableFeature, currentUsage?: number, isGrandfathered?: boolean) => ({
featureKey,
currentUsage,
isGrandfathered,
}),
hideUpgradeModal: true,
}),
reducers({
upgradeModalFeatureKey: [
null as AvailableFeature | null,
{
showUpgradeModal: (_, { featureKey }) => featureKey,
hideUpgradeModal: () => null,
},
],
upgradeModalFeatureUsage: [
null as number | null,
{
showUpgradeModal: (_, { currentUsage }) => currentUsage ?? null,
hideUpgradeModal: () => null,
},
],
upgradeModalIsGrandfathered: [
null as boolean | null,
{
showUpgradeModal: (_, { isGrandfathered }) => isGrandfathered ?? null,
hideUpgradeModal: () => null,
},
],
}),
selectors(({ actions }) => ({
guardAvailableFeature: [
(s) => [s.preflight, s.hasAvailableFeature],
(preflight, hasAvailableFeature): GuardAvailableFeatureFn => {
return (featureKey, featureAvailableCallback, options): boolean => {
const {
guardOnCloud = true,
guardOnSelfHosted = true,
currentUsage,
isGrandfathered,
} = options || {}
let featureAvailable: boolean
if (!preflight) {
featureAvailable = false
} else if (!guardOnCloud && preflight.cloud) {
featureAvailable = true
} else if (!guardOnSelfHosted && !preflight.cloud) {
featureAvailable = true
} else {
featureAvailable = hasAvailableFeature(featureKey, currentUsage)
}

if (!featureAvailable) {
actions.showUpgradeModal(featureKey, currentUsage, isGrandfathered)
} else {
featureAvailableCallback?.()
}

return featureAvailable
}
},
],
})),
listeners(() => ({
showUpgradeModal: ({ featureKey }) => {
eventUsageLogic.actions.reportUpgradeModalShown(featureKey)
},
})),
])
66 changes: 1 addition & 65 deletions frontend/src/scenes/sceneLogic.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,14 +4,13 @@ import { commandBarLogic } from 'lib/components/CommandBar/commandBarLogic'
import { BarStatus } from 'lib/components/CommandBar/types'
import { FEATURE_FLAGS, TeamMembershipLevel } from 'lib/constants'
import { featureFlagLogic } from 'lib/logic/featureFlagLogic'
import { eventUsageLogic } from 'lib/utils/eventUsageLogic'
import { addProjectIdIfMissing, removeProjectIdIfPresent } from 'lib/utils/router-utils'
import posthog from 'posthog-js'
import { emptySceneParams, preloadedScenes, redirects, routes, sceneConfigurations } from 'scenes/scenes'
import { LoadedScene, Params, Scene, SceneConfig, SceneExport, SceneParams } from 'scenes/sceneTypes'
import { urls } from 'scenes/urls'

import { AvailableFeature, ProductKey } from '~/types'
import { ProductKey } from '~/types'

import { handleLoginRedirect } from './authentication/loginLogic'
import { onboardingLogic, OnboardingStepKey } from './onboarding/onboardingLogic'
Expand Down Expand Up @@ -52,27 +51,6 @@ export const sceneLogic = kea<sceneLogicType>([
setLoadedScene: (loadedScene: LoadedScene) => ({
loadedScene,
}),
showUpgradeModal: (featureKey: AvailableFeature, currentUsage?: number, isGrandfathered?: boolean) => ({
featureKey,
currentUsage,
isGrandfathered,
}),
guardAvailableFeature: (
featureKey: AvailableFeature,
featureAvailableCallback?: () => void,
guardOn: {
cloud: boolean
selfHosted: boolean
} = {
cloud: true,
selfHosted: true,
},
// how much of the feature has been used (eg. number of recording playlists created),
// which will be compared to the limit for their subscriptions
currentUsage?: number,
isGrandfathered?: boolean
) => ({ featureKey, featureAvailableCallback, guardOn, currentUsage, isGrandfathered }),
hideUpgradeModal: true,
reloadBrowserDueToImportError: true,
}),
reducers({
Expand Down Expand Up @@ -105,27 +83,6 @@ export const sceneLogic = kea<sceneLogicType>([
setScene: () => null,
},
],
upgradeModalFeatureKey: [
null as AvailableFeature | null,
{
showUpgradeModal: (_, { featureKey }) => featureKey,
hideUpgradeModal: () => null,
},
],
upgradeModalFeatureUsage: [
null as number | null,
{
showUpgradeModal: (_, { currentUsage }) => currentUsage ?? null,
hideUpgradeModal: () => null,
},
],
upgradeModalIsGrandfathered: [
null as boolean | null,
{
showUpgradeModal: (_, { isGrandfathered }) => isGrandfathered ?? null,
hideUpgradeModal: () => null,
},
],
lastReloadAt: [
null as number | null,
{ persist: true },
Expand Down Expand Up @@ -170,27 +127,6 @@ export const sceneLogic = kea<sceneLogicType>([
hashParams: [(s) => [s.sceneParams], (sceneParams): Record<string, any> => sceneParams.hashParams || {}],
}),
listeners(({ values, actions, props, selectors }) => ({
showUpgradeModal: ({ featureKey }) => {
eventUsageLogic.actions.reportUpgradeModalShown(featureKey)
},
guardAvailableFeature: ({ featureKey, featureAvailableCallback, guardOn, currentUsage, isGrandfathered }) => {
const { preflight } = preflightLogic.values
let featureAvailable: boolean
if (!preflight) {
featureAvailable = false
} else if (!guardOn.cloud && preflight.cloud) {
featureAvailable = true
} else if (!guardOn.selfHosted && !preflight.cloud) {
featureAvailable = true
} else {
featureAvailable = userLogic.values.hasAvailableFeature(featureKey, currentUsage)
}
if (featureAvailable) {
featureAvailableCallback?.()
} else {
actions.showUpgradeModal(featureKey, currentUsage, isGrandfathered)
}
},
setScene: ({ scene, scrollToTop }, _, __, previousState) => {
posthog.capture('$pageview')

Expand Down
Loading

0 comments on commit b4aac7a

Please sign in to comment.