2021-05-04 16:12:24 +02:00
|
|
|
import pytest
|
2023-05-24 12:51:24 -06:00
|
|
|
from io import StringIO
|
|
|
|
from fontTools.misc.xmlWriter import XMLWriter
|
2021-05-04 16:12:24 +02:00
|
|
|
from fontTools.varLib.models import VariationModel
|
|
|
|
from fontTools.varLib.varStore import OnlineVarStoreBuilder, VarStoreInstancer
|
2021-05-04 17:08:27 +02:00
|
|
|
from fontTools.ttLib import TTFont, newTable
|
2021-05-04 16:12:24 +02:00
|
|
|
from fontTools.ttLib.tables._f_v_a_r import Axis
|
2021-05-04 17:08:27 +02:00
|
|
|
from fontTools.ttLib.tables.otBase import OTTableReader, OTTableWriter
|
|
|
|
from fontTools.ttLib.tables.otTables import VarStore
|
2021-05-04 16:12:24 +02:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"locations, masterValues",
|
|
|
|
[
|
|
|
|
(
|
|
|
|
[{}, {"a": 1}],
|
|
|
|
[
|
2022-06-23 11:58:50 -06:00
|
|
|
[10, 10], # Test NO_VARIATION_INDEX
|
2021-05-04 16:12:24 +02:00
|
|
|
[100, 2000],
|
|
|
|
[100, 22000],
|
|
|
|
],
|
|
|
|
),
|
|
|
|
(
|
|
|
|
[{}, {"a": 1}, {"b": 1}, {"a": 1, "b": 1}],
|
|
|
|
[
|
|
|
|
[10, 20, 40, 60],
|
|
|
|
[100, 2000, 400, 6000],
|
2021-05-04 17:08:27 +02:00
|
|
|
[7100, 22000, 4000, 30000],
|
2021-05-04 16:12:24 +02:00
|
|
|
],
|
|
|
|
),
|
2021-05-04 18:21:37 +02:00
|
|
|
(
|
|
|
|
[{}, {"a": 1}],
|
|
|
|
[
|
|
|
|
[10, 20],
|
|
|
|
[42000, 100],
|
|
|
|
[100, 52000],
|
|
|
|
],
|
|
|
|
),
|
|
|
|
(
|
|
|
|
[{}, {"a": 1}, {"b": 1}, {"a": 1, "b": 1}],
|
|
|
|
[
|
|
|
|
[10, 20, 40, 60],
|
|
|
|
[40000, 42000, 400, 6000],
|
|
|
|
[100, 22000, 4000, 173000],
|
|
|
|
],
|
|
|
|
),
|
2021-05-04 16:12:24 +02:00
|
|
|
],
|
|
|
|
)
|
|
|
|
def test_onlineVarStoreBuilder(locations, masterValues):
|
|
|
|
axisTags = sorted({k for loc in locations for k in loc})
|
|
|
|
model = VariationModel(locations)
|
|
|
|
builder = OnlineVarStoreBuilder(axisTags)
|
|
|
|
builder.setModel(model)
|
2021-05-04 16:39:20 +02:00
|
|
|
varIdxs = []
|
2021-05-04 16:12:24 +02:00
|
|
|
for masters in masterValues:
|
2021-05-04 16:39:20 +02:00
|
|
|
_, varIdx = builder.storeMasters(masters)
|
|
|
|
varIdxs.append(varIdx)
|
2021-05-04 16:12:24 +02:00
|
|
|
|
|
|
|
varStore = builder.finish()
|
2021-05-04 16:19:48 +02:00
|
|
|
mapping = varStore.optimize()
|
2021-05-04 16:39:20 +02:00
|
|
|
varIdxs = [mapping[varIdx] for varIdx in varIdxs]
|
2021-05-04 16:12:24 +02:00
|
|
|
|
2021-05-04 19:29:39 +02:00
|
|
|
dummyFont = TTFont()
|
2021-05-04 17:08:27 +02:00
|
|
|
writer = OTTableWriter()
|
2021-05-04 19:29:39 +02:00
|
|
|
varStore.compile(writer, dummyFont)
|
2021-05-04 17:08:27 +02:00
|
|
|
data = writer.getAllData()
|
|
|
|
reader = OTTableReader(data)
|
|
|
|
varStore = VarStore()
|
2021-05-04 19:29:39 +02:00
|
|
|
varStore.decompile(reader, dummyFont)
|
2021-05-04 17:08:27 +02:00
|
|
|
|
2021-05-04 19:29:39 +02:00
|
|
|
fvarAxes = [buildAxis(axisTag) for axisTag in axisTags]
|
2021-05-04 16:12:24 +02:00
|
|
|
instancer = VarStoreInstancer(varStore, fvarAxes)
|
2021-05-04 16:39:20 +02:00
|
|
|
for masters, varIdx in zip(masterValues, varIdxs):
|
2021-05-04 16:12:24 +02:00
|
|
|
base, *rest = masters
|
|
|
|
for expectedValue, loc in zip(masters, locations):
|
|
|
|
instancer.setLocation(loc)
|
|
|
|
value = base + instancer[varIdx]
|
|
|
|
assert expectedValue == value
|
|
|
|
|
|
|
|
|
|
|
|
def buildAxis(axisTag):
|
|
|
|
axis = Axis()
|
|
|
|
axis.axisTag = axisTag
|
|
|
|
return axis
|
2023-05-24 12:51:24 -06:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"numRegions, varData, expectedNumVarData, expectedBytes",
|
|
|
|
[
|
|
|
|
(
|
|
|
|
5,
|
|
|
|
[
|
|
|
|
[10, 10, 0, 0, 20],
|
|
|
|
{3: 300},
|
|
|
|
],
|
|
|
|
1,
|
|
|
|
156,
|
|
|
|
),
|
|
|
|
(
|
|
|
|
5,
|
|
|
|
[
|
|
|
|
[10, 10, 0, 0, 20],
|
|
|
|
[10, 11, 0, 0, 20],
|
|
|
|
[10, 12, 0, 0, 20],
|
|
|
|
[10, 13, 0, 0, 20],
|
|
|
|
{3: 300},
|
|
|
|
],
|
|
|
|
2,
|
|
|
|
186,
|
|
|
|
),
|
2023-05-24 13:00:59 -06:00
|
|
|
(
|
|
|
|
5,
|
|
|
|
[
|
|
|
|
[10, 11, 0, 0, 20],
|
|
|
|
[10, 300, 0, 0, 20],
|
|
|
|
[10, 301, 0, 0, 20],
|
|
|
|
[10, 302, 0, 0, 20],
|
|
|
|
[10, 303, 0, 0, 20],
|
|
|
|
[10, 304, 0, 0, 20],
|
|
|
|
],
|
|
|
|
1,
|
|
|
|
180,
|
|
|
|
),
|
2023-05-24 12:51:24 -06:00
|
|
|
],
|
|
|
|
)
|
|
|
|
def test_optimize(numRegions, varData, expectedNumVarData, expectedBytes):
|
|
|
|
locations = [{i: i / 16384.0} for i in range(numRegions)]
|
|
|
|
axisTags = sorted({k for loc in locations for k in loc})
|
|
|
|
|
|
|
|
model = VariationModel(locations)
|
|
|
|
builder = OnlineVarStoreBuilder(axisTags)
|
|
|
|
builder.setModel(model)
|
|
|
|
|
|
|
|
for data in varData:
|
|
|
|
if type(data) is dict:
|
|
|
|
newData = [0] * numRegions
|
|
|
|
for k, v in data.items():
|
|
|
|
newData[k] = v
|
|
|
|
data = newData
|
|
|
|
|
|
|
|
builder.storeMasters(data)
|
|
|
|
|
|
|
|
varStore = builder.finish()
|
|
|
|
mapping = varStore.optimize()
|
|
|
|
|
|
|
|
assert len(varStore.VarData) == expectedNumVarData
|
|
|
|
|
|
|
|
dummyFont = TTFont()
|
|
|
|
|
|
|
|
writer = XMLWriter(StringIO())
|
|
|
|
varStore.toXML(writer, dummyFont)
|
|
|
|
xml = writer.file.getvalue()
|
|
|
|
|
|
|
|
writer = OTTableWriter()
|
|
|
|
varStore.compile(writer, dummyFont)
|
|
|
|
data = writer.getAllData()
|
|
|
|
|
|
|
|
assert len(data) == expectedBytes, xml
|