From e48195d69ac08974984c8bae85b7f41a77a67adf Mon Sep 17 00:00:00 2001
From: Jon Reid
Date: Mon, 4 Jun 2012 16:37:18 -0700
Subject: Timeline: Fixes to timeline to handle document switching & breadcrumb
interactions under new DOM Architecture.
---
.../Timeline/TimelinePanel.reel/TimelinePanel.js | 3055 ++++++++++----------
1 file changed, 1535 insertions(+), 1520 deletions(-)
(limited to 'js/panels/Timeline/TimelinePanel.reel/TimelinePanel.js')
diff --git a/js/panels/Timeline/TimelinePanel.reel/TimelinePanel.js b/js/panels/Timeline/TimelinePanel.reel/TimelinePanel.js
index 6e9513f2..a5737334 100644
--- a/js/panels/Timeline/TimelinePanel.reel/TimelinePanel.js
+++ b/js/panels/Timeline/TimelinePanel.reel/TimelinePanel.js
@@ -1,1521 +1,1536 @@
-/*
- This file contains proprietary software owned by Motorola Mobility, Inc.
- No rights, expressed or implied, whatsoever to this software are provided by Motorola Mobility, Inc. hereunder.
- (c) Copyright 2011 Motorola Mobility, Inc. All Rights Reserved.
- */
-
-var Montage = require("montage/core/core").Montage;
-var Component = require("montage/ui/component").Component;
-var nj = require("js/lib/NJUtils").NJUtils;
-
-var TimelinePanel = exports.TimelinePanel = Montage.create(Component, {
-
- hasTemplate:{
- value:true
- },
-
- /* === BEGIN: Models === */
- _currentDocument: {
- value : null
- },
-
- currentDocument : {
- get : function() {
- return this._currentDocument;
- },
- set : function(value) {
- if (value === this._currentDocument) {
- return;
- }
-
- if(!this._currentDocument && value.currentView === "design") {
- this.enablePanel(true);
- }
-
- this._currentDocument = value;
-
- if(!value) {
- this.enablePanel(false);
- } else if(this._currentDocument.currentView === "design") {
- this._boolCacheArrays = false;
- this.clearTimelinePanel();
- this._boolCacheArrays = true;
-
- // Rebind the document events for the new document context
- this._bindDocumentEvents();
-
- // TODO: Fix the init function so that it can be called here instead of when container changes
- // this.initTimelineForDocument();
- }
- }
- },
-
- _arrLayers:{
- value:[]
- },
-
- arrLayers:{
- serializable:true,
- get:function () {
- return this._arrLayers;
- },
- set:function (newVal) {
- this._arrLayers = newVal;
- this.needsDraw = true;
- this.cacheTimeline();
- }
- },
-
- _temparrLayers:{
- value:[]
- },
-
- temparrLayers:{
- get:function () {
- return this._temparrLayers;
- },
- set:function (newVal) {
- this._temparrLayers = newVal;
- }
- },
-
-
- _layerRepetition:{
- value:null
- },
-
- layerRepetition:{
- get:function () {
- return this._layerRepetition;
- },
- set:function (newVal) {
- this._layerRepetition = newVal;
- }
- },
-
- // Set to false to skip array caching array sets in current document
- _boolCacheArrays:{
- value:true
- },
-
- _currentLayerNumber:{
- value:0
- },
-
- currentLayerNumber:{
- get:function () {
- return this._currentLayerNumber;
- },
- set:function (newVal) {
- if (newVal !== this._currentLayerNumber) {
- this._currentLayerNumber = newVal;
- this.cacheTimeline();
- }
- }
- },
-
- _currentLayerSelected:{
- value: false
- },
- currentLayerSelected:{
- get:function () {
- return this._currentLayerSelected;
- },
- set:function (newVal) {
- this._currentLayerSelected = newVal;
- this.cacheTimeline();
- }
- },
-
- _selectedLayerID:{
- value:false
- },
- selectedLayerID:{
- get:function () {
- return this._selectedLayerID;
- },
- set:function (newVal) {
- if (newVal === false) {
- // We are clearing the timeline, so just set the value and return.
- this._selectedLayerID = newVal;
- return;
- }
- if (newVal !== this._selectedLayerID) {
- var selectIndex = this.getLayerIndexByID(newVal);
- this._selectedLayerID = newVal;
- this._captureSelection = true;
- if (this.currentLayerSelected !== false) {
- this.selectLayer(selectIndex, false);
- }
- if (this.currentLayersSelected !== false) {
- this.selectLayers(this.currentLayersSelected);
- }
- if ((this.currentLayersSelected === false) && (this.currentLayerSelected === false)) {
- this.selectLayers([]);
- }
-
- }
- }
- },
-
- _currentLayersSelected:{
- value:[]
- },
- currentLayersSelected:{
- get:function () {
- return this._currentLayersSelected;
- },
- set:function (newVal) {
- this._currentLayersSelected = newVal;
- this.cacheTimeline();
- }
- },
-
- _currentSelectedContainer: {
- value: null
- },
- currentSelectedContainer: {
- get: function() {
- return this._currentSelectedContainer;
- },
- set: function(newVal) {
- if(this._currentSelectedContainer !== newVal) {
- this._currentSelectedContainer = newVal;
-
- this._boolCacheArrays = false;
- this.clearTimelinePanel();
- this._boolCacheArrays = true;
-
- // TODO: Fix the function so that we can remove this call here.
- this.initTimelineForDocument();
- }
- }
- },
-
- _millisecondsOffset:{
- value:1000
- },
-
- millisecondsOffset:{
- get:function () {
- return this._millisecondsOffset;
- },
- set:function (newVal) {
- if (newVal !== this._millisecondsOffset) {
- this._millisecondsOffset= newVal;
- this.drawTimeMarkers();
- NJevent('tlZoomSlider',this);
- }
- }
- },
-
- _masterDuration:{
- value:0
- },
-
- masterDuration:{
- serializable:true,
- get:function () {
- return this._masterDuration;
- },
- set:function (val) {
- this._masterDuration = val;
- this.timebar.style.width = (this._masterDuration / 12) + "px";
- }
- },
-
- _trackRepetition:{
- value:null
- },
-
- trackRepetition:{
- get:function () {
- return this._trackRepetition;
- },
- set:function (newVal) {
- this._trackRepetition = newVal;
- }
- },
-
- _selectedKeyframes:{
- value:[]
- },
-
- selectedKeyframes:{
- serializable:true,
- get:function () {
- return this._selectedKeyframes;
- },
- set:function (newVal) {
- this._selectedKeyframes = newVal;
- }
- },
-
- _selectedTweens:{
- value:[]
- },
-
- selectedTweens:{
- serializable:true,
- get:function () {
- return this._selectedTweens;
- },
- set:function (newVal) {
- this._selectedTweens = newVal;
- }
- },
-
- _breadCrumbContainer:{
- value:null
- },
-
- breadCrumbContainer:{
- get:function () {
- return this._breadCrumbContainer;
- },
- set:function (value) {
- if (this._breadCrumbContainer !== value) {
- this._breadCrumbContainer = value;
- }
- }
- },
-
- _isLayer:{
- value:false
- },
-
- _firstTimeLoaded:{
- value:true
- },
-
- _captureSelection:{
- value:false
- },
-
- _openDoc:{
- value:false
- },
-
- timeMarkerHolder:{
- value:null
- },
- _dragAndDropHelper : {
- value: false
- },
- _dragAndDropHelperCoords: {
- value: false
- },
- _dragAndDropHelperOffset : {
- value: false
- },
- _dragLayerID : {
- value: null
- },
-
- layersDragged:{
- value:[],
- writable:true
- },
-
- dragLayerID : {
- get: function() {
- return this._dragLayerID;
- },
- set: function(newVal) {
- if (newVal !== this._dragLayerID) {
- this._dragLayerID = newVal;
- }
- }
- },
- _dropLayerID : {
- value: null
- },
- dropLayerID : {
- get: function() {
- return this._dropLayerID;
- },
- set: function(newVal) {
- if (newVal !== this._dropLayerID) {
- this._dropLayerID = newVal;
-
- var dragLayerIndex = this.getLayerIndexByID(this.dragLayerID),
- dropLayerIndex = this.getLayerIndexByID(this.dropLayerID),
- dragLayer = this.arrLayers[dragLayerIndex];
- this.layersDragged.push(dragLayer);
- this._layerDroppedInPlace = this.arrLayers[dropLayerIndex];
-
- this.arrLayers.splice(dragLayerIndex, 1);
- this.arrLayers.splice(dropLayerIndex, 0, dragLayer);
- this.cacheTimeline();
-
- // Clear for future DnD
- this._dropLayerID = null;
- this._dragLayerID = null;
-
- // Sometimes, just to be fun, the drop and dragend events don't fire.
- // So just in case, set the draw routine to delete the helper.
- this._deleteHelper = true;
- this.needsDraw = true;
- }
- }
- },
- _appendHelper: {
- value: false
- },
- _deleteHelper: {
- value: false
- },
- _scrollTracks: {
- value: false
- },
- useAbsolutePosition:{
- value:true
- },
- /* === END: Models === */
- /* === BEGIN: Draw cycle === */
- prepareForDraw:{
- value:function () {
- this.initTimeline();
-
- // Bind drag and drop event handlers
- this.container_layers.addEventListener("dragstart", this.handleLayerDragStart.bind(this), false);
- this.container_layers.addEventListener("dragend", this.handleLayerDragEnd.bind(this), false);
- this.container_layers.addEventListener("dragover", this.handleLayerDragover.bind(this), false);
- this.container_layers.addEventListener("drop", this.handleLayerDrop.bind(this), false);
-
- // Bind the handlers for the config menu
- this.checkable_animated.addEventListener("click", this.handleAnimatedClick.bind(this), false);
- this.checkable_relative.addEventListener("click", this.handleRelativeClick.bind(this), false);
- this.checkable_absolute.addEventListener("click", this.handleAbsoluteClick.bind(this), false);
- this.tl_configbutton.addEventListener("click", this.handleConfigButtonClick.bind(this), false);
- document.addEventListener("click", this.handleDocumentClick.bind(this), false);
-
- }
- },
-
- willDraw:{
- value:function () {
- if (this._isLayer) {
- this._isLayer = false;
- }
- }
- },
-
- draw: {
- value: function() {
- // Drag and Drop:
- // Do we have a helper to append?
- if (this._appendHelper === true) {
- this.container_layers.appendChild(this._dragAndDropHelper);
- this._appendHelper = false;
- }
- // Do we need to move the helper?
- if (this._dragAndDropHelperCoords !== false) {
- if (this._dragAndDropHelper !== null) {
- this._dragAndDropHelper.style.top = this._dragAndDropHelperCoords;
- }
- this._dragAndDropHelperCoords = false;
- }
- // Do we need to scroll the tracks?
- if (this._scrollTracks !== false) {
- this.layout_tracks.scrollTop = this._scrollTracks;
- this._scrollTracks = false;
- }
- // Do we have a helper to delete?
- if (this._deleteHelper === true) {
- if (this._dragAndDropHelper === null) {
- // Problem....maybe a helper didn't get appended, or maybe it didn't get stored.
- // Try and recover the helper so we can delete it.
- var myHelper = this.container_layers.querySelector(".timeline-dnd-helper");
- if (myHelper != null) {
- this._dragAndDropHelper = myHelper;
- }
- }
- if (this._dragAndDropHelper !== null) {
- // We need to delete the helper. Can we delete it from container_layers?
- if (this._dragAndDropHelper && this._dragAndDropHelper.parentNode === this.container_layers) {
- this.container_layers.removeChild(this._dragAndDropHelper);
- this._dragAndDropHelper = null;
- this._deleteHelper = false;
- }
- }
- this.application.ninja.elementMediator.reArrangeDOM(this.layersDragged , this._layerDroppedInPlace);
- this.layersDragged =[];
- }
- }
- },
- /* === END: Draw cycle === */
- /* === BEGIN: Controllers === */
- // Create an empty layer template object with minimal defaults and return it for use
- createLayerTemplate:{
- value:function () {
- var returnObj = {};
-
- returnObj.layerData = {};
- returnObj.layerData.layerName = null;
- returnObj.layerData.layerID = null;
- returnObj.layerData.isMainCollapsed = true;
- returnObj.layerData.isPositionCollapsed = true;
- returnObj.layerData.isTransformCollapsed = true;
- returnObj.layerData.isStyleCollapsed = true;
- returnObj.layerData.arrLayerStyles = [];
- returnObj.layerData.arrLayerStyles = [];
- returnObj.layerData.elementsList = [];
- returnObj.layerData.deleted = false;
- returnObj.layerData.isSelected = false;
- returnObj.layerData.layerPosition = null;
- returnObj.layerData.created = false;
- returnObj.layerData.isTrackAnimated = false;
- returnObj.layerData.currentKeyframeRule = null;
- returnObj.layerData.trackPosition = 0;
- returnObj.layerData.arrStyleTracks = [];
- returnObj.layerData.tweens = [];
- returnObj.layerData.layerTag = "";
- returnObj.layerData.isVisible = true;
- returnObj.layerData.docUUID = this.application.ninja.currentDocument._uuid;
- returnObj.layerData.isTrackAnimated = false;
- returnObj.layerData.triggerBinding = false;
- returnObj.parentElementUUID = null;
- returnObj.parentElement = null;
-
- return returnObj;
- }
- },
-
- // cache Timeline data in currentDocument.
- cacheTimeline: {
- value: function() {
- // Store the timeline data in currentDocument...
- if (this._boolCacheArrays) {
- // ... but only if we're supposed to.
- this.application.ninja.currentDocument.tlArrLayers = this.arrLayers;
- this.application.ninja.currentDocument.tlCurrentSelectedContainer = this.application.ninja.currentSelectedContainer;
- this.application.ninja.currentDocument.tllayerNumber = this.currentLayerNumber;
- this.application.ninja.currentDocument.tlCurrentLayerSelected = this.currentLayerSelected;
- this.application.ninja.currentDocument.tlCurrentLayersSelected = this.currentLayersSelected;
- }
- }
- },
- // Initialize Timeline cache in currentDocument.
- initTimelineCache: {
- value: function() {
- // Initialize the currentDocument for a new set of timeline data.
- this.application.ninja.currentDocument.isTimelineInitialized = true;
- this.application.ninja.currentDocument.tlArrLayers = [];
- this.application.ninja.currentDocument.tlCurrentSelectedContainer = this.application.ninja.currentSelectedContainer;
- this.application.ninja.currentDocument.tllayerNumber = this.currentLayerNumber;
- this.application.ninja.currentDocument.tlCurrentLayerSelected = false;
- this.application.ninja.currentDocument.tlCurrentLayersSelected = false;
- }
- },
-
- // Create an array of style objects for an element, for use
- // in creating a new layer
- createLayerStyles : {
- value: function(ptrElement) {
- // TODO: Create logic to loop through
- // CSS properties on element and build
- // array of layer styles for return.
- // Right now this method just returns an array of one bogus style.
-
- var returnArray = [],
- newStyle = {},
- styleID = "1@0"; // format: layerID + "@" + style counter
-
- /* Example new style
- newStyle.styleID = styleID;
- newStyle.whichView = "propval"; // Which view do we want to show, usually property/value view (see Style)
- newStyle.editorProperty = "top"; // the style property
- newStyle.editorValue = 0; // The current value
- newStyle.ruleTweener = false;
- newStyle.isSelected = false;
-
- returnArray.push(newStyle);
- */
-
- return returnArray;
-
- }
- },
-
- // Create an array of style track objects for an element, for use
- // in creating a new layer
- createStyleTracks : {
- value: function(ptrElement) {
- // TODO: Create logic to loop through
- // CSS properties on element and build
- // array of layer styles for return.
- // Right now this method just returns an array of one bogus style.
-
- var returnArray = [];
-
- return returnArray;
-
- }
- },
-
- // Bind all document-specific events (pass in true to unbind)
- _bindDocumentEvents : {
- value: function(boolUnbind) {
- var arrEvents = ["deleteLayerClick",
- "newLayer",
- "deleteLayer",
- "elementAdded",
- "elementsRemoved",
- "elementReplaced",
- "selectionChange"],
- i,
- arrEventsLength = arrEvents.length;
-
- if (boolUnbind) {
- for (i = 0; i < arrEventsLength; i++) {
- this.eventManager.removeEventListener(arrEvents[i], this, false);
- }
- } else {
- for (i = 0; i < arrEventsLength; i++) {
- this.eventManager.addEventListener(arrEvents[i], this, false);
- }
- }
- }
- },
-
- // Initialize the timeline, runs only once when the timeline component is first loaded
- initTimeline:{
- value:function () {
-
- // Get some selectors
- this.layout_tracks = this.element.querySelector(".layout-tracks");
- this.layout_markers = this.element.querySelector(".layout_markers");
-
- // Add some event handlers
- this.timeline_leftpane.addEventListener("mousedown", this.timelineLeftPaneMousedown.bind(this), false);
- this.timeline_leftpane.addEventListener("mouseup", this.timelineLeftPaneMouseup.bind(this), false);
- this.layout_tracks.addEventListener("scroll", this.updateLayerScroll.bind(this), false);
- this.user_layers.addEventListener("scroll", this.updateLayerScroll.bind(this), false);
- this.end_hottext.addEventListener("changing", this.updateTrackContainerWidth.bind(this), false);
- this.playhead.addEventListener("mousedown", this.startPlayheadTracking.bind(this), false);
- this.playhead.addEventListener("mouseup", this.stopPlayheadTracking.bind(this), false);
- this.time_markers.addEventListener("click", this.updatePlayhead.bind(this), false);
-
- // Bind some bindings
- Object.defineBinding(this, "currentSelectedContainer", {
- boundObject:this.application.ninja,
- boundObjectPropertyPath:"currentSelectedContainer",
- oneway:true
- });
-
- // Start the panel out in disabled mode by default
- // (Will be switched on later, if appropriate).
- this.enablePanel(false);
-
- }
- },
-
- // Initialize the timeline for a document.
- // Called when a document is opened (new or existing), or when documents are switched.
- initTimelineForDocument:{
- value:function () {
-
-
- var myIndex,
- boolAlreadyInitialized = false;
- this.drawTimeMarkers();
- // Document switching
- // Check to see if we have saved timeline information in the currentDocument.
- if ((typeof(this.application.ninja.currentDocument.isTimelineInitialized) === "undefined")) {
- //console.log('TimelinePanel.initTimelineForDocument: new Document');
- // No, we have no information stored.
- // This could mean we are creating a new file, OR are opening an existing file.
-
- // First, initialize the caches.
- this.initTimelineCache();
- this.temparrLayers = [];
-
- // That's all we need to do for a brand new file.
- // But what if we're opening an existing document?
- if (!this.application.ninja.documentController.creatingNewFile && this.application.ninja.currentDocument.currentView !== "code") {
- // Opening an existing document. If it has DOM elements we need to restore their timeline info
- if (this.application.ninja.currentDocument.model.documentRoot.children[0]) {
- // Yes, it has DOM elements. Loop through them and create a new object for each.
- for (myIndex = 0; this.application.ninja.currentDocument.model.documentRoot.children[myIndex]; myIndex++) {
- this._openDoc = true;
- this.restoreLayer(this.application.ninja.currentDocument.model.documentRoot.children[myIndex]);
- }
- }
- }
-
- // Draw the repetition.
- this.arrLayers = this.temparrLayers;
- this.currentLayerNumber = this.arrLayers.length;
- boolAlreadyInitialized = true;
-
- } else if (this.application.ninja.currentDocument.setLevel) {
- //console.log('TimelinePanel.initTimelineForDocument: breadCrumbClick');
- // Information stored, but we're moving up or down in the breadcrumb.
- // Get the current selection and restore timeline info for its children.
- var parentNode = this.application.ninja.currentSelectedContainer,
- storedCurrentLayerNumber = this.application.ninja.currentDocument.tllayerNumber;
- this.temparrLayers = [];
-
- for (myIndex = 0; parentNode.children[myIndex]; myIndex++) {
- this._openDoc = true;
- this.restoreLayer(parentNode.children[myIndex]);
-
- }
- // Draw the repetition.
- this.arrLayers = this.temparrLayers;
- this.currentLayerNumber = storedCurrentLayerNumber;
- boolAlreadyInitialized = true;
- this.application.ninja.currentDocument.setLevel = false;
-
-
- } else {
- //console.log('TimelinePanel.initTimelineForDocument: else fallback');
- // we do have information stored. Use it.
- var i = 0,
- tlArrLayersLength = this.application.ninja.currentDocument.tlArrLayers.length;
-
- // We're reading from the cache, not writing to it.
- this._boolCacheArrays = false;
- for (i = 0; i < tlArrLayersLength; i++) {
- if (this.application.ninja.currentDocument.tlArrLayers[i].layerData.isSelected === true) {
- this.application.ninja.currentDocument.tlArrLayers[i].layerData._isFirstDraw = true;
- } else {
- this.application.ninja.currentDocument.tlArrLayers[i].layerData._isFirstDraw = false;
- }
- }
- this.arrLayers = this.application.ninja.currentDocument.tlArrLayers;
- this.currentLayerNumber = this.application.ninja.currentDocument.tllayerNumber;
- this.currentLayerSelected = this.application.ninja.currentDocument.tlCurrentLayerSelected;
- this.currentLayersSelected = this.application.ninja.currentDocument.tlCurrentLayersSelected;
-
-
- //debugger;
- if (typeof(this.application.ninja.currentDocument.tlCurrentSelectedContainer) !== "undefined") {
-// this.application.ninja.currentSelectedContainer=this.application.ninja.currentDocument.tlCurrentSelectedContainer;
- }
-
- // Are we only showing animated layers?
- if (this.application.ninja.currentDocument.boolShowOnlyAnimated) {
- // Fake a click.
- var evt = document.createEvent("MouseEvents");
- evt.initMouseEvent("click");
- this.checkable_animated.dispatchEvent(evt);
- }
-
- // Ok, done reading from the cache.
- this._boolCacheArrays = true;
-
- // Reset master duration
- this.resetMasterDuration();
- }
- }
- },
-
- // Clear the currently-displayed document (and its events) from the timeline.
- clearTimelinePanel:{
- value:function () {
- // Remove events
- this._bindDocumentEvents(true);
-
- // Remove every event listener for every selected tween in the timeline
- this.deselectTweens();
-
- // Reset visual appearance
- // Todo: Maybe this should be stored per document, so we can persist between document switch?
- this.application.ninja.timeline.playhead.style.left = "-2px";
- this.application.ninja.timeline.playheadmarker.style.left = "0px";
- this.application.ninja.timeline.updateTimeText(0.00);
- this.timebar.style.width = "0px";
- this.checkable_animated.classList.remove("checked");
- this.currentLayerNumber = 0;
- this.currentLayerSelected = false;
- this.currentLayersSelected = false;
- this.selectedKeyframes = [];
- this.selectedTweens = [];
- this._captureSelection = false;
- this._openDoc = false;
- this.end_hottext.value = 25;
- this.updateTrackContainerWidth();
- this.masterDuration = 0;
- // Clear the repetitions
- if (this.arrLayers.length > 0) {
- this.arrLayers = [];
- this.arrLayers.length = 0;
- }
- }
- },
-
- handleDocumentChange:{
- value:function () {
-
- }
- },
-
- updateTrackContainerWidth:{
- value:function () {
- this.container_tracks.style.width = (this.end_hottext.value * 80) + "px";
- this.master_track.style.width = (this.end_hottext.value * 80) + "px";
- this.time_markers.style.width = (this.end_hottext.value * 80) + "px";
- if (this.timeMarkerHolder) {
- this.time_markers.removeChild(this.timeMarkerHolder);
- }
- this.drawTimeMarkers();
- }
- },
-
- updateLayerScroll:{
- value:function () {
- this.user_layers.scrollTop = this.layout_tracks.scrollTop;
- this.layout_markers.scrollLeft = this.layout_tracks.scrollLeft;
- this.playheadmarker.style.top = this.layout_tracks.scrollTop + "px";
- }
- },
-
- startPlayheadTracking:{
- value:function () {
- this.time_markers.onmousemove = this.updatePlayhead.bind(this);
- }
- },
-
- stopPlayheadTracking:{
- value:function () {
- this.time_markers.onmousemove = null;
- }
- },
-
- updatePlayhead:{
- value:function (event) {
- var clickedPosition = event.target.offsetLeft + event.offsetX;
- this.playhead.style.left = (clickedPosition - 2) + "px";
- this.playheadmarker.style.left = clickedPosition + "px";
- var currentMillisecPerPixel = Math.floor(this.millisecondsOffset / 80);
- var currentMillisec = currentMillisecPerPixel * clickedPosition;
- this.updateTimeText(currentMillisec);
- }
- },
-
- handleSelectionChange:{
- value:function () {
- var layerIndex,
- i = 0,
- j = 0,
- arrLayersLength = this.arrLayers.length,
- intNumSelected = this.application.ninja.selectedElements.length,
- checkIndex = 0;
-
- this.deselectTweens();
- //console.log("TimelinePanel.handleSelectionChange")
- if (intNumSelected === 0) {
- this.selectLayers([]);
- this.currentLayerSelected = false;
- this.currentLayersSelected = false;
- }
-
- if (intNumSelected === 1) {
- this.currentLayersSelected = false;
- if (this.application.ninja.selectedElements[0]) {
- checkIndex = this.application.ninja.selectedElements[0].uuid;
- for (i = 0; i < arrLayersLength; i++) {
- var currIndex = this.arrLayers[i].layerData.elementsList[0].uuid,
- layerID = this.arrLayers[i].layerData.layerID,
- layerIndex = 0;
- if (checkIndex === currIndex) {
- layerIndex = this.getLayerIndexByID(layerID);
- this._captureSelection = false;
- this.selectLayer(layerIndex);
- this._captureSelection = true;
- }
- }
- }
- }
-
- if (intNumSelected > 1) {
- // Build an array of indexes of selected layers to give to the selectLayers method
- var arrSelectedIndexes = [];
- this.currentLayerSelected = false;
- for (i = 0; i < intNumSelected; i++) {
- var currentCheck = this.application.ninja.selectedElements[i].uuid;
- //console.log("checking ", currentCheck);
- for (j = 0; j < arrLayersLength; j++) {
- //console.log(".......... ", this.arrLayers[j].layerData.elementsList[0].uuid)
- if (currentCheck === this.arrLayers[j].layerData.elementsList[0].uuid) {
- //console.log("...............Yes!")
- arrSelectedIndexes.push(j);
- }
- }
- }
- this.selectLayers(arrSelectedIndexes);
- }
- }
- },
-
-
-
- selectLayers:{
- value:function (arrSelectedIndexes) {
-
- var i = 0,
- arrLayersLength = this.arrLayers.length,
- arrSelectedIndexesLength = arrSelectedIndexes.length,
- userSelection = false;
-
- //console.log(arrSelectedIndexes);
-
-
- if (this.selectedKeyframes) {
- this.deselectTweens();
- }
-
- for (i = 0; i < arrLayersLength; i++) {
- this.arrLayers[i].layerData.isSelected = false;
- this.triggerLayerBinding(i);
- }
-
- this.currentLayersSelected = false;
- if (arrSelectedIndexesLength > 0) {
- this.currentLayersSelected = [];
- }
-
-
- for (i = 0; i < arrLayersLength; i++) {
- if (arrSelectedIndexes.indexOf(i) > -1) {
- this.arrLayers[i].layerData.isSelected = true;
- this.arrLayers[i].isSelected = true;
- this.triggerLayerBinding(i);
- this.currentLayersSelected.push(i);
- }
- }
-
- this.layerRepetition.selectedIndexes = arrSelectedIndexes;
-
- // TODO: Set up for user selection.
- if (userSelection) {
- if (this._captureSelection) {
-
- if (this.currentLayerSelected.layerData.elementsList.length >= 1) {
- this.application.ninja.selectionController.selectElements(this.currentLayerSelected.layerData.elementsList);
- } else {
- this.application.ninja.selectionController.executeSelectElement();
- }
-
- }
- this._captureSelection = true;
- }
-
- // Finally, reset the master duration.
- this.resetMasterDuration();
- }
- },
-
- deselectTweens:{
- value:function () {
- for (var i = 0; i < this.selectedTweens.length; i++) {
- this.selectedTweens[i].deselectTween();
- }
- this.selectedTweens = null;
- this.selectedTweens = new Array();
- }
- },
-
- timelineLeftPaneMousedown:{
- value:function (event) {
- var ptrParent = nj.queryParentSelector(event.target, ".container-layer");
- if (ptrParent !== false) {
- var myIndex = this.getActiveLayerIndex();
- if (myIndex !== false) {
- this.selectLayer(myIndex, true);
- }
-
- }
- this._isMousedown = true;
- }
- },
-
- timelineLeftPaneMouseup:{
- value:function (event) {
- this._isMousedown = false;
- }
- },
-
- createNewLayer:{
- value:function (object) {
- var newLayerName = "",
- thingToPush = this.createLayerTemplate(),
- myIndex = 0,
- i = 0,
- arrLayersLength = this.arrLayers.length;
-
- // Make up a layer name.
- this.currentLayerNumber = this.currentLayerNumber + 1;
- newLayerName = "Layer " + this.currentLayerNumber;
-
- // Possibly currentLayerNumber doesn't correctly reflect the
- // number of layers. Check that.
- // Commented out to fix WebGL rendering bug
- /*for(k = 0; k < arrLayersLength; k++){
- if(this.arrLayers[k].layerData.layerName === newLayerName){
- this.currentLayerNumber = this.currentLayerNumber + 1;
- newLayerName = "Layer " + this.currentLayerNumber;
- break;
- }
- }*/
- // We will no longer have multiple things selected, so wipe that info out
- // if it isn't already gone.
- this.currentLayersSelected = false;
-
- // thingToPush is the template we just got. Now fill it in.
- thingToPush.layerData.layerName = newLayerName;
- thingToPush.layerData.layerTag = "<" + object.nodeName.toLowerCase() + ">";
- thingToPush.layerData.layerID = this.currentLayerNumber;
- thingToPush.parentElement = this.application.ninja.currentSelectedContainer;
- thingToPush.layerData.isSelected = true;
- thingToPush.layerData._isFirstDraw = true;
- thingToPush.layerData.created = true;
-
- if (this.checkable_animated.classList.contains("checked")) {
- thingToPush.layerData.isVisible = false;
- }
-
- if (this.layerRepetition.selectedIndexes) {
- // There is a selected layer, so we need to splice the new layer on top of it.
- myIndex = this.layerRepetition.selectedIndexes[0];
- if (typeof(myIndex) === "undefined") {
- // Edge case: sometimes there's nothing selected, so this will be "undefined"
- // In that case, set it to 0, the first layer.
- myIndex = 0;
- }
- for (var i = 0; i < this.layerRepetition.selectedIndexes.length; i++) {
- if (myIndex > this.layerRepetition.selectedIndexes[i]) {
- myIndex = this.layerRepetition.selectedIndexes[i];
- }
- }
- thingToPush.layerData.layerPosition = myIndex;
- thingToPush.layerData.trackPosition = myIndex;
- this.arrLayers.splice(myIndex, 0, thingToPush);
- } else {
- thingToPush.layerData.layerPosition = myIndex;
- this.arrLayers.splice(myIndex, 0, thingToPush);
-
- }
- this.selectLayer(myIndex, false);
- }
- },
-
- restoreLayer:{
- value:function (ele) {
-
- var newLayerName, thingToPush = this.createLayerTemplate();
-
- this.currentLayerNumber = this.currentLayerNumber + 1;
- newLayerName = "Layer " + this.currentLayerNumber;
-
- if(ele.dataset.storedLayerName){
- newLayerName = ele.dataset.storedLayerName;
- }
- thingToPush.layerData.layerName = newLayerName;
- thingToPush.layerData.layerID = this.currentLayerNumber;
- thingToPush.layerData.layerTag = "<" + ele.nodeName.toLowerCase() + ">";
- thingToPush.parentElement = this.application.ninja.currentSelectedContainer;
- if (this.checkable_animated.classList.contains("checked")) {
- thingToPush.layerData.isVisible = false;
- }
- // Are there styles to add?
- thingToPush.layerData.arrLayerStyles = this.createLayerStyles();
- thingToPush.layerData.arrStyleTracks = this.createStyleTracks();
-
- if (this._openDoc) {
- thingToPush.layerData.elementsList.push(ele);
- }
-
- this.temparrLayers.splice(0, 0, thingToPush);
- thingToPush.layerData.trackPosition = this.temparrLayers.length - 1;
- thingToPush.layerData.layerPosition = this.temparrLayers.length - 1;
-
- this._openDoc = false;
-
- }
- },
-
- deleteLayer:{
- value:function (arrElements) {
- // Only delete a selected layers. If no layers are selected, do nothing.
- var i = 0,
- arrLayers = document.querySelectorAll(".container-layers .container-layer"),
- arrLayersLength = arrLayers.length;
-
- for (i = arrLayersLength -1; i >= 0; i--) {
- if (arrLayers[i].classList.contains("selected")) {
- this.arrLayers.splice(i, 1);
- }
- }
-
- this.currentLayerSelected = false;
- this.currentLayersSelected = false;
- this.resetMasterDuration();
-
-
- /*
- var length = elements.length;
-
- while(length>0){
- if (this.layerRepetition.selectedIndexes.length > 0) {
- // Delete the selected layer.
- var myIndex = this.layerRepetition.selectedIndexes[0];
- this.arrLayers.splice(myIndex, 1);
- var selectIndex = this.arrLayers.length;
- this.resetMasterDuration();
- if(selectIndex>0){
- this.selectLayer(selectIndex-1);
- }
- length--;
- }
- }
- */
- }
- },
-
- resetMasterDuration:{
- value:function(){
- var trackDuration = 0,
- arrLayersLength = this.arrLayers.length,
- i = 0;
-
- if (arrLayersLength > 0) {
- for (i = 0; i < arrLayersLength; i++) {
- var currLength = this.arrLayers[i].layerData.trackDuration;
- if (currLength > trackDuration) {
- trackDuration = currLength;
- }
- }
- }
- this.masterDuration = trackDuration;
- }
- },
-
- handleElementAdded:{
- value:function() {
- this.createNewLayer(this.application.ninja.selectedElements[0]);
-
- if (typeof(this.currentLayerSelected) === "undefined") {
- // Edge case: currentLayerSelected needs to be initialized.
- this.currentLayerSelected = {};
- this.currentLayerSelected.layerData = {};
- this.currentLayerSelected.layerData.elementsList = [];
- }
- this.currentLayerSelected.layerData.elementsList.push(this.application.ninja.selectedElements[0]);
- this.currentLayerSelected.layerData.elementsList[0].dataset.storedLayerName = this.currentLayerSelected.layerData.layerName;
- }
- },
-
- handleElementsRemoved:{
- value:function (event) {
- var deleteElements = event.detail;
- //console.log("TimelinePanel.handleElementsRemoved; event.detail is ", event.detail);
- //debugger;
- this.deleteLayer(deleteElements);
- }
- },
-
- handleElementReplaced:{
- value:function(event){
- this.currentLayerSelected.layerData.elementsList.pop();
- this.currentLayerSelected.layerData.elementsList.push(event.detail.data.newChild);
- this.currentLayerSelected.layerData.animatedElement = event.detail.data.newChild;
- }
- },
-
- drawTimeMarkers:{
- value:function () {
- this.timeMarkerHolder = document.createElement("div");
-
- if(this.time_markers.children[0]){
- this.time_markers.removeChild(this.time_markers.children[0]);
- }
-
- this.time_markers.appendChild(this.timeMarkerHolder);
- var i;
- var totalMarkers = Math.floor(this.time_markers.offsetWidth / 80);
- for (i = 0; i < totalMarkers; i++) {
- var timeMark = document.createElement("div");
- var markValue = this.calculateTimeMarkerValue(i);
- timeMark.className = "timemark";
- timeMark.innerHTML = markValue;
- this.timeMarkerHolder.appendChild(timeMark);
- }
- }
- },
-
- calculateTimeMarkerValue:{
- value:function (currentMarker) {
- var currentMilliseconds = currentMarker * this.millisecondsOffset;
- return this.convertMillisecondsToTime(currentMilliseconds);
- }
- },
-
- updateTimeText:{
- value:function (millisec) {
- this.timetext.innerHTML = this.convertMillisecondsToTime(millisec);
- }
- },
-
- convertMillisecondsToTime:{
- value:function(millisec){
- var timeToReturn;
- var sec = (Math.floor((millisec / 1000))) % 60;
- var min = (Math.floor((millisec / 1000) / 60)) % 60;
- var milliSeconds = String(Math.round(millisec / 10));
- var returnMillisec = milliSeconds.slice(milliSeconds.length - 2, milliSeconds.length);
- var returnSec;
- var returnMin;
- if (sec < 10) {
- returnSec = "0" + sec;
- } else {
- returnSec = sec;
- }
- if (min < 10) {
- returnMin = "0" + min;
- } else {
- returnMin = min;
- }
- if (returnMillisec == "0") {
- returnMillisec = "0" + returnMillisec;
- }
- timeToReturn = returnMin + ":" + returnSec + ":" + returnMillisec;
- return timeToReturn;
- }
- },
-
- createLayerHashTable:{
- value:function (key, value) {
- var hashLayerObject;
- hashLayerObject = Object.create(Object.prototype, {
- counter:{
- value:0,
- writable:true
- },
-
- setItem:{
- value:function (key, value, index) {
- if (hashLayerObject[key] === undefined) {
- hashLayerObject[key] = {};
- }
- if (hashLayerObject[key][index] !== undefined) {
-
-
- for (this.counter = index; hashLayerObject[key][this.counter]; this.counter++) {
- }
-
- for (; this.counter !== index; this.counter--) {
- hashLayerObject[key][this.counter] = hashLayerObject[key][this.counter - 1];
- }
- }
- hashLayerObject[key][index] = value;
- this.counter = 0;
- }
- },
-
- getItem:{
- value:function (key) {
- return hashLayerObject[key];
- }
- }
- });
- return hashLayerObject;
- }
- },
-
- selectLayer:{
- value:function (layerIndex, userSelection) {
-
- var i = 0;
- var arrLayersLength = this.arrLayers.length;
-
- if (this.selectedKeyframes) {
- this.deselectTweens();
- }
-
- for (i = 0; i < arrLayersLength; i++) {
- if (i === layerIndex) {
- this.arrLayers[i].layerData.isSelected = true;
- } else {
- this.arrLayers[i].layerData.isSelected = false;
- }
-
- this.triggerLayerBinding(i);
- }
-
- this.layerRepetition.selectedIndexes = [layerIndex];
- this.currentLayerSelected = this.arrLayers[layerIndex];
- if (userSelection) {
- if (this._captureSelection) {
-
- if (this.currentLayerSelected.layerData.elementsList.length >= 1) {
- this.application.ninja.selectionController.selectElements(this.currentLayerSelected.layerData.elementsList);
- } else {
- this.application.ninja.selectionController.executeSelectElement();
- }
-
- }
- this._captureSelection = true;
- }
- this.resetMasterDuration();
- }
- },
-
- getLayerIndexByID:{
- value:function (layerID, tempArr) {
- var i = 0,
- returnVal = false,
- arrLayersLength = this.arrLayers.length;
-
- if (tempArr) {
- var tempArrLength = this.temparrLayers.length;
-
- for (i = 0; i < tempArrLength; i++) {
- if (this.temparrLayers[i].layerData.layerID === layerID) {
- returnVal = i;
- }
- }
-
- } else {
- for (i = 0; i < arrLayersLength; i++) {
- if (this.arrLayers[i].layerData.layerID === layerID) {
- returnVal = i;
- }
- }
- }
- return returnVal;
- }
- },
-
- getLayerIndexByName:{
- value:function (layerName) {
- var i = 0,
- returnVal = false,
- arrLayersLength = this.arrLayers.length;
-
- for (i = 0; i < arrLayersLength; i++) {
- if (this.arrLayers[i].layerData.layerName === layerName) {
- returnVal = i;
- }
- }
- return returnVal;
- }
- },
-
- getActiveLayerIndex:{
- value:function () {
- var i = 0,
- returnVal = false,
- arrLayersLength = this.arrLayers.length;
- for (i = 0; i < arrLayersLength; i++) {
- if (this.arrLayers[i].layerData.isActive === true) {
- returnVal = i;
- this.arrLayers[i].layerData.isActive = false;
- }
- }
- return returnVal;
- }
- },
-
- enablePanel:{
- value:function (boolEnable) {
- if (boolEnable) {
- this.timeline_disabler.style.display = "none";
- } else {
- this.timeline_disabler.style.display = "block";
- }
- }
- },
- handleConfigButtonClick: {
- value: function(event) {
- event.stopPropagation();
- this.handleCheckableClick(event);
-
- }
- },
- handleDocumentClick: {
- value: function(event) {
- if (this.tl_configbutton.classList.contains("checked")) {
- this.tl_configbutton.classList.remove("checked");
- }
- }
- },
-
- handleAnimatedClick: {
- value: function(event) {
- if (typeof(this.application.ninja.currentDocument) === "undefined") {
- return;
- }
- if (this.application.ninja.currentDocument == null) {
- return;
- }
- this.handleCheckableClick(event);
- this.application.ninja.currentDocument.boolShowOnlyAnimated = event.currentTarget.classList.contains("checked");
- var boolHide = false,
- i = 0,
- arrLayersLength = this.arrLayers.length;
- if (event.currentTarget.classList.contains("checked")) {
- // Hide layers with isAnimated = false;
- boolHide = true;
- }
-
- for (i = 0; i < arrLayersLength; i++) {
- if (boolHide) {
- // Hide layers with isAnimated = false
- if (this.arrLayers[i].layerData.isTrackAnimated === false) {
- this.arrLayers[i].layerData.isVisible = false;
- this.triggerLayerBinding(i);
- }
- } else {
- this.arrLayers[i].layerData.isVisible = true;
- this.triggerLayerBinding(i);
- }
- }
-
- }
- },
- handleRelativeClick: {
- value: function(event) {
- if (!event.currentTarget.classList.contains("checked")) {
- this.handleCheckableClick(event);
- }
- this.checkable_absolute.classList.remove("checked");
- this.useAbsolutePosition = false;
- }
- },
- handleAbsoluteClick: {
- value: function(event) {
- if (!event.currentTarget.classList.contains("checked")) {
- this.handleCheckableClick(event);
- }
- this.checkable_relative.classList.remove("checked");
- this.useAbsolutePosition = true;
- }
- },
- handleCheckableClick: {
- value: function(event) {
- if (event.currentTarget.classList.contains("checked")) {
- event.currentTarget.classList.remove("checked");
- } else {
- event.currentTarget.classList.add("checked");
- }
- }
- },
- // Trigger the layer/track data binding
- triggerLayerBinding : {
- value: function(intIndex) {
- this.arrLayers[intIndex].layerData.triggerBinding = !this.arrLayers[intIndex].layerData.triggerBinding;
- }
- },
-
- handleLayerDragStart : {
- value: function(event) {
- var dragIcon = document.createElement("img");
- event.dataTransfer.effectAllowed = 'move';
- event.dataTransfer.setData('Text', this.identifier);
- // dragIcon.src = "/images/transparent.png";
- dragIcon.src = ""
- dragIcon.width = 1;
- event.dataTransfer.setDragImage(dragIcon, 0, 0);
-
- // Clone the element we're dragging
- this._dragAndDropHelper = event.target.cloneNode(true);
- this._dragAndDropHelper.style.opacity = 0.8;
- this._dragAndDropHelper.style.position = "absolute";
- this._dragAndDropHelper.style.top = "0px";
- this._dragAndDropHelper.style.left = "0px";
- this._dragAndDropHelper.style.zIndex = 700;
-
- this._dragAndDropHelper.style.width = window.getComputedStyle(this.container_layers, null).getPropertyValue("width");
- this._dragAndDropHelper.classList.add("timeline-dnd-helper");
-
- // Get the offset
- var findYOffset = function(obj) {
- var curleft = curtop = 0;
-
- if (obj.offsetParent) {
- do {
- curleft += obj.offsetLeft;
- curtop += obj.offsetTop;
-
- } while (obj = obj.offsetParent);
- }
- return curtop;
- }
- this._dragAndDropHelperOffset = findYOffset(this.container_layers);
- this._appendHelper = true;
- this._deleteHelper = false;
- }
- },
- handleLayerDragover: {
- value: function(event) {
- var currPos = 0,
- myScrollTest = ((event.y - (this._dragAndDropHelperOffset - this.user_layers.scrollTop)) + 28) - this.user_layers.scrollTop;
- if ((myScrollTest < 60) && (this.user_layers.scrollTop >0)) {
- this._scrollTracks = (this.user_layers.scrollTop - 10)
- }
- if ((myScrollTest < 50) && (this.user_layers.scrollTop >0)) {
- this._scrollTracks = (this.user_layers.scrollTop - 20)
- }
- if ((myScrollTest > (this.user_layers.clientHeight + 10))) {
- this._scrollTracks = (this.user_layers.scrollTop + 10)
- }
- if ((myScrollTest > (this.user_layers.clientHeight + 20))) {
- this._scrollTracks = (this.user_layers.scrollTop + 20)
-
- }
- currPos = event.y - (this._dragAndDropHelperOffset - this.user_layers.scrollTop)- 28;
- this._dragAndDropHelperCoords = currPos + "px";
- this.needsDraw = true;
- }
- },
- handleLayerDragEnd : {
- value: function(event) {
- this._deleteHelper = true;
- this.needsDraw = true;
-
- }
- },
- handleLayerDrop : {
- value: function(event) {
- event.stopPropagation();
- event.preventDefault();
- this._deleteHelper = true;
- this.needsDraw = true;
- }
- },
- /* === END: Controllers === */
-
- /* === BEGIN: Logging routines === */
- _boolDebug:{
- enumerable:false,
- value:false // set to true to enable debugging to console; false for turning off all debugging.
- },
- boolDebug:{
- get:function () {
- return this._boolDebug;
- },
- set:function (boolDebugSwitch) {
- this._boolDebug = boolDebugSwitch;
- }
- },
- log:{
- value:function (strMessage) {
- if (this.boolDebug) {
- console.log(this.getLineNumber() + ": " + strMessage);
- }
- }
- },
- getLineNumber:{
- value:function () {
- try {
- throw new Error('bazinga')
- } catch (e) {
- return e.stack.split("at")[3].split(":")[2];
- }
- }
- }
- /* === END: Logging routines === */
+/*
+ This file contains proprietary software owned by Motorola Mobility, Inc.
+ No rights, expressed or implied, whatsoever to this software are provided by Motorola Mobility, Inc. hereunder.
+ (c) Copyright 2011 Motorola Mobility, Inc. All Rights Reserved.
+ */
+
+var Montage = require("montage/core/core").Montage;
+var Component = require("montage/ui/component").Component;
+var nj = require("js/lib/NJUtils").NJUtils;
+
+var TimelinePanel = exports.TimelinePanel = Montage.create(Component, {
+
+ hasTemplate:{
+ value:true
+ },
+
+ /* === BEGIN: Models === */
+ _currentDocument: {
+ value : null
+ },
+
+ currentDocument : {
+ get : function() {
+ return this._currentDocument;
+ },
+ set : function(value) {
+ // If it's the same document, do nothing.
+ if (value === this._currentDocument) {
+ return;
+ }
+
+ if(!this._currentDocument && value.currentView === "design") {
+ this.enablePanel(true);
+ }
+
+ this._currentDocument = value;
+
+ if(!value) {
+ this._boolCacheArrays = false;
+ this.clearTimelinePanel();
+ this._boolCacheArrays = true;
+ this.enablePanel(false);
+ } else if(this._currentDocument.currentView === "design") {
+ this._boolCacheArrays = false;
+ this.clearTimelinePanel();
+ this._boolCacheArrays = true;
+
+ // Rebind the document events for the new document context
+ this._bindDocumentEvents();
+
+ // Initialize the timeline for the document.
+ this.initTimelineForDocument();
+ }
+ }
+ },
+
+ _currentSelectedContainer: {
+ value: null
+ },
+ currentSelectedContainer: {
+ get: function() {
+ return this._currentSelectedContainer;
+ },
+ set: function(newVal) {
+ if(this._currentSelectedContainer !== newVal) {
+ this._currentSelectedContainer = newVal;
+ if (this._ignoreNextContainerChange === true) {
+ this._ignoreNextContainerChange = false;
+ return;
+ }
+ this.application.ninja.currentDocument.setLevel = true;
+
+ if(this._currentDocument.currentView === "design") {
+ this._boolCacheArrays = false;
+ this.clearTimelinePanel();
+ this._boolCacheArrays = true;
+
+ // Rebind the document events for the new document context
+ this._bindDocumentEvents();
+
+ // Initialize the timeline for the document.
+ this.initTimelineForDocument();
+ }
+ }
+ }
+ },
+
+ _arrLayers:{
+ value:[]
+ },
+
+ arrLayers:{
+ serializable:true,
+ get:function () {
+ return this._arrLayers;
+ },
+ set:function (newVal) {
+ this._arrLayers = newVal;
+ this.needsDraw = true;
+ this.cacheTimeline();
+ }
+ },
+
+ _temparrLayers:{
+ value:[]
+ },
+
+ temparrLayers:{
+ get:function () {
+ return this._temparrLayers;
+ },
+ set:function (newVal) {
+ this._temparrLayers = newVal;
+ }
+ },
+
+
+ _layerRepetition:{
+ value:null
+ },
+
+ layerRepetition:{
+ get:function () {
+ return this._layerRepetition;
+ },
+ set:function (newVal) {
+ this._layerRepetition = newVal;
+ }
+ },
+
+ // Set to false to skip array caching array sets in current document
+ _boolCacheArrays:{
+ value:true
+ },
+
+ _currentLayerNumber:{
+ value:0
+ },
+
+ currentLayerNumber:{
+ get:function () {
+ return this._currentLayerNumber;
+ },
+ set:function (newVal) {
+ if (newVal !== this._currentLayerNumber) {
+ this._currentLayerNumber = newVal;
+ this.cacheTimeline();
+ }
+ }
+ },
+
+ _currentLayerSelected:{
+ value: false
+ },
+ currentLayerSelected:{
+ get:function () {
+ return this._currentLayerSelected;
+ },
+ set:function (newVal) {
+ this._currentLayerSelected = newVal;
+ this.cacheTimeline();
+ }
+ },
+
+ _selectedLayerID:{
+ value:false
+ },
+ selectedLayerID:{
+ get:function () {
+ return this._selectedLayerID;
+ },
+ set:function (newVal) {
+ if (newVal === false) {
+ // We are clearing the timeline, so just set the value and return.
+ this._selectedLayerID = newVal;
+ return;
+ }
+ if (newVal !== this._selectedLayerID) {
+ var selectIndex = this.getLayerIndexByID(newVal);
+ this._selectedLayerID = newVal;
+ this._captureSelection = true;
+ if (this.currentLayerSelected !== false) {
+ this.selectLayer(selectIndex, false);
+ }
+ if (this.currentLayersSelected !== false) {
+ this.selectLayers(this.currentLayersSelected);
+ }
+ if ((this.currentLayersSelected === false) && (this.currentLayerSelected === false)) {
+ this.selectLayers([]);
+ }
+
+ }
+ }
+ },
+
+ _currentLayersSelected:{
+ value:[]
+ },
+ currentLayersSelected:{
+ get:function () {
+ return this._currentLayersSelected;
+ },
+ set:function (newVal) {
+ this._currentLayersSelected = newVal;
+ this.cacheTimeline();
+ }
+ },
+
+
+ _millisecondsOffset:{
+ value:1000
+ },
+
+ millisecondsOffset:{
+ get:function () {
+ return this._millisecondsOffset;
+ },
+ set:function (newVal) {
+ if (newVal !== this._millisecondsOffset) {
+ this._millisecondsOffset= newVal;
+ this.drawTimeMarkers();
+ NJevent('tlZoomSlider',this);
+ }
+ }
+ },
+
+ _masterDuration:{
+ value:0
+ },
+
+ masterDuration:{
+ serializable:true,
+ get:function () {
+ return this._masterDuration;
+ },
+ set:function (val) {
+ this._masterDuration = val;
+ this.timebar.style.width = (this._masterDuration / 12) + "px";
+ }
+ },
+
+ _trackRepetition:{
+ value:null
+ },
+
+ trackRepetition:{
+ get:function () {
+ return this._trackRepetition;
+ },
+ set:function (newVal) {
+ this._trackRepetition = newVal;
+ }
+ },
+
+ _selectedKeyframes:{
+ value:[]
+ },
+
+ selectedKeyframes:{
+ serializable:true,
+ get:function () {
+ return this._selectedKeyframes;
+ },
+ set:function (newVal) {
+ this._selectedKeyframes = newVal;
+ }
+ },
+
+ _selectedTweens:{
+ value:[]
+ },
+
+ selectedTweens:{
+ serializable:true,
+ get:function () {
+ return this._selectedTweens;
+ },
+ set:function (newVal) {
+ this._selectedTweens = newVal;
+ }
+ },
+
+ _breadCrumbContainer:{
+ value:null
+ },
+
+ breadCrumbContainer:{
+ get:function () {
+ return this._breadCrumbContainer;
+ },
+ set:function (value) {
+ if (this._breadCrumbContainer !== value) {
+ this._breadCrumbContainer = value;
+ }
+ }
+ },
+
+ _isLayer:{
+ value:false
+ },
+
+ _firstTimeLoaded:{
+ value:true
+ },
+
+ _captureSelection:{
+ value:false
+ },
+
+ _openDoc:{
+ value:false
+ },
+
+ timeMarkerHolder:{
+ value:null
+ },
+ _dragAndDropHelper : {
+ value: false
+ },
+ _dragAndDropHelperCoords: {
+ value: false
+ },
+ _dragAndDropHelperOffset : {
+ value: false
+ },
+ _dragLayerID : {
+ value: null
+ },
+
+ layersDragged:{
+ value:[],
+ writable:true
+ },
+
+ dragLayerID : {
+ get: function() {
+ return this._dragLayerID;
+ },
+ set: function(newVal) {
+ if (newVal !== this._dragLayerID) {
+ this._dragLayerID = newVal;
+ }
+ }
+ },
+ _dropLayerID : {
+ value: null
+ },
+ dropLayerID : {
+ get: function() {
+ return this._dropLayerID;
+ },
+ set: function(newVal) {
+ if (newVal !== this._dropLayerID) {
+ this._dropLayerID = newVal;
+
+ var dragLayerIndex = this.getLayerIndexByID(this.dragLayerID),
+ dropLayerIndex