extract the code from fixedToStr into its own function (defined in roundTools), to be used for serializing fractional F2Dot14 angles for COLRv1 PaintRotate etc.
252 lines
6.6 KiB
Python
252 lines
6.6 KiB
Python
"""
|
|
The `OpenType specification <https://docs.microsoft.com/en-us/typography/opentype/spec/otff#data-types>`_
|
|
defines two fixed-point data types:
|
|
|
|
``Fixed``
|
|
A 32-bit signed fixed-point number with a 16 bit twos-complement
|
|
magnitude component and 16 fractional bits.
|
|
``F2DOT14``
|
|
A 16-bit signed fixed-point number with a 2 bit twos-complement
|
|
magnitude component and 14 fractional bits.
|
|
|
|
To support reading and writing data with these data types, this module provides
|
|
functions for converting between fixed-point, float and string representations.
|
|
|
|
.. data:: MAX_F2DOT14
|
|
|
|
The maximum value that can still fit in an F2Dot14. (1.99993896484375)
|
|
"""
|
|
|
|
from .roundTools import otRound, nearestMultipleShortestRepr
|
|
import logging
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
__all__ = [
|
|
"MAX_F2DOT14",
|
|
"fixedToFloat",
|
|
"floatToFixed",
|
|
"floatToFixedToFloat",
|
|
"floatToFixedToStr",
|
|
"fixedToStr",
|
|
"strToFixed",
|
|
"strToFixedToFloat",
|
|
"ensureVersionIsLong",
|
|
"versionToFixed",
|
|
]
|
|
|
|
|
|
MAX_F2DOT14 = 0x7FFF / (1 << 14)
|
|
|
|
|
|
def fixedToFloat(value, precisionBits):
|
|
"""Converts a fixed-point number to a float given the number of
|
|
precision bits.
|
|
|
|
Args:
|
|
value (int): Number in fixed-point format.
|
|
precisionBits (int): Number of precision bits.
|
|
|
|
Returns:
|
|
Floating point value.
|
|
|
|
Examples::
|
|
|
|
>>> import math
|
|
>>> f = fixedToFloat(-10139, precisionBits=14)
|
|
>>> math.isclose(f, -0.61883544921875)
|
|
True
|
|
"""
|
|
return value / (1 << precisionBits)
|
|
|
|
|
|
def floatToFixed(value, precisionBits):
|
|
"""Converts a float to a fixed-point number given the number of
|
|
precision bits.
|
|
|
|
Args:
|
|
value (float): Floating point value.
|
|
precisionBits (int): Number of precision bits.
|
|
|
|
Returns:
|
|
int: Fixed-point representation.
|
|
|
|
Examples::
|
|
|
|
>>> floatToFixed(-0.61883544921875, precisionBits=14)
|
|
-10139
|
|
>>> floatToFixed(-0.61884, precisionBits=14)
|
|
-10139
|
|
"""
|
|
return otRound(value * (1 << precisionBits))
|
|
|
|
|
|
def floatToFixedToFloat(value, precisionBits):
|
|
"""Converts a float to a fixed-point number and back again.
|
|
|
|
By converting the float to fixed, rounding it, and converting it back
|
|
to float again, this returns a floating point values which is exactly
|
|
representable in fixed-point format.
|
|
|
|
Note: this **is** equivalent to ``fixedToFloat(floatToFixed(value))``.
|
|
|
|
Args:
|
|
value (float): The input floating point value.
|
|
precisionBits (int): Number of precision bits.
|
|
|
|
Returns:
|
|
float: The transformed and rounded value.
|
|
|
|
Examples::
|
|
>>> import math
|
|
>>> f1 = -0.61884
|
|
>>> f2 = floatToFixedToFloat(-0.61884, precisionBits=14)
|
|
>>> f1 != f2
|
|
True
|
|
>>> math.isclose(f2, -0.61883544921875)
|
|
True
|
|
"""
|
|
scale = 1 << precisionBits
|
|
return otRound(value * scale) / scale
|
|
|
|
|
|
def fixedToStr(value, precisionBits):
|
|
"""Converts a fixed-point number to a string representing a decimal float.
|
|
|
|
This chooses the float that has the shortest decimal representation (the least
|
|
number of fractional decimal digits).
|
|
|
|
For example, to convert a fixed-point number in a 2.14 format, use
|
|
``precisionBits=14``::
|
|
|
|
>>> fixedToStr(-10139, precisionBits=14)
|
|
'-0.61884'
|
|
|
|
This is pretty slow compared to the simple division used in ``fixedToFloat``.
|
|
Use sporadically when you need to serialize or print the fixed-point number in
|
|
a human-readable form.
|
|
It uses nearestMultipleShortestRepr under the hood.
|
|
|
|
Args:
|
|
value (int): The fixed-point value to convert.
|
|
precisionBits (int): Number of precision bits, *up to a maximum of 16*.
|
|
|
|
Returns:
|
|
str: A string representation of the value.
|
|
"""
|
|
scale = 1 << precisionBits
|
|
return nearestMultipleShortestRepr(value/scale, factor=1.0/scale)
|
|
|
|
|
|
def strToFixed(string, precisionBits):
|
|
"""Converts a string representing a decimal float to a fixed-point number.
|
|
|
|
Args:
|
|
string (str): A string representing a decimal float.
|
|
precisionBits (int): Number of precision bits, *up to a maximum of 16*.
|
|
|
|
Returns:
|
|
int: Fixed-point representation.
|
|
|
|
Examples::
|
|
|
|
>>> ## to convert a float string to a 2.14 fixed-point number:
|
|
>>> strToFixed('-0.61884', precisionBits=14)
|
|
-10139
|
|
"""
|
|
value = float(string)
|
|
return otRound(value * (1 << precisionBits))
|
|
|
|
|
|
def strToFixedToFloat(string, precisionBits):
|
|
"""Convert a string to a decimal float with fixed-point rounding.
|
|
|
|
This first converts string to a float, then turns it into a fixed-point
|
|
number with ``precisionBits`` fractional binary digits, then back to a
|
|
float again.
|
|
|
|
This is simply a shorthand for fixedToFloat(floatToFixed(float(s))).
|
|
|
|
Args:
|
|
string (str): A string representing a decimal float.
|
|
precisionBits (int): Number of precision bits.
|
|
|
|
Returns:
|
|
float: The transformed and rounded value.
|
|
|
|
Examples::
|
|
|
|
>>> import math
|
|
>>> s = '-0.61884'
|
|
>>> bits = 14
|
|
>>> f = strToFixedToFloat(s, precisionBits=bits)
|
|
>>> math.isclose(f, -0.61883544921875)
|
|
True
|
|
>>> f == fixedToFloat(floatToFixed(float(s), precisionBits=bits), precisionBits=bits)
|
|
True
|
|
"""
|
|
value = float(string)
|
|
scale = 1 << precisionBits
|
|
return otRound(value * scale) / scale
|
|
|
|
|
|
def floatToFixedToStr(value, precisionBits):
|
|
"""Convert float to string with fixed-point rounding.
|
|
|
|
This uses the shortest decimal representation (ie. the least
|
|
number of fractional decimal digits) to represent the equivalent
|
|
fixed-point number with ``precisionBits`` fractional binary digits.
|
|
It uses nearestMultipleShortestRepr under the hood.
|
|
|
|
>>> floatToFixedToStr(-0.61883544921875, precisionBits=14)
|
|
'-0.61884'
|
|
|
|
Args:
|
|
value (float): The float value to convert.
|
|
precisionBits (int): Number of precision bits, *up to a maximum of 16*.
|
|
|
|
Returns:
|
|
str: A string representation of the value.
|
|
|
|
"""
|
|
scale = 1 << precisionBits
|
|
return nearestMultipleShortestRepr(value, factor=1.0/scale)
|
|
|
|
|
|
def ensureVersionIsLong(value):
|
|
"""Ensure a table version is an unsigned long.
|
|
|
|
OpenType table version numbers are expressed as a single unsigned long
|
|
comprising of an unsigned short major version and unsigned short minor
|
|
version. This function detects if the value to be used as a version number
|
|
looks too small (i.e. is less than ``0x10000``), and converts it to
|
|
fixed-point using :func:`floatToFixed` if so.
|
|
|
|
Args:
|
|
value (Number): a candidate table version number.
|
|
|
|
Returns:
|
|
int: A table version number, possibly corrected to fixed-point.
|
|
"""
|
|
if value < 0x10000:
|
|
newValue = floatToFixed(value, 16)
|
|
log.warning(
|
|
"Table version value is a float: %.4f; "
|
|
"fix to use hex instead: 0x%08x", value, newValue)
|
|
value = newValue
|
|
return value
|
|
|
|
|
|
def versionToFixed(value):
|
|
"""Ensure a table version number is fixed-point.
|
|
|
|
Args:
|
|
value (str): a candidate table version number.
|
|
|
|
Returns:
|
|
int: A table version number, possibly corrected to fixed-point.
|
|
"""
|
|
value = int(value, 0) if value.startswith("0") else float(value)
|
|
value = ensureVersionIsLong(value)
|
|
return value
|