mirror of
https://github.com/wailsapp/wails.git
synced 2025-05-12 06:59:30 +08:00
240 lines
8.0 KiB
JavaScript
240 lines
8.0 KiB
JavaScript
/*
|
|
_ __ _ __
|
|
| | / /___ _(_) /____
|
|
| | /| / / __ `/ / / ___/
|
|
| |/ |/ / /_/ / / (__ )
|
|
|__/|__/\__,_/_/_/____/
|
|
The lightweight framework for web-like apps
|
|
(c) Lea Anthony 2019-present
|
|
*/
|
|
/* jshint esversion: 6 */
|
|
|
|
function init() {
|
|
|
|
// Bridge object
|
|
window.wailsbridge = {
|
|
reconnectOverlay: null,
|
|
reconnectTimer: 300,
|
|
wsURL: 'ws://' + window.location.hostname + ':34115/bridge',
|
|
connectionState: null,
|
|
config: {},
|
|
websocket: null,
|
|
callback: null,
|
|
overlayHTML:
|
|
'<div class="wails-reconnect-overlay"><div class="wails-reconnect-overlay-content"><div class="wails-reconnect-overlay-loadingspinner"></div></div></div>',
|
|
overlayCSS:
|
|
'.wails-reconnect-overlay{position:fixed;top:0;left:0;width:100%;height:100%;backdrop-filter: blur(20px) saturate(160%) contrast(45%) brightness(140%);display:none;z-index:999999}.wails-reconnect-overlay-content{position:relative;top:50%;transform:translateY(-50%);margin: 0;background-image:url();background-repeat:no-repeat;background-position:center}.wails-reconnect-overlay-loadingspinner{pointer-events:none;width:2.5em;height:2.5em;border:.4em solid transparent;border-color:#f00 #eee0 #f00 #eee0;border-radius:50%;animation:loadingspin 1s linear infinite;margin:auto;padding:2.5em}@keyframes loadingspin{100%{transform:rotate(360deg); opacity}}',
|
|
log: function (message) {
|
|
// eslint-disable-next-line
|
|
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'
|
|
);
|
|
}
|
|
};
|
|
|
|
window.onbeforeunload = function() {
|
|
if( window.wails.websocket ) {
|
|
window.wails.websocket.onclose = function () { };
|
|
window.wails.websocket.close();
|
|
window.wails.websocket = null;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
function setupIPCBridge() {
|
|
// darwin
|
|
window.webkit = {
|
|
messageHandlers: {
|
|
external: {
|
|
postMessage: (message) => {
|
|
window.wailsbridge.websocket.send(message);
|
|
}
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
// Adapted from webview - thanks zserge!
|
|
function injectCSS(css) {
|
|
const elem = document.createElement('style');
|
|
elem.setAttribute('type', 'text/css');
|
|
elem.appendChild(document.createTextNode(css));
|
|
const head = document.head || document.getElementsByTagName('head')[0];
|
|
head.appendChild(elem);
|
|
}
|
|
|
|
// Creates a node in the Dom
|
|
/**
|
|
* @param {HTMLElement} parent
|
|
* @param {string} elementType
|
|
* @param {string} id
|
|
* @param {string|null} className
|
|
* @param {string|null} content
|
|
*/
|
|
function createNode(parent, elementType, id, className, content) {
|
|
const 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() {
|
|
const body = document.body;
|
|
const wailsBridgeNode = createNode(body, 'div', 'wails-bridge', null, null);
|
|
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
|
|
const hideReconnectOverlay = function () {
|
|
window.wailsbridge.reconnectOverlay.style.display = 'none';
|
|
}
|
|
window.wailsbridge.hideReconnectOverlay = hideReconnectOverlay;
|
|
|
|
// Adds a script to the Dom.
|
|
// Removes it if second parameter is true.
|
|
function addScript(script, remove) {
|
|
const s = document.createElement('script');
|
|
s.setAttribute('type', 'text/javascript');
|
|
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');
|
|
setupIPCBridge();
|
|
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 1s (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 'b':
|
|
message = message.data.slice(1)
|
|
addScript(message);
|
|
window.wailsbridge.log('Loaded Wails Runtime');
|
|
|
|
// We need to now send a message to the backend telling it
|
|
// we have loaded (System Start)
|
|
window.webkit.messageHandlers.external.postMessage("SS");
|
|
|
|
// Now wails runtime is loaded, wails for the ready event
|
|
// and callback to the main app
|
|
// window.wails.Events.On('wails:loaded', function () {
|
|
if (window.wailsbridge.callback) {
|
|
window.wailsbridge.log('Notifying application');
|
|
window.wailsbridge.callback(window.wails);
|
|
}
|
|
// });
|
|
break;
|
|
// // Notifications
|
|
// case 'n':
|
|
// addScript(message.data.slice(1), true);
|
|
// break;
|
|
// // Binding
|
|
// case 'b':
|
|
// const binding = message.data.slice(1);
|
|
// //log("Binding: " + binding)
|
|
// window.wails._.NewBinding(binding);
|
|
// break;
|
|
// // Call back
|
|
case 'c':
|
|
const callbackData = message.data.slice(1);
|
|
window.wails._.Callback(callbackData);
|
|
break;
|
|
default:
|
|
window.wailsbridge.log('Unknown message: ' + message.data);
|
|
}
|
|
}
|
|
|
|
// Start by showing the overlay...
|
|
showReconnectOverlay();
|
|
|
|
// ...and attempt to connect
|
|
connect();
|
|
}
|
|
|
|
function InitBridge(callback) {
|
|
// Set up the bridge
|
|
init();
|
|
|
|
// Save the callback
|
|
window.wailsbridge.callback = callback;
|
|
|
|
// Start Bridge
|
|
startBridge();
|
|
}
|
|
|
|
module.exports = {
|
|
InitBridge: InitBridge,
|
|
}
|