diff --git a/frontend/src/lib/components/IframedToolbarBrowser/IframedToolbarBrowser.tsx b/frontend/src/lib/components/IframedToolbarBrowser/IframedToolbarBrowser.tsx
index 3e34575b31c96..aa32b83e36ee7 100644
--- a/frontend/src/lib/components/IframedToolbarBrowser/IframedToolbarBrowser.tsx
+++ b/frontend/src/lib/components/IframedToolbarBrowser/IframedToolbarBrowser.tsx
@@ -35,10 +35,10 @@ export function IframedToolbarBrowser({
iframeRef,
userIntent,
}: {
- iframeRef?: React.MutableRefObject
+ iframeRef: React.MutableRefObject
userIntent: ToolbarUserIntent
}): JSX.Element | null {
- const logic = iframedToolbarBrowserLogic()
+ const logic = iframedToolbarBrowserLogic({ iframeRef, userIntent: userIntent })
const { browserUrl } = useValues(logic)
const { onIframeLoad, setIframeWidth } = useActions(logic)
diff --git a/frontend/src/lib/components/IframedToolbarBrowser/iframedToolbarBrowserLogic.ts b/frontend/src/lib/components/IframedToolbarBrowser/iframedToolbarBrowserLogic.ts
index 71a59fdde32a1..a2d74018d6aa8 100644
--- a/frontend/src/lib/components/IframedToolbarBrowser/iframedToolbarBrowserLogic.ts
+++ b/frontend/src/lib/components/IframedToolbarBrowser/iframedToolbarBrowserLogic.ts
@@ -10,11 +10,14 @@ import { LemonBannerProps } from 'lib/lemon-ui/LemonBanner'
import posthog from 'posthog-js'
import { RefObject } from 'react'
+import { ToolbarUserIntent } from '~/types'
+
import type { iframedToolbarBrowserLogicType } from './iframedToolbarBrowserLogicType'
export type IframedToolbarBrowserLogicProps = {
iframeRef: RefObject
clearBrowserUrlOnUnmount?: boolean
+ userIntent?: ToolbarUserIntent
}
export interface IFrameBanner {
@@ -50,9 +53,13 @@ export const iframedToolbarBrowserLogic = kea([
setIframeBanner: (banner: IFrameBanner | null) => ({ banner }),
startTrackingLoading: true,
stopTrackingLoading: true,
+ enableElementSelector: true,
+ disableElementSelector: true,
+ setNewActionName: (name: string | null) => ({ name }),
+ toolbarMessageReceived: (type: PostHogAppToolbarEvent, payload: Record) => ({ type, payload }),
}),
- reducers({
+ reducers(({ props }) => ({
// they're called common filters in the toolbar because they're shared between heatmaps and clickmaps
// the name is continued here since they're passed down into the embedded iframe
commonFilters: [
@@ -87,7 +94,7 @@ export const iframedToolbarBrowserLogic = kea([
],
browserUrl: [
null as string | null,
- { persist: true },
+ { persist: props.userIntent == 'heatmaps' },
{
setBrowserUrl: (_, { url }) => url,
},
@@ -107,7 +114,7 @@ export const iframedToolbarBrowserLogic = kea([
setIframeBanner: (_, { banner }) => banner,
},
],
- }),
+ })),
selectors({
isBrowserUrlAuthorized: [
@@ -138,7 +145,7 @@ export const iframedToolbarBrowserLogic = kea([
'*'
)
},
-
+ // heatmaps
patchHeatmapFilters: ({ filters }) => {
actions.sendToolbarMessage(PostHogAppToolbarEvent.PH_PATCH_HEATMAP_FILTERS, { filters })
},
@@ -159,6 +166,17 @@ export const iframedToolbarBrowserLogic = kea([
actions.sendToolbarMessage(PostHogAppToolbarEvent.PH_HEATMAPS_COMMON_FILTERS, { commonFilters: filters })
},
+ // actions
+ enableElementSelector: () => {
+ actions.sendToolbarMessage(PostHogAppToolbarEvent.PH_ELEMENT_SELECTOR, { enabled: true })
+ },
+ disableElementSelector: () => {
+ actions.sendToolbarMessage(PostHogAppToolbarEvent.PH_ELEMENT_SELECTOR, { enabled: false })
+ },
+ setNewActionName: ({ name }) => {
+ actions.sendToolbarMessage(PostHogAppToolbarEvent.PH_NEW_ACTION_NAME, { name })
+ },
+
onIframeLoad: () => {
// we get this callback whether the iframe loaded successfully or not
// and don't get a signal if the load was successful, so we have to check
@@ -171,13 +189,20 @@ export const iframedToolbarBrowserLogic = kea([
fixedPositionMode: values.heatmapFixedPositionMode,
commonFilters: values.commonFilters,
})
- actions.sendToolbarMessage(PostHogAppToolbarEvent.PH_HEATMAPS_CONFIG, {
- enabled: true,
- })
+ switch (props.userIntent) {
+ case 'heatmaps':
+ actions.sendToolbarMessage(PostHogAppToolbarEvent.PH_HEATMAPS_CONFIG, {
+ enabled: true,
+ })
+ break
+ }
}
const onIframeMessage = (e: MessageEvent): void => {
const type: PostHogAppToolbarEvent = e?.data?.type
+ const payload = e?.data?.payload
+
+ actions.toolbarMessageReceived(type, payload)
if (!type || !type.startsWith('ph-')) {
return
@@ -195,14 +220,17 @@ export const iframedToolbarBrowserLogic = kea([
case PostHogAppToolbarEvent.PH_TOOLBAR_INIT:
return init()
case PostHogAppToolbarEvent.PH_TOOLBAR_READY:
- posthog.capture('in-app heatmap frame loaded', {
- inapp_heatmap_page_url_visited: values.browserUrl,
- inapp_heatmap_filters: values.heatmapFilters,
- inapp_heatmap_color_palette: values.heatmapColorPalette,
- inapp_heatmap_fixed_position_mode: values.heatmapFixedPositionMode,
- })
- // reset loading tracking - if we're e.g. slow this will avoid a flash of warning message
- return actions.startTrackingLoading()
+ if (props.userIntent === 'heatmaps') {
+ posthog.capture('in-app heatmap frame loaded', {
+ inapp_heatmap_page_url_visited: values.browserUrl,
+ inapp_heatmap_filters: values.heatmapFilters,
+ inapp_heatmap_color_palette: values.heatmapColorPalette,
+ inapp_heatmap_fixed_position_mode: values.heatmapFixedPositionMode,
+ })
+ // reset loading tracking - if we're e.g. slow this will avoid a flash of warning message
+ return actions.startTrackingLoading()
+ }
+ return
case PostHogAppToolbarEvent.PH_TOOLBAR_HEATMAP_LOADING:
return actions.startTrackingLoading()
case PostHogAppToolbarEvent.PH_TOOLBAR_HEATMAP_LOADED:
@@ -223,6 +251,10 @@ export const iframedToolbarBrowserLogic = kea([
actions.stopTrackingLoading()
actions.setIframeBanner({ level: 'error', message: 'The heatmap failed to load.' })
return
+ case PostHogAppToolbarEvent.PH_NEW_ACTION_CREATED:
+ actions.setNewActionName(null)
+ actions.disableElementSelector()
+ return
default:
console.warn(`[PostHog Heatmaps] Received unknown child window message: ${type}`)
}
diff --git a/frontend/src/lib/components/IframedToolbarBrowser/utils.ts b/frontend/src/lib/components/IframedToolbarBrowser/utils.ts
index dbb47dd057d3c..87d9cae13cf9f 100644
--- a/frontend/src/lib/components/IframedToolbarBrowser/utils.ts
+++ b/frontend/src/lib/components/IframedToolbarBrowser/utils.ts
@@ -14,6 +14,9 @@ export enum PostHogAppToolbarEvent {
PH_TOOLBAR_HEATMAP_LOADING = 'ph-toolbar-heatmap-loading',
PH_TOOLBAR_HEATMAP_LOADED = 'ph-toolbar-heatmap-loaded',
PH_TOOLBAR_HEATMAP_FAILED = 'ph-toolbar-heatmap-failed',
+ PH_ELEMENT_SELECTOR = 'ph-element-selector',
+ PH_NEW_ACTION_NAME = 'ph-new-action-name',
+ PH_NEW_ACTION_CREATED = 'ph-new-action-created',
}
export const DEFAULT_HEATMAP_FILTERS: HeatmapFilters = {
diff --git a/frontend/src/scenes/dashboard/dashboardTemplateVariablesLogic.ts b/frontend/src/scenes/dashboard/dashboardTemplateVariablesLogic.ts
index da23d22466d25..e12513e025d38 100644
--- a/frontend/src/scenes/dashboard/dashboardTemplateVariablesLogic.ts
+++ b/frontend/src/scenes/dashboard/dashboardTemplateVariablesLogic.ts
@@ -1,7 +1,18 @@
-import { actions, kea, listeners, path, props, propsChanged, reducers, selectors } from 'kea'
+import { actions, connect, kea, listeners, path, props, propsChanged, reducers, selectors } from 'kea'
+import { iframedToolbarBrowserLogic } from 'lib/components/IframedToolbarBrowser/iframedToolbarBrowserLogic'
+import { PostHogAppToolbarEvent } from 'lib/components/IframedToolbarBrowser/utils'
import { isEmptyObject } from 'lib/utils'
-import { DashboardTemplateVariableType, FilterType, Optional } from '~/types'
+import {
+ ActionType,
+ BaseMathType,
+ DashboardTemplateVariableType,
+ EntityType,
+ EntityTypes,
+ FilterType,
+ Optional,
+ TemplateVariableStep,
+} from '~/types'
import type { dashboardTemplateVariablesLogicType } from './dashboardTemplateVariablesLogicType'
@@ -18,17 +29,22 @@ const FALLBACK_EVENT = {
export const dashboardTemplateVariablesLogic = kea([
path(['scenes', 'dashboard', 'DashboardTemplateVariablesLogic']),
props({ variables: [] } as DashboardTemplateVariablesLogicProps),
+ connect({
+ actions: [iframedToolbarBrowserLogic, ['toolbarMessageReceived', 'disableElementSelector']],
+ }),
actions({
setVariables: (variables: DashboardTemplateVariableType[]) => ({ variables }),
setVariable: (variableName: string, filterGroup: Optional) => ({
variable_name: variableName,
filterGroup,
}),
+ setVariableFromAction: (variableName: string, action: ActionType) => ({ variableName, action }),
setActiveVariableIndex: (index: number) => ({ index }),
incrementActiveVariableIndex: true,
possiblyIncrementActiveVariableIndex: true,
resetVariable: (variableId: string) => ({ variableId }),
goToNextUntouchedActiveVariableIndex: true,
+ setIsCurrentlySelectingElement: (isSelecting: boolean) => ({ isSelecting }),
}),
reducers({
variables: [
@@ -43,10 +59,23 @@ export const dashboardTemplateVariablesLogic = kea {
- // TODO: handle actions as well as events
+ // There is only one type with contents at a time
+ // So iterate through the types to find the first one with contents
+ const typeWithContents: EntityType = Object.keys(filterGroup).filter(
+ (group) => (filterGroup[group as EntityType] || [])?.length > 0
+ )?.[0] as EntityType
+
+ if (!typeWithContents) {
+ return state
+ }
+
return state.map((v: DashboardTemplateVariableType) => {
- if (v.name === variableName && filterGroup?.events?.length && filterGroup.events[0]) {
- return { ...v, default: filterGroup.events[0], touched: true }
+ if (
+ v.name === variableName &&
+ filterGroup?.[typeWithContents]?.length &&
+ filterGroup?.[typeWithContents]?.[0]
+ ) {
+ return { ...v, default: filterGroup[typeWithContents]?.[0] || {}, touched: true }
}
return { ...v }
})
@@ -68,6 +97,12 @@ export const dashboardTemplateVariablesLogic = kea state + 1,
},
],
+ isCurrentlySelectingElement: [
+ false as boolean,
+ {
+ setIsCurrentlySelectingElement: (_, { isSelecting }) => isSelecting,
+ },
+ ],
}),
selectors(() => ({
activeVariable: [
@@ -82,6 +117,12 @@ export const dashboardTemplateVariablesLogic = kea v.touched)
},
],
+ hasTouchedAnyVariable: [
+ (s) => [s.variables],
+ (variables: DashboardTemplateVariableType[]) => {
+ return variables.some((v) => v.touched)
+ },
+ ],
})),
listeners(({ actions, props, values }) => ({
possiblyIncrementActiveVariableIndex: () => {
@@ -103,6 +144,30 @@ export const dashboardTemplateVariablesLogic = kea {
+ const originalVariableName = variableName.replace(/\s-\s\d+/g, '')
+ const step: TemplateVariableStep = {
+ id: action.id.toString(),
+ math: BaseMathType.UniqueUsers,
+ name: action.name,
+ order: 0,
+ type: EntityTypes.ACTIONS,
+ selector: action.steps?.[0]?.selector,
+ href: action.steps?.[0]?.href,
+ url: action.steps?.[0]?.url,
+ }
+ const filterGroup: FilterType = {
+ actions: [step],
+ }
+ actions.setVariable(originalVariableName, filterGroup)
+ actions.setIsCurrentlySelectingElement(false)
+ },
+ toolbarMessageReceived: ({ type, payload }) => {
+ if (type === PostHogAppToolbarEvent.PH_NEW_ACTION_CREATED) {
+ actions.setVariableFromAction(payload.action.name, payload.action as ActionType)
+ actions.disableElementSelector()
+ }
+ },
})),
propsChanged(({ actions, props }, oldProps) => {
if (props.variables !== oldProps.variables) {
diff --git a/frontend/src/scenes/onboarding/productAnalyticsSteps/DashboardTemplateConfigureStep.tsx b/frontend/src/scenes/onboarding/productAnalyticsSteps/DashboardTemplateConfigureStep.tsx
index 2bed8e8a07d2f..476cef6de6275 100644
--- a/frontend/src/scenes/onboarding/productAnalyticsSteps/DashboardTemplateConfigureStep.tsx
+++ b/frontend/src/scenes/onboarding/productAnalyticsSteps/DashboardTemplateConfigureStep.tsx
@@ -8,7 +8,7 @@ import { useRef, useState } from 'react'
import { dashboardTemplateVariablesLogic } from 'scenes/dashboard/dashboardTemplateVariablesLogic'
import { newDashboardLogic } from 'scenes/dashboard/newDashboardLogic'
-import { OnboardingStepKey } from '../onboardingLogic'
+import { onboardingLogic, OnboardingStepKey } from '../onboardingLogic'
import { OnboardingStep } from '../OnboardingStep'
import { sdksLogic } from '../sdks/sdksLogic'
import { DashboardTemplateVariables } from './DashboardTemplateVariables'
@@ -30,7 +30,8 @@ export const OnboardingDashboardTemplateConfigureStep = ({
const theDashboardTemplateVariablesLogic = dashboardTemplateVariablesLogic({
variables: activeDashboardTemplate?.variables || [],
})
- const { variables, allVariablesAreTouched } = useValues(theDashboardTemplateVariablesLogic)
+ const { variables, allVariablesAreTouched, hasTouchedAnyVariable } = useValues(theDashboardTemplateVariablesLogic)
+ const { goToNextStep, setStepKey } = useActions(onboardingLogic)
const [isSubmitting, setIsSubmitting] = useState(false)
@@ -58,32 +59,50 @@ export const OnboardingDashboardTemplateConfigureStep = ({
Select where you want to track events from.
-
- Not seeing the site you want? Install posthog-js or the HTML snippet
- wherever you want to track events, then come back here.
-
{snippetHosts.length > 0 ? (
-
- {snippetHosts.map((host) => (
- {
- addUrl(host)
- setBrowserUrl(host)
- }}
- sideIcon={ }
- >
- {host}
-
- ))}
-
+ <>
+
+ Not seeing the site you want? Install posthog-js or the HTML
+ snippet wherever you want to track events, then come back here.
+
+
+ {snippetHosts.map((host) => (
+ {
+ addUrl(host)
+ setBrowserUrl(host)
+ }}
+ sideIcon={ }
+ >
+ {host}
+
+ ))}
+
+ >
) : (
-
- Hm, we're not finding any available hosts. Head back to the install
- step to install posthog-js in your frontend.
-
+ <>
+
+ Hm, it looks like you haven't ingested any events from a website
+ yet. To select actions from your site, head back to the{' '}
+ setStepKey(OnboardingStepKey.INSTALL)}>
+ install step
+ {' '}
+ to install posthog-js in your frontend.
+
+
+ You can still create a dashboard using custom event names,
+ though it's not quite as fun.
+
+ setStepKey(OnboardingStepKey.INSTALL)}
+ type="primary"
+ >
+ Install posthog-js
+
+ >
)}
@@ -116,24 +135,35 @@ export const OnboardingDashboardTemplateConfigureStep = ({
{' '}
(no need to send it now) .
-
- {
- if (activeDashboardTemplate) {
- setIsSubmitting(true)
- createDashboardFromTemplate(activeDashboardTemplate, variables, false)
- }
- }}
- loading={isLoading}
- fullWidth
- center
- className="mt-6"
- disabledReason={!allVariablesAreTouched && 'Please select an event for each variable'}
- >
- Create dashboard
-
+
+
+
+ {
+ if (activeDashboardTemplate) {
+ setIsSubmitting(true)
+ createDashboardFromTemplate(activeDashboardTemplate, variables, false)
+ }
+ }}
+ loading={isLoading}
+ fullWidth
+ center
+ className="grow"
+ disabledReason={
+ !allVariablesAreTouched && 'Please select an event for each variable'
+ }
+ >
+ Create dashboard
+
+
+
+ goToNextStep()} fullWidth center>
+ {hasTouchedAnyVariable ? 'Discard dashboard & skip' : 'Skip for now'}
+
+
+
>
diff --git a/frontend/src/scenes/onboarding/productAnalyticsSteps/DashboardTemplateVariables.tsx b/frontend/src/scenes/onboarding/productAnalyticsSteps/DashboardTemplateVariables.tsx
index e22f822eedcfe..9f87ff0aeb0f4 100644
--- a/frontend/src/scenes/onboarding/productAnalyticsSteps/DashboardTemplateVariables.tsx
+++ b/frontend/src/scenes/onboarding/productAnalyticsSteps/DashboardTemplateVariables.tsx
@@ -1,6 +1,7 @@
-import { IconCheckCircle, IconInfo, IconTrash } from '@posthog/icons'
-import { LemonBanner, LemonButton, LemonCollapse, LemonInput, LemonLabel } from '@posthog/lemon-ui'
+import { IconCheckCircle, IconInfo, IconTarget, IconTrash } from '@posthog/icons'
+import { LemonBanner, LemonButton, LemonCollapse, LemonInput, LemonLabel, Spinner } from '@posthog/lemon-ui'
import { useActions, useValues } from 'kea'
+import { iframedToolbarBrowserLogic } from 'lib/components/IframedToolbarBrowser/iframedToolbarBrowserLogic'
import { useEffect, useState } from 'react'
import { dashboardTemplateVariablesLogic } from 'scenes/dashboard/dashboardTemplateVariablesLogic'
import { newDashboardLogic } from 'scenes/dashboard/newDashboardLogic'
@@ -8,26 +9,37 @@ import { newDashboardLogic } from 'scenes/dashboard/newDashboardLogic'
import { DashboardTemplateVariableType } from '~/types'
function VariableSelector({
- variable,
+ variableName,
hasSelectedSite,
+ iframeRef,
}: {
- variable: DashboardTemplateVariableType
+ variableName: string
hasSelectedSite: boolean
+ iframeRef: React.RefObject
}): JSX.Element {
const { activeDashboardTemplate } = useValues(newDashboardLogic)
const theDashboardTemplateVariablesLogic = dashboardTemplateVariablesLogic({
variables: activeDashboardTemplate?.variables || [],
})
- const { setVariable, resetVariable, goToNextUntouchedActiveVariableIndex, incrementActiveVariableIndex } =
- useActions(theDashboardTemplateVariablesLogic)
- const { allVariablesAreTouched, variables, activeVariableIndex } = useValues(theDashboardTemplateVariablesLogic)
+ const {
+ setVariable,
+ resetVariable,
+ goToNextUntouchedActiveVariableIndex,
+ incrementActiveVariableIndex,
+ setIsCurrentlySelectingElement,
+ } = useActions(theDashboardTemplateVariablesLogic)
+ const { allVariablesAreTouched, variables, activeVariableIndex, isCurrentlySelectingElement } = useValues(
+ theDashboardTemplateVariablesLogic
+ )
const [customEventName, setCustomEventName] = useState(null)
const [showCustomEventField, setShowCustomEventField] = useState(false)
+ const { enableElementSelector, disableElementSelector, setNewActionName } = useActions(
+ iframedToolbarBrowserLogic({ iframeRef, clearBrowserUrlOnUnmount: true })
+ )
- const FALLBACK_EVENT = {
- id: '$other_event',
- math: 'dau',
- type: 'events',
+ const variable: DashboardTemplateVariableType | undefined = variables.find((v) => v.name === variableName)
+ if (!variable) {
+ return <>>
}
return (
@@ -37,12 +49,32 @@ function VariableSelector({
{variable.description}
+ {!showCustomEventField && activeVariableIndex == 0 && hasSelectedSite && !variable.touched && (
+
+
+ Tip: Navigate to the page you want before you start selecting.
+
+
+ )}
{variable.touched && !customEventName && (
-
{' '}
-
Selected
-
.md-invite-button
+
+ {' '}
+ Selected
+
+
+
+ CSS selector: {' '}
+ {variable.default.selector || 'not set'}
+
+
+ Element href: {variable.default.href || 'not set'}
+
+
+ Page URL: {variable.default.url || 'any url'}
+
+
{
+ disableElementSelector()
+ setNewActionName(null)
resetVariable(variable.id)
setCustomEventName(null)
setShowCustomEventField(false)
@@ -101,59 +135,100 @@ function VariableSelector({
)}
- {!hasSelectedSite ? (
- Please select a site to continue.
- ) : (
-
- {variable.touched ? (
- <>
- {!allVariablesAreTouched ||
- (allVariablesAreTouched && variables.length !== activeVariableIndex + 1) ? (
-
- !allVariablesAreTouched
- ? goToNextUntouchedActiveVariableIndex()
- : variables.length !== activeVariableIndex + 1
- ? incrementActiveVariableIndex()
- : null
- }
- >
- Continue
-
- ) : null}
- >
- ) : (
-
+
+
+ {variable.touched ? (
+ <>
+ {!allVariablesAreTouched ||
+ (allVariablesAreTouched && variables.length !== activeVariableIndex + 1) ? (
+
+ !allVariablesAreTouched
+ ? goToNextUntouchedActiveVariableIndex()
+ : variables.length !== activeVariableIndex + 1
+ ? incrementActiveVariableIndex()
+ : null
+ }
+ >
+ Continue
+
+ ) : null}
+ >
+ ) : (
+
+ {isCurrentlySelectingElement ? (
+ {
+ disableElementSelector()
+ setNewActionName(null)
+ setIsCurrentlySelectingElement(false)
+ }}
+ icon={ }
+ center
+ className="min-w-44"
+ >
+ Cancel selection
+
+ ) : (
{
setShowCustomEventField(false)
- setVariable(variable.name, { events: [FALLBACK_EVENT] })
+ enableElementSelector()
+ setNewActionName(variable.name)
+ setIsCurrentlySelectingElement(true)
}}
+ icon={ }
+ center
+ className="min-w-44"
+ disabledReason={!hasSelectedSite && 'Please select a site to continue'}
>
Select from site
- setShowCustomEventField(true)}>
- Use custom event
-
-
- )}
-
- )}
+ )}
+
{
+ disableElementSelector()
+ setNewActionName(null)
+ setShowCustomEventField(true)
+ setIsCurrentlySelectingElement(false)
+ }}
+ fullWidth
+ center
+ className="grow max-w-44"
+ >
+ Use custom event
+
+
+ )}
+
)
}
-export function DashboardTemplateVariables({ hasSelectedSite }: { hasSelectedSite: boolean }): JSX.Element {
+export function DashboardTemplateVariables({
+ hasSelectedSite,
+ iframeRef,
+}: {
+ hasSelectedSite: boolean
+ iframeRef: React.RefObject
+}): JSX.Element {
const { activeDashboardTemplate } = useValues(newDashboardLogic)
const theDashboardTemplateVariablesLogic = dashboardTemplateVariablesLogic({
variables: activeDashboardTemplate?.variables || [],
})
const { variables, activeVariableIndex } = useValues(theDashboardTemplateVariablesLogic)
- const { setVariables, setActiveVariableIndex } = useActions(theDashboardTemplateVariablesLogic)
+ const { setVariables, setActiveVariableIndex, setIsCurrentlySelectingElement } = useActions(
+ theDashboardTemplateVariablesLogic
+ )
+ const { setNewActionName, disableElementSelector } = useActions(
+ iframedToolbarBrowserLogic({ iframeRef, clearBrowserUrlOnUnmount: true })
+ )
// TODO: onboarding-dashboard-templates: this is a hack, I'm not sure why it's not set properly initially.
useEffect(() => {
@@ -172,10 +247,20 @@ export function DashboardTemplateVariables({ hasSelectedSite }: { hasSelectedSit
{v.touched && }
),
- content: ,
+ content: (
+
+ ),
className: 'p-4 bg-white',
onHeaderClick: () => {
setActiveVariableIndex(i)
+ disableElementSelector()
+ setNewActionName(null)
+ setIsCurrentlySelectingElement(false)
},
}))}
embedded
diff --git a/frontend/src/scenes/onboarding/sdks/sdksLogic.tsx b/frontend/src/scenes/onboarding/sdks/sdksLogic.tsx
index 0ab48594d2825..6d4a0d4641eb2 100644
--- a/frontend/src/scenes/onboarding/sdks/sdksLogic.tsx
+++ b/frontend/src/scenes/onboarding/sdks/sdksLogic.tsx
@@ -126,26 +126,25 @@ export const sdksLogic = kea([
loadSnippetEvents: async () => {
const query: HogQLQuery = {
kind: NodeKind.HogQLQuery,
- query: hogql`SELECT properties.$lib_version AS lib_version,
+ query: hogql`SELECT
max(timestamp) AS latest_timestamp,
- count(lib_version) AS count,
concat(
if(startsWith(properties.current_url, 'https://'), 'https://', 'http://'),
properties.$host
) AS full_host
FROM events
- WHERE timestamp >= now() - INTERVAL 3 DAY
+ WHERE timestamp >= now() - INTERVAL 5 DAY
AND timestamp <= now()
AND properties.$lib = 'web'
AND properties.$host is not null
- GROUP BY lib_version, full_host
+ GROUP BY full_host
ORDER BY latest_timestamp DESC
LIMIT 10`,
}
const res = await api.query(query)
const hasEvents = !!(res.results?.length ?? 0 > 0)
- const snippetHosts = res.results?.map((result) => result[3]).filter((val) => !!val) ?? []
+ const snippetHosts = res.results?.map((result) => result[1]).filter((val) => !!val) ?? []
if (hasEvents) {
actions.setSnippetHosts(snippetHosts)
}
diff --git a/frontend/src/toolbar/actions/ActionAttribute.tsx b/frontend/src/toolbar/actions/ActionAttribute.tsx
index c8cd2c68898b0..ca74a79363c07 100644
--- a/frontend/src/toolbar/actions/ActionAttribute.tsx
+++ b/frontend/src/toolbar/actions/ActionAttribute.tsx
@@ -1,6 +1,10 @@
-import { Link } from '@posthog/lemon-ui'
+import { LemonSwitch, Link } from '@posthog/lemon-ui'
+import { useActions, useValues } from 'kea'
import { IconBranch, IconClipboardEdit, IconLink, IconTextSize } from 'lib/lemon-ui/icons'
+import { actionsTabLogic } from './actionsTabLogic'
+import { ActionStepPropertyKey } from './ActionStep'
+
function SelectorString({ value }: { value: string }): JSX.Element {
const [last, ...rest] = value.split(' ').reverse()
return (
@@ -10,7 +14,16 @@ function SelectorString({ value }: { value: string }): JSX.Element {
)
}
-export function ActionAttribute({ attribute, value }: { attribute: string; value?: string }): JSX.Element {
+export function ActionAttribute({
+ attribute,
+ value,
+}: {
+ attribute: ActionStepPropertyKey
+ value?: string
+}): JSX.Element {
+ const { automaticCreationIncludedPropertyKeys, automaticActionCreationEnabled } = useValues(actionsTabLogic)
+ const { removeAutomaticCreationIncludedPropertyKey, addAutomaticCreationIncludedPropertyKey } =
+ useActions(actionsTabLogic)
const icon =
attribute === 'text' ? (
@@ -44,6 +57,17 @@ export function ActionAttribute({ attribute, value }: { attribute: string; value
return (
+ {automaticActionCreationEnabled && (
+
+ checked
+ ? addAutomaticCreationIncludedPropertyKey(attribute)
+ : removeAutomaticCreationIncludedPropertyKey(attribute)
+ }
+ />
+ )}
{icon}
{text}
diff --git a/frontend/src/toolbar/actions/ActionStep.tsx b/frontend/src/toolbar/actions/ActionStep.tsx
index f0d6272d01b8b..a2a34a43412a7 100644
--- a/frontend/src/toolbar/actions/ActionStep.tsx
+++ b/frontend/src/toolbar/actions/ActionStep.tsx
@@ -1,16 +1,26 @@
+import { useValues } from 'kea'
+
import { ActionAttribute } from '~/toolbar/actions/ActionAttribute'
import { ActionStepType } from '~/types'
+import { actionsTabLogic } from './actionsTabLogic'
+
interface ActionStepProps {
actionStep: ActionStepType
}
-type ActionStepPropertyKey = 'text' | 'name' | 'href' | 'selector'
+export type ActionStepPropertyKey = 'text' | 'name' | 'href' | 'selector' | 'url'
export function ActionStep({ actionStep }: ActionStepProps): JSX.Element {
+ const { automaticActionCreationEnabled } = useValues(actionsTabLogic)
+
+ const stepTypes = ['text', 'name', 'href', 'selector', automaticActionCreationEnabled ? 'url' : null].filter(
+ (key) => key
+ ) as ActionStepPropertyKey[]
+
return (
- {(['text', 'name', 'href', 'selector'] as ActionStepPropertyKey[]).map((attr) =>
+ {stepTypes.map((attr) =>
actionStep[attr] || attr === 'selector' ? (
) : null
diff --git a/frontend/src/toolbar/actions/actionsTabLogic.tsx b/frontend/src/toolbar/actions/actionsTabLogic.tsx
index 759d410d00f97..be2b0fa13f160 100644
--- a/frontend/src/toolbar/actions/actionsTabLogic.tsx
+++ b/frontend/src/toolbar/actions/actionsTabLogic.tsx
@@ -14,11 +14,25 @@ import { actionStepToActionStepFormItem, elementToActionStep, stepToDatabaseForm
import { ActionType, ElementType } from '~/types'
import type { actionsTabLogicType } from './actionsTabLogicType'
+import { ActionStepPropertyKey } from './ActionStep'
-function newAction(element: HTMLElement | null, dataAttributes: string[] = []): ActionDraftType {
+function newAction(
+ element: HTMLElement | null,
+ dataAttributes: string[] = [],
+ name: string | null,
+ includedPropertyKeys?: ActionStepPropertyKey[]
+): ActionDraftType {
return {
- name: '',
- steps: [element ? actionStepToActionStepFormItem(elementToActionStep(element, dataAttributes), true) : {}],
+ name: name || '',
+ steps: [
+ element
+ ? actionStepToActionStepFormItem(
+ elementToActionStep(element, dataAttributes),
+ true,
+ includedPropertyKeys
+ )
+ : {},
+ ],
pinned_at: null,
}
}
@@ -67,6 +81,11 @@ export const actionsTabLogic = kea
([
hideButtonActions: true,
setShowActionsTooltip: (showActionsTooltip: boolean) => ({ showActionsTooltip }),
setElementSelector: (selector: string, index: number) => ({ selector, index }),
+ setAutomaticActionCreationEnabled: (enabled: boolean, name?: string) => ({ enabled, name }),
+ actionCreatedSuccess: (action: ActionType) => ({ action }),
+ setautomaticCreationIncludedPropertyKeys: (keys: ActionStepPropertyKey[]) => ({ keys }),
+ removeAutomaticCreationIncludedPropertyKey: (key: ActionStepPropertyKey) => ({ key }),
+ addAutomaticCreationIncludedPropertyKey: (key: ActionStepPropertyKey) => ({ key }),
}),
connect(() => ({
@@ -142,6 +161,30 @@ export const actionsTabLogic = kea([
setShowActionsTooltip: (_, { showActionsTooltip }) => showActionsTooltip,
},
],
+ // we automatically create actions for people from analytics onboarding. This flag controls that experience.
+ automaticActionCreationEnabled: [
+ false as boolean,
+ {
+ setAutomaticActionCreationEnabled: (_, { enabled, name }) => (enabled && !!name) || false,
+ },
+ ],
+ newActionName: [
+ null as string | null,
+ {
+ setAutomaticActionCreationEnabled: (_, { enabled, name }) => (enabled && name ? name : null),
+ },
+ ],
+ automaticCreationIncludedPropertyKeys: [
+ [] as ActionStepPropertyKey[],
+ {
+ setAutomaticActionCreationEnabled: (_, { enabled }) =>
+ enabled ? ['text', 'href', 'name', 'selector', 'url'] : [],
+ setautomaticCreationIncludedPropertyKeys: (_, { keys }) => keys || [],
+ removeAutomaticCreationIncludedPropertyKey: (state, { key }) => state.filter((k) => k !== key),
+ addAutomaticCreationIncludedPropertyKey: (state, { key }) =>
+ !state.includes(key) ? [...state, key] : state,
+ },
+ ],
}),
forms(({ values, actions }) => ({
@@ -158,6 +201,19 @@ export const actionsTabLogic = kea([
const { apiURL, temporaryToken } = values
const { selectedActionId } = values
+ const findUniqueActionName = (baseName: string, index = 0): string => {
+ const proposedName = index === 0 ? baseName : `${baseName} - ${index}`
+ if (!values.allActions.find((action) => action.name === proposedName)) {
+ return proposedName
+ }
+ return findUniqueActionName(baseName, index + 1)
+ }
+
+ if (values.newActionName) {
+ // newActionName is programmatically set, but they may already have an existing action with that name. Append an index.
+ actionToSave.name = findUniqueActionName(values.newActionName)
+ }
+
let response: ActionType
if (selectedActionId && selectedActionId !== 'new') {
response = await api.update(
@@ -172,15 +228,19 @@ export const actionsTabLogic = kea([
}
breakpoint()
- actionsLogic.actions.updateAction({ action: response })
actions.selectAction(null)
+ actionsLogic.actions.updateAction({ action: response })
- lemonToast.success('Action saved', {
- button: {
- label: 'Open in PostHog',
- action: () => window.open(`${apiURL}${urls.action(response.id)}`, '_blank'),
- },
- })
+ if (!values.automaticActionCreationEnabled) {
+ lemonToast.success('Action saved', {
+ button: {
+ label: 'Open in PostHog',
+ action: () => window.open(`${apiURL}${urls.action(response.id)}`, '_blank'),
+ },
+ })
+ }
+
+ actions.actionCreatedSuccess(response)
},
// whether we show errors after touch (true) or submit (false)
@@ -211,22 +271,47 @@ export const actionsTabLogic = kea([
},
],
selectedAction: [
- (s) => [s.selectedActionId, s.newActionForElement, s.allActions, s.dataAttributes],
+ (s) => [
+ s.selectedActionId,
+ s.newActionForElement,
+ s.allActions,
+ s.dataAttributes,
+ s.newActionName,
+ s.automaticCreationIncludedPropertyKeys,
+ ],
(
selectedActionId,
newActionForElement,
allActions,
- dataAttributes
+ dataAttributes,
+ newActionName,
+ automaticCreationIncludedPropertyKeys
): ActionType | ActionDraftType | null => {
if (selectedActionId === 'new') {
- return newAction(newActionForElement, dataAttributes)
+ return newAction(
+ newActionForElement,
+ dataAttributes,
+ newActionName,
+ automaticCreationIncludedPropertyKeys
+ )
}
return allActions.find((a) => a.id === selectedActionId) || null
},
],
+ isReadyForAutomaticSubmit: [
+ (s) => [s.automaticActionCreationEnabled, s.selectedAction, s.actionForm],
+ (automaticActionCreationEnabled, selectedAction, actionForm): boolean => {
+ return (
+ (automaticActionCreationEnabled &&
+ selectedAction?.name &&
+ actionForm.steps?.[0]?.selector_selected) ||
+ false
+ )
+ },
+ ],
}),
- subscriptions(({ actions }) => ({
+ subscriptions(({ actions, values }) => ({
selectedAction: (selectedAction: ActionType | ActionDraftType | null) => {
if (!selectedAction) {
actions.setActionFormValues({ name: null, steps: [{}] })
@@ -237,6 +322,9 @@ export const actionsTabLogic = kea([
? selectedAction.steps.map((step) => actionStepToActionStepFormItem(step, false))
: [{}],
})
+ if (values.isReadyForAutomaticSubmit) {
+ actions.submitActionForm()
+ }
}
},
})),
diff --git a/frontend/src/toolbar/bar/toolbarLogic.ts b/frontend/src/toolbar/bar/toolbarLogic.ts
index 2176e5650cfee..dd5a8dd6a2f2b 100644
--- a/frontend/src/toolbar/bar/toolbarLogic.ts
+++ b/frontend/src/toolbar/bar/toolbarLogic.ts
@@ -31,7 +31,13 @@ export const toolbarLogic = kea([
connect(() => ({
actions: [
actionsTabLogic,
- ['showButtonActions', 'hideButtonActions', 'selectAction'],
+ [
+ 'showButtonActions',
+ 'hideButtonActions',
+ 'selectAction',
+ 'setAutomaticActionCreationEnabled',
+ 'actionCreatedSuccess',
+ ],
elementsLogic,
['enableInspect', 'disableInspect', 'createAction'],
heatmapLogic,
@@ -393,6 +399,10 @@ export const toolbarLogic = kea([
// if embedded we need to signal start and finish of heatmap loading to the parent
window.parent.postMessage({ type: PostHogAppToolbarEvent.PH_TOOLBAR_HEATMAP_FAILED }, '*')
},
+ actionCreatedSuccess: (action) => {
+ // if embedded, we need to tell the parent window that a new action was created
+ window.parent.postMessage({ type: PostHogAppToolbarEvent.PH_NEW_ACTION_CREATED, payload: action }, '*')
+ },
})),
afterMount(({ actions, values, cache }) => {
cache.clickListener = (e: MouseEvent): void => {
@@ -440,6 +450,17 @@ export const toolbarLogic = kea([
case PostHogAppToolbarEvent.PH_HEATMAPS_COMMON_FILTERS:
actions.setCommonFilters(e.data.payload.commonFilters)
return
+ case PostHogAppToolbarEvent.PH_ELEMENT_SELECTOR:
+ if (e.data.payload.enabled) {
+ actions.enableInspect()
+ } else {
+ actions.disableInspect()
+ actions.hideButtonActions()
+ }
+ return
+ case PostHogAppToolbarEvent.PH_NEW_ACTION_NAME:
+ actions.setAutomaticActionCreationEnabled(true, e.data.payload.name)
+ return
default:
console.warn(`[PostHog Toolbar] Received unknown parent window message: ${type}`)
}
diff --git a/frontend/src/toolbar/elements/ElementInfo.tsx b/frontend/src/toolbar/elements/ElementInfo.tsx
index 151367d229ad3..c4459838ed561 100644
--- a/frontend/src/toolbar/elements/ElementInfo.tsx
+++ b/frontend/src/toolbar/elements/ElementInfo.tsx
@@ -7,6 +7,8 @@ import { ActionStep } from '~/toolbar/actions/ActionStep'
import { elementsLogic } from '~/toolbar/elements/elementsLogic'
import { heatmapLogic } from '~/toolbar/elements/heatmapLogic'
+import { actionsTabLogic } from '../actions/actionsTabLogic'
+
function ElementStatistic({
prefix,
suffix,
@@ -34,6 +36,7 @@ export function ElementInfo(): JSX.Element | null {
const { activeMeta } = useValues(elementsLogic)
const { createAction } = useActions(elementsLogic)
+ const { automaticActionCreationEnabled } = useValues(actionsTabLogic)
if (!activeMeta) {
return null
@@ -79,16 +82,20 @@ export function ElementInfo(): JSX.Element | null {
) : null}
{/* eslint-disable-next-line react/forbid-dom-props */}
-
Actions ({activeMeta.actions.length})
+ {!automaticActionCreationEnabled && (
+ <>
+
Actions ({activeMeta.actions.length})
- {activeMeta.actions.length === 0 ? (
-
No actions include this element
- ) : (
-
a.action)} />
+ {activeMeta.actions.length === 0 ? (
+ No actions include this element
+ ) : (
+ a.action)} />
+ )}
+ >
)}
createAction(element)} icon={ }>
- Create a new action
+ {automaticActionCreationEnabled ? 'Select element' : 'Create a new action'}
>
diff --git a/frontend/src/toolbar/elements/elementsLogic.ts b/frontend/src/toolbar/elements/elementsLogic.ts
index eb019fdfe3edb..c7060ea5ba547 100644
--- a/frontend/src/toolbar/elements/elementsLogic.ts
+++ b/frontend/src/toolbar/elements/elementsLogic.ts
@@ -405,6 +405,7 @@ export const elementsLogic = kea([
},
createAction: ({ element }) => {
actions.selectElement(null)
+ // this just sets the action form
actions.newAction(element)
},
})),
diff --git a/frontend/src/toolbar/utils.ts b/frontend/src/toolbar/utils.ts
index ade5110122b76..65f8285cbc189 100644
--- a/frontend/src/toolbar/utils.ts
+++ b/frontend/src/toolbar/utils.ts
@@ -7,6 +7,8 @@ import { CSSProperties } from 'react'
import { ActionStepForm, ElementRect } from '~/toolbar/types'
import { ActionStepType } from '~/types'
+import { ActionStepPropertyKey } from './actions/ActionStep'
+
export const TOOLBAR_ID = '__POSTHOG_TOOLBAR__'
export const LOCALSTORAGE_KEY = '_postHogToolbarParams'
@@ -265,7 +267,11 @@ export function getBoxColors(color: 'blue' | 'red' | 'green', hover = false, opa
}
}
-export function actionStepToActionStepFormItem(step: ActionStepType, isNew = false): ActionStepForm {
+export function actionStepToActionStepFormItem(
+ step: ActionStepType,
+ isNew = false,
+ includedPropertyKeys?: ActionStepPropertyKey[]
+): ActionStepForm {
if (!step) {
return {}
}
@@ -281,24 +287,24 @@ export function actionStepToActionStepFormItem(step: ActionStepType, isNew = fal
...step,
href_selected: true,
selector_selected: hasSelector,
- text_selected: false,
- url_selected: false,
+ text_selected: includedPropertyKeys?.includes('text') || false,
+ url_selected: includedPropertyKeys?.includes('url') || false,
}
} else if (step.tag_name === 'button') {
return {
...step,
text_selected: true,
selector_selected: hasSelector,
- href_selected: false,
- url_selected: false,
+ href_selected: includedPropertyKeys?.includes('href') || false,
+ url_selected: includedPropertyKeys?.includes('url') || false,
}
}
return {
...step,
selector_selected: hasSelector,
- text_selected: false,
- url_selected: false,
- href_selected: false,
+ text_selected: includedPropertyKeys?.includes('text') || false,
+ url_selected: includedPropertyKeys?.includes('url') || false,
+ href_selected: includedPropertyKeys?.includes('href') || false,
}
}
diff --git a/frontend/src/types.ts b/frontend/src/types.ts
index 94cedbd1c1566..a6db8171f301e 100644
--- a/frontend/src/types.ts
+++ b/frontend/src/types.ts
@@ -568,6 +568,7 @@ export interface ActionStepType {
url?: string | null
/** @default StringMatching.Contains */
url_matching?: ActionStepStringMatching | null
+ name?: string | null
}
export interface ElementType {
@@ -1805,6 +1806,9 @@ export interface DashboardTemplateVariableType {
default: Record
required: boolean
touched?: boolean
+ selector?: string
+ href?: string
+ url?: string
}
export type DashboardLayoutSize = 'sm' | 'xs'
@@ -2136,6 +2140,17 @@ export interface FilterType {
aggregation_group_type_index?: integer // Groups aggregation
}
+export interface TemplateVariableStep {
+ id: string
+ math: BaseMathType
+ name: string | null
+ order: number
+ type: EntityTypes
+ selector?: string | null
+ href?: string | null
+ url?: string | null
+}
+
export interface PropertiesTimelineFilterType {
date_from?: string | null // DateMixin
date_to?: string | null // DateMixin