Skip to content

Commit

Permalink
bpo-42759: Fix equality comparison of Variable and Font in Tkinter (p…
Browse files Browse the repository at this point in the history
…ythonGH-23968)

Objects which belong to different Tcl interpreters are now always
different, even if they have the same name.
  • Loading branch information
serhiy-storchaka authored and adorilson committed Mar 11, 2021
1 parent 5064d6c commit 30b5aa5
Show file tree
Hide file tree
Showing 5 changed files with 27 additions and 11 deletions.
10 changes: 3 additions & 7 deletions Lib/tkinter/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -516,15 +516,11 @@ def trace_vinfo(self):
self._tk.call("trace", "vinfo", self._name))]

def __eq__(self, other):
"""Comparison for equality (==).
Note: if the Variable's master matters to behavior
also compare self._master == other._master
"""
if not isinstance(other, Variable):
return NotImplemented
return self.__class__.__name__ == other.__class__.__name__ \
and self._name == other._name
return (self._name == other._name
and self.__class__.__name__ == other.__class__.__name__
and self._tk == other._tk)


class StringVar(Variable):
Expand Down
2 changes: 1 addition & 1 deletion Lib/tkinter/font.py
Original file line number Diff line number Diff line change
Expand Up @@ -107,7 +107,7 @@ def __repr__(self):
def __eq__(self, other):
if not isinstance(other, Font):
return NotImplemented
return self.name == other.name
return self.name == other.name and self._tk == other._tk

def __getitem__(self, key):
return self.cget(key)
Expand Down
9 changes: 8 additions & 1 deletion Lib/tkinter/test/test_tkinter/test_font.py
Original file line number Diff line number Diff line change
Expand Up @@ -63,15 +63,22 @@ def test_name(self):
self.assertEqual(self.font.name, fontname)
self.assertEqual(str(self.font), fontname)

def test_eq(self):
def test_equality(self):
font1 = font.Font(root=self.root, name=fontname, exists=True)
font2 = font.Font(root=self.root, name=fontname, exists=True)
self.assertIsNot(font1, font2)
self.assertEqual(font1, font2)
self.assertNotEqual(font1, font1.copy())

self.assertNotEqual(font1, 0)
self.assertEqual(font1, ALWAYS_EQ)

root2 = tkinter.Tk()
self.addCleanup(root2.destroy)
font3 = font.Font(root=root2, name=fontname, exists=True)
self.assertEqual(str(font1), str(font3))
self.assertNotEqual(font1, font3)

def test_measure(self):
self.assertIsInstance(self.font.measure('abc'), int)

Expand Down
14 changes: 12 additions & 2 deletions Lib/tkinter/test/test_tkinter/test_variables.py
Original file line number Diff line number Diff line change
Expand Up @@ -58,22 +58,32 @@ def test_dont_unset_not_existing(self):
del v2
self.assertFalse(self.info_exists("name"))

def test___eq__(self):
def test_equality(self):
# values doesn't matter, only class and name are checked
v1 = Variable(self.root, name="abc")
v2 = Variable(self.root, name="abc")
self.assertIsNot(v1, v2)
self.assertEqual(v1, v2)

v3 = StringVar(self.root, name="abc")
v3 = Variable(self.root, name="cba")
self.assertNotEqual(v1, v3)

v4 = StringVar(self.root, name="abc")
self.assertEqual(str(v1), str(v4))
self.assertNotEqual(v1, v4)

V = type('Variable', (), {})
self.assertNotEqual(v1, V())

self.assertNotEqual(v1, object())
self.assertEqual(v1, ALWAYS_EQ)

root2 = tkinter.Tk()
self.addCleanup(root2.destroy)
v5 = Variable(root2, name="abc")
self.assertEqual(str(v1), str(v5))
self.assertNotEqual(v1, v5)

def test_invalid_name(self):
with self.assertRaises(TypeError):
Variable(self.root, name=123)
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
Fixed equality comparison of :class:`tkinter.Variable` and
:class:`tkinter.font.Font`. Objects which belong to different Tcl
interpreters are now always different, even if they have the same name.

0 comments on commit 30b5aa5

Please sign in to comment.