From f9e1ad80a9c85a71786537a73b23245ec7a17a9f Mon Sep 17 00:00:00 2001 From: Denys Vuika Date: Wed, 6 Apr 2016 14:35:46 +0100 Subject: [PATCH] demo shell (ng2) basic angular2-based demo shell --- demo-shell-ng2/.gitignore | 7 + demo-shell-ng2/LICENSE | 21 ++ demo-shell-ng2/README.md | 21 ++ demo-shell-ng2/app/app.component.html | 109 ++++++++++ demo-shell-ng2/app/app.component.ts | 67 ++++++ .../app/components/AuthRouterOutlet.ts | 36 ++++ .../app/components/core/SideMenu.ts | 47 ++++ .../app/components/core/navbar.component.ts | 24 +++ demo-shell-ng2/app/components/core/tabs.ts | 51 +++++ .../form-design-surface.component.ts | 96 +++++++++ .../form-design-toolbar.component.ts | 91 ++++++++ demo-shell-ng2/app/components/forms.view.ts | 23 ++ demo-shell-ng2/app/components/home.view.ts | 17 ++ demo-shell-ng2/app/components/login.ts | 48 +++++ demo-shell-ng2/app/components/page1.view.ts | 14 ++ demo-shell-ng2/app/components/page2.view.ts | 14 ++ demo-shell-ng2/app/css/app.css | 7 + demo-shell-ng2/app/css/designer.css | 116 ++++++++++ demo-shell-ng2/app/css/menu.css | 201 ++++++++++++++++++ demo-shell-ng2/app/css/theme/navbar.css | 25 +++ demo-shell-ng2/app/css/widgets.css | 148 +++++++++++++ demo-shell-ng2/app/img/blank.gif | Bin 0 -> 43 bytes demo-shell-ng2/app/js/dropZone.js | 60 ++++++ demo-shell-ng2/app/main.ts | 9 + demo-shell-ng2/app/services/authentication.ts | 32 +++ demo-shell-ng2/app/services/form-service.ts | 55 +++++ demo-shell-ng2/app/widgets/template.js | 16 ++ demo-shell-ng2/app/widgets/widgetUtils.js | 30 +++ demo-shell-ng2/app/widgets/widgets-button.js | 95 +++++++++ demo-shell-ng2/app/widgets/widgets-col-12.js | 18 ++ .../app/widgets/widgets-col-2x6x4.js | 20 ++ .../app/widgets/widgets-col-4x4x4.js | 20 ++ demo-shell-ng2/app/widgets/widgets-col-6x6.js | 19 ++ demo-shell-ng2/app/widgets/widgets-col-8x4.js | 20 ++ .../app/widgets/widgets-container.js | 22 ++ demo-shell-ng2/app/widgets/widgets-input.js | 71 +++++++ demo-shell-ng2/app/widgets/widgets-row.js | 18 ++ demo-shell-ng2/index.html | 61 ++++++ demo-shell-ng2/package.json | 30 +++ demo-shell-ng2/tsconfig.json | 17 ++ demo-shell-ng2/typings.json | 7 + 41 files changed, 1803 insertions(+) create mode 100644 demo-shell-ng2/.gitignore create mode 100644 demo-shell-ng2/LICENSE create mode 100644 demo-shell-ng2/README.md create mode 100644 demo-shell-ng2/app/app.component.html create mode 100644 demo-shell-ng2/app/app.component.ts create mode 100644 demo-shell-ng2/app/components/AuthRouterOutlet.ts create mode 100644 demo-shell-ng2/app/components/core/SideMenu.ts create mode 100644 demo-shell-ng2/app/components/core/navbar.component.ts create mode 100644 demo-shell-ng2/app/components/core/tabs.ts create mode 100644 demo-shell-ng2/app/components/form-design-surface.component.ts create mode 100644 demo-shell-ng2/app/components/form-design-toolbar.component.ts create mode 100644 demo-shell-ng2/app/components/forms.view.ts create mode 100644 demo-shell-ng2/app/components/home.view.ts create mode 100644 demo-shell-ng2/app/components/login.ts create mode 100644 demo-shell-ng2/app/components/page1.view.ts create mode 100644 demo-shell-ng2/app/components/page2.view.ts create mode 100644 demo-shell-ng2/app/css/app.css create mode 100644 demo-shell-ng2/app/css/designer.css create mode 100644 demo-shell-ng2/app/css/menu.css create mode 100644 demo-shell-ng2/app/css/theme/navbar.css create mode 100644 demo-shell-ng2/app/css/widgets.css create mode 100644 demo-shell-ng2/app/img/blank.gif create mode 100644 demo-shell-ng2/app/js/dropZone.js create mode 100644 demo-shell-ng2/app/main.ts create mode 100644 demo-shell-ng2/app/services/authentication.ts create mode 100644 demo-shell-ng2/app/services/form-service.ts create mode 100644 demo-shell-ng2/app/widgets/template.js create mode 100644 demo-shell-ng2/app/widgets/widgetUtils.js create mode 100644 demo-shell-ng2/app/widgets/widgets-button.js create mode 100644 demo-shell-ng2/app/widgets/widgets-col-12.js create mode 100644 demo-shell-ng2/app/widgets/widgets-col-2x6x4.js create mode 100644 demo-shell-ng2/app/widgets/widgets-col-4x4x4.js create mode 100644 demo-shell-ng2/app/widgets/widgets-col-6x6.js create mode 100644 demo-shell-ng2/app/widgets/widgets-col-8x4.js create mode 100644 demo-shell-ng2/app/widgets/widgets-container.js create mode 100644 demo-shell-ng2/app/widgets/widgets-input.js create mode 100644 demo-shell-ng2/app/widgets/widgets-row.js create mode 100644 demo-shell-ng2/index.html create mode 100644 demo-shell-ng2/package.json create mode 100644 demo-shell-ng2/tsconfig.json create mode 100644 demo-shell-ng2/typings.json diff --git a/demo-shell-ng2/.gitignore b/demo-shell-ng2/.gitignore new file mode 100644 index 0000000000..dc8ef00369 --- /dev/null +++ b/demo-shell-ng2/.gitignore @@ -0,0 +1,7 @@ +typings/ +node_modules/ +*.js +*.js.map +.idea +!app/widgets/*.js +!app/js/*.js diff --git a/demo-shell-ng2/LICENSE b/demo-shell-ng2/LICENSE new file mode 100644 index 0000000000..6d9cb0c109 --- /dev/null +++ b/demo-shell-ng2/LICENSE @@ -0,0 +1,21 @@ +The MIT License (MIT) + +Copyright (c) 2016 Alfresco + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/demo-shell-ng2/README.md b/demo-shell-ng2/README.md new file mode 100644 index 0000000000..9bafbdae51 --- /dev/null +++ b/demo-shell-ng2/README.md @@ -0,0 +1,21 @@ +### Start development + +Install the npm packages described in the `package.json` and verify that it works: + +```bash +$ npm install +$ npm start +``` +You're ready to write your application. + +Remember the npm scripts in `package.json`: + +* `npm start` - runs the compiler and a server at the same time, both in "watch mode". +* `npm run tsc` - runs the TypeScript compiler once. +* `npm run tsc:w` - runs the TypeScript compiler in watch mode; the process keeps running, awaiting changes to TypeScript files and re-compiling when it sees them. +* `npm run lite` - runs the [lite-server](https://www.npmjs.com/package/lite-server), a light-weight, static file server, written and maintained by +[John Papa](https://github.com/johnpapa) and +[Christopher Martin](https://github.com/cgmartin) +with excellent support for Angular apps that use routing. +* `npm run typings` - runs the typings tool. +* `npm run postinstall` - called by *npm* automatically *after* it successfully completes package installation. This script installs the TypeScript definition files this app requires. diff --git a/demo-shell-ng2/app/app.component.html b/demo-shell-ng2/app/app.component.html new file mode 100644 index 0000000000..ece8142775 --- /dev/null +++ b/demo-shell-ng2/app/app.component.html @@ -0,0 +1,109 @@ + + + + + +
+ + +
+
+ + + Property 1 + Property 2 + Property 3 + Property 4 + Property 5 + Property 6 + + + + + + + + + + +
diff --git a/demo-shell-ng2/app/app.component.ts b/demo-shell-ng2/app/app.component.ts new file mode 100644 index 0000000000..adb1ce2f22 --- /dev/null +++ b/demo-shell-ng2/app/app.component.ts @@ -0,0 +1,67 @@ +import {Component} from 'angular2/core'; +import {FormService} from './services/form-service'; +import {Router, RouteConfig, ROUTER_DIRECTIVES} from "angular2/router"; +import {Login} from "./components/login"; +import {Authentication} from "./services/authentication"; +import {AuthRouterOutlet} from "./components/AuthRouterOutlet"; +import {SideMenu} from "./components/core/SideMenu"; +import {AppNavBar} from "./components/core/navbar.component"; +import {FormDesignToolbar} from "./components/form-design-toolbar.component"; +import {HomeView} from "./components/home.view"; +import {FormsView} from "./components/forms.view"; +import {Page1View} from "./components/page1.view"; +import {Page2View} from "./components/page2.view"; + +@Component({ + selector: 'my-app', + templateUrl: 'app/app.component.html', + directives: [ROUTER_DIRECTIVES, AuthRouterOutlet, SideMenu, AppNavBar, FormDesignToolbar], + providers: [FormService] +}) +@RouteConfig([ + {path: '/', name: 'Home', component: HomeView, useAsDefault: true}, + {path: '/login', name: 'Login', component: Login}, + {path: '/forms', name: 'Forms', component: FormsView}, + {path: '/page1', name: 'Page1', component: Page1View}, + {path: '/page2', name: 'Page2', component: Page2View} +]) +export class AppComponent { + + constructor( + public auth: Authentication, + public router: Router + ){} + + toggleMenu(menu: SideMenu, $event) { + if (menu) { + menu.toggle(); + } + if ($event) { + $event.preventDefault(); + } + } + + isActive(instruction: any[]): boolean { + return this.router.isRouteActive(this.router.generate(instruction)); + } + + isLoggedIn(): boolean { + return this.auth.isLoggedIn(); + } + + onLogout(event) { + event.preventDefault(); + this.auth.logout() + .subscribe( + () => this.router.navigate(['Login']) + ); + } + + /* + hideMenu(menu: SideMenu) { + if (menu && menu.isOpen) { + menu.close(); + } + } + */ +} diff --git a/demo-shell-ng2/app/components/AuthRouterOutlet.ts b/demo-shell-ng2/app/components/AuthRouterOutlet.ts new file mode 100644 index 0000000000..592be07d13 --- /dev/null +++ b/demo-shell-ng2/app/components/AuthRouterOutlet.ts @@ -0,0 +1,36 @@ +import { ElementRef, DynamicComponentLoader, Directive, Attribute } from 'angular2/core'; +import { Router, RouterOutlet, ComponentInstruction } from 'angular2/router'; +import {Authentication} from '../services/authentication'; + +@Directive({selector: 'auth-router-outlet'}) +export class AuthRouterOutlet extends RouterOutlet { + + publicRoutes: Array; + private router: Router; + + constructor( + _elementRef: ElementRef, _loader: DynamicComponentLoader, + _parentRouter: Router, @Attribute('name') nameAttr: string, + private authentication: Authentication + ) { + super(_elementRef, _loader, _parentRouter, nameAttr); + + this.router = _parentRouter; + this.publicRoutes = [ + '', 'login', 'signup' + ]; + } + + activate(instruction: ComponentInstruction) { + if (this._canActivate(instruction.urlPath)) { + return super.activate(instruction); + } + + this.router.navigate(['Login']); + } + + _canActivate(url) { + return this.publicRoutes.indexOf(url) !== -1 + || this.authentication.isLoggedIn(); + } +} diff --git a/demo-shell-ng2/app/components/core/SideMenu.ts b/demo-shell-ng2/app/components/core/SideMenu.ts new file mode 100644 index 0000000000..42be3f9c93 --- /dev/null +++ b/demo-shell-ng2/app/components/core/SideMenu.ts @@ -0,0 +1,47 @@ +import {Component, Input, ElementRef} from "angular2/core"; + +@Component({ + selector: 'side-menu', + host: { + '(click)': 'onClick($event)', + //'(document:click)': 'onOutsideClick($event)' + }, + template: ` + + ` +}) +export class SideMenu { + @Input() title: string = ''; + @Input() direction: string = 'left'; + isOpen: boolean = false; + + constructor(private el: ElementRef) { + + } + + onClick(event) { + event.preventDefault(); + event.stopPropagation(); + } + + toggle() { + this.isOpen = !this.isOpen; + } + + open() { + this.isOpen = true; + } + + close() { + this.isOpen = false; + } +} diff --git a/demo-shell-ng2/app/components/core/navbar.component.ts b/demo-shell-ng2/app/components/core/navbar.component.ts new file mode 100644 index 0000000000..3a997c996a --- /dev/null +++ b/demo-shell-ng2/app/components/core/navbar.component.ts @@ -0,0 +1,24 @@ +import {Component} from "angular2/core"; + +@Component({ + selector: 'app-navbar', + template: ` + + `, + styles: [ + ` + :host .image-button { + padding-bottom: 10px; + padding-top: 12px; + max-height: 50px; + } + ` + ] +}) +export class AppNavBar { + +} diff --git a/demo-shell-ng2/app/components/core/tabs.ts b/demo-shell-ng2/app/components/core/tabs.ts new file mode 100644 index 0000000000..ba2d09b31a --- /dev/null +++ b/demo-shell-ng2/app/components/core/tabs.ts @@ -0,0 +1,51 @@ +import {Component, Input} from 'angular2/core'; + +@Component({ + selector: 'tabs', + template: ` + + + ` +}) +export class Tabs { + tabs: Tab[] = []; + + selectTab(tab:Tab, $event) { + this.tabs.forEach(tab => { + tab.active = false; + }); + tab.active = true; + if ($event) { + $event.preventDefault(); + } + } + + addTab(tab:Tab) { + if (this.tabs.length === 0) { + tab.active = true; + } + this.tabs.push(tab); + } +} + + +@Component({ + selector: 'tab', + template: ` +
+ +
+ ` +}) +export class Tab { + @Input('tabTitle') title; + active: Boolean; + + constructor(tabs: Tabs) { + tabs.addTab(this); + } +} diff --git a/demo-shell-ng2/app/components/form-design-surface.component.ts b/demo-shell-ng2/app/components/form-design-surface.component.ts new file mode 100644 index 0000000000..e213eb479b --- /dev/null +++ b/demo-shell-ng2/app/components/form-design-surface.component.ts @@ -0,0 +1,96 @@ +import {Component, ElementRef, OnInit} from "angular2/core"; + +declare var dropZone: any; +declare var widgets: any; + +@Component({ + selector: 'form-design-surface', + template: '
' +}) +export class FormDesignSurface implements OnInit { + + private _selectedWidget: Element; + + constructor(public elementRef: ElementRef) { + //el.nativeElement.style.backgroundColor = 'yellow'; + } + + get selectedWidget(): Element { + return this._selectedWidget; + } + + set selectedWidget(val: Element) { + if (this.selectedWidget && this.selectedWidget != val) { + this._selectedWidget.classList.remove('selected'); + } + this._selectedWidget = val; + if (this._selectedWidget) { + this._selectedWidget.classList.add('selected'); + } + } + + ngOnInit() { + // Create root container + var container = widgets.container.create(); + container.dataset.widgetType = 'container'; + this.setupWidget(container); + this.elementRef.nativeElement.appendChild(container); + } + + private setupWidget(widget: HTMLElement) { + // initialize all drop placeholders + var dropPlaceholders = widget.querySelectorAll('.drop-zone'); + for (var i = 0; i < dropPlaceholders.length; i++) { + var placeholder = dropPlaceholders[i]; + var z = new dropZone({ + element: placeholder, + onDrop: this.onWidgetDrop.bind(this) + }); + } + + // initialize clicks + if (widget.dataset['widgetId']) { + widget.addEventListener('mouseup', this.onWidgetMouseUp.bind(this), false); + } + + // wire child element clicks + var nested = widget.querySelectorAll('[data-widget-id]'); + for (var x = 0; x < nested.length; x++) { + nested[x].addEventListener('mouseup', this.onWidgetMouseUp.bind(this), false); + } + } + + private onWidgetMouseUp(e) { + var wid = e.currentTarget.dataset.widgetId; + if (wid) { + console.log('Selected Widget Id: ' + wid); + this.selectedWidget = e.currentTarget; + e.stopPropagation(); + } + } + + private onWidgetDrop(dz, opts) { + var widgetType = opts.widgetType; + if (widgetType) { + var component = widgets[widgetType]; + if (component) { + var widget = component.create(); + if (widget) { + widget.dataset.widgetType = widgetType; + this.setupWidget(widget); + + // insert widget before drop zone + var container = dz.parentElement; + container.insertBefore(widget, dz); + // create new drop zone + var zone = new dropZone({ + onDrop: this.onWidgetDrop.bind(this), + minHeight: '5px' + }); + // insert new drop zone before widget + container.insertBefore(zone.element, widget); + } + } + } + } +} diff --git a/demo-shell-ng2/app/components/form-design-toolbar.component.ts b/demo-shell-ng2/app/components/form-design-toolbar.component.ts new file mode 100644 index 0000000000..53ad6d6be9 --- /dev/null +++ b/demo-shell-ng2/app/components/form-design-toolbar.component.ts @@ -0,0 +1,91 @@ +import {Component, OnInit} from "angular2/core"; +import {FormService} from "../services/form-service"; + +@Component({ + selector: 'form-design-toolbar', + providers: [FormService], + //encapsulation: ViewEncapsulation.Native, + styles: [` + .category-header { + color: #555; + padding: 11px; + margin: 0; + background: #e7e7e7; + cursor: default; + font-size: 14px; + font-weight: bold; + } + + a.toolbar-item { + cursor: move; + + display: block; + color: #777; + font-size: 1.1em; + font-weight: 300; + text-decoration: none; + + border-bottom: 1px solid #e7e7e7; + padding: 1em; + } + + a.toolbar-item:hover { + color: #555; + background: #f8f8f8; + } + + a.toolbar-item:active, a.toolbar-item.active { + color: #fff; + border-color: #428bca; + background-color: #428bca; + } + `], + template: ` + + ` +}) +export class FormDesignToolbar implements OnInit { + + categories: any[]; + private dragCache; + + constructor( + private _formService: FormService + ) {} + + ngOnInit() { + this.categories = this._formService.getWidgetCategories(); + // Stores drag ghost elements + this.dragCache = document.getElementById('drag-images-cache'); + } + + onElementDragStart(e) { + e.dataTransfer.effectAllowed = 'move'; + + var widgetType = e.target.dataset.widgetType; + + var payload = { "widgetType": widgetType }; + e.dataTransfer.setData('text', JSON.stringify(payload)); + + //var dragImage = getDragImage(widgetType); + var dragImage = this._formService.getDragImage(widgetType); + this.dragCache.appendChild(dragImage); + //e.dataTransfer.setDragImage(dragImage, dragImage.offsetWidth / 2, 0); + e.dataTransfer.setDragImage(dragImage, 0, 0); + + e.stopPropagation(); + } + + onElementDragEnd(e) { + if (this.dragCache) { + this.dragCache.innerHTML = ''; + } + } + +} diff --git a/demo-shell-ng2/app/components/forms.view.ts b/demo-shell-ng2/app/components/forms.view.ts new file mode 100644 index 0000000000..2efdfdf7aa --- /dev/null +++ b/demo-shell-ng2/app/components/forms.view.ts @@ -0,0 +1,23 @@ +import {Component} from 'angular2/core'; +import {FormDesignSurface} from "./form-design-surface.component"; + +@Component({ + selector: 'forms-view', + template: ` +
+
+
+ + +
+
+
+ `, + directives: [FormDesignSurface] +}) +export class FormsView { + /* + @ViewChild(FormDesignSurface) + private _surface: FormDesignSurface; + */ +} diff --git a/demo-shell-ng2/app/components/home.view.ts b/demo-shell-ng2/app/components/home.view.ts new file mode 100644 index 0000000000..7b5f345768 --- /dev/null +++ b/demo-shell-ng2/app/components/home.view.ts @@ -0,0 +1,17 @@ +import {Component} from 'angular2/core'; + +@Component({ + selector: 'home-view', + template: ` +
+
+
+

Home View

+
+
+
+ `, + directives: [] +}) +export class HomeView { +} diff --git a/demo-shell-ng2/app/components/login.ts b/demo-shell-ng2/app/components/login.ts new file mode 100644 index 0000000000..f8db849843 --- /dev/null +++ b/demo-shell-ng2/app/components/login.ts @@ -0,0 +1,48 @@ +import {Component} from "angular2/core"; +import {Router, ROUTER_DIRECTIVES} from "angular2/router"; +import {FORM_DIRECTIVES, ControlGroup, FormBuilder, Validators} from "angular2/common"; +import {Authentication} from "../services/authentication"; + +@Component({ + selector: 'login', + directives: [ROUTER_DIRECTIVES, FORM_DIRECTIVES], + template: ` +
+
+
+
Check your password
+
+ + +
+
+ + +
+ +
+
+
+ ` +}) +export class Login { + form: ControlGroup; + error: boolean = false; + + constructor(fb: FormBuilder, public auth: Authentication, public router: Router) { + this.form = fb.group({ + username: ['', Validators.required], + password: ['', Validators.required] + }); + } + + onSubmit(value: any, event) { + //event.preventDefault(); + this.auth.login(value.username, value.password) + .subscribe( + //(token: any) => this.router.navigate(['../Home']), + (token: any) => this.router.navigate(['Home']), + () => { this.error = true; } + ); + } +} diff --git a/demo-shell-ng2/app/components/page1.view.ts b/demo-shell-ng2/app/components/page1.view.ts new file mode 100644 index 0000000000..95be92bcd6 --- /dev/null +++ b/demo-shell-ng2/app/components/page1.view.ts @@ -0,0 +1,14 @@ +import {Component} from "angular2/core"; +@Component({ + selector: 'page1-view', + template: ` +
+
+

Page 1

+
+
+ ` +}) +export class Page1View { + +} diff --git a/demo-shell-ng2/app/components/page2.view.ts b/demo-shell-ng2/app/components/page2.view.ts new file mode 100644 index 0000000000..0c1b3031bf --- /dev/null +++ b/demo-shell-ng2/app/components/page2.view.ts @@ -0,0 +1,14 @@ +import {Component} from "angular2/core"; +@Component({ + selector: 'page2-view', + template: ` +
+
+

Page 2

+
+
+ ` +}) +export class Page2View { + +} diff --git a/demo-shell-ng2/app/css/app.css b/demo-shell-ng2/app/css/app.css new file mode 100644 index 0000000000..f52219b4a0 --- /dev/null +++ b/demo-shell-ng2/app/css/app.css @@ -0,0 +1,7 @@ +body { padding-top: 70px; } + +/* Utils */ + +.p-10 { + padding: 10px; +} diff --git a/demo-shell-ng2/app/css/designer.css b/demo-shell-ng2/app/css/designer.css new file mode 100644 index 0000000000..21f29e9bfa --- /dev/null +++ b/demo-shell-ng2/app/css/designer.css @@ -0,0 +1,116 @@ +/* Generated by less 2.2.0 */ +.drag-image { + left: 0; + position: absolute; + top: 0; + z-index: -1; +} +.drop-zone { + border-radius: 4px; + visibility: visible; +} +.drop-zone.over { + outline: 1px dashed #ddd; +} +.widget-button { + cursor: default; +} +.widget-button[contenteditable="true"] { + cursor: text; +} +.widget-input:read-only { + background-color: #ffffff; + cursor: default; +} +.widget-row { + background-color: #F5F5F5; + border: 1px solid #dddddd; + border-radius: 4px; + margin: 5px 0; + padding: 25px 14px 0; + position: relative; +} +.widget-row:before { + background-color: #F5F5F5; + border: 1px solid #dddddd; + border-radius: 4px 0 4px 0; + color: #9DA0A4; + content: "Row"; + font-size: 12px; + font-weight: bold; + left: -1px; + padding: 3px 7px; + position: absolute; + top: -1px; +} +.widget-row.selected { + border-color: green; + transition: border-color 0.5s; +} +.widget-row.selected:before { + background-color: green; + border-color: green; + color: white; +} +.widget-col { + background-color: #ffffff; + border: 1px solid #dddddd; + border-radius: 4px; + margin: 15px 0; + padding: 39px 19px 24px; + position: relative; +} +.widget-col:before { + background-color: #F5F5F5; + border: 1px solid #dddddd; + border-radius: 4px 0 4px 0; + color: #9DA0A4; + content: "Column"; + font-size: 12px; + font-weight: bold; + left: -1px; + padding: 3px 7px; + position: absolute; + top: -1px; +} +.widget-col.selected { + border-color: green; + transition: border-color 0.5s; +} +.widget-col.selected:before { + background-color: green; + border-color: green; + color: white; +} +.widget-container { + border: 1px solid #dddddd; + border-radius: 4px; + margin-left: 0; + margin-top: 10px; + padding: 30px 15px 15px; + position: relative; + width: 100%; + word-wrap: break-word; +} +.widget-container:before { + background-color: #F5F5F5; + border: 1px solid #dddddd; + border-radius: 4px 0 4px 0; + color: #9DA0A4; + content: "Container"; + font-size: 12px; + font-weight: bold; + left: -1px; + padding: 3px 7px; + position: absolute; + top: -1px; +} +.widget-container.selected { + border-color: green; + transition: border-color 0.5s; +} +.widget-container.selected:before { + background-color: green; + border-color: green; + color: white; +} diff --git a/demo-shell-ng2/app/css/menu.css b/demo-shell-ng2/app/css/menu.css new file mode 100644 index 0000000000..1a1e670d04 --- /dev/null +++ b/demo-shell-ng2/app/css/menu.css @@ -0,0 +1,201 @@ +/* General styles for all menus */ + +.cbp-spmenu { + background-color: #fff; + position: fixed; + border-color: #e7e7e7; +} + +.cbp-spmenu > h3 { + color: #555; + font-size: 1.9em; + padding: 11px; + margin: 0; + font-weight: 300; + background: #e7e7e7; + cursor: default; +} + +.cbp-spmenu > a { + display: block; + color: #777; + font-size: 1.1em; + font-weight: 300; + text-decoration: none; +} + +.cbp-spmenu > a:hover { + color: #555; + background: #f8f8f8; +} + +.cbp-spmenu > a:active, +.cbp-spmenu > a.active { + color: #fff; + border-color: #428bca; + background-color: #428bca; +} + +/* Orientation-dependent styles for the content of the menu */ + +.cbp-spmenu-vertical { + width: 240px; + height: 100%; + top: 0; + /*z-index: 1000;*/ + z-index: 1040; /* default TBS navbar index is 1030 */ + overflow-y: auto; /* vertical scrollbar */ + overflow-x: hidden; +} + +.cbp-spmenu-vertical > a { + border-bottom: 1px solid #e7e7e7; + padding: 1em; +} + +.cbp-spmenu-horizontal { + width: 100%; + height: 150px; + left: 0; + z-index: 1000; + overflow: hidden; +} + +.cbp-spmenu-horizontal > h3 { + height: 100%; + width: 20%; + float: left; +} + +.cbp-spmenu-horizontal > a { + float: left; + width: 20%; + padding: 0.8em; + border-left: 1px solid #258ecd; +} + +/* Vertical menu that slides from the left or right */ + +.cbp-spmenu-left { + left: -240px; + border-right: 1px solid #e7e7e7; +} + +.cbp-spmenu-right { + right: -240px; + border-left: 1px solid #e7e7e7; +} + +.cbp-spmenu-left.cbp-spmenu-open { + left: 0; +} + +.cbp-spmenu-right.cbp-spmenu-open { + right: 0; +} + +/* Horizontal menu that slides from the top or bottom */ + +.cbp-spmenu-top { + top: -150px; +} + +.cbp-spmenu-bottom { + bottom: -150px; +} + +.cbp-spmenu-top.cbp-spmenu-open { + top: 0; +} + +.cbp-spmenu-bottom.cbp-spmenu-open { + bottom: 0; +} + +/* Push classes applied to the body */ + +.cbp-spmenu-push { + overflow-x: hidden; + position: relative; + left: 0; +} + +.cbp-spmenu-push-toright { + left: 240px; +} + +.cbp-spmenu-push-toleft { + left: -240px; +} + +/* Transitions */ + +.cbp-spmenu, +.cbp-spmenu-push { + -webkit-transition: all 0.3s ease; + -moz-transition: all 0.3s ease; + transition: all 0.3s ease; +} + +/* Example media queries */ + +@media screen and (max-width: 55.1875em){ + + .cbp-spmenu-horizontal { + font-size: 75%; + height: 110px; + } + + .cbp-spmenu-top { + top: -110px; + } + + .cbp-spmenu-bottom { + bottom: -110px; + } +} + +@media screen and (max-height: 26.375em){ + + .cbp-spmenu-vertical { + font-size: 90%; + width: 190px; + } + + .cbp-spmenu-left, + .cbp-spmenu-push-toleft { + left: -190px; + } + + .cbp-spmenu-right { + right: -190px; + } + + .cbp-spmenu-push-toright { + left: 190px; + } +} + +/* Extra stuff */ + +/* move scrollbar for right menu to the right */ + +.cbp-spmenu-right { direction: rtl; } +.cbp-spmenu-right > * { direction: ltr; } + +/* misc */ + +a.menu-close { + color: #555; +} + +/* Show menu inside main content rather than page */ +.inline-menu { + padding-bottom: 50px; + margin-top: 50px; +} + +.inline-menu > h3 { + font-size: 14px; + font-weight: bold; +} diff --git a/demo-shell-ng2/app/css/theme/navbar.css b/demo-shell-ng2/app/css/theme/navbar.css new file mode 100644 index 0000000000..00bee7d787 --- /dev/null +++ b/demo-shell-ng2/app/css/theme/navbar.css @@ -0,0 +1,25 @@ +/* Custom theme for AppNavBar component */ + +app-navbar .navbar-default { + background-color: #0c79bf; +} + +app-navbar .navbar-default .navbar-nav > li > a, +app-navbar .navbar-default .navbar-nav > li > a:focus { + color: #ffffff; +} + +app-navbar .navbar-alfresco-logo { + padding: 0; +} + +app-navbar .navbar-alfresco-logo > img { + border: none; + background-image: url(data:image/svg+xml;base64,<?xml version="1.0" encoding="utf-8"?>
<!-- Generator: Adobe Illustrator 19.1.0, SVG Export Plug-In . SVG Version: 6.00 Build 0)  -->
<svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px"
	 viewBox="0 0 145 42" style="enable-background:new 0 0 145 42;" width="145" height="42" xml:space="preserve">
<style type="text/css">
	.st0{fill:#FFFFFF;}
	.st1{fill:#8BDC01;}
	.st2{fill:#FBA100;}
	.st3{fill:#33AAFF;}
	.st4{fill:#017DFF;}
	.st5{fill:#FFE000;}
	.st6{fill:#47B41B;}
	.st7{fill-rule:evenodd;clip-rule:evenodd;fill:#FFFFFF;}
</style>
<g>
	<path class="st0" d="M21,21l-5.8-5.8l-0.2-0.2c-3.4-3.4-8.8-3.4-12.1,0c-3.4,3.4-3.4,8.8,0,12.1c3.4,3.4,8.8,3.3,12.1,0L21,21
		L21,21z"/>
	<path class="st0" d="M21,21l0,8.3l0,0.3c0,4.7-3.8,8.6-8.6,8.6c-4.7,0-8.6-3.8-8.6-8.6c0-4.7,3.8-8.6,8.6-8.6L21,21L21,21z"/>
	<path class="st0" d="M21,21l0,8.3l0,0.3c0,4.7-3.8,8.6-8.6,8.6c-4.7,0-8.6-3.8-8.6-8.6c0-4.7,3.8-8.6,8.6-8.6L21,21L21,21z"/>
	<path class="st0" d="M21,21l0,8.3l0,0.3c0,4.7-3.8,8.6-8.6,8.6c-4.7,0-8.6-3.8-8.6-8.6c0-0.5,0-1,0.1-1.5c3.3,2.3,8,2,11-1L21,21
		L21,21L21,21z"/>
	<path class="st1" d="M12.4,22.1c-4.1,0-7.5,3.4-7.5,7.5c0,4.1,3.4,7.5,7.5,7.5c4.1,0,7.5-3.4,7.5-7.5l0-0.3l0-7.2L12.4,22.1
		L12.4,22.1L12.4,22.1z"/>
	<path class="st0" d="M21,21l5.8,5.8l0.2,0.2c3.4,3.4,3.4,8.8,0,12.1c-3.4,3.4-8.8,3.4-12.1,0c-3.4-3.4-3.3-8.8,0-12.1L21,21L21,21z
		"/>
	<path class="st1" d="M15.7,27.8c-2.9,2.9-2.9,7.7,0,10.6c2.9,2.9,7.7,2.9,10.6,0c2.9-2.9,2.9-7.7,0-10.6l-0.2-0.2L21,22.5
		L15.7,27.8L15.7,27.8L15.7,27.8z"/>
	<path class="st0" d="M21,21l8.3,0l0.3,0c4.7,0,8.6,3.8,8.6,8.6c0,4.7-3.8,8.6-8.6,8.6c-4.7,0-8.6-3.8-8.6-8.6L21,21L21,21z"/>
	<path class="st1" d="M22.1,29.6c0,4.1,3.4,7.5,7.5,7.5c4.1,0,7.5-3.4,7.5-7.5c0-4.1-3.4-7.5-7.5-7.5l-0.3,0l-7.2,0L22.1,29.6
		L22.1,29.6L22.1,29.6z"/>
	<path class="st0" d="M21,21l5.8-5.8l0.2-0.2c3.4-3.4,8.8-3.4,12.1,0c3.4,3.4,3.4,8.8,0,12.1c-3.4,3.4-8.8,3.3-12.1,0L21,21L21,21z"
		/>
	<path class="st1" d="M27.8,26.3c2.9,2.9,7.7,2.9,10.6,0c2.9-2.9,2.9-7.7,0-10.6c-2.9-2.9-7.7-2.9-10.6,0l-0.2,0.2L22.5,21
		L27.8,26.3L27.8,26.3L27.8,26.3z"/>
	<path class="st0" d="M21,21l0-8.3l0-0.3c0-4.7,3.8-8.6,8.6-8.6c4.7,0,8.6,3.8,8.6,8.6c0,4.7-3.8,8.6-8.6,8.6L21,21L21,21z"/>
	<path class="st2" d="M29.6,19.9c4.1,0,7.5-3.4,7.5-7.5c0-4.1-3.4-7.5-7.5-7.5c-4.1,0-7.5,3.4-7.5,7.5l0,0.3l0,7.2H29.6L29.6,19.9
		L29.6,19.9z"/>
	<path class="st0" d="M21,21l-5.8-5.8l-0.2-0.2c-3.4-3.4-3.4-8.8,0-12.1c3.4-3.4,8.8-3.4,12.1,0c3.4,3.4,3.3,8.8,0,12.1L21,21L21,21
		z"/>
	<path class="st3" d="M26.3,14.2c2.9-2.9,2.9-7.7,0-10.6c-2.9-2.9-7.7-2.9-10.6,0c-2.9,2.9-2.9,7.7,0,10.6l0.2,0.2l5.1,5.1
		L26.3,14.2L26.3,14.2L26.3,14.2z"/>
	<path class="st0" d="M21,21l-8.3,0l-0.3,0c-4.7,0-8.6-3.8-8.6-8.6c0-4.7,3.8-8.6,8.6-8.6c4.7,0,8.6,3.8,8.6,8.6L21,21L21,21z"/>
	<path class="st3" d="M19.9,12.4c0-4.1-3.4-7.5-7.5-7.5c-4.1,0-7.5,3.4-7.5,7.5c0,4.1,3.4,7.5,7.5,7.5l0.3,0l7.2,0L19.9,12.4
		L19.9,12.4L19.9,12.4z"/>
	<path class="st0" d="M21,21L0.3,21c0-2.2,0.8-4.4,2.5-6.1c3.4-3.4,8.8-3.3,12.1,0L21,21L21,21z"/>
	<path class="st3" d="M14.2,15.7l5.3,5.3l-7,0c-3.8,0-7,2.5-8.2,5.9c-0.2-0.2-0.5-0.4-0.7-0.6c-2.9-2.9-2.9-7.7,0-10.6
		C6.5,12.8,11.2,12.8,14.2,15.7L14.2,15.7z"/>
	<path class="st4" d="M14.2,15.7l4.2,4.2h-5.6l-0.3,0c-3.4,0-6.3-2.3-7.2-5.4C8.1,12.9,11.7,13.3,14.2,15.7L14.2,15.7L14.2,15.7z"/>
	<path class="st4" d="M19.9,12.4l0,5.9l-4-4l-0.2-0.2c-2.4-2.4-2.8-6.1-1.2-8.9C17.6,6.1,19.9,9,19.9,12.4L19.9,12.4L19.9,12.4z"/>
	<path class="st4" d="M26.3,14.2l-4.2,4.2l0-5.6l0-0.3c0-3.4,2.3-6.3,5.4-7.2C29.2,8,28.7,11.7,26.3,14.2L26.3,14.2L26.3,14.2z"/>
	<path class="st5" d="M29.6,19.9l-5.9,0l4-4l0.2-0.2c2.4-2.4,6.1-2.8,9-1.2C35.9,17.6,33,19.9,29.6,19.9L29.6,19.9L29.6,19.9z"/>
	<path class="st6" d="M27.8,26.3l-4.2-4.2l5.6,0l0.3,0c3.4,0,6.3,2.3,7.2,5.4C33.9,29.1,30.2,28.7,27.8,26.3L27.8,26.3L27.8,26.3z"
		/>
	<path class="st6" d="M22.1,29.5l0-5.9l4,4l0.2,0.2c2.4,2.4,2.8,6.1,1.2,8.9C24.4,35.9,22.1,33,22.1,29.5L22.1,29.5L22.1,29.5z"/>
	<path class="st6" d="M15.7,27.8l4.2-4.2l0,5.6l0,0.3c0,3.4-2.3,6.3-5.4,7.2C12.9,34,13.3,30.3,15.7,27.8L15.7,27.8L15.7,27.8z"/>
	<path class="st6" d="M12.4,22.1l5.9,0l-4,4l-0.2,0.2c-2.4,2.4-6.1,2.8-9,1.2C6.1,24.4,9,22.1,12.4,22.1L12.4,22.1L12.4,22.1z"/>
	<path class="st7" d="M43.2,32.4h2.3l2.7-6.9h10.2l2.7,6.9h2.3l-8.8-22.3h-2.4L43.2,32.4L43.2,32.4z M53.3,12.2l4.3,11.4h-8.8
		L53.3,12.2L53.3,12.2z"/>
	<polygon class="st0" points="64.1,32.4 66.1,32.4 66.1,10.1 64.1,10.1 64.1,32.4 	"/>
	<path class="st0" d="M74.5,16.3h-3.2v-1.4c0-0.4,0-0.8,0-1.2c0-0.4,0.1-0.7,0.2-1c0.1-0.3,0.3-0.5,0.6-0.7c0.3-0.2,0.7-0.3,1.3-0.3
		c0.2,0,0.5,0,0.7,0c0.2,0,0.5,0.1,0.8,0.1v-1.7c-0.3,0-0.6-0.1-0.8-0.1c-0.3,0-0.5,0-0.8,0c-0.8,0-1.4,0.1-2,0.3
		c-0.5,0.2-0.9,0.6-1.2,1c-0.3,0.4-0.5,0.9-0.6,1.5c-0.1,0.6-0.2,1.2-0.2,1.9v1.5h-2.8V18h2.8v14.5h2V18h3.2V16.3L74.5,16.3z"/>
	<path class="st0" d="M75.1,32.4h2V24c0-0.8,0.1-1.6,0.4-2.3c0.3-0.7,0.7-1.3,1.2-1.8c0.5-0.5,1.2-0.9,1.9-1.2
		c0.7-0.3,1.5-0.4,2.4-0.4v-2c-1.5-0.1-2.7,0.2-3.8,0.9c-1.1,0.6-1.8,1.6-2.3,2.9h-0.1v-3.8h-1.8V32.4L75.1,32.4z"/>
	<path class="st7" d="M96.4,24.9c0-1.1-0.1-2.1-0.3-3.2c-0.3-1-0.7-2-1.2-2.8c-0.6-0.8-1.3-1.4-2.2-1.9c-0.9-0.5-2-0.7-3.3-0.7
		c-1.3,0-2.4,0.2-3.3,0.7c-0.9,0.5-1.7,1.1-2.3,1.8c-0.6,0.8-1,1.6-1.3,2.6c-0.3,1-0.4,1.9-0.4,3c0,1.1,0.1,2.1,0.4,3.1
		c0.3,1,0.7,1.8,1.3,2.6c0.6,0.7,1.3,1.3,2.3,1.8c0.9,0.4,2,0.6,3.3,0.6c2,0,3.6-0.5,4.7-1.5c1.1-1,1.9-2.4,2.3-4.3h-2
		c-0.3,1.2-0.9,2.2-1.7,3c-0.8,0.8-1.9,1.1-3.4,1.1c-0.9,0-1.7-0.2-2.4-0.5c-0.7-0.4-1.2-0.8-1.6-1.4c-0.4-0.6-0.7-1.2-0.9-1.9
		c-0.2-0.7-0.3-1.4-0.3-2.1H96.4L96.4,24.9z M84,23.2c0.1-0.7,0.3-1.3,0.5-2c0.3-0.6,0.6-1.2,1-1.7c0.4-0.5,1-0.9,1.6-1.2
		c0.6-0.3,1.3-0.4,2.2-0.4c0.8,0,1.5,0.1,2.1,0.4c0.6,0.3,1.2,0.7,1.6,1.2c0.4,0.5,0.8,1,1,1.7c0.2,0.6,0.4,1.3,0.4,2H84L84,23.2z"
		/>
	<path class="st0" d="M109.5,21c0-0.9-0.2-1.6-0.5-2.3c-0.3-0.6-0.7-1.2-1.3-1.6c-0.5-0.4-1.1-0.8-1.8-1c-0.7-0.2-1.5-0.3-2.3-0.3
		c-0.7,0-1.4,0.1-2.1,0.3c-0.7,0.2-1.4,0.4-2,0.8c-0.6,0.4-1.1,0.8-1.4,1.4c-0.4,0.6-0.5,1.3-0.5,2.1c0,0.7,0.1,1.3,0.3,1.7
		c0.2,0.5,0.5,0.9,1,1.2c0.4,0.3,0.9,0.6,1.4,0.8c0.5,0.2,1.1,0.4,1.8,0.6l2.5,0.6c0.4,0.1,0.9,0.2,1.3,0.4c0.4,0.1,0.8,0.3,1.1,0.5
		c0.3,0.2,0.6,0.5,0.8,0.8c0.2,0.3,0.3,0.7,0.3,1.2c0,0.6-0.1,1-0.4,1.4c-0.3,0.4-0.6,0.7-1.1,1c-0.4,0.3-0.9,0.4-1.4,0.5
		c-0.5,0.1-1,0.2-1.5,0.2c-1.3,0-2.4-0.3-3.3-1c-0.9-0.7-1.4-1.7-1.5-3h-2c0.2,2,0.8,3.4,2,4.3c1.2,0.9,2.7,1.4,4.6,1.4
		c0.8,0,1.5-0.1,2.3-0.3c0.8-0.2,1.5-0.5,2.1-0.8c0.6-0.4,1.1-0.9,1.5-1.5c0.4-0.6,0.6-1.4,0.6-2.2c0-0.7-0.1-1.3-0.4-1.8
		c-0.3-0.5-0.6-1-1-1.3c-0.4-0.4-0.9-0.6-1.5-0.9c-0.6-0.2-1.1-0.4-1.7-0.5l-2.6-0.6c-0.3-0.1-0.7-0.2-1.1-0.3
		c-0.4-0.1-0.8-0.3-1.1-0.5c-0.3-0.2-0.6-0.5-0.8-0.8c-0.2-0.3-0.3-0.7-0.3-1.1c0-0.5,0.1-1,0.3-1.3c0.2-0.4,0.5-0.6,0.9-0.9
		c0.4-0.2,0.8-0.4,1.2-0.5c0.4-0.1,0.9-0.1,1.3-0.1c0.6,0,1.1,0.1,1.6,0.2c0.5,0.1,0.9,0.3,1.3,0.6c0.4,0.3,0.7,0.7,0.9,1.1
		c0.2,0.4,0.4,1,0.4,1.6H109.5L109.5,21z"/>
	<path class="st0" d="M124.4,21.8c-0.2-1.9-0.9-3.2-2.1-4.2c-1.2-0.9-2.7-1.4-4.4-1.4c-1.2,0-2.3,0.2-3.3,0.6
		c-0.9,0.4-1.7,1-2.4,1.7c-0.6,0.7-1.1,1.6-1.5,2.6c-0.3,1-0.5,2-0.5,3.1c0,1.1,0.2,2.1,0.5,3.1c0.3,1,0.8,1.8,1.5,2.6
		c0.6,0.7,1.4,1.3,2.4,1.7c0.9,0.4,2,0.6,3.3,0.6c1.8,0,3.3-0.6,4.5-1.7c1.2-1.1,1.9-2.6,2.2-4.6h-2c-0.1,0.7-0.2,1.3-0.5,1.8
		c-0.3,0.6-0.6,1-1,1.5c-0.4,0.4-0.9,0.7-1.4,1c-0.5,0.2-1.1,0.3-1.7,0.3c-0.9,0-1.8-0.2-2.5-0.6c-0.7-0.4-1.3-0.9-1.8-1.5
		c-0.5-0.6-0.8-1.3-1.1-2.1c-0.2-0.8-0.4-1.5-0.4-2.3c0-0.8,0.1-1.6,0.4-2.3c0.2-0.8,0.6-1.5,1.1-2.1c0.5-0.6,1.1-1.1,1.8-1.5
		c0.7-0.4,1.5-0.6,2.5-0.6c1.3,0,2.3,0.3,3,1c0.7,0.7,1.2,1.6,1.5,2.8H124.4L124.4,21.8z"/>
	<path class="st7" d="M132.8,16.3c-1.2,0-2.3,0.2-3.3,0.6c-0.9,0.4-1.7,1-2.4,1.7c-0.6,0.7-1.1,1.6-1.5,2.6c-0.3,1-0.5,2-0.5,3.1
		c0,1.1,0.2,2.1,0.5,3.1c0.3,1,0.8,1.8,1.5,2.6c0.6,0.7,1.4,1.3,2.4,1.7c0.9,0.4,2,0.6,3.3,0.6c1.2,0,2.3-0.2,3.3-0.6
		c0.9-0.4,1.7-1,2.4-1.7c0.6-0.7,1.1-1.6,1.5-2.6c0.3-1,0.5-2,0.5-3.1c0-1.1-0.2-2.1-0.5-3.1c-0.3-1-0.8-1.8-1.5-2.6
		c-0.6-0.7-1.4-1.3-2.4-1.7C135.1,16.5,134.1,16.3,132.8,16.3L132.8,16.3z M132.8,18c0.9,0,1.8,0.2,2.5,0.6c0.7,0.4,1.3,0.9,1.8,1.5
		c0.5,0.6,0.8,1.3,1.1,2.1c0.2,0.8,0.4,1.5,0.4,2.3c0,0.8-0.1,1.6-0.4,2.3c-0.2,0.8-0.6,1.5-1.1,2.1c-0.5,0.6-1.1,1.1-1.8,1.5
		c-0.7,0.4-1.5,0.6-2.5,0.6c-0.9,0-1.8-0.2-2.5-0.6c-0.7-0.4-1.3-0.9-1.8-1.5c-0.5-0.6-0.8-1.3-1.1-2.1c-0.2-0.8-0.4-1.5-0.4-2.3
		c0-0.8,0.1-1.6,0.4-2.3c0.2-0.8,0.6-1.5,1.1-2.1c0.5-0.6,1.1-1.1,1.8-1.5C131.1,18.1,131.9,18,132.8,18L132.8,18z"/>
	<polygon class="st0" points="139.5,16.4 140.3,16.4 140.3,18.5 140.5,18.5 140.5,16.4 141.3,16.4 141.3,16.2 139.5,16.2 
		139.5,16.4 	"/>
	<polygon class="st0" points="141.5,18.5 141.8,18.5 141.8,16.6 141.8,16.6 142.5,18.5 142.8,18.5 143.5,16.6 143.5,16.6 
		143.5,18.5 143.8,18.5 143.8,16.2 143.4,16.2 142.6,18.2 141.9,16.2 141.5,16.2 141.5,18.5 	"/>
</g>
</svg>
); + margin-top: 2px; + width: 155px; + height: 48px; + background-size: auto auto; + background-position: 5px 0; + background-repeat: no-repeat; +} diff --git a/demo-shell-ng2/app/css/widgets.css b/demo-shell-ng2/app/css/widgets.css new file mode 100644 index 0000000000..f190616a75 --- /dev/null +++ b/demo-shell-ng2/app/css/widgets.css @@ -0,0 +1,148 @@ +/* Layout */ + +.drag-image { + position: absolute; + left: 0; + top: 0; + z-index: -1; +} + +.drop-zone { + border-radius: 4px; + /*height: 20px;*/ + visibility: visible /*!Important*/; +} + +.drop-zone.over { + /*height: 100px;*/ + outline: 1px dashed #ddd; +} + +ul.toolbar { + list-style: none; + margin: 0 0 1em; + padding: 0; +} + +ul.toolbar li { + background: #F7F7F7; + border: 1px solid #D9D9D9; + border-radius: 3px; + display: inline-block; + margin: 0 1em 1em 0; + padding: 0.5em 1em; +} + +/* Widgets */ + +/*.widget {}*/ + +.widget.widget-button { + cursor: default; +} + +.widget.widget-button[contentEditable="true"] { + cursor: text; +} + +/*.widget.widget-input {}*/ + +.widget.widget-input:read-only { + background-color: #fff; + cursor: default; +} + +/*.widget.widget-input.drag:before { + content: "Input"; +}*/ + +.widget.widget-input:before { + content: "Input"; +} + +.widget.widget-row { + background-color: #F5F5F5; + border: 1px solid #DDDDDD; + border-radius: 4px; + margin: 5px 0; + padding: 25px 14px 0; + position: relative; +} + +.widget.widget-row:before { + background-color: #F5F5F5; + border: 1px solid #DDDDDD; + border-radius: 4px 0 4px 0; + color: #9DA0A4; + content: "Row"; + font-size: 12px; + font-weight: bold; + left: -1px; + line-height: 2; + padding: 3px 7px; + position: absolute; + top: -1px; +} + +.widget.widget-col { + background-color: #FFFFFF; + border: 1px solid #DDDDDD; + border-radius: 4px; + margin: 15px 0; + padding: 39px 19px 24px; + position: relative; +} + +.widget.widget-col:before { + background-color: #F5F5F5; + border: 1px solid #DDDDDD; + border-radius: 4px 0 4px 0; + color: #9DA0A4; + content: "Column"; + font-size: 12px; + font-weight: bold; + left: -1px; + padding: 3px 7px; + position: absolute; + top: -1px; +} + +.widget.widget-container { + margin-left: 0; + margin-top: 10px; + padding: 30px 15px 15px; + border: 1px solid #DDDDDD; + border-radius: 4px; + position: relative; + width: 100%; + word-wrap: break-word; +} + +.widget.widget-container:before { + background-color: #F5F5F5; + border: 1px solid #DDDDDD; + border-radius: 4px 0 4px 0; + color: #9DA0A4; + content: "Container"; + font-size: 12px; + font-weight: bold; + left: -1px; + padding: 3px 7px; + position: absolute; + top: -1px; +} + +.widget.widget-row.selected, +.widget.widget-container.selected, +.widget.widget-col.selected { + border-color: green; + transition: border-color .5s; +} + +.widget.widget-row.selected:before, +.widget.widget-container.selected:before, +.widget.widget-col.selected:before { + border-color: green; + background-color: green; + color: white; +} diff --git a/demo-shell-ng2/app/img/blank.gif b/demo-shell-ng2/app/img/blank.gif new file mode 100644 index 0000000000000000000000000000000000000000..e565824aafafe632011b281cba976baf8b3ba89a GIT binary patch literal 43 qcmZ?wbhEHbWMp7uXkcLY4+e@qSs1y10y+#p0Fq%~V)9{Rum%7ZWeN!Z literal 0 HcmV?d00001 diff --git a/demo-shell-ng2/app/js/dropZone.js b/demo-shell-ng2/app/js/dropZone.js new file mode 100644 index 0000000000..84177a2478 --- /dev/null +++ b/demo-shell-ng2/app/js/dropZone.js @@ -0,0 +1,60 @@ +function dropZone(opts) { + var options = opts || {}, + element = opts.element || document.createElement('div'), + minHeight = opts.minHeight || '20px', + maxHeight = opts.maxHeight || '100px'; + + element.className = 'drop-zone'; + element.style.height = minHeight; + element.textAlign = 'center'; + + // Event listener for when the dragged element enters the drop zone. + element.addEventListener('dragenter', function (e) { + this.classList.add('over'); + this.style.height = maxHeight; + }); + + // Event listener for when the dragged element is over the drop zone. + element.addEventListener('dragover', function (e) { + if (e.preventDefault) { + e.preventDefault(); + } + e.dataTransfer.dropEffect = 'move'; + return false; + }); + + // Event listener for when the dragged element leaves the drop zone. + element.addEventListener('dragleave', function (e) { + this.classList.remove('over'); + this.style.height = minHeight; + }); + + // Event listener for when the dragged element dropped in the drop zone. + element.addEventListener('drop', function (e) { + if (e.preventDefault) e.preventDefault(); + if (e.stopPropagation) e.stopPropagation(); + + this.classList.remove('over'); + this.style.height = minHeight; + + if (typeof opts.onDrop === 'function') { + var dropData = e.dataTransfer.getData('text'); + var payload = {}; + if (dropData) { + try { + payload = JSON.parse(dropData); + } catch (err) { + payload = {}; + } + } + + opts.onDrop(this, payload); + } + + return false; + }); + + return { + "element": element + }; +} \ No newline at end of file diff --git a/demo-shell-ng2/app/main.ts b/demo-shell-ng2/app/main.ts new file mode 100644 index 0000000000..0800830073 --- /dev/null +++ b/demo-shell-ng2/app/main.ts @@ -0,0 +1,9 @@ +import {bootstrap} from 'angular2/platform/browser'; +import {AppComponent} from './app.component'; +import {ROUTER_PROVIDERS} from "angular2/router"; +import {Authentication} from "./services/authentication"; + +bootstrap(AppComponent, [ + ROUTER_PROVIDERS, + Authentication +]); diff --git a/demo-shell-ng2/app/services/authentication.ts b/demo-shell-ng2/app/services/authentication.ts new file mode 100644 index 0000000000..3ccf2c7be6 --- /dev/null +++ b/demo-shell-ng2/app/services/authentication.ts @@ -0,0 +1,32 @@ +import {Injectable} from "angular2/core"; +import {Observable} from 'rxjs/Rx'; + +@Injectable() +export class Authentication { + token: string; + + constructor() { + this.token = localStorage.getItem('token'); + } + + isLoggedIn() { + return !!localStorage.getItem('token'); + } + + login(username: String, password: String) { + if (username === 'test' && password === 'test') { + this.token = 'token'; + localStorage.setItem('token', this.token); + return Observable.of('token'); + } + + return Observable.throw('authentication failure'); + } + + logout() { + this.token = undefined; + localStorage.removeItem('token'); + + return Observable.of(true); + } +} diff --git a/demo-shell-ng2/app/services/form-service.ts b/demo-shell-ng2/app/services/form-service.ts new file mode 100644 index 0000000000..8ad1ac68fc --- /dev/null +++ b/demo-shell-ng2/app/services/form-service.ts @@ -0,0 +1,55 @@ +import {Injectable} from 'angular2/core'; + +// TODO: move to some namespace +declare var widgets: any; + +@Injectable() +export class FormService { + getWidgetCategories() { + var result = []; + var categories = {}; + var keys = Object.keys(widgets); + keys.forEach(function (key) { + var w = widgets[key]; + var categoryName = w.category || 'Misc'; + var category = categories[categoryName]; + if (!category) { + category = { + name: categoryName, + widgets: [] + }; + categories[categoryName] = category; + //result.push(category); + } + category.widgets.push({ + type: key, + name: w.name, + iconClass: w.iconClass + }); + }); + + Object.keys(categories).sort().forEach(function (key) { + result.push(categories[key]); + }); + + return result; + } + + getDragImage(widgetType: string): Element { + var w = widgets[widgetType]; + // try getting exported drag image + if (w && typeof w.getDragImage === 'function') { + var img = w.getDragImage(); + if (img) { + return img; + } + } + + // create default drag image + var dragImage = document.createElement('button'); + dragImage.className = 'btn btn-default drag-image'; + dragImage.textContent = w.name; + dragImage.style.minWidth = '100px'; + return dragImage; + } +} diff --git a/demo-shell-ng2/app/widgets/template.js b/demo-shell-ng2/app/widgets/template.js new file mode 100644 index 0000000000..92ffa05a3f --- /dev/null +++ b/demo-shell-ng2/app/widgets/template.js @@ -0,0 +1,16 @@ +var widgets = (function (widgets) { + widgets.COMPONENT = { + create: createComponent, + name: 'COMPONENT NAME', + iconClass: 'fa fa-puzzle-piece' + }; + + function createComponent() { + var widget = document.createElement('button'); + widget.textContent = 'Button'; + widget.classList.add('btn', 'btn-default', 'widget', 'widget-button'); + return widget; + } + + return widgets; +})(widgets || {}); \ No newline at end of file diff --git a/demo-shell-ng2/app/widgets/widgetUtils.js b/demo-shell-ng2/app/widgets/widgetUtils.js new file mode 100644 index 0000000000..e141bc7eca --- /dev/null +++ b/demo-shell-ng2/app/widgets/widgetUtils.js @@ -0,0 +1,30 @@ +var widgetUtils = (function (utils) { + utils.col = function(className) { + var col = document.createElement('div'); + col.dataset.widgetId = utils.uid(); + col.classList.add(className, 'widget', 'widget-col'); + + var dropPlaceholder = document.createElement('div'); + dropPlaceholder.className = 'drop-zone'; + col.appendChild(dropPlaceholder); + + return col; + }; + + utils.getRandomColor = function () { + var color = '#'+(0x1000000+(Math.random())*0xffffff).toString(16).substr(1,6); + return color; + }; + + utils.uid = (function(){ + var id = 0; + return function(){ + if (arguments[0] === 0) { + id = 0; + } + return id++; + }; + })(); + + return utils; +})(widgetUtils || {}); \ No newline at end of file diff --git a/demo-shell-ng2/app/widgets/widgets-button.js b/demo-shell-ng2/app/widgets/widgets-button.js new file mode 100644 index 0000000000..c4385b8cb2 --- /dev/null +++ b/demo-shell-ng2/app/widgets/widgets-button.js @@ -0,0 +1,95 @@ +var widgets = (function (widgets, utils) { + widgets.button = { + create: createComponent, + name: 'Button', + iconClass: 'fa fa-square', + category: 'Components' + }; + + function createComponent() { + var widget = document.createElement('button'); + widget.textContent = 'Button'; + widget.classList.add('btn', 'btn-default', 'widget', 'widget-button'); + widget.dataset.widgetId = utils.uid(); + + widget.addEventListener('dblclick', function (e) { + if (this.contentEditable !== 'true') { + this.dataset.originalValue = this.textContent; + } + + this.contentEditable = true; + selectElementContents(this); + }); + + widget.addEventListener('blur', function (e) { + if (!this.textContent) { + this.textContent = 'Button'; + } + this.contentEditable = false; + }); + + widget.addEventListener('input', function (e) { + // when button has no text - place the stub and autoselect it + if (!this.textContent) { + this.textContent = 'Button'; + selectElementContents(this); + } + }); + + widget.addEventListener('keydown', function (e) { + // special handling of 'spacebar' key + if (e.keyCode == 32) { + insertHtmlAtCursor(' '); + e.stopPropagation(); + e.preventDefault(); + return false; + } + + // 'Enter' key + if (e.keyCode == 13) { + e.stopPropagation(); + e.preventDefault(); + this.blur(); + return false; + } + + // 'Esc' key + if (e.keyCode == 27) { + var originalValue = this.dataset.originalValue || 'Button'; + this.textContent = originalValue; + delete this.dataset.originalValue; + this.blur(); + return false; + } + }); + + return widget; + } + + // TODO: move to shared library + function selectElementContents(el) { + var range = document.createRange(); + range.selectNodeContents(el); + var sel = window.getSelection(); + sel.removeAllRanges(); + sel.addRange(range); + } + + // TODO: move to shared library + function insertHtmlAtCursor(html) { + var range, node; + if (window.getSelection && window.getSelection().getRangeAt) { + range = window.getSelection().getRangeAt(0); + node = range.createContextualFragment(html); + range.insertNode(node); + window.getSelection().collapseToEnd(); + window.getSelection().modify('move', 'forward', 'character'); + } else if (document.selection && document.selection.createRange) { + document.selection.createRange().pasteHTML(html); + document.selection.collapseToEnd(); + document.selection.modify('move', 'forward', 'character'); + } + } + + return widgets; +})(widgets || {}, widgetUtils); \ No newline at end of file diff --git a/demo-shell-ng2/app/widgets/widgets-col-12.js b/demo-shell-ng2/app/widgets/widgets-col-12.js new file mode 100644 index 0000000000..9090dc7a30 --- /dev/null +++ b/demo-shell-ng2/app/widgets/widgets-col-12.js @@ -0,0 +1,18 @@ +var widgets = (function (widgets, utils) { + widgets.col12 = { + create: createComponent, + name: 'column 12', + iconClass: 'fa fa-puzzle-piece', + category: 'Grid system' + }; + + function createComponent() { + var widget = document.createElement('div'); + widget.dataset.widgetId = utils.uid(); + widget.classList.add('row', 'widget', 'widget-row'); + widget.appendChild(utils.col('col-md-12')); + return widget; + } + + return widgets; +})(widgets || {}, widgetUtils); \ No newline at end of file diff --git a/demo-shell-ng2/app/widgets/widgets-col-2x6x4.js b/demo-shell-ng2/app/widgets/widgets-col-2x6x4.js new file mode 100644 index 0000000000..6d2cee88ce --- /dev/null +++ b/demo-shell-ng2/app/widgets/widgets-col-2x6x4.js @@ -0,0 +1,20 @@ +var widgets = (function (widgets, utils) { + widgets.col2x6x4 = { + create: createComponent, + name: 'column 2:6:4', + iconClass: 'fa fa-puzzle-piece', + category: 'Grid system' + }; + + function createComponent() { + var widget = document.createElement('div'); + widget.dataset.widgetId = utils.uid(); + widget.classList.add('row', 'widget', 'widget-row'); + widget.appendChild(utils.col('col-md-2')); + widget.appendChild(utils.col('col-md-6')); + widget.appendChild(utils.col('col-md-4')); + return widget; + } + + return widgets; +})(widgets || {}, widgetUtils); \ No newline at end of file diff --git a/demo-shell-ng2/app/widgets/widgets-col-4x4x4.js b/demo-shell-ng2/app/widgets/widgets-col-4x4x4.js new file mode 100644 index 0000000000..7bc2568125 --- /dev/null +++ b/demo-shell-ng2/app/widgets/widgets-col-4x4x4.js @@ -0,0 +1,20 @@ +var widgets = (function (widgets, utils) { + widgets.col4x4x4 = { + create: createComponent, + name: 'column 4:4:4', + iconClass: 'fa fa-puzzle-piece', + category: 'Grid system' + }; + + function createComponent() { + var widget = document.createElement('div'); + widget.dataset.widgetId = utils.uid(); + widget.classList.add('row', 'widget', 'widget-row'); + widget.appendChild(utils.col('col-md-4')); + widget.appendChild(utils.col('col-md-4')); + widget.appendChild(utils.col('col-md-4')); + return widget; + } + + return widgets; +})(widgets || {}, widgetUtils); \ No newline at end of file diff --git a/demo-shell-ng2/app/widgets/widgets-col-6x6.js b/demo-shell-ng2/app/widgets/widgets-col-6x6.js new file mode 100644 index 0000000000..a2f2ca3575 --- /dev/null +++ b/demo-shell-ng2/app/widgets/widgets-col-6x6.js @@ -0,0 +1,19 @@ +var widgets = (function (widgets, utils) { + widgets.col6x6 = { + create: createComponent, + name: 'column 6:6', + iconClass: 'fa fa-puzzle-piece', + category: 'Grid system' + }; + + function createComponent() { + var widget = document.createElement('div'); + widget.dataset.widgetId = utils.uid(); + widget.classList.add('row', 'widget', 'widget-row'); + widget.appendChild(utils.col('col-md-6')); + widget.appendChild(utils.col('col-md-6')); + return widget; + } + + return widgets; +})(widgets || {}, widgetUtils); \ No newline at end of file diff --git a/demo-shell-ng2/app/widgets/widgets-col-8x4.js b/demo-shell-ng2/app/widgets/widgets-col-8x4.js new file mode 100644 index 0000000000..6b45b2cac6 --- /dev/null +++ b/demo-shell-ng2/app/widgets/widgets-col-8x4.js @@ -0,0 +1,20 @@ +var widgets = (function (widgets, utils) { + widgets.col8x4 = { + create: createComponent, + name: 'column 8:4', + iconClass: 'fa fa-puzzle-piece', + category: 'Grid system' + }; + + function createComponent() { + var widget = document.createElement('div'); + widget.dataset.widgetId = utils.uid(); + widget.classList.add('row', 'widget', 'widget-row'); + widget.appendChild(utils.col('col-md-8')); + widget.appendChild(utils.col('col-md-4')); + + return widget; + } + + return widgets; +})(widgets || {}, widgetUtils); \ No newline at end of file diff --git a/demo-shell-ng2/app/widgets/widgets-container.js b/demo-shell-ng2/app/widgets/widgets-container.js new file mode 100644 index 0000000000..f1fe6a53ef --- /dev/null +++ b/demo-shell-ng2/app/widgets/widgets-container.js @@ -0,0 +1,22 @@ +var widgets = (function (widgets, utils) { + widgets.container = { + create: createComponent, + name: 'container', + iconClass: 'fa fa-puzzle-piece' + }; + + function createComponent() { + var widget = document.createElement('div'); + widget.className = 'container widget widget-container'; + widget.dataset.widgetId = utils.uid(); + //widget.style.borderColor = utils.getRandomColor(); + + var dropPlaceholder = document.createElement('div'); + dropPlaceholder.className = 'drop-zone'; + widget.appendChild(dropPlaceholder); + + return widget; + } + + return widgets; +})(widgets || {}, widgetUtils); \ No newline at end of file diff --git a/demo-shell-ng2/app/widgets/widgets-input.js b/demo-shell-ng2/app/widgets/widgets-input.js new file mode 100644 index 0000000000..36faf40d87 --- /dev/null +++ b/demo-shell-ng2/app/widgets/widgets-input.js @@ -0,0 +1,71 @@ +var widgets = (function (widgets, utils) { + widgets.input = { + create: createComponent, + name: 'Input', + iconClass: 'fa fa-file-text-o', + category: 'Components', + getDragImage: getDragImage + }; + + function createComponent() { + var widget = document.createElement('input'); + widget.type = 'text'; + widget.readOnly = true; + widget.classList.add('form-control', 'widget', 'widget-input'); + widget.dataset.widgetId = utils.uid(); + return widget; + } + + /*function getDragImage() { + var container = document.createElement('div'); + container.style.width = '200px'; + + var label = document.createElement('label'); + label.textContent = "Input"; + label.style.display = 'block'; + label.style.margin = '0'; + label.style.fontSize = '12px'; + container.appendChild(label); + + var element = document.createElement('input'); + element.className = 'form-control'; + element.style.display = 'block'; + container.appendChild(element); + + return container; + }*/ + + function getDragImage() { + var element = document.createElement('input'); + element.style.width = '200px'; + element.className = 'form-control'; + return element; + } + + // TODO: move to shared library + function moveCaretToEnd(el) { + if (typeof el.selectionStart == "number") { + el.selectionStart = el.selectionEnd = el.value.length; + } else if (typeof el.createTextRange != "undefined") { + el.focus(); + var range = el.createTextRange(); + range.collapse(false); + range.select(); + } + } + + // TODO: move to shared library + function setCursorPosition (element, pos) { + if (element.setSelectionRange) { + element.setSelectionRange(pos, pos); + } else if (element.createTextRange) { + var range = element.createTextRange(); + range.collapse(true); + range.moveEnd('character', pos); + range.moveStart('character', pos); + range.select(); + } + } + + return widgets; +})(widgets || {}, widgetUtils); \ No newline at end of file diff --git a/demo-shell-ng2/app/widgets/widgets-row.js b/demo-shell-ng2/app/widgets/widgets-row.js new file mode 100644 index 0000000000..7cb116234c --- /dev/null +++ b/demo-shell-ng2/app/widgets/widgets-row.js @@ -0,0 +1,18 @@ +var widgets = (function (widgets, utils) { + widgets.row = { + create: createComponent, + name: 'row', + iconClass: 'fa fa-puzzle-piece', + category: 'Grid system' + }; + + function createComponent() { + var widget = document.createElement('div'); + widget.classList.add('row', 'widget', 'widget-row'); + widget.dataset.widgetId = utils.uid(); + widget.appendChild(utils.col('col-md-12')); + return widget; + } + + return widgets; +})(widgets || {}, widgetUtils); \ No newline at end of file diff --git a/demo-shell-ng2/index.html b/demo-shell-ng2/index.html new file mode 100644 index 0000000000..f87bee8a62 --- /dev/null +++ b/demo-shell-ng2/index.html @@ -0,0 +1,61 @@ + + + + Angular 2 QuickStart + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Loading... + + + diff --git a/demo-shell-ng2/package.json b/demo-shell-ng2/package.json new file mode 100644 index 0000000000..cf1b1228df --- /dev/null +++ b/demo-shell-ng2/package.json @@ -0,0 +1,30 @@ +{ + "name": "angular2-quickstart", + "version": "1.0.0", + "scripts": { + "start": "concurrently \"npm run tsc:w\" \"npm run lite\" ", + "tsc": "tsc", + "tsc:w": "tsc -w", + "lite": "lite-server", + "typings": "typings", + "postinstall": "typings install" + }, + "license": "ISC", + "dependencies": { + "angular2": "2.0.0-beta.13", + "bootstrap": "^3.3.6", + "es6-shim": "^0.35.0", + "font-awesome": "^4.5.0", + "jquery": "^2.2.2", + "reflect-metadata": "0.1.2", + "rxjs": "5.0.0-beta.2", + "systemjs": "0.19.25", + "zone.js": "0.6.6" + }, + "devDependencies": { + "concurrently": "^2.0.0", + "lite-server": "^2.1.0", + "typescript": "^1.8.9", + "typings": "^0.7.11" + } +} diff --git a/demo-shell-ng2/tsconfig.json b/demo-shell-ng2/tsconfig.json new file mode 100644 index 0000000000..9be71e4c6e --- /dev/null +++ b/demo-shell-ng2/tsconfig.json @@ -0,0 +1,17 @@ +{ + "compilerOptions": { + "target": "es5", + "module": "system", + "moduleResolution": "node", + "sourceMap": true, + "emitDecoratorMetadata": true, + "experimentalDecorators": true, + "removeComments": false, + "noImplicitAny": false + }, + "exclude": [ + "node_modules", + "typings/main", + "typings/main.d.ts" + ] +} diff --git a/demo-shell-ng2/typings.json b/demo-shell-ng2/typings.json new file mode 100644 index 0000000000..5ad6252b13 --- /dev/null +++ b/demo-shell-ng2/typings.json @@ -0,0 +1,7 @@ +{ + "ambientDependencies": { + "es6-shim": "github:DefinitelyTyped/DefinitelyTyped/es6-shim/es6-shim.d.ts#4de74cb527395c13ba20b438c3a7a419ad931f1c", + "jasmine": "github:DefinitelyTyped/DefinitelyTyped/jasmine/jasmine.d.ts#d594ef506d1efe2fea15f8f39099d19b39436b71", + "jquery": "registry:dt/jquery#1.10.0+20160316155526" + } +}