[3.8] bpo-42759: Fix equality comparison of Variable and Font in Tkinter (GH-23968) (GH-24026)

Objects which belong to different Tcl interpreters are now always
different, even if they have the same name.
(cherry picked from commit 1df56bc059)
This commit is contained in:
Serhiy Storchaka 2021-01-01 19:39:36 +02:00 committed by GitHub
parent 02639c3bc6
commit e3a9adba32
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 38 additions and 12 deletions

View File

@ -491,13 +491,11 @@ class Variable:
self._tk.call("trace", "vinfo", self._name))] self._tk.call("trace", "vinfo", self._name))]
def __eq__(self, other): def __eq__(self, other):
"""Comparison for equality (==). if not isinstance(other, Variable):
return NotImplemented
Note: if the Variable's master matters to behavior return (self._name == other._name
also compare self._master == other._master and self.__class__.__name__ == other.__class__.__name__
""" and self._tk == other._tk)
return self.__class__.__name__ == other.__class__.__name__ \
and self._name == other._name
class StringVar(Variable): class StringVar(Variable):

View File

@ -100,7 +100,9 @@ class Font:
return self.name return self.name
def __eq__(self, other): def __eq__(self, other):
return isinstance(other, Font) and self.name == other.name if not isinstance(other, Font):
return NotImplemented
return self.name == other.name and self._tk == other._tk
def __getitem__(self, key): def __getitem__(self, key):
return self.cget(key) return self.cget(key)

View File

@ -63,14 +63,21 @@ class FontTest(AbstractTkTest, unittest.TestCase):
self.assertEqual(self.font.name, fontname) self.assertEqual(self.font.name, fontname)
self.assertEqual(str(self.font), 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) font1 = font.Font(root=self.root, name=fontname, exists=True)
font2 = font.Font(root=self.root, name=fontname, exists=True) font2 = font.Font(root=self.root, name=fontname, exists=True)
self.assertIsNot(font1, font2) self.assertIsNot(font1, font2)
self.assertEqual(font1, font2) self.assertEqual(font1, font2)
self.assertNotEqual(font1, font1.copy()) self.assertNotEqual(font1, font1.copy())
self.assertNotEqual(font1, 0) self.assertNotEqual(font1, 0)
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): def test_measure(self):
self.assertIsInstance(self.font.measure('abc'), int) self.assertIsInstance(self.font.measure('abc'), int)

View File

@ -1,6 +1,7 @@
import unittest import unittest
import gc import gc
import tkinter import tkinter
from test.support import ALWAYS_EQ
from tkinter import (Variable, StringVar, IntVar, DoubleVar, BooleanVar, Tcl, from tkinter import (Variable, StringVar, IntVar, DoubleVar, BooleanVar, Tcl,
TclError) TclError)
from tkinter.test.support import AbstractDefaultRootTest from tkinter.test.support import AbstractDefaultRootTest
@ -57,15 +58,30 @@ class TestVariable(TestBase):
del v2 del v2
self.assertFalse(self.info_exists("name")) self.assertFalse(self.info_exists("name"))
def test___eq__(self): def test_equality(self):
# values doesn't matter, only class and name are checked # values doesn't matter, only class and name are checked
v1 = Variable(self.root, name="abc") v1 = Variable(self.root, name="abc")
v2 = Variable(self.root, name="abc") v2 = Variable(self.root, name="abc")
self.assertEqual(v1, v2) self.assertEqual(v1, v2)
v3 = Variable(self.root, name="abc") v3 = Variable(self.root, name="cba")
self.assertNotEqual(v1, v3)
v4 = StringVar(self.root, name="abc") v4 = StringVar(self.root, name="abc")
self.assertNotEqual(v3, v4) 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): def test_invalid_name(self):
with self.assertRaises(TypeError): with self.assertRaises(TypeError):

View File

@ -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.