"""Convert fontFeatures.Routine objects to binary GPOS/GSUB lookups."""
import fontTools.otlLib.builder as otl
from fontTools.varLib.builder import buildVarDevTable
from fontTools.feaLib.variableScalar import VariableScalar
import itertools


def toOTLookup(self, font, ff):
    """Converts a fontFeatures.Routine object to binary.

    Args:
        font: A ``TTFont`` object.
        ff: The parent ``FontFeatures`` object containing this routine.

    Returns a list of ``fontTools.otlLib.builder`` Builder objects allowing this
    routine to be converted to binary layout format.
    """

    lookuptypes = [x.lookup_type() for x in self.rules]
    if not all([lu == lookuptypes[0] for lu in lookuptypes]):
        raise ValueError("For now, a routine can only contain rules of the same type")

    if not all([self.rules[0].flags == rule.flags for rule in self.rules]):
        raise ValueError("For now, a routine can only contain rules of the same flags")
    self.flags = self.rules[0].flags

    if self.stage == "pos":
        return buildPos(self, font, lookuptypes[0], ff)
    if self.stage == "sub":
        return buildSub(self, font, lookuptypes[0], ff)


def makeAnchor(anchor, ff):
    """Creates an OpenType anchor object from. If the anchor position contains
     a variable scalar, these variable scalars are saved in the ``GDEF`` variation
     store.

     Args:
        anchor: a tuple of x,y position. These elements may be numbers or
            :py:class:`VariableScalar` objects.

    Returns: an ``otTables.Anchor`` object.
    """
    x, y = anchor
    if isinstance(x, VariableScalar):
        x_def, x_index = x.add_to_variation_store(ff.varstorebuilder)
    else:
        x_def, x_index = int(x), None
    if isinstance(y, VariableScalar):
        y_def, y_index = y.add_to_variation_store(ff.varstorebuilder)
    else:
        y_def, y_index = int(y), None
    anchor = otl.buildAnchor(x_def, y_def)
    if x_index is not None and x_index != 0xFFFFFFFF:
        anchor.XDeviceTable = buildVarDevTable(x_index)
        anchor.Format = 3
    if y_index is not None and x_index != 0xFFFFFFFF:
        anchor.YDeviceTable = buildVarDevTable(y_index)
        anchor.Format = 3
    return anchor


def buildPos(self, font, lookuptype, ff):
    """Build a GPOS subtable."""
    builders = []
    if lookuptype == 1:
        builder = otl.SinglePosBuilder(font, self.address)
        for rule in self.rules:
            ot_valuerecs = [
                x.toOTValueRecord(ff, pairPosContext=False) for x in rule.valuerecords
            ]
            for glyph in rule.glyphs[0]:
                builder.add_pos(rule.address, glyph, ot_valuerecs[0])
    elif lookuptype == 2:
        builder = otl.PairPosBuilder(font, self.address)
        for rule in self.rules:
            ot_valuerecs = [
                x.toOTValueRecord(ff, pairPosContext=True) for x in rule.valuerecords
            ]
            if len(rule.glyphs[0]) == 1 and len(rule.glyphs[1]) == 1:
                builder.addGlyphPair(
                    rule.address,
                    rule.glyphs[0][0],
                    ot_valuerecs[0],
                    rule.glyphs[1][0],
                    ot_valuerecs[1],
                )
            else:
                builder.addClassPair(
                    rule.address,
                    tuple(rule.glyphs[0]),
                    ot_valuerecs[0],
                    tuple(rule.glyphs[1]),
                    ot_valuerecs[1],
                )
    elif lookuptype == 3:
        builder = otl.CursivePosBuilder(font, self.address)
        for rule in self.rules:
            allglyphs = set(rule.bases.keys()) | set(rule.marks.keys())
            for g in allglyphs:
                builder.attachments[g] = (
                    g in rule.bases and makeAnchor(rule.bases[g], ff) or None,
                    g in rule.marks and makeAnchor(rule.marks[g], ff) or None,
                )
    elif lookuptype == 4 or lookuptype == 6:
        if lookuptype == 4:
            builder = otl.MarkBasePosBuilder(font, self.address)
            baseholder = builder.bases
        else:
            builder = otl.MarkMarkPosBuilder(font, self.address)
            baseholder = builder.baseMarks
        for r in self.rules:
            for mark, anchor in r.marks.items():
                if mark in builder.marks:
                    raise ValueError(
                        "A mark glyph %s tried to be in two categories (%s, %s)"
                        % (mark, r.mark_name, builder.marks[mark][0])
                    )
                builder.marks[mark] = (r.mark_name, makeAnchor(anchor, ff))
            for base, anchor in r.bases.items():
                if base not in baseholder:
                    baseholder[base] = {}
                baseholder[base][r.mark_name] = makeAnchor(anchor, ff)
        # XXX. We may need to express this as multiple lookups
    elif lookuptype == 8:
        builder = otl.ChainContextPosBuilder(font, self.address)
        for r in self.rules:
            new_lookup_list = []
            import fontFeatures

            if isinstance(r, fontFeatures.Positioning):
                for glyphs, vr in zip(r.glyphs, r.valuerecords):
                    # Make a fake pos routine
                    subbuilder = buildPos(
                        fontFeatures.Routine(
                            rules=[
                                fontFeatures.Positioning([glyphs], valuerecords=[vr])
                            ]
                        ),
                        font,
                        1,
                        ff,
                    )
                    builders.extend(subbuilder)
                    new_lookup_list.append(subbuilder)
                glyphs = r.glyphs
            else:
                for list_of_lookups in r.lookups:
                    new_lookup_list.append(
                        [lu.routine.__builder for lu in (list_of_lookups or [])]
                    )
                glyphs = r.input
            builder.rules.append(
                otl.ChainContextualRule(
                    r.precontext or [],
                    glyphs or [],
                    r.postcontext or [],
                    new_lookup_list,
                )
            )
    else:
        raise ValueError("Don't know how to build a POS type %i lookup" % lookuptype)
    builder.lookupflag = self.flags
    # XXX mark filtering set
    self.__builder = builder
    builders.append(builder)
    return builders


def buildSub(self, font, lookuptype, ff):
    """Build a GSUB subtable."""
    builders = []
    if lookuptype == 1:
        builder = otl.SingleSubstBuilder(font, self.address)
        for rule in self.rules:
            for left, right in zip(rule.input[0], rule.replacement[0]):
                builder.mapping[left] = right
    elif lookuptype == 2:
        builder = otl.MultipleSubstBuilder(font, self.address)
        for rule in self.rules:
            builder.mapping[rule.input[0][0]] = [x[0] for x in rule.replacement]
    elif lookuptype == 4:
        builder = otl.LigatureSubstBuilder(font, self.address)
        for rule in self.rules:
            for sequence in itertools.product(*rule.input):
                builder.ligatures[sequence] = rule.replacement[0][0]
    elif lookuptype == 6:
        builder = otl.ChainContextSubstBuilder(font, self.address)
        for r in self.rules:
            import fontFeatures

            if isinstance(r, fontFeatures.Substitution) and r.replacement:
                lookups = []
                for left, right in zip(r.input, r.replacement):
                    # Make a fake sub routine

                    subbuilder = buildSub(
                        fontFeatures.Routine(
                            rules=[
                                fontFeatures.Substitution([left], replacement=[right])
                            ]
                        ),
                        font,
                        1,
                        ff,
                    )
                    builders.extend(subbuilder)
                    lookups.append(subbuilder)
            else:
                lookups = []
                for list_of_lookups in r.lookups:
                    lookups.append(
                        [lu.routine.__builder for lu in (list_of_lookups or [])]
                    )
            builder.rules.append(
                otl.ChainContextualRule(
                    r.precontext or [],
                    r.input or [],
                    r.postcontext or [],
                    lookups,
                )
            )
    elif lookuptype == 8:
        builder = otl.ReverseChainSingleSubstBuilder(font, self.address)
        for rule in self.rules:
            mapping = {l[0]: r[0] for l, r in zip(rule.input, rule.replacement)}
            builder.rules.append((rule.precontext, rule.postcontext, mapping))
    else:
        raise ValueError("Don't know how to build a SUB type %i lookup" % lookuptype)

    builder.lookupflag = self.flags
    # XXX mark filtering set
    self.__builder = builder
    builders.append(builder)
    return builders
