HOME


Mini Shell 1.0
DIR: /home/otwalrll/feedafricafarm.com/wp-content/themes/blocksy/static/js/
Upload File :
Current File : /home/otwalrll/feedafricafarm.com/wp-content/themes/blocksy/static/js/dynamic-chunks.js
import $script from 'scriptjs'

import { handleTrigger } from './frontend/dynamic-chunks/triggers'

let loadedChunks = {}
let intersectionObserver = null

export const loadDynamicChunk = (chunkId) => {
	return new Promise((resolve, reject) => {
		let chunk = ct_localizations.dynamic_js_chunks.find(
			(chunk) => chunk.id === chunkId
		)

		if (!chunk) {
			reject()
		}

		if (loadedChunks[chunk.id]) {
			resolve({ chunk: loadedChunks[chunk.id], isInitial: false })
		} else {
			loadedChunks[chunk.id] = {
				state: 'loading',
			}

			if (chunk.global_data) {
				chunk.global_data.map((data) => {
					if (!data.var || !data.data) {
						return
					}

					window[data.var] = data.data
				})
			}

			if (chunk.raw_html) {
				if (!document.querySelector(chunk.raw_html.selector)) {
					document.body.insertAdjacentHTML(
						'beforeend',
						chunk.raw_html.html
					)
				}
			}

			if (chunk.deps) {
				const depsThatAreNotLoadedIds = chunk.deps.filter(
					(id) =>
						!document.querySelector(
							`script[src*="${chunk.deps_data[id]}"]`
						)
				)
				const depsThatAreNotLoaded = depsThatAreNotLoadedIds.map(
					(id) => chunk.deps_data[id]
				)

				;[...depsThatAreNotLoadedIds, 'root']
					.map((x) => () => {
						return new Promise((depsResolve) => {
							if (x === 'root') {
								$script([chunk.url], () => {
									depsResolve()
									resolve({
										chunk: loadedChunks[chunk.id],
										isInitial: true,
									})
								})
								return
							}

							$script([chunk.deps_data[x]], () => {
								depsResolve()
							})
						})
					})
					.reduce(
						(before, after) => before.then((_) => after()),
						Promise.resolve()
					)
			} else {
				$script(chunk.url, () => {
					resolve({ chunk: loadedChunks[chunk.id], isInitial: true })
				})
			}
		}
	})
}

const loadChunkWithPayload = (chunk, payload = {}, el = null) => {
	const immediateMount = () => {
		if (!loadedChunks[chunk.id].mount) {
			return
		}

		if (el) {
			loadedChunks[chunk.id].mount(el, payload)
		} else {
			;[...document.querySelectorAll(chunk.selector)].map((el) => {
				loadedChunks[chunk.id].mount(el, payload)
			})
		}
	}

	loadDynamicChunk(chunk.id)
		.then(({ isInitial }) => {
			if (isInitial) {
				immediateMount()
			}

			// This is special case when re-triggered & payload present.
			if (!isInitial && payload) {
				immediateMount()
			}
		})
		.catch((e) => {
			console.error('Cannot load chunk', chunk.id, e)
		})
}

const addChunkToIntersectionObserver = (chunk) => {
	if (!window.IntersectionObserver) {
		return
	}

	if (!intersectionObserver) {
		intersectionObserver = new IntersectionObserver((entries) => {
			entries.map(({ boundingClientRect, target, isIntersecting }) => {
				const chunk = target.__chunk__

				if (!isIntersecting && boundingClientRect.y > 0) {
					return
				}

				let state = `target-before-bottom`

				if (!isIntersecting && boundingClientRect.y < 0) {
					state = 'target-after-bottom'
				}

				if (
					state === 'target-before-bottom' &&
					!loadedChunks[chunk.id]
				) {
					return
				}

				loadChunkWithPayload(chunk, { state, target }, chunk.el)
			})
		})
	}

	;[...document.querySelectorAll(chunk.selector)].map((el) => {
		if (el.ioObserving) {
			return
		}

		el.ioObserving = true

		const target = document.querySelector(chunk.target)

		if (!target) {
			return
		}

		target.__chunk__ = { ...chunk, el }

		intersectionObserver.observe(target)
	})
}

export const mountDynamicChunks = () => {
	ct_localizations.dynamic_js_chunks.map((chunk) => {
		if (!chunk.id) {
			return
		}

		// This is a potential problem for cases when we have multiple triggers
		if (!document.querySelector(chunk.selector)) {
			return
		}

		if (!chunk.trigger) {
			loadChunkWithPayload(chunk, null)
			return
		}

		;(Array.isArray(chunk.trigger) ? chunk.trigger : [chunk.trigger]).map(
			(trigger) => {
				trigger = trigger.trigger
					? trigger
					: {
							trigger,
							selector: chunk.selector,
					  }

				if (trigger.trigger === 'intersection-observer') {
					addChunkToIntersectionObserver(chunk)
					return
				}

				handleTrigger(
					trigger,
					chunk,
					loadChunkWithPayload,
					loadedChunks
				)
			}
		)
	})
}

export const registerDynamicChunk = (id, implementation) => {
	if (loadedChunks[id] && loadedChunks[id].state !== 'loading') {
		return
	}

	loadedChunks[id] = implementation
}