Skip to content

Commit

Permalink
Merge pull request umami-software#162 from freeyourmusic/bh/unregister
Browse files Browse the repository at this point in the history
Ability to unregister umami
  • Loading branch information
mikecao authored Sep 18, 2020
2 parents c932dc2 + 17b4f51 commit bddda92
Show file tree
Hide file tree
Showing 6 changed files with 121 additions and 38 deletions.
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@
# production
/build
/public/umami.js
/public/snippet.js
/lang-compiled
/lang-formatted

Expand Down
4 changes: 2 additions & 2 deletions components/forms/TrackingCodeForm.js
Original file line number Diff line number Diff line change
Expand Up @@ -27,9 +27,9 @@ export default function TrackingCodeForm({ values, onClose }) {
/>
</FormRow>
<FormButtons>
<CopyButton type="submit" variant="action" element={ref} />
<CopyButton type="submit" variant="action" element={ref}/>
<Button onClick={onClose}>
<FormattedMessage id="button.cancel" defaultMessage="Cancel" />
<FormattedMessage id="button.cancel" defaultMessage="Cancel"/>
</Button>
</FormButtons>
</FormLayout>
Expand Down
1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
"start": "next start",
"build-app": "next build",
"build-tracker": "rollup -c rollup.tracker.config.js",
"build-snippet": "rollup -c rollup.snippet.config.js",
"copy-db-schema": "node scripts/copy-db-schema.js",
"build-db-schema": "dotenv prisma introspect",
"build-db-client": "dotenv prisma generate",
Expand Down
19 changes: 19 additions & 0 deletions rollup.snippet.config.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
import 'dotenv/config';
import buble from '@rollup/plugin-buble';
import replace from '@rollup/plugin-replace';
import resolve from '@rollup/plugin-node-resolve';
import { terser } from 'rollup-plugin-terser';

export default {
input: 'tracker/snippet.js',
output: {
file: 'public/snippet.js',
format: 'iife',
},
plugins: [
replace({ __DNT__: !!process.env.ENABLE_DNT }),
resolve(),
buble(),
terser({ compress: { evaluate: false } }),
],
};
91 changes: 55 additions & 36 deletions tracker/index.js
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import 'promise-polyfill/src/polyfill';
import 'unfetch/polyfill';
import { post, hook, doNotTrack } from '../lib/web';
import { doNotTrack, hook, post } from '../lib/web';
import { removeTrailingSlash } from '../lib/url';

(window => {
Expand All @@ -19,6 +19,7 @@ import { removeTrailingSlash } from '../lib/url';

const website = script.getAttribute('data-website-id');
const hostUrl = script.getAttribute('data-host-url');
const skipAuto = script.getAttribute('data-skip-auto');
const root = hostUrl
? removeTrailingSlash(hostUrl)
: new URL(script.src).href.split('/').slice(0, -1).join('/');
Expand All @@ -28,13 +29,31 @@ import { removeTrailingSlash } from '../lib/url';
let currentUrl = `${pathname}${search}`;
let currentRef = document.referrer;

/* Collect metrics */
/* Handle events */

const removeEvents = () => {
listeners.forEach(([element, type, listener]) => {
element && element.removeEventListener(type, listener, true);
});
listeners.length = 0;
};

const loadEvents = () => {
document.querySelectorAll('[class*=\'umami--\']').forEach(element => {
element.className.split(' ').forEach(className => {
if (/^umami--([a-z]+)--([a-z0-9_]+[a-z0-9-_]+)$/.test(className)) {
const [, type, value] = className.split('--');
const listener = () => collectEvent(type, value);

const collect = (type, params) => {
listeners.push([element, type, listener]);
element.addEventListener(type, listener, true);
}
});
});
};
const collect = (type, params, uuid) => {
const payload = {
url: currentUrl,
referrer: currentRef,
website,
website: uuid,
hostname,
screen,
language,
Expand All @@ -51,13 +70,15 @@ import { removeTrailingSlash } from '../lib/url';
payload,
});
};
const pageView = (url = currentUrl, referrer = currentRef, uuid = website) => collect('pageview', {
url,
referrer,
}, uuid);

const pageView = () => collect('pageview').then(() => setTimeout(loadEvents, 300));

const pageEvent = (event_type, event_value) => collect('event', { event_type, event_value });
/* Collect metrics */
const pageViewWithAutoEvents = (url, referrer) => pageView(url, referrer).then(() => setTimeout(loadEvents, 300));

/* Handle history */

const handlePush = (state, title, url) => {
removeEvents();
currentRef = currentUrl;
Expand All @@ -70,40 +91,38 @@ import { removeTrailingSlash } from '../lib/url';
currentUrl = newUrl;
}

pageView();
pageViewWithAutoEvents(currentUrl, currentRef);
};

history.pushState = hook(history, 'pushState', handlePush);
history.replaceState = hook(history, 'replaceState', handlePush);

/* Handle events */
const collectEvent = (event_type, event_value, url = currentUrl, uuid = website) => collect('event', {
url,
event_type,
event_value,
}, uuid);

const removeEvents = () => {
listeners.forEach(([element, type, listener]) => {
element && element.removeEventListener(type, listener, true);
});
listeners.length = 0;
const registerAutoEvents = () => {
history.pushState = hook(history, 'pushState', handlePush);
history.replaceState = hook(history, 'replaceState', handlePush);
return pageViewWithAutoEvents(currentUrl, currentRef);
};

const loadEvents = () => {
document.querySelectorAll("[class*='umami--']").forEach(element => {
element.className.split(' ').forEach(className => {
if (/^umami--([a-z]+)--([a-z0-9_]+[a-z0-9-_]+)$/.test(className)) {
const [, type, value] = className.split('--');
const listener = () => pageEvent(type, value);

listeners.push([element, type, listener]);
element.addEventListener(type, listener, true);
}
});
});
};
const umamiFunctions = { collect, pageView, collectEvent, registerAutoEvents };
const scheduledCalls = window.umami.calls;

/* Start */
window.umami = event_value => collect('event', { event_type: 'custom', event_value });
Object.keys(umamiFunctions).forEach((key) => {
window.umami[key] = umamiFunctions[key];
});

pageView();
if (scheduledCalls) {
scheduledCalls.forEach(([fnName, ...params]) => {
window.umami[fnName].apply(window.umami, params);
});
}

if (!window.umami) {
window.umami = event_value => collect('event', { event_type: 'custom', event_value });
/* Start */
if (!skipAuto) {
registerAutoEvents().catch(e => console.error(e));
}
})(window);
43 changes: 43 additions & 0 deletions tracker/snippet.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
(window => {
const umami = window.umami = window.umami || [];
if (!umami.registerAutoEvents) {
if (umami.invoked) {
window.console && console.error && console.error('Umami snippet included twice.');
} else {
umami.invoked = true;
umami.calls = [];
umami.methods = ['registerAutoEvents', 'event', 'pageView'];
umami.factory = t => {
return function() {
const e = Array.prototype.slice.call(arguments);
e.unshift(t);
umami.calls.push(e);
return umami;
};
};
for (let t = 0; t < umami.methods.length; t++) {
let e = umami.methods[t];
umami[e] = umami.factory(e);
}
umami.load = function(umamiScript, umamiUUID, skipAuto) {
const scriptElement = document.createElement('script');
scriptElement.type = 'text/javascript';
scriptElement.defer = true;
scriptElement.async = true;
scriptElement.setAttribute('data-website-id', umamiUUID);
if (skipAuto) {
scriptElement.setAttribute('data-skip-auto', 'true');
}
scriptElement.src = umamiScript;
const otherScript = document.getElementsByTagName('script')[0];
otherScript.parentNode.insertBefore(scriptElement, otherScript);
};

umami.load('[HOST]/umami.js', '[UMAMI_UUID]', false);
}
}
})(window);
// This snippet is for more advanced use case of Umami. If you want to track custom events,
// and not worry about having blocking script in the header,
// use this snippet (compiled version available in /public/snippet.js).
// Just remember to replace [HOST] and [UMAMI_UUID] when pasting it.

0 comments on commit bddda92

Please sign in to comment.