test_config_key.py 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291
  1. """Test config_key, coverage 98%.
  2. Coverage is effectively 100%. Tkinter dialog is mocked, Mac-only line
  3. may be skipped, and dummy function in bind test should not be called.
  4. Not tested: exit with 'self.advanced or self.keys_ok(keys)) ...' False.
  5. """
  6. from idlelib import config_key
  7. from test.support import requires
  8. import unittest
  9. from unittest import mock
  10. from tkinter import Tk, TclError
  11. from idlelib.idle_test.mock_idle import Func
  12. from idlelib.idle_test.mock_tk import Mbox_func
  13. gkd = config_key.GetKeysDialog
  14. class ValidationTest(unittest.TestCase):
  15. "Test validation methods: ok, keys_ok, bind_ok."
  16. class Validator(gkd):
  17. def __init__(self, *args, **kwargs):
  18. config_key.GetKeysDialog.__init__(self, *args, **kwargs)
  19. class list_keys_final:
  20. get = Func()
  21. self.list_keys_final = list_keys_final
  22. get_modifiers = Func()
  23. showerror = Mbox_func()
  24. @classmethod
  25. def setUpClass(cls):
  26. requires('gui')
  27. cls.root = Tk()
  28. cls.root.withdraw()
  29. keylist = [['<Key-F12>'], ['<Control-Key-x>', '<Control-Key-X>']]
  30. cls.dialog = cls.Validator(
  31. cls.root, 'Title', '<<Test>>', keylist, _utest=True)
  32. @classmethod
  33. def tearDownClass(cls):
  34. cls.dialog.cancel()
  35. cls.root.update_idletasks()
  36. cls.root.destroy()
  37. del cls.dialog, cls.root
  38. def setUp(self):
  39. self.dialog.showerror.message = ''
  40. # A test that needs a particular final key value should set it.
  41. # A test that sets a non-blank modifier list should reset it to [].
  42. def test_ok_empty(self):
  43. self.dialog.key_string.set(' ')
  44. self.dialog.ok()
  45. self.assertEqual(self.dialog.result, '')
  46. self.assertEqual(self.dialog.showerror.message, 'No key specified.')
  47. def test_ok_good(self):
  48. self.dialog.key_string.set('<Key-F11>')
  49. self.dialog.list_keys_final.get.result = 'F11'
  50. self.dialog.ok()
  51. self.assertEqual(self.dialog.result, '<Key-F11>')
  52. self.assertEqual(self.dialog.showerror.message, '')
  53. def test_keys_no_ending(self):
  54. self.assertFalse(self.dialog.keys_ok('<Control-Shift'))
  55. self.assertIn('Missing the final', self.dialog.showerror.message)
  56. def test_keys_no_modifier_bad(self):
  57. self.dialog.list_keys_final.get.result = 'A'
  58. self.assertFalse(self.dialog.keys_ok('<Key-A>'))
  59. self.assertIn('No modifier', self.dialog.showerror.message)
  60. def test_keys_no_modifier_ok(self):
  61. self.dialog.list_keys_final.get.result = 'F11'
  62. self.assertTrue(self.dialog.keys_ok('<Key-F11>'))
  63. self.assertEqual(self.dialog.showerror.message, '')
  64. def test_keys_shift_bad(self):
  65. self.dialog.list_keys_final.get.result = 'a'
  66. self.dialog.get_modifiers.result = ['Shift']
  67. self.assertFalse(self.dialog.keys_ok('<a>'))
  68. self.assertIn('shift modifier', self.dialog.showerror.message)
  69. self.dialog.get_modifiers.result = []
  70. def test_keys_dup(self):
  71. for mods, final, seq in (([], 'F12', '<Key-F12>'),
  72. (['Control'], 'x', '<Control-Key-x>'),
  73. (['Control'], 'X', '<Control-Key-X>')):
  74. with self.subTest(m=mods, f=final, s=seq):
  75. self.dialog.list_keys_final.get.result = final
  76. self.dialog.get_modifiers.result = mods
  77. self.assertFalse(self.dialog.keys_ok(seq))
  78. self.assertIn('already in use', self.dialog.showerror.message)
  79. self.dialog.get_modifiers.result = []
  80. def test_bind_ok(self):
  81. self.assertTrue(self.dialog.bind_ok('<Control-Shift-Key-a>'))
  82. self.assertEqual(self.dialog.showerror.message, '')
  83. def test_bind_not_ok(self):
  84. self.assertFalse(self.dialog.bind_ok('<Control-Shift>'))
  85. self.assertIn('not accepted', self.dialog.showerror.message)
  86. class ToggleLevelTest(unittest.TestCase):
  87. "Test toggle between Basic and Advanced frames."
  88. @classmethod
  89. def setUpClass(cls):
  90. requires('gui')
  91. cls.root = Tk()
  92. cls.root.withdraw()
  93. cls.dialog = gkd(cls.root, 'Title', '<<Test>>', [], _utest=True)
  94. @classmethod
  95. def tearDownClass(cls):
  96. cls.dialog.cancel()
  97. cls.root.update_idletasks()
  98. cls.root.destroy()
  99. del cls.dialog, cls.root
  100. def test_toggle_level(self):
  101. dialog = self.dialog
  102. def stackorder():
  103. """Get the stack order of the children of the frame.
  104. winfo_children() stores the children in stack order, so
  105. this can be used to check whether a frame is above or
  106. below another one.
  107. """
  108. for index, child in enumerate(dialog.frame.winfo_children()):
  109. if child._name == 'keyseq_basic':
  110. basic = index
  111. if child._name == 'keyseq_advanced':
  112. advanced = index
  113. return basic, advanced
  114. # New window starts at basic level.
  115. self.assertFalse(dialog.advanced)
  116. self.assertIn('Advanced', dialog.button_level['text'])
  117. basic, advanced = stackorder()
  118. self.assertGreater(basic, advanced)
  119. # Toggle to advanced.
  120. dialog.toggle_level()
  121. self.assertTrue(dialog.advanced)
  122. self.assertIn('Basic', dialog.button_level['text'])
  123. basic, advanced = stackorder()
  124. self.assertGreater(advanced, basic)
  125. # Toggle to basic.
  126. dialog.button_level.invoke()
  127. self.assertFalse(dialog.advanced)
  128. self.assertIn('Advanced', dialog.button_level['text'])
  129. basic, advanced = stackorder()
  130. self.assertGreater(basic, advanced)
  131. class KeySelectionTest(unittest.TestCase):
  132. "Test selecting key on Basic frames."
  133. class Basic(gkd):
  134. def __init__(self, *args, **kwargs):
  135. super().__init__(*args, **kwargs)
  136. class list_keys_final:
  137. get = Func()
  138. select_clear = Func()
  139. yview = Func()
  140. self.list_keys_final = list_keys_final
  141. def set_modifiers_for_platform(self):
  142. self.modifiers = ['foo', 'bar', 'BAZ']
  143. self.modifier_label = {'BAZ': 'ZZZ'}
  144. showerror = Mbox_func()
  145. @classmethod
  146. def setUpClass(cls):
  147. requires('gui')
  148. cls.root = Tk()
  149. cls.root.withdraw()
  150. cls.dialog = cls.Basic(cls.root, 'Title', '<<Test>>', [], _utest=True)
  151. @classmethod
  152. def tearDownClass(cls):
  153. cls.dialog.cancel()
  154. cls.root.update_idletasks()
  155. cls.root.destroy()
  156. del cls.dialog, cls.root
  157. def setUp(self):
  158. self.dialog.clear_key_seq()
  159. def test_get_modifiers(self):
  160. dialog = self.dialog
  161. gm = dialog.get_modifiers
  162. eq = self.assertEqual
  163. # Modifiers are set on/off by invoking the checkbutton.
  164. dialog.modifier_checkbuttons['foo'].invoke()
  165. eq(gm(), ['foo'])
  166. dialog.modifier_checkbuttons['BAZ'].invoke()
  167. eq(gm(), ['foo', 'BAZ'])
  168. dialog.modifier_checkbuttons['foo'].invoke()
  169. eq(gm(), ['BAZ'])
  170. @mock.patch.object(gkd, 'get_modifiers')
  171. def test_build_key_string(self, mock_modifiers):
  172. dialog = self.dialog
  173. key = dialog.list_keys_final
  174. string = dialog.key_string.get
  175. eq = self.assertEqual
  176. key.get.result = 'a'
  177. mock_modifiers.return_value = []
  178. dialog.build_key_string()
  179. eq(string(), '<Key-a>')
  180. mock_modifiers.return_value = ['mymod']
  181. dialog.build_key_string()
  182. eq(string(), '<mymod-Key-a>')
  183. key.get.result = ''
  184. mock_modifiers.return_value = ['mymod', 'test']
  185. dialog.build_key_string()
  186. eq(string(), '<mymod-test>')
  187. @mock.patch.object(gkd, 'get_modifiers')
  188. def test_final_key_selected(self, mock_modifiers):
  189. dialog = self.dialog
  190. key = dialog.list_keys_final
  191. string = dialog.key_string.get
  192. eq = self.assertEqual
  193. mock_modifiers.return_value = ['Shift']
  194. key.get.result = '{'
  195. dialog.final_key_selected()
  196. eq(string(), '<Shift-Key-braceleft>')
  197. class CancelTest(unittest.TestCase):
  198. "Simulate user clicking [Cancel] button."
  199. @classmethod
  200. def setUpClass(cls):
  201. requires('gui')
  202. cls.root = Tk()
  203. cls.root.withdraw()
  204. cls.dialog = gkd(cls.root, 'Title', '<<Test>>', [], _utest=True)
  205. @classmethod
  206. def tearDownClass(cls):
  207. cls.dialog.cancel()
  208. cls.root.update_idletasks()
  209. cls.root.destroy()
  210. del cls.dialog, cls.root
  211. def test_cancel(self):
  212. self.assertEqual(self.dialog.winfo_class(), 'Toplevel')
  213. self.dialog.button_cancel.invoke()
  214. with self.assertRaises(TclError):
  215. self.dialog.winfo_class()
  216. self.assertEqual(self.dialog.result, '')
  217. class HelperTest(unittest.TestCase):
  218. "Test module level helper functions."
  219. def test_translate_key(self):
  220. tr = config_key.translate_key
  221. eq = self.assertEqual
  222. # Letters return unchanged with no 'Shift'.
  223. eq(tr('q', []), 'Key-q')
  224. eq(tr('q', ['Control', 'Alt']), 'Key-q')
  225. # 'Shift' uppercases single lowercase letters.
  226. eq(tr('q', ['Shift']), 'Key-Q')
  227. eq(tr('q', ['Control', 'Shift']), 'Key-Q')
  228. eq(tr('q', ['Control', 'Alt', 'Shift']), 'Key-Q')
  229. # Convert key name to keysym.
  230. eq(tr('Page Up', []), 'Key-Prior')
  231. # 'Shift' doesn't change case when it's not a single char.
  232. eq(tr('*', ['Shift']), 'Key-asterisk')
  233. if __name__ == '__main__':
  234. unittest.main(verbosity=2)