@mtdt.temp/browser-core
Version:
Datadog browser core utilities.
57 lines (49 loc) • 2.15 kB
text/typescript
import { Observable } from '../tools/observable'
import { objectValues } from '../tools/utils/polyfills'
import type { Configuration } from '../domain/configuration'
import { addEventListeners, addEventListener, DOM_EVENT } from './addEventListener'
export const PageExitReason = {
HIDDEN: 'visibility_hidden',
UNLOADING: 'before_unload',
PAGEHIDE: 'page_hide',
FROZEN: 'page_frozen',
} as const
export type PageExitReason = (typeof PageExitReason)[keyof typeof PageExitReason]
export interface PageMayExitEvent {
reason: PageExitReason
}
export function createPageMayExitObservable(configuration: Configuration): Observable<PageMayExitEvent> {
return new Observable<PageMayExitEvent>((observable) => {
const { stop: stopListeners } = addEventListeners(
configuration,
window,
[DOM_EVENT.VISIBILITY_CHANGE, DOM_EVENT.FREEZE],
(event) => {
if (event.type === DOM_EVENT.VISIBILITY_CHANGE && document.visibilityState === 'hidden') {
/**
* Only event that guarantee to fire on mobile devices when the page transitions to background state
* (e.g. when user switches to a different application, goes to homescreen, etc), or is being unloaded.
*/
observable.notify({ reason: PageExitReason.HIDDEN })
} else if (event.type === DOM_EVENT.FREEZE) {
/**
* After transitioning in background a tab can be freezed to preserve resources. (cf: https://developer.chrome.com/blog/page-lifecycle-api)
* Allow to collect events happening between hidden and frozen state.
*/
observable.notify({ reason: PageExitReason.FROZEN })
}
},
{ capture: true }
)
const stopBeforeUnloadListener = addEventListener(configuration, window, DOM_EVENT.BEFORE_UNLOAD, () => {
observable.notify({ reason: PageExitReason.UNLOADING })
}).stop
return () => {
stopListeners()
stopBeforeUnloadListener()
}
})
}
export function isPageExitReason(reason: string): reason is PageExitReason {
return objectValues(PageExitReason).includes(reason as PageExitReason)
}