/md-ui/src/main/webapp/js/fc/flowchart_directive.spec.js

https://gitlab.com/kingofhappy/openbdre · JavaScript · 618 lines · 390 code · 189 blank · 39 comment · 7 complexity · ca21788d16db1cb584e872f5cf752da4 MD5 · raw file

  1. describe('flowchart-directive', function () {
  2. var testObject;
  3. var mockScope;
  4. var mockDragging;
  5. var mockSvgElement;
  6. //
  7. // Bring in the flowChart module before each test.
  8. //
  9. beforeEach(module('flowChart'));
  10. //
  11. // Helper function to create the controller for each test.
  12. //
  13. var createController = function ($rootScope, $controller) {
  14. mockScope = $rootScope.$new();
  15. mockDragging = createMockDragging();
  16. mockSvgElement = {
  17. get: function () {
  18. return createMockSvgElement();
  19. }
  20. };
  21. testObject = $controller('FlowChartController', {
  22. $scope: mockScope,
  23. dragging: mockDragging,
  24. $element: mockSvgElement,
  25. });
  26. };
  27. //
  28. // Setup the controller before each test.
  29. //
  30. beforeEach(inject(function ($rootScope, $controller) {
  31. createController($rootScope, $controller);
  32. }));
  33. //
  34. // Create a mock DOM element.
  35. //
  36. var createMockElement = function(attr, parent, scope) {
  37. return {
  38. attr: function() {
  39. return attr;
  40. },
  41. parent: function () {
  42. return parent;
  43. },
  44. scope: function () {
  45. return scope || {};
  46. },
  47. };
  48. }
  49. //
  50. // Create a mock node data model.
  51. //
  52. var createMockNode = function (inputConnectors, outputConnectors) {
  53. return {
  54. x: function () { return 0 },
  55. y: function () { return 0 },
  56. inputConnectors: inputConnectors || [],
  57. outputConnectors: outputConnectors || [],
  58. select: jasmine.createSpy(),
  59. selected: function () { return false; },
  60. };
  61. };
  62. //
  63. // Create a mock chart.
  64. //
  65. var createMockChart = function (mockNodes, mockConnections) {
  66. return {
  67. nodes: mockNodes,
  68. connections: mockConnections,
  69. handleNodeClicked: jasmine.createSpy(),
  70. handleConnectionMouseDown: jasmine.createSpy(),
  71. updateSelectedNodesLocation: jasmine.createSpy(),
  72. deselectAll: jasmine.createSpy(),
  73. createNewConnection: jasmine.createSpy(),
  74. applySelectionRect: jasmine.createSpy(),
  75. };
  76. };
  77. //
  78. // Create a mock dragging service.
  79. //
  80. var createMockDragging = function () {
  81. var mockDragging = {
  82. startDrag: function (evt, config) {
  83. mockDragging.evt = evt;
  84. mockDragging.config = config;
  85. },
  86. };
  87. return mockDragging;
  88. };
  89. //
  90. // Create a mock version of the SVG element.
  91. //
  92. var createMockSvgElement = function () {
  93. return {
  94. getScreenCTM: function () {
  95. return {
  96. inverse: function () {
  97. return this;
  98. },
  99. };
  100. },
  101. createSVGPoint: function () {
  102. return {
  103. x: 0,
  104. y: 0 ,
  105. matrixTransform: function () {
  106. return this;
  107. },
  108. };
  109. }
  110. };
  111. };
  112. it('searchUp returns null when at root 1', function () {
  113. expect(testObject.searchUp(null, "some-class")).toBe(null);
  114. });
  115. it('searchUp returns null when at root 2', function () {
  116. expect(testObject.searchUp([], "some-class")).toBe(null);
  117. });
  118. it('searchUp returns element when it has requested class', function () {
  119. var whichClass = "some-class";
  120. var mockElement = createMockElement(whichClass);
  121. expect(testObject.searchUp(mockElement, whichClass)).toBe(mockElement);
  122. });
  123. it('searchUp returns parent when it has requested class', function () {
  124. var whichClass = "some-class";
  125. var mockParent = createMockElement(whichClass);
  126. var mockElement = createMockElement('', mockParent);
  127. expect(testObject.searchUp(mockElement, whichClass)).toBe(mockParent);
  128. });
  129. it('hitTest returns result of elementFromPoint', function () {
  130. var mockElement = {};
  131. // Mock out the document.
  132. testObject.document = {
  133. elementFromPoint: function () {
  134. return mockElement;
  135. },
  136. };
  137. expect(testObject.hitTest(12, 30)).toBe(mockElement);
  138. });
  139. it('checkForHit returns null when the hit element has no parent with requested class', function () {
  140. var mockElement = createMockElement(null, null);
  141. testObject.jQuery = function (input) {
  142. return input;
  143. };
  144. expect(testObject.checkForHit(mockElement, "some-class")).toBe(null);
  145. });
  146. it('checkForHit returns the result of searchUp when found', function () {
  147. var mockConnectorScope = {};
  148. var whichClass = "some-class";
  149. var mockElement = createMockElement(whichClass, null, mockConnectorScope);
  150. testObject.jQuery = function (input) {
  151. return input;
  152. };
  153. expect(testObject.checkForHit(mockElement, whichClass)).toBe(mockConnectorScope);
  154. });
  155. it('checkForHit returns null when searchUp fails', function () {
  156. var mockElement = createMockElement(null, null, null);
  157. testObject.jQuery = function (input) {
  158. return input;
  159. };
  160. expect(testObject.checkForHit(mockElement, "some-class")).toBe(null);
  161. });
  162. it('test node dragging is started on node mouse down', function () {
  163. mockDragging.startDrag = jasmine.createSpy();
  164. var mockEvt = {};
  165. var mockNode = createMockNode();
  166. mockScope.nodeMouseDown(mockEvt, mockNode);
  167. expect(mockDragging.startDrag).toHaveBeenCalled();
  168. });
  169. it('test node click handling is forwarded to view model', function () {
  170. mockScope.chart = createMockChart([mockNode]);
  171. var mockEvt = {
  172. ctrlKey: false,
  173. };
  174. var mockNode = createMockNode();
  175. mockScope.nodeMouseDown(mockEvt, mockNode);
  176. mockDragging.config.clicked();
  177. expect(mockScope.chart.handleNodeClicked).toHaveBeenCalledWith(mockNode, false);
  178. });
  179. it('test control + node click handling is forwarded to view model', function () {
  180. var mockNode = createMockNode();
  181. mockScope.chart = createMockChart([mockNode]);
  182. var mockEvt = {
  183. ctrlKey: true,
  184. };
  185. mockScope.nodeMouseDown(mockEvt, mockNode);
  186. mockDragging.config.clicked();
  187. expect(mockScope.chart.handleNodeClicked).toHaveBeenCalledWith(mockNode, true);
  188. });
  189. it('test node dragging updates selected nodes location', function () {
  190. var mockEvt = {};
  191. mockScope.chart = createMockChart([createMockNode()]);
  192. mockScope.nodeMouseDown(mockEvt, mockScope.chart.nodes[0]);
  193. var xIncrement = 5;
  194. var yIncrement = 15;
  195. mockDragging.config.dragStarted(0, 0);
  196. mockDragging.config.dragging(xIncrement, yIncrement);
  197. expect(mockScope.chart.updateSelectedNodesLocation).toHaveBeenCalledWith(xIncrement, yIncrement);
  198. });
  199. it('test node dragging doesnt modify selection when node is already selected', function () {
  200. var mockNode1 = createMockNode();
  201. var mockNode2 = createMockNode();
  202. mockScope.chart = createMockChart([mockNode1, mockNode2]);
  203. mockNode2.selected = function () { return true; }
  204. var mockEvt = {};
  205. mockScope.nodeMouseDown(mockEvt, mockNode2);
  206. mockDragging.config.dragStarted(0, 0);
  207. expect(mockScope.chart.deselectAll).not.toHaveBeenCalled();
  208. });
  209. it('test node dragging selects node, when the node is not already selected', function () {
  210. var mockNode1 = createMockNode();
  211. var mockNode2 = createMockNode();
  212. mockScope.chart = createMockChart([mockNode1, mockNode2]);
  213. var mockEvt = {};
  214. mockScope.nodeMouseDown(mockEvt, mockNode2);
  215. mockDragging.config.dragStarted(0, 0);
  216. expect(mockScope.chart.deselectAll).toHaveBeenCalled();
  217. expect(mockNode2.select).toHaveBeenCalled();
  218. });
  219. it('test connection click handling is forwarded to view model', function () {
  220. var mockNode = createMockNode();
  221. var mockEvt = {
  222. stopPropagation: jasmine.createSpy(),
  223. preventDefault: jasmine.createSpy(),
  224. ctrlKey: false,
  225. };
  226. var mockConnection = {};
  227. mockScope.chart = createMockChart([mockNode]);
  228. mockScope.connectionMouseDown(mockEvt, mockConnection);
  229. expect(mockScope.chart.handleConnectionMouseDown).toHaveBeenCalledWith(mockConnection, false);
  230. expect(mockEvt.stopPropagation).toHaveBeenCalled();
  231. expect(mockEvt.preventDefault).toHaveBeenCalled();
  232. });
  233. it('test control + connection click handling is forwarded to view model', function () {
  234. var mockNode = createMockNode();
  235. var mockEvt = {
  236. stopPropagation: jasmine.createSpy(),
  237. preventDefault: jasmine.createSpy(),
  238. ctrlKey: true,
  239. };
  240. var mockConnection = {};
  241. mockScope.chart = createMockChart([mockNode]);
  242. mockScope.connectionMouseDown(mockEvt, mockConnection);
  243. expect(mockScope.chart.handleConnectionMouseDown).toHaveBeenCalledWith(mockConnection, true);
  244. });
  245. it('test selection is cleared when background is clicked', function () {
  246. var mockEvt = {};
  247. mockScope.chart = createMockChart([createMockNode()]);
  248. mockScope.chart.nodes[0].selected = true;
  249. mockScope.mouseDown(mockEvt);
  250. expect(mockScope.chart.deselectAll).toHaveBeenCalled();
  251. });
  252. it('test background mouse down commences selection dragging', function () {
  253. var mockNode = createMockNode();
  254. var mockConnector = {};
  255. var mockEvt = {};
  256. mockScope.chart = createMockChart([mockNode]);
  257. mockScope.mouseDown(mockEvt);
  258. mockDragging.config.dragStarted(0, 0);
  259. expect(mockScope.dragSelecting).toBe(true);
  260. });
  261. it('test can end selection dragging', function () {
  262. var mockNode = createMockNode();
  263. var mockConnector = {};
  264. var mockEvt = {};
  265. mockScope.chart = createMockChart([mockNode]);
  266. mockScope.mouseDown(mockEvt);
  267. mockDragging.config.dragStarted(0, 0, mockEvt);
  268. mockDragging.config.dragging(0, 0, mockEvt);
  269. mockDragging.config.dragEnded();
  270. expect(mockScope.dragSelecting).toBe(false);
  271. });
  272. it('test selection dragging ends by selecting nodes', function () {
  273. var mockNode = createMockNode();
  274. var mockConnector = {};
  275. var mockEvt = {};
  276. mockScope.chart = createMockChart([mockNode]);
  277. mockScope.mouseDown(mockEvt);
  278. mockDragging.config.dragStarted(0, 0, mockEvt);
  279. mockDragging.config.dragging(0, 0, mockEvt);
  280. var selectionRect = {
  281. x: 1,
  282. y: 2,
  283. width: 3,
  284. height: 4,
  285. };
  286. mockScope.dragSelectionRect = selectionRect;
  287. mockDragging.config.dragEnded();
  288. expect(mockScope.chart.applySelectionRect).toHaveBeenCalledWith(selectionRect);
  289. });
  290. it('test mouse down commences connection dragging', function () {
  291. var mockNode = createMockNode();
  292. var mockConnector = {};
  293. var mockEvt = {};
  294. mockScope.chart = createMockChart([mockNode]);
  295. mockScope.connectorMouseDown(mockEvt, mockScope.chart.nodes[0], mockScope.chart.nodes[0].inputConnectors[0], 0, false);
  296. mockDragging.config.dragStarted(0, 0);
  297. expect(mockScope.draggingConnection).toBe(true);
  298. });
  299. it('test can end connection dragging', function () {
  300. var mockNode = createMockNode();
  301. var mockConnector = {};
  302. var mockEvt = {};
  303. mockScope.chart = createMockChart([mockNode]);
  304. mockScope.connectorMouseDown(mockEvt, mockScope.chart.nodes[0], mockScope.chart.nodes[0].inputConnectors[0], 0, false);
  305. mockDragging.config.dragStarted(0, 0, mockEvt);
  306. mockDragging.config.dragging(0, 0, mockEvt);
  307. mockDragging.config.dragEnded();
  308. expect(mockScope.draggingConnection).toBe(false);
  309. });
  310. it('test can make a connection by dragging', function () {
  311. var mockNode = createMockNode();
  312. var mockDraggingConnector = {};
  313. var mockDragOverConnector = {};
  314. var mockEvt = {};
  315. mockScope.chart = createMockChart([mockNode]);
  316. mockScope.connectorMouseDown(mockEvt, mockScope.chart.nodes[0], mockDraggingConnector, 0, false);
  317. mockDragging.config.dragStarted(0, 0, mockEvt);
  318. mockDragging.config.dragging(0, 0, mockEvt);
  319. // Fake out the mouse over connector.
  320. mockScope.mouseOverConnector = mockDragOverConnector;
  321. mockDragging.config.dragEnded();
  322. expect(mockScope.chart.createNewConnection).toHaveBeenCalledWith(mockDraggingConnector, mockDragOverConnector);
  323. });
  324. it('test connection creation by dragging is cancelled when dragged over invalid connector', function () {
  325. var mockNode = createMockNode();
  326. var mockDraggingConnector = {};
  327. var mockDragOverConnector = {};
  328. var mockEvt = {};
  329. mockScope.chart = createMockChart([mockNode]);
  330. mockScope.connectorMouseDown(mockEvt, mockScope.chart.nodes[0], mockDraggingConnector, 0, false);
  331. mockDragging.config.dragStarted(0, 0, mockEvt);
  332. mockDragging.config.dragging(0, 0, mockEvt);
  333. // Fake out the invalid connector.
  334. mockScope.mouseOverConnector = null;
  335. mockDragging.config.dragEnded();
  336. expect(mockScope.chart.createNewConnection).not.toHaveBeenCalled();
  337. });
  338. it('mouse move over connection caches the connection', function () {
  339. var mockElement = {};
  340. var mockConnection = {};
  341. var mockConnectionScope = {
  342. connection: mockConnection
  343. };
  344. var mockEvent = {};
  345. //
  346. // Fake out the function that check if a connection has been hit.
  347. //
  348. testObject.checkForHit = function (element, whichClass) {
  349. if (whichClass === testObject.connectionClass) {
  350. return mockConnectionScope;
  351. }
  352. return null;
  353. };
  354. testObject.hitTest = function () {
  355. return mockElement;
  356. };
  357. mockScope.mouseMove(mockEvent);
  358. expect(mockScope.mouseOverConnection).toBe(mockConnection);
  359. });
  360. it('test mouse over connection clears mouse over connector and node', function () {
  361. var mockElement = {};
  362. var mockConnection = {};
  363. var mockConnectionScope = {
  364. connection: mockConnection
  365. };
  366. var mockEvent = {};
  367. //
  368. // Fake out the function that check if a connection has been hit.
  369. //
  370. testObject.checkForHit = function (element, whichClass) {
  371. if (whichClass === testObject.connectionClass) {
  372. return mockConnectionScope;
  373. }
  374. return null;
  375. };
  376. testObject.hitTest = function () {
  377. return mockElement;
  378. };
  379. mockScope.mouseOverConnector = {};
  380. mockScope.mouseOverNode = {};
  381. mockScope.mouseMove(mockEvent);
  382. expect(mockScope.mouseOverConnector).toBe(null);
  383. expect(mockScope.mouseOverNode).toBe(null);
  384. });
  385. it('test mouseMove handles mouse over connector', function () {
  386. var mockElement = {};
  387. var mockConnector = {};
  388. var mockConnectorScope = {
  389. connector: mockConnector
  390. };
  391. var mockEvent = {};
  392. //
  393. // Fake out the function that check if a connector has been hit.
  394. //
  395. testObject.checkForHit = function (element, whichClass) {
  396. if (whichClass === testObject.connectorClass) {
  397. return mockConnectorScope;
  398. }
  399. return null;
  400. };
  401. testObject.hitTest = function () {
  402. return mockElement;
  403. };
  404. mockScope.mouseMove(mockEvent);
  405. expect(mockScope.mouseOverConnector).toBe(mockConnector);
  406. });
  407. it('test mouseMove handles mouse over node', function () {
  408. var mockElement = {};
  409. var mockNode = {};
  410. var mockNodeScope = {
  411. node: mockNode
  412. };
  413. var mockEvent = {};
  414. //
  415. // Fake out the function that check if a connector has been hit.
  416. //
  417. testObject.checkForHit = function (element, whichClass) {
  418. if (whichClass === testObject.nodeClass) {
  419. return mockNodeScope;
  420. }
  421. return null;
  422. };
  423. testObject.hitTest = function () {
  424. return mockElement;
  425. };
  426. mockScope.mouseMove(mockEvent);
  427. expect(mockScope.mouseOverNode).toBe(mockNode);
  428. });
  429. });