Подтвердить что ты не робот

Почему расчетная ширина и высота в пикселе строки в 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 таких исключений не избежать. Какое другое кросс-платформенное решение вы бы порекомендовали?

4b9b3361

Ответ 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 имеет значение, но отличия - это только один пиксель, поэтому это может быть другое округление или различный рендеринг одного и того же шрифта.