$axure.internal(function($ax) { var _actionHandlers = {}; var _action = $ax.action = {}; var queueTypes = _action.queueTypes = { none: 0, move: 1, setState: 2, fade: 3, resize: 4, rotate: 5 }; var animationQueue = {}; // using array as the key doesn't play nice var nextAnimationId = 1; var animationsToCount = {}; var actionToActionGroups = {}; var getAnimation = function(id, type) { return animationQueue[id] && animationQueue[id][type] && animationQueue[id][type][0]; }; var _addAnimation = _action.addAnimation = function (id, type, func, suppressFire) { var wasEmpty = !getAnimation(id, type); // Add the func to the queue. Create the queue if necessary. var idQueue = animationQueue[id]; if(!idQueue) animationQueue[id] = idQueue = {}; var queue = idQueue[type]; if(!queue) idQueue[type] = queue = []; queue[queue.length] = func; // If it was empty, there isn't a callback waiting to be called on this. You have to fire it manually. // If this is waiting on something, suppress it, and it will fire when it's ready if(wasEmpty && !suppressFire) func(); }; var _addAnimations = function (animations) { if(animations.length == 1) { _addAnimation(animations[0].id, animations[0].type, animations[0].func); return; } var allReady = true; var readyCount = 0; for(var i = 0; i < animations.length; i++) { var animation = animations[i]; var thisReady = !getAnimation(animation.id, animation.type); allReady = allReady && thisReady; if (thisReady) readyCount++; else { var typeToGroups = actionToActionGroups[animation.id]; if (!typeToGroups) actionToActionGroups[animation.id] = typeToGroups = {}; var groups = typeToGroups[animation.type]; if (!groups) typeToGroups[animation.type] = groups = []; groups[groups.length] = animations; } } for(i = 0; i < animations.length; i++) { animation = animations[i]; _addAnimation(animation.id, animation.type, animation.func, true); } if (allReady) { for (i = 0; i < animations.length; i++) animations[i].func(); } else { animations.id = nextAnimationId++; animationsToCount[animations.id] = readyCount; } } var _fireAnimationFromQueue = _action.fireAnimationFromQueue = function (id, type) { // Remove the function that was just fired if (animationQueue[id] && animationQueue[id][type]) $ax.splice(animationQueue[id][type], 0, 1); // Fire the next func if there is one var func = getAnimation(id, type); if(func && !_checkFireActionGroup(id, type, func)) func(); }; var _checkFireActionGroup = function(id, type, func) { var group = actionToActionGroups[id]; group = group && group[type]; if (!group || group.length == 0) return false; var animations = group[0]; var found = false; for (var i = 0; i < animations.length; i++) { var animation = animations[i]; if (animation.id == id && animation.type == type) { found = func == animation.func; break; } } // if found then update this action group, otherwise, keep waiting for right action to fire if(!found) return false; $ax.splice(group, 0, 1); var count = animationsToCount[animations.id] + 1; if(count != animations.length) { animationsToCount[animations.id] = count; return true; } delete animationsToCount[animations.id]; // Funcs is needed because an earlier func can try to cascade right away (when no animation for example) and will kill this func and move on to the // next one (which may not even exist). If we get all funcs before calling any, then we know they are all the func we want. var funcs = []; for(i = 0; i < animations.length; i++) { animation = animations[i]; funcs.push(getAnimation(animation.id, animation.type)); } for(i = 0; i < funcs.length; i++) { funcs[i](); } return true; } var _refreshing = []; _action.refreshStart = function(repeaterId) { _refreshing.push(repeaterId); }; _action.refreshEnd = function() { _refreshing.pop(); }; // TODO: [ben] Consider moving this to repeater.js var _repeatersToRefresh = _action.repeatersToRefresh = []; var _ignoreAction = function(repeaterId) { for(var i = 0; i < _refreshing.length; i++) if(_refreshing[i] == repeaterId) return true; return false; }; var _addRefresh = function(repeaterId) { if(_repeatersToRefresh.indexOf(repeaterId) == -1) _repeatersToRefresh.push(repeaterId); }; var _getIdToResizeMoveState = function(eventInfo) { if(!eventInfo.idToResizeMoveState) eventInfo.idToResizeMoveState = {}; return eventInfo.idToResizeMoveState; } var _queueResizeMove = function (id, type, eventInfo, actionInfo) { if (type == queueTypes.resize || type == queueTypes.rotate) $ax.public.fn.convertToSingleImage($jobj(id)); var idToResizeMoveState = _getIdToResizeMoveState(eventInfo); if(!idToResizeMoveState[id]) { idToResizeMoveState[id] = {}; idToResizeMoveState[id][queueTypes.move] = { queue: [], used: 0 }; idToResizeMoveState[id][queueTypes.resize] = { queue: [], used: 0 }; idToResizeMoveState[id][queueTypes.rotate] = { queue: [], used: 0 }; } var state = idToResizeMoveState[id]; // If this is not a type being queued (no action of it's type waiting already) then if it is an instant, fire right away. var myOptions = type == queueTypes.resize ? actionInfo : actionInfo.options; if(!state[type].queue.length && (!myOptions.easing || myOptions.easing == 'none' || !myOptions.duration)) { var func = type == queueTypes.resize ? _addResize : type == queueTypes.rotate ? _addRotate : _addMove; func(id, eventInfo, actionInfo, { easing: 'none', duration: 0, stop: { instant: true } }); return; } // Check other 2 types to see if either is empty, if so, we can't do anything, so just queue it up var otherType1 = type == queueTypes.move ? queueTypes.resize : queueTypes.move; var otherType2 = type == queueTypes.rotate ? queueTypes.resize : queueTypes.rotate; if (!state[otherType1].queue.length || !state[otherType2].queue.length) { state[type].queue.push({ eventInfo: eventInfo, actionInfo: actionInfo }); } else { var duration = myOptions.duration; var used1 = state[otherType1].used; var used2 = state[otherType2].used; while(state[otherType1].queue.length && state[otherType2].queue.length && duration != 0) { var other1 = state[otherType1].queue[0]; var otherOptions1 = otherType1 == queueTypes.resize ? other1.actionInfo : other1.actionInfo.options; // If queue up action is a non animation, then don't combo it, just queue it and move on if(!otherOptions1.easing || otherOptions1.easing == 'none' || !otherOptions1.duration) { func = otherType1 == queueTypes.resize ? _addResize : otherType1 == queueTypes.rotate ? _addRotate : _addMove; func(id, eventInfo, actionInfo, { easing: 'none', duration: 0, stop: { instant: true } }); continue; } var other2 = state[otherType2].queue[0]; var otherOptions2 = otherType2 == queueTypes.resize ? other2.actionInfo : other2.actionInfo.options; // If queue up action is a non animation, then don't combo it, just queue it and move on if(!otherOptions2.easing || otherOptions2.easing == 'none' || !otherOptions2.duration) { func = otherType2 == queueTypes.resize ? _addResize : otherType2 == queueTypes.rotate ? _addRotate : _addMove; func(id, eventInfo, actionInfo, { easing: 'none', duration: 0, stop: { instant: true } }); continue; } // Other duration is what is left over. When in queue it may be partly finished already var otherDuration1 = otherOptions1.duration - used1; var otherDuration2 = otherOptions2.duration - used2; var resizeInfo = type == queueTypes.resize ? actionInfo : otherType1 == queueTypes.resize ? other1.actionInfo : other2.actionInfo; var rotateInfo = type == queueTypes.rotate ? actionInfo : otherType1 == queueTypes.rotate ? other1.actionInfo : other2.actionInfo; var moveInfo = type == queueTypes.move ? actionInfo : otherType1 == queueTypes.move ? other1.actionInfo : other2.actionInfo; var options = { easing: moveInfo.options.easing, duration: Math.min(duration, otherDuration1, otherDuration2) }; // Start for self is whole duration - duration left, end is start plus duration of combo to be queued, length is duration var stop = { start: myOptions.duration - duration, len: myOptions.duration }; stop.end = stop.start + options.duration; // Start for other is used (will be 0 after 1st round), end is start plus length is duration of combo to be queued, length is other duration var otherStop1 = { start: used1, end: options.duration + used1, len: otherOptions1.duration }; var otherStop2 = { start: used2, end: options.duration + used2, len: otherOptions2.duration }; options.stop = type == queueTypes.resize ? stop : otherType1 == queueTypes.resize ? otherStop1 : otherStop2; options.moveStop = type == queueTypes.move ? stop : otherType1 == queueTypes.move ? otherStop1 : otherStop2; options.rotateStop = type == queueTypes.rotate ? stop : otherType1 == queueTypes.rotate ? otherStop1 : otherStop2; _addResize(id, eventInfo, resizeInfo, options, moveInfo, rotateInfo); // Update duration for this animation duration -= options.duration; // For others update used and remove from queue if necessary if(otherDuration1 == options.duration) { $ax.splice(state[otherType1].queue, 0, 1); used1 = 0; } else used1 += options.duration; if(otherDuration2 == options.duration) { $ax.splice(state[otherType2].queue, 0, 1); used2 = 0; } else used2 += options.duration; } // Start queue for new type if necessary if(duration) { state[type].queue.push({ eventInfo: eventInfo, actionInfo: actionInfo }); state[type].used = myOptions.duration - duration; } // Update used for others state[otherType1].used = used1; state[otherType2].used = used2; } }; _action.flushAllResizeMoveActions = function (eventInfo) { var idToResizeMoveState = _getIdToResizeMoveState(eventInfo); for(var id in idToResizeMoveState) _flushResizeMoveActions(id, idToResizeMoveState); }; var _flushResizeMoveActions = function(id, idToResizeMoveState) { var state = idToResizeMoveState[id]; var move = state[queueTypes.move]; var moveInfo = move.queue[0]; var resize = state[queueTypes.resize]; var resizeInfo = resize.queue[0]; var rotate = state[queueTypes.rotate]; var rotateInfo = rotate.queue[0]; while (moveInfo || resizeInfo || rotateInfo) { var eventInfo = moveInfo ? moveInfo.eventInfo : resizeInfo ? resizeInfo.eventInfo : rotateInfo.eventInfo; moveInfo = moveInfo && moveInfo.actionInfo; resizeInfo = resizeInfo && resizeInfo.actionInfo; rotateInfo = rotateInfo && rotateInfo.actionInfo; // Resize is used by default, then rotate if(resizeInfo) { // Check for instant resize if(!resizeInfo.duration || resizeInfo.easing == 'none') { _addResize(id, resize.queue[0].eventInfo, resizeInfo, { easing: 'none', duration: 0, stop: { instant: true } }); _updateResizeMoveUsed(id, queueTypes.resize, 0, idToResizeMoveState); resizeInfo = resize.queue[0]; continue; } var duration = resizeInfo.duration - resize.used; if(moveInfo) duration = Math.min(duration, moveInfo.options.duration - move.used); if(rotateInfo) duration = Math.min(duration, rotateInfo.options.duration - rotate.used); var baseOptions = moveInfo ? moveInfo.options : resizeInfo; var options = { easing: baseOptions.easing, duration: duration }; options.stop = { start: resize.used, end: resize.used + duration, len: resizeInfo.duration }; if(moveInfo) options.moveStop = { start: move.used, end: move.used + duration, len: moveInfo.options.duration }; if(rotateInfo) options.rotateStop = { start: rotate.used, end: rotate.used + duration, len: rotateInfo.options.duration }; _addResize(id, eventInfo, resizeInfo, options, moveInfo, rotateInfo); _updateResizeMoveUsed(id, queueTypes.resize, duration, idToResizeMoveState); resizeInfo = resize.queue[0]; if(rotateInfo) { _updateResizeMoveUsed(id, queueTypes.rotate, duration, idToResizeMoveState); rotateInfo = rotate.queue[0]; } if(moveInfo) { _updateResizeMoveUsed(id, queueTypes.move, duration, idToResizeMoveState); moveInfo = move.queue[0]; } } else if (rotateInfo) { // Check for instant rotate if(!rotateInfo.options.duration || rotateInfo.options.easing == 'none') { _addRotate(id, rotate.queue[0].eventInfo, rotateInfo, { easing: 'none', duration: 0, stop: { instant: true } }); _updateResizeMoveUsed(id, queueTypes.rotate, 0, idToResizeMoveState); rotateInfo = rotate.queue[0]; continue; } duration = rotateInfo.options.duration - rotate.used; if(moveInfo) duration = Math.min(duration, moveInfo.options.duration - move.used); baseOptions = moveInfo ? moveInfo.options : rotateInfo.options; options = { easing: baseOptions.easing, duration: duration }; options.stop = { start: rotate.used, end: rotate.used + duration, len: rotateInfo.options.duration }; if(moveInfo) options.moveStop = { start: move.used, end: move.used + duration, len: moveInfo.options.duration }; _addRotate(id, eventInfo, rotateInfo, options, moveInfo); _updateResizeMoveUsed(id, queueTypes.rotate, duration, idToResizeMoveState); rotateInfo = rotate.queue[0]; if(moveInfo) { _updateResizeMoveUsed(id, queueTypes.move, duration, idToResizeMoveState); moveInfo = move.queue[0]; } } else { if(!moveInfo.options.duration || moveInfo.options.easing == 'none') { _addMove(id, eventInfo, moveInfo, { easing: 'none', duration: 0, stop: { instant: true } }); _updateResizeMoveUsed(id, queueTypes.move, 0, idToResizeMoveState); moveInfo = move.queue[0]; continue; } duration = moveInfo.options.duration - move.used; options = { easing: moveInfo.options.easing, duration: duration }; options.stop = { start: move.used, end: moveInfo.options.duration, len: moveInfo.options.duration }; _addMove(id, eventInfo, moveInfo, options); _updateResizeMoveUsed(id, queueTypes.move, duration, idToResizeMoveState); moveInfo = move.queue[0]; } } }; var _updateResizeMoveUsed = function(id, type, duration, idToResizeMoveState) { var state = idToResizeMoveState[id][type]; state.used += duration; var options = state.queue[0].actionInfo; if(options.options) options = options.options; var optionDur = (options.easing && options.easing != 'none' && options.duration) || 0; if(optionDur <= state.used) { $ax.splice(state.queue, 0, 1); state.used = 0; } } var _dispatchAction = $ax.action.dispatchAction = function(eventInfo, actions, currentIndex) { currentIndex = currentIndex || 0; //If no actions, you can bubble if(currentIndex >= actions.length) return; //actions are responsible for doing their own dispatching _actionHandlers[actions[currentIndex].action](eventInfo, actions, currentIndex); }; _actionHandlers.wait = function(eventInfo, actions, index) { var action = actions[index]; var infoCopy = $ax.eventCopy(eventInfo); window.setTimeout(function() { infoCopy.now = new Date(); infoCopy.idToResizeMoveState = undefined; _dispatchAction(infoCopy, actions, index + 1); _action.flushAllResizeMoveActions(infoCopy); }, action.waitTime); }; _actionHandlers.expr = function(eventInfo, actions, index) { var action = actions[index]; $ax.expr.evaluateExpr(action.expr, eventInfo); //this should be a block _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.setFunction = _actionHandlers.expr; _actionHandlers.linkWindow = function(eventInfo, actions, index) { linkActionHelper(eventInfo, actions, index); }; _actionHandlers.closeCurrent = function(eventInfo, actions, index) { $ax.closeWindow(); _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.linkFrame = function(eventInfo, actions, index) { linkActionHelper(eventInfo, actions, index); }; _actionHandlers.setAdaptiveView = function(eventInfo, actions, index) { var action = actions[index]; var view = action.setAdaptiveViewTo; if(view) $ax.adaptive.setAdaptiveView(view); }; var linkActionHelper = function(eventInfo, actions, index) { var action = actions[index]; eventInfo.link = true; if(action.linkType != 'frame') { var includeVars = _includeVars(action.target, eventInfo); if(action.target.targetType == "reloadPage") { $ax.reload(action.target.includeVariables); } else if(action.target.targetType == "backUrl") { $ax.back(); } var url = action.target.url; if(!url && action.target.urlLiteral) { url = $ax.expr.evaluateExpr(action.target.urlLiteral, eventInfo, true); } if(url) { let useStartHtml = shouldUseStartHtml(action); if(useStartHtml) { //use start.html to load player url = urlWithStartHtml(url); //collapse player for popup if(action.linkType == "popup") url = urlWithCollapseSitemap(url); } //set useGlobalVarNameInUrl to true to use GLOBAL_VAR_NAME in the url, so player knows how to parse it //without this, we are assuming everything after '#' are global vars if(action.linkType == "popup") { $ax.navigate({ url: url, target: action.linkType, includeVariables: includeVars, popupOptions: action.popup, useGlobalVarNameInUrl : useStartHtml }); } else { $ax.navigate({ url: url, target: action.linkType, includeVariables: includeVars, useGlobalVarNameInUrl : useStartHtml }); } } } else linkFrame(eventInfo, action); eventInfo.link = false; _dispatchAction(eventInfo, actions, index + 1); }; //use start.html will add a player to the prototype var shouldUseStartHtml = function(linkAction) { return linkAction.target.targetType == 'page' //only adding player for page, not external links && (linkAction.linkType == "popup" || linkAction.linkType == "new") //only add for popup and new tabs && $axure.utils.isInPlayer() //allow user to view without player (maybe useful for user testing) && !$axure.utils.isShareApp() //share app use special handling on its link, add start.html breaks the handling } var urlWithStartHtml = function(url) { var pageName = url.substring(0, url.lastIndexOf('.html')); var pageHash = $axure.utils.setHashStringVar(START_URL_NAME, PAGE_URL_NAME, pageName); return START_URL_NAME + pageHash; } var urlWithCollapseSitemap = function(url) { return url + '&' + SITEMAP_COLLAPSE_VAR_NAME + '=' + SITEMAP_COLLAPSE_VALUE; } var _includeVars = function(target, eventInfo) { if(target.includeVariables) return true; // If it is a url literal, that is a string literal, that has only 1 sto, that is an item that is a page, include vars. if(target.urlLiteral) { var literal = target.urlLiteral; var sto = literal.stos[0]; if(literal.exprType == 'stringLiteral' && literal.value.indexOf('[[') == 0 && literal.value.indexOf(']]' == literal.value.length - 2) && literal.stos.length == 1 && sto.sto == 'item' && eventInfo.item) { var data = $ax.repeater.getData(eventInfo, eventInfo.item.repeater.elementId, eventInfo.item.index, sto.name, 'data'); if (data && $ax.public.fn.IsPage(data.type)) return true; } } return false; }; var linkFrame = function(eventInfo, action) { for(var i = 0; i < action.framesToTargets.length; i++) { var framePath = action.framesToTargets[i].framePath; var target = action.framesToTargets[i].target; var includeVars = _includeVars(target, eventInfo); var url = target.url; if(!url && target.urlLiteral) { url = $ax.expr.evaluateExpr(target.urlLiteral, eventInfo, true); } var id = $ax.getElementIdsFromPath(framePath, eventInfo)[0]; if(id) $ax('#' + $ax.INPUT(id)).openLink(url, includeVars); } }; var _repeatPanelMap = {}; _actionHandlers.setPanelState = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.panelsToStates.length; i++) { var panelToState = action.panelsToStates[i]; var stateInfo = panelToState.stateInfo; var elementIds = $ax.getElementIdsFromPath(panelToState.panelPath, eventInfo); for(var j = 0; j < elementIds.length; j++) { var elementId = elementIds[j]; // Need new scope for elementId and info (function(elementId, stateInfo) { _addAnimation(elementId, queueTypes.setState, function() { var stateNumber = stateInfo.stateNumber; if(stateInfo.setStateType == "value") { var oldTarget = eventInfo.targetElement; eventInfo.targetElement = elementId; var stateName = $ax.expr.evaluateExpr(stateInfo.stateValue, eventInfo); eventInfo.targetElement = oldTarget; // Try for state name first var states = $ax.getObjectFromElementId(elementId).diagrams; var stateNameFound = false; for(var k = 0; k < states.length; k++) { if(states[k].label == stateName) { stateNumber = k + 1; stateNameFound = true; } } // Now check for index if(!stateNameFound) { stateNumber = Number(stateName); var panelCount = $('#' + elementId).children().length; // Make sure number is not NaN, is in range, and is a whole number. // Wasn't a state name or number, so return if(isNaN(stateNumber) || stateNumber <= 0 || stateNumber > panelCount || Math.round(stateNumber) != stateNumber) return _fireAnimationFromQueue(elementId, queueTypes.setState); } } else if(stateInfo.setStateType == 'next' || stateInfo.setStateType == 'previous') { var info = $ax.deepCopy(stateInfo); var repeat = info.repeat; // Only map it, if repeat exists. if(typeof (repeat) == 'number') _repeatPanelMap[elementId] = info; return _progessPanelState(elementId, info, info.repeatSkipFirst); } delete _repeatPanelMap[elementId]; // If setting to current (to stop repeat) break here if(stateInfo.setStateType == 'current') return _fireAnimationFromQueue(elementId, queueTypes.setState); $ax('#' + elementId).SetPanelState(stateNumber, stateInfo.options, stateInfo.showWhenSet); }); })(elementId, stateInfo); } } _dispatchAction(eventInfo, actions, index + 1); }; var _progessPanelState = function(id, info, skipFirst) { var direction = info.setStateType; var loop = info.loop; var repeat = info.repeat; var options = info.options; var hasRepeat = typeof (repeat) == 'number'; var currentStateId = $ax.visibility.GetPanelState(id); var stateNumber = ''; if(currentStateId != '') { currentStateId = $ax.repeater.getScriptIdFromElementId(currentStateId); var currentStateNumber = Number(currentStateId.substr(currentStateId.indexOf('state') + 5)); if(direction == "next") { stateNumber = currentStateNumber + 2; if(stateNumber > $ax.visibility.GetPanelStateCount(id)) { if(loop) stateNumber = 1; else { delete _repeatPanelMap[id]; return _fireAnimationFromQueue(id, queueTypes.setState); } } } else if(direction == "previous") { stateNumber = currentStateNumber; if(stateNumber <= 0) { if(loop) stateNumber = $ax.visibility.GetPanelStateCount(id); else { delete _repeatPanelMap[id]; return _fireAnimationFromQueue(id, queueTypes.setState); } } } if(hasRepeat && _repeatPanelMap[id] != info) return _fireAnimationFromQueue(id, queueTypes.setState); if (!skipFirst) $ax('#' + id).SetPanelState(stateNumber, options, info.showWhenSet); else _fireAnimationFromQueue(id, queueTypes.setState); if(hasRepeat) { var animate = options && options.animateIn; if(animate && animate.easing && animate.easing != 'none' && animate.duration > repeat) repeat = animate.duration; animate = options && options.animateOut; if(animate && animate.easing && animate.easing != 'none' && animate.duration > repeat) repeat = animate.duration; window.setTimeout(function() { // Either new repeat, or no repeat anymore. if(_repeatPanelMap[id] != info) return; _addAnimation(id, queueTypes.setState, function() { _progessPanelState(id, info, false); }); }, repeat); } else delete _repeatPanelMap[id]; } }; _actionHandlers.fadeWidget = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.objectsToFades.length; i++) { var fadeInfo = action.objectsToFades[i].fadeInfo; var elementIds = $ax.getElementIdsFromPath(action.objectsToFades[i].objectPath, eventInfo); for(var j = 0; j < elementIds.length; j++) { var elementId = elementIds[j]; // Need new scope for elementId and info (function(elementId, fadeInfo) { _addAnimation(elementId, queueTypes.fade, function() { if(fadeInfo.fadeType == "hide") { $ax('#' + elementId).hide(fadeInfo.options); } else if(fadeInfo.fadeType == "show") { $ax('#' + elementId).show(fadeInfo.options, eventInfo); } else if(fadeInfo.fadeType == "toggle") { $ax('#' + elementId).toggleVisibility(fadeInfo.options); } }); })(elementId, fadeInfo); } } _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.setOpacity = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.objectsToSetOpacity.length; i++) { var opacityInfo = action.objectsToSetOpacity[i].opacityInfo; var elementIds = $ax.getElementIdsFromPath(action.objectsToSetOpacity[i].objectPath, eventInfo); for(var j = 0; j < elementIds.length; j++) { var elementId = elementIds[j]; (function(elementId, opacityInfo) { _addAnimation(elementId, queueTypes.fade, function () { var oldTarget = eventInfo.targetElement; eventInfo.targetElement = elementId; var opacity = $ax.expr.evaluateExpr(opacityInfo.opacity, eventInfo); eventInfo.targetElement = oldTarget; opacity = Math.min(100, Math.max(0, opacity)); $ax('#' + elementId).setOpacity(opacity/100, opacityInfo.easing, opacityInfo.duration); }) })(elementId, opacityInfo); } } _dispatchAction(eventInfo, actions, index + 1); } _actionHandlers.moveWidget = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.objectsToMoves.length; i++) { var moveInfo = action.objectsToMoves[i].moveInfo; var elementIds = $ax.getElementIdsFromPath(action.objectsToMoves[i].objectPath, eventInfo); for(var j = 0; j < elementIds.length; j++) { var elementId = elementIds[j]; _queueResizeMove(elementId, queueTypes.move, eventInfo, moveInfo); //_addMove(eventInfo, elementId, moveInfo, eventInfo.dragInfo); } } _dispatchAction(eventInfo, actions, index + 1); }; //var _compoundChildrenShallow = function (id) { // var deep = []; // var children = $ax('#' + id).getChildren()[0].children; // var piecePrefix = id + 'p'; // for (var i = 0; i < children.length; i++) { // if(children[i].substring(0, id.length + 1) == piecePrefix) { // deep.push(children[i]); // } // } // return deep; //}; var _addMove = function (elementId, eventInfo, moveInfo, optionsOverride) { var eventInfoCopy = $ax.eventCopy(eventInfo); var idToResizeMoveState = _getIdToResizeMoveState(eventInfoCopy); eventInfoCopy.targetElement = elementId; var options = $ax.deepCopy(moveInfo.options); options.easing = optionsOverride.easing; options.duration = optionsOverride.duration; options.dragInfo = eventInfo.dragInfo; if($ax.public.fn.IsLayer($obj(elementId).type)) { var childrenIds = $ax.public.fn.getLayerChildrenDeep(elementId, true); if(childrenIds.length == 0) return; var animations = []; // Get move delta once, then apply to all children animations.push({ id: elementId, type: queueTypes.move, func: function () { var layerInfo = $ax('#' + elementId).offsetBoundingRect(); //var layerInfo = $ax.public.fn.getWidgetBoundingRect(elementId); var deltaLoc = _getMoveLoc(elementId, moveInfo, eventInfoCopy, optionsOverride.stop, idToResizeMoveState[elementId], options, layerInfo); // $ax.event.raiseSyntheticEvent(elementId, "onMove"); $ax.visibility.pushContainer(elementId, false); options.onComplete = function () { _fireAnimationFromQueue(elementId, queueTypes.move); $ax.visibility.popContainer(elementId, false); }; $ax('#' + elementId).moveBy(deltaLoc.x, deltaLoc.y, options); } }); //for(var i = 0; i < childrenIds.length; i++) { // (function(childId) { // animations.push({ // id: childId, // type: queueTypes.move, // func: function () { // // Nop, while trying to move as container // //$ax.event.raiseSyntheticEvent(childId, "onMove"); // //if($ax.public.fn.IsLayer($obj(childId).type)) _fireAnimationFromQueue(childId, queueTypes.move); // //else $ax('#' + childId).moveBy(deltaLoc.x, deltaLoc.y, moveInfo.options); // } // }); // })(childrenIds[i]); //} _addAnimations(animations); } else { _addAnimation(elementId, queueTypes.move, function() { var loc = _getMoveLoc(elementId, moveInfo, eventInfoCopy, optionsOverride.stop, idToResizeMoveState[elementId], options); // $ax.event.raiseSyntheticEvent(elementId, "onMove"); if(loc.moveTo) $ax('#' + elementId).moveTo(loc.x, loc.y, options); else $ax('#' + elementId).moveBy(loc.x, loc.y, options); }); } }; var _moveSingleWidget = function (elementId, delta, options, onComplete) { if(!delta.x && !delta.y) { $ax.action.fireAnimationFromQueue(elementId, $ax.action.queueTypes.move); if (onComplete) onComplete(); return; } var fixedInfo = $ax.dynamicPanelManager.getFixedInfo(elementId); var xProp = 'left'; var xDiff = '+='; if(fixedInfo) { if(fixedInfo.horizontal == 'right') { xProp = 'right'; xDiff = '-='; } else if(fixedInfo.horizontal == 'center') { xProp = 'margin-left'; } } var yProp = 'top'; var yDiff = '+='; if(fixedInfo) { if(fixedInfo.vertical == 'bottom') { yProp = 'bottom'; yDiff = '-='; } else if(fixedInfo.vertical == 'middle') { yProp = 'margin-top'; } } var css = {}; css[xProp] = xDiff + delta.x; css[yProp] = yDiff + delta.y; $ax.visibility.moveMovedLocation(elementId, delta.x, delta.y); var moveInfo = $ax.move.PrepareForMove(elementId, delta.x, delta.y,false, options); $jobjAll(elementId).animate(css, { duration: options.duration, easing: options.easing, queue: false, complete: function () { if(onComplete) onComplete(); if(moveInfo.rootLayer) $ax.visibility.popContainer(moveInfo.rootLayer, false); $ax.dynamicPanelManager.fitParentPanel(elementId); $ax.action.fireAnimationFromQueue(elementId, $ax.action.queueTypes.move); } }); } var _getMoveLoc = function (elementId, moveInfo, eventInfoCopy, stopInfo, comboState, options, layerInfo) { var moveTo = false; var moveWithThis = false; var xValue = 0; var yValue = 0; var moveResult = comboState.moveResult; var widgetDragInfo = eventInfoCopy.dragInfo; var jobj = $jobj(elementId); var startX; var startY; switch(moveInfo.moveType) { case "location": // toRatio is ignoring anything before start since that has already taken effect we just know whe have from start to len to finish // getting to the location we want to get to. var toRatio = stopInfo.instant ? 1 : (stopInfo.end - stopInfo.start) / (stopInfo.len - stopInfo.start); // If result already caluculated, don't recalculate again, other calculate and save if (moveResult) { xValue = moveResult.x; yValue = moveResult.y; } else { comboState.moveResult = moveResult = { x: $ax.expr.evaluateExpr(moveInfo.xValue, eventInfoCopy), y: $ax.expr.evaluateExpr(moveInfo.yValue, eventInfoCopy) }; xValue = moveResult.x; yValue = moveResult.y; } // If this is final stop for this move, then clear out the result so next move won't use it if(stopInfo.instant || stopInfo.end == stopInfo.len) comboState.moveResult = undefined; if (layerInfo) { startX = layerInfo.left; startY = layerInfo.top; //} else if ($ax.public.fn.isCompoundVectorHtml(jobj[0])) { // var dimensions = $ax.public.fn.compoundWidgetDimensions(jobj); // startX = dimensions.left; // startY = dimensions.top; } else { var offsetLocation = $ax('#' + elementId).offsetLocation(); startX = offsetLocation.left; startY = offsetLocation.top; //startX = $ax('#' + elementId).locRelativeIgnoreLayer(false); //startY = $ax('#' + elementId).locRelativeIgnoreLayer(true); if(jobj.css('position') == 'fixed') { startX -= $(window).scrollLeft(); startY -= $(window).scrollTop(); } } xValue = xValue == '' ? 0 : (xValue - startX) * toRatio; yValue = yValue == '' ? 0 : (yValue - startY) * toRatio; break; case "delta": var ratio = stopInfo.instant ? 1 : (stopInfo.end - stopInfo.start) / stopInfo.len; // See case location above if(moveResult) { xValue = moveResult.x * ratio; yValue = moveResult.y * ratio; } else { comboState.moveResult = moveResult = { x: $ax.expr.evaluateExpr(moveInfo.xValue, eventInfoCopy), y: $ax.expr.evaluateExpr(moveInfo.yValue, eventInfoCopy) }; xValue = moveResult.x * ratio; yValue = moveResult.y * ratio; } if (stopInfo.instant || stopInfo.end == stopInfo.len) comboState.moveResult = undefined; break; case "drag": xValue = widgetDragInfo.xDelta; yValue = widgetDragInfo.yDelta; break; case "dragX": xValue = widgetDragInfo.xDelta; yValue = 0; break; case "dragY": xValue = 0; yValue = widgetDragInfo.yDelta; break; case "locationBeforeDrag": var location = widgetDragInfo.movedWidgets[eventInfoCopy.targetElement]; if (location) { var axObj = $ax('#' + eventInfoCopy.targetElement); //This may require using the css value var viewportLocation = axObj.viewportLocation(); xValue = location.x - viewportLocation.left; yValue = location.y - viewportLocation.top; //xValue = location.x - axObj.left(); //yValue = location.y - axObj.top(); } else { _fireAnimationFromQueue(eventInfoCopy.srcElement, queueTypes.move); return { x: 0, y: 0 }; } //moveTo = true; break; case "withThis": moveWithThis = true; var widgetMoveInfo = $ax.move.GetWidgetMoveInfo(); var srcElementId = $ax.getElementIdsFromEventAndScriptId(eventInfoCopy, eventInfoCopy.srcElement)[0]; var delta = widgetMoveInfo[srcElementId]; options.easing = delta.options.easing; options.duration = delta.options.duration; xValue = delta.x; yValue = delta.y; break; } if (options && options.boundaryExpr) { //$ax.public.fn.removeCompound(jobj); //if(jobj.css('position') == 'fixed') { // //swap page coordinates with fixed coordinates // options.boundaryExpr.leftExpr.value = options.boundaryExpr.leftExpr.value.replace('.top', '.topfixed').replace('.left', '.leftfixed').replace('.bottom', '.bottomfixed').replace('.right', '.rightfixed'); // options.boundaryExpr.leftExpr.stos[0].leftSTO.prop = options.boundaryExpr.leftExpr.stos[0].leftSTO.prop + 'fixed'; // options.boundaryStos.boundaryScope.direcval0.value = options.boundaryStos.boundaryScope.direcval0.value.replace('.top', '.topfixed').replace('.left', '.leftfixed').replace('.bottom', '.bottomfixed').replace('.right', '.rightfixed'); // options.boundaryStos.boundaryScope.direcval0.stos[0].leftSTO.prop = options.boundaryStos.boundaryScope.direcval0.stos[0].leftSTO.prop + 'fixed'; //} if(moveWithThis && (xValue || yValue)) { _updateLeftExprVariable(options.boundaryExpr, xValue.toString(), yValue.toString()); } if(!$ax.expr.evaluateExpr(options.boundaryExpr, eventInfoCopy)) { var boundaryStoInfo = options.boundaryStos; if(boundaryStoInfo) { if(moveWithThis) { var stoScopes = boundaryStoInfo.boundaryScope; if(stoScopes) { for(var s in stoScopes) { var boundaryScope = stoScopes[s]; if(!boundaryScope.localVariables) continue; if(boundaryScope.localVariables.withx) boundaryScope.localVariables.withx.value = xValue.toString(); if(boundaryScope.localVariables.withy) boundaryScope.localVariables.withy.value = yValue.toString(); } } } if(layerInfo) { startX = layerInfo.left; startY = layerInfo.top; } else { offsetLocation = $ax('#' + elementId).offsetLocation(); startX = offsetLocation.left; startY = offsetLocation.top; //startX = $ax('#' + elementId).locRelativeIgnoreLayer(false); //startY = $ax('#' + elementId).locRelativeIgnoreLayer(true); if(jobj.css('position') == 'fixed') { startX -= $(window).scrollLeft(); startY -= $(window).scrollTop(); } } if(boundaryStoInfo.ySto) { var currentTop = layerInfo ? layerInfo.top : startY; var newTop = $ax.evaluateSTO(boundaryStoInfo.ySto, boundaryStoInfo.boundaryScope, eventInfoCopy); if(moveTo) yValue = newTop; else yValue = newTop - currentTop; } if(boundaryStoInfo.xSto) { var currentLeft = layerInfo ? layerInfo.left : startX; var newLeft = $ax.evaluateSTO(boundaryStoInfo.xSto, boundaryStoInfo.boundaryScope, eventInfoCopy); if(moveTo) xValue = newLeft; else xValue = newLeft - currentLeft; } } } //$ax.public.fn.restoreCompound(jobj); } return { x: Number(xValue), y: Number(yValue), moveTo: moveTo }; }; //we will have something like [[Target.right + withX]] for leftExpr, and this function set the value of withX var _updateLeftExprVariable = function (exprTree, xValue, yValue) { if(exprTree.leftExpr && !exprTree.leftExpr.op) { var localVars = exprTree.leftExpr.localVariables; if(localVars) { if(localVars.withx) localVars.withx.value = xValue; if(localVars.withy) localVars.withy.value = yValue; } } //traversal if(exprTree.op) { if(exprTree.leftExpr) _updateLeftExprVariable(exprTree.leftExpr, xValue, yValue); if(exprTree.rightExpr) _updateLeftExprVariable(exprTree.rightExpr, xValue, yValue); } } var widgetRotationFilter = [ $ax.constants.IMAGE_BOX_TYPE, $ax.constants.IMAGE_MAP_REGION_TYPE, $ax.constants.DYNAMIC_PANEL_TYPE, $ax.constants.VECTOR_SHAPE_TYPE, $ax.constants.VERTICAL_LINE_TYPE, $ax.constants.HORIZONTAL_LINE_TYPE ]; _actionHandlers.rotateWidget = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.objectsToRotate.length; i++) { var rotateInfo = action.objectsToRotate[i].rotateInfo; var elementIds = $ax.getElementIdsFromPath(action.objectsToRotate[i].objectPath, eventInfo); for(var j = 0; j < elementIds.length; j++) { var elementId = elementIds[j]; _queueResizeMove(elementId, queueTypes.rotate, eventInfo, rotateInfo); } } _dispatchAction(eventInfo, actions, index + 1); }; var _addRotate = function (elementId, eventInfo, rotateInfo, options, moveInfo) { var idToResizeMoveState = _getIdToResizeMoveState(eventInfo); rotateInfo = $ax.deepCopy(rotateInfo); rotateInfo.options.easing = options.easing; rotateInfo.options.duration = options.duration; var eventInfoCopy = $ax.eventCopy(eventInfo); eventInfoCopy.targetElement = elementId; //calculate degree value at start of animation var rotateDegree; var offset = {}; var eval = function(boundingRect) { rotateDegree = parseFloat($ax.expr.evaluateExpr(rotateInfo.degree, eventInfoCopy)); offset.x = Number($ax.expr.evaluateExpr(rotateInfo.offsetX, eventInfoCopy)); offset.y = Number($ax.expr.evaluateExpr(rotateInfo.offsetY, eventInfoCopy)); if(!rotateInfo.options.clockwise) rotateDegree = -rotateDegree; _updateOffset(offset, rotateInfo.anchor, boundingRect); } if(moveInfo) { var moveOptions = { dragInfo: eventInfoCopy.dragInfo, duration: options.duration, easing: options.easing, boundaryExpr: moveInfo.options.boundaryExpr, boundaryStos: moveInfo.options.boundaryStos }; } var obj = $obj(elementId); if($ax.public.fn.IsLayer(obj.type)) { var childrenIds = $ax.public.fn.getLayerChildrenDeep(elementId, true, true); if(childrenIds.length == 0) return; var animations = []; //get center point of the group, and degree delta var centerPoint, degreeDelta, moveDelta; animations.push({ id: elementId, type: queueTypes.rotate, func: function () { var boundingRect = $ax('#' + elementId).offsetBoundingRect(); //var boundingRect = $axure.fn.getWidgetBoundingRect(elementId); eval(boundingRect); centerPoint = boundingRect.centerPoint; centerPoint.x += offset.x; centerPoint.y += offset.y; degreeDelta = _initRotateLayer(elementId, rotateInfo, rotateDegree, options, options.stop); _fireAnimationFromQueue(elementId, queueTypes.rotate); moveDelta = { x: 0, y: 0 }; if (moveInfo) { moveDelta = _getMoveLoc(elementId, moveInfo, eventInfoCopy, options.moveStop, idToResizeMoveState[elementId], moveOptions, boundingRect); if (moveDelta.moveTo) { moveDelta.x -= $ax.getNumFromPx($jobj(elementId).css('left')); moveDelta.y -= $ax.getNumFromPx($jobj(elementId).css('top')); } $ax.event.raiseSyntheticEvent(elementId, 'onMove'); } } }); for(var idIndex = 0; idIndex < childrenIds.length; idIndex++) { var childId = childrenIds[idIndex]; (function(id) { var childObj = $obj(id); var rotate = $.inArray(childObj.type, widgetRotationFilter) != -1; var isLayer = $ax.public.fn.IsLayer(childObj.type); animations.push({ id: id, type: queueTypes.rotate, func: function() { $ax.event.raiseSyntheticEvent(id, "onRotate"); if(isLayer) _fireAnimationFromQueue(id, queueTypes.rotate); else $ax('#' + id).circularMoveAndRotate(degreeDelta, options, centerPoint.x, centerPoint.y, rotate, moveDelta); } }); if(!isLayer) animations.push({ id: id, type: queueTypes.move, func: function() {} }); })(childId); } _addAnimations(animations); } else { animations = []; animations.push({ id: elementId, type: queueTypes.rotate, func: function () { var jobj = $jobj(elementId); var unrotatedDim = { width: $ax.getNumFromPx(jobj.css('width')), height: $ax.getNumFromPx(jobj.css('height')) }; eval(unrotatedDim); var delta = { x: 0, y: 0 }; if(moveInfo) { delta = _getMoveLoc(elementId, moveInfo, eventInfoCopy, options.moveStop, idToResizeMoveState[elementId], moveOptions); if(delta.moveTo) { delta.x -= $ax.getNumFromPx($jobj(elementId).css('left')); delta.y -= $ax.getNumFromPx($jobj(elementId).css('top')); } } $ax.event.raiseSyntheticEvent(elementId, 'onRotate'); if(offset.x == 0 && offset.y == 0) _rotateSingle(elementId, rotateDegree, rotateInfo.rotateType == 'location', delta, options, options.stop, true); else _rotateSingleOffset(elementId, rotateDegree, rotateInfo.rotateType == 'location', delta, { x: offset.x, y: offset.y }, options, options.stop); if(moveInfo) $ax.event.raiseSyntheticEvent(elementId, 'onMove'); } }); animations.push({ id: elementId, type: queueTypes.move, func: function () { } }); _addAnimations(animations); } } var _updateOffset = function(offset, anchor, boundingRect) { if (anchor.indexOf('left') != -1) offset.x -= boundingRect.width / 2; if (anchor.indexOf('right') != -1) offset.x += boundingRect.width / 2; if (anchor.indexOf('top') != -1) offset.y -= boundingRect.height / 2; if (anchor.indexOf('bottom') != -1) offset.y += boundingRect.height / 2; } var _rotateSingle = function(elementId, rotateDegree, rotateTo, delta, options, stop, handleMove) { var degreeDelta = _applyRotateStop(rotateDegree, $ax.move.getRotationDegree(elementId), rotateTo, stop); $ax('#' + elementId).rotate(degreeDelta, options.easing, options.duration, false, true); if(handleMove) { if (delta.x || delta.y) _moveSingleWidget(elementId, delta, options); else $ax.action.fireAnimationFromQueue(elementId, $ax.action.queueTypes.move); } }; var _rotateSingleOffset = function (elementId, rotateDegree, rotateTo, delta, offset, options, stop, resizeOffset) { var obj = $obj(elementId); var currRotation = $ax.move.getRotationDegree(elementId); // Need to fix offset. Want to to stay same place on widget after rotation, so need to take the offset and rotate it to where it should be. if(currRotation) { offset = $axure.fn.getPointAfterRotate(currRotation, offset, { x: 0, y: 0 }); } var degreeDelta = _applyRotateStop(rotateDegree, currRotation, rotateTo, stop); var widgetCenter = $ax('#' + elementId).offsetBoundingRect().centerPoint; //var widgetCenter = $axure.fn.getWidgetBoundingRect(elementId).centerPoint; var rotate = $.inArray(obj.type, widgetRotationFilter) != -1; $ax('#' + elementId).circularMoveAndRotate(degreeDelta, options, widgetCenter.x + offset.x, widgetCenter.y + offset.y, rotate, delta, resizeOffset); } var _applyRotateStop = function(rotateDegree, currRotation, to, stop) { var degreeDelta; var ratio; if(to) { degreeDelta = rotateDegree - currRotation; ratio = stop.instant ? 1 : (stop.end - stop.start) / (stop.len - stop.start); } else { degreeDelta = rotateDegree; ratio = stop.instant ? 1 : (stop.end - stop.start) / stop.len; } return degreeDelta * ratio; } var _initRotateLayer = function(elementId, rotateInfo, rotateDegree, options, stop) { var layerDegree = $jobj(elementId).data('layerDegree'); if (layerDegree === undefined) layerDegree = 0; else layerDegree = parseFloat(layerDegree); var to = rotateInfo.rotateType == 'location'; var newDegree = to ? rotateDegree : layerDegree + rotateDegree; var degreeDelta = newDegree - layerDegree; var ratio = stop.instant ? 1 : (stop.end - stop.start) / (stop.len - stop.start); degreeDelta *= ratio; $jobj(elementId).data('layerDegree', newDegree); $ax.event.raiseSyntheticEvent(elementId, "onRotate"); return degreeDelta; } _actionHandlers.setWidgetSize = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.objectsToResize.length; i++) { var resizeInfo = action.objectsToResize[i].sizeInfo; var objPath = action.objectsToResize[i].objectPath; if(objPath == 'thisItem') { var thisId = eventInfo.srcElement; var repeaterId = $ax.getParentRepeaterFromElementId(thisId); var itemId = $ax.repeater.getItemIdFromElementId(thisId); var currSize = $ax.repeater.getItemSize(repeaterId, itemId); var newSize = _getSizeFromInfo(resizeInfo, eventInfo, currSize.width, currSize.height); $ax.repeater.setItemSize(repeaterId, itemId, newSize.width, newSize.height); continue; } var elementIds = $ax.getElementIdsFromPath(objPath, eventInfo); for(var j = 0; j < elementIds.length; j++) { var elementId = elementIds[j]; _queueResizeMove(elementId, queueTypes.resize, eventInfo, resizeInfo); //_addResize(elementId, resizeInfo); } } _dispatchAction(eventInfo, actions, index + 1); }; // Move info undefined unless this move/resize actions are being merged var _addResize = function(elementId, eventInfo, resizeInfo, options, moveInfo, rotateInfo) { var axObject = $obj(elementId); resizeInfo = $ax.deepCopy(resizeInfo); resizeInfo.easing = options.easing; resizeInfo.duration = options.duration; var eventInfoCopy = $ax.eventCopy(eventInfo); eventInfoCopy.targetElement = elementId; var moves = moveInfo || resizeInfo.anchor != "top left" || ($ax.public.fn.IsDynamicPanel(axObject.type) && ((axObject.fixedHorizontal && axObject.fixedHorizontal == 'center') || (axObject.fixedVertical && axObject.fixedVertical == 'middle'))) || (rotateInfo && (rotateInfo.offsetX || rotateInfo.offsetY)); if(moveInfo) { var moveOptions = { dragInfo: eventInfoCopy.dragInfo, duration: options.duration, easing: options.easing, boundaryExpr: moveInfo.options.boundaryExpr, boundaryStos: moveInfo.options.boundaryStos }; } var idToResizeMoveState = _getIdToResizeMoveState(eventInfoCopy); var animations = []; if($ax.public.fn.IsLayer(axObject.type)) { moves = true; // Assume widgets will move will layer, even though not all widgets may move var childrenIds = $ax.public.fn.getLayerChildrenDeep(elementId, true, true); if(childrenIds.length === 0) return; // Need to wait to calculate new size, until time to animate, but animates are in separate queues // best option seems to be to calculate in a "animate" for the layer itself and all children will use that. // May just have to be redundant if this doesn't work well. var boundingRect, widthChangedPercent, heightChangedPercent, unchanged, deltaLoc, degreeDelta, resizeOffset; animations.push({ id: elementId, type: queueTypes.resize, func: function () { $ax.visibility.pushContainer(elementId, false); boundingRect = $ax('#' + elementId).offsetBoundingRect(); //boundingRect = $ax.public.fn.getWidgetBoundingRect(elementId); var size = _getSizeFromInfo(resizeInfo, eventInfoCopy, boundingRect.width, boundingRect.height, elementId); deltaLoc = { x: 0, y: 0 }; var stop = options.stop; var ratio = stop.instant ? 1 : (stop.end - stop.start) / (stop.len - stop.start); widthChangedPercent = Math.round(size.width - boundingRect.width) / boundingRect.width * ratio; heightChangedPercent = Math.round(size.height - boundingRect.height) / boundingRect.height * ratio; resizeOffset = _applyAnchorToResizeOffset(widthChangedPercent * boundingRect.width, heightChangedPercent * boundingRect.height, resizeInfo.anchor); if(stop.instant || stop.end == stop.len) idToResizeMoveState[elementId].resizeResult = undefined; unchanged = widthChangedPercent === 0 && heightChangedPercent === 0; $ax.event.raiseSyntheticEvent(elementId, 'onResize'); _fireAnimationFromQueue(elementId, queueTypes.resize); } }); if(moveInfo) animations.push({ id: elementId, type: queueTypes.move, func: function() { deltaLoc = _getMoveLoc(elementId, moveInfo, eventInfoCopy, options.moveStop, idToResizeMoveState[elementId], moveOptions, boundingRect); $ax.visibility.pushContainer(elementId, false); _fireAnimationFromQueue(elementId, queueTypes.move); $ax.event.raiseSyntheticEvent(elementId, 'onMove'); } }); if (rotateInfo) animations.push({ id: elementId, type: queueTypes.rotate, func: function () { resizeOffset = _applyAnchorToResizeOffset(widthChangedPercent * boundingRect.width, heightChangedPercent * boundingRect.height, resizeInfo.anchor); var rotateDegree = parseFloat($ax.expr.evaluateExpr(rotateInfo.degree, eventInfoCopy)); degreeDelta = _initRotateLayer(elementId, rotateInfo, rotateDegree, options, options.rotateStop); _fireAnimationFromQueue(elementId, queueTypes.rotate); $ax.event.raiseSyntheticEvent(elementId, 'onRotate'); } }); var completeCount = childrenIds.length*2; // Because there is a resize and move complete, it needs to be doubled for(var idIndex = 0; idIndex < childrenIds.length; idIndex++) { // Need to use scoping trick here to make sure childId doesn't change on next loop (function(childId) { //use ax obj to get width and height, jquery css give us the value without border var isLayer = $ax.public.fn.IsLayer($obj(childId).type); var rotate = $.inArray($obj(childId).type, widgetRotationFilter) != -1; animations.push({ id: childId, type: queueTypes.resize, func: function() { //$ax.event.raiseSyntheticEvent(childId, 'onResize'); if(isLayer) { completeCount -= 2; _fireAnimationFromQueue(childId, queueTypes.resize); $ax.event.raiseSyntheticEvent(childId, 'onResize'); } else { var currDeltaLoc = { x: deltaLoc.x, y: deltaLoc.y }; var resizeDeltaMove = { x: 0, y: 0 }; var css = _getCssForResizingLayerChild(childId, resizeInfo.anchor, boundingRect, widthChangedPercent, heightChangedPercent, resizeDeltaMove); var onComplete = function() { if(--completeCount == 0) $ax.visibility.popContainer(elementId, false); }; $ax('#' + childId).resize(css, resizeInfo, true, moves, onComplete); if(rotateInfo) { var offset = { x: Number($ax.expr.evaluateExpr(rotateInfo.offsetX, eventInfoCopy)), y: Number($ax.expr.evaluateExpr(rotateInfo.offsetY, eventInfo)) }; _updateOffset(offset, resizeInfo.anchor, boundingRect); var centerPoint = { x: boundingRect.centerPoint.x + offset.x, y: boundingRect.centerPoint.y + offset.y }; $ax('#' + childId).circularMoveAndRotate(degreeDelta, options, centerPoint.x, centerPoint.y, rotate, currDeltaLoc, resizeOffset, resizeDeltaMove, onComplete); } else { currDeltaLoc.x += resizeDeltaMove.x; currDeltaLoc.y += resizeDeltaMove.y; _moveSingleWidget(childId, currDeltaLoc, options, onComplete); } } } }); if(!isLayer) animations.push({ id: childId, type: queueTypes.move, func: function () {} }); if(!isLayer && rotateInfo) animations.push({ id: childId, type: queueTypes.rotate, func: function () {} }); })(childrenIds[idIndex]); } } else { // Not func, obj with func animations.push({ id: elementId, type: queueTypes.resize, func: function() { //textarea can be resized manully by the user, but doesn't update div size yet, so doing this for now. //alternatively axquery get for size can account for this var sizeId = $ax.public.fn.IsTextArea(axObject.type) ? $jobj(elementId).children('textarea').attr('id') : elementId; var oldSize = $ax('#' + sizeId).size(); var oldWidth = oldSize.width; var oldHeight = oldSize.height; var stop = options.stop; var ratio = stop.instant ? 1 : (stop.end - stop.start) / (stop.len - stop.start); var size = _getSizeFromInfo(resizeInfo, eventInfoCopy, oldWidth, oldHeight, elementId); var newWidth = size.width; var newHeight = size.height; var deltaWidth = Math.round(newWidth - oldWidth) * ratio; var deltaHeight = Math.round(newHeight - oldHeight) * ratio; newWidth = oldWidth + deltaWidth; newHeight = oldHeight + deltaHeight; var delta = { x: 0, y: 0 }; if(moveInfo) { delta = _getMoveLoc(elementId, moveInfo, eventInfoCopy, options.moveStop, idToResizeMoveState[elementId], moveOptions); if (delta.moveTo) { delta.x -= $ax.getNumFromPx($jobj(elementId).css('left')); delta.y -= $ax.getNumFromPx($jobj(elementId).css('top')); } } var rotateHandlesMove = false; var offset = { x: 0, y: 0 }; if(rotateInfo) { offset.x = Number($ax.expr.evaluateExpr(rotateInfo.offsetX, eventInfoCopy)); offset.y = Number($ax.expr.evaluateExpr(rotateInfo.offsetY, eventInfoCopy)); _updateOffset(offset, rotateInfo.anchor, $ax('#' + elementId).offsetBoundingRect()); //_updateOffset(offset, rotateInfo.anchor, $axure.fn.getWidgetBoundingRect(elementId)); rotateHandlesMove = Boolean(rotateInfo && (offset.x || offset.y || rotateInfo.anchor != 'center')); $ax.event.raiseSyntheticEvent(elementId, 'onRotate'); } var css = null; var rootLayer = null; if(deltaHeight != 0 || deltaWidth != 0) { rootLayer = $ax.move.getRootLayer(elementId); if(rootLayer) $ax.visibility.pushContainer(rootLayer, false); css = _getCssForResizingWidget(elementId, eventInfoCopy, resizeInfo.anchor, newWidth, newHeight, oldWidth, oldHeight, delta, options.stop, !rotateHandlesMove); idToResizeMoveState[elementId].resizeResult = undefined; } if(rotateInfo) { var rotateDegree = parseFloat($ax.expr.evaluateExpr(rotateInfo.degree, eventInfoCopy)); if(rotateHandlesMove) { var resizeOffset = _applyAnchorToResizeOffset(deltaWidth, deltaHeight, rotateInfo.anchor); _rotateSingleOffset(elementId, rotateDegree, rotateInfo.rotateType == 'location', delta, offset, options, options.rotateStop, resizeOffset); } else { // Not handling move so pass in nop delta _rotateSingle(elementId, rotateDegree, rotateInfo.rotateType == 'location', { x: 0, y: 0 }, options, options.rotateStop); if (moves) _fireAnimationFromQueue(elementId, queueTypes.move); } } else if(!css && moves) _moveSingleWidget(elementId, delta, options); // Have to do it down here to make sure move info is registered if(moveInfo) $ax.event.raiseSyntheticEvent(elementId, 'onMove'); //$ax.event.raiseSyntheticEvent(elementId, 'onResize'); if (css) { $ax('#' + elementId).resize(css, resizeInfo, true, moves, function () { if(rootLayer) $ax.visibility.popContainer(rootLayer, false); }); } else { _fireAnimationFromQueue(elementId, queueTypes.resize); $ax.event.raiseSyntheticEvent(elementId, 'onResize'); } } }); // Nop move (move handled by resize) if(rotateInfo) animations.push({ id: elementId, type: queueTypes.rotate, func: function () { } }); if(moves) animations.push({ id: elementId, type: queueTypes.move, func: function () { } }); } _addAnimations(animations); }; var _applyAnchorToResizeOffset = function (deltaWidth, deltaHeight, anchor) { var offset = {}; if (anchor.indexOf('left') != -1) offset.x = -deltaWidth / 2; else if (anchor.indexOf('right') != -1) offset.x = deltaWidth / 2; if (anchor.indexOf('top') != -1) offset.y = -deltaHeight / 2; else if (anchor.indexOf('bottom') != -1) offset.y = deltaHeight / 2; return offset; } //var _getOldAndNewSize = function (resizeInfo, eventInfo, targetElement) { // var axObject = $obj(targetElement); // var oldWidth, oldHeight; // //textarea can be resized manully by the user, use the textarea child to get the current size // //because this new size may not be reflected on its parents yet // if ($ax.public.fn.IsTextArea(axObject.type)) { // var jObject = $jobj(elementId); // var textObj = $ax('#' + jObject.children('textarea').attr('id')); // //maybe we shouldn't use ax obj to get width and height here anymore... // oldWidth = textObj.width(); // oldHeight = textObj.height(); // } else { // oldWidth = $ax('#' + elementId).width(); // oldHeight = $ax('#' + elementId).height(); // } // var size = _getSizeFromInfo(resizeInfo, eventInfo, oldHeight, oldWidth, elementId); // return { oldWidth: oldWidth, oldHeight: oldHeight, newWidth: size.width, newHeight: size.height, change: oldWidth != size.width || oldHeight != size.height }; //} var _getSizeFromInfo = function(resizeInfo, eventInfo, oldWidth, oldHeight, targetElement) { var oldTarget = eventInfo.targetElement; eventInfo.targetElement = targetElement; var state = _getIdToResizeMoveState(eventInfo)[targetElement]; if(state && state.resizeResult) return state.resizeResult; var width = $ax.expr.evaluateExpr(resizeInfo.width, eventInfo); var height = $ax.expr.evaluateExpr(resizeInfo.height, eventInfo); eventInfo.targetElement = oldTarget; // If either one is not a number, use the old value width = width != "" ? Number(width) : oldWidth; height = height != "" ? Number(height) : oldHeight; width = isNaN(width) ? oldWidth : width; height = isNaN(height) ? oldHeight : height; // can't be negative var result = { width: Math.max(width, 0), height: Math.max(height, 0) }; if(state) state.resizeResult = result; return result; } //var _queueResize = function (elementId, css, resizeInfo) { // var resizeFunc = function() { // $ax('#' + elementId).resize(css, resizeInfo, true); // //$ax.public.fn.resize(elementId, css, resizeInfo, true); // }; // var obj = $obj(elementId); // var moves = resizeInfo.anchor != "top left" || ($ax.public.fn.IsDynamicPanel(obj.type) && ((obj.fixedHorizontal && obj.fixedHorizontal == 'center') || (obj.fixedVertical && obj.fixedVertical == 'middle'))) // if(!moves) { // _addAnimation(elementId, queueTypes.resize, resizeFunc); // } else { // var animations = []; // animations[0] = { id: elementId, type: queueTypes.resize, func: resizeFunc }; // animations[1] = { id: elementId, type: queueTypes.move, func: function() {}}; // Nop func - resize handles move and firing from queue // _addAnimations(animations); // } //}; //should clean this function and var _getCssForResizingWidget = function (elementId, eventInfo, anchor, newWidth, newHeight, oldWidth, oldHeight, delta, stop, handleMove) { var ratio = stop.instant ? 1 : (stop.end - stop.start) / (stop.len - stop.start); var deltaWidth = (newWidth - oldWidth) * ratio; var deltaHeight = (newHeight - oldHeight) * ratio; if(stop.instant || stop.end == stop.len) { var idToResizeMoveState = _getIdToResizeMoveState(eventInfo); if(idToResizeMoveState[elementId]) idToResizeMoveState[elementId].resizeResult = undefined; } var css = {}; css.height = oldHeight + deltaHeight; var obj = $obj(elementId); //if it's 100% width, don't change its width if($ax.dynamicPanelManager.isPercentWidthPanel(obj)) var is100Dp = true; else css.width = oldWidth + deltaWidth; var jobj = $jobj(elementId); //if this is pinned dp, we will mantain the pin, no matter how you resize it; so no need changes left or top //NOTE: currently only pinned DP has position == fixed if(jobj.css('position') == 'fixed') { if(obj.fixedHorizontal && obj.fixedHorizontal == 'center') css['margin-left'] = '+=' + delta.x; if(obj.fixedVertical && obj.fixedVertical == 'middle') css['margin-top'] = '+=' + delta.y; return css; } // If it is pinned, but temporarily not fixed because it is wrappen in a container, then just make sure to anchor it correctly if(obj.fixedVertical) { if(obj.fixedVertical == 'middle') anchor = obj.fixedHorizontal; else anchor = obj.fixedVertical + (obj.fixedHorizontal == 'center' ? '' : ' ' + obj.fixedHorizontal); } //use position relative to parents //var position = obj.generateCompound ? $ax.public.fn.getWidgetBoundingRect(elementId) : $ax.public.fn.getPositionRelativeToParent(elementId); var locationShift; switch(anchor) { case "top left": locationShift = { x: 0, y: 0 }; break; case "top": locationShift = { x: -deltaWidth / 2.0, y: 0.0 }; break; case "top right": locationShift = { x: -deltaWidth, y: 0.0 }; break; case "left": locationShift = { x: 0.0, y: -deltaHeight / 2.0 }; break; case "center": locationShift = { x: -deltaWidth / 2.0, y: -deltaHeight / 2.0 }; break; case "right": locationShift = { x: -deltaWidth, y: -deltaHeight / 2.0 }; break; case "bottom left": locationShift = { x: 0.0, y: -deltaHeight }; break; case "bottom": locationShift = { x: -deltaWidth/2.0, y: -deltaHeight }; break; case "bottom right": locationShift = { x: -deltaWidth, y: -deltaHeight }; break; } if(handleMove) { if(jobj.css('position') === 'absolute') { css.left = $ax.getNumFromPx(jobj.css('left')) + locationShift.x + delta.x; css.top = $ax.getNumFromPx(jobj.css('top')) + locationShift.y + delta.y; } else { var axQuery = $ax('#' + elementId); var offsetLocation = axQuery.offsetLocation(); css.left = offsetLocation.left + locationShift.x + delta.x; css.top = offsetLocation.top + locationShift.y + delta.y; //css.left = axQuery.left(true) + locationShift.x + delta.x; //css.top = axQuery.top(true) + locationShift.y + delta.y; } } else { delta.x += locationShift.x; delta.y += locationShift.y; } css.deltaX = locationShift.x + delta.x; css.deltaY = locationShift.y + delta.y; return css; }; var _getCssForResizingLayerChild = function (elementId, anchor, layerBoundingRect, widthChangedPercent, heightChangedPercent, deltaLoc) { var boundingRect = $ax('#' + elementId).offsetBoundingRect(); //var boundingRect = $ax.public.fn.getWidgetBoundingRect(elementId); var childCenterPoint = boundingRect.centerPoint; var currentSize = $ax('#' + elementId).size(); var newWidth = currentSize.width + currentSize.width * widthChangedPercent; var newHeight = currentSize.height + currentSize.height * heightChangedPercent; var css = {}; css.height = newHeight; var obj = $obj(elementId); //if it's 100% width, don't change its width and left var changeLeft = true; if($ax.dynamicPanelManager.isPercentWidthPanel(obj)) changeLeft = false; else css.width = newWidth; var jobj = $jobj(elementId); //if this is pinned dp, we will mantain the pin, no matter how you resize it; so no need changes left or top //NOTE: currently only pinned DP has position == fixed if(jobj.css('position') == 'fixed') return css; //use bounding rect position relative to parents to calculate delta //var axObj = $ax('#' + elementId); // This will be absolute world coordinates, but we want body coordinates. var offsetLocation = $ax('#' + elementId).offsetLocation(); var currentLeft = offsetLocation.left; var currentTop = offsetLocation.top; //var currentLeft = axObj.locRelativeIgnoreLayer(false); //var currentTop = axObj.locRelativeIgnoreLayer(true); var resizable = $ax.public.fn.IsResizable(obj.type); if(anchor.indexOf("top") > -1) { var topDelta = (currentTop - layerBoundingRect.top) * heightChangedPercent; if(!resizable && Math.round(topDelta)) topDelta += currentSize.height * heightChangedPercent; } else if(anchor.indexOf("bottom") > -1) { if(resizable) topDelta = (currentTop - layerBoundingRect.bottom) * heightChangedPercent; else { var bottomDelta = Math.round(currentTop + currentSize.height - layerBoundingRect.bottom) * heightChangedPercent; if(bottomDelta) topDelta = bottomDelta - currentSize.height * heightChangedPercent; else topDelta = 0; } } else { //center vertical if(resizable) topDelta = (childCenterPoint.y - layerBoundingRect.centerPoint.y)*heightChangedPercent - currentSize.height*heightChangedPercent/2; else { var centerTopChange = Math.round(childCenterPoint.y - layerBoundingRect.centerPoint.y)*heightChangedPercent; if(centerTopChange > 0) topDelta = centerTopChange + Math.abs(currentSize.height * heightChangedPercent / 2); else if(centerTopChange < 0) topDelta = centerTopChange - Math.abs(currentSize.height * heightChangedPercent / 2); else topDelta = 0; } } if(changeLeft) { if(anchor.indexOf("left") > -1) { var leftDelta = (currentLeft - layerBoundingRect.left) * widthChangedPercent; if(!resizable && Math.round(leftDelta)) leftDelta += currentSize.width * widthChangedPercent; } else if(anchor.indexOf("right") > -1) { if(resizable) leftDelta = (currentLeft - layerBoundingRect.right) * widthChangedPercent; else { var rightDelta = Math.round(currentLeft + currentSize.width - layerBoundingRect.right) * widthChangedPercent; if(rightDelta) leftDelta = rightDelta - currentSize.width * widthChangedPercent; else leftDelta = 0; } } else { //center horizontal if(resizable) leftDelta = (childCenterPoint.x - layerBoundingRect.centerPoint.x)*widthChangedPercent - currentSize.width*widthChangedPercent/2; else { var centerLeftChange = Math.round(childCenterPoint.x - layerBoundingRect.centerPoint.x) * widthChangedPercent; if(centerLeftChange > 0) leftDelta = centerLeftChange + Math.abs(currentSize.width * widthChangedPercent / 2); else if(centerLeftChange < 0) leftDelta = centerLeftChange - Math.abs(currentSize.width * widthChangedPercent / 2); else leftDelta = 0; } } } if(topDelta) deltaLoc.y += topDelta; if(leftDelta && changeLeft) deltaLoc.x += leftDelta; return css; }; _actionHandlers.setPanelOrder = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.panelPaths.length; i++) { var func = action.panelPaths[i].setOrderInfo.bringToFront ? 'bringToFront' : 'sendToBack'; var elementIds = $ax.getElementIdsFromPath(action.panelPaths[i].panelPath, eventInfo); for(var j = 0; j < elementIds.length; j++) $ax('#' + elementIds[j])[func](); } _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.modifyDataSetEditItems = function(eventInfo, actions, index) { var action = actions[index]; var add = action.repeatersToAddTo; var repeaters = add || action.repeatersToRemoveFrom; var itemId; for(var i = 0; i < repeaters.length; i++) { var data = repeaters[i]; // Grab the first one because repeaters must have only element id, as they cannot be inside repeaters // or none if unplaced var id = $ax.getElementIdsFromPath(data.path, eventInfo)[0]; if(!id) continue; if(data.addType == 'this') { var scriptId = $ax.repeater.getScriptIdFromElementId(eventInfo.srcElement); itemId = $ax.repeater.getItemIdFromElementId(eventInfo.srcElement); var repeaterId = $ax.getParentRepeaterFromScriptId(scriptId); if(add) $ax.repeater.addEditItems(repeaterId, [itemId]); else $ax.repeater.removeEditItems(repeaterId, [itemId]); } else if(data.addType == 'all') { var allItems = $ax.repeater.getAllItemIds(id); if(add) $ax.repeater.addEditItems(id, allItems); else $ax.repeater.removeEditItems(id, allItems); } else { var oldTarget = eventInfo.targetElement; var itemIds = $ax.repeater.getAllItemIds(id); var itemIdsToAdd = []; for(var j = 0; j < itemIds.length; j++) { itemId = itemIds[j]; eventInfo.targetElement = $ax.repeater.createElementId(id, itemId); if($ax.expr.evaluateExpr(data.query, eventInfo) == "true") { itemIdsToAdd[itemIdsToAdd.length] = String(itemId); } eventInfo.targetElement = oldTarget; } if(add) $ax.repeater.addEditItems(id, itemIdsToAdd); else $ax.repeater.removeEditItems(id, itemIdsToAdd); } } _dispatchAction(eventInfo, actions, index + 1); }; _action.repeaterInfoNames = { addItemsToDataSet: 'dataSetsToAddTo', deleteItemsFromDataSet: 'dataSetItemsToRemove', updateItemsInDataSet: 'dataSetsToUpdate', addFilterToRepeater: 'repeatersToAddFilter', removeFilterFromRepeater: 'repeatersToRemoveFilter', addSortToRepeater: 'repeaterToAddSort', removeSortFromRepeater: 'repeaterToRemoveSort', setRepeaterToPage: 'repeatersToSetPage', setItemsPerRepeaterPage: 'repeatersToSetItemCount' }; _actionHandlers.addItemsToDataSet = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.dataSetsToAddTo.length; i++) { var datasetInfo = action.dataSetsToAddTo[i]; // Grab the first one because repeaters must have only element id, as they cannot be inside repeaters // or none if unplaced var id = $ax.getElementIdsFromPath(datasetInfo.path, eventInfo)[0]; if(!id || _ignoreAction(id)) continue; var dataset = datasetInfo.data; for(var j = 0; j < dataset.length; j++) $ax.repeater.addItem(id, $ax.deepCopy(dataset[j]), eventInfo); if(dataset.length) _addRefresh(id); } _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.deleteItemsFromDataSet = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.dataSetItemsToRemove.length; i++) { // Grab the first one because repeaters must have only element id, as they cannot be inside repeaters // or none if unplaced var deleteInfo = action.dataSetItemsToRemove[i]; var id = $ax.getElementIdsFromPath(deleteInfo.path, eventInfo)[0]; if(!id || _ignoreAction(id)) continue; $ax.repeater.deleteItems(id, eventInfo, deleteInfo.type, deleteInfo.rule); _addRefresh(id); } _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.updateItemsInDataSet = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.dataSetsToUpdate.length; i++) { var dataSet = action.dataSetsToUpdate[i]; // Grab the first one because repeaters must have only element id, as they cannot be inside repeaters // or none if unplaced var id = $ax.getElementIdsFromPath(dataSet.path, eventInfo)[0]; if(!id || _ignoreAction(id)) continue; $ax.repeater.updateEditItems(id, dataSet.props, eventInfo, dataSet.type, dataSet.rule); _addRefresh(id); } _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.setRepeaterToDataSet = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.repeatersToSet.length; i++) { var setRepeaterInfo = action.repeatersToSet[i]; // Grab the first one because repeaters must have only element id, as they cannot be inside repeaters // or none if unplaced var id = $ax.getElementIdsFromPath(setRepeaterInfo.path, eventInfo)[0]; if(!id) continue; $ax.repeater.setDataSet(id, setRepeaterInfo.localDataSetId); } _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.addFilterToRepeater = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.repeatersToAddFilter.length; i++) { var addFilterInfo = action.repeatersToAddFilter[i]; // Grab the first one because repeaters must have only element id, as they cannot be inside repeaters // or none if unplaced var id = $ax.getElementIdsFromPath(addFilterInfo.path, eventInfo)[0]; if(!id || _ignoreAction(id)) continue; $ax.repeater.addFilter(id, addFilterInfo.removeOtherFilters, addFilterInfo.label, addFilterInfo.filter, eventInfo.srcElement); _addRefresh(id); } _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.removeFilterFromRepeater = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.repeatersToRemoveFilter.length; i++) { var removeFilterInfo = action.repeatersToRemoveFilter[i]; // Grab the first one because repeaters must have only element id, as they cannot be inside repeaters // or none if unplaced var id = $ax.getElementIdsFromPath(removeFilterInfo.path, eventInfo)[0]; if(!id || _ignoreAction(id)) continue; if(removeFilterInfo.removeAll) $ax.repeater.removeFilter(id); else if(removeFilterInfo.filterName != '') { $ax.repeater.removeFilter(id, removeFilterInfo.filterName); } _addRefresh(id); } _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.addSortToRepeater = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.repeatersToAddSort.length; i++) { var addSortInfo = action.repeatersToAddSort[i]; // Grab the first one because repeaters must have only element id, as they cannot be inside repeaters // or none if unplaced var id = $ax.getElementIdsFromPath(addSortInfo.path, eventInfo)[0]; if(!id || _ignoreAction(id)) continue; $ax.repeater.addSort(id, addSortInfo.label, addSortInfo.columnName, addSortInfo.ascending, addSortInfo.toggle, addSortInfo.sortType); _addRefresh(id); } _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.removeSortFromRepeater = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.repeatersToRemoveSort.length; i++) { var removeSortInfo = action.repeatersToRemoveSort[i]; // Grab the first one because repeaters must have only element id, as they cannot be inside repeaters // or none if unplaced var id = $ax.getElementIdsFromPath(removeSortInfo.path, eventInfo)[0]; if(!id || _ignoreAction(id)) continue; if(removeSortInfo.removeAll) $ax.repeater.removeSort(id); else if(removeSortInfo.sortName != '') $ax.repeater.removeSort(id, removeSortInfo.sortName); _addRefresh(id); } _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.setRepeaterToPage = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.repeatersToSetPage.length; i++) { var setPageInfo = action.repeatersToSetPage[i]; // Grab the first one because repeaters must have only element id, as they cannot be inside repeaters // or none if unplaced var id = $ax.getElementIdsFromPath(setPageInfo.path, eventInfo)[0]; if(!id || _ignoreAction(id)) continue; var oldTarget = eventInfo.targetElement; eventInfo.targetElement = id; $ax.repeater.setRepeaterToPage(id, setPageInfo.pageType, setPageInfo.pageValue, eventInfo); eventInfo.targetElement = oldTarget; _addRefresh(id); } _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.setItemsPerRepeaterPage = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.repeatersToSetItemCount.length; i++) { var setItemCountInfo = action.repeatersToSetItemCount[i]; // Grab the first one because repeaters must have only element id, as they cannot be inside repeaters // or none if unplaced var id = $ax.getElementIdsFromPath(setItemCountInfo.path, eventInfo)[0]; if(!id || _ignoreAction(id)) continue; if(setItemCountInfo.noLimit) $ax.repeater.setNoItemLimit(id); else $ax.repeater.setItemLimit(id, setItemCountInfo.itemCountValue, eventInfo); _addRefresh(id); } _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.refreshRepeater = function(eventInfo, actions, index) { // We use this as a psudo action now. var action = actions[index]; for (var i = 0; i < action.repeatersToRefresh.length; i++) { // Grab the first one because repeaters must have only element id, as they cannot be inside repeaters // or none if unplaced var id = $ax.getElementIdsFromPath(action.repeatersToRefresh[i], eventInfo)[0]; if(id) _tryRefreshRepeater(id, eventInfo); } _dispatchAction(eventInfo, actions, index + 1); }; var _tryRefreshRepeater = function(id, eventInfo) { var idIndex = _repeatersToRefresh.indexOf(id); if(idIndex == -1) return; $ax.splice(_repeatersToRefresh, idIndex, 1); $ax.repeater.refreshRepeater(id, eventInfo); }; _action.tryRefreshRepeaters = function(ids, eventInfo) { for(var i = 0; i < ids.length; i++) _tryRefreshRepeater(ids[i], eventInfo); }; _actionHandlers.scrollToWidget = function(eventInfo, actions, index) { var action = actions[index]; var elementIds = $ax.getElementIdsFromPath(action.objectPath, eventInfo); if(elementIds.length > 0) $ax('#' + elementIds[0]).scroll(action.options); _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.enableDisableWidgets = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.pathToInfo.length; i++) { var elementIds = $ax.getElementIdsFromPath(action.pathToInfo[i].objectPath, eventInfo); var enable = action.pathToInfo[i].enableDisableInfo.enable; for(var j = 0; j < elementIds.length; j++) $ax('#' + elementIds[j]).enabled(enable); } _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.setImage = function(eventInfo, actions, index) { var oldTarget = eventInfo.targetElement; var action = actions[index]; var view = $ax.adaptive.currentViewId; eventInfo.image = true; for(var i = 0; i < action.imagesToSet.length; i++) { var imgInfo = action.imagesToSet[i]; if (view && imgInfo.adaptive[view]) imgInfo = imgInfo.adaptive[view]; else imgInfo = imgInfo.base; var elementIds = $ax.getElementIdsFromPath(action.imagesToSet[i].objectPath, eventInfo); for(var j = 0; j < elementIds.length; j++) { var elementId = elementIds[j]; eventInfo.targetElement = elementId; var evaluatedImgs = _evaluateImages(imgInfo, eventInfo); var img = evaluatedImgs.normal; if($ax.style.IsWidgetDisabled(elementId)) { if(imgInfo.disabled) img = evaluatedImgs.disabled; } else if($ax.style.IsWidgetSelected(elementId)) { if(imgInfo.selected) img = evaluatedImgs.selected; } else if($ax.event.mouseDownObjectId == elementId && imgInfo.mouseDown) img = evaluatedImgs.mouseDown; else if($ax.event.mouseOverIds.indexOf(elementId) != -1 && imgInfo.mouseOver) { img = evaluatedImgs.mouseOver; //Update mouseOverObjectId var currIndex = $ax.event.mouseOverIds.indexOf($ax.event.mouseOverObjectId); var imgIndex = $ax.event.mouseOverIds.indexOf(elementId); if(currIndex < imgIndex) $ax.event.mouseOverObjectId = elementId; } else if(imgInfo.mouseOver && elementId == eventInfo.srcElement) { img = evaluatedImgs.mouseOver; } // $('#' + $ax.repeater.applySuffixToElementId(elementId, '_img')).attr('src', img); $jobj($ax.GetImageIdFromShape(elementId)).attr('src', img); //Set up overrides $ax.style.mapElementIdToImageOverrides(elementId, evaluatedImgs); $ax.style.updateElementIdImageStyle(elementId); if(evaluatedImgs.mouseOver || evaluatedImgs.mouseDown) $ax.event.updateIxStyleEvents(elementId); } } eventInfo.targetElement = oldTarget; eventInfo.image = false; _dispatchAction(eventInfo, actions, index + 1); }; var _evaluateImages = function(imgInfo, eventInfo) { var retVal = {}; for(var state in imgInfo) { if(!imgInfo.hasOwnProperty(state)) continue; var img = imgInfo[state][$ax.adaptive.getSketchKey()] || $ax.expr.evaluateExpr(imgInfo[state].literal, eventInfo); if(!img) img = $axure.utils.getTransparentGifPath(); retVal[state] = img; } return retVal; }; $ax.clearRepeaterImageOverrides = function(repeaterId) { var childIds = $ax.getChildElementIdsForRepeater(repeaterId); for(var i = childIds; i < childIds.length; i++) $ax.style.deleteElementIdToImageOverride(childIds[i]); }; _actionHandlers.setFocusOnWidget = function(eventInfo, actions, index) { var action = actions[index]; if(action.objectPaths.length > 0) { var elementIds = $ax.getElementIdsFromPath(action.objectPaths[0], eventInfo); if(elementIds.length > 0) { $ax('#' + elementIds[0]).focus(); //if select text and not in placeholder mode, then select all text if(action.selectText && !$ax.placeholderManager.isActive(elementIds[0])) { var elementChildren = document.getElementById(elementIds[0]).children; //find the input or textarea element for(var i = 0; i < elementChildren.length; i++) { if (elementChildren[i].id.indexOf('_input') == -1) continue; var elementTagName = elementChildren[i].tagName; if(elementTagName && (elementTagName.toLowerCase() == "input" || elementTagName.toLowerCase() == "textarea")) { elementChildren[i].select(); } } } } } _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.expandCollapseTree = function(eventInfo, actions, index) { var action = actions[index]; for(var i = 0; i < action.pathToInfo.length; i++) { var pair = action.pathToInfo[i]; var elementIds = $ax.getElementIdsFromPath(pair.treeNodePath, eventInfo); for(var j = 0; j < elementIds.length; j++) $ax('#' + elementIds[j]).expanded(pair.expandCollapseInfo.expand); } _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.other = function(eventInfo, actions, index) { var action = actions[index]; $ax.navigate({ url: $axure.utils.getOtherPath() + "#other=" + encodeURI(action.otherDescription), target: "popup", includeVariables: false, popupOptions: action.popup }); _dispatchAction(eventInfo, actions, index + 1); }; _actionHandlers.fireEvents = function(eventInfo, actions, index) { var action = actions[index]; //look for the nearest element id var objId = eventInfo.srcElement; var thisWidget = eventInfo.thiswidget; var obj = $ax.getObjectFromElementId(objId); var rdoId = obj ? $ax.getRdoParentFromElementId(objId) : ""; var rdo = $ax.getObjectFromElementId(rdoId); var page = rdo ? $ax.pageData.masters[rdo.masterId] : $ax.pageData.page; // Check if rdo should be this var oldIsMasterEvent = eventInfo.isMasterEvent; if (obj && $ax.public.fn.IsReferenceDiagramObject(obj.type) && eventInfo.isMasterEvent) { rdoId = objId; rdo = obj; page = $ax.pageData.masters[rdo.masterId]; } for(var i = 0; i < action.firedEvents.length; i++) { var firedEvent = action.firedEvents[i]; var isPage = firedEvent.objectPath.length == 0; var targetObjIds = isPage ? [rdoId] : $ax.getElementIdsFromPath(firedEvent.objectPath, eventInfo); for (var j = 0; j < targetObjIds.length; j++) { var targetObjId = targetObjIds[j]; var targetObj = isPage ? rdo : $ax.getObjectFromElementId(targetObjId); eventInfo.srcElement = targetObjId || ''; eventInfo.thiswidget = $ax.getWidgetInfo(eventInfo.srcElement); eventInfo.isMasterEvent = false; var raisedEvents = firedEvent.raisedEventIds; if(raisedEvents) { for(var k = 0; k < raisedEvents.length; k++) { var event = targetObj.interactionMap && targetObj.interactionMap.raised && targetObj.interactionMap.raised[raisedEvents[k]]; if(event) $ax.event.handleEvent(targetObjId, eventInfo, event, false, true); } } if(isPage) { eventInfo.isMasterEvent = true; eventInfo.label = $ax.pageData.page.name; eventInfo.friendlyType = 'Page'; } var firedTarget = isPage ? page : targetObj; var firedEventNames = firedEvent.firedEventNames; if(firedEventNames) { for(k = 0; k < firedEventNames.length; k++) { event = firedTarget.interactionMap && firedTarget.interactionMap[firedEventNames[k]]; if(event) $ax.event.handleEvent(isPage ? '' : targetObjId, eventInfo, event, false, true); } } if(isPage) eventInfo.isMasterEvent = oldIsMasterEvent; } eventInfo.srcElement = objId; eventInfo.thiswidget = thisWidget; eventInfo.isMasterEvent = oldIsMasterEvent; } _dispatchAction(eventInfo, actions, index + 1); }; });