123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253 |
- """
- 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
|