Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Implementing Typography for Skia #376

Merged
merged 13 commits into from
Aug 13, 2022
1 change: 1 addition & 0 deletions p5/core/attribs.py
Original file line number Diff line number Diff line change
Expand Up @@ -85,6 +85,7 @@ def stroke(*color_args, **color_kwargs):
"""
stroke_color = Color(*color_args, **color_kwargs)
p5.renderer.style.stroke_enabled = True
p5.renderer.style.stroke_set = True
p5.renderer.style.stroke_color = stroke_color.normalized


Expand Down
6 changes: 6 additions & 0 deletions p5/core/constants.py
Original file line number Diff line number Diff line change
Expand Up @@ -40,10 +40,16 @@
CORNER = "CORNER"
CORNERS = "CORNERS"

# Typography Setting Options
LEFT = "LEFT"
RIGHT = "RIGHT"
TOP = "TOP"
BOTTOM = "BOTTOM"
BASELINE = "BASELINE"
NORMAL = "NORMAL"
ITALIC = "ITALIC"
BOLD = "BOLD"
BOLDITALIC = "BOLDITALIC"

# Colors
colour_codes = {
Expand Down
40 changes: 22 additions & 18 deletions p5/core/font.py
Original file line number Diff line number Diff line change
Expand Up @@ -15,18 +15,6 @@
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
import textwrap

from PIL import Image
from PIL import ImageDraw
from PIL import ImageFont
from PIL import ImageFilter
from PIL import ImageChops

from .image import image
from .image import PImage
from .structure import push_style

from . import p5

__all__ = [
Expand All @@ -40,6 +28,7 @@
"text_width",
"text_ascent",
"text_descent",
"text_style",
]


Expand Down Expand Up @@ -102,14 +91,15 @@ def text(*args, wrap_at=None):
return p5.renderer.text(text_string, position, wrap_at)


def text_font(font, size=10):
def text_font(font, size=None):
"""Set current text font.

:param font:
:param font: PIL.ImageFont.ImageFont for Vispy, Object|String: a font loaded via loadFont(), or a String
representing a web safe font (a font that is generally available across all systems)
:type font: PIL.ImageFont.ImageFont

"""
p5.renderer.font_family = font
p5.renderer.text_font(font, size)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Instead of passing in a default size of 10, we should probably use the current font size.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, fixed it.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks! Looks like the size parameter is being ignored by vispy's text_font, could you also do a quick fix for that as well?



def text_align(align_x, align_y=None):
Expand All @@ -122,9 +112,9 @@ def text_align(align_x, align_y=None):
:type align_y: string

"""
p5.renderer.text_align_x = align_x
p5.renderer.style.text_align_x = align_x
if align_y:
p5.renderer.text_align_y = align_y
p5.renderer.style.text_align_y = align_y


def text_leading(leading):
Expand All @@ -135,7 +125,7 @@ def text_leading(leading):

"""

p5.renderer.text_leading = leading
p5.renderer.style.text_leading = leading


def text_size(size):
Expand Down Expand Up @@ -182,3 +172,17 @@ def text_descent():

"""
return p5.renderer.text_descent()


def text_style(s):
"""
Sets/Gets the style of the text for system fonts to NORMAL, ITALIC, BOLD or BOLDITALIC
For non-system fonts (opentype, truetype, etc.) please load styled fonts instead.

:param s: Style for the font
:type s: NORMAL | ITALIC | BOLD | BOLDITALIC

:returns: Current text style
:rtype: NORMAL | ITALIC | BOLD | BOLDITALIC
"""
return p5.renderer.text_style(s)
71 changes: 53 additions & 18 deletions p5/sketch/Skia2DRenderer/renderer2d.py
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,16 @@ class Style2D:
stroke_cap = skia.Paint.kRound_Cap
stroke_join = skia.Paint.kMiter_Join

# Flag that holds whether stroke() is called user
stroke_set = False
text_font = skia.Font(skia.Typeface())
text_leading = 0
text_size = 15
text_align = (constants.LEFT, constants.TOP)
text_wrap = None
text_baseline = constants.BASELINE
text_style = constants.NORMAL

def set_stroke_cap(self, c):
if c == constants.ROUND:
self.stroke_cap = skia.Paint.kRound_Cap
Expand All @@ -53,9 +63,6 @@ def __init__(self):
self.style = Style2D()
self.style_stack = []
self.path = None
self.font = skia.Font()
self.typeface = skia.Typeface.MakeDefault()
self.font.setTypeface(self.typeface)
self.curve_tightness = 0

# Transforms functions
Expand Down Expand Up @@ -131,36 +138,63 @@ def render_text(self, text, x, y):
# full path works relative does not
# assert (typeface is not None), "should not be NULL"
# handle alignment manually
if self.style.stroke_enabled and self.style.stroke_set:
self.paint.setStyle(skia.Paint.kStroke_Style)
self.paint.setColor(skia.Color4f(*self.style.stroke_color))
self.paint.setStrokeWidth(self.style.stroke_weight)
self.canvas.drawSimpleText(text, x, y, self.style.text_font, self.paint)

if self.style.fill_enabled:
self.paint.setStyle(skia.Paint.kFill_Style)
self.paint.setColor(self.style.fill_color)
self.canvas.drawPath(self.path, self.paint)
self.paint.setColor(skia.Color4f(*self.style.fill_color))
self.canvas.drawSimpleText(text, x, y, self.style.text_font, self.paint)

if self.style.stroke_enabled:
self.paint.setStyle(skia.Paint.kStroke_Style)
self.paint.setColor(self.style.stroke_color)
self.paint.setStrokeWidth(self.style.stroke_weight)
self.canvas.drawPath(self.path, self.paint)
self.canvas.drawSimpleText(text, x, y, self.font, self.paint)
def text(self, text, position, max_width=None, max_height=None):
self.render_text(text, *position)

def load_font(self, path):
"""
path: string
Absolute path of the font file
returns: skia.Typeface
"""
typeface = skia.Typeface.MakeFromFile(path)
typeface = skia.Typeface().MakeFromFile(path=path)
return typeface

def set_font(self, font):
def text_font(self, font, size=None):
"""
Sets the current font to be used for rendering text
"""
self.typeface = font
self.font.setTypeface(self.typeface)
if size:
self.style.text_size = size
if isinstance(font, str):
self.style.text_font.setTypeface(skia.Typeface.MakeFromName(font))
else:
self.style.text_font.setTypeface(font)

def text_size(self, size):
self.style.text_font.setSize(size)

def text_style(self, s):
skia_font_style = None
if s == constants.BOLD:
skia_font_style = skia.FontStyle.Bold()
elif s == constants.BOLDITALIC:
skia_font_style = skia.FontStyle.BoldItalic()
elif s == constants.ITALIC:
skia_font_style = skia.FontStyle.BoldItalic()
elif s == constants.NORMAL:
skia_font_style = skia.FontStyle.Normal()
else:
return self.style.text_style

self.style.text_style = s
# skia.Font.
family_name = self.style.text_font.getTypeface().getFamilyName()
typeface = skia.Typeface.MakeFromName(family_name, skia_font_style)
self.style.text_font.setTypeface(typeface)

def set_font_size(self, size):
self.font.setSize(size)
return self.style.text_style

def render_image(self, img, *args):
self.canvas.drawImage(img, *args)
Expand Down Expand Up @@ -213,7 +247,8 @@ def render(self, fill=True, stroke=True, rewind=True):

def reset(self):
self.reset_matrix()
self.font.setSize(15)
# NOTE: We have to handle PGraphics stroke_set as well separately
self.style.stroke_set = False

def line(self, path):
x1, y1, x2, y2 = path[0].x, path[0].y, path[1].x, path[1].y
Expand Down
15 changes: 6 additions & 9 deletions p5/sketch/Vispy2DRenderer/openglrenderer.py
Original file line number Diff line number Diff line change
Expand Up @@ -238,6 +238,12 @@ class Style2D:
stroke_cap = ROUND
stroke_join = MITER

# typography variables
font_family = ImageFont.load_default()
text_align_x = LEFT
text_align_y = TOP
text_leading = 0

def set_stroke_cap(self, c):
self.stroke_cap = c

Expand Down Expand Up @@ -287,15 +293,6 @@ def __init__(self, src_fbuffer, src_default):
self.matrix_stack = []

self.curve_tightness = 0

# typography variables
self.font_family = ImageFont.load_default()
self.text_align_x = LEFT
self.text_align_y = TOP
self.text_leading = 0




def render_default(self, draw_type, draw_queue):
# 1. Get the maximum number of vertices persent in the shapes
Expand Down
52 changes: 31 additions & 21 deletions p5/sketch/Vispy2DRenderer/renderer2d.py
Original file line number Diff line number Diff line change
Expand Up @@ -408,14 +408,14 @@ def text(self, text_string, position, wrap_at):
multiline = False
if not (wrap_at is None):
text_string = textwrap.fill(text_string, wrap_at)
size = self.font_family.getsize_multiline(text_string)
size = self.style.font_family.getsize_multiline(text_string)
multiline = True
elif "\n" in text_string:
multiline = True
size = list(self.font_family.getsize_multiline(text_string))
size[1] += self.text_leading * text_string.count("\n")
size = list(self.style.font_family.getsize_multiline(text_string))
size[1] += self.style.text_leading * text_string.count("\n")
else:
size = self.font_family.getsize(text_string)
size = self.style.font_family.getsize(text_string)

is_stroke_valid = False # True when stroke_weight != 0
is_min_filter = False # True when stroke_weight <0
Expand Down Expand Up @@ -445,11 +445,11 @@ def text(self, text_string, position, wrap_at):
canvas_draw.multiline_text(
text_xy,
text_string,
font=self.font_family,
spacing=self.text_leading,
font=self.style.style.font_family,
spacing=self.style.text_leading,
)
else:
canvas_draw.text(text_xy, text_string, font=self.font_family)
canvas_draw.text(text_xy, text_string, font=self.style.font_family)

text_image = PImage(*new_size)
text_image._img = canvas
Expand All @@ -465,18 +465,18 @@ def text(self, text_string, position, wrap_at):

width, height = new_size
position = list(position)
if self.text_align_x == LEFT:
if self.style.text_align_x == LEFT:
position[0] += 0
elif self.text_align_x == RIGHT:
elif self.style.text_align_x == RIGHT:
position[0] -= width
elif self.text_align_x == CENTER:
elif self.style.text_align_x == CENTER:
position[0] -= width / 2

if self.text_align_y == TOP:
if self.style.text_align_y == TOP:
position[1] += 0
elif self.text_align_y == BOTTOM:
elif self.style.text_align_y == BOTTOM:
position[1] -= height
elif self.text_align_y == CENTER:
elif self.style.text_align_y == CENTER:
position[1] -= height / 2

with push_style():
Expand All @@ -491,22 +491,32 @@ def text(self, text_string, position, wrap_at):

return text_string

def text_font(self, font, size):
self.style.font_family = font
if size:
self.text_size(size)

def text_size(self, size):
if hasattr(self.font_family, "path"):
if self.font_family.path.endswith("ttf") or self.font_family.path.endswith(
"otf"
):
self.font_family = ImageFont.truetype(self.font_family.path, size)
if hasattr(self.style.font_family, "path"):
if self.style.font_family.path.endswith(
"ttf"
) or self.style.font_family.path.endswith("otf"):
self.style.font_family = ImageFont.truetype(
self.style.font_family.path, size
)
else:
raise ValueError("text_size is not supported for Bitmap Fonts")

def text_width(self, text):
return self.font_family.getsize(text)[0]
return self.style.font_family.getsize(text)[0]

def text_ascent(self):
ascent, descent = self.font_family.getmetrics()
ascent, descent = self.style.font_family.getmetrics()
return ascent

def text_descent(self):
ascent, descent = self.font_family.getmetrics()
ascent, descent = self.style.font_family.getmetrics()
return descent

def text_style(self):
raise NotImplementedError("Not Implemented error in Vispy")
Loading