api-v2/deps/phoenix_live_reload/priv/static/phoenix_live_reload.js
2025-04-16 10:03:13 -03:00

186 lines
5.2 KiB
JavaScript

let buildFreshUrl = (link) => {
let date = Math.round(Date.now() / 1000).toString()
let url = link.href.replace(/(\&|\\?)vsn=\d*/, "")
let newLink = document.createElement('link')
let onComplete = () => {
if(link.parentNode !== null){
link.parentNode.removeChild(link)
}
}
newLink.onerror = onComplete
newLink.onload = onComplete
link.setAttribute("data-pending-removal", "")
newLink.setAttribute("rel", "stylesheet");
newLink.setAttribute("type", "text/css");
newLink.setAttribute("href", url + (url.indexOf("?") >= 0 ? "&" : "?") + "vsn=" + date)
link.parentNode.insertBefore(newLink, link.nextSibling)
return newLink
}
let repaint = () => {
let browser = navigator.userAgent.toLowerCase()
if(browser.indexOf("chrome") > -1){
setTimeout(() => document.body.offsetHeight, 25)
}
}
let cssStrategy = () => {
let reloadableLinkElements = window.parent.document.querySelectorAll(
"link[rel=stylesheet]:not([data-no-reload]):not([data-pending-removal])"
)
Array.from(reloadableLinkElements)
.filter(link => link.href)
.forEach(link => buildFreshUrl(link))
repaint()
};
let pageStrategy = channel => {
channel.off("assets_change")
window[targetWindow].location.reload()
}
let reloadStrategies = {
css: reloadPageOnCssChanges ? pageStrategy : cssStrategy,
page: pageStrategy
};
class LiveReloader {
constructor(socket){
this.socket = socket
this.logsEnabled = false
this.enabledOnce = false
this.editorURL = null
}
enable(){
this.socket.onOpen(() => {
if(this.enabledOnce){ return }
this.enabledOnce = true
if(["complete", "loaded", "interactive"].indexOf(parent.document.readyState) >= 0){
this.dispatchConnected()
} else {
parent.addEventListener("load", () => this.dispatchConnected())
}
})
this.channel = socket.channel("phoenix:live_reload", {})
this.channel.on("assets_change", msg => {
let reloadStrategy = reloadStrategies[msg.asset_type] || reloadStrategies.page
setTimeout(() => reloadStrategy(this.channel), interval)
})
this.channel.on("log", ({msg, level}) => this.logsEnabled && this.log(level, msg))
this.channel.join().receive("ok", ({editor_url}) => {
this.editorURL = editor_url
})
this.socket.connect()
}
disable(){
this.channel.leave()
socket.disconnect()
}
enableServerLogs(){ this.logsEnabled = true }
disableServerLogs(){ this.logsEnabled = false }
openEditorAtCaller(targetNode){
if(!this.editorURL){
return console.error("phoenix_live_reload cannot openEditorAtCaller without configured PLUG_EDITOR")
}
let fileLineApp = this.closestCallerFileLine(targetNode)
if(fileLineApp){
this.openFullPath(...fileLineApp)
}
}
openEditorAtDef(targetNode){
if(!this.editorURL){
return console.error("phoenix_live_reload cannot openEditorAtDef without configured PLUG_EDITOR")
}
let fileLineApp = this.closestDefFileLine(targetNode)
if(fileLineApp){
this.openFullPath(...fileLineApp)
}
}
// private
openFullPath(file, line, app){
console.log("opening full path", file, line, app)
this.channel.push("full_path", {rel_path: file, app: app})
.receive("ok", ({full_path}) => {
console.log("full path", full_path)
let url = this.editorURL
.replace("__RELATIVEFILE__", file)
.replace("__FILE__", full_path)
.replace("__LINE__", line)
window.open(url, "_self")
})
.receive("error", reason => console.error("failed to resolve full path", reason))
}
dispatchConnected(){
parent.dispatchEvent(new CustomEvent("phx:live_reload:attached", {detail: this}))
}
log(level, str){
let levelColor = level === "debug" ? "darkcyan" : "inherit"
let consoleFunc = this.logFunc(level)
this.logMsg(consoleFunc, str, levelColor)
}
logMsg(fun, str, color) {
fun(`%c📡 ${str}`, `color: ${color};`)
}
logFunc(level){
switch(level) {
case "debug":
return console.debug;
case "info":
return console.info;
case "warning":
return console.warn;
default:
return console.error;
}
}
closestCallerFileLine(node){
while(node.previousSibling){
node = node.previousSibling
if(node.nodeType === Node.COMMENT_NODE){
let callerComment = node.previousSibling
let callerMatch = callerComment &&
callerComment.nodeType === Node.COMMENT_NODE &&
callerComment.nodeValue.match(/\s@caller\s+(.+):(\d+)\s\((.*)\)\s/i)
if(callerMatch){
return [callerMatch[1], callerMatch[2], callerMatch[3]]
}
}
}
if(node.parentNode){ return this.closestCallerFileLine(node.parentNode) }
}
closestDefFileLine(node){
while(node.previousSibling){
node = node.previousSibling
if(node.nodeType === Node.COMMENT_NODE){
let fcMatch = node.nodeValue.match(/.*>\s([\w\/]+.*ex):(\d+)\s\((.*)\)\s/i)
if(fcMatch){
return [fcMatch[1], fcMatch[2], fcMatch[3]]
}
}
}
if(node.parentNode){ return this.closestDefFileLine(node.parentNode) }
}
}
reloader = new LiveReloader(socket)
reloader.enable()