Почему рассчитанная ширина и высота в пикселях строки в Tkinter отличаются между платформами? - PullRequest
29 голосов
/ 27 мая 2010

У меня есть скрипт на Python, который должен рассчитать точный размер произвольных строк, отображаемых в произвольных шрифтах, чтобы генерировать простые диаграммы. Я легко могу сделать это с Tkinter.

import Tkinter as tk
import tkFont
root = tk.Tk()
canvas = tk.Canvas(root, width=300, height=200)
canvas.pack()
(x,y) = (5,5)
text = "yellow world"
fonts = []
for (family,size) in [("times",12),("times",24)]:
    font = tkFont.Font(family=family, size=size)
    (w,h) = (font.measure(text),font.metrics("linespace"))
    print "%s %s: (%s,%s)" % (family,size,w,h)
    canvas.create_rectangle(x,y,x+w,y+h)
    canvas.create_text(x,y,text=text,font=font,anchor=tk.NW)
    fonts.append(font) # save object from garbage collecting
    y += h+5
tk.mainloop()

Результаты зависят от версии Python и / или системы:

Python 2,5 Mac 0S X, времена 12: (63,12), времена 24: (128,24). Python 2.6 Mac OS X, время 12: (64,14), время 24: (127,27). Python 2.6 Windows XP, времена 12: (78,19), времена 24: (169,36) http://grab.by/grabs/d24a5035cce0d8032ea4e04cb8c85959.png

После того, как Нед Батчелдер упомянул об этом, я обнаружил, что размер шрифтов отличается от платформы к платформе. Возможно, это не нарушит условия сделки, если вы придерживаетесь Tkinter, который остается согласованным с самим собой. Но моя полная программа не не использует Tkinter для выполнения фактического рисования: он просто полагается на свои вычисления размера шрифта для генерации вывода (в SVG или в виде скрипта Python для отправки в Nodebox ). И там все идет не так:

Выход mocodo http://grab.by/grabs/f67b951d092dd1f4f490e1469a53bca2.png

(Пожалуйста, посмотрите на изображение в реальном размере . Обратите внимание, что основным шрифтом, используемым для этих выходных данных, является не Times, а Trebuchet MS.)

Теперь я подозреваю, что таких расхождений нельзя избежать с помощью Tkinter. Какое другое кроссплатформенное решение вы бы порекомендовали?

Ответы [ 3 ]

19 голосов
/ 01 июня 2010

У вас две проблемы. Давайте решать их по одному

1: разница между python 2.5 и 2.6 на одной платформе с одинаковым шрифтом

Эти две версии python используют разные версии tk. На моем компьютере Mac 2.5 используется версия 8.4.19, а 2.6 - 8.5.7. В версии 8.5.2 в tk были внесены некоторые изменения в функции измерения шрифта в tk. Предполагая, что изменения были улучшениями, я думаю, что можно с уверенностью предположить, что числа, которые вы получаете из Python 2.6, более точны, чем числа из 2.5.

2: разница между Python 2.6 на Mac и 2.6 на ПК.

Очевидно, что на скриншотах, которые вы включили, ПК использует больший шрифт, и, таким образом, вы получаете большие цифры для измерения. Вопрос в том, почему? Вы указываете размер шрифта в точках (1/72 дюйма). Чтобы Tk (или любая система рендеринга) отображала шрифт, ему нужно знать, сколько пикселей в дюйме на реальном дисплее. Это будет отличаться в разных системах, и Tk не всегда получает точное число от базовой ОС для выполнения своих расчетов.

Исторически Apple и Microsoft стандартизировали 72ppi и 96ppi независимо от фактического дисплея, поэтому цифры всегда будут разными. Для получения дополнительной информации о различиях в том, как Mac и Windows вычисляют плотность пикселей, см. Статью Точек на дюйм в Википедии.

Вы можете попытаться решить эту проблему, указав шрифт в пикселях, а не в точках. Вы можете сделать это, используя отрицательные числа для размера шрифта.

Наконец, одна вещь, которую вы можете добавить в свой маленький пример кода, это распечатать результат команды font.actual() - вы можете увидеть что-то другое между вашими окнами и блоками Mac, что объясняет различия там. Это говорит вам точно, какой шрифт используется Tk.

7 голосов
/ 21 декабря 2015

После поиска возрастов я наконец нашел способ получить ширину текста любого шрифта и размера!

from tkinter import *

Window = Tk()
Window.geometry("500x500+80+80")

frame = Frame(Window) # this will hold the label
frame.pack(side = "top")

# CALCULATE:
measure = Label(frame, font = ("Purisa", 10), text = "The width of this in pixels is.....", bg = "yellow")
measure.grid(row = 0, column = 0) # put the label in
measure.update_idletasks() # this is VERY important, it makes python calculate the width
width = measure.winfo_width() # get the width

# PROOF IT WORKS:
canvas = Canvas(frame, width = 400, height = 200, bg = "light green")
canvas.grid(row = 1, column = 0, columnspan = 100) # collumnspan is 100 so that the line lines up with the text
line = canvas.create_line(0, 10, width, 10, width = 4) # make a line the same length as the text
canvas.create_text(10, 20, font = ("Purisa", 10), text = "... "+str(width)+" Pixels", anchor = "nw")

Строка, которую я делаю, является доказательством того, что это работает для любого шрифта.

Я проверил это для разных шрифтов и размеров, насколько я знаю, это работает.

Это изображение вывода: This is a picture of the output

5 голосов
/ 27 мая 2010

Вы не сделали ничего плохого: размер шрифтов действительно отличается от платформы к платформе.

Я не уверен, почему версия Python имеет значение, но различия составляют всего один пиксель, поэтому это может быть разное округление или различное отображение одного и того же шрифта.

...