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.
+
+[](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.
+
+[](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`
+
`;
+ 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`
+