diff --git a/README.md b/README.md index ffd5b13..7e4ef5e 100644 --- a/README.md +++ b/README.md @@ -9,14 +9,21 @@ This project seed includes the following [Platform API](https://openfin.co/platf * Examples of visual customization via CSS * Examples of visual customization by providing a custom Platform Window * Examples of behavioral customization by overriding the Platform APIs +* Examples of how to make use of the platform apis to treat views inside of a window as a group with shared context and how messages could be shared to views in a group and how you can maintain that relationship when you pull out a view from a window (so there is a linked relationship). +* How your view can make use of knowledge of it being moved from window to window and how it can keep track of the number of views that exist so it can react to that. +* How you can lock all the views on a window (to prevent views from being pulled out or views dragged in and to also hide the tabs if they are not needed). +* How you can pin platform windows so they are always on top. + +[![Click to watch on Vimeo](openfin-seed-project-first-demo-preview.png)](https://vimeo.com/401935037) [Launch in OpenFin](https://openfin.github.io/start/?manifest=https%3A%2F%2Fopenfin.github.io%2Fplatform-api-project-seed%2Fpublic.json) ## How to use this repository: * Clone this repository * Install the dependencies: `npm install` -* Start the live-server and launch the application: `npm start` +* To launch the default application: Start the live-server and launch the application: `npm start` +* To launch the main window application: Start the live-server and launch the application: `npm run mainwindow` ## Understanding the code @@ -30,25 +37,93 @@ This project seed includes the following [Platform API](https://openfin.co/platf * A [stylesheet](https://developers.openfin.co/docs/platform-api#section-standard-window-customization) is linked in the [platform-window.html](platform-window.html) file, and allows for [visual customization](styles/frame-styles.css). For a complete view of all properties, please refer to the [example stylesheet](https://github.com/openfin/layouts-v2-style-examples) ### Platform Window +The [platform-window.html](platform-window.html) file contains the [layout-container](https://developers.openfin.co/docs/platform-api#section-5-2-complete-window-customization) element and two custom elements: `left-menu` and `title-bar`. These elements, in conjunction with the [js/left-menu.js](js/left-menu.js) and [js/title-bar.js](js/title-bar.js) files, enable the following functionality: The [platform-window.html](platform-window.html) file contains the [layout-container](https://developers.openfin.co/docs/platform-api#section-5-2-complete-window-customization) element and two custom elements: `left-menu` and `title-bar`. These elements, in conjunction with the [js/platform-window.js](js/platform-window.js) file, enable the following functionality: ##### left-menu Provides examples of the following functionality: * Adding a View to an existing Window +* Adding a View that makes use of shared context if there is more than one view of that type * Adding a View in a new Window -* Saving the Window's current Layout -* Restoring the Window's current Layout * Creating a regular OpenFin Window * Saving/Restoring Platform Snapshots -* Applying a preset arrangement on the current window (Grid, Tabs, Rows) +* Applying a preset arrangement on the current window (Grid, Tabs, Rows, Columns) +* Added a view New View Tab Example that showed a way of preventing specific tabs from being moved or closed (this is just to give an idea of an approach if needed) ##### title-bar Provides examples of the following functionality: * Draggable area +* Save/Restore a layout +* Lock/Unlock the current layout +* Pin/Unpin window so it is always on top +* Close/Maximize/Minimize buttons + +### Provider +Our [custom Provider](js/platform-provider.js) includes an [extension](js/external-window-snapshot.js) that will look for a pre-configured list of [externalWindows](https://cdn.openfin.co/docs/javascript/15.80.49.21/ExternalWindow.html) (the default being the provided [my_platform_notes.txt](my_platform_notes.txt) file opened in notepad.exe) and: + +* Override `getSnapshot` to include a [externalWindows](https://cdn.openfin.co/docs/javascript/15.80.49.21/ExternalWindow.html) section containing information on any any external window included in the configuration. +* Override `applySnapshot` to look for an [externalWindows](https://cdn.openfin.co/docs/javascript/15.80.49.21/ExternalWindow.html) section and restore the position and state of any external window included in the configuration. + +## Understanding the second example- Main Window Example + +This is similar to the main example but is configured to act like an application that has a main window and child windows that may or may not use the platform layout. + +This example also shows how you can automatically capture layout when the main window closes (to local storage but it could be any store) and have the main window support a layout but not close if the last view is removed. + +[![Click to watch on Vimeo](openfin-seed-project-second-demo-preview.png)](https://vimeo.com/401942382) + +### Platform configuration +* [Platform configuration](https://developers.openfin.co/docs/platform-api#section-1-launching-a-platform) has been included in the provided [app-window-main.json](app-window-main.json) file. This config does not include a default window as this is managed by the custom provider: [provider-window-main.html](provider-window-main.html) +* The [defaultWindowOptions](https://developers.openfin.co/docs/platform-api#section-standard-window-customization) key in the provided `app-window-main.json` will instruct the platform to replace our Standard Window with the provided [platform-window-common.html](platform-window-common.html) file for new windows. +* The [commands](https://developers.openfin.co/docs/platform-api#section-5-3-using-keyboard-commands) key in the provided `app-window-main.json` will allow keyboard access to the next tab command. +* The [providerUrl]() key in the provided `app-window-main.json` will allow you to make custom changes to the Platform APIs, by pointing to a hosted [provider-window-main.html](provider-window-main.html) file. (STARTING IN v15.80.50.x) + +### CSS Customization +* A [stylesheet](https://developers.openfin.co/docs/platform-api#section-standard-window-customization) is linked in the [platform-window-common.html](platform-window-common.html) file, and allows for [visual customization](styles/frame-styles.css). For a complete view of all properties, please refer to the [example stylesheet](https://github.com/openfin/layouts-v2-style-examples) +* A [stylesheet](https://developers.openfin.co/docs/platform-api#section-standard-window-customization) is linked in the [platform-window-main.html](platform-window-main.html) file, and allows for [main window visual customization](styles/frame-styles-window-main.css) with some additional classes that only apply to the main window. For a complete view of all properties, please refer to the [example stylesheet](https://github.com/openfin/layouts-v2-style-examples) + +### Platform Window Common +The [platform-window-common.html](platform-window-common.html) file contains the [layout-container](https://developers.openfin.co/docs/platform-api#section-5-2-complete-window-customization) element and one custom element: `title-bar-common`. This element, in conjunction with the [js/title-bar-common.js](js/title-bar-common.js) file, enable the following functionality: + +##### title-bar-common +Provides examples of the following functionality: +* Draggable area +* Save/Restore a layout +* Lock/Unlock the current layout +* Pin/Unpin window so it is always on top +* Clone the current window (experimental) * Close/Maximize/Minimize buttons +### Platform Window Main +The [platform-window-main.html](platform-window-main.html) file contains the [layout-container](https://developers.openfin.co/docs/platform-api#section-5-2-complete-window-customization) element and two custom elements: `left-menu` and `title-bar-main`. These elements, in conjunction with the [js/left-menu.js](js/left-menu.js) and [js/title-bar-main.js](js/title-bar-main.js) files, enable the following functionality: + +##### left-menu +Provides examples of the following functionality: +* Adding a View to an existing Window +* Adding a View that makes use of shared context if there is more than one view of that type +* Adding a View in a new Window +* Creating a regular OpenFin Window +* Saving/Restoring Platform Snapshots +* Applying a preset arrangement on the current window (Grid, Tabs, Rows, Columns) +* Added a view New View Tab Example that showed a way of preventing specific tabs from being moved or closed (this is just to give an idea of an approach if needed) + +##### title-bar-main +Provides examples of the following functionality: +* Same button functionality as title-bar-common (except clone) +* When you close the window it saves a snapshot of the layout for all windows before closing the whole application. +* If the window is being closed because you have dragged the last view out of the main window into another window it adds a default view and doesn't close the window. If you want to have layout support in the main window then you must have at least one default view that you always want displayed or you have a main window without layout support. +* If only one view remains in the main window then it removes the close icon via the custom class in the [main window visual customization](styles/frame-styles-window-main.css) stylesheet and removes it when more than one view exists. + + ### Provider Our [custom Provider](js/platform-provider.js) includes an [extension](js/external-window-snapshot.js) that will look for a pre-configured list of [externalWindows](https://cdn.openfin.co/docs/javascript/15.80.49.21/ExternalWindow.html) (the default being the provided [my_platform_notes.txt](my_platform_notes.txt) file opened in notepad.exe) and: * Override `getSnapshot` to include a [externalWindows](https://cdn.openfin.co/docs/javascript/15.80.49.21/ExternalWindow.html) section containing information on any any external window included in the configuration. * Override `applySnapshot` to look for an [externalWindows](https://cdn.openfin.co/docs/javascript/15.80.49.21/ExternalWindow.html) section and restore the position and state of any external window included in the configuration. + +We also include an additional script [custom Provider Window Main](js/platform-provider-window-main.js) that: + +* Checks for the last saved snapshot +* If retrieved, clears it and uses the local copy to launch the last saved set of windows +* If there is no snapshot it uses a default snapshot + diff --git a/app-window-main.json b/app-window-main.json new file mode 100644 index 0000000..ac42004 --- /dev/null +++ b/app-window-main.json @@ -0,0 +1,55 @@ +{ + "runtime": { + "arguments": "--v=1 --inspect", + "version": "alpha" + }, + "shortcut": { + "company": "OpenFin", + "description": "Platform app seed local with a main window", + "icon": "https://openfin.github.io/golden-prototype/favicon.ico", + "name": "Platform app seed local main window" + }, + "platform": { + "uuid": "platform_main_window_customization_local", + "applicationIcon": "https://openfin.github.io/golden-prototype/favicon.ico", + "autoShow": false, + "providerUrl": "http://localhost:5556/provider-window-main.html", + "permissions": { + "ExternalWindow": { + "wrap": true + }, + "System": { + "getAllExternalWindows": true, + "launchExternalProcess": true, + "readRegistryValue": false, + "terminateExternalProcess": true + } + }, + "defaultWindowOptions": { + "url": "http://localhost:5556/platform-window-common.html", + "contextMenu": true, + "defaultWidth": 600, + "defaultHeight": 600, + "defaultLeft": 0, + "defaultTop": 0, + "saveWindowState": false, + "backgroundThrottling": true + }, + "defaultViewOptions": { + "experimental": { + "childWindows": true + } + }, + "commands": [ + { + "command": "stack.nextTab", + "keys": "Ctrl+Tab" + } + ] + }, + "snapshot": { + "windows": [ + + ] + } +} diff --git a/app.json b/app.json index bf0c0ac..3eb2644 100644 --- a/app.json +++ b/app.json @@ -1,7 +1,7 @@ { "runtime": { "arguments": "--v=1 --inspect", - "version": "canary" + "version": "alpha" }, "shortcut": { "company": "OpenFin", diff --git a/js/component-name-generator.js b/js/component-name-generator.js new file mode 100644 index 0000000..1597e8a --- /dev/null +++ b/js/component-name-generator.js @@ -0,0 +1,3 @@ +export function componentNameGenerator() { + return `component_A${Date.now() + Math.floor(Math.random() * 10000)}`; +} diff --git a/js/components/header/clone-layout-component.js b/js/components/header/clone-layout-component.js new file mode 100644 index 0000000..8e0dc19 --- /dev/null +++ b/js/components/header/clone-layout-component.js @@ -0,0 +1,34 @@ +import { html, render } from 'https://unpkg.com/lit-html@1.0.0/lit-html.js'; + +class CloneLayoutComponent extends HTMLElement { + + constructor() { + super(); + this.render = this.render.bind(this); + this.render(); + } + + async clone() { + let winLayout = fin.Platform.Layout.getCurrentSync(); + let config = await winLayout.getConfig(); + + // work around (this is not the official way this is a workaround until I can investigate why the name of the views is trying to be re-used) + let content = JSON.stringify(config.content); + content = content.replace(/internal-generated-view-/gi, 'internal-generated-view-' + Date.now() + "-"); + + fin.Platform.getCurrentSync().createWindow({ + layout: { + content: JSON.parse(content) + } + }); + } + + async render() { + const clone = html` +
this.clone().catch(console.error)}>🧬
+ `; + return render(clone, this); + } +} + +customElements.define('clone-layout', CloneLayoutComponent); diff --git a/js/components/header/lock-unlock-component.js b/js/components/header/lock-unlock-component.js new file mode 100644 index 0000000..cfa609e --- /dev/null +++ b/js/components/header/lock-unlock-component.js @@ -0,0 +1,41 @@ +import { html, render } from 'https://unpkg.com/lit-html@1.0.0/lit-html.js'; +import { isLayoutLocked, lockLayout, unlockLayout } from '../../layout-locking.js'; + +class LockUnlockComponent extends HTMLElement { + + constructor() { + super(); + this.render = this.render.bind(this); + this.render(); + } + + async lockLayout() { + await lockLayout(); + this.saveLockStatus(true); + this.render(); + } + + saveLockStatus(isLocked) { + sessionStorage.setItem(fin.me.identity.name + '-locked', isLocked); + } + + async unlockLayout() { + await unlockLayout(); + this.saveLockStatus(false); + this.render(); + } + + async render() { + const lockUnlock = html` + + ${ sessionStorage.getItem(fin.me.identity.name + '-locked') === "true" ? html` +
this.unlockLayout().catch(console.error)}>🔓
` + : html` +
this.lockLayout().catch(console.error)}>🔒
+ `} + `; + return render(lockUnlock, this); + } +} + +customElements.define('lock-unlock', LockUnlockComponent); diff --git a/js/components/header/pin-unpin-component.js b/js/components/header/pin-unpin-component.js new file mode 100644 index 0000000..a03c070 --- /dev/null +++ b/js/components/header/pin-unpin-component.js @@ -0,0 +1,41 @@ +import { html, render } from 'https://unpkg.com/lit-html@1.0.0/lit-html.js'; +import { isPinned, pin, unPin } from '../../pin-unpin.js'; + +class PinUnPinComponent extends HTMLElement { + + constructor() { + super(); + this.render = this.render.bind(this); + this.render(); + } + + async pin() { + await pin(); + this.savePinStatus(true); + this.render(); + } + + savePinStatus(isPinned) { + sessionStorage.setItem(fin.me.identity.name + '-pinned', isPinned); + } + + async unpin() { + await unPin(); + this.savePinStatus(false); + this.render(); + } + + async render() { + const pinUnpin = html` + + ${ sessionStorage.getItem(fin.me.identity.name + '-pinned') === "true" ? html` +
this.unpin().catch(console.error)}>📌
` + : html` +
this.pin().catch(console.error)}>📍
+ `} + `; + return render(pinUnpin, this); + } +} + +customElements.define('pin-unpin', PinUnPinComponent); diff --git a/js/components/header/save-restore-layout-component.js b/js/components/header/save-restore-layout-component.js new file mode 100644 index 0000000..1df1a81 --- /dev/null +++ b/js/components/header/save-restore-layout-component.js @@ -0,0 +1,69 @@ +import { html, render } from 'https://unpkg.com/lit-html@1.0.0/lit-html.js'; +import { isLayoutLocked } from '../../layout-locking.js'; +import hostViewContainer from '../../host-view-container.js'; + +class SaveRestoreLayoutComponent extends HTMLElement { + + constructor() { + super(); + this.render = this.render.bind(this); + this._viewContainerOptions = {}; + this.render(); + } + + saveLockStatus(isLocked) { + sessionStorage.setItem(fin.me.identity.name + '-locked', isLocked); + } + + async saveWindowLayout() { + let winLayout; + let winLayoutConfig; + + try { + console.log("About to capture and stored the current window layout."); + winLayout = fin.Platform.Layout.getCurrentSync(); + winLayoutConfig = await winLayout.getConfig(); + sessionStorage.setItem(fin.me.identity.name, JSON.stringify(winLayoutConfig)); + this._viewContainerOptions = hostViewContainer.getAllOptions(); + console.log("Captured and stored the current window layout."); + this.render(); + } catch (err) { + console.error("Error trying to capture the layout of this window.", err); + } + + } + + async restoreWindowLayout() { + const storedWinLayout = sessionStorage.getItem(fin.me.identity.name); + if (storedWinLayout) { + let winLayout = JSON.parse(storedWinLayout); + let isWinLayoutLocked = await isLayoutLocked(winLayout); + this.saveLockStatus(isWinLayoutLocked); + await fin.Platform.Layout.getCurrentSync().replace(winLayout); + + // if the view tab example view exists on this window we wish to re-apply any changes that were present + if(this._viewContainerOptions !== undefined) { + setTimeout(()=> { + console.log("Trying to restore view container settings"); + // because we tie into the tab created event elsewhere we will need to wait a while before applying the captured snapshot + hostViewContainer.updateAllOptions(this._viewContainerOptions); + }, 500); + } + } else { + throw new Error("No snapshot found in sessionstorage"); + } + } + + async render() { + const saveRestoreLayout = html` + +
this.saveWindowLayout().catch(console.error)}>💾
+ ${ sessionStorage.getItem(fin.me.identity.name) !== null ? html` +
this.restoreWindowLayout().catch(console.error)}>📂
` + : html``} + `; + return render(saveRestoreLayout, this); + } +} + +customElements.define('save-restore-layout', SaveRestoreLayoutComponent); diff --git a/js/components/header/save-snapshot-close-all-component.js b/js/components/header/save-snapshot-close-all-component.js new file mode 100644 index 0000000..ac1cd05 --- /dev/null +++ b/js/components/header/save-snapshot-close-all-component.js @@ -0,0 +1,26 @@ +import { html, render } from 'https://unpkg.com/lit-html@1.0.0/lit-html.js'; +import { saveSnapShot } from '../../platform-store.js'; + +class SaveSnapshotCloseAllComponent extends HTMLElement { + + constructor() { + super(); + this.render = this.render.bind(this); + this.render(); + } + + async render() { + const close = html` +
this.close().catch(console.error)}>
+ `; + return render(close, this); + } + + async close() { + await saveSnapShot(); + const platform = await fin.Platform.getCurrent(); + platform.quit(); + } +} + +customElements.define('save-snapshot-close-all', SaveSnapshotCloseAllComponent); diff --git a/js/components/header/window-close-component.js b/js/components/header/window-close-component.js new file mode 100644 index 0000000..ed6c633 --- /dev/null +++ b/js/components/header/window-close-component.js @@ -0,0 +1,23 @@ +import { html, render } from 'https://unpkg.com/lit-html@1.0.0/lit-html.js'; + +class WindowCloseComponent extends HTMLElement { + + constructor() { + super(); + this.render = this.render.bind(this); + this.render(); + } + + async render() { + const close = html` +
this.close().catch(console.error)}>
+ `; + return render(close, this); + } + + async close() { + fin.me.close().catch(console.error); + } +} + +customElements.define('window-close', WindowCloseComponent); diff --git a/js/components/header/window-maximize-component.js b/js/components/header/window-maximize-component.js new file mode 100644 index 0000000..1d19d1b --- /dev/null +++ b/js/components/header/window-maximize-component.js @@ -0,0 +1,30 @@ +import { html, render } from 'https://unpkg.com/lit-html@1.0.0/lit-html.js'; + +class WindowMaximizeComponent extends HTMLElement { + + constructor() { + super(); + this.render = this.render.bind(this); + this.render(); + this.maxOrRestore = this.maxOrRestore.bind(this); + } + + async render() { + const maximize = html` +
this.maxOrRestore().catch(console.error)}>
+ `; + return render(maximize, this); + } + + async maxOrRestore() { + if (await fin.me.getState() === "normal") { + await fin.me.maximize(); + this.raiseEvent('maxOrRestore', 'maximized'); + } else { + fin.me.restore(); + this.raiseEvent('maxOrRestore', 'restored'); + } + } +} + +customElements.define('window-maximize', WindowMaximizeComponent); diff --git a/js/components/header/window-minimize-component.js b/js/components/header/window-minimize-component.js new file mode 100644 index 0000000..655fa54 --- /dev/null +++ b/js/components/header/window-minimize-component.js @@ -0,0 +1,23 @@ +import { html, render } from 'https://unpkg.com/lit-html@1.0.0/lit-html.js'; + +class WindowMinimizeComponent extends HTMLElement { + + constructor() { + super(); + this.render = this.render.bind(this); + this.render(); + } + + async render() { + const minimize = html` +
this.minimize().catch(console.error)}>
+ `; + return render(minimize, this); + } + + async minimize() { + fin.me.minimize().catch(console.error); + } +} + +customElements.define('window-minimize', WindowMinimizeComponent); diff --git a/js/host-view-container.js b/js/host-view-container.js new file mode 100644 index 0000000..0ca1a66 --- /dev/null +++ b/js/host-view-container.js @@ -0,0 +1,125 @@ +class HostViewContainer { + constructor(){ + if(! HostViewContainer.instance) { + this._init(); + HostViewContainer.instance = this; + } + + return HostViewContainer.instance; + } + + getAllOptions() { + return JSON.parse(JSON.stringify(this._viewContainerOptions)); + } + + updateAllOptions(allOptions){ + + if(this._pendingAction.updateAll !== null) { + clearTimeout(this._pendingAction.updateAll); + this._pendingAction.updateAll = null; + } + this._pendingAction.updateAll = setTimeout(()=> { + // before apply this give the DOM some space to be rebuilt in case it is a case of layout replace being used (we don't want to apply the change to the layout that is about to be replaced). + let keys = Object.keys(allOptions); + + keys.forEach(key => { + this.updateOptions(allOptions[key]); + }); + }, 100); + } + + updateOptions(options) { + + if(options === undefined) { + return; + } + + let id; + + if(options.viewId !== undefined) { + id = 'tab-' + options.viewId; + } else { + this._log("A viewId was not passed so we cannot apply options to a view container"); + return; + } + + let containerOptions = this._viewContainerOptions[options.viewId]; + + if(containerOptions === undefined) { + containerOptions = {}; + } + + let keys = Object.keys(options); + + keys.forEach(key => { + containerOptions[key] = options[key]; + }); + + this._viewContainerOptions[options.viewId] = containerOptions; + + let tab = document.getElementById(id); + + if(tab !== undefined && tab !== null){ + + if (options.canMove === true || options.canMove === false) { + if(options.canMove) { + tab.style.cursor = null; + tab.classList.remove('move-disabled'); + } else { + tab.style.cursor = 'default'; + tab.classList.add('move-disabled'); + } + tab.draggable = options.canMove; + } + + if (options.canClose === true) { + tab.classList.remove('close-disabled'); + } else if(options.canClose === false) { + tab.classList.add('close-disabled'); + } + } + } + + _init() { + this._log("Init called."); + this._viewContainerOptions = {}; + this._pendingAction = { + updateAll: null + }; + + fin.InterApplicationBus.subscribe({ uuid: fin.me.identity.uuid},fin.me.identity.name + '-view-container-options', options => { + this.updateOptions(options); + }).then(() => this._log('Listening for host view container options')).catch(err => this._log(err)); + + const finWindow = fin.Window.getCurrentSync(); + + finWindow.on("view-detached", function(event) { + if(this._viewContainerOptions !== undefined && + event !== undefined && + event.viewIdentity !== undefined && + event.viewIdentity.name !== undefined) { + delete this._viewContainerOptions[event.viewIdentity.name]; + } + }); + + // document.body.addEventListener("drop", ()=> { + // this.updateAllOptions(this.getAllOptions()); + // }); + + const myLayoutContainer = document.getElementById('layout-container'); + myLayoutContainer.addEventListener('tab-created', async (event)=> { + console.log("Tab created"); + this.updateAllOptions(this.getAllOptions()); + }); + } + + _log(message) { + console.log("HostViewContainer: " + message); + } + } + + const instance = new HostViewContainer(); + Object.freeze(instance); + + + export default instance; diff --git a/js/layout-container-binding.js b/js/layout-container-binding.js new file mode 100644 index 0000000..965b0fb --- /dev/null +++ b/js/layout-container-binding.js @@ -0,0 +1,11 @@ + + window.addEventListener('DOMContentLoaded', () => { + const containerId = 'layout-container'; + try { + console.log("About to bind the layout."); + fin.Platform.Layout.init({containerId}); + console.log("Layout has been bound to: " + containerId); + } catch(e) { + // don't throw me - after API version .50 it won't error anymore + } + }); diff --git a/js/layout-locking.js b/js/layout-locking.js new file mode 100644 index 0000000..5f5f891 --- /dev/null +++ b/js/layout-locking.js @@ -0,0 +1,90 @@ +let _wrappedLayout; + +function getWrappedLayout() { + if(_wrappedLayout !== undefined) { + return _wrappedLayout; + } + + _wrappedLayout = fin.Platform.Layout.getCurrentSync(); + return _wrappedLayout; +} + +export async function toggleLockedLayout() { + + let wrappedLayout = getWrappedLayout(); + const currentLayout = await wrappedLayout.getConfig(); + isLocked = await isLayoutLocked(oldLayout); + + if(isLocked){ + await unlockLayout(currentLayout); + } else { + await lockLayout(currentLayout); + } +} + +export async function isLayoutLocked(layoutConfig) { + let layout; + + if(layoutConfig === undefined || layoutConfig.settings === undefined) { + let wrappedLayout = getWrappedLayout(); + layout = await wrappedLayout.getConfig(); + } + + if(layout === undefined || layout.settings=== undefined) { + return false; + } + + return layout.settings.hasHeaders === false && layout.settings.reorderEnabled === false; +} + +export async function lockLayout(layoutConfig) { + let settings; + let currentLayout; + let wrappedLayout = getWrappedLayout(); + + if(layoutConfig === undefined) { + currentLayout = await wrappedLayout.getConfig(); + } else { + currentLayout = layoutConfig; + } + + settings = currentLayout.settings; + + wrappedLayout.replace({ + ...currentLayout, + settings: { + ...settings, + hasHeaders: false, + reorderEnabled: false, + } + }); +} + +export async function unlockLayout(layoutConfig) { + let settings; + let oldLayout; + let dimensions; + let wrappedLayout = getWrappedLayout(); + + if(layoutConfig === undefined) { + oldLayout = await wrappedLayout.getConfig(); + } else { + oldLayout = layoutConfig; + } + + settings = oldLayout.settings; + dimensions = oldLayout.dimensions; + + wrappedLayout.replace({ + ...oldLayout, + settings: { + ...settings, + hasHeaders: true, + reorderEnabled: true, + }, + dimensions: { + ...dimensions, + headerHeight: 20 + } + }); +} diff --git a/js/platform-window.js b/js/left-menu.js similarity index 64% rename from js/platform-window.js rename to js/left-menu.js index d5f6793..7041072 100644 --- a/js/platform-window.js +++ b/js/left-menu.js @@ -1,15 +1,8 @@ import { html, render } from 'https://unpkg.com/lit-html@1.0.0/lit-html.js'; -window.addEventListener('DOMContentLoaded', () => { - const containerId = 'layout-container'; - try { - fin.Platform.Layout.init({containerId}); - } catch(e) { - // don't throw me - after .50/.51 it won't error anymore - } -}); - const chartUrl = 'https://cdn.openfin.co/embed-web/chart.html'; +const contextViewUrl = document.location.host + '/platform-view-context.html'; +const tabViewUrl = document.location.host + '/platform-view-tab-example.html'; //Our Left Menu element class LeftMenu extends HTMLElement { @@ -17,31 +10,34 @@ class LeftMenu extends HTMLElement { super(); this.render = this.render.bind(this); this.createChart = this.createChart.bind(this); - this.saveSnapshot = this.saveSnapshot.bind(this); - this.restoreSnapshot = this.restoreSnapshot.bind(this); + this.createContextView = this.createContextView.bind(this); this.toGrid = this.toGrid.bind(this); this.toTabbed = this.toTabbed.bind(this); this.toRows = this.toRows.bind(this); + this.toColumns = this.toColumns.bind(this); this.newWindow = this.newDefaultWindow.bind(this); + this.newContextWindow = this.newContextWindow.bind(this); this.nonLayoutWindow = this.nonLayoutWindow.bind(this); - this.saveWindowLayout = this.saveWindowLayout.bind(this); - this.restoreWindowLayout = this.restoreWindowLayout.bind(this); + this.saveSnapshot = this.saveSnapshot.bind(this); + this.restoreSnapshot = this.restoreSnapshot.bind(this); this.applySnapshot = this.applySnapshot.bind(this); this.render(); } - async render() { + async render() { const menuItems = html`
`; return render(menuItems, this); - } + } - async createChart() { - //we want to add a chart to the current window. + async createContextView() { + //we want to add a context view to the current window. return fin.Platform.getCurrentSync().createView({ - url: chartUrl + url: contextViewUrl }, fin.me.identity); } - async saveWindowLayout() { - const winLayoutConfig = await fin.Platform.Layout.getCurrentSync().getConfig(); - localStorage.setItem(fin.me.identity.name, JSON.stringify(winLayoutConfig)); + async createTabView() { + return fin.Platform.getCurrentSync().createView({ + url: tabViewUrl + }, fin.me.identity); } - async restoreWindowLayout() { - const storedWinLayout = localStorage.getItem(fin.me.identity.name); - if (storedWinLayout) { - return fin.Platform.Layout.getCurrentSync().replace(JSON.parse(storedWinLayout)); - } else { - throw new Error("No snapshot found in localstorage"); - } + + async createChart() { + //we want to add a chart to the current window. + return fin.Platform.getCurrentSync().createView({ + url: chartUrl + }, fin.me.identity); } async toGrid() { @@ -89,6 +85,12 @@ class LeftMenu extends HTMLElement { }); } + async toColumns() { + await fin.Platform.Layout.getCurrentSync().applyPreset({ + presetType: 'columns' + }); + } + async newDefaultWindow() { //we want to add a chart in a new window. return fin.Platform.getCurrentSync().createView({ @@ -96,6 +98,13 @@ class LeftMenu extends HTMLElement { }, undefined); } + async newContextWindow() { + //we want to add a context view in a new window. + return fin.Platform.getCurrentSync().createView({ + url: contextViewUrl + }, undefined); + } + async nonLayoutWindow() { return fin.Platform.getCurrentSync().applySnapshot({ windows: [{ @@ -118,6 +127,9 @@ class LeftMenu extends HTMLElement { async restoreSnapshot() { const storedSnapshot = localStorage.getItem('snapShot'); if (storedSnapshot) { + // this will prevent the reload event + await fin.Window.getCurrentSync().removeAllListeners('close-requested'); + return fin.Platform.getCurrentSync().applySnapshot(JSON.parse(storedSnapshot), { closeExistingWindows: true }); @@ -138,39 +150,4 @@ class LeftMenu extends HTMLElement { } } -//Our Title bar element -class TitleBar extends HTMLElement { - constructor() { - super(); - this.render = this.render.bind(this); - - this.render(); - this.maxOrRestore = this.maxOrRestore.bind(this); - } - - async render() { - const titleBar = html` -
-
-
-
-
-
fin.me.minimize().catch(console.error)}>
-
this.maxOrRestore().catch(console.error)}>
-
fin.me.close().catch(console.error)}>
-
-
`; - return render(titleBar, this); - } - - async maxOrRestore() { - if (await fin.me.getState() === "normal") { - return await fin.me.maximize(); - } - - return fin.me.restore(); - } -} - customElements.define('left-menu', LeftMenu); -customElements.define('title-bar', TitleBar); diff --git a/js/messagebus-color.js b/js/messagebus-color.js new file mode 100644 index 0000000..ba02a45 --- /dev/null +++ b/js/messagebus-color.js @@ -0,0 +1,103 @@ +import { html, render } from 'https://unpkg.com/lit-html@1.0.0/lit-html.js'; +import viewHost from './view-host.js'; + +class MessageBusColor extends HTMLElement { + + constructor() { + super(); + this._viewCount = ""; + this.render = this.render.bind(this); + this.render(); + this.init(); + } + + async updateViews(event) { + let backgroundColor = event.target.value; + viewHost.sendToViews({ backgroundColor }, this._linkedHostId); + } + + async onViewUpdate(update) { + await viewHost.setContext({ backgroundColor:update.message.backgroundColor }); + document.body.style.backgroundColor = update.message.backgroundColor; + this._colorWell.value = update.message.backgroundColor; + } + + async init() { + let context = await viewHost.getContext(); + console.log("Context returned:" + JSON.stringify(context)); + + await this.applyContext(); + await this.saveContext(); + + this._colorWell = document.querySelector("#colorWell-iab"); + this._colorWell.addEventListener("change", this.updateViews.bind(this), false); + this._colorWell.select(); + await this.onViewUpdate({ + message: {backgroundColor:this._backgroundColor} + }); + + await this.listenToViews(); + + viewHost.onHostChange(async (data)=>{ + if(data.viewCount !== 1) { + await this.applyContext(); + await this.listenToViews(); + await this.onViewUpdate({ + message: { + backgroundColor: this._backgroundColor + } + }) + } else { + await this.saveContext(); + } + }); + + viewHost.onContextChange((newContext)=> { + console.log("On Host Context Change: " + JSON.stringify(newContext)); + }); + } + + async applyContext(){ + let context = await viewHost.getContext(); + + if(context !== undefined && context !== null) { + if(context.backgroundColor !== undefined) { + this._backgroundColor = context.backgroundColor; + } + + if(context.linkedHostId !== undefined) { + this._linkedHostId = context.linkedHostId; + } + } + + if(this._linkedHostId === undefined) { + this._linkedHostId = await viewHost.getCurrentId(); + } + + if(this._backgroundColor === undefined) { + this._backgroundColor ="#ff0000"; + } + } + + async saveContext(){ + await viewHost.setContext({backgroundColor: this._backgroundColor, linkedHostId:this._linkedHostId}); + } + + async listenToViews() { + if(this._unsubscribeFromMessages !== undefined) { + this._unsubscribeFromMessages.dispose(); + } + this._unsubscribeFromMessages = await viewHost.listenToViews(this.onViewUpdate.bind(this), false, this._linkedHostId); + } + + async render() { + const colorWell = html` +
+ + +
`; + return render(colorWell, this); + } +} + +customElements.define('messagebus-color', MessageBusColor); diff --git a/js/pin-unpin.js b/js/pin-unpin.js new file mode 100644 index 0000000..6920813 --- /dev/null +++ b/js/pin-unpin.js @@ -0,0 +1,46 @@ +let _currentWindow; + +export async function isPinned() { + + let win = getCurrentWindow(); + + let options = await win.getOptions(); + + return options.alwaysOnTop === true; +} + +export async function pin() { + const options = { + "alwaysOnTop": true + }; + + let win = getCurrentWindow(); + + await win.updateOptions(options); +} + +export async function unPin() { + const options = { + "alwaysOnTop": false + }; + + let win = getCurrentWindow(); + + await win.updateOptions(options); +} + +export async function togglePinState(){ + let pinned = await isPinned(); + if(pinned) { + await unPin(); + } else { + await pin(); + } +} + +function getCurrentWindow() { + if(_currentWindow === undefined) { + _currentWindow = fin.Window.getCurrentSync(); + } + return _currentWindow; +} diff --git a/js/platform-provider-window-main.js b/js/platform-provider-window-main.js new file mode 100644 index 0000000..57dfae4 --- /dev/null +++ b/js/platform-provider-window-main.js @@ -0,0 +1,47 @@ +import { getSnapShot, clearSnapShot } from './platform-store.js'; + +const defaultLayout = { + windows: [ + { + url: "http://localhost:5556/platform-window-main.html", + layout: { + content: [ + { + type: "stack", + id: "no-drop-target", + content: [ + { + type: "component", + componentName: "view", + componentState: { + name: "component_A1", + processAffinity: "ps_1", + url: "https://cdn.openfin.co/embed-web/chart.html" + } + } + ] + } + ] + } + } + ] +}; + +async function init() { + const storedSnapshot = getSnapShot(); + if (storedSnapshot) { + clearSnapShot(); + return fin.Platform.getCurrentSync().applySnapshot(storedSnapshot, { + closeExistingWindows: true + }); + } else { + return fin.Platform.getCurrentSync().applySnapshot(defaultLayout, { + closeExistingWindows: true + }); + } +} + +window.addEventListener('DOMContentLoaded', () => { + let platform = fin.Platform.getCurrentSync(); + platform.once('platform-api-ready', init.bind(this)); +}); diff --git a/js/platform-store.js b/js/platform-store.js new file mode 100644 index 0000000..b7be8bd --- /dev/null +++ b/js/platform-store.js @@ -0,0 +1,17 @@ +const key = "main-window-snapshot"; +export function getSnapShot() { + const storedSnapshot = localStorage.getItem(key); + if (storedSnapshot) { + return JSON.parse(storedSnapshot); + } + return null; +} + +export async function saveSnapShot() { + const snapshot = await fin.Platform.getCurrentSync().getSnapshot(); + localStorage.setItem(key, JSON.stringify(snapshot)); +} + +export function clearSnapShot() { + localStorage.removeItem(key); +} diff --git a/js/title-bar-common.js b/js/title-bar-common.js new file mode 100644 index 0000000..d042521 --- /dev/null +++ b/js/title-bar-common.js @@ -0,0 +1,46 @@ +import { html, render } from 'https://unpkg.com/lit-html@1.0.0/lit-html.js'; + +import './layout-container-binding.js'; +import './components/header/save-restore-layout-component.js'; +import './components/header/lock-unlock-component.js'; +import './components/header/pin-unpin-component.js'; +import './components/header/clone-layout-component.js'; +import './components/header/window-maximize-component.js'; +import './components/header/window-minimize-component.js'; +import './components/header/window-close-component.js'; + + +class TitleBarCommon extends HTMLElement { + + constructor() { + super(); + this.render = this.render.bind(this); + this.addEventListener("maximize-minimize-close-clicked", (e)=>{ + if(e.detail.controlType === "close") { + fin.me.close().catch(console.error); + } + }); + this.render(); + } + + async render() { + const titleBar = html` +
+
+
+
+
+ + + + + + + +
+
`; + return render(titleBar, this); + } +} + +customElements.define('title-bar-common', TitleBarCommon); diff --git a/js/title-bar-main.js b/js/title-bar-main.js new file mode 100644 index 0000000..ace6edc --- /dev/null +++ b/js/title-bar-main.js @@ -0,0 +1,128 @@ +import { html, render } from 'https://unpkg.com/lit-html@1.0.0/lit-html.js'; +import { saveSnapShot } from './platform-store.js'; +import './layout-container-binding.js'; +import './components/header/lock-unlock-component.js'; +import './components/header/pin-unpin-component.js'; +import './components/header/window-maximize-component.js'; +import './components/header/window-minimize-component.js'; +import './components/header/save-snapshot-close-all-component.js'; +import './components/header/save-restore-layout-component.js'; + +class TitleBarMain extends HTMLElement { + + constructor() { + super(); + this.render = this.render.bind(this); + this.render(); + this.init(); + } + + checkForLastView() { + // could use the ability to return the views for a window but we want the tab element as we are adding/removing a class + let tabs = window.document.querySelectorAll('.lm_tab'); + if(tabs.length === 1){ + tabs[0].classList.add('last-tab'); + if(!tabs[0].classList.contains('move-disabled')) { + tabs[0].style.cursor = 'default'; + tabs[0].draggable = false; + } + } else { + let markedTab = window.document.querySelector('.last-tab'); + if(markedTab !== undefined && markedTab !== null) { + markedTab.classList.remove('last-tab'); + if(!markedTab.classList.contains('move-disabled')) { + markedTab.style.cursor = null; + markedTab.draggable = true; + } + } + } + } + + async init() { + const finWindow = await fin.Window.getCurrent(); + + finWindow.on("view-attached", this.checkForLastView); + + finWindow.on("view-detached", this.checkForLastView); + + const platform = await fin.Platform.getCurrent(); + + platform.on("window-blurred", async (args)=> { + // taking a snapshot so that we have the last good snapshot since the last interaction with a window + // putting this on the event loop after two seconds so it doesn't trigger straight away when a user clicks + // from one platform window to another in case they are clicking a button to also capture the layout. + setTimeout(async () => { + await saveSnapShot(); + console.log("Platform has detected a window blur event and saved a snapshot."); + }, 1000); + }); + + finWindow.on('close-requested', async () => { + let currentLayout; + let currentLayoutConfig; + try { + // in case it is not available we still want to close the platform + currentLayout = fin.Platform.Layout.getCurrentSync(); + currentLayoutConfig = await currentLayout.getConfig(); + } catch(err) { + console.error("Error when trying to get current layout config: ", err); + } + + if(currentLayoutConfig !== undefined && currentLayoutConfig.content.length === 0) { + // this is just a way of enforcing that a main window should not close because the last view was removed + // and that it should always have at least one view in there + const layout = fin.Platform.Layout.wrapSync(finWindow.identity); + + const newLayout = { + content: [ + { + type: "stack", + content: [ + { + type: "component", + componentName: "view", + componentState: { + processAffinity: "ps_1", + url: "https://cdn.openfin.co/embed-web/chart.html" + } + } + ] + } + ] + }; + + layout.replace(newLayout); + } else { + // it was closed by the taskbar for example + await this.closePlatform(); + } + }); + + await finWindow.setAsForeground(); + } + + async closePlatform() { + const platform = await fin.Platform.getCurrent(); + platform.quit(); + } + + async render() { + const titleBar = html` +
+
+
+
+
+ + + + + + +
+
`; + return render(titleBar, this); + } +} + +customElements.define('title-bar-main', TitleBarMain); diff --git a/js/title-bar.js b/js/title-bar.js new file mode 100644 index 0000000..8076664 --- /dev/null +++ b/js/title-bar.js @@ -0,0 +1,37 @@ +import { html, render } from 'https://unpkg.com/lit-html@1.0.0/lit-html.js'; +import './layout-container-binding.js'; +import './components/header/lock-unlock-component.js'; +import './components/header/pin-unpin-component.js'; +import './components/header/window-maximize-component.js'; +import './components/header/window-minimize-component.js'; +import './components/header/window-close-component.js'; +import './components/header/save-restore-layout-component.js'; + +class TitleBar extends HTMLElement { + + constructor() { + super(); + this.render = this.render.bind(this); + this.render(); + } + + async render() { + const titleBar = html` +
+
+
+
+
+ + + + + + +
+
`; + return render(titleBar, this); + } +} + +customElements.define('title-bar', TitleBar); diff --git a/js/view-container.js b/js/view-container.js new file mode 100644 index 0000000..2dabeab --- /dev/null +++ b/js/view-container.js @@ -0,0 +1,85 @@ +class ViewContainer { + constructor(){ + if(! ViewContainer.instance) { + this._init(); + ViewContainer.instance = this; + } + + return ViewContainer.instance; + } + + getOptions() { + return this._options; + } + + async updateOptions(options) { + let view = fin.View.getCurrentSync(); + let win = await view.getCurrentWindow(); + let identity = win.identity; + + let keys = Object.keys(options); + + keys.forEach(key => { + this._options[key] = options[key]; + }); + + options.viewId = fin.me.identity.name; + + fin.InterApplicationBus.send( + identity, + identity.name + '-view-container-options', + options + ) + .then(() => this._log("View Container Options sent.")) + .catch(err => this._log(err)); + } + + onViewContainerChange(handler) { + const id = 'view-container-change-subscriber-' + Date.now() + Math.floor(Math.random() * 10000); + this._onViewContainerChangeSubscribers[id] = handler; + this._log("Listening to view container changes. Id: " + id); + return { + dispose: ()=> { + this._log("Removing view container change listener. Id: " + id); + delete this._onViewContainerChangeSubscribers[id]; + } + } + } + + async _init() { + this._log("Init called."); + this._onViewContainerChangeSubscribers = {}; + this._disposables = {}; + this._options = {}; + + const view = fin.View.getCurrentSync(); + + await view.addListener("target-changed", async (event)=> { + this._log("View Container Changed"); + this._notifyViewContainerChangeSubscribers(); + }); + + await view.addListener("destroyed", ()=> { + this._log("View destroyed."); + this._onViewContainerChangeSubscribers = {}; + }); + } + + _notifyViewContainerChangeSubscribers() { + this._log("Notifying view container change subscribers"); + let subscriberList = Object.keys(this._onViewContainerChangeSubscribers); + subscriberList.forEach(subscriberId => { + this._onViewContainerChangeSubscribers[subscriberId](); + }); + } + + _log(message) { + console.log("ViewContainer: " + message); + } + } + + const instance = new ViewContainer(); + Object.freeze(instance); + + + export default instance; diff --git a/js/view-count.js b/js/view-count.js new file mode 100644 index 0000000..76045cd --- /dev/null +++ b/js/view-count.js @@ -0,0 +1,44 @@ +import { html, render } from 'https://unpkg.com/lit-html@1.0.0/lit-html.js'; +import viewHost from './view-host.js'; + +class ViewCount extends HTMLElement { + + constructor() { + super(); + this.render = this.render.bind(this); + this.render(); + this.init(); + } + + async init() { + viewHost.onHostChange(async (data)=>{ + await this.watchViewCount(); + }); + await this.watchViewCount(); + } + + async watchViewCount(){ + if(this.disposable !== undefined) { + this.disposable.dispose(); + } + + this.disposable = viewHost.onHostViewCountChange(async (data) => { + console.log("On view host view count change:" + JSON.stringify(data)); + this.viewCount = data.viewCount; + this.render(); + }); + + this.viewCount = await viewHost.getViewCount(); + this.render(); + } + + async render() { + const viewCount = html` +
+ View Count: ${this.viewCount} +
`; + return render(viewCount, this); + } +} + +customElements.define('view-count', ViewCount); diff --git a/js/view-host.js b/js/view-host.js new file mode 100644 index 0000000..0c5540b --- /dev/null +++ b/js/view-host.js @@ -0,0 +1,290 @@ +class ViewHost { + constructor(){ + if(! ViewHost.instance) { + this._init(); + ViewHost.instance = this; + } + + return ViewHost.instance; + } + + onContextChange(handler) { + const id = 'host-context-subscriber-' + Date.now() + Math.floor(Math.random() * 10000); + this._contextSubscribers[id] = handler; + this._log("Listening to host context changes. Id: " + id); + return { + dispose: ()=> { + this._log("Removing host context change listener. Id: " + id); + delete this._contextSubscribers[id]; + } + } + } + + onHostChange(handler) { + const id = 'host-change-subscriber-' + Date.now() + Math.floor(Math.random() * 10000); + this._onHostChangeSubscribers[id] = handler; + this._log("Listening to host changes. Id: " + id); + return { + dispose: ()=> { + this._log("Removing host change listener. Id: " + id); + delete this._onHostChangeSubscribers[id]; + } + } + } + + onHostViewCountChange(handler) { + const id = 'host-viewcount-subscriber-' + Date.now() + Math.floor(Math.random() * 10000); + this._onHostViewCountSubscribers[id] = handler; + this._log("Listening to host view count changes. Id: " + id); + return { + dispose: ()=> { + this._log("Removing host view count change listener. Id: " + id); + delete this._onHostViewCountSubscribers[id]; + } + } + } + + async getViewCount() { + this._log("Get view count requested"); + let view = fin.View.getCurrentSync(); + let win = await view.getCurrentWindow(); + let views = await win.getCurrentViews(); + return views.length; + } + + async getCurrentId() { + let view = fin.View.getCurrentSync(); + let win = await view.getCurrentWindow(); + return win.identity.name; + } + + async sendToViews(message, linkedHostId = null) { + let view = fin.View.getCurrentSync(); + let win = await view.getCurrentWindow(); + let from = view.me.name; + let targetHostId; + let isCurrentHost; + + if(linkedHostId === null) { + targetHostId = win.identity.name; + isCurrentHost = true; + } else { + targetHostId = linkedHostId; + isCurrentHost = linkedHostId === win.identity.name; + } + + try { + this._log("Trying to send message to views: from: " + from + " host id: " + targetHostId + " is current host: " + isCurrentHost); + fin.InterApplicationBus.send({ + uuid: win.identity.uuid + }, targetHostId + '/views', + { + from, + message + }); + } catch (error) { + this._log("No other listeners for message from: " + from + " and targeting host: " + targetHostId); + } + } + + async listenToViews(handler, autoDisposeOnHostChange = true, linkedHostId = null) { + let view = fin.View.getCurrentSync(); + let win = await view.getCurrentWindow(); + let targetHostId; + let isCurrentHost; + + if(linkedHostId === null) { + targetHostId = win.identity.name; + isCurrentHost = true; + } else { + targetHostId = linkedHostId; + isCurrentHost = linkedHostId === win.identity.name; + } + let identity = { + uuid: win.identity.uuid + }; + let topic = targetHostId + '/views'; + const id = 'host-views-message-subscription-' + Date.now() + Math.floor(Math.random() * 10000); + this._log("Listening to view messages: auto dispose on host change: " + autoDisposeOnHostChange + " host id: " + targetHostId + " is current host: " + isCurrentHost); + fin.InterApplicationBus.subscribe(identity, topic, handler); + this._onHostViewMessageSubscriptions[id] = { + autoDisposeOnHostChange, + dispose: ()=>{ + this._log("Unsubscribing from listening to view messages: auto dispose on host change: " + autoDisposeOnHostChange + " host id: " + targetHostId + " was current host: " + isCurrentHost); + fin.InterApplicationBus.unsubscribe(identity, topic, handler); + } + }; + + return { + dispose: ()=> { + if(this._onHostViewMessageSubscriptions[id] !== undefined) { + this._onHostViewMessageSubscriptions[id].dispose(); + } + } + }; + } + + async getContext() { + this._log("Request to get host shared context."); + let view = fin.View.getCurrentSync(); + let win = await view.getCurrentWindow(); + let options = await win.getOptions(); + if(options.customData === undefined || options.customData === null || options.customData.hostContext === undefined) { + return undefined; + } + + return options.customData.hostContext; + } + + async setContext(customContext) { + this._log("Request to set host shared context."); + let view = fin.View.getCurrentSync(); + let win = await view.getCurrentWindow(); + await win.updateOptions({customData: {hostContext: customContext}}); + } + + async _init() { + this._log("Init called."); + this._contextSubscribers = {}; + this._onHostChangeSubscribers = {}; + this._onHostViewCountSubscribers = {}; + this._disposables = {}; + this._onHostViewMessageSubscriptions = {}; + + const view = fin.View.getCurrentSync(); + + await view.addListener("target-changed", async (event)=> { + this._log("Host Changed"); + this._notifyHostChangeSubscribers( + { + newHostName: event.target.name, + previousHostName: event.previousTarget === undefined ? undefined : event.previousTarget.name, + viewCount: await this.getViewCount() + } + ); + await this._listenToHostViewCountChange(); + await this._listenToHostContextChange(); + + let subscriptions = Object.keys(this._onHostViewMessageSubscriptions); + let disposablesToClear = []; + + subscriptions.forEach(subscriptionId => { + let subscription = this._onHostViewMessageSubscriptions[subscriptionId]; + if(subscription.autoDisposeOnHostChange) { + this._log("Disposing of host view message subscription as the host has changed and autoDispose on host change was: " + subscription.autoDisposeOnHostChange); + subscription.dispose(); + disposablesToClear.push(subscriptionId); + } + }); + + disposablesToClear.forEach(disposableId => { + delete this._onHostViewMessageSubscriptions[disposableId]; + }); + }); + + await this._listenToHostViewCountChange(); + await this._listenToHostContextChange(); + + await view.addListener("destroyed", ()=> { + this._log("View destroyed."); + this._contextSubscribers = {}; + this._onHostChangeSubscribers = {}; + this._onHostViewCountSubscribers = {}; + + if(this._disposables.disposeViewCountListeners !== null && this._disposeables.disposeViewCountListeners !== undefined) { + this._disposables.disposeViewCountListeners(); + } + + if(this._disposables.disposeHostContextListeners !== null && this._disposeables.disposeHostContextListeners !== undefined) { + this._disposables.disposeHostContextListeners(); + } + + let subscriptions = Object.keys(this._onHostViewMessageSubscriptions); + + subscriptions.forEach(subscriptionId => { + this._log("View Destruction: Clearing host view message subscription: " + subscriptionId); + this._onHostViewMessageSubscriptions[subscriptionId].dispose(); + }); + + this._onHostViewMessageSubscriptions = {}; + }); + } + + _notifyContextSubscribers(newContext) { + this._log("Notifying context subscribers"); + let subscriberList = Object.keys(this._contextSubscribers); + subscriberList.forEach(subscriberId => { + this._contextSubscribers[subscriberId](newContext); + }); + } + + _notifyHostChangeSubscribers(data) { + this._log("Notifying host change subscribers"); + let subscriberList = Object.keys(this._onHostChangeSubscribers); + subscriberList.forEach(subscriberId => { + this._onHostChangeSubscribers[subscriberId](data); + }); + } + + _notifyHostViewCountSubscribers(data) { + this._log("Notifying host view count subscribers"); + let subscriberList = Object.keys(this._onHostViewCountSubscribers); + subscriberList.forEach(subscriberId => { + this._onHostViewCountSubscribers[subscriberId](data); + }); + } + + async _listenToHostViewCountChange() { + + if(this._disposables.disposeViewCountListeners !== null && this._disposables.disposeViewCountListeners !== undefined) { + this._log("Disposing of previous view count listeners."); + this._disposables.disposeViewCountListeners(); + } + + let view = fin.View.getCurrentSync(); + let win = await view.getCurrentWindow(); + win.on("view-attached", this._updateViewCount.bind(this)); + win.on("view-detached", this._updateViewCount.bind(this)); + this._disposables.disposeViewCountListeners = ()=> { + win.removeListener("view-attached", this._updateViewCount.bind(this)); + win.removeListener("view-detached", this._updateViewCount.bind(this)); + }; + } + + async _listenToHostContextChange() { + + if(this._disposables.disposeHostContextListeners !== null && this._disposables.disposeHostContextListeners !== undefined) { + this._log("Disposing of previous host context change listeners."); + this._disposables.disposeHostContextListeners(); + } + + let view = fin.View.getCurrentSync(); + let win = await view.getCurrentWindow(); + win.on('options-changed', this._updateHostContext.bind(this)); + this._disposables.disposeHostContextListeners = ()=> { + win.removeListener('options-changed', this._updateHostContext.bind(this)); + }; + } + + async _updateViewCount(){ + this._notifyHostViewCountSubscribers({ + viewCount: await this.getViewCount() + }); + } + + async _updateHostContext(event) { + if (event.diff.customData && event.diff.customData) { + this._notifyContextSubscribers(event.options.customData.hostContext); + } + } + + _log(message) { + console.log("ViewHost: " + message); + } + } + + const instance = new ViewHost(); + Object.freeze(instance); + + + export default instance; diff --git a/js/view-tab-example.js b/js/view-tab-example.js new file mode 100644 index 0000000..c5b71e6 --- /dev/null +++ b/js/view-tab-example.js @@ -0,0 +1,56 @@ +import { html, render } from 'https://unpkg.com/lit-html@1.0.0/lit-html.js'; +import viewContainer from './view-container.js'; + +class ViewTabExample extends HTMLElement { + + constructor() { + super(); + this.render = this.render.bind(this); + this.render(); + this.init(); + } + + async init() { + viewContainer.onViewContainerChange(async ()=> { + let options = viewContainer.getOptions(); + await viewContainer.updateOptions(options); + }) + } + + async canClose() { + await viewContainer.updateOptions({ + canClose:true + }); + } + + async cannotClose() { + await viewContainer.updateOptions({ + canClose:false + }); + } + + async canMove() { + await viewContainer.updateOptions({ + canMove:true + }); + } + + async cannotMove() { + await viewContainer.updateOptions({ + canMove:false + }); + } + + async render() { + const viewContainerControls = html` +
+

+

+

+

+
`; + return render(viewContainerControls, this); + } +} + +customElements.define('view-tab-example', ViewTabExample); diff --git a/openfin-seed-project-first-demo-preview.png b/openfin-seed-project-first-demo-preview.png new file mode 100644 index 0000000..26842a7 Binary files /dev/null and b/openfin-seed-project-first-demo-preview.png differ diff --git a/openfin-seed-project-second-demo-preview.png b/openfin-seed-project-second-demo-preview.png new file mode 100644 index 0000000..bca1b41 Binary files /dev/null and b/openfin-seed-project-second-demo-preview.png differ diff --git a/package.json b/package.json index e552492..f4c21dd 100644 --- a/package.json +++ b/package.json @@ -1,10 +1,13 @@ { - "name": "openfin-template", - "version": "0.0.0", - "devDependencies": { - "@types/openfin": "^34.0.0", - "hadouken-js-adapter": "^0.34.3", - "http-server": "^0.12.1" - }, - "license": "Apache-2.0" + "name": "openfin-template", + "version": "0.0.0", + "devDependencies": { + "@types/openfin": "^34.0.0", + "hadouken-js-adapter": "^0.34.3", + "http-server": "^0.12.1" + }, + "license": "Apache-2.0", + "scripts": { + "mainwindow": "node server-window-main.js" + } } diff --git a/platform-view-context.html b/platform-view-context.html new file mode 100644 index 0000000..95e8a10 --- /dev/null +++ b/platform-view-context.html @@ -0,0 +1,18 @@ + + + + + View With Context + + + + + + + + + + + diff --git a/platform-view-tab-example.html b/platform-view-tab-example.html new file mode 100644 index 0000000..863d060 --- /dev/null +++ b/platform-view-tab-example.html @@ -0,0 +1,16 @@ + + + + + View Tab Example + + + + + + + + + diff --git a/platform-window-common.html b/platform-window-common.html new file mode 100644 index 0000000..06556c8 --- /dev/null +++ b/platform-window-common.html @@ -0,0 +1,23 @@ + + + + + + OpenFin Template + + + + + + + + +
+ +
+
+
+
+ + + diff --git a/platform-window-main.html b/platform-window-main.html new file mode 100644 index 0000000..cf73422 --- /dev/null +++ b/platform-window-main.html @@ -0,0 +1,26 @@ + + + + + + OpenFin Template + + + + + + + + + + +
+ +
+ +
+
+
+ + + diff --git a/platform-window.html b/platform-window.html index 762e01d..f6ad9ab 100644 --- a/platform-window.html +++ b/platform-window.html @@ -7,7 +7,11 @@ - + + + + +
diff --git a/provider-window-main.html b/provider-window-main.html new file mode 100644 index 0000000..47c5305 --- /dev/null +++ b/provider-window-main.html @@ -0,0 +1,19 @@ + + + + + + + OpenFin Template + + + + + + +
+ Custom Provider... +
+ + + diff --git a/public.json b/public.json index 8654b95..64c22a5 100644 --- a/public.json +++ b/public.json @@ -1,7 +1,7 @@ { "runtime": { "arguments": "--v=1 --inspect", - "version": "canary" + "version": "alpha" }, "shortcut": { "company": "OpenFin", diff --git a/server-window-main.js b/server-window-main.js new file mode 100644 index 0000000..ac71ffb --- /dev/null +++ b/server-window-main.js @@ -0,0 +1,37 @@ +const httpServer = require('http-server'); +const path = require('path'); +const { launch, connect } = require('hadouken-js-adapter'); + +const serverParams = { + root: path.resolve('./'), + port: 5556, + open: false, + logLevel: 2, + cache: -1 +}; + +//To Launch the OpenFin Application we need a manifestUrl. +const manifestUrl = `http://localhost:${serverParams.port}/app-window-main.json`; + +//Start the server server +const server = httpServer.createServer(serverParams); +server.listen(serverParams.port); +(async() => { + try { + console.log(manifestUrl); + //Once the server is running we can launch OpenFin and retrieve the port. + const port = await launch({ manifestUrl }); + + //We will use the port to connect from Node to determine when OpenFin exists. + const fin = await connect({ + uuid: 'server-connection', //Supply an addressable Id for the connection + address: `ws://localhost:${port}`, //Connect to the given port. + nonPersistent: true //We want OpenFin to exit as our application exists. + }); + + //Once OpenFin exists we shut down the server. + fin.once('disconnected', process.exit); + } catch (err) { + console.error(err); + } +})(); diff --git a/styles/frame-styles-window-main.css b/styles/frame-styles-window-main.css new file mode 100644 index 0000000..0de70c5 --- /dev/null +++ b/styles/frame-styles-window-main.css @@ -0,0 +1,5 @@ +@import 'frame-styles.css'; + +.last-tab .tab-button { + visibility: hidden; +} diff --git a/styles/frame-styles.css b/styles/frame-styles.css index a961816..6b72af5 100644 --- a/styles/frame-styles.css +++ b/styles/frame-styles.css @@ -19,7 +19,7 @@ --sidebar-width: 120px; - --layout-container-width: calc(100% - var(--layout-container-padding-left) - var(--layout-container-padding-right) - var(--sidebar-width)); + --layout-container-width: calc(100% - var(--layout-container-padding-left) - var(--layout-container-padding-right)); --frame-background-color: var(--frame-background); --title-bar-background-color: var(--title-bar); @@ -44,27 +44,20 @@ body { display:flex; } -.left-menu ul { - list-style: none; - padding-inline-start: 0px; - width: 100%; - margin-block-start: 20px; -} -.left-menu li { - width: 100%; +.wrapper_title { + visibility: hidden; } -.left-menu button { - width: var(--sidebar-width); - padding: 4px; - height: 40px; - background-color: var(--active-tab); - color: var(--tab-text); - border-color: var(--frame-background); +.last-tab-button { + visibility: hidden; } -.left-menu button:hover { +save-restore-layout { + -webkit-app-region: none; cursor: pointer; - background-color: var(--tab-bar-background); -} \ No newline at end of file + user-select: none; + display: flex; + justify-content: space-between; + align-items: center; +} diff --git a/styles/left-menu.css b/styles/left-menu.css new file mode 100644 index 0000000..3662558 --- /dev/null +++ b/styles/left-menu.css @@ -0,0 +1,30 @@ +@import 'frame-styles.css'; + +:root { + --sidebar-width: 120px; + --layout-container-width: calc(100% - var(--layout-container-padding-left) - var(--layout-container-padding-right) - var(--sidebar-width)); +} +.left-menu ul { + list-style: none; + padding-inline-start: 0px; + width: 100%; + margin-block-start: 20px; +} + +.left-menu li { + width: 100%; +} + +.left-menu button { + width: var(--sidebar-width); + padding: 4px; + height: 40px; + background-color: var(--active-tab); + color: var(--tab-text); + border-color: var(--frame-background); +} + +.left-menu button:hover { + cursor: pointer; + background-color: var(--tab-bar-background); +} diff --git a/styles/view-container-styles.css b/styles/view-container-styles.css new file mode 100644 index 0000000..c5fa709 --- /dev/null +++ b/styles/view-container-styles.css @@ -0,0 +1,8 @@ + +.close-disabled .tab-button { + visibility: hidden; +} + +.move-disabled { + cursor: pointer; +}