Added possiblity to extend the dashboard programmatically

Change-Id: Ibf2c2f7e6d51e6f5a661021f3f9f4b15c9cbefa1
diff --git a/src/app/core/index.ts b/src/app/core/index.ts
index e6f740e..4b89ac8 100644
--- a/src/app/core/index.ts
+++ b/src/app/core/index.ts
@@ -18,6 +18,7 @@
 import {ModelSetup} from './services/helpers/model-setup.helpers';
 import {xosSidePanel} from './side-panel/side-panel';
 import {XosSidePanel} from './side-panel/side-panel.service';
+import {XosComponentInjector} from './services/helpers/component-injector.helpers';
 
 export const xosCore = 'xosCore';
 
@@ -35,6 +36,7 @@
   .service('ConfigHelpers', ConfigHelpers)
   .service('ModelSetup', ModelSetup)
   .service('XosSidePanel', XosSidePanel)
+  .service('XosComponentInjector', XosComponentInjector)
   .directive('xosLinkWrapper', xosLinkWrapper)
   .component('xosHeader', xosHeader)
   .component('xosFooter', xosFooter)
diff --git a/src/app/core/nav/nav.html b/src/app/core/nav/nav.html
index 402d2fa..ce495da 100644
--- a/src/app/core/nav/nav.html
+++ b/src/app/core/nav/nav.html
@@ -44,7 +44,8 @@
             <a ng-click="vm.logout()" class="btn btn-accent btn-block btn-logout">Logout</a>
           </div>
         </div>
-        <!--<a ng-click="vm.togglePanel()" class="btn btn-success">Open XSP</a>-->
+        <a ng-click="vm.togglePanel()" class="btn btn-success">Open XSP</a>
+        <a ng-click="vm.addToDashboard()" class="btn btn-success">Add to home</a>
       </li>
     </ul>
   </nav>
diff --git a/src/app/core/nav/nav.spec.ts b/src/app/core/nav/nav.spec.ts
index f443d91..5101ba8 100644
--- a/src/app/core/nav/nav.spec.ts
+++ b/src/app/core/nav/nav.spec.ts
@@ -32,7 +32,8 @@
       .service('NavigationService', NavigationService)
       .value('AuthService', AuthMock)
       .value('StyleConfig', {})
-      .value('XosSidePanel', {});
+      .value('XosSidePanel', {})
+      .value('XosComponentInjector', {});
     angular.mock.module('xosNav');
   });
 
diff --git a/src/app/core/nav/nav.ts b/src/app/core/nav/nav.ts
index ec2d8b2..08ad48a 100644
--- a/src/app/core/nav/nav.ts
+++ b/src/app/core/nav/nav.ts
@@ -3,9 +3,10 @@
 import {IXosAuthService} from '../../datasources/rest/auth.rest';
 import {IXosStyleConfig} from '../../../index';
 import {IXosSidePanelService} from '../side-panel/side-panel.service';
+import {IXosComponentInjectorService} from '../services/helpers/component-injector.helpers';
 
 class NavCtrl {
-  static $inject = ['$scope', '$state', 'NavigationService', 'AuthService', 'StyleConfig', 'XosSidePanel'];
+  static $inject = ['$scope', '$state', 'NavigationService', 'AuthService', 'StyleConfig', 'XosSidePanel', 'XosComponentInjector'];
   public routes: IXosNavigationRoute[];
   public navSelected: string;
   public appName: string;
@@ -17,7 +18,8 @@
     private navigationService: IXosNavigationService,
     private authService: IXosAuthService,
     private StyleConfig: IXosStyleConfig,
-    private XosSidePanel: IXosSidePanelService
+    private XosSidePanel: IXosSidePanelService,
+    private XosComponentInjector: IXosComponentInjectorService
   ) {
     // NOTE we'll need to have:
     // - Base routes (defined from configuration based on BRAND)
@@ -62,6 +64,9 @@
   togglePanel() {
     this.XosSidePanel.injectComponent('xosAlert', {config: {type: 'danger'}, show: true}, 'Sample message');
   }
+  addToDashboard() {
+    this.XosComponentInjector.injectComponent('#dashboard-component-container', 'xosAlert', {config: {type: 'danger'}, show: true}, 'Sample message', false);
+  }
 
   logout() {
     this.authService.logout()
diff --git a/src/app/core/services/helpers/component-injector.helpers.spec.ts b/src/app/core/services/helpers/component-injector.helpers.spec.ts
new file mode 100644
index 0000000..0ba501f
--- /dev/null
+++ b/src/app/core/services/helpers/component-injector.helpers.spec.ts
@@ -0,0 +1,52 @@
+import * as angular from 'angular';
+import 'angular-mocks';
+import 'angular-ui-router';
+import * as $ from 'jquery';
+import {XosComponentInjector, IXosComponentInjectorService} from './component-injector.helpers';
+
+let service: IXosComponentInjectorService;
+let element, scope: angular.IRootScopeService, compile: ng.ICompileService;
+
+describe('The XosComponentInjector service', () => {
+  beforeEach(() => {
+    angular
+      .module('test', [])
+      .component('extension', {
+        template: 'extended'
+      })
+      .component('target', {
+        template: `<div id="target"></div>`
+      })
+      .service('XosComponentInjector', XosComponentInjector);
+
+    angular.mock.module('test');
+  });
+
+  beforeEach(angular.mock.inject((
+    XosComponentInjector: IXosComponentInjectorService,
+  ) => {
+    service = XosComponentInjector;
+  }));
+
+  beforeEach(angular.mock.inject(($rootScope: ng.IRootScopeService, $compile: ng.ICompileService) => {
+    scope = $rootScope;
+    compile = $compile;
+    element = $compile('<target></target>')($rootScope);
+    $rootScope.$digest();
+  }));
+
+  it('should have an InjectComponent method', () => {
+    expect(service.injectComponent).toBeDefined();
+  });
+
+  it('should have an removeInjectedComponents method', () => {
+    expect(service.removeInjectedComponents).toBeDefined();
+  });
+
+  it('should add a component to the target container', () => {
+    service.injectComponent($('#target', element), 'extension');
+    scope.$apply();
+    const extension = $('extension', element);
+    expect(extension.text()).toBe('extended');
+  });
+});
diff --git a/src/app/core/services/helpers/component-injector.helpers.ts b/src/app/core/services/helpers/component-injector.helpers.ts
new file mode 100644
index 0000000..8fd7a50
--- /dev/null
+++ b/src/app/core/services/helpers/component-injector.helpers.ts
@@ -0,0 +1,61 @@
+import * as $ from 'jquery';
+import * as _ from 'lodash';
+
+export interface IXosComponentInjectorService {
+  injectComponent(target: string | JQuery, componentName: string, attributes?: any, transclude?: string, clean?: boolean): void;
+  removeInjectedComponents(target: string | JQuery): void;
+}
+
+export class XosComponentInjector implements IXosComponentInjectorService {
+  static $inject = ['$rootScope', '$compile'];
+
+  constructor (
+    private $rootScope: ng.IRootScopeService,
+    private $compile: ng.ICompileService
+  ) {
+  }
+
+  public injectComponent(target: string | JQuery, componentName: string, attributes?: any, transclude?: string, clean?: boolean) {
+    let targetEl;
+    if (angular.isString(target)) {
+      targetEl = $(target);
+    }
+    else {
+      targetEl = target;
+    }
+
+    const componentTagName = this.camelToSnakeCase(componentName);
+    let scope = this.$rootScope.$new();
+    let attr: string = '';
+
+    if (clean) {
+      this.removeInjectedComponents(target);
+    }
+
+    if (angular.isDefined(attributes) && angular.isObject(attributes)) {
+      attr = this.stringifyAttributes(attributes);
+      scope = angular.merge(scope, attributes);
+    }
+
+    const componentTag = `<${componentTagName} ${attr}>${transclude || ''}</${componentTagName}>`;
+    const element = this.$compile(componentTag)(scope);
+
+    targetEl.append(element);
+  }
+
+  public removeInjectedComponents(target: string | JQuery) {
+    const targetEl = $(target);
+    targetEl.html('');
+  }
+
+  private stringifyAttributes(attributes: any): string {
+    return _.reduce(Object.keys(attributes), (string: string, a: string) => {
+      string += `${a}="${a}"`;
+      return string;
+    }, '');
+  }
+
+  private camelToSnakeCase(name: string): string {
+    return name.split(/(?=[A-Z])/).map(w => w.toLowerCase()).join('-');
+  };
+}
diff --git a/src/app/core/side-panel/side-panel.service.ts b/src/app/core/side-panel/side-panel.service.ts
index 06dfef9..a7471f5 100644
--- a/src/app/core/side-panel/side-panel.service.ts
+++ b/src/app/core/side-panel/side-panel.service.ts
@@ -1,5 +1,5 @@
 import * as $ from 'jquery';
-import * as _ from 'lodash';
+import {IXosComponentInjectorService} from '../services/helpers/component-injector.helpers';
 
 export interface IXosSidePanelService {
   open(): void;
@@ -8,14 +8,15 @@
 }
 
 export class XosSidePanel implements IXosSidePanelService {
-  static $inject = ['$rootScope', '$compile'];
+  static $inject = ['$rootScope', '$compile', 'XosComponentInjector'];
   public sidePanelElName = 'xos-side-panel';
   public sidePanelElClass = '.xos-side-panel';
   public sidePanelEl: JQuery;
 
   constructor (
     private $rootScope: ng.IRootScopeService,
-    private $compile: ng.ICompileService
+    private $compile: ng.ICompileService,
+    private XosComponentInjector: IXosComponentInjectorService
   ) {
     this.sidePanelEl = $(`${this.sidePanelElName} > ${this.sidePanelElClass}`);
   }
@@ -29,36 +30,7 @@
   };
 
   public injectComponent(componentName: string, attributes?: any, transclude?: string) {
-    const componentTagName = this.camelToSnakeCase(componentName);
-    let scope = this.$rootScope.$new();
-    let attr: string = '';
-
-    // NOTE add a flag to keep the loaded compoenents?
-    this.removeInjectedComponents();
-
-    if (angular.isDefined(attributes) && angular.isObject(attributes)) {
-      attr = this.stringifyAttributes(attributes);
-      scope = angular.merge(scope, attributes);
-    }
-
-    const componentTag = `<${componentTagName} ${attr}>${transclude || ''}</${componentTagName}>`;
-    const element = this.$compile(componentTag)(scope);
-    this.sidePanelEl.find('#side-panel-container').append(element);
+    this.XosComponentInjector.injectComponent('#side-panel-container', componentName, attributes, transclude, true);
     this.open();
   }
-
-  public removeInjectedComponents() {
-    this.sidePanelEl.find('#side-panel-container').html('');
-  }
-
-  private stringifyAttributes(attributes: any): string {
-    return _.reduce(Object.keys(attributes), (string: string, a: string) => {
-      string += `${a}="${a}"`;
-      return string;
-    }, '');
-  }
-
-  private camelToSnakeCase(name: string): string {
-    return name.split(/(?=[A-Z])/).map(w => w.toLowerCase()).join('-');
-  };
 }