Почему расчетная ширина и высота в пикселе строки в Tkinter различаются между платформами?
У меня есть Python script, которому нужно рассчитать точный размер произвольных строк, отображаемых в произвольных шрифтах, для создания простых диаграмм. Я легко могу сделать это с помощью 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
После того, как Ned Batchelder упомянул об этом, я обнаружил, что размер шрифтов отличается от платформы к платформе. Это может быть не прерывание сделки, если вы придерживаетесь Tkinter, который остается совместимым с самим собой. Но моя полная программа делает не использование Tkinter для выполнения фактического чертежа: он просто полагается на свои вычисления размера шрифта для генерации выход (в SVG или как Python script, который будет отправлен на Nodebox). И это там, что все идет по-настоящему неправильно:
Вывод файла mocodo http://grab.by/grabs/f67b951d092dd1f4f490e1469a53bca2.png
(Посмотрите на образ в реальном размере. Обратите внимание, что основным шрифтом, используемым для этих выходов, является не Times, а Trebuchet MS.)
Теперь я подозреваю, что с Tkinter таких исключений не избежать. Какое другое кросс-платформенное решение вы бы порекомендовали?
Ответы
Ответ 1
У вас две проблемы. Позвольте решать их по очереди
1: разница между python 2.5 и 2.6 на той же платформе с тем же шрифтом
Эти две версии python используют разные версии tk. В моем поле mac, 2.5 использует tk версии 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 независимо от фактического отображения, поэтому цифры всегда будут отличаться. Для получения дополнительной информации о различиях в том, как макросы и окна вычисляют плотность пикселей, см. Статью Dots Per Inch о википедии.
Вы можете попытаться решить это, указав шрифт в пикселях, а не в точках. Вы можете сделать это, используя отрицательные числа для размера шрифта.
Наконец, одна вещь, которую вы можете добавить в свой маленький примерный код, - это распечатать результат команды font.actual()
- вы можете увидеть что-то другое между окнами и блоками mac, что объясняет различия. Это говорит вам точно, какой шрифт используется Tk.
Ответ 2
После поиска веков я наконец узнал способ получить ширину текста в любом шрифте и размере!
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]()
Ответ 3
Вы ничего не сделали неправильно: размер шрифтов отличается от платформы к платформе.
Я не уверен, почему версия Python имеет значение, но отличия - это только один пиксель, поэтому это может быть другое округление или различный рендеринг одного и того же шрифта.