events.js 62 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361
  1. // ******* Features MANAGER ******** //
  2. $axure.internal(function($ax) {
  3. var _features = $ax.features = {};
  4. var _supports = _features.supports = {};
  5. _supports.touchstart = typeof window.ontouchstart !== 'undefined';
  6. _supports.touchmove = typeof window.ontouchmove !== 'undefined';
  7. _supports.touchend = typeof window.ontouchend !== 'undefined';
  8. _supports.mobile = _supports.touchstart && _supports.touchend && _supports.touchmove;
  9. // Got this from http://stackoverflow.com/questions/11381673/javascript-solution-to-detect-mobile-browser
  10. var check = navigator.userAgent.match(/Android/i)
  11. || navigator.userAgent.match(/webOS/i)
  12. || navigator.userAgent.match(/iPhone/i)
  13. || navigator.userAgent.match(/iPad/i)
  14. || navigator.userAgent.match(/iPod/i)
  15. || navigator.userAgent.match(/BlackBerry/i)
  16. || navigator.userAgent.match(/Tablet PC/i)
  17. || navigator.userAgent.match(/Windows Phone/i);
  18. _supports.windowsMobile = navigator.userAgent.match(/Tablet PC/i) || navigator.userAgent.match(/Windows Phone/i);
  19. if(!check && _supports.mobile) {
  20. _supports.touchstart = false;
  21. _supports.touchmove = false;
  22. _supports.touchend = false;
  23. _supports.mobile = false;
  24. }
  25. var _eventNames = _features.eventNames = {};
  26. _eventNames.mouseDownName = _supports.touchstart ? 'touchstart' : 'mousedown';
  27. _eventNames.mouseUpName = _supports.touchend ? 'touchend' : 'mouseup';
  28. _eventNames.mouseMoveName = _supports.touchmove ? 'touchmove' : 'mousemove';
  29. });
  30. // ******* EVENT MANAGER ******** //
  31. $axure.internal(function($ax) {
  32. var _objectIdToEventHandlers = {};
  33. var _jBrowserEvent = undefined;
  34. $ax.setjBrowserEvent = function(event) {
  35. _jBrowserEvent = event;
  36. };
  37. $ax.getjBrowserEvent = function() {
  38. return _jBrowserEvent;
  39. };
  40. var _event = {};
  41. $ax.event = _event;
  42. //initilize state
  43. _event.mouseOverObjectId = '';
  44. _event.mouseDownObjectId = '';
  45. _event.mouseOverIds = [];
  46. var EVENT_NAMES = ['mouseenter', 'mouseleave', 'contextmenu', 'change', 'focus', 'blur'];
  47. // Tap, double tap, and touch move, or synthetic.
  48. if(!$ax.features.supports.mobile) {
  49. EVENT_NAMES[EVENT_NAMES.length] = 'click';
  50. EVENT_NAMES[EVENT_NAMES.length] = 'dblclick';
  51. EVENT_NAMES[EVENT_NAMES.length] = 'mousemove';
  52. }
  53. // add the event names for the touch events
  54. EVENT_NAMES[EVENT_NAMES.length] = $ax.features.eventNames.mouseDownName;
  55. EVENT_NAMES[EVENT_NAMES.length] = $ax.features.eventNames.mouseUpName;
  56. for(var i = 0; i < EVENT_NAMES.length; i++) {
  57. var eventName = EVENT_NAMES[i];
  58. //we need the function here to circumvent closure modifying eventName
  59. _event[eventName] = (function(event_Name) {
  60. return function(elementId, fn) {
  61. var elementIdQuery = $jobj(elementId);
  62. var type = $ax.getTypeFromElementId(elementId);
  63. //we need specially track link events so we can enable and disable them along with
  64. //their parent widgets
  65. if(elementIdQuery.is('a')) _attachCustomObjectEvent(elementId, event_Name, fn);
  66. //see notes below
  67. else if($ax.IsTreeNodeObject(type)) _attachTreeNodeEvent(elementId, event_Name, fn);
  68. else if($ax.IsImageFocusable(type) && (event_Name == 'focus' || event_Name == 'blur')) {
  69. _attachDefaultObjectEvent($jobj($ax.repeater.applySuffixToElementId(elementId, '_img')), elementId, event_Name, fn);
  70. } else {
  71. var inputId = $ax.INPUT(elementId);
  72. var isInput = $jobj(inputId).length != 0;
  73. var id = isInput && (event_Name == 'focus' || event_Name == 'blur') ? inputId : elementId;
  74. _attachDefaultObjectEvent($jobj(id), elementId, event_Name, fn);
  75. }
  76. };
  77. })(eventName);
  78. }
  79. var AXURE_TO_JQUERY_EVENT_NAMES = {
  80. 'onMouseOver': 'mouseenter',
  81. 'onMouseOut': 'mouseleave',
  82. 'onContextMenu': 'contextmenu',
  83. 'onChange': 'change',
  84. 'onFocus': 'focus',
  85. 'onLostFocus': 'blur'
  86. };
  87. // Tap, double tap, and touch move, or synthetic.
  88. if(!$ax.features.supports.mobile) {
  89. AXURE_TO_JQUERY_EVENT_NAMES.onClick = 'click';
  90. AXURE_TO_JQUERY_EVENT_NAMES.onDoubleClick = 'dblclick';
  91. AXURE_TO_JQUERY_EVENT_NAMES.onMouseMove = 'mousemove';
  92. }
  93. AXURE_TO_JQUERY_EVENT_NAMES.onMouseDown = $ax.features.eventNames.mouseDownName;
  94. AXURE_TO_JQUERY_EVENT_NAMES.onMouseUp = $ax.features.eventNames.mouseUpName;
  95. var _attachEvents = function(diagramObject, elementId) {
  96. var inputId = $ax.repeater.applySuffixToElementId(elementId, '_input');
  97. var id = $jobj(inputId).length ? inputId : elementId;
  98. for(var eventName in diagramObject.interactionMap) {
  99. var jQueryEventName = AXURE_TO_JQUERY_EVENT_NAMES[eventName];
  100. if(!jQueryEventName) continue;
  101. _event[jQueryEventName](id,
  102. //this is needed to escape closure
  103. (function(axEventObject) {
  104. return function(e) {
  105. $ax.setjBrowserEvent(e);
  106. _handleEvent(elementId, $ax.getEventInfoFromEvent($ax.getjBrowserEvent(), false, elementId), axEventObject);
  107. };
  108. })(diagramObject.interactionMap[eventName])
  109. );
  110. }
  111. };
  112. var preventDefaultEvents = ['OnContextMenu', 'OnKeyUp', 'OnKeyDown'];
  113. var allowBubble = ['OnFocus', 'OnResize', 'OnMouseOut', 'OnMouseOver'];
  114. var _canClick = true;
  115. var _startScroll = [];
  116. var _setCanClick = function(canClick) {
  117. _canClick = canClick;
  118. if(_canClick) _startScroll = [$(window).scrollLeft(), $(window).scrollTop()];
  119. };
  120. var _getCanClick = function() {
  121. if(!$ax.features.supports.mobile) return true;
  122. var endScroll = [$(window).scrollLeft(), $(window).scrollTop()];
  123. return _canClick && _startScroll[0] == endScroll[0] && _startScroll[1] == endScroll[1];
  124. };
  125. var _handleEvent = $ax.event.handleEvent = function(elementId, eventInfo, axEventObject, skipShowDescriptions, synthetic) {
  126. var eventDescription = axEventObject.description;
  127. if(!_getCanClick() && eventDescription == 'OnClick') return;
  128. // If you are supposed to suppress, do that right away.
  129. if(suppressedEventStatus[eventDescription]) {
  130. return;
  131. }
  132. var currentEvent = $ax.getjBrowserEvent();
  133. if(!synthetic && currentEvent && currentEvent.originalEvent && currentEvent.originalEvent.handled && !eventInfo.isMasterEvent) return;
  134. if(!synthetic && elementId && !$ax.style.getObjVisible(elementId) && $ax.getTypeFromElementId(elementId) != 'referenceDiagramObject') return;
  135. var bubble = true;
  136. if(skipShowDescriptions || !_shouldShowCaseDescriptions(axEventObject)) {
  137. //handle case descriptions
  138. var caseGroups = [];
  139. var currentCaseGroup = [];
  140. caseGroups[0] = currentCaseGroup;
  141. // Those refreshes not after a wait
  142. var guaranteedRefreshes = {};
  143. var caseGroupIndex = 0;
  144. for(var i = 0; i < axEventObject.cases.length; i++) {
  145. var currentCase = axEventObject.cases[i];
  146. if(currentCase.isNewIfGroup && i != 0) {
  147. caseGroupIndex++;
  148. currentCaseGroup = [];
  149. caseGroups[caseGroups.length] = currentCaseGroup;
  150. }
  151. currentCaseGroup[currentCaseGroup.length] = currentCase;
  152. for(var j = 0; j < currentCase.actions.length; j++) {
  153. var action = currentCase.actions[j];
  154. if(action.action == 'wait') break;
  155. if(action.action != 'refreshRepeater') continue;
  156. for(var k = 0; k < action.repeatersToRefresh.length; k++) {
  157. var id = $ax.getElementIdsFromPath(action.repeatersToRefresh[k], eventInfo)[0];
  158. guaranteedRefreshes[id] = caseGroupIndex;
  159. }
  160. }
  161. }
  162. for(var i = 0; i < caseGroups.length; i++) {
  163. var groupRefreshes = [];
  164. for(var key in guaranteedRefreshes) {
  165. if(guaranteedRefreshes[key] == i) groupRefreshes[groupRefreshes.length] = key;
  166. }
  167. bubble = _handleCaseGroup(eventInfo, caseGroups[i], groupRefreshes) && bubble;
  168. }
  169. } else {
  170. _showCaseDescriptions(elementId, eventInfo, axEventObject, synthetic);
  171. bubble = false;
  172. }
  173. // Only trigger a supression if it handled this event
  174. if(!bubble && suppressingEvents[eventDescription]) {
  175. suppressedEventStatus[suppressingEvents[eventDescription]] = true;
  176. }
  177. // This should not be needed anymore. All refreshes should be inserted, or handled earlier.
  178. var repeaters = $ax.deepCopy($ax.action.repeatersToRefresh);
  179. while($ax.action.repeatersToRefresh.length) $ax.action.repeatersToRefresh.pop();
  180. for(i = 0; i < repeaters.length; i++) $ax.repeater.refreshRepeater(repeaters[i], eventInfo);
  181. if(currentEvent && currentEvent.originalEvent) {
  182. currentEvent.originalEvent.handled = !synthetic && !bubble && allowBubble.indexOf(eventDescription) == -1;
  183. //currentEvent.originalEvent.donotdrag = currentEvent.donotdrag || (!bubble && eventDescription == 'OnMouseDown');
  184. // Prevent default if necessary
  185. if(currentEvent.originalEvent.handled && preventDefaultEvents.indexOf(eventDescription) != -1) {
  186. currentEvent.preventDefault();
  187. }
  188. }
  189. };
  190. var _showCaseDescriptions = function(elementId, eventInfo, axEventObject, synthetic) {
  191. if(axEventObject.cases.length == 0) return true;
  192. var linksId = elementId + "linkBox";
  193. $('#' + linksId).remove();
  194. var $container = $("<div class='intcases' id='" + linksId + "'></div>");
  195. if(!_isEventSimulating(axEventObject)) {
  196. var copy = $ax.eventCopy(eventInfo);
  197. for(var i = 0; i < axEventObject.cases.length; i++) {
  198. var $link = $("<div class='intcaselink'>" + axEventObject.cases[i].description + "</div>");
  199. $link.click(function(j) {
  200. return function() {
  201. var bubble = $ax.action.dispatchAction(copy, axEventObject.cases[j].actions);
  202. $('#' + linksId).remove();
  203. return bubble;
  204. };
  205. } (i)
  206. );
  207. $container.append($link);
  208. }
  209. } else {
  210. var fullDescription = axEventObject.description + ":<br>";
  211. for(var i = 0; i < axEventObject.cases.length; i++) {
  212. var currentCase = axEventObject.cases[i];
  213. fullDescription += "&nbsp;&nbsp;" + currentCase.description.replace(/<br>/g, '<br>&nbsp;&nbsp;') + ":<br>";
  214. for(var j = 0; j < currentCase.actions.length; j++) {
  215. fullDescription += "&nbsp;&nbsp;&nbsp;&nbsp;" + currentCase.actions[j].description.replace(/<br>/g, '<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;') + "<br>";
  216. }
  217. }
  218. fullDescription = fullDescription.substring(0, fullDescription.length - 4);
  219. var $link = $("<div class='intcaselink'>" + fullDescription + "</div>");
  220. $link.click(function() {
  221. _handleEvent(elementId, eventInfo, axEventObject, true, synthetic);
  222. $('#' + linksId).remove();
  223. return;
  224. });
  225. $container.append($link);
  226. }
  227. $container.mouseleave(function(e) { $ax.legacy.SuppressBubble(e); });
  228. $('body').append($container);
  229. _showCaseLinks(eventInfo, linksId);
  230. };
  231. var _showCaseLinks = function(eventInfo, linksId) {
  232. var links = window.document.getElementById(linksId);
  233. links.style.top = eventInfo.pageY;
  234. var left = eventInfo.pageX;
  235. links.style.left = left;
  236. $ax.visibility.SetVisible(links, true);
  237. $ax.legacy.BringToFront(linksId, true);
  238. $ax.legacy.RefreshScreen();
  239. };
  240. var _shouldShowCaseDescriptions = function(axEventObject) {
  241. if($ax.document.configuration.linkStyle == "alwaysDisplayTargets") return true;
  242. if($ax.document.configuration.linkStyle == "neverDisplayTargets") return false;
  243. if(axEventObject.cases.length == 0) return false;
  244. if(_isEventSimulating(axEventObject)) return false;
  245. if(axEventObject.cases.length >= 2) return true;
  246. return false;
  247. };
  248. var _isEventSimulating = function(axEventObject) {
  249. for(var i = 0; i < axEventObject.cases.length; i++) {
  250. if(axEventObject.cases[i].condition) return true;
  251. }
  252. return false;
  253. };
  254. var _handleCaseGroup = function(eventInfo, caseGroup, groupRefreshes) {
  255. for(var i = 0; i < caseGroup.length; i++) {
  256. var currentCase = caseGroup[i];
  257. if(!currentCase.condition || _processCondition(currentCase.condition, eventInfo)) {
  258. for(var j = 0; j < currentCase.actions.length; j++) {
  259. var action = currentCase.actions[j];
  260. if(action.action == 'wait') break;
  261. if(action.action != 'refreshRepeater') continue;
  262. for(var k = 0; k < action.repeatersToRefresh.length; k++) {
  263. var id = $ax.getElementIdsFromPath(action.repeatersToRefresh[i], eventInfo)[i];
  264. var index = groupRefreshes.indexOf(id);
  265. if(index != -1) $ax.splice(groupRefreshes, index);
  266. }
  267. }
  268. // Any guaranteed refreshes that aren't accounted for must be run still.
  269. $ax.action.tryRefreshRepeaters(groupRefreshes, eventInfo);
  270. $ax.action.dispatchAction(eventInfo, currentCase.actions);
  271. return false;
  272. }
  273. }
  274. // Any guaranteed refreshes that aren't accounted for must be run still.
  275. $ax.action.tryRefreshRepeaters(groupRefreshes, eventInfo);
  276. return true;
  277. };
  278. var _processCondition = function(expr, eventInfo) {
  279. return $ax.expr.evaluateExpr(expr, eventInfo);
  280. };
  281. var _attachTreeNodeEvent = function(elementId, eventName, fn) {
  282. //we need to set the cursor here because we want to make sure that every tree node has the default
  283. //cursor set and then it's overridden if it has a click
  284. if(eventName == 'click') window.document.getElementById(elementId).style.cursor = 'pointer';
  285. _attachCustomObjectEvent(elementId, eventName, fn);
  286. };
  287. var _attachDefaultObjectEvent = function(elementIdQuery, elementId, eventName, fn) {
  288. var func = function() {
  289. if(!$ax.style.IsWidgetDisabled(elementId)) return fn.apply(this, arguments);
  290. return true;
  291. };
  292. var bind = !elementIdQuery[eventName];
  293. if(bind) elementIdQuery.bind(eventName, func);
  294. else elementIdQuery[eventName](func);
  295. };
  296. var _attachCustomObjectEvent = function(elementId, eventName, fn) {
  297. var handlers = _objectIdToEventHandlers[elementId];
  298. if(!handlers) _objectIdToEventHandlers[elementId] = handlers = {};
  299. var fnList = handlers[eventName];
  300. if(!fnList) handlers[eventName] = fnList = [];
  301. fnList[fnList.length] = fn;
  302. };
  303. var _fireObjectEvent = function(elementId, event, originalArgs) {
  304. var element = window.document.getElementById(elementId);
  305. var handlerList = _objectIdToEventHandlers[elementId] && _objectIdToEventHandlers[elementId][event];
  306. if(handlerList) {
  307. for(var i = 0; i < handlerList.length; i++) handlerList[i].apply(element, originalArgs);
  308. }
  309. };
  310. //for button shapes and images the img is focusable instead of the div to get better outlines
  311. $ax.event.getFocusableWidgetOrChildId = function(elementId) {
  312. var imgId = $ax.repeater.applySuffixToElementId(elementId, '_img');
  313. var imgQuery = $jobj(imgId);
  314. var inputId = $ax.repeater.applySuffixToElementId(elementId, '_input');
  315. var inputQuery = $jobj(inputId);
  316. return inputQuery.length > 0 ? inputId : imgQuery.length > 0 ? imgId : elementId;
  317. };
  318. // key is the suppressing event, and the value is the event that is supressed
  319. var suppressingEvents = {};
  320. // key is the event that will cancel the suppression, and value is the event that was being suppressed
  321. var cancelSuppressions = {};
  322. // suppressed event maps to true if it is supressed
  323. var suppressedEventStatus = {};
  324. // Attempt at a generic way to supress events
  325. var initSuppressingEvents = function(query) {
  326. suppressingEvents['OnLongClick'] = 'OnClick';
  327. cancelSuppressions['onMouseDown'] = 'OnClick';
  328. // Have to cancel suppressed event here. Only works for non-synthetic events currently
  329. for(var key in cancelSuppressions) {
  330. var eventName = AXURE_TO_JQUERY_EVENT_NAMES[key];
  331. if(!eventName) continue;
  332. (function(eventName, suppressed) {
  333. query.bind(eventName, function() {
  334. suppressedEventStatus[suppressed] = false;
  335. });
  336. })(eventName, cancelSuppressions[key]);
  337. }
  338. // Otherwise see if you have the chance to cancel a supression
  339. // if(cancelSuppressions[eventDescription]) {
  340. // suppressedEventStatus[cancelSuppressions[eventDescription]] = false;
  341. // }
  342. };
  343. // TODO: It may be a good idea to split this into multiple functions, or at least pull out more similar functions into private methods
  344. var _initializeObjectEvents = function (query) {
  345. // Must init the supressing eventing before the handlers, so that it has the ability to supress those events.
  346. initSuppressingEvents(query);
  347. query.each(function(dObj, elementId) {
  348. var $element = $jobj(elementId);
  349. // Focus has to be done before on focus fires
  350. // Set up focus
  351. if(dObj.type == 'textArea' || dObj.type == 'textBox' || dObj.type == 'checkBox' || dObj.type == 'radioButton' ||
  352. dObj.type == 'listBox' || dObj.type == 'comboBox' || dObj.type == 'button' || dObj.type == 'imageBox' ||
  353. dObj.type == 'buttonShape' || dObj.type == 'flowShape' || dObj.type == 'treeNodeObject' || dObj.type == 'tableCell') {
  354. var focusObj = $jobj($ax.event.getFocusableWidgetOrChildId(elementId));
  355. focusObj.focus(function() {
  356. window.lastFocusedControl = elementId;
  357. });
  358. }
  359. // [MAS: Supressing events were here]
  360. if(dObj.interactionMap) {
  361. _attachEvents(dObj, elementId);
  362. };
  363. //attach button shape alternate styles
  364. var needsMouseFilter = dObj.type != 'hyperlink' && dObj.type != 'dynamicPanel' && dObj.type != 'richTextPanel' &&
  365. dObj.type != 'repeater' && dObj.type != 'checkbox' && dObj.type != 'radioButton' && dObj.type != 'treeNodeObject';
  366. if(needsMouseFilter) {
  367. $element.mouseenter(function() {
  368. var elementId = this.id;
  369. var parent = $ax.dynamicPanelManager.parentHandlesStyles(elementId);
  370. if(parent && parent.direct) return;
  371. if($.inArray(elementId, _event.mouseOverIds) != -1) return;
  372. _event.mouseOverIds[_event.mouseOverIds.length] = elementId;
  373. if(elementId == _event.mouseOverObjectId) return;
  374. _event.mouseOverObjectId = elementId;
  375. $ax.style.SetWidgetHover(elementId, true);
  376. var textId = $ax.style.GetTextIdFromShape(elementId);
  377. if(textId) $ax.annotation.updateLinkLocations(textId);
  378. }).mouseleave(function() {
  379. var elementId = this.id;
  380. var parent = $ax.dynamicPanelManager.parentHandlesStyles(elementId);
  381. if(parent && parent.direct) return;
  382. $ax.splice(_event.mouseOverIds, $.inArray(elementId, _event.mouseOverIds), 1);
  383. if(elementId == _event.mouseOverObjectId) {
  384. _event.mouseOverObjectId = '';
  385. }
  386. $ax.style.SetWidgetHover(elementId, false);
  387. var textId = $ax.style.GetTextIdFromShape(elementId);
  388. if(textId) $ax.annotation.updateLinkLocations(textId);
  389. });
  390. $element.bind($ax.features.eventNames.mouseDownName, function() {
  391. var elementId = this.id;
  392. var parent = $ax.dynamicPanelManager.parentHandlesStyles(elementId);
  393. if(parent) {
  394. dynamicPanelMouseDown(parent.id);
  395. if(parent.direct) return;
  396. }
  397. _event.mouseDownObjectId = elementId;
  398. $ax.style.SetWidgetMouseDown(this.id, true);
  399. $ax.annotation.updateLinkLocations($ax.style.GetTextIdFromShape(elementId));
  400. }).bind($ax.features.eventNames.mouseUpName, function() {
  401. var elementId = this.id;
  402. var parent = $ax.dynamicPanelManager.parentHandlesStyles(elementId);
  403. if(parent) {
  404. dynamicPanelMouseUp(parent.id);
  405. if(parent.direct) return;
  406. }
  407. var mouseDownId = _event.mouseDownObjectId;
  408. _event.mouseDownObjectId = '';
  409. if(!$ax.style.ObjHasMouseDown(elementId)) return;
  410. $ax.style.SetWidgetMouseDown(elementId, false);
  411. $ax.annotation.updateLinkLocations($ax.style.GetTextIdFromShape(elementId));
  412. //there used to be something we needed to make images click, because swapping out the images prevents the click
  413. // this is a note that we can eventually delete.
  414. });
  415. }
  416. var $axElement = undefined;
  417. // Initialize selected elements
  418. if((dObj.type == 'flowShape' || dObj.type == 'buttonShape' || dObj.type == 'imageBox' || dObj.type == 'dynamicPanel') && dObj.selected) {
  419. if(!$axElement) $axElement = $ax('#' + elementId);
  420. $axElement.selected(true);
  421. }
  422. //initialize disabled elements
  423. if((dObj.type == 'flowShape' || dObj.type == 'buttonShape' || dObj.type == 'imageBox' || dObj.type == 'dynamicPanel') && dObj.disabled) {
  424. if(!$axElement) $axElement = $ax('#' + elementId);
  425. $axElement.enabled(false);
  426. }
  427. if(OS_MAC && WEBKIT) {
  428. if(dObj.type == 'comboBox' && dObj.disabled) {
  429. $jobj($ax.INPUT(elementId)).css('color', 'grayText');
  430. }
  431. };
  432. // Initialize Placeholders. Right now this is text boxes and text areas.
  433. // Also, the assuption is being made that these widgets with the placeholder, have no other styles (this may change...)
  434. var hasPlaceholder = dObj.placeholderText == '' ? true : Boolean(dObj.placeholderText);
  435. if((dObj.type == 'textArea' || dObj.type == 'textBox') && hasPlaceholder) {
  436. // This is needed to initialize the placeholder state
  437. $jobj($ax.INPUT(elementId)).bind('keydown', function() {
  438. var id = this.id;
  439. var inputIndex = id.indexOf('_input');
  440. if(inputIndex == -1) return;
  441. var inputId = id.substring(0, inputIndex);
  442. if(!$ax.placeholderManager.isActive(inputId)) return;
  443. $ax.placeholderManager.updatePlaceholder(inputId, false, true);
  444. }).bind('keyup', function() {
  445. var id = this.id;
  446. var inputIndex = id.indexOf('_input');
  447. if(inputIndex == -1) return;
  448. var inputId = id.substring(0, inputIndex);
  449. if($ax.placeholderManager.isActive(inputId)) return;
  450. if(!$jobj(id).val()) {
  451. $ax.placeholderManager.updatePlaceholder(inputId, true);
  452. $ax.placeholderManager.moveCaret(id, 0);
  453. }
  454. }).bind('focus', function() {
  455. $ax.placeholderManager.moveCaret(this.id);
  456. }).bind('mousedown', function() {
  457. $ax.placeholderManager.moveCaret(this.id);
  458. }).bind('mouseup', function() {
  459. $ax.placeholderManager.moveCaret(this.id);
  460. }).bind('blur', function() {
  461. var id = this.id;
  462. var inputIndex = id.indexOf('_input');
  463. if(inputIndex == -1) return;
  464. var inputId = id.substring(0, inputIndex);
  465. if($jobj(id).val()) return;
  466. $ax.placeholderManager.updatePlaceholder(inputId, true);
  467. });
  468. $ax.placeholderManager.registerPlaceholder(elementId, dObj.placeholderText, $jobj($ax.INPUT(elementId)).attr('type') == 'password');
  469. $ax.placeholderManager.updatePlaceholder(elementId, !($jobj($ax.repeater.applySuffixToElementId(elementId, '_input')).val()));
  470. }
  471. // Initialize assigned submit buttons
  472. if(dObj.submitButton) {
  473. $element.keyup(function(e) {
  474. if(e.keyCode == '13') {
  475. var scriptId = $ax.repeater.getScriptIdFromElementId(elementId);
  476. var path = $ax.deepCopy(dObj.submitButton.path);
  477. path[path.length] = dObj.submitButton.id;
  478. var itemNum = $ax.repeater.getItemIdFromElementId(elementId);
  479. var submitId = $ax.getScriptIdFromPath(path, scriptId);
  480. if(itemNum && $ax.getParentRepeaterFromScriptId(submitId) == $ax.getParentRepeaterFromScriptId(scriptId)) {
  481. submitId = $ax.repeater.createElementId(submitId, itemNum);
  482. }
  483. var inputId = $ax.INPUT(submitId);
  484. if($jobj(inputId).length) submitId = inputId;
  485. $ax.setjBrowserEvent(e);
  486. $ax.event.fireClick(submitId);
  487. }
  488. }).keydown(function(e) {
  489. if(e.keyCode == '13') {
  490. e.preventDefault();
  491. }
  492. });
  493. }
  494. // Don't drag after mousing down on a plain text object
  495. if(dObj.type == 'textArea' || dObj.type == 'textBox' || dObj.type == 'listBox' || dObj.type == 'comboBox' || dObj.type == 'checkBox' || dObj.type == 'radioButton') {
  496. $element.bind($ax.features.eventNames.mouseDownName, function(event) {
  497. event.originalEvent.donotdrag = true;
  498. });
  499. }
  500. if($ax.features.supports.mobile) {
  501. $element.bind($ax.features.eventNames.mouseDownName, function() { _setCanClick(true); });
  502. if(dObj.type == 'dynamicPanel') {
  503. $element.scroll(function() { _setCanClick(false); });
  504. }
  505. }
  506. //initialize tree node cursors to default so they will override their parent
  507. if(dObj.type == 'treeNodeObject' && !(dObj.interactionMap && dObj.interactionMap.onClick)) {
  508. $element.css('cursor', 'default');
  509. }
  510. //initialize widgets that are clickable to have the pointer over them when hovering
  511. if(dObj.interactionMap && dObj.interactionMap.onClick) {
  512. if($element) $element.css('cursor', 'pointer');
  513. }
  514. // TODO: not sure if we need this. It appears to be working without
  515. //initialize panels for DynamicPanels
  516. if(dObj.type == 'dynamicPanel') {
  517. $element.children().each(function() {
  518. var parts = this.id.split('_');
  519. var state = parts[parts.length - 1].substring(5);
  520. if(state != 0) $ax.visibility.SetVisible(this, false);
  521. });
  522. }
  523. //initialize TreeNodes
  524. if(dObj.type == 'treeNodeObject') {
  525. if($element.hasClass('treeroot')) return;
  526. var childrenId = elementId + '_children';
  527. var children = $element.children('[id="' + childrenId + '"]:first');
  528. if(children.length > 0) {
  529. var plusMinusId = 'u' + (parseInt($ax.repeater.getScriptIdFromElementId(elementId).substring(1)) + 1);
  530. var itemId = $ax.repeater.getItemIdFromElementId(elementId);
  531. if(itemId) plusMinusId = $ax.repeater.createElementId(plusMinusId, itemId);
  532. if(!$jobj(plusMinusId).hasClass('ax_image')) plusMinusId = '';
  533. $ax.tree.InitializeTreeNode(elementId, plusMinusId, childrenId);
  534. }
  535. $element.click(function() { $ax.tree.SelectTreeNode(elementId, true); });
  536. }
  537. //initialize submenus
  538. if(dObj.type == 'menuObject') {
  539. if($element.hasClass('sub_menu')) {
  540. var tableCellElementId = $ax.getElementIdFromPath([dObj.parentCellId], { relativeTo: elementId });
  541. $ax.menu.InitializeSubmenu(elementId, tableCellElementId);
  542. }
  543. }
  544. // Attach handles for dynamic panels that propagate styles to inner items.
  545. if(dObj.type == 'dynamicPanel' && dObj.propagate) {
  546. $element.mouseenter(function() {
  547. var elementId = this.id;
  548. dynamicPanelMouseOver(elementId);
  549. }).mouseleave(function() {
  550. var elementId = this.id;
  551. dynamicPanelMouseLeave(elementId);
  552. }).bind($ax.features.eventNames.mouseDownName, function() {
  553. var elementId = this.id;
  554. dynamicPanelMouseDown(elementId);
  555. }).bind($ax.features.eventNames.mouseUpName, function() {
  556. var elementId = this.id;
  557. dynamicPanelMouseUp(elementId);
  558. });
  559. }
  560. // These are the dynamic panel functions for propagating rollover styles and mouse down styles to inner objects
  561. var dynamicPanelMouseOver = function(elementId, fromChild) {
  562. var parent = $ax.dynamicPanelManager.parentHandlesStyles(elementId);
  563. if(parent) {
  564. dynamicPanelMouseOver(parent.id, true);
  565. if(parent.direct) return;
  566. }
  567. if($.inArray(elementId, _event.mouseOverIds) != -1) return;
  568. // If this event is coming from a child, don't mark that it's actually entered.
  569. // Only mark that this has been entered if this event has naturally been triggered. (For reason see mouseleave)
  570. if(!fromChild) _event.mouseOverIds[_event.mouseOverIds.length] = elementId;
  571. if(elementId == _event.mouseOverObjectId) return;
  572. _event.mouseOverObjectId = elementId;
  573. $ax.dynamicPanelManager.propagateMouseOver(elementId, true);
  574. };
  575. var dynamicPanelMouseLeave = function(elementId, fromChild) {
  576. var parent = $ax.dynamicPanelManager.parentHandlesStyles(elementId);
  577. if(parent) {
  578. dynamicPanelMouseLeave(parent.id, true);
  579. if(parent.direct) return;
  580. }
  581. var index = $.inArray(elementId, _event.mouseOverIds);
  582. // If index != -1, this has been natuarally entered. If naturally entered, then leaving child should not trigger leaving,
  583. // but instead wait for natural mouse leave. If natural mouse enter never triggered, natural mouse leave won't so do this now.
  584. if((index != -1) && fromChild) return;
  585. $ax.splice(_event.mouseOverIds, index, 1);
  586. if(elementId == _event.mouseOverObjectId) {
  587. _event.mouseOverObjectId = '';
  588. }
  589. $ax.dynamicPanelManager.propagateMouseOver(elementId, false);
  590. };
  591. var dynamicPanelMouseDown = function(elementId) {
  592. var parent = $ax.dynamicPanelManager.parentHandlesStyles(elementId);
  593. if(parent) {
  594. dynamicPanelMouseDown(parent.id);
  595. if(parent.direct) return;
  596. }
  597. _event.mouseDownObjectId = elementId;
  598. $ax.dynamicPanelManager.propagateMouseDown(elementId, true);
  599. };
  600. var dynamicPanelMouseUp = function(elementId) {
  601. var parent = $ax.dynamicPanelManager.parentHandlesStyles(elementId);
  602. if(parent) {
  603. dynamicPanelMouseUp(parent.id);
  604. if(parent.direct) return;
  605. }
  606. _event.mouseDownObjectId = '';
  607. $ax.dynamicPanelManager.propagateMouseDown(elementId, false);
  608. };
  609. //attach handlers for button shape and tree node mouse over styles
  610. // TODO: Can this really be removed? Trees seem to work with out (the generic hover case works for it).
  611. // query.filter(function(obj) {
  612. // return obj.type == 'buttonShape' && obj.parent.type == 'treeNodeObject' &&
  613. // obj.parent.style && obj.parent.style.stateStyles &&
  614. // obj.parent.style.stateStyles.mouseOver;
  615. // }).mouseenter(function() {
  616. // $ax.style.SetWidgetHover(this.id, true);
  617. // }).mouseleave(function() {
  618. // $ax.style.SetWidgetHover(this.id, false);
  619. // });
  620. //handle treeNodeObject events and prevent them from bubbling up. this is necessary because otherwise
  621. //both a sub menu and it's parent would get a click
  622. if(dObj.type == 'treeNodeObject') {
  623. $element.click(function() {
  624. //todo -- this was bubbling, but then selecting a child tree node would bubble and select the parent (don't know if there is a better way)
  625. _fireObjectEvent(this.id, 'click', arguments);
  626. return false;
  627. }).each(function() {
  628. if(!this.style.cursor) {
  629. this.style.cursor = 'default';
  630. }
  631. });
  632. }
  633. // Synthetic events
  634. var map = dObj.interactionMap;
  635. // Attach dynamic panel synthetic drag and swipe events
  636. if(dObj.type == "dynamicPanel" && map && (
  637. map.onDragStart || map.onDrag ||
  638. map.onDragDrop || map.onSwipeLeft || map.onSwipeRight || map.onSwipeUp || map.onSwipeDown)) {
  639. $element.bind($ax.features.eventNames.mouseDownName, function(e) { $ax.drag.StartDragWidget(e.originalEvent, elementId); });
  640. }
  641. // Attach dynamic panel synthetic scroll event
  642. if(dObj.type == 'dynamicPanel' && map && map.onScroll) {
  643. var diagrams = dObj.diagrams;
  644. for(var i = 0; i < diagrams.length; i++) {
  645. var panelId = $ax.repeater.applySuffixToElementId(elementId, '_state' + i);
  646. (function(id) {
  647. _attachDefaultObjectEvent($('#' + id), elementId, 'scroll', function(e) {
  648. $ax.setjBrowserEvent(e);
  649. _handleEvent(elementId, $ax.getEventInfoFromEvent($ax.getjBrowserEvent(), false, elementId), map.onScroll);
  650. });
  651. })(panelId);
  652. }
  653. }
  654. // Attach synthetic hover event
  655. if (map && map.onMouseHover) {
  656. var MIN_HOVER_HOLD_TIME = 1000;
  657. // So when the timeout fires, you know whether it is the same mouseenter that is active or not.
  658. var hoverMouseCount = 0;
  659. // Update eventInfo regularly, so position is accurate.
  660. var hoverEventInfo;
  661. $element.mouseenter(function(e) {
  662. $ax.setjBrowserEvent(e);
  663. hoverEventInfo = $ax.getEventInfoFromEvent($ax.getjBrowserEvent(), false, elementId);
  664. (function(currCount) {
  665. window.setTimeout(function() {
  666. if(currCount == hoverMouseCount) _raiseSyntheticEvent(elementId, 'onMouseHover', false, hoverEventInfo, true);
  667. }, MIN_HOVER_HOLD_TIME);
  668. })(hoverMouseCount);
  669. }).mouseleave(function(e) {
  670. $ax.setjBrowserEvent(e);
  671. hoverMouseCount++;
  672. }).mousemove(function(e) {
  673. $ax.setjBrowserEvent(e);
  674. hoverEventInfo = $ax.getEventInfoFromEvent($ax.getjBrowserEvent(), false, elementId);
  675. });
  676. }
  677. // Attach synthetic tap and hold event.
  678. if (map && map.onLongClick) {
  679. var MIN_LONG_CLICK_HOLD_TIME = 750;
  680. // So when the timeout fires, you know whether it is the same mousedown that is active or not.
  681. var longClickMouseCount = 0;
  682. $element.bind($ax.features.eventNames.mouseDownName, function(e) {
  683. (function(currCount) {
  684. $ax.setjBrowserEvent(e);
  685. var eventInfo = $ax.getEventInfoFromEvent($ax.getjBrowserEvent(), false, elementId);
  686. window.setTimeout(function() {
  687. if(currCount == longClickMouseCount) _raiseSyntheticEvent(elementId, 'onLongClick', false, eventInfo, true);
  688. }, MIN_LONG_CLICK_HOLD_TIME);
  689. if(e.preventDefault) e.preventDefault();
  690. })(longClickMouseCount);
  691. }).bind($ax.features.eventNames.mouseUpName, function(e) {
  692. $ax.setjBrowserEvent(e);
  693. longClickMouseCount++;
  694. });
  695. };
  696. // Attach synthetic onSelectionChange event to droplist and listbox elements
  697. if ($ax.event.HasSelectionChanged(dObj)) {
  698. $element.bind('change', function(e) {
  699. $ax.setjBrowserEvent(e);
  700. _raiseSyntheticEvent(elementId, 'onSelectionChange');
  701. });
  702. };
  703. // Highjack key up and key down to keep track of state of keyboard.
  704. _event.initKeyEvents($element);
  705. // Attach synthetic onTextChange event to textbox and textarea elements
  706. if ($ax.event.HasTextChanged(dObj)) {
  707. var element = $jobj($ax.INPUT(elementId));
  708. $ax.updateElementText(elementId, element.val());
  709. //Key down needed because when holding a key down, key up only fires once, but keydown fires repeatedly.
  710. //Key up because last mouse down will only show the state before the last character.
  711. element.bind('keydown', function(e) {
  712. $ax.setjBrowserEvent(e);
  713. $ax.event.TryFireTextChanged(elementId);
  714. }).bind('keyup', function(e) {
  715. $ax.setjBrowserEvent(e);
  716. $ax.event.TryFireTextChanged(elementId);
  717. });
  718. };
  719. // Attach synthetic onCheckedChange event to radiobutton and checkbox elements
  720. if (dObj.type == 'checkbox' || dObj.type == 'radioButton') {
  721. var input = $jobj($ax.INPUT(elementId));
  722. if(dObj.type == 'radioButton' && input.prop('checked')) {
  723. $ax.updateRadioButtonSelected(input.attr('name'), elementId);
  724. }
  725. $element.bind('change', function(e) {
  726. $ax.setjBrowserEvent(e);
  727. _tryFireCheckedChanged(elementId, true);
  728. });
  729. };
  730. var hasTap = map && (map.onClick || map.onDoubleClick);
  731. var hasMove = map && map.onMouseMove;
  732. _event.initMobileEvents(hasTap ? $element : $(),
  733. hasMove ? $element : $(), elementId);
  734. //attach link alternate styles
  735. if(dObj.type == 'hyperlink') {
  736. $element.mouseenter(function() {
  737. var elementId = this.id;
  738. if(_event.mouseOverIds.indexOf(elementId) != -1) return true;
  739. _event.mouseOverIds[_event.mouseOverIds.length] = elementId;
  740. var mouseOverObjectId = _event.mouseOverObjectId;
  741. if(mouseOverObjectId && $ax.style.IsWidgetDisabled(mouseOverObjectId)) return true;
  742. $ax.style.SetLinkHover(elementId);
  743. var bubble = _fireObjectEvent(elementId, 'mouseenter', arguments);
  744. $ax.annotation.updateLinkLocations($ax.style.GetTextIdFromLink(elementId));
  745. return bubble;
  746. }).mouseleave(function() {
  747. var elementId = this.id;
  748. $ax.splice(_event.mouseOverIds, _event.mouseOverIds.indexOf(elementId), 1);
  749. var mouseOverObjectId = _event.mouseOverObjectId;
  750. if(mouseOverObjectId && $ax.style.IsWidgetDisabled(mouseOverObjectId)) return true;
  751. $ax.style.SetLinkNotHover(elementId);
  752. var bubble = _fireObjectEvent(elementId, 'mouseleave', arguments);
  753. $ax.annotation.updateLinkLocations($ax.style.GetTextIdFromLink(elementId));
  754. return bubble;
  755. }).bind($ax.features.eventNames.mouseDownName, function() {
  756. var elementId = this.id;
  757. var mouseOverObjectId = _event.mouseOverObjectId;
  758. if($ax.style.IsWidgetDisabled(mouseOverObjectId)) return undefined;
  759. if(mouseOverObjectId) $ax.style.SetWidgetMouseDown(mouseOverObjectId, true);
  760. $ax.style.SetLinkMouseDown(elementId);
  761. $ax.annotation.updateLinkLocations($ax.style.GetTextIdFromLink(elementId));
  762. return false;
  763. }).bind($ax.features.eventNames.mouseUpName, function() {
  764. var elementId = this.id;
  765. var mouseOverObjectId = _event.mouseOverObjectId;
  766. if(mouseOverObjectId && $ax.style.IsWidgetDisabled(mouseOverObjectId)) return;
  767. if(mouseOverObjectId) $ax.style.SetWidgetMouseDown(mouseOverObjectId, false);
  768. $ax.style.SetLinkNotMouseDown(elementId);
  769. $ax.annotation.updateLinkLocations($ax.style.GetTextIdFromLink(elementId));
  770. }).click(function() {
  771. var elementId = this.id;
  772. var mouseOverObjectId = _event.mouseOverObjectId;
  773. if(mouseOverObjectId && $ax.style.IsWidgetDisabled(mouseOverObjectId)) return undefined;
  774. return _fireObjectEvent(elementId, 'click', arguments);
  775. });
  776. }
  777. // Init inline frames
  778. if (dObj.type == 'inlineFrame') {
  779. var target = dObj.target;
  780. var url = '';
  781. if(target.includeVariables && target.url) {
  782. var origSrc = target.url;
  783. url = origSrc.toLowerCase().indexOf('http://') == -1 ? $ax.globalVariableProvider.getLinkUrl(origSrc) : origSrc;
  784. } else if(target.urlLiteral) {
  785. url = $ax.expr.evaluateExpr(target.urlLiteral, $ax.getEventInfoFromEvent(undefined, true, elementId), true);
  786. }
  787. if(url) $jobj($ax.INPUT(elementId)).attr('src', url);
  788. };
  789. });
  790. }
  791. $ax.initializeObjectEvents = _initializeObjectEvents;
  792. // Handle key up and key down events
  793. (function() {
  794. var _keyState = {};
  795. _keyState.ctrl = false;
  796. _keyState.alt = false;
  797. _keyState.shift = false;
  798. _keyState.keyCode = 0;
  799. $ax.event.keyState = function() {
  800. return $ax.deepCopy(_keyState);
  801. };
  802. var modifierCodes = [16, 17, 18];
  803. $ax.event.initKeyEvents = function($query) {
  804. $query.keydown(function(e) {
  805. var elementId = this.id;
  806. _keyState.ctrl = e.ctrlKey;
  807. _keyState.alt = e.altKey;
  808. _keyState.shift = e.shiftKey;
  809. // If a modifier was pressed, then don't set the keyCode;
  810. if(modifierCodes.indexOf(e.keyCode) == -1) _keyState.keyCode = e.keyCode;
  811. $ax.setjBrowserEvent(e);
  812. if(!elementId) fireEventThroughContainers('onKeyDown', undefined, false, ['page', 'referenceDiagramObject', 'dynamicPanel', 'repeater'], ['page', 'referenceDiagramObject']);
  813. else _raiseSyntheticEvent(elementId, 'onKeyDown', false, undefined, true);
  814. });
  815. $query.keyup(function(e) {
  816. var elementId = this.id;
  817. $ax.setjBrowserEvent(e);
  818. // Fire event before updating modifiers.
  819. if(!elementId) fireEventThroughContainers('onKeyUp', undefined, false, ['page', 'referenceDiagramObject', 'dynamicPanel', 'repeater'], ['page', 'referenceDiagramObject']);
  820. else _raiseSyntheticEvent(elementId, 'onKeyUp', false, undefined, true);
  821. _keyState.ctrl = e.ctrlKey;
  822. _keyState.alt = e.altKey;
  823. _keyState.shift = e.shiftKey;
  824. // If a non-modifier was lifted, clear the keycode
  825. if(modifierCodes.indexOf(e.keyCode) == -1) _keyState.keyCode = 0;
  826. });
  827. };
  828. })();
  829. // Handle adding mobile events
  830. (function() {
  831. // NOTE: Multi touch is NOT handled currently.
  832. var CLICK_THRESHOLD_PX = 25;
  833. var CLICK_THRESHOLD_PX_SQ = CLICK_THRESHOLD_PX * CLICK_THRESHOLD_PX;
  834. var DBLCLICK_THRESHOLD_MS = 500;
  835. // Location in page cooridinates
  836. var tapDownLoc;
  837. var lastClickEventTime;
  838. _event.initMobileEvents = function($tapQuery, $moveQuery, elementId) {
  839. if(!$ax.features.supports.mobile) return;
  840. // Handle touch start
  841. $tapQuery.bind('touchstart', function(e) {
  842. // We do NOT support multiple touches. This isn't necessarily the touch we want.
  843. var touch = e.originalEvent && e.originalEvent.changedTouches && e.originalEvent.changedTouches[0];
  844. if(!touch) return;
  845. tapDownLoc = [touch.pageX, touch.pageY];
  846. var time = (new Date()).getTime();
  847. if(time - lastClickEventTime < DBLCLICK_THRESHOLD_MS) {
  848. var dObj = elementId === '' ? $ax.pageData.page : $ax.getObjectFromElementId(elementId);
  849. var axEventObject = dObj && dObj.interactionMap && dObj.interactionMap['onDoubleClick'];
  850. if(axEventObject) e.preventDefault(); //for Chrome on Android
  851. }
  852. }).bind('touchend', function(e) {
  853. var touch = e.originalEvent && e.originalEvent.changedTouches && e.originalEvent.changedTouches[0];
  854. if(!touch || !tapDownLoc) return;
  855. var tapUpLoc = [touch.pageX, touch.pageY];
  856. var xDiff = tapUpLoc[0] - tapDownLoc[0];
  857. var yDiff = tapUpLoc[1] - tapDownLoc[1];
  858. if((xDiff * xDiff + yDiff * yDiff) < CLICK_THRESHOLD_PX_SQ) {
  859. $ax.setjBrowserEvent(e);
  860. _raiseSyntheticEvent(elementId, 'onClick', false, undefined, true);
  861. var time = (new Date()).getTime();
  862. if(time - lastClickEventTime < DBLCLICK_THRESHOLD_MS) {
  863. _raiseSyntheticEvent(elementId, 'onDoubleClick', false, undefined, true);
  864. if(e.originalEvent && e.originalEvent.handled) e.preventDefault(); //for iOS
  865. }
  866. lastClickEventTime = time;
  867. }
  868. });
  869. // Handles touch move
  870. $moveQuery.bind('touchmove', function(e) {
  871. $ax.setjBrowserEvent(e);
  872. _raiseSyntheticEvent(elementId, 'onMouseMove', false, undefined, true);
  873. if(e.originalEvent && e.originalEvent.handled) e.preventDefault();
  874. });
  875. };
  876. })();
  877. // Handle adding device independent click events to non-widgets
  878. (function() {
  879. var CLICK_THRESHOLD_PX = 25;
  880. var CLICK_THRESHOLD_PX_SQ = CLICK_THRESHOLD_PX * CLICK_THRESHOLD_PX;
  881. // Location in page cooridinates
  882. var tapDownLoc;
  883. _event.attachClick = function(query, clickHandler) {
  884. if(!$ax.features.supports.mobile) {
  885. query.click(clickHandler);
  886. return;
  887. }
  888. $(query).bind('touchstart', function(e) {
  889. // We do NOT support multiple touches. This isn't necessarily the touch we want.
  890. var touch = e.originalEvent && e.originalEvent.changedTouches && e.originalEvent.changedTouches[0];
  891. if(!touch) return;
  892. tapDownLoc = [touch.pageX, touch.pageY];
  893. });
  894. $(query).bind('touchend', function(e) {
  895. var touch = e.originalEvent && e.originalEvent.changedTouches && e.originalEvent.changedTouches[0];
  896. if(!touch) return;
  897. var tapUpLoc = [touch.pageX, touch.pageY];
  898. var xDiff = tapUpLoc[0] - tapDownLoc[0];
  899. var yDiff = tapUpLoc[1] - tapDownLoc[1];
  900. if((xDiff * xDiff + yDiff * yDiff) < CLICK_THRESHOLD_PX_SQ) {
  901. clickHandler();
  902. }
  903. });
  904. };
  905. })();
  906. // Handle firing device independent click events on widgets
  907. (function() {
  908. _event.fireClick = function(elementId) {
  909. if(!$ax.features.supports.mobile) {
  910. $('#' + elementId).click();
  911. return;
  912. }
  913. _raiseSyntheticEvent(elementId, 'onClick', false, undefined, true);
  914. };
  915. })();
  916. var _mouseLocation = $ax.mouseLocation = { x: 0, y: 0 };
  917. var _lastmouseLocation = $ax.lastMouseLocation = { x: 0, y: 0 };
  918. var _updateMouseLocation = function(e, end) {
  919. if(!e) return;
  920. if(IE && typeof (e.type) == 'unknown') return;
  921. if(e.type != 'mousemove' && e.type != 'touchstart' && e.type != 'touchmove' && e.type != 'touchend') return;
  922. var newX;
  923. var newY;
  924. if(IE) {
  925. newX = e.clientX + $('html').scrollLeft();
  926. newY = e.clientY + $('html').scrollTop();
  927. } else {
  928. newX = e.pageX;
  929. newY = e.pageY;
  930. }
  931. var body = $('body');
  932. if(body.css('position') == 'relative') newX = Math.round(newX - Number(body.css('left').replace('px', '')) - Math.max(0, ($(window).width() - body.width()) / 2));
  933. if(_mouseLocation.x == newX && _mouseLocation.y == newY) return;
  934. _lastmouseLocation.x = _mouseLocation.x;
  935. _lastmouseLocation.y = _mouseLocation.y;
  936. _mouseLocation.x = newX;
  937. _mouseLocation.y = newY;
  938. $ax.geometry.tick(_mouseLocation.x, _mouseLocation.y, end);
  939. };
  940. _event.updateMouseLocation = _updateMouseLocation;
  941. var _leavingState = function(stateId) {
  942. var mouseOverIds = _event.mouseOverIds;
  943. if(mouseOverIds.length == 0) return;
  944. var stateQuery = $jobj(stateId);
  945. for(var i = mouseOverIds.length - 1; i >= 0; i--) {
  946. var id = mouseOverIds[i];
  947. if(stateQuery.find('#' + id).length) {
  948. $ax.splice(mouseOverIds, $.inArray(id, mouseOverIds), 1);
  949. $ax.style.SetWidgetMouseDown(id, false);
  950. $ax.style.SetWidgetHover(id, false);
  951. }
  952. }
  953. };
  954. _event.leavingState = _leavingState;
  955. var _raiseSyntheticEvent = function(elementId, eventName, skipShowDescription, eventInfo, nonSynthetic) {
  956. // Empty string used when this is an event directly on the page.
  957. var dObj = elementId === '' ? $ax.pageData.page : $ax.getObjectFromElementId(elementId);
  958. var axEventObject = dObj && dObj.interactionMap && dObj.interactionMap[eventName];
  959. if(!axEventObject) return;
  960. eventInfo = eventInfo || $ax.getEventInfoFromEvent($ax.getjBrowserEvent(), skipShowDescription, elementId);
  961. _handleEvent(elementId, eventInfo, axEventObject, false, !nonSynthetic);
  962. };
  963. $ax.event.raiseSyntheticEvent = _raiseSyntheticEvent;
  964. var _hasSyntheticEvent = function(scriptId, eventName) {
  965. var dObj = $ax.getObjectFromScriptId(scriptId);
  966. var axEventObject = dObj && dObj.interactionMap && dObj.interactionMap[eventName];
  967. return Boolean(axEventObject);
  968. };
  969. $ax.event.hasSyntheticEvent = _hasSyntheticEvent;
  970. var _initialize = function() {
  971. $ax.repeater.load();
  972. // Make sure key events for page are initialized first. That way they will update the value of key pressed before any other events occur.
  973. _event.initKeyEvents($(window));
  974. _initializeObjectEvents($ax('*'));
  975. //finally, process the pageload
  976. _pageLoad();
  977. // _loadDynamicPanelsAndMasters();
  978. // $ax.repeater.init();
  979. // and wipe out the basic links.
  980. $('.basiclink').click(function() {
  981. return false;
  982. });
  983. };
  984. _event.initialize = _initialize;
  985. $ax.event.HasTextChanged = function(diagramObject) {
  986. if(diagramObject.type != 'textBox' && diagramObject.type != 'textArea') return false;
  987. var map = diagramObject.interactionMap;
  988. return map && map.onTextChange;
  989. };
  990. $ax.event.TryFireTextChanged = function(elementId) {
  991. var query = $jobj($ax.repeater.applySuffixToElementId(elementId, '_input'));
  992. if(!$ax.hasElementTextChanged(elementId, query.val())) return;
  993. $ax.updateElementText(elementId, query.val());
  994. $ax.event.raiseSyntheticEvent(elementId, 'onTextChange');
  995. };
  996. $ax.event.HasSelectionChanged = function(diagramObject) {
  997. if(diagramObject.type != 'listBox' && diagramObject.type != 'comboBox') return false;
  998. var map = diagramObject.interactionMap;
  999. return map && map.onSelectionChange;
  1000. };
  1001. $ax.event.HasCheckedChanged = function(diagramObject) {
  1002. if(diagramObject.type != 'checkbox' && diagramObject.type != 'radioButton') return false;
  1003. var map = diagramObject.interactionMap;
  1004. return map && map.onCheckedChange;
  1005. };
  1006. var _tryFireCheckedChanged = $ax.event.TryFireCheckChanged = function(elementId, value) {
  1007. var isRadio = $obj(elementId).type == 'radioButton';
  1008. if(isRadio) {
  1009. if(!value) {
  1010. $ax.updateRadioButtonSelected($jobj($ax.INPUT(elementId)).attr('name'), undefined);
  1011. } else {
  1012. var last = $ax.updateRadioButtonSelected($jobj($ax.INPUT(elementId)).attr('name'), elementId);
  1013. // If no change, this should not fire
  1014. if(last == elementId) return;
  1015. // Initially selecting one, last may be undefined
  1016. if(last) $ax.event.raiseSyntheticEvent(last, 'onCheckedChange');
  1017. }
  1018. }
  1019. $ax.event.raiseSyntheticEvent(elementId, 'onCheckedChange');
  1020. };
  1021. var _loadDynamicPanelsAndMasters = function(objects, path, itemId) {
  1022. fireEventThroughContainers('onLoad', objects, true, ['page', 'referenceDiagramObject', 'dynamicPanel'], ['page', 'referenceDiagramObject', 'dynamicPanel', 'repeater'], path, itemId);
  1023. };
  1024. $ax.loadDynamicPanelsAndMasters = _loadDynamicPanelsAndMasters;
  1025. var _viewChangePageAndMasters = function() {
  1026. fireEventThroughContainers('onAdaptiveViewChange', undefined, true, ['page', 'referenceDiagramObject', 'dynamicPanel'], ['page', 'referenceDiagramObject']);
  1027. _postAdaptiveViewChanged();
  1028. };
  1029. $ax.viewChangePageAndMasters = _viewChangePageAndMasters;
  1030. var _postAdaptiveViewChanged = function() {
  1031. //only trigger adaptive view changed if the window is on the mainframe. Also triggered on init, even if default.
  1032. try {
  1033. if(window.name == 'mainFrame' ||
  1034. (!CHROME_5_LOCAL && window.parent.$ && window.parent.$('#mainFrame').length > 0)) {
  1035. $axure.messageCenter.postMessage('adaptiveViewChange', $ax.adaptive.currentViewId);
  1036. }
  1037. } catch(e) { }
  1038. };
  1039. $ax.postAdaptiveViewChanged = _postAdaptiveViewChanged;
  1040. var _postResize = $ax.postResize = function(e) {
  1041. $ax.setjBrowserEvent(e);
  1042. return fireEventThroughContainers('onResize', undefined, false, ['page', 'referenceDiagramObject', 'dynamicPanel', 'repeater'], ['page', 'referenceDiagramObject']);
  1043. };
  1044. // Filters include page, referenceDiagramObject, dynamicPanel, and repeater.
  1045. var fireEventThroughContainers = function(eventName, objects, synthetic, searchFilter, callFilter, path, itemId) {
  1046. // TODO: may want to pass in this as a parameter. At that point, may want to convert some of them to an option parameter. For now this is the only case
  1047. var skipShowDescription = eventName == 'onLoad';
  1048. // If objects undefined, load page
  1049. if(!objects) {
  1050. if(callFilter.indexOf('page') != -1) {
  1051. var map = $ax.pageData.page.interactionMap;
  1052. var pageEventInfo = $ax.getEventInfoFromEvent($ax.getjBrowserEvent(), skipShowDescription, '');
  1053. var pageEvent = map && map[eventName];
  1054. if(pageEvent) _handleEvent('', pageEventInfo, pageEvent, skipShowDescription, synthetic);
  1055. }
  1056. if(searchFilter.indexOf('page') != -1) fireEventThroughContainers(eventName, $ax.pageData.page.diagram.objects, synthetic, searchFilter, callFilter);
  1057. return;
  1058. }
  1059. if(!path) path = [];
  1060. var pathCopy = [];
  1061. for(var j = 0; j < path.length; j++) pathCopy[j] = path[j];
  1062. for(var i = 0; i < objects.length; i++) {
  1063. var obj = objects[i];
  1064. if(obj.type != 'referenceDiagramObject' && obj.type != 'dynamicPanel' && obj.type != 'repeater') continue;
  1065. pathCopy[path.length] = obj.id;
  1066. var objId = $ax.getScriptIdFromPath(pathCopy);
  1067. objId = $ax.repeater.createElementId(objId, itemId);
  1068. if(obj.type == 'referenceDiagramObject') {
  1069. if(callFilter.indexOf('referenceDiagramObject') != -1) {
  1070. var axEvent = $ax.pageData.masters[obj.masterId].interactionMap[eventName];
  1071. if(axEvent) {
  1072. var eventInfo = $ax.getEventInfoFromEvent($ax.getjBrowserEvent(), skipShowDescription, objId);
  1073. eventInfo.isMasterEvent = true;
  1074. _handleEvent(objId, eventInfo, axEvent, skipShowDescription, synthetic);
  1075. }
  1076. }
  1077. if(searchFilter.indexOf('referenceDiagramObject') != -1) fireEventThroughContainers(eventName, $ax.pageData.masters[obj.masterId].diagram.objects, synthetic, searchFilter, callFilter, pathCopy, itemId);
  1078. } else if(obj.type == 'dynamicPanel') {
  1079. if(callFilter.indexOf('dynamicPanel') != -1) $ax.event.raiseSyntheticEvent(objId, eventName, skipShowDescription, undefined, !synthetic);
  1080. if(searchFilter.indexOf('dynamicPanel') != -1) {
  1081. var diagrams = obj.diagrams;
  1082. for(var j = 0; j < diagrams.length; j++) {
  1083. fireEventThroughContainers(eventName, diagrams[j].objects, synthetic, searchFilter, callFilter, path, itemId);
  1084. }
  1085. }
  1086. } else if(obj.type == 'repeater') {
  1087. // TODO: possible an option for repeater item? Now fires overall for the repeater
  1088. if(callFilter.indexOf('repeater') != -1) $ax.event.raiseSyntheticEvent(objId, eventName, skipShowDescription, undefined, !synthetic);
  1089. if(searchFilter.indexOf('repeater') != -1) {
  1090. var itemIds = $ax.getItemIdsForRepeater(objId);
  1091. for(var j = 0; j < itemIds.length; j++) {
  1092. fireEventThroughContainers(eventName, obj.objects, synthetic, searchFilter, callFilter, path, itemIds[j]);
  1093. }
  1094. }
  1095. }
  1096. }
  1097. };
  1098. // FOCUS stuff
  1099. (function() {
  1100. })();
  1101. var _pageLoad = function() {
  1102. // Map of axure event names to pair of what it should attach to, and what the jquery event name is.
  1103. var PAGE_AXURE_TO_JQUERY_EVENT_NAMES = {
  1104. 'onScroll': [window, 'scroll'],
  1105. //'onResize': [window, 'resize'],
  1106. 'onContextMenu': [window, 'contextmenu']
  1107. };
  1108. var $win = $(window);
  1109. if(!$ax.features.supports.mobile) {
  1110. PAGE_AXURE_TO_JQUERY_EVENT_NAMES.onClick = ['html', 'click'];
  1111. PAGE_AXURE_TO_JQUERY_EVENT_NAMES.onDoubleClick = ['html', 'dblclick'];
  1112. PAGE_AXURE_TO_JQUERY_EVENT_NAMES.onMouseMove = ['html', 'mousemove'];
  1113. } else {
  1114. _event.initMobileEvents($win, $win, '');
  1115. $win.bind($ax.features.eventNames.mouseDownName, _updateMouseLocation);
  1116. $win.bind($ax.features.eventNames.mouseUpName, function(e) { _updateMouseLocation(e, true); });
  1117. $win.scroll(function() { _setCanClick(false); });
  1118. $win.bind($ax.features.eventNames.mouseDownName, (function() {
  1119. _setCanClick(true);
  1120. }));
  1121. }
  1122. $win.bind($ax.features.eventNames.mouseMoveName, _updateMouseLocation);
  1123. $win.scroll($ax.flyoutManager.reregisterAllFlyouts);
  1124. for(key in PAGE_AXURE_TO_JQUERY_EVENT_NAMES) {
  1125. if(!PAGE_AXURE_TO_JQUERY_EVENT_NAMES.hasOwnProperty(key)) continue;
  1126. (function(axureName) {
  1127. var jqueryEventNamePair = PAGE_AXURE_TO_JQUERY_EVENT_NAMES[axureName];
  1128. $(jqueryEventNamePair[0])[jqueryEventNamePair[1]](function(e) {
  1129. $ax.setjBrowserEvent(e);
  1130. return fireEventThroughContainers(axureName, undefined, false, ['page', 'referenceDiagramObject', 'dynamicPanel', 'repeater'], ['page', 'referenceDiagramObject']);
  1131. });
  1132. })(key);
  1133. }
  1134. };
  1135. _event.pageLoad = _pageLoad;
  1136. });