/* Wails Bridge (c) 2019-present Lea Anthony This library creates a bridge between your application and the frontend, allowing you to develop your app using standard tooling (browser extensions, live reload, etc). Usage: ``` import Bridge from "./wailsbridge"; Bridge.Start(startApp); ``` The given callback (startApp in the example) will be called when the bridge has successfully initialised. It passes the window.wails object back, in case it is not accessible directly. */ // Bridge object window.wailsbridge = { reconnectOverlay: null, reconnectTimer: 300, wsURL: "ws://localhost:34115/bridge", connectionState: null, config: {}, websocket: null, callback: null, overlayHTML: '
', overlayCSS: ".wails-reconnect-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,.6);font-family:sans-serif;display:none;z-index:999999}.wails-reconnect-overlay-content{padding:20px 30px;text-align:center;width:20em;position:relative;height:14em;border-radius:1em;margin:5% auto 0;background-color:#fff;box-shadow:1px 1px 20px 3px;background-image:url();background-repeat:no-repeat;background-position:center}.wails-reconnect-overlay-title{font-size:2em}.wails-reconnect-overlay-message{font-size:1.3em}.wails-reconnect-overlay-loadingspinner{pointer-events:none;width:2.5em;height:2.5em;border:.4em solid transparent;border-color:#3E67EC #eee #eee;border-radius:50%;animation:loadingspin 1s linear infinite;margin:auto;padding:2.5em}@keyframes loadingspin{100%{transform:rotate(360deg)}}", log: function (message) { console.log( "%c wails bridge %c " + message + " ", "background: #aa0000; color: #fff; border-radius: 3px 0px 0px 3px; padding: 1px; font-size: 0.7rem", "background: #009900; color: #fff; border-radius: 0px 3px 3px 0px; padding: 1px; font-size: 0.7rem" ); } }; // Adapted from webview - thanks zserge! function injectCSS(css) { var elem = document.createElement("style"); elem.setAttribute("type", "text/css"); if (elem.styleSheet) { elem.styleSheet.cssText = css; } else { elem.appendChild(document.createTextNode(css)); } var head = document.head || document.getElementsByTagName("head")[0]; head.appendChild(elem); } // Creates a node in the Dom function createNode(parent, elementType, id, className, content) { var d = document.createElement(elementType); if (id) { d.id = id; } if (className) { d.className = className; } if (content) { d.innerHTML = content; } parent.appendChild(d); return d; } // Sets up the overlay function setupOverlay() { var body = document.body; var wailsBridgeNode = createNode(body, "div", "wails-bridge"); wailsBridgeNode.innerHTML = window.wailsbridge.overlayHTML; // Inject the overlay CSS injectCSS(window.wailsbridge.overlayCSS); } // Start the Wails Bridge function startBridge() { // Setup the overlay setupOverlay(); window.wailsbridge.websocket = null; window.wailsbridge.connectTimer = null; window.wailsbridge.reconnectOverlay = document.querySelector( ".wails-reconnect-overlay" ); window.wailsbridge.connectionState = "disconnected"; // Shows the overlay function showReconnectOverlay() { window.wailsbridge.reconnectOverlay.style.display = "block"; } // Hides the overlay function hideReconnectOverlay() { window.wailsbridge.reconnectOverlay.style.display = "none"; } // Bridge external.invoke window.external = { invoke: function (msg) { window.wailsbridge.websocket.send(msg); } }; // Adds a script to the Dom. // Removes it if second parameter is true. function addScript(script, remove) { var s = document.createElement("script"); s.textContent = script; document.head.appendChild(s); // Remove internal messages from the DOM if (remove) { s.parentNode.removeChild(s); } } // Handles incoming websocket connections function handleConnect() { window.wailsbridge.log("Connected to backend"); hideReconnectOverlay(); clearInterval(window.wailsbridge.connectTimer); window.wailsbridge.websocket.onclose = handleDisconnect; window.wailsbridge.websocket.onmessage = handleMessage; window.wailsbridge.connectionState = "connected"; } // Handles websocket disconnects function handleDisconnect() { window.wailsbridge.log("Disconnected from backend"); window.wailsbridge.websocket = null; window.wailsbridge.connectionState = "disconnected"; showReconnectOverlay(); connect(); } // Try to connect to the backend every 300ms (default value). // Change this value in the main wailsbridge object. function connect() { window.wailsbridge.connectTimer = setInterval(function () { if (window.wailsbridge.websocket == null) { window.wailsbridge.websocket = new WebSocket(window.wailsbridge.wsURL); window.wailsbridge.websocket.onopen = handleConnect; window.wailsbridge.websocket.onerror = function (e) { e.stopImmediatePropagation(); e.stopPropagation(); e.preventDefault(); window.wailsbridge.websocket = null; return false; }; } }, window.wailsbridge.reconnectTimer); } function handleMessage(message) { // As a bridge we ignore js and css injections switch (message.data[0]) { // Wails library - inject! case "w": addScript(message.data.slice(1)); // Now wails runtime is loaded, wails for the ready event // and callback to the main app window.wails.events.on("wails:loaded", function () { window.wailsbridge.log("Wails Ready"); if (window.wailsbridge.callback) { window.wailsbridge.log("Notifying application"); window.wailsbridge.callback(window.wails); } }); window.wailsbridge.log("Loaded Wails Runtime"); break; // Notifications case "n": addScript(message.data.slice(1), true); break; // Binding case "b": var binding = message.data.slice(1) //log("Binding: " + binding) window.wails._.newBinding(binding); break; // Call back case "c": var callbackData = message.data.slice(1); log("Callback = " + callbackData); window.wails._.callback(callbackData); break; } } // Start by showing the overlay... showReconnectOverlay(); // ...and attempt to connect connect(); } export default { // The main function // Passes the main Wails object to the callback if given. Start: function (callback) { // Save the callback window.wailsbridge.callback = callback; // Start Bridge startBridge(); } };