fix: invert macOS and PlayStation icons in dark mode #64

Merged
uz1mani merged 2 commits from staging into main 2026-03-19 12:46:02 +00:00
3 changed files with 52 additions and 13 deletions

View File

@@ -2298,15 +2298,32 @@ export default defineConfig({
<p>Follow these steps to add Pulse through GTM:</p>
<ol>
<li>Go to <strong>Tags &rarr; New &rarr; Custom HTML</strong></li>
<li>Paste the Pulse script</li>
<li>Paste the snippet below (replace <code>your-site.com</code> with your domain)</li>
<li>Set the trigger to <strong>All Pages</strong></li>
<li>Publish your container</li>
</ol>
<CodeBlock filename="GTM → Custom HTML Tag">{`<script
<CodeBlock filename="GTM → Custom HTML Tag (recommended)">{`<script>
window.pulseConfig = { domain: "your-site.com" };
</script>
<script defer src="https://pulse.ciphera.net/script.js"></script>`}</CodeBlock>
<p>
This uses a global config object so that GTM does not need to preserve{' '}
<code>data-*</code> attributes on the injected script element. You can
also pass <code>api</code>, <code>storage</code>, and other options via{' '}
<code>pulseConfig</code>.
</p>
<details>
<summary className="cursor-pointer text-sm text-neutral-500 dark:text-neutral-400 hover:text-neutral-700 dark:hover:text-neutral-300">
Alternative: inline data attributes (may not work in all GTM setups)
</summary>
<CodeBlock filename="GTM → Custom HTML Tag (alternative)">{`<script
defer
data-domain="your-site.com"
src="https://pulse.ciphera.net/script.js"
></script>`}</CodeBlock>
</details>
<p>
For more details, see the{' '}

View File

@@ -58,6 +58,8 @@ export function getBrowserIcon(browserName: string) {
return <img src={src} alt={browserName} width={18} height={18} className="inline-block" style={{ verticalAlign: '-0.125em' }} />
}
const OS_DARK_INVERT = new Set(['macos', 'playstation'])
const OS_ICON_MAP: Record<string, string> = {
'windows': 'windows',
'macos': 'macos',
@@ -81,7 +83,8 @@ export function getOSIcon(osName: string) {
if (!osName) return <Question className="text-neutral-400" />
const file = OS_ICON_MAP[osName.toLowerCase()]
if (!file) return <Question className="text-neutral-400" />
return <img src={`/icons/os/${file}.png`} alt={osName} width={18} height={18} className="inline-block" style={{ verticalAlign: '-0.125em' }} />
const cls = OS_DARK_INVERT.has(file) ? 'inline-block dark:invert' : 'inline-block'
return <img src={`/icons/os/${file}.png`} alt={osName} width={18} height={18} className={cls} style={{ verticalAlign: '-0.125em' }} />
}
export function getDeviceIcon(deviceName: string) {

View File

@@ -19,18 +19,37 @@
}
// * Get domain from script tag
const script = document.currentScript || document.querySelector('script[data-domain]');
if (!script || !script.getAttribute('data-domain')) {
// * Get config from script tag, or fall back to window.pulseConfig for GTM / tag managers
// * GTM Custom HTML tags may not preserve data-* attributes on the injected <script> element,
// * so we also search by src URL and support a global config object.
const script = document.currentScript
|| document.querySelector('script[data-domain]')
|| document.querySelector('script[src*="pulse.ciphera.net/script"]');
const globalConfig = window.pulseConfig || {};
// * Helper: read a config value from script data-* attribute or globalConfig
function attr(name) {
// * Support both "storage-ttl" (data-attr style) and "storageTtl" (camelCase config style)
var camel = name.replace(/-([a-z])/g, function(_, c) { return c.toUpperCase(); });
return (script && script.getAttribute('data-' + name)) || globalConfig[name] || globalConfig[camel] || null;
}
function hasAttr(name) {
// * Support both "no-scroll" (data-attr style) and "noScroll" (camelCase config style)
var camel = name.replace(/-([a-z])/g, function(_, c) { return c.toUpperCase(); });
return (script && script.hasAttribute('data-' + name)) || globalConfig[name] === true || globalConfig[camel] === true;
}
const domain = attr('domain');
if (!domain) {
return;
}
const domain = script.getAttribute('data-domain');
const apiUrl = script.getAttribute('data-api') || 'https://pulse-api.ciphera.net';
const apiUrl = attr('api') || 'https://pulse-api.ciphera.net';
// * Visitor ID storage: "local" (default, cross-tab) or "session" (ephemeral per-tab)
const storageMode = (script.getAttribute('data-storage') || 'local').toLowerCase() === 'session' ? 'session' : 'local';
const storageMode = (attr('storage') || 'local').toLowerCase() === 'session' ? 'session' : 'local';
// * When storage is "local", optional TTL in hours; after TTL the ID is regenerated (e.g. 24 = one day)
const ttlHours = storageMode === 'local' ? parseFloat(script.getAttribute('data-storage-ttl') || '24') : 0;
const ttlHours = storageMode === 'local' ? parseFloat(attr('storage-ttl') || '24') : 0;
const ttlMs = ttlHours > 0 ? ttlHours * 60 * 60 * 1000 : 0;
let currentEventId = null;
@@ -343,7 +362,7 @@
// * Auto-track scroll depth at 25%, 50%, 75%, and 100% (on by default)
// * Each threshold fires once per pageview; resets on SPA navigation
// * Opt-out: add data-no-scroll to the script tag
var trackScroll = !script.hasAttribute('data-no-scroll');
var trackScroll = !hasAttr('no-scroll');
if (trackScroll) {
var scrollThresholds = [25, 50, 75, 100];
@@ -378,8 +397,8 @@
// * Auto-track outbound link clicks and file downloads (on by default)
// * Opt-out: add data-no-outbound or data-no-downloads to the script tag
var trackOutbound = !script.hasAttribute('data-no-outbound');
var trackDownloads = !script.hasAttribute('data-no-downloads');
var trackOutbound = !hasAttr('no-outbound');
var trackDownloads = !hasAttr('no-downloads');
if (trackOutbound || trackDownloads) {
var FILE_EXT_REGEX = /\.(pdf|zip|gz|tar|xlsx|xls|csv|docx|doc|pptx|ppt|mp4|mp3|wav|avi|mov|exe|dmg|pkg|deb|rpm|iso|7z|rar)($|\?|#)/i;