/qt/python/mantidqt/mantidqt/widgets/workspacedisplay/table/test/test_tableworkspacedisplay_marked_columns.py

https://github.com/mantidproject/mantid · Python · 360 lines · 242 code · 63 blank · 55 comment · 1 complexity · 2e2c62e406c4d245c8b7b3cb74a35d12 MD5 · raw file

  1. # Mantid Repository : https://github.com/mantidproject/mantid
  2. #
  3. # Copyright © 2018 ISIS Rutherford Appleton Laboratory UKRI,
  4. # NScD Oak Ridge National Laboratory, European Spallation Source,
  5. # Institut Laue - Langevin & CSNS, Institute of High Energy Physics, CAS
  6. # SPDX - License - Identifier: GPL - 3.0 +
  7. # This file is part of the mantid workbench.
  8. #
  9. #
  10. import unittest
  11. from itertools import permutations
  12. from mantidqt.widgets.workspacedisplay.table.error_column import ErrorColumn
  13. from mantidqt.widgets.workspacedisplay.table.marked_columns import MarkedColumns
  14. class ReferenceHolder:
  15. def __init__(self, test_func, test_list):
  16. self.func = test_func
  17. self.list = test_list
  18. class MarkedColumnsTest(unittest.TestCase):
  19. def test_add_x(self):
  20. mc = MarkedColumns()
  21. self.execute_add(mc.add_x, mc.as_x)
  22. def test_add_y(self):
  23. mc = MarkedColumns()
  24. self.execute_add(mc.add_y, mc.as_y)
  25. def execute_add(self, func_to_add, list_to_check):
  26. func_to_add(2)
  27. self.assertEqual(1, len(list_to_check))
  28. func_to_add(3)
  29. self.assertEqual(2, len(list_to_check))
  30. func_to_add(4000000)
  31. self.assertEqual(3, len(list_to_check))
  32. def test_add_y_err(self):
  33. """
  34. Test adding YErr columns that do not overlap in any way
  35. """
  36. mc = MarkedColumns()
  37. ec = ErrorColumn(2, 4)
  38. mc.add_y_err(ec)
  39. self.assertEqual(1, len(mc.as_y_err))
  40. ec = ErrorColumn(3, 5)
  41. mc.add_y_err(ec)
  42. self.assertEqual(2, len(mc.as_y_err))
  43. ec = ErrorColumn(1, 6)
  44. mc.add_y_err(ec)
  45. self.assertEqual(3, len(mc.as_y_err))
  46. def test_add_x_duplicate_column(self):
  47. mc = MarkedColumns()
  48. self.execute_add_duplicate_column(mc.add_x, mc.as_x)
  49. def test_add_y_duplicate_column(self):
  50. mc = MarkedColumns()
  51. self.execute_add_duplicate_column(mc.add_y, mc.as_y)
  52. def execute_add_duplicate_column(self, func_to_add, list_to_check):
  53. func_to_add(2)
  54. self.assertEqual(1, len(list_to_check))
  55. func_to_add(2)
  56. self.assertEqual(1, len(list_to_check))
  57. func_to_add(55)
  58. self.assertEqual(2, len(list_to_check))
  59. func_to_add(55)
  60. self.assertEqual(2, len(list_to_check))
  61. def test_add_y_err_duplicate_column(self):
  62. mc = MarkedColumns()
  63. ec = ErrorColumn(2, 4)
  64. mc.add_y_err(ec)
  65. self.assertEqual(1, len(mc.as_y_err))
  66. mc.add_y_err(ec)
  67. self.assertEqual(1, len(mc.as_y_err))
  68. ec2 = ErrorColumn(3, 5)
  69. mc.add_y_err(ec2)
  70. self.assertEqual(2, len(mc.as_y_err))
  71. mc.add_y_err(ec2)
  72. self.assertEqual(2, len(mc.as_y_err))
  73. def test_add_already_marked(self):
  74. mc = MarkedColumns()
  75. relevant_funcs = [ReferenceHolder(mc.add_x, mc.as_x),
  76. ReferenceHolder(mc.add_y, mc.as_y)]
  77. all_combinations = permutations(relevant_funcs, 2)
  78. for combination in all_combinations:
  79. self.execute_add_already_marked(*combination)
  80. def execute_add_already_marked(self, first, two):
  81. """
  82. If trying to mark a column that is already marked -> all other markings must be removed
  83. :type first: ReferenceHolder
  84. :type two: ReferenceHolder
  85. :return:
  86. """
  87. # add column in first
  88. first.func(33)
  89. self.assertEqual(1, len(first.list))
  90. # add the same column in the second
  91. two.func(33)
  92. # it should have been removed from the first and only present in the second
  93. self.assertEqual(0, len(first.list))
  94. self.assertEqual(1, len(two.list))
  95. def test_add_y_err_duplicate_column_same_source_column(self):
  96. """
  97. Test for adding a new YErr column with the same source column
  98. -> The new YErr must replace the old one
  99. """
  100. mc = MarkedColumns()
  101. ec = ErrorColumn(column=2, related_y_column=4)
  102. mc.add_y_err(ec)
  103. self.assertEqual(1, len(mc.as_y_err))
  104. self.assertEqual(2, mc.as_y_err[0].column)
  105. self.assertEqual(4, mc.as_y_err[0].related_y_column)
  106. # different source column but contains error for the same column
  107. # adding this one should replace the first one
  108. ec2 = ErrorColumn(column=2, related_y_column=5)
  109. mc.add_y_err(ec2)
  110. self.assertEqual(1, len(mc.as_y_err))
  111. self.assertEqual(2, mc.as_y_err[0].column)
  112. self.assertEqual(5, mc.as_y_err[0].related_y_column)
  113. def test_add_y_err_duplicate_column_different_reference_col(self):
  114. """
  115. Test for adding a new YErr column with a _different_ source column but same reference column
  116. -> The new YErr must replace the old one
  117. """
  118. mc = MarkedColumns()
  119. ec = ErrorColumn(column=2, related_y_column=4)
  120. mc.add_y_err(ec)
  121. self.assertEqual(1, len(mc.as_y_err))
  122. self.assertEqual(2, mc.as_y_err[0].column)
  123. self.assertEqual(4, mc.as_y_err[0].related_y_column)
  124. # different source column but contains error for the same column
  125. # adding this one should replace the first one
  126. ec2 = ErrorColumn(column=3, related_y_column=4)
  127. mc.add_y_err(ec2)
  128. self.assertEqual(1, len(mc.as_y_err))
  129. self.assertEqual(3, mc.as_y_err[0].column)
  130. self.assertEqual(4, mc.as_y_err[0].related_y_column)
  131. def test_changing_y_to_x_removes_associated_yerr_columns(self):
  132. """
  133. Test to check if a first column is marked as Y, a second column YErr is associated with it, but then
  134. the first one is changed to X - the YErr mark should be removed
  135. """
  136. mc = MarkedColumns()
  137. mc.add_y(4)
  138. ec = ErrorColumn(column=2, related_y_column=4)
  139. mc.add_y_err(ec)
  140. # check that we have both a Y col and an associated YErr
  141. self.assertEqual(1, len(mc.as_y))
  142. self.assertEqual(1, len(mc.as_y_err))
  143. mc.add_x(4)
  144. # changing the column to X should have removed it from Y and Yerr
  145. self.assertEqual(1, len(mc.as_x))
  146. self.assertEqual(0, len(mc.as_y))
  147. self.assertEqual(0, len(mc.as_y_err))
  148. # check setting the column back to Y does not automatically reinstate the error column
  149. mc.add_y(4)
  150. self.assertEqual(1, len(mc.as_y))
  151. self.assertEqual(0, len(mc.as_y_err))
  152. def test_changing_y_to_none_removes_associated_yerr_columns(self):
  153. """
  154. Test to check if a first column is marked as Y, a second column YErr is associated with it, but then
  155. the first one is changed to X - the YErr mark should be removed
  156. """
  157. mc = MarkedColumns()
  158. mc.add_y(4)
  159. ec = ErrorColumn(column=2, related_y_column=4)
  160. mc.add_y_err(ec)
  161. # check that we have both a Y col and an associated YErr
  162. self.assertEqual(1, len(mc.as_y))
  163. self.assertEqual(1, len(mc.as_y_err))
  164. mc.remove(4)
  165. # changing the column to NONE should have removed it from X, Y and YErr
  166. self.assertEqual(0, len(mc.as_x))
  167. self.assertEqual(0, len(mc.as_y))
  168. self.assertEqual(0, len(mc.as_y_err))
  169. # check adding the Y column back in does not automatically reinstate the error column
  170. mc.add_y(4)
  171. self.assertEqual(1, len(mc.as_y))
  172. self.assertEqual(0, len(mc.as_y_err))
  173. def test_remove_column(self):
  174. mc = MarkedColumns()
  175. mc.add_y(4)
  176. mc.add_x(3)
  177. ec = ErrorColumn(column=2, related_y_column=6)
  178. mc.add_y_err(ec)
  179. self.assertEqual(1, len(mc.as_x))
  180. self.assertEqual(1, len(mc.as_y))
  181. self.assertEqual(1, len(mc.as_y_err))
  182. mc.remove(4)
  183. self.assertEqual(0, len(mc.as_y))
  184. self.assertEqual(1, len(mc.as_y_err))
  185. self.assertEqual(1, len(mc.as_x))
  186. mc.remove(3)
  187. self.assertEqual(0, len(mc.as_x))
  188. self.assertEqual(0, len(mc.as_y))
  189. self.assertEqual(1, len(mc.as_y_err))
  190. mc.remove(2)
  191. self.assertEqual(0, len(mc.as_x))
  192. self.assertEqual(0, len(mc.as_y))
  193. self.assertEqual(0, len(mc.as_y_err))
  194. def test_build_labels_x_y(self):
  195. # TODO test this edge case: mark all columns Y, remove one that is not the last one!
  196. mc = MarkedColumns()
  197. mc.add_y(0)
  198. mc.add_y(1)
  199. mc.add_y(2)
  200. mc.add_y(3)
  201. # note that the max Y label number will decrease as more Y columns are being changed to X
  202. expected = [(0, '[Y0]'), (1, '[Y1]'), (2, '[Y2]'), (3, '[Y3]')]
  203. self.assertEqual(expected, mc.build_labels())
  204. expected = [(1, '[X0]'), (0, '[Y0]'), (2, '[Y1]'), (3, '[Y2]')]
  205. mc.add_x(1)
  206. self.assertEqual(expected, mc.build_labels())
  207. expected = [(1, '[X0]'), (3, '[X1]'), (0, '[Y0]'), (2, '[Y1]')]
  208. mc.add_x(3)
  209. self.assertEqual(expected, mc.build_labels())
  210. def test_build_labels_y_and_yerr_change_middle(self):
  211. mc = MarkedColumns()
  212. mc.add_y(0)
  213. mc.add_y(1)
  214. mc.add_y(2)
  215. # change one of the columns to YErr
  216. mc.add_y_err(ErrorColumn(1, 0))
  217. expected = [(0, '[Y0]'), (2, '[Y1]'), (1, '[Y0_YErr]')]
  218. self.assertEqual(expected, mc.build_labels())
  219. # change the last Y column to YErr
  220. mc.add_y_err(ErrorColumn(2, 0))
  221. expected = [(0, '[Y0]'), (2, '[Y0_YErr]')]
  222. self.assertEqual(expected, mc.build_labels())
  223. def test_build_labels_y_and_yerr_change_first(self):
  224. mc = MarkedColumns()
  225. mc.add_y(0)
  226. mc.add_y(1)
  227. mc.add_y(2)
  228. # change one of the columns to YErr
  229. mc.add_y_err(ErrorColumn(0, 1))
  230. # note: the first column is being set -> this decreases the label index of all columns to its right by 1
  231. expected = [(1, '[Y0]'), (2, '[Y1]'), (0, '[Y0_YErr]')]
  232. self.assertEqual(expected, mc.build_labels())
  233. # change the last Y column to YErr
  234. mc.add_y_err(ErrorColumn(2, 1))
  235. expected = [(1, '[Y0]'), (2, '[Y0_YErr]')]
  236. self.assertEqual(expected, mc.build_labels())
  237. def test_build_labels_x_y_and_yerr(self):
  238. mc = MarkedColumns()
  239. mc.add_y(0)
  240. mc.add_y(1)
  241. mc.add_y(2)
  242. mc.add_y(3)
  243. mc.add_y_err(ErrorColumn(1, 0))
  244. expected = [(0, '[Y0]'), (2, '[Y1]'), (3, '[Y2]'), (1, '[Y0_YErr]')]
  245. self.assertEqual(expected, mc.build_labels())
  246. expected = [(1, '[X0]'), (0, '[Y0]'), (2, '[Y1]'), (3, '[Y2]')]
  247. mc.add_x(1)
  248. self.assertEqual(expected, mc.build_labels())
  249. expected = [(1, '[X0]'), (2, '[Y0]'), (3, '[Y1]'), (0, '[Y1_YErr]')]
  250. mc.add_y_err(ErrorColumn(0, 3))
  251. self.assertEqual(expected, mc.build_labels())
  252. def test_build_labels_y_with_only_some_having_yerr(self):
  253. mc = MarkedColumns()
  254. mc.add_y(0)
  255. mc.add_y(1)
  256. mc.add_y_err(ErrorColumn(2,1))
  257. expected = [(0, '[Y0]'), (1, '[Y1]'), (2, '[Y1_YErr]')]
  258. self.assertEqual(expected, mc.build_labels())
  259. def test_fail_to_add_yerr_for_x(self):
  260. mc = MarkedColumns()
  261. mc.add_y(0)
  262. mc.add_y(1)
  263. mc.add_y(2)
  264. mc.add_y(3)
  265. mc.add_y_err(ErrorColumn(1, 0))
  266. expected = [(0, '[Y0]'), (2, '[Y1]'), (3, '[Y2]'), (1, '[Y0_YErr]')]
  267. self.assertEqual(expected, mc.build_labels())
  268. expected = [(1, '[X0]'), (0, '[Y0]'), (2, '[Y1]'), (3, '[Y2]')]
  269. mc.add_x(1)
  270. self.assertEqual(expected, mc.build_labels())
  271. self.assertRaises(ValueError, lambda: mc.add_y_err(ErrorColumn(0, 1)))
  272. def test_fail_to_add_yerr_for_another_yerr(self):
  273. mc = MarkedColumns()
  274. mc.add_y(0)
  275. mc.add_y(1)
  276. mc.add_y(2)
  277. mc.add_y(3)
  278. mc.add_y_err(ErrorColumn(1, 0))
  279. expected = [(0, '[Y0]'), (2, '[Y1]'), (3, '[Y2]'), (1, '[Y0_YErr]')]
  280. self.assertEqual(expected, mc.build_labels())
  281. self.assertRaises(ValueError, lambda: mc.add_y_err(ErrorColumn(0, 1)))
  282. def test_find_yerr(self):
  283. mc = MarkedColumns()
  284. mc.add_y(0)
  285. mc.add_y(1)
  286. mc.add_y(2)
  287. mc.add_y(3)
  288. mc.add_y_err(ErrorColumn(4, 1))
  289. expected = {1: 4}
  290. self.assertEqual(expected, mc.find_yerr([1]))
  291. # Replace the Y column, which has an associated YErr. This should remove the YErr as well
  292. mc.add_y_err(ErrorColumn(1, 3))
  293. expected = {3: 1}
  294. self.assertEqual(expected, mc.find_yerr([0, 1, 2, 3]))
  295. mc.add_y_err(ErrorColumn(4, 2))
  296. expected = {2: 4, 3: 1}
  297. self.assertEqual(expected, mc.find_yerr([0, 1, 2, 3]))