/*****************************************************************************
 * Open MCT, Copyright (c) 2014-2018, United States Government
 * as represented by the Administrator of the National Aeronautics and Space
 * Administration. All rights reserved.
 *
 * Open MCT is licensed under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 * http://www.apache.org/licenses/LICENSE-2.0.
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
 * License for the specific language governing permissions and limitations
 * under the License.
 *
 * Open MCT includes source code licensed under additional open source
 * licenses. See the Open Source Licenses file (LICENSES.md) included with
 * this source code distribution or the Licensing information page available
 * at runtime from the About dialog for additional information.
 *****************************************************************************/

/**
 * MCTRepresentationSpec. Created by vwoeltje on 11/6/14.
 */
define(
    ["../src/MCTRepresentation"],
    function (MCTRepresentation) {

        var JQLITE_FUNCTIONS = ["on", "off", "attr", "removeAttr"],
            LOG_FUNCTIONS = ["error", "warn", "info", "debug"],
            DOMAIN_OBJECT_METHODS = ["getId", "getModel", "getCapability", "hasCapability", "useCapability"];

        describe("The mct-representation directive", function () {
            var testRepresentations,
                testViews,
                testUrls,
                mockRepresenters,
                mockMutationCapability,
                mockQ,
                mockLinker,
                mockLog,
                mockChangeTemplate,
                mockScope,
                mockElement,
                mockDomainObject,
                testModel,
                mctRepresentation;

            function mockPromise(value) {
                return (value && value.then) ? value : {
                    then: function (callback) {
                        return mockPromise(callback(value));
                    }
                };
            }

            function fireWatch(expr, value) {
                mockScope.$watch.calls.all().forEach(function (call) {
                    if (call.args[0] === expr) {
                        call.args[1](value);
                    }
                });
            }

            beforeEach(function () {
                testUrls = {};

                testRepresentations = [
                    {
                        key: "abc",
                        bundle: { path: "a", resources: "b" },
                        templateUrl: "c/template.html"
                    },
                    {
                        key: "def",
                        bundle: { path: "d", resources: "e" },
                        templateUrl: "f/template.html",
                        uses: ["testCapability", "otherTestCapability"]
                    }
                ];

                testViews = [
                    {
                        key: "uvw",
                        bundle: { path: "u", resources: "v" },
                        templateUrl: "w/template.html",
                        gestures: ["testGesture", "otherTestGesture"]
                    },
                    {
                        key: "xyz",
                        bundle: { path: "x", resources: "y" },
                        templateUrl: "z/template.html"
                    }
                ];

                testModel = { someKey: "some value" };

                testUrls = {};
                testViews.concat(testRepresentations).forEach(function (t, i) {
                    testUrls[t.key] = "some URL " + String(i);
                });

                mockRepresenters = ["A", "B"].map(function (name) {
                    var constructor = jasmine.createSpy("Representer" + name),
                        representer = jasmine.createSpyObj(
                            "representer" + name,
                            ["represent", "destroy"]
                        );
                    constructor.and.returnValue(representer);
                    return constructor;
                });

                mockQ = { when: mockPromise };
                mockLinker = jasmine.createSpyObj(
                    'templateLinker',
                    ['link', 'getPath']
                );
                mockChangeTemplate = jasmine.createSpy('changeTemplate');
                mockLog = jasmine.createSpyObj("$log", LOG_FUNCTIONS);

                mockMutationCapability =
                    jasmine.createSpyObj("mutation", ["listen"]);

                mockScope = jasmine.createSpyObj("scope", ["$watch", "$on"]);
                mockElement = jasmine.createSpyObj("element", JQLITE_FUNCTIONS);
                mockDomainObject = jasmine.createSpyObj("domainObject", DOMAIN_OBJECT_METHODS);

                mockDomainObject.getModel.and.returnValue(testModel);
                mockLinker.link.and.returnValue(mockChangeTemplate);
                mockLinker.getPath.and.callFake(function (ext) {
                    return testUrls[ext.key];
                });

                mockDomainObject.getCapability.and.callFake(function (c) {
                    return c === 'mutation' && mockMutationCapability;
                });

                mctRepresentation = new MCTRepresentation(
                    testRepresentations,
                    testViews,
                    mockRepresenters,
                    mockQ,
                    mockLinker,
                    mockLog
                );
                mctRepresentation.link(mockScope, mockElement);
            });

            it("is restricted to elements", function () {
                expect(mctRepresentation.restrict).toEqual("E");
            });

            it("exposes templates via the templateLinker", function () {
                expect(mockLinker.link)
                    .toHaveBeenCalledWith(mockScope, mockElement);
            });

            it("watches scope when linked", function () {
                expect(mockScope.$watch).toHaveBeenCalledWith(
                    "key",
                    jasmine.any(Function)
                );
                expect(mockScope.$watch).toHaveBeenCalledWith(
                    "domainObject",
                    jasmine.any(Function)
                );
            });

            it("recognizes keys for representations", function () {
                mockScope.key = "abc";
                mockScope.domainObject = mockDomainObject;

                // Trigger the watch
                fireWatch('key', mockScope.key);
                fireWatch('domainObject', mockDomainObject);

                expect(mockChangeTemplate)
                    .toHaveBeenCalledWith(testRepresentations[0]);
            });

            it("recognizes keys for views", function () {
                mockScope.key = "xyz";
                mockScope.domainObject = mockDomainObject;

                // Trigger the watches
                fireWatch('key', mockScope.key);
                fireWatch('domainObject', mockDomainObject);

                expect(mockChangeTemplate)
                    .toHaveBeenCalledWith(testViews[1]);
            });

            it("does not load templates until there is an object", function () {
                mockScope.key = "xyz";

                // Trigger the watch
                fireWatch('key', mockScope.key);

                expect(mockChangeTemplate)
                    .not.toHaveBeenCalledWith(jasmine.any(Object));

                mockScope.domainObject = mockDomainObject;
                fireWatch('domainObject', mockDomainObject);

                expect(mockChangeTemplate)
                    .toHaveBeenCalledWith(jasmine.any(Object));
            });

            it("loads declared capabilities", function () {
                mockScope.key = "def";
                mockScope.domainObject = mockDomainObject;

                // Trigger the watch
                mockScope.$watch.calls.all()[0].args[1]();

                expect(mockDomainObject.useCapability)
                    .toHaveBeenCalledWith("testCapability");
                expect(mockDomainObject.useCapability)
                    .toHaveBeenCalledWith("otherTestCapability");
            });

            it("logs when no representation is available for a key", function () {
                mockScope.key = "someUnknownThing";

                // Verify precondition
                expect(mockLog.warn).not.toHaveBeenCalled();

                // Trigger the watch
                mockScope.$watch.calls.all()[0].args[1]();

                // Should have gotten a warning - that's an unknown key
                expect(mockLog.warn).toHaveBeenCalled();
            });

            it("clears out obsolete properties from scope", function () {
                mockScope.key = "def";
                mockScope.domainObject = mockDomainObject;
                mockDomainObject.useCapability.and.returnValue("some value");

                // Trigger the watch
                mockScope.$watch.calls.all()[0].args[1]();
                expect(mockScope.testCapability).toBeDefined();

                // Change the view
                mockScope.key = "xyz";

                // Trigger the watch again; should clear capability from scope
                mockScope.$watch.calls.all()[0].args[1]();
                expect(mockScope.testCapability).toBeUndefined();
            });

            describe("when a domain object has been observed", function () {
                var mockContext,
                    mockContext2,
                    mockLink,
                    mockParent;

                beforeEach(function () {
                    mockContext = jasmine.createSpyObj('context', ['getPath']);
                    mockContext2 = jasmine.createSpyObj('context', ['getPath']);
                    mockLink = jasmine.createSpyObj(
                        'linkedObject',
                        DOMAIN_OBJECT_METHODS
                    );
                    mockParent = jasmine.createSpyObj(
                        'parentObject',
                        DOMAIN_OBJECT_METHODS
                    );

                    mockDomainObject.getCapability.and.callFake(function (c) {
                        return {
                            context: mockContext,
                            mutation: mockMutationCapability
                        }[c];
                    });
                    mockLink.getCapability.and.callFake(function (c) {
                        return {
                            context: mockContext2,
                            mutation: mockMutationCapability
                        }[c];
                    });
                    mockDomainObject.hasCapability.and.callFake(function (c) {
                        return c === 'context';
                    });
                    mockLink.hasCapability.and.callFake(function (c) {
                        return c === 'context';
                    });
                    mockLink.getModel.and.returnValue({});

                    mockContext.getPath.and.returnValue([mockDomainObject]);
                    mockContext2.getPath.and.returnValue([mockParent, mockLink]);

                    mockLink.getId.and.returnValue('test-id');
                    mockDomainObject.getId.and.returnValue('test-id');

                    mockParent.getId.and.returnValue('parent-id');

                    mockScope.key = "abc";
                    mockScope.domainObject = mockDomainObject;

                    mockScope.$watch.calls.all()[0].args[1]();
                });

                it("listens for mutation of that object", function () {
                    expect(mockMutationCapability.listen)
                        .toHaveBeenCalledWith(jasmine.any(Function));
                });

                it("detects subsequent changes among linked instances", function () {
                    var callCount = mockChangeTemplate.calls.count();

                    mockScope.domainObject = mockLink;
                    mockScope.$watch.calls.all()[0].args[1]();

                    expect(mockChangeTemplate.calls.count())
                        .toEqual(callCount + 1);
                });

                it("does not trigger excess template changes for same instances", function () {
                    var callCount = mockChangeTemplate.calls.count();
                    mockScope.$watch.calls.all()[0].args[1]();
                    expect(mockChangeTemplate.calls.count()).toEqual(callCount);
                });

            });


        });
    }
);