|
3 | 3 | Half the class creates dialog, half works with user customizations. |
4 | 4 | Coverage: 46% just by creating dialog, 60% with current tests. |
5 | 5 | """ |
6 | | -from idlelib.configdialog import ConfigDialog, idleConf, changes |
| 6 | +from idlelib.configdialog import ConfigDialog, idleConf, changes, VarTrace |
7 | 7 | from test.support import requires |
8 | 8 | requires('gui') |
9 | | -from tkinter import Tk |
| 9 | +from tkinter import Tk, IntVar, BooleanVar |
10 | 10 | import unittest |
| 11 | +from unittest import mock |
11 | 12 | import idlelib.config as config |
12 | 13 | from idlelib.idle_test.mock_idle import Func |
13 | 14 |
|
@@ -248,5 +249,94 @@ def test_editor_size(self): |
248 | 249 | #def test_help_sources(self): pass # TODO |
249 | 250 |
|
250 | 251 |
|
| 252 | +class TestVarTrace(unittest.TestCase): |
| 253 | + |
| 254 | + def setUp(self): |
| 255 | + changes.clear() |
| 256 | + self.v1 = IntVar(root) |
| 257 | + self.v2 = BooleanVar(root) |
| 258 | + self.called = 0 |
| 259 | + self.tracers = VarTrace() |
| 260 | + |
| 261 | + def tearDown(self): |
| 262 | + del self.v1, self.v2 |
| 263 | + |
| 264 | + def var_changed_increment(self, *params): |
| 265 | + self.called += 13 |
| 266 | + |
| 267 | + def var_changed_boolean(self, *params): |
| 268 | + pass |
| 269 | + |
| 270 | + def test_init(self): |
| 271 | + self.assertEqual(self.tracers.untraced, []) |
| 272 | + self.assertEqual(self.tracers.traced, []) |
| 273 | + |
| 274 | + def test_add(self): |
| 275 | + tr = self.tracers |
| 276 | + func = Func() |
| 277 | + cb = tr.make_callback = mock.Mock(return_value=func) |
| 278 | + |
| 279 | + v1 = tr.add(self.v1, self.var_changed_increment) |
| 280 | + self.assertIsInstance(v1, IntVar) |
| 281 | + v2 = tr.add(self.v2, self.var_changed_boolean) |
| 282 | + self.assertIsInstance(v2, BooleanVar) |
| 283 | + |
| 284 | + v3 = IntVar(root) |
| 285 | + v3 = tr.add(v3, ('main', 'section', 'option')) |
| 286 | + cb.assert_called_once() |
| 287 | + cb.assert_called_with(v3, ('main', 'section', 'option')) |
| 288 | + |
| 289 | + expected = [(v1, self.var_changed_increment), |
| 290 | + (v2, self.var_changed_boolean), |
| 291 | + (v3, func)] |
| 292 | + self.assertEqual(tr.traced, []) |
| 293 | + self.assertEqual(tr.untraced, expected) |
| 294 | + |
| 295 | + del tr.make_callback |
| 296 | + |
| 297 | + def test_make_callback(self): |
| 298 | + tr = self.tracers |
| 299 | + cb = tr.make_callback(self.v1, ('main', 'section', 'option')) |
| 300 | + self.assertTrue(callable(cb)) |
| 301 | + self.v1.set(42) |
| 302 | + # Not attached, so set didn't invoke the callback. |
| 303 | + self.assertNotIn('section', changes['main']) |
| 304 | + # Invoke callback manually. |
| 305 | + cb() |
| 306 | + self.assertIn('section', changes['main']) |
| 307 | + self.assertEqual(changes['main']['section']['option'], '42') |
| 308 | + |
| 309 | + def test_attach_detach(self): |
| 310 | + tr = self.tracers |
| 311 | + v1 = tr.add(self.v1, self.var_changed_increment) |
| 312 | + v2 = tr.add(self.v2, self.var_changed_boolean) |
| 313 | + expected = [(v1, self.var_changed_increment), |
| 314 | + (v2, self.var_changed_boolean)] |
| 315 | + |
| 316 | + # Attach callbacks and test call increment. |
| 317 | + tr.attach() |
| 318 | + self.assertEqual(tr.untraced, []) |
| 319 | + self.assertCountEqual(tr.traced, expected) |
| 320 | + v1.set(1) |
| 321 | + self.assertEqual(v1.get(), 1) |
| 322 | + self.assertEqual(self.called, 13) |
| 323 | + |
| 324 | + # Check that only one callback is attached to a variable. |
| 325 | + # If more than one callback were attached, then var_changed_increment |
| 326 | + # would be called twice and the counter would be 2. |
| 327 | + self.called = 0 |
| 328 | + tr.attach() |
| 329 | + v1.set(1) |
| 330 | + self.assertEqual(self.called, 13) |
| 331 | + |
| 332 | + # Detach callbacks. |
| 333 | + self.called = 0 |
| 334 | + tr.detach() |
| 335 | + self.assertEqual(tr.traced, []) |
| 336 | + self.assertCountEqual(tr.untraced, expected) |
| 337 | + v1.set(1) |
| 338 | + self.assertEqual(self.called, 0) |
| 339 | + |
| 340 | + |
251 | 341 | if __name__ == '__main__': |
252 | 342 | unittest.main(verbosity=2) |
0 commit comments