/* 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 Collapser = require("js/panels/Timeline/Collapser").Collapser; var defaultEventManager = require("montage/core/event/event-manager").defaultEventManager; var TimelineTrack = exports.TimelineTrack = Montage.create(Component, { // ==== Begin Models _trackID:{ value:null }, trackID:{ serializable:true, get:function () { return this._trackID; }, set:function (value) { if (value !== this._trackID) { this._trackID = value; this.trackData.layerID = value; } } }, _tween:{ value:[] }, tween:{ serializable:true, get:function () { return this._tween; }, set:function (newVal) { this._tween = newVal; } }, positionPropertyTrack:{ value:null }, _isFirstDraw: { value: true }, _isVisible:{ value: true }, isVisible:{ get:function(){ return this._isVisible; }, set:function(value){ if (this._isVisible !== value) { this._isVisible = value; if (this.element !== null) { if (value === true) { this.element.classList.remove("layer-hidden"); } else { this.element.classList.add("layer-hidden"); } } } this.trackData.isVisible = value; } }, _bindingPoint : { value : {} }, bindingPoint: { get: function() { return this._bindingPoint; }, set: function(newVal) { if (newVal !== this._bindingPoint) { this._bindingPoint = newVal; this.setData(); } } }, _stageElement: { value: null }, stageElement: { get: function() { return this._stageElement; }, set: function(newVal) { this._stageElement = newVal; this.trackData.stageElement = newVal; } }, _isMainCollapsed:{ value: true }, isMainCollapsed:{ get:function () { return this._isMainCollapsed; }, set:function (newVal) { this._isMainCollapsed = newVal; this.trackData.isMainCollapsed = newVal; } }, _isPositionCollapsed:{ value:true }, isPositionCollapsed:{ get:function () { return this._isPositionCollapsed; }, set:function (newVal) { this._isPositionCollapsed = newVal; this.trackData.isPositionCollapsed = newVal; } }, _isStyleCollapsed:{ value:true }, isStyleCollapsed:{ get:function () { return this._isStyleCollapsed; }, set:function (newVal) { this._isStyleCollapsed = newVal; this.trackData.isStyleCollapsed = newVal; } }, _bypassAnimation : { value: false }, bypassAnimation : { serializable: true, get: function() { return this._bypassAnimation; }, set: function(newVal) { if (typeof(this.trackData) !== "undefined") { this._bypassAnimation = newVal; this.trackData.bypassAnimation = newVal; } } }, _arrStyleTracks : { value: [] }, arrStyleTracks: { serializable:true, get: function() { return this._arrStyleTracks; }, set: function(newVal) { this._arrStyleTracks = newVal; this.trackData.arrStyleTracks = newVal; } }, _styleTracksRepetition: { value: null }, styleTracksRepetition : { serializable:true, get: function() { return this._styleTracksRepetition; }, set: function(newVal) { this._styleTracksRepetition = newVal; } }, _arrPositionTracks : { value: [] }, arrPositionTracks: { serializable:true, get: function() { return this._arrPositionTracks; }, set: function(newVal) { this._arrPositionTracks = newVal; this.trackData.arrPositionTracks = newVal; } }, _positionTracksRepetition: { value: null }, positionTracksRepetition : { get: function() { return this._positionTracksRepetition; }, set: function(newVal) { this._positionTracksRepetition = newVal; } }, _tweens:{ value:[] }, tweens:{ serializable: true, get:function () { return this._tweens; }, set:function (newVal) { this._tweens = newVal; this.trackData.tweens = newVal; } }, _tweenRepetition:{ value:null }, tweenRepetition:{ get:function () { return this._tweenRepetition; }, set:function (newVal) { this._tweenRepetition = newVal; } }, _trackDuration:{ value:0 }, trackDuration:{ serializable:true, get:function () { return this._trackDuration; }, set:function (val) { this._trackDuration = val; if(this._trackDuration > this.application.ninja.timeline.masterDuration){ this.application.ninja.timeline.masterDuration = this._trackDuration; } this.trackData.trackDuration = val; } }, _trackPosition:{ value:0 }, trackPosition:{ serializable:true, get:function () { return this._trackPosition; }, set:function (val) { this._trackPosition = val; this.trackData.trackPosition = val; } }, _currentKeyframeRule:{ value:null }, currentKeyframeRule:{ serializable: true, get:function(){ return this._currentKeyframeRule; }, set:function(val){ this._currentKeyframeRule = val; this.trackData.currentKeyframeRule = val; } }, nextKeyframe:{ value:1 }, currentMillisecClicked:{ value:0 }, _isTrackAnimated:{ value:null }, isTrackAnimated:{ serializable: true, get:function(){ return this._isTrackAnimated; }, set:function(val){ this._isTrackAnimated = val; this.trackData.isTrackAnimated = val; } }, _animatedElement:{ value:null }, animatedElement:{ serializable:true, get:function () { return this._animatedElement; }, set:function (val) { this._animatedElement = val; this.trackData.animatedElement = val; } }, _animationName:{ value:null }, animationName:{ serializable:true, get:function () { return this._animationName; }, set:function (val) { this._animationName = val; this.trackData.animationName = val; } }, _ruleList:{ value:[] }, ruleList:{ get:function () { return this._ruleList; }, set:function (val) { this._ruleList = val; } }, animationNamesString:{ value:"" }, ninjaStylesContoller:{ value:null }, _positionCollapser:{ value:null }, positionCollapser:{ serializable:true, get:function(){ return this._positionCollapser; }, set:function(val){ this._positionCollapser = val; } }, _mainCollapser:{ value:null }, mainCollapser:{ serializable:true, get:function () { return this._mainCollapser; }, set:function (val) { this._mainCollapser = val; } }, _styleCollapser:{ value:null }, styleCollapser:{ serializable:true, get:function () { return this._styleCollapser; }, set:function (val) { this._styleCollapser = val; } }, _dragAndDropHelper : { value: false }, _dragAndDropHelperCoords: { value: false }, dragAndDropHelperCoords: { get: function() { return this._dragAndDropHelperCoords; }, set: function(newVal) { this._dragAndDropHelperCoords = newVal; } }, _draggingIndex: { value: false }, draggingIndex: { get: function() { return this._draggingIndex; }, set: function(newVal) { this._draggingIndex = newVal; } }, _dragAndDropHelperOffset : { value: false }, _appendHelper: { value: false }, _deleteHelper: { value: false }, _trackData:{ value: false }, trackData:{ serializable: true, get:function(){ return this._trackData; }, set:function(val){ this._trackData = val; if(this._trackData){ this.setData(); } } }, _setDataTimestamp : { value: false }, setData:{ value:function(){ if (typeof(this.trackData) === "undefined") { return; } this.bypassAnimation = this.trackData.bypassAnimation; this.trackID = this.trackData.layerID; this.tweens = this.trackData.tweens; this.arrPositionTracks = this.trackData.arrPositionTracks; this.animatedElement = this.trackData.animatedElement; this.arrStyleTracks = this.trackData.arrStyleTracks; this.isTrackAnimated = this.trackData.isTrackAnimated; this.trackDuration = this.trackData.trackDuration; this.animationName = this.trackData.animationName; this.currentKeyframeRule = this.trackData.currentKeyframeRule; this.isMainCollapsed = this.trackData.isMainCollapsed; this.isPositionCollapsed = this.trackData.isPositionCollapsed; this.isTransformCollapsed = this.trackData.isTransformCollapsed; this.isStyleCollapsed = this.trackData.isStyleCollapsed; this.trackPosition = this.trackData.trackPosition; this.isVisible = this.trackData.isVisible; this.stageElement = this.trackData.stageElement; this.trackEditorProperty = "master"; this.needsDraw = true; } }, // ==== End Models // ==== Begin Draw cycle methods prepareForDraw:{ value:function () { this.init(); this.ninjaStylesContoller = this.application.ninja.stylesController; this.element.addEventListener("click", this, false); this.eventManager.addEventListener("tlZoomSlider", this, false); // Drag and Drop event handlers this.element.addEventListener("dragstart", this.handleKeyframeDragstart.bind(this), false); this.element.addEventListener("dragend", this.handleKeyframeDragend.bind(this), false); } }, draw:{ value:function () { this.ninjaStylesContoller = this.application.ninja.stylesController; var selectedIndex = this.application.ninja.timeline.getLayerIndexByID(this.trackID); if (selectedIndex !== false) { if(this.application.ninja.timeline.arrLayers[selectedIndex].layerData.stageElement){ this.animatedElement = this.application.ninja.timeline.arrLayers[selectedIndex].layerData.stageElement; } } // Drag and Drop: if (this._appendHelper === true) { this.track_lanes.appendChild(this._dragAndDropHelper); this._appendHelper = false; } if (this._dragAndDropHelperCoords !== false) { if (this._dragAndDropHelper !== null) { if (typeof(this._dragAndDropHelper.style) !== "undefined") { this._dragAndDropHelper.style.left = this._dragAndDropHelperCoords; } } this._dragAndDropHelperCoords = false; } if (this._deleteHelper === true) { if (this._dragAndDropHelper === null) { var myHelper = this.element.querySelector(".track-dnd-helper"); if (myHelper != null) { this._dragAndDropHelper = myHelper; } } if (this._dragAndDropHelper !== null) { if (this._dragAndDropHelper && this._dragAndDropHelper.parentNode === this.track_lanes) { this.track_lanes.removeChild(this._dragAndDropHelper); this._dragAndDropHelper = null; this._deleteHelper = false; } } } } }, didDraw:{ value:function () { if ((!this.application.ninja.documentController.creatingNewFile)||(!this.application.ninja.currentDocument.setLevel)) { if (this.application.ninja.currentDocument.model.documentRoot.children[0]) { var selectedIndex = this.application.ninja.timeline.getLayerIndexByID(this.trackID); if (selectedIndex !== false) { if (!this.application.ninja.timeline.arrLayers[selectedIndex].layerData.created) { this.retrieveStoredTweens(); } } } } if (this._isFirstDraw === true) { if (this.isMainCollapsed === false) { this._mainCollapser.myContent.style.height = "auto"; this._mainCollapser.myContent.classList.remove(this._mainCollapser.collapsedClass); this._mainCollapser.clicker.classList.remove(this._mainCollapser.collapsedClass); } if (this.isPositionCollapsed === false) { this._positionCollapser.myContent.style.height = "auto"; this._positionCollapser.myContent.classList.remove(this._positionCollapser.collapsedClass); this._positionCollapser.clicker.classList.remove(this._positionCollapser.collapsedClass); } if (this.isStyleCollapsed === false) { this._styleCollapser.myContent.style.height = "auto"; this._styleCollapser.myContent.classList.remove(this._styleCollapser.collapsedClass); this._styleCollapser.clicker.classList.remove(this._styleCollapser.collapsedClass); } this._isFirstDraw = false; } } }, // ==== End Draw cycle methods // ==== Begin Event handlers handleTlZoomSlider: { value: function(event) { var currentMilliSecPerPixel , currentMilliSec , clickPos,thingToPush; var i = 0, tweensLength = this.tweens.length; for (i = 0; i < tweensLength; i++) { if (i === 0) { // Exception: 0th item does not depend on anything // If 0th tween is draggable, this will need to be fixed. this.tweens[i].tweenData.spanWidth=0; this.tweens[i].tweenData.spanPosition=0; this.tweens[i].tweenData.keyFramePosition=0; this.tweens[i].tweenData.keyFrameMillisec=0; } else { var prevKeyFramePosition = this.tweens[i - 1].tweenData.keyFramePosition, myObj = {}, thing = {}; currentMilliSecPerPixel = Math.floor(this.application.ninja.timeline.millisecondsOffset / 80); currentMilliSec = this.tweens[i].tweenData.keyFrameMillisec; clickPos = currentMilliSec / currentMilliSecPerPixel; for (thing in this.tweens[i].tweenData) { myObj[thing] = this.tweens[i].tweenData[thing]; } myObj.spanWidth = clickPos - prevKeyFramePosition; myObj.keyFramePosition = clickPos; myObj.spanPosition = clickPos - (clickPos - prevKeyFramePosition); this.tweens[i].tweenData = myObj; } } } }, handleClick:{ value:function (ev) { var targetElementOffset = this.findXOffset(ev.currentTarget), position = (event.pageX - targetElementOffset) - 18; this.application.ninja.timeline.playheadmarker.style.left = position + "px"; var currentMillisecPerPixel = Math.floor(this.application.ninja.timeline.millisecondsOffset / 80); var currentMillisec = currentMillisecPerPixel * position; this.application.ninja.timeline.updateTimeText(currentMillisec); if (ev.shiftKey) { var selectedIndex = this.application.ninja.timeline.getLayerIndexByID(this.trackID); this.application.ninja.timeline.selectLayer(selectedIndex, true); if (this.tweens.length < 1) { this.insertTween(0); this.addAnimationRuleToElement(ev); this.updateKeyframeRule(); } else { if (ev.target.className === "tracklane") { this.handleNewTween(ev); this.updateKeyframeRule(); } else if (ev.target.className === "tween_span_bar" && ev.target.parentElement.parentElement.parentElement.className === "tracklane") { this.handleNewTween(ev); this.updateKeyframeRule(); } } } } }, handleKeyboardShortcut:{ value:function (ev) { if (ev.actionType == "insert") { if (this.tweens.length < 1) { this.insertTween(0); this.addAnimationRuleToElement(ev); this.updateKeyframeRule(); } else { this.handleNewTween(ev); this.updateKeyframeRule(); } } else if (ev.actionType == "remove") { this.removeTween(); this.updateKeyframeRule(); } } }, // Drag and drop event handlers handleKeyframeDragstart:{ value:function (event) { var dragIcon = document.createElement("img"), minPosition = 0, maxPosition = 100000000000; event.dataTransfer.effectAllowed = 'move'; event.dataTransfer.setData('Text', this.identifier); 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 = "5px"; this._dragAndDropHelper.style.left = "0px"; this._dragAndDropHelper.style.zIndex = 700; this._dragAndDropHelper.classList.add("keyframeSelected"); this._dragAndDropHelper.classList.add("track-dnd-helper"); if (this.draggingIndex < (this.tweens.length - 1)) { maxPosition = this.tweenRepetition.childComponents[this.draggingIndex + 1].keyFramePosition; } if (this.draggingIndex > 1) { minPosition = this.tweenRepetition.childComponents[this.draggingIndex - 1].keyFramePosition; } this._keyframeMinPosition = minPosition + 2; this._keyframeMaxPosition = maxPosition - 9; this._appendHelper = true; this._deleteHelper = false; var i = 0, arrLayersLength = this.parentComponent.parentComponent.arrLayers.length, myId = null; for (i = 0; i < arrLayersLength; i++) { var currUuid = this.parentComponent.parentComponent.trackRepetition.childComponents[i].uuid; if (currUuid === this.uuid) { myId = i; } } this.parentComponent.parentComponent.draggingTrackId = myId; this.parentComponent.parentComponent.draggingType = "keyframe"; } }, handleKeyframeDragend:{ value:function (event) { if (this.parentComponent.parentComponent.draggingType !== "keyframe") { return; } this._deleteHelper = true; this.needsDraw = true; } }, handleNewTween:{ value:function (ev) { if (ev.offsetX > this.tweens[this.tweens.length - 1].tweenData.keyFramePosition) { var selectedIndex = this.application.ninja.timeline.getLayerIndexByID(this.trackID); this.application.ninja.timeline.selectLayer(selectedIndex, false); this.insertTween(ev.offsetX); } else { if (typeof(ev.currentTarget) === "undefined") { this.splitTweenAt(ev.offsetX); } else { var targetElementOffset = this.findXOffset(ev.currentTarget), position = event.pageX - targetElementOffset; this.splitTweenAt(position-18); } } } }, handleLayerEvent:{ value:function (layerEvent) { if (layerEvent.layerID !== this.trackID) { return; } if (layerEvent.layerEventType === "newStyle") { var newStyleTrack = {}; newStyleTrack.propTrackData = {}; newStyleTrack.propTrackData.styleSelection = layerEvent.styleSelection; newStyleTrack.propTrackData.propTweens = []; newStyleTrack.propTrackData.trackType = "style"; newStyleTrack.propTrackData.trackEditorProperty = ""; newStyleTrack.propTrackData.styleIndex = layerEvent.styleIndex; newStyleTrack.propTrackData.existingRule = ""; this.arrStyleTracks.push(newStyleTrack); } else if (layerEvent.layerEventType === "restoreStyle") { var restoredStyleTrack = {}; restoredStyleTrack.propTrackData = {}; restoredStyleTrack.propTrackData.styleSelection = layerEvent.styleSelection; restoredStyleTrack.propTrackData.propTweens = []; restoredStyleTrack.propTrackData.trackType = "style"; restoredStyleTrack.propTrackData.trackEditorProperty = layerEvent.trackEditorProperty; restoredStyleTrack.propTrackData.styleIndex = layerEvent.styleIndex; restoredStyleTrack.propTrackData.existingRule = layerEvent.existingRule; this.arrStyleTracks.push(restoredStyleTrack); } else if (layerEvent.layerEventType === "deleteStyle") { this.arrStyleTracks.splice(layerEvent._event.selectedStyleIndex, 1); } } }, // ==== End Event handlers // ==== Begin Controllers init:{ value:function () { this.createPositionTracks(); this.element.addEventListener("layerEvent", this, false); } }, createPositionTracks:{ value:function(){ if (this.arrPositionTracks.length > 0) { return; } var newLeftTrack = {}; newLeftTrack.propTrackData = {}; newLeftTrack.propTrackData.propTweens = []; newLeftTrack.propTrackData.styleIndex = 1; newLeftTrack.propTrackData.trackType = "position"; newLeftTrack.propTrackData.trackEditorProperty = "left"; this.arrPositionTracks.push(newLeftTrack); var newTopTrack = {}; newTopTrack.propTrackData = {}; newTopTrack.propTrackData.propTweens = []; newTopTrack.propTrackData.styleIndex = 0; newTopTrack.propTrackData.trackType = "position"; newTopTrack.propTrackData.trackEditorProperty = "top"; this.arrPositionTracks.push(newTopTrack); var newWidthTrack = {}; newWidthTrack.propTrackData = {}; newWidthTrack.propTrackData.propTweens = []; newWidthTrack.propTrackData.styleIndex = 2; newWidthTrack.propTrackData.trackType = "position"; newWidthTrack.propTrackData.trackEditorProperty = "width"; this.arrPositionTracks.push(newWidthTrack); var newHeightTrack = {}; newHeightTrack.propTrackData = {}; newHeightTrack.propTrackData.propTweens = []; newHeightTrack.propTrackData.styleIndex = 3; newHeightTrack.propTrackData.trackType = "position"; newHeightTrack.propTrackData.trackEditorProperty = "height"; this.arrPositionTracks.push(newHeightTrack); } }, createTrackData:{ value:function () { tempData = {}; tempData.bypassAnimation = this.bypassAnimation; tempData.trackID = this.layerID; tempData.tweens = this.tweens; tempData.animatedElement = this.animatedElement; tempData.arrStyleTracks = this.arrStyleTracks; tempData.arrPositionTracks = this.arrPositionTracks; tempData.isTrackAnimated = this.isTrackAnimated; tempData.trackDuration = this.trackDuration; tempData.animationName = this.animationName; tempData.currentKeyframeRule = this.currentKeyframeRule; tempData.isMainCollapsed = this.isMainCollapsed; tempData.isPositionCollapsed = this.isPositionCollapsed; tempData.isTransformCollapsed = this.isTransformCollapsed; tempData.isStyleCollapsed = this.isStyleCollapsed; tempData.trackPosition = this.trackPosition; tempData.isVisible = this.isVisible; this.trackData = tempData; } }, triggerOutgoingBinding:{ value:function () { this.trackData.triggerBinding = !this.trackData.triggerBinding; } }, insertTween:{ value:function (clickPos) { var currentMillisecPerPixel = Math.floor(this.application.ninja.timeline.millisecondsOffset / 80); var currentMillisec = currentMillisecPerPixel * clickPos; this.trackDuration = currentMillisec; var newTween = {}; newTween.tweenData = {}; if (clickPos == 0) { this.animatedElement = this.application.ninja.timeline.arrLayers[this.application.ninja.timeline.currentLayersSelected[0]].layerData.stageElement; newTween.tweenData.spanWidth = 0; newTween.tweenData.keyFramePosition = 0; newTween.tweenData.keyFrameMillisec = 0; newTween.tweenData.tweenID = 0; newTween.tweenData.spanPosition = 0; newTween.tweenData.easing = "none"; newTween.tweenData.initSelect = true; newTween.tweenData.tweenedProperties = []; newTween.tweenData.tweenedProperties["top"] = this.animatedElement.offsetTop + "px"; newTween.tweenData.tweenedProperties["left"] = this.animatedElement.offsetLeft + "px"; newTween.tweenData.tweenedProperties["width"] = this.animatedElement.offsetWidth + "px"; newTween.tweenData.tweenedProperties["height"] = this.animatedElement.offsetHeight + "px"; this.tweens.push(newTween); this.createMatchingPositionSizeTween(newTween); } else { newTween.tweenData.spanWidth = clickPos - this.tweens[this.tweens.length - 1].tweenData.keyFramePosition; newTween.tweenData.keyFramePosition = clickPos; newTween.tweenData.keyFrameMillisec = currentMillisec; newTween.tweenData.tweenID = this.nextKeyframe; newTween.tweenData.spanPosition = clickPos - newTween.tweenData.spanWidth; newTween.tweenData.easing = "none"; newTween.tweenData.initSelect = true; newTween.tweenData.tweenedProperties = []; newTween.tweenData.tweenedProperties["top"] = this.animatedElement.offsetTop + "px"; newTween.tweenData.tweenedProperties["left"] = this.animatedElement.offsetLeft + "px"; newTween.tweenData.tweenedProperties["width"] = this.animatedElement.offsetWidth + "px"; newTween.tweenData.tweenedProperties["height"] = this.animatedElement.offsetHeight + "px"; this.tweens.push(newTween); var animationDuration = (this.trackDuration / 1000) + "s"; this.ninjaStylesContoller.setElementStyle(this.animatedElement, "-webkit-animation-duration", animationDuration); this.nextKeyframe += 1; this.createMatchingPositionSizeTween(newTween); } this.application.ninja.currentDocument.model.needsSave = true; } }, removeTween:{ value:function(){ var tweenIDToRemove = this.application.ninja.timeline.selectedTweens[0].tweenID, oldPosition = this.application.ninja.timeline.selectedTweens[0].spanPosition, oldSpanWidth = this.application.ninja.timeline.selectedTweens[0].spanWidth; if(tweenIDToRemove == this.tweens[this.tweens.length-1].tweenData.tweenID){ this.trackDuration = this.tweens[this.tweens.length-2].tweenData.keyFrameMillisec; this.tweens.pop(); return; } this.tweens[tweenIDToRemove + 1].tweenData.spanPosition = oldPosition; this.tweens[tweenIDToRemove + 1].spanPosition = oldPosition; this.tweens[tweenIDToRemove + 1].tweenData.spanWidth = this.tweens[tweenIDToRemove + 1].tweenData.spanWidth + oldSpanWidth; this.tweens[tweenIDToRemove + 1].spanWidth = this.tweens[tweenIDToRemove + 1].spanWidth + oldSpanWidth; for(var i in this.tweenRepetition.childComponents){ this.tweenRepetition.childComponents[i].setData(); } this.tweens.splice(tweenIDToRemove, 1); this.application.ninja.currentDocument.model.needsSave = true; for (var j = 0; j < this.tweens.length; j++) { this.tweens[j].tweenID = j; this.tweens[j].tweenData.tweenID = j; } } }, // splitTweenAt: Split a tween at a particular position (x coordinate) splitTweenAt: { value:function (position) { var i, j, nextComponentIndex, tweensLength = this.tweens.length-1, prevTween, nextTween, splitTweenIndex; // Search through the tweens and find the pair whose keyframes bracket position. for(i=0; i prevTween && position < nextTween) { // We will insert a new tween at this index splitTweenIndex = i+1; // Update the next tween to have new span position and width. this.tweens[i+1].tweenData.spanPosition = position; this.tweens[i+1].spanPosition = position; this.tweens[i+1].tweenData.spanWidth = this.tweens[i+1].tweenData.keyFramePosition - position; this.tweens[i+1].spanWidth = this.tweens[i+1].keyFramePosition - position; // You'd think that would be enough to make the component associated with that part of the array redraw, wouldn't you? // Turns out we have to manually poke the desired childComponent in the repetition to register its new changes. // So we have to get the index of the actual componentin the repetition, which may not match our iteration index. for (j = 0; j < tweensLength +1; j++) { if (this.tweenRepetition.childComponents[j].keyFramePosition === nextTween) { nextComponentIndex = j; } } this.tweenRepetition.childComponents[nextComponentIndex].setData(); // Create the new tween and splice it into the model var newTweenToInsert = {}; newTweenToInsert.tweenData = {}; newTweenToInsert.tweenData.spanWidth = position - prevTween; newTweenToInsert.tweenData.keyFramePosition = position; newTweenToInsert.tweenData.keyFrameMillisec = Math.floor(this.application.ninja.timeline.millisecondsOffset / 80) * position; newTweenToInsert.tweenData.tweenID = this.tweens.length; newTweenToInsert.tweenData.spanPosition = position - newTweenToInsert.tweenData.spanWidth; newTweenToInsert.tweenData.tweenedProperties = []; newTweenToInsert.tweenData.easing = "none"; newTweenToInsert.tweenData.initSelect = true; newTweenToInsert.tweenData.tweenedProperties["top"] = this.animatedElement.offsetTop + "px"; newTweenToInsert.tweenData.tweenedProperties["left"] = this.animatedElement.offsetLeft + "px"; newTweenToInsert.tweenData.tweenedProperties["width"] = this.animatedElement.offsetWidth + "px"; newTweenToInsert.tweenData.tweenedProperties["height"] = this.animatedElement.offsetHeight + "px"; this.tweens.splice(splitTweenIndex, 0, newTweenToInsert); i = tweensLength; } } this.application.ninja.currentDocument.model.needsSave = true; for (i = 0; i <= tweensLength+1; i++) { this.tweens[i].tweenID = i; this.tweens[i].tweenData.tweenID = i; } } }, retrieveStoredTweens:{ value:function () { var percentValue, fraction, splitValue,offsetAttribute,topOffSetAttribute,leftOffsetAttribute,widthOffsetAttribute,heightOffsetAttribute; var currentMilliSec,currentMilliSecPerPixel,clickPosition,tempTiming,tempTimingFloat,trackTiming,i = 0; var selectedIndex = this.application.ninja.timeline.getLayerIndexByID(this.trackID); this.application.ninja.timeline.arrLayers[selectedIndex].layerData.created=true; this.animatedElement = this.application.ninja.timeline.arrLayers[selectedIndex].layerData.stageElement; if(this.animatedElement!==undefined){ this.animationName = this.application.ninja.stylesController.getElementStyle(this.animatedElement, "-webkit-animation-name"); if(this.animationName){ // check for multiple animation names var animationNameList = this.animationName.split(","); if (animationNameList.length > 1) { this.animationNamesString = this.animationName; this.animationName = animationNameList[0]; this.getAllAnimationRules(animationNameList); } else { this.animationNamesString = this.animationName; } trackTiming = this.application.ninja.stylesController.getElementStyle(this.animatedElement, "-webkit-animation-duration"); this.nextKeyframe = 0; this.currentKeyframeRule = this.application.ninja.stylesController.getAnimationRuleWithName(this.animationName, this.application.ninja.currentDocument.model.views.design.document); for (i =0; this.currentKeyframeRule[i] ;i++) { var newTween = {}; newTween.tweenData = {}; var j, styleLength = this.currentKeyframeRule[i].style.length, keyframeStyles = []; for (j = 0; j < styleLength; j++) { var currProp = this.currentKeyframeRule[i].style[j]; var propVal = this.currentKeyframeRule[i].style[currProp]; keyframeStyles.push([currProp, propVal]); } // recreate tween properties array for timeline tween newTween.tweenData.tweenedProperties = []; for(var k in keyframeStyles){ newTween.tweenData.tweenedProperties[keyframeStyles[k][0]] = keyframeStyles[k][1]; } if (this.currentKeyframeRule[i].keyText === "0%") { newTween.tweenData.spanWidth = 0; newTween.tweenData.keyFramePosition = 0; newTween.tweenData.keyFrameMillisec = 0; newTween.tweenData.tweenID = 0; newTween.tweenData.spanPosition = 0; newTween.tweenData.initSelect = false; this.tweens.push(newTween); this.createMatchingPositionSizeTween(newTween); } else { tempTiming = trackTiming.split("s"); tempTimingFloat = parseFloat(tempTiming[0]); this.trackDuration = tempTimingFloat *1000; percentValue = this.currentKeyframeRule[i].keyText; splitValue = percentValue.split("%"); fraction = splitValue[0] / 100; currentMilliSec = fraction * this.trackDuration; currentMilliSecPerPixel = Math.floor(this.application.ninja.timeline.millisecondsOffset / 80); clickPosition = currentMilliSec / currentMilliSecPerPixel; newTween.tweenData.spanWidth = clickPosition - this.tweens[this.tweens.length - 1].tweenData.keyFramePosition; newTween.tweenData.keyFramePosition = clickPosition; newTween.tweenData.keyFrameMillisec = currentMilliSec; newTween.tweenData.tweenID = this.nextKeyframe; newTween.tweenData.initSelect = false; newTween.tweenData.spanPosition =clickPosition - newTween.tweenData.spanWidth; newTween.tweenData.easing = this.currentKeyframeRule[i].style.webkitAnimationName; if (newTween.tweenData.easing == "") { newTween.tweenData.easing = "none"; } this.tweens.push(newTween); this.createMatchingPositionSizeTween(newTween); } this.nextKeyframe += 1; } this.isTrackAnimated = true; } } } }, getAllAnimationRules:{ value:function(ruleNames){ ruleNames.splice(0,1); // temp remove first animation already retrieved for main track for(var i in ruleNames){ var currentName = ruleNames[i].replace(/^\s+|\s+$/g,""); // trim whitespace var currentRule = this.application.ninja.stylesController.getAnimationRuleWithName(currentName, this.application.ninja.currentDocument._document); this.ruleList[currentName] = currentRule; } this.recreatePropertyTracks(this.ruleList); } }, recreatePropertyTracks:{ value:function(ruleSet){ for(var i in ruleSet){ var styleProp = ruleSet[i][0].style[0]; this.application.ninja.timeline.layerRepetition.childComponents[0].addStyle(styleProp, ruleSet[i]); } } }, addAnimationRuleToElement:{ value:function (tweenEvent) { this.tweens[0].tweenData.tweenedProperties["top"] = this.animatedElement.offsetTop + "px"; this.tweens[0].tweenData.tweenedProperties["left"] = this.animatedElement.offsetLeft + "px"; this.tweens[0].tweenData.tweenedProperties["width"] = this.animatedElement.offsetWidth + "px"; this.tweens[0].tweenData.tweenedProperties["height"] = this.animatedElement.offsetHeight + "px"; var animationDuration = Math.round(this.trackDuration / 1000) + "s"; this.animationName = this.animatedElement.classList[0] + "_PositionSize"; if(this.animationNamesString.length == 0){ this.animationNamesString = this.animationName; } else { this.animationNamesString = this.animationName + ", " + this.animationNamesString; } this.ninjaStylesContoller.setElementStyle(this.animatedElement, "-webkit-animation-name", this.animationNamesString); this.ninjaStylesContoller.setElementStyle(this.animatedElement, "-webkit-animation-duration", animationDuration); this.ninjaStylesContoller.setElementStyle(this.animatedElement, "-webkit-animation-fill-mode", "forwards"); this.ninjaStylesContoller.setElementStyle(this.animatedElement, "-webkit-animation-iteration-count", 1); var initRule = "@-webkit-keyframes " + this.animationName + " { 0% {top: " + this.animatedElement.offsetTop + "px; left: " + this.animatedElement.offsetLeft + "px;} 100% {top: " + this.animatedElement.offsetTop + "px; left: " + this.animatedElement.offsetLeft + "px;} }"; this.currentKeyframeRule = this.ninjaStylesContoller.addRule(initRule); this.insertTween(tweenEvent.offsetX); this.isTrackAnimated = true; } }, updateKeyframeRule:{ value:function () { this.ninjaStylesContoller.deleteRule(this.currentKeyframeRule); var keyframeString = "@-webkit-keyframes " + this.animationName + " {"; for (var i = 0; i < this.tweens.length; i++) { var keyMill = parseInt(this.tweens[i].tweenData.keyFrameMillisec); // trackDur should be parseFloat rounded to significant digits var trackDur = parseInt(this.trackDuration); var keyframePercent = Math.round((keyMill / trackDur) * 100) + "%"; var keyframePropertyString = " " + keyframePercent + " {"; for(var prop in this.tweens[i].tweenData.tweenedProperties){ keyframePropertyString += prop + ": " + this.tweens[i].tweenData.tweenedProperties[prop] + ";"; } keyframePropertyString += "}"; keyframeString += keyframePropertyString; } keyframeString += " }"; this.currentKeyframeRule = this.ninjaStylesContoller.addRule(keyframeString); this.application.ninja.currentDocument.model.needsSave = true; } }, createMatchingPositionSizeTween:{ value:function (newTween) { var i; var posTracks = this.positionTracksRepetition.childComponents.length; for (i = 0; i < posTracks; i++) { this.positionTracksRepetition.childComponents[i].propTweens.push(newTween); } } }, findXOffset:{ value:function (obj) { var curleft = 0; if (typeof(obj) === "undefined") { } if (obj.offsetParent) { do { curleft += (obj.offsetLeft - obj.scrollLeft); } while (obj = obj.offsetParent); } return curleft; } }, getTweenIndexById: { value: function(intID) { var i = 0, arrTweensLength = this.tweens.length; for (i = 0; i < arrTweensLength; i++) { if (this.tweens[i].tweenData.tweenID === intID) { returnVal = i; } } return returnVal; } } // ==== End Controllers });