-
Notifications
You must be signed in to change notification settings - Fork 329
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #1068 from datopian/analytics
[core][m]: Add analytics component to the core packages
- Loading branch information
Showing
10 changed files
with
240 additions
and
2 deletions.
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,36 @@ | ||
import Script from 'next/script.js' | ||
|
||
export interface GoogleAnalyticsProps { | ||
googleAnalyticsId: string | ||
} | ||
|
||
export const GA = ({ googleAnalyticsId }: GoogleAnalyticsProps) => { | ||
return ( | ||
<> | ||
<Script | ||
strategy="afterInteractive" | ||
src={`https://www.googletagmanager.com/gtag/js?id=${googleAnalyticsId}`} | ||
/> | ||
|
||
<Script strategy="afterInteractive" id="ga-script"> | ||
{` | ||
window.dataLayer = window.dataLayer || []; | ||
function gtag(){dataLayer.push(arguments);} | ||
gtag('js', new Date()); | ||
gtag('config', '${googleAnalyticsId}'); | ||
`} | ||
</Script> | ||
</> | ||
) | ||
} | ||
|
||
// https://developers.google.com/analytics/devguides/collection/gtagjs/events | ||
export const logEvent = (action, category, label, value) => { | ||
// eslint-disable-next-line @typescript-eslint/ban-ts-comment | ||
// @ts-ignore | ||
window.gtag?.('event', action, { | ||
event_category: category, | ||
event_label: label, | ||
value: value, | ||
}) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,41 @@ | ||
import Script from 'next/script.js' | ||
|
||
export interface PlausibleProps { | ||
plausibleDataDomain: string | ||
dataApi?: string | ||
src?: string | ||
} | ||
|
||
/** | ||
* Plausible analytics component. | ||
* To proxy the requests through your own domain, you can use the dataApi and src attribute. | ||
* See [Plausible docs](https://plausible.io/docs/proxy/guides/nextjs#step-2-adjust-your-deployed-script) | ||
* for more information. | ||
* | ||
*/ | ||
export const Plausible = ({ | ||
plausibleDataDomain, | ||
dataApi = undefined, | ||
src = 'https://plausible.io/js/plausible.js', | ||
}: PlausibleProps) => { | ||
return ( | ||
<> | ||
<Script | ||
strategy="lazyOnload" | ||
data-domain={plausibleDataDomain} | ||
data-api={dataApi} | ||
src={src} | ||
/> | ||
<Script strategy="lazyOnload" id="plausible-script"> | ||
{` | ||
window.plausible = window.plausible || function() { (window.plausible.q = window.plausible.q || []).push(arguments) } | ||
`} | ||
</Script> | ||
</> | ||
) | ||
} | ||
|
||
// https://plausible.io/docs/custom-event-goals | ||
export const logEvent = (eventName, ...rest) => { | ||
return window.plausible?.(eventName, ...rest) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,25 @@ | ||
import Script from 'next/script.js' | ||
|
||
export interface PosthogProps { | ||
posthogProjectApiKey: string | ||
apiHost?: string | ||
} | ||
|
||
/** | ||
* Posthog analytics component. | ||
* See [Posthog docs](https://posthog.com/docs/libraries/js#option-1-add-javascript-snippet-to-your-html-badgerecommendedbadge) for more information. | ||
* | ||
*/ | ||
export const Posthog = ({ | ||
posthogProjectApiKey, | ||
apiHost = 'https://app.posthog.com', | ||
}: PosthogProps) => { | ||
return ( | ||
<Script strategy="lazyOnload" id="posthog-script"> | ||
{` | ||
!function(t,e){var o,n,p,r;e.__SV||(window.posthog=e,e._i=[],e.init=function(i,s,a){function g(t,e){var o=e.split(".");2==o.length&&(t=t[o[0]],e=o[1]),t[e]=function(){t.push([e].concat(Array.prototype.slice.call(arguments,0)))}}(p=t.createElement("script")).type="text/javascript",p.async=!0,p.src=s.api_host+"/static/array.js",(r=t.getElementsByTagName("script")[0]).parentNode.insertBefore(p,r);var u=e;for(void 0!==a?u=e[a]=[]:a="posthog",u.people=u.people||[],u.toString=function(t){var e="posthog";return"posthog"!==a&&(e+="."+a),t||(e+=" (stub)"),e},u.people.toString=function(){return u.toString(1)+".people (stub)"},o="capture identify alias people.set people.set_once set_config register register_once unregister opt_out_capturing has_opted_out_capturing opt_in_capturing reset isFeatureEnabled onFeatureFlags".split(" "),n=0;n<o.length;n++)g(u,o[n]);e._i.push([i,s,a])},e.__SV=1)}(document,window.posthog||[]); | ||
posthog.init('${posthogProjectApiKey}',{api_host:'${apiHost}'}) | ||
`} | ||
</Script> | ||
) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
import Script from 'next/script.js' | ||
|
||
export interface SimpleAnalyticsProps { | ||
src?: string | ||
} | ||
|
||
export const SimpleAnalytics = ({ | ||
src = 'https://scripts.simpleanalyticscdn.com/latest.js', | ||
}: SimpleAnalyticsProps) => { | ||
return ( | ||
<> | ||
<Script strategy="lazyOnload" id="sa-script"> | ||
{` | ||
window.sa_event=window.sa_event||function(){var a=[].slice.call(arguments);window.sa_event.q?window.sa_event.q.push(a):window.sa_event.q=[a]}; | ||
`} | ||
</Script> | ||
<Script strategy="lazyOnload" src={src} /> | ||
</> | ||
) | ||
} | ||
|
||
// https://docs.simpleanalytics.com/events | ||
export const logEvent = (eventName, callback) => { | ||
if (callback) { | ||
return window.sa_event?.(eventName, callback) | ||
} else { | ||
return window.sa_event?.(eventName) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,20 @@ | ||
import Script from 'next/script.js' | ||
|
||
export interface UmamiProps { | ||
umamiWebsiteId: string | ||
src?: string | ||
} | ||
|
||
export const Umami = ({ | ||
umamiWebsiteId, | ||
src = 'https://analytics.umami.is/script.js', | ||
}: UmamiProps) => { | ||
return ( | ||
<Script | ||
async | ||
defer | ||
data-website-id={umamiWebsiteId} | ||
src={src} // Replace with your umami instance | ||
/> | ||
) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,82 @@ | ||
/* eslint-disable @typescript-eslint/no-explicit-any */ | ||
import { GA, GoogleAnalyticsProps } from "./GoogleAnalytics"; | ||
import { Plausible, PlausibleProps } from "./Plausible"; | ||
import { SimpleAnalytics, SimpleAnalyticsProps } from "./SimpleAnalytics"; | ||
import { Umami, UmamiProps } from "./Umami"; | ||
import { Posthog, PosthogProps } from "./Posthog"; | ||
|
||
declare global { | ||
interface Window { | ||
// eslint-disable-next-line @typescript-eslint/ban-ts-comment | ||
// @ts-ignore | ||
gtag?: (...args: any[]) => void; | ||
plausible?: (...args: any[]) => void; | ||
sa_event?: (...args: any[]) => void; | ||
} | ||
} | ||
|
||
export interface AnalyticsConfig { | ||
googleAnalytics?: GoogleAnalyticsProps; | ||
plausibleAnalytics?: PlausibleProps; | ||
umamiAnalytics?: UmamiProps; | ||
posthogAnalytics?: PosthogProps; | ||
simpleAnalytics?: SimpleAnalyticsProps; | ||
} | ||
|
||
/** | ||
* @example | ||
* const analytics: AnalyticsConfig = { | ||
* plausibleDataDomain: '', // e.g. tailwind-nextjs-starter-blog.vercel.app | ||
* simpleAnalytics: false, // true or false | ||
* umamiWebsiteId: '', // e.g. 123e4567-e89b-12d3-a456-426614174000 | ||
* posthogProjectApiKey: '', // e.g. AhnJK8392ndPOav87as450xd | ||
* googleAnalyticsId: '', // e.g. UA-000000-2 or G-XXXXXXX | ||
* } | ||
*/ | ||
export interface AnalyticsProps { | ||
analyticsConfig: AnalyticsConfig; | ||
} | ||
|
||
const isProduction = true || process.env["NODE_ENV"] === "production"; | ||
|
||
/** | ||
* Supports Plausible, Simple Analytics, Umami, Posthog or Google Analytics. | ||
* All components default to the hosted service, but can be configured to use a self-hosted | ||
* or proxied version of the script by providing the `src` / `apiHost` props. | ||
* | ||
* Note: If you want to use an analytics provider you have to add it to the | ||
* content security policy in the `next.config.js` file. | ||
* @param {AnalyticsProps} { analytics } | ||
* @return {*} | ||
*/ | ||
export const Analytics = ({ analyticsConfig }: AnalyticsProps) => { | ||
return ( | ||
<> | ||
{isProduction && analyticsConfig.plausibleAnalytics && ( | ||
<Plausible {...analyticsConfig.plausibleAnalytics} /> | ||
)} | ||
{isProduction && analyticsConfig.simpleAnalytics && ( | ||
<SimpleAnalytics {...analyticsConfig.simpleAnalytics} /> | ||
)} | ||
{isProduction && analyticsConfig.posthogAnalytics && ( | ||
<Posthog {...analyticsConfig.posthogAnalytics} /> | ||
)} | ||
{isProduction && analyticsConfig.umamiAnalytics && ( | ||
<Umami {...analyticsConfig.umamiAnalytics} /> | ||
)} | ||
{isProduction && analyticsConfig.googleAnalytics && ( | ||
<GA {...analyticsConfig.googleAnalytics} /> | ||
)} | ||
</> | ||
); | ||
}; | ||
|
||
export { GA, Plausible, SimpleAnalytics, Umami, Posthog }; | ||
|
||
export type { | ||
GoogleAnalyticsProps, | ||
PlausibleProps, | ||
UmamiProps, | ||
PosthogProps, | ||
SimpleAnalyticsProps, | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters