Hot-keys on this page
r m x p toggle line displays
j k next/prev highlighted chunk
0 (zero) top of page
1 (one) first highlighted chunk
# -*- coding: utf-8 -*- Tamari Interval-posets
This module implements Tamari interval-posets: combinatorial objects which represent intervals of the Tamari order. They have been introduced in [ChP2015]_ and allow for many combinatorial operations on Tamari intervals. In particular, they are linked to :class:`DyckWords` and :class:`BinaryTrees`. An introduction into Tamari interval-posets is given in Chapter 7 of [Pons2013]_.
The Tamari lattice can be defined as a lattice structure on either of several classes of Catalan objects, especially binary trees and Dyck paths [TamBrack1962]_ [HuangTamari1972]_ [Sta-EC2]_. An interval can be seen as a pair of comparable elements. The number of intervals has been given in [ChapTamari08]_.
REFERENCES:
.. [ChP2015] Grégory Châtel and Viviane Pons. *Counting smaller elements in the tamari and m-tamari lattices*. Journal of Combinatorial Theory, Series A. (2015). :arxiv:`1311.3922`.
.. [Pons2013] Viviane Pons, *Combinatoire algébrique liée aux ordres sur les permutations*. PhD Thesis. (2013). :arxiv:`1310.1805v1`.
.. [TamBrack1962] Dov Tamari. *The algebra of bracketings and their enumeration*. Nieuw Arch. Wisk. (1962).
.. [HuangTamari1972] Samuel Huang and Dov Tamari. *Problems of associativity: A simple proof for the lattice property of systems ordered by a semi-associative law*. J. Combinatorial Theory Ser. A. (1972). http://www.sciencedirect.com/science/article/pii/0097316572900039 .
.. [ChapTamari08] Frédéric Chapoton. *Sur le nombre d'intervalles dans les treillis de Tamari*. Sem. Lothar. Combin. (2008). :arxiv:`math/0602368v1`.
.. [FPR15] Wenjie Fang and Louis-François Préville-Ratelle, *From generalized Tamari intervals to non-separable planar maps*. :arxiv:`1511.05937`
.. [Pons2018] Viviane Pons, *The Rise-Contact involution on Tamari intervals*
.. [Rog2018] Baptiste Rognerud, *Exceptional and modern intervals of the Tamari lattice*. :arxiv:`1801.04097`
AUTHORS:
- Viviane Pons 2014: initial implementation - Frédéric Chapoton 2014: review - Darij Grinberg 2014: review - Travis Scrimshaw 2014: review """ # **************************************************************************** # Copyright (C) 2013 Viviane Pons <viviane.pons@univie.ac.at>, # # Distributed under the terms of the GNU General Public License (GPL) # as published by the Free Software Foundation; either version 2 of # the License, or (at your option) any later version. # http://www.gnu.org/licenses/ # ****************************************************************************
r""" The class of Tamari interval-posets.
An interval-poset is a labelled poset of size `n`, with labels `1, 2, \ldots, n`, satisfying the following conditions:
- if `a < c` (as integers) and `a` precedes `c` in the poset, then, for all `b` such that `a < b < c`, `b` precedes `c`,
- if `a < c` (as integers) and `c` precedes `a` in the poset, then, for all `b` such that `a < b < c`, `b` precedes `a`.
We use the word "precedes" here to distinguish the poset order and the natural order on numbers. "Precedes" means "is smaller than with respect to the poset structure"; this does not imply a covering relation.
Interval-posets of size `n` are in bijection with intervals of the Tamari lattice of binary trees of size `n`. Specifically, if `P` is an interval-poset of size `n`, then the set of linear extensions of `P` (as permutations in `S_n`) is an interval in the right weak order (see :meth:`~sage.combinat.permutation.Permutation.permutohedron_lequal`), and is in fact the preimage of an interval in the Tamari lattice (of binary trees of size `n`) under the operation which sends a permutation to its right-to-left binary search tree (:meth:`~sage.combinat.permutation.Permutation.binary_search_tree` with the ``left_to_right`` variable set to ``False``) without its labelling.
INPUT:
- ``size`` -- an integer, the size of the interval-posets (number of vertices)
- ``relations`` -- a list (or tuple) of pairs ``(a,b)`` (themselves lists or tuples), each representing a relation of the form '`a` precedes `b`' in the poset.
- ``check`` -- (default: ``True``) whether to check the interval-poset condition or not.
.. WARNING::
The ``relations`` input can be a list or tuple, but not an iterator (nor should its entries be iterators).
NOTATION:
Here and in the following, the signs `<` and `>` always refer to the natural ordering on integers, whereas the word "precedes" refers to the order of the interval-poset. "Minimal" and "maximal" refer to the natural ordering on integers.
The *increasing relations* of an interval-poset `P` mean the pairs `(a, b)` of elements of `P` such that `a < b` as integers and `a` precedes `b` in `P`. The *initial forest* of `P` is the poset obtained by imposing (only) the increasing relations on the ground set of `P`. It is a sub-interval poset of `P`, and is a forest with its roots on top. This forest is usually given the structure of a planar forest by ordering brother nodes by their labels; it then has the property that if its nodes are traversed in post-order (see :meth:`~sage.combinat.abstract_tree.AbstractTree.post_order_traversal`, and traverse the trees of the forest from left to right as well), then the labels encountered are `1, 2, \ldots, n` in this order.
The *decreasing relations* of an interval-poset `P` mean the pairs `(a, b)` of elements of `P` such that `b < a` as integers and `a` precedes `b` in `P`. The *final forest* of `P` is the poset obtained by imposing (only) the decreasing relations on the ground set of `P`. It is a sub-interval poset of `P`, and is a forest with its roots on top. This forest is usually given the structure of a planar forest by ordering brother nodes by their labels; it then has the property that if its nodes are traversed in pre-order (see :meth:`~sage.combinat.abstract_tree.AbstractTree.pre_order_traversal`, and traverse the trees of the forest from left to right as well), then the labels encountered are `1, 2, \ldots, n` in this order.
EXAMPLES::
sage: TamariIntervalPoset(0,[]) The Tamari interval of size 0 induced by relations [] sage: TamariIntervalPoset(3,[]) The Tamari interval of size 3 induced by relations [] sage: TamariIntervalPoset(3,[(1,2)]) The Tamari interval of size 3 induced by relations [(1, 2)] sage: TamariIntervalPoset(3,[(1,2),(2,3)]) The Tamari interval of size 3 induced by relations [(1, 2), (2, 3)] sage: TamariIntervalPoset(3,[(1,2),(2,3),(1,3)]) The Tamari interval of size 3 induced by relations [(1, 2), (2, 3)] sage: TamariIntervalPoset(3,[(1,2),(3,2)]) The Tamari interval of size 3 induced by relations [(1, 2), (3, 2)] sage: TamariIntervalPoset(3,[[1,2],[2,3]]) The Tamari interval of size 3 induced by relations [(1, 2), (2, 3)] sage: TamariIntervalPoset(3,[[1,2],[2,3],[1,2],[1,3]]) The Tamari interval of size 3 induced by relations [(1, 2), (2, 3)]
sage: TamariIntervalPoset(3,[(3,4)]) Traceback (most recent call last): ... ValueError: The relations do not correspond to the size of the poset.
sage: TamariIntervalPoset(2,[(2,1),(1,2)]) Traceback (most recent call last): ... ValueError: The graph is not directed acyclic
sage: TamariIntervalPoset(3,[(1,3)]) Traceback (most recent call last): ... ValueError: This does not satisfy the Tamari interval-poset condition.
It is also possible to transform a poset directly into an interval-poset::
sage: TIP = TamariIntervalPosets() sage: p = Poset(([1,2,3], [(1,2)])) sage: TIP(p) The Tamari interval of size 3 induced by relations [(1, 2)] sage: TIP(Poset({1: []})) The Tamari interval of size 1 induced by relations [] sage: TIP(Poset({})) The Tamari interval of size 0 induced by relations [] """ def __classcall_private__(cls, *args, **opts): r""" Ensure that interval-posets created by the enumerated sets and directly are the same and that they are instances of :class:`TamariIntervalPoset`.
TESTS::
sage: ip = TamariIntervalPoset(4,[(2,4),(3,4),(2,1),(3,1)]) sage: ip.parent() Interval-posets sage: type(ip) <class 'sage.combinat.interval_posets.TamariIntervalPosets_all_with_category.element_class'>
sage: ip2 = TamariIntervalPosets()(4,[(2,4),(3,4),(2,1),(3,1)]) sage: ip2.parent() is ip.parent() True sage: type(ip) is type(ip2) True
sage: ip3 = TamariIntervalPosets(4)([(2,4),(3,4),(2,1),(3,1)]) sage: ip3.parent() is ip.parent() False sage: type(ip3) is type(ip) True """
r""" TESTS::
sage: TamariIntervalPoset(3,[(1,2),(3,2)]).parent() Interval-posets """ # This can happen as the Poset constructor automatically adds # in elements from the relations.
r""" Set the latex options for use in the ``_latex_`` function. The default values are set in the ``__init__`` function.
- ``tikz_scale`` -- (default: 1) scale for use with the tikz package
- ``line_width`` -- (default: 1 * ``tikz_scale``) value representing the line width
- ``color_decreasing`` -- (default: red) the color for decreasing relations
- ``color_increasing`` -- (default: blue) the color for increasing relations
- ``hspace`` -- (default: 1) the difference between horizontal coordinates of adjacent vertices
- ``vspace`` -- (default: 1) the difference between vertical coordinates of adjacent vertices
INPUT:
- ``D`` -- a dictionary with a list of latex parameters to change
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(2,4),(3,4),(2,1),(3,1)]) sage: ip.latex_options()["color_decreasing"] 'red' sage: ip.set_latex_options({"color_decreasing":'green'}) sage: ip.latex_options()["color_decreasing"] 'green' sage: ip.set_latex_options({"color_increasing":'black'}) sage: ip.latex_options()["color_increasing"] 'black'
To change the default options for all interval-posets, use the parent's latex options::
sage: ip = TamariIntervalPoset(4,[(2,4),(3,4),(2,1),(3,1)]) sage: ip2 = TamariIntervalPoset(4,[(1,2),(2,3)]) sage: ip.latex_options()["color_decreasing"] 'red' sage: ip2.latex_options()["color_decreasing"] 'red' sage: TamariIntervalPosets.options(latex_color_decreasing='green') sage: ip.latex_options()["color_decreasing"] 'green' sage: ip2.latex_options()["color_decreasing"] 'green'
Next we set a local latex option and show the global option does not override it::
sage: ip.set_latex_options({"color_decreasing": 'black'}) sage: ip.latex_options()["color_decreasing"] 'black' sage: TamariIntervalPosets.options(latex_color_decreasing='blue') sage: ip.latex_options()["color_decreasing"] 'black' sage: ip2.latex_options()["color_decreasing"] 'blue' sage: TamariIntervalPosets.options._reset() """
r""" Return the latex options for use in the ``_latex_`` function as a dictionary. The default values are set using the options.
- ``tikz_scale`` -- (default: 1) scale for use with the tikz package
- ``line_width`` -- (default: 1) value representing the line width (additionally scaled by ``tikz_scale``)
- ``color_decreasing`` -- (default: ``'red'``) the color for decreasing relations
- ``color_increasing`` -- (default: ``'blue'``) the color for increasing relations
- ``hspace`` -- (default: 1) the difference between horizontal coordinates of adjacent vertices
- ``vspace`` -- (default: 1) the difference between vertical coordinates of adjacent vertices
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(2,4),(3,4),(2,1),(3,1)]) sage: ip.latex_options()['color_decreasing'] 'red' sage: ip.latex_options()['hspace'] 1 """
""" Compute a nice embedding.
If `x` precedes `y`, then `y` will always be placed on top of `x` and/or to the right of `x`. Decreasing relations tend to be drawn vertically and increasing relations horizontally. The algorithm tries to avoid superposition but on big interval-posets, it might happen.
OUTPUT:
a dictionary {vertex: (x,y)}
EXAMPLES::
sage: ti = TamariIntervalPosets(4)[2] sage: list(ti._find_node_positions().values()) [[0, 0], [0, -1], [0, -2], [1, -2]] """
decreasing_parent < to_draw[-1][0]): else: else: current_parent.append(increasing_parent) parenty.append(nodey)
""" Return a picture.
The picture represents the Hasse diagram, where the covers are colored in blue if they are increasing and in red if they are decreasing.
This uses the same coordinates as the latex view.
EXAMPLES::
sage: ti = TamariIntervalPosets(4)[2] sage: ti.plot() Graphics object consisting of 6 graphics primitives """ else: G.set_edge_label(a, b, 1)
r""" A latex representation of ``self`` using the tikzpicture package.
This picture shows the union of the Hasse diagrams of the initial and final forests.
If `x` precedes `y`, then `y` will always be placed on top of `x` and/or to the right of `x`. Decreasing relations tend to be drawn vertically and increasing relations horizontally. The algorithm tries to avoid superposition but on big interval-posets, it might happen.
You can use ``self.set_latex_options()`` to change default latex options. Or you can use the parent's options.
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(2,4),(3,4),(2,1),(3,1)]) sage: print(ip._latex_()) \begin{tikzpicture}[scale=1] \node(T1) at (1,0) {1}; \node(T2) at (0,-1) {2}; \node(T3) at (1,-2) {3}; \node(T4) at (2,-1) {4}; \draw[line width = 0.5, color=red] (T3) -- (T1); \draw[line width = 0.5, color=red] (T2) -- (T1); \draw[line width = 0.5, color=blue] (T2) -- (T4); \draw[line width = 0.5, color=blue] (T3) -- (T4); \end{tikzpicture} """
r""" Internal method to draw vertices """
r""" Internal method to draw increasing relations """
r""" Internal method to draw decreasing relations """
nodes = "\\node(T0) at (0,0){$\emptyset$};" relations = "" else:
r""" Return ``self`` as a labelled poset.
An interval-poset is indeed constructed from a labelled poset which is stored internally. This method allows to access the poset and all the associated methods.
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(1,2),(3,2),(2,4),(3,4)]) sage: pos = ip.poset(); pos Finite poset containing 4 elements sage: pos.maximal_chains() [[3, 2, 4], [1, 2, 4]] sage: pos.maximal_elements() [4] sage: pos.is_lattice() False """
""" Return the hash of ``self``.
EXAMPLES::
sage: len(set(hash(u) for u in TamariIntervalPosets(4))) 68 """
def increasing_cover_relations(self): r""" Return the cover relations of the initial forest of ``self`` (the poset formed by keeping only the relations of the form `a` precedes `b` with `a < b`).
The initial forest of ``self`` is a forest with its roots being on top. It is also called the increasing poset of ``self``.
.. WARNING::
This method computes the cover relations of the initial forest. This is not identical with the cover relations of ``self`` which happen to be increasing!
.. SEEALSO::
:meth:`initial_forest`
EXAMPLES::
sage: TamariIntervalPoset(4,[(1,2),(3,2),(2,4),(3,4)]).increasing_cover_relations() [(1, 2), (2, 4), (3, 4)] sage: TamariIntervalPoset(3,[(1,2),(1,3),(2,3)]).increasing_cover_relations() [(1, 2), (2, 3)] """
r""" Return the root vertices of the initial forest of ``self``, i.e., the vertices `a` of ``self`` such that there is no `b > a` with `a` precedes `b`.
OUTPUT:
The list of all roots of the initial forest of ``self``, in decreasing order.
EXAMPLES::
sage: ip = TamariIntervalPoset(6,[(3,2),(4,3),(5,2),(6,5),(1,2),(3,5),(4,5)]); ip The Tamari interval of size 6 induced by relations [(1, 2), (3, 5), (4, 5), (6, 5), (5, 2), (4, 3), (3, 2)] sage: ip.increasing_roots() [6, 5, 2] sage: ip.initial_forest().increasing_roots() [6, 5, 2] """ return []
r""" Return the children of ``v`` in the initial forest of ``self``.
INPUT:
- ``v`` -- an integer representing a vertex of ``self`` (between 1 and ``size``)
OUTPUT:
The list of all children of ``v`` in the initial forest of ``self``, in decreasing order.
EXAMPLES::
sage: ip = TamariIntervalPoset(6,[(3,2),(4,3),(5,2),(6,5),(1,2),(3,5),(4,5)]); ip The Tamari interval of size 6 induced by relations [(1, 2), (3, 5), (4, 5), (6, 5), (5, 2), (4, 3), (3, 2)] sage: ip.increasing_children(2) [1] sage: ip.increasing_children(5) [4, 3] sage: ip.increasing_children(1) [] """
r""" Return the vertex parent of ``v`` in the initial forest of ``self``.
This is the lowest (as integer!) vertex `b > v` such that `v` precedes `b`. If there is no such vertex (that is, `v` is an increasing root), then ``None`` is returned.
INPUT:
- ``v`` -- an integer representing a vertex of ``self`` (between 1 and ``size``)
EXAMPLES::
sage: ip = TamariIntervalPoset(6,[(3,2),(4,3),(5,2),(6,5),(1,2),(3,5),(4,5)]); ip The Tamari interval of size 6 induced by relations [(1, 2), (3, 5), (4, 5), (6, 5), (5, 2), (4, 3), (3, 2)] sage: ip.increasing_parent(1) 2 sage: ip.increasing_parent(3) 5 sage: ip.increasing_parent(4) 5 sage: ip.increasing_parent(5) is None True """
def decreasing_cover_relations(self): r""" Return the cover relations of the final forest of ``self`` (the poset formed by keeping only the relations of the form `a` precedes `b` with `a > b`).
The final forest of ``self`` is a forest with its roots being on top. It is also called the decreasing poset of ``self``.
.. WARNING::
This method computes the cover relations of the final forest. This is not identical with the cover relations of ``self`` which happen to be decreasing!
.. SEEALSO::
:meth:`final_forest`
EXAMPLES::
sage: TamariIntervalPoset(4,[(2,1),(3,2),(3,4),(4,2)]).decreasing_cover_relations() [(4, 2), (3, 2), (2, 1)] sage: TamariIntervalPoset(4,[(2,1),(4,3),(2,3)]).decreasing_cover_relations() [(4, 3), (2, 1)] sage: TamariIntervalPoset(3,[(2,1),(3,1),(3,2)]).decreasing_cover_relations() [(3, 2), (2, 1)] """
r""" Return the root vertices of the final forest of ``self``, i.e., the vertices `b` such that there is no `a < b` with `b` preceding `a`.
OUTPUT:
The list of all roots of the final forest of ``self``, in increasing order.
EXAMPLES::
sage: ip = TamariIntervalPoset(6,[(3,2),(4,3),(5,2),(6,5),(1,2),(3,5),(4,5)]); ip The Tamari interval of size 6 induced by relations [(1, 2), (3, 5), (4, 5), (6, 5), (5, 2), (4, 3), (3, 2)] sage: ip.decreasing_roots() [1, 2] sage: ip.final_forest().decreasing_roots() [1, 2] """
r""" Return the children of ``v`` in the final forest of ``self``.
INPUT:
- ``v`` -- an integer representing a vertex of ``self`` (between 1 and ``size``)
OUTPUT:
The list of all children of ``v`` in the final forest of ``self``, in increasing order.
EXAMPLES::
sage: ip = TamariIntervalPoset(6,[(3,2),(4,3),(5,2),(6,5),(1,2),(3,5),(4,5)]); ip The Tamari interval of size 6 induced by relations [(1, 2), (3, 5), (4, 5), (6, 5), (5, 2), (4, 3), (3, 2)] sage: ip.decreasing_children(2) [3, 5] sage: ip.decreasing_children(3) [4] sage: ip.decreasing_children(1) [] """
r""" Return the vertex parent of ``v`` in the final forest of ``self``. This is the highest (as integer!) vertex `a < v` such that ``v`` precedes ``a``. If there is no such vertex (that is, `v` is a decreasing root), then ``None`` is returned.
INPUT:
- ``v`` -- an integer representing a vertex of ``self`` (between 1 and ``size``)
EXAMPLES::
sage: ip = TamariIntervalPoset(6,[(3,2),(4,3),(5,2),(6,5),(1,2),(3,5),(4,5)]); ip The Tamari interval of size 6 induced by relations [(1, 2), (3, 5), (4, 5), (6, 5), (5, 2), (4, 3), (3, 2)] sage: ip.decreasing_parent(4) 3 sage: ip.decreasing_parent(3) 2 sage: ip.decreasing_parent(5) 2 sage: ip.decreasing_parent(2) is None True """
r""" Return whether ``e1`` precedes or equals ``e2`` in ``self``.
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(1,2),(2,3)]) sage: ip.le(1,2) True sage: ip.le(1,3) True sage: ip.le(2,3) True sage: ip.le(3,4) False sage: ip.le(1,1) True """
r""" Return whether ``e1`` strictly precedes ``e2`` in ``self``.
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(1,2),(2,3)]) sage: ip.lt(1,2) True sage: ip.lt(1,3) True sage: ip.lt(2,3) True sage: ip.lt(3,4) False sage: ip.lt(1,1) False """
r""" Return whether ``e2`` precedes or equals ``e1`` in ``self``.
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(1,2),(2,3)]) sage: ip.ge(2,1) True sage: ip.ge(3,1) True sage: ip.ge(3,2) True sage: ip.ge(4,3) False sage: ip.ge(1,1) True """
r""" Return whether ``e2`` strictly precedes ``e1`` in ``self``.
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(1,2),(2,3)]) sage: ip.gt(2,1) True sage: ip.gt(3,1) True sage: ip.gt(3,2) True sage: ip.gt(4,3) False sage: ip.gt(1,1) False """
r""" Return the size (number of vertices) of the interval-poset.
EXAMPLES::
sage: TamariIntervalPoset(3,[(2,1),(3,1)]).size() 3 """
r""" Return the complement of the interval-poset ``self``.
If `P` is a Tamari interval-poset of size `n`, then the *complement* of `P` is defined as the interval-poset `Q` whose base set is `[n] = \{1, 2, \ldots, n\}` (just as for `P`), but whose order relation has `a` precede `b` if and only if `n + 1 - a` precedes `n + 1 - b` in `P`.
In terms of the Tamari lattice, the *complement* is the symmetric of ``self``. It is formed from the left-right symmeterized of the binary trees of the interval (switching left and right subtrees, see :meth:`~sage.combinat.binary_tree.BinaryTree.left_right_symmetry`). In particular, initial intervals are sent to final intervals and vice-versa.
EXAMPLES::
sage: TamariIntervalPoset(3, [(2, 1), (3, 1)]).complement() The Tamari interval of size 3 induced by relations [(1, 3), (2, 3)] sage: TamariIntervalPoset(0, []).complement() The Tamari interval of size 0 induced by relations [] sage: ip = TamariIntervalPoset(4, [(1, 2), (2, 4), (3, 4)]) sage: ip.complement() == TamariIntervalPoset(4, [(2, 1), (3, 1), (4, 3)]) True sage: ip.lower_binary_tree() == ip.complement().upper_binary_tree().left_right_symmetry() True sage: ip.upper_binary_tree() == ip.complement().lower_binary_tree().left_right_symmetry() True sage: ip.is_initial_interval() True sage: ip.complement().is_final_interval() True """ for i in self._poset.cover_relations_iterator()]
""" Return the image of ``self`` by the left-branch involution.
OUTPUT: an interval-poset
.. SEEALSO:: :meth:`rise_contact_involution`
EXAMPLES::
sage: tip = TamariIntervalPoset(8, [(1,2), (2,4), (3,4), (6,7), (3,2), (5,4), (6,4), (8,7)]) sage: t = tip.left_branch_involution(); t The Tamari interval of size 8 induced by relations [(1, 6), (2, 6), (3, 5), (4, 5), (5, 6), (6, 8), (7, 8), (7, 6), (4, 3), (3, 1), (2, 1)] sage: t.left_branch_involution() == tip True
REFERENCES:
- [Pons2018]_ """
""" Return the image of ``self`` by the rise-contact involution.
OUTPUT: an interval-poset
This is defined by conjugating the complement involution by the left-branch involution
.. SEEALSO:: :meth:`left_branch_involution`, :meth:`complement`
EXAMPLES::
sage: tip = TamariIntervalPoset(8, [(1,2), (2,4), (3,4), (6,7), (3,2), (5,4), (6,4), (8,7)]) sage: t = tip.rise_contact_involution(); t The Tamari interval of size 8 induced by relations [(2, 8), (3, 8), (4, 5), (5, 7), (6, 7), (7, 8), (8, 1), (7, 2), (6, 2), (5, 3), (4, 3), (3, 2), (2, 1)] sage: t.rise_contact_involution() == tip True sage: tip.lower_dyck_word().number_of_touch_points() == t.upper_dyck_word().number_of_initial_rises() True sage: tip.number_of_tamari_inversions() == t.number_of_tamari_inversions() True
REFERENCES:
- [Pons2018]_ """
r""" Return the Tamari insertion of an integer `i` into the interval-poset ``self``.
If `P` is a Tamari interval-poset of size `n` and `i` is an integer with `1 \leq i \leq n+1`, then the Tamari insertion of `i` into `P` is defined as the Tamari interval-poset of size `n+1` which corresponds to the interval `[C_1, C_2]` on the Tamari lattice, where the binary trees `C_1` and `C_2` are defined as follows: We write the interval-poset `P` as `[B_1, B_2]` for two binary trees `B_1` and `B_2`. We label the vertices of each of these two trees with the integers `1, 2, \ldots, i-1, i+1, i+2, \ldots, n+1` in such a way that the trees are binary search trees (this labelling is unique). Then, we insert `i` into each of these trees (in the way as explained in :meth:`~sage.combinat.binary_tree.LabelledBinaryTree.binary_search_insert`). The shapes of the resulting two trees are denoted `C_1` and `C_2`.
An alternative way to construct the insertion of `i` into `P` is by relabeling each vertex `u` of `P` satisfying `u \geq i` (as integers) as `u+1`, and then adding a vertex `i` which should precede `i-1` and `i+1`.
.. TODO::
To study this, it would be more natural to define interval-posets on arbitrary ordered sets rather than just on `\{1, 2, \ldots, n\}`.
EXAMPLES::
sage: ip = TamariIntervalPoset(4, [(2, 3), (4, 3)]); ip The Tamari interval of size 4 induced by relations [(2, 3), (4, 3)] sage: ip.insertion(1) The Tamari interval of size 5 induced by relations [(1, 2), (3, 4), (5, 4)] sage: ip.insertion(2) The Tamari interval of size 5 induced by relations [(2, 3), (3, 4), (5, 4), (2, 1)] sage: ip.insertion(3) The Tamari interval of size 5 induced by relations [(2, 4), (3, 4), (5, 4), (3, 2)] sage: ip.insertion(4) The Tamari interval of size 5 induced by relations [(2, 3), (4, 5), (5, 3), (4, 3)] sage: ip.insertion(5) The Tamari interval of size 5 induced by relations [(2, 3), (5, 4), (4, 3)]
sage: ip = TamariIntervalPoset(0, []) sage: ip.insertion(1) The Tamari interval of size 1 induced by relations []
sage: ip = TamariIntervalPoset(1, []) sage: ip.insertion(1) The Tamari interval of size 2 induced by relations [(1, 2)] sage: ip.insertion(2) The Tamari interval of size 2 induced by relations [(2, 1)]
TESTS:
Verifying that the two ways of computing insertion are equivalent::
sage: def insert_alternative(T, i): ....: # Just another way to compute the insertion of i into T. ....: from sage.combinat.binary_tree import LabelledBinaryTree ....: B1 = T.lower_binary_tree().canonical_labelling() ....: B2 = T.upper_binary_tree().canonical_labelling() ....: # We should relabel the trees to "make space" for a label i, ....: # but we don't, because it doesn't make a difference: The ....: # binary search insertion will go precisely the same, because ....: # an integer equal to the label of the root gets sent onto ....: # the left branch. ....: C1 = B1.binary_search_insert(i) ....: C2 = B2.binary_search_insert(i) ....: return TamariIntervalPosets.from_binary_trees(C1, C2) sage: def test_equivalence(n): ....: for T in TamariIntervalPosets(n): ....: for i in range(1, n + 2): ....: if not (insert_alternative(T, i) == T.insertion(i)): ....: print(T, i) ....: return False ....: return True sage: test_equivalence(3) True """ raise ValueError("integer to be inserted not " "in the appropriate interval")
for (a, b) in self.decreasing_cover_relations()] for (a, b) in self.increasing_cover_relations()]
r""" TESTS::
sage: TamariIntervalPoset(3,[(2,1),(3,1)]) The Tamari interval of size 3 induced by relations [(3, 1), (2, 1)] sage: TamariIntervalPoset(3,[(3,1),(2,1)]) The Tamari interval of size 3 induced by relations [(3, 1), (2, 1)] sage: TamariIntervalPoset(3,[(2,3),(2,1)]) The Tamari interval of size 3 induced by relations [(2, 3), (2, 1)] """ self.increasing_cover_relations() + self.decreasing_cover_relations())
""" Return an ascii art picture of ``self``.
This is a picture of the Hasse diagram. Vertices from `1` to `n` are placed on the diagonal from top-left to bottom-right. Then increasing covers are drawn above the diagonal and decreasing covers are drawn below the diagonal.
EXAMPLES::
sage: T = TamariIntervalPosets(5)[56] sage: ascii_art(T) O-----------+ O--------+ +--O--+ | O--+ O sage: T.poset().cover_relations() [[3, 4], [3, 2], [4, 5], [2, 5], [1, 5]] """
# put symbol b at position x, y # on top of existing symbols there if b == a: pass elif b == '---': M[i][j] = '-+-' elif b == ' | ': M[i][j] = '-+ ' if b == a: pass elif b == '---': M[i][j] = '-+-' elif b == ' | ': M[i][j] = ' +-' if b == a: pass elif b == '-+ ': M[i][j] = '-+-' elif b == ' +-': M[i][j] = '-+-' elif b == ' +-': M[i][j] = ' +-'
else: M[i][i] = '-O-'
superpose(k, j, ' | ') superpose(i, k, '---') else:
""" Return an unicode picture of ``self``.
This is a picture of the Hasse diagram. Vertices from `1` to `n` are placed on the diagonal from top-left to bottom-right. Then increasing covers are drawn above the diagonal and decreasing covers are drawn below the diagonal.
EXAMPLES::
sage: T = TamariIntervalPosets(5)[56] sage: unicode_art(T) o───╮ o──┤ ╰o╮│ o┤ o sage: T.poset().cover_relations() [[3, 4], [3, 2], [4, 5], [2, 5], [1, 5]] """
# put symbol b at position x, y # on top of existing symbols there if b == a: pass elif b == u'─': M[i][j] = u'┬' elif b == u'│': M[i][j] = u'┤' if b == a: pass elif b == u'─': M[i][j] = u'┴' elif b == u'│': M[i][j] = u'├' if b == a: pass elif b == u'╮': M[i][j] = u'┬' elif b == u'╰': M[i][j] = u'┴' elif b == u'╰': M[i][j] = u'├'
superpose(k, j, u'│') superpose(i, k, u'─') else:
r""" TESTS::
sage: TamariIntervalPoset(0,[]) == TamariIntervalPoset(0,[]) True sage: TamariIntervalPoset(1,[]) == TamariIntervalPoset(0,[]) False sage: TamariIntervalPoset(3,[(1,2),(3,2)]) == TamariIntervalPoset(3,[(3,2),(1,2)]) True sage: TamariIntervalPoset(3,[(1,2),(3,2)]) == TamariIntervalPoset(3,[(1,2)]) False sage: TamariIntervalPoset(3,[(1,2),(3,2)]) != TamariIntervalPoset(3,[(3,2),(1,2)]) False
sage: ip1 = TamariIntervalPoset(4,[(1,2),(2,3),(4,3)]) sage: ip2 = TamariIntervalPoset(4,[(1,2),(2,3)]) sage: ip1 <= ip2 True sage: ip1 <= ip1 True sage: ip2 <= ip1 False """ return op == op_NE self._cover_relations == other._cover_relations) self._cover_relations == other._cover_relations) return self.parent().lt(self, other) if op == op_GT: return self.parent().gt(self, other) if op == op_GE: return self.parent().ge(self, other)
r""" Iterate through the vertices of ``self``.
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(1,2),(3,2)]) sage: [i for i in ip] [1, 2, 3, 4] """
r""" Return whether the interval represented by ``other`` is contained in ``self`` as an interval of the Tamari lattice.
In terms of interval-posets, it means that all relations of ``self`` are relations of ``other``.
INPUT:
- ``other`` -- an interval-poset
EXAMPLES::
sage: ip1 = TamariIntervalPoset(4,[(1,2),(2,3),(4,3)]) sage: ip2 = TamariIntervalPoset(4,[(2,3)]) sage: ip2.contains_interval(ip1) True sage: ip3 = TamariIntervalPoset(4,[(2,1)]) sage: ip2.contains_interval(ip3) False sage: ip4 = TamariIntervalPoset(3,[(2,3)]) sage: ip2.contains_interval(ip4) False """
r""" Return whether the interval represented by ``other`` is contained in ``self`` as an interval of the Tamari lattice and if they share the same lower bound.
As interval-posets, it means that ``other`` contains the relations of ``self`` plus some extra increasing relations.
INPUT:
- ``other`` -- an interval-poset
EXAMPLES::
sage: ip1 = TamariIntervalPoset(4,[(1,2),(2,3),(4,3)]); sage: ip2 = TamariIntervalPoset(4,[(4,3)]) sage: ip2.lower_contains_interval(ip1) True sage: ip2.contains_interval(ip1) and ip2.lower_binary_tree() == ip1.lower_binary_tree() True sage: ip3 = TamariIntervalPoset(4,[(4,3),(2,1)]) sage: ip2.contains_interval(ip3) True sage: ip2.lower_binary_tree() == ip3.lower_binary_tree() False sage: ip2.lower_contains_interval(ip3) False """ return False
r""" Return whether the interval represented by ``other`` is contained in ``self`` as an interval of the Tamari lattice and if they share the same upper bound.
As interval-posets, it means that ``other`` contains the relations of ``self`` plus some extra decreasing relations.
INPUT:
- ``other`` -- an interval-poset
EXAMPLES::
sage: ip1 = TamariIntervalPoset(4,[(1,2),(2,3),(4,3)]) sage: ip2 = TamariIntervalPoset(4,[(1,2),(2,3)]) sage: ip2.upper_contains_interval(ip1) True sage: ip2.contains_interval(ip1) and ip2.upper_binary_tree() == ip1.upper_binary_tree() True sage: ip3 = TamariIntervalPoset(4,[(1,2),(2,3),(3,4)]) sage: ip2.upper_contains_interval(ip3) False sage: ip2.contains_interval(ip3) True sage: ip2.upper_binary_tree() == ip3.upper_binary_tree() False """ return False
r""" Return whether the permutation ``perm`` is a linear extension of ``self``.
INPUT:
- ``perm`` -- a permutation of the size of ``self``
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(1,2),(2,3),(4,3)]) sage: ip.is_linear_extension([1,4,2,3]) True sage: ip.is_linear_extension(Permutation([1,4,2,3])) True sage: ip.is_linear_extension(Permutation([1,4,3,2])) False """
r""" Return whether the interval represented by ``self`` contains the binary tree ``binary_tree``.
INPUT:
- ``binary_tree`` -- a binary tree
.. SEEALSO:: :meth:`contains_dyck_word`
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(2,4),(3,4),(2,1),(3,1)]) sage: ip.contains_binary_tree(BinaryTree([[None,[None,[]]],None])) True sage: ip.contains_binary_tree(BinaryTree([None,[[[],None],None]])) True sage: ip.contains_binary_tree(BinaryTree([[],[[],None]])) False sage: ip.contains_binary_tree(ip.lower_binary_tree()) True sage: ip.contains_binary_tree(ip.upper_binary_tree()) True sage: all(ip.contains_binary_tree(bt) for bt in ip.binary_trees()) True
"""
r""" Return whether the interval represented by ``self`` contains the Dyck word ``dyck_word``.
INPUT:
- ``dyck_word`` -- a Dyck word
.. SEEALSO:: :meth:`contains_binary_tree`
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(2,4),(3,4),(2,1),(3,1)]) sage: ip.contains_dyck_word(DyckWord([1,1,1,0,0,0,1,0])) True sage: ip.contains_dyck_word(DyckWord([1,1,0,1,0,1,0,0])) True sage: ip.contains_dyck_word(DyckWord([1,0,1,1,0,1,0,0])) False sage: ip.contains_dyck_word(ip.lower_dyck_word()) True sage: ip.contains_dyck_word(ip.upper_dyck_word()) True sage: all(ip.contains_dyck_word(bt) for bt in ip.dyck_words()) True """
r""" Return the interval-poset formed by combining the relations from both ``self`` and ``other``. It corresponds to the intersection of the two corresponding intervals of the Tamari lattice.
INPUT:
- ``other`` -- an interval-poset of the same size as ``self``
EXAMPLES::
sage: ip1 = TamariIntervalPoset(4,[(1,2),(2,3)]) sage: ip2 = TamariIntervalPoset(4,[(4,3)]) sage: ip1.intersection(ip2) The Tamari interval of size 4 induced by relations [(1, 2), (2, 3), (4, 3)] sage: ip3 = TamariIntervalPoset(4,[(2,1)]) sage: ip1.intersection(ip3) Traceback (most recent call last): ... ValueError: This intersection is empty, it does not correspond to an interval-poset. sage: ip4 = TamariIntervalPoset(3,[(2,3)]) sage: ip2.intersection(ip4) Traceback (most recent call last): ... ValueError: Intersections are only possible on interval-posets of the same size. """
r""" Return the initial forest of ``self``, i.e., the interval-poset formed from only the increasing relations of ``self``.
.. SEEALSO:: :meth:`final_forest`
EXAMPLES::
sage: TamariIntervalPoset(4,[(1,2),(3,2),(2,4),(3,4)]).initial_forest() The Tamari interval of size 4 induced by relations [(1, 2), (2, 4), (3, 4)] sage: ip = TamariIntervalPoset(4,[(1,2),(2,3)]) sage: ip.initial_forest() == ip True """
r""" Return the final forest of ``self``, i.e., the interval-poset formed with only the decreasing relations of ``self``.
.. SEEALSO:: :meth:`initial_forest`
EXAMPLES::
sage: TamariIntervalPoset(4,[(2,1),(3,2),(3,4),(4,2)]).final_forest() The Tamari interval of size 4 induced by relations [(4, 2), (3, 2), (2, 1)] sage: ip = TamariIntervalPoset(3,[(2,1),(3,1)]) sage: ip.final_forest() == ip True """
r""" Return if ``self`` corresponds to an initial interval of the Tamari lattice, i.e. if its lower end is the smallest element of the lattice. It consists of checking that ``self`` does not contain any decreasing relations.
.. SEEALSO:: :meth:`is_final_interval`
EXAMPLES::
sage: ip = TamariIntervalPoset(4, [(1, 2), (2, 4), (3, 4)]) sage: ip.is_initial_interval() True sage: ip.lower_dyck_word() [1, 0, 1, 0, 1, 0, 1, 0] sage: ip = TamariIntervalPoset(4, [(1, 2), (2, 4), (3, 4), (3, 2)]) sage: ip.is_initial_interval() False sage: ip.lower_dyck_word() [1, 0, 1, 1, 0, 0, 1, 0] sage: all(DyckWord([1,0,1,0,1,0]).tamari_interval(dw).is_initial_interval() for dw in DyckWords(3)) True """
r""" Return if ``self`` corresponds to a final interval of the Tamari lattice, i.e. if its upper end is the largest element of the lattice. It consists of checking that ``self`` does not contain any increasing relations.
.. SEEALSO:: :meth:`is_initial_interval`
EXAMPLES::
sage: ip = TamariIntervalPoset(4, [(4, 3), (3, 1), (2, 1)]) sage: ip.is_final_interval() True sage: ip.upper_dyck_word() [1, 1, 1, 1, 0, 0, 0, 0] sage: ip = TamariIntervalPoset(4, [(4, 3), (3, 1), (2, 1), (2, 3)]) sage: ip.is_final_interval() False sage: ip.upper_dyck_word() [1, 1, 0, 1, 1, 0, 0, 0] sage: all(dw.tamari_interval(DyckWord([1, 1, 1, 0, 0, 0])).is_final_interval() for dw in DyckWords(3)) True """
r""" Return the lowest binary tree in the interval of the Tamari lattice represented by ``self``.
This is a binary tree. It is the shape of the unique binary search tree whose left-branch ordered forest (i.e., the result of applying :meth:`~sage.combinat.binary_tree.BinaryTree.to_ordered_tree_left_branch` and cutting off the root) is the final forest of ``self``.
.. SEEALSO:: :meth:`lower_dyck_word`
EXAMPLES::
sage: ip = TamariIntervalPoset(6,[(3,2),(4,3),(5,2),(6,5),(1,2),(4,5)]); ip The Tamari interval of size 6 induced by relations [(1, 2), (4, 5), (6, 5), (5, 2), (4, 3), (3, 2)] sage: ip.lower_binary_tree() [[., .], [[., [., .]], [., .]]] sage: TamariIntervalPosets.final_forest(ip.lower_binary_tree()) == ip.final_forest() True sage: ip == TamariIntervalPosets.from_binary_trees(ip.lower_binary_tree(),ip.upper_binary_tree()) True """
r""" Return the lowest Dyck word in the interval of the Tamari lattice represented by ``self``.
.. SEEALSO:: :meth:`lower_binary_tree`
EXAMPLES::
sage: ip = TamariIntervalPoset(6,[(3,2),(4,3),(5,2),(6,5),(1,2),(4,5)]); ip The Tamari interval of size 6 induced by relations [(1, 2), (4, 5), (6, 5), (5, 2), (4, 3), (3, 2)] sage: ip.lower_dyck_word() [1, 0, 1, 1, 1, 0, 0, 1, 1, 0, 0, 0] sage: TamariIntervalPosets.final_forest(ip.lower_dyck_word()) == ip.final_forest() True sage: ip == TamariIntervalPosets.from_dyck_words(ip.lower_dyck_word(),ip.upper_dyck_word()) True """
r""" Return the highest binary tree in the interval of the Tamari lattice represented by ``self``.
This is a binary tree. It is the shape of the unique binary search tree whose right-branch ordered forest (i.e., the result of applying :meth:`~sage.combinat.binary_tree.BinaryTree.to_ordered_tree_right_branch` and cutting off the root) is the initial forest of ``self``.
.. SEEALSO:: :meth:`upper_dyck_word`
EXAMPLES::
sage: ip = TamariIntervalPoset(6,[(3,2),(4,3),(5,2),(6,5),(1,2),(4,5)]); ip The Tamari interval of size 6 induced by relations [(1, 2), (4, 5), (6, 5), (5, 2), (4, 3), (3, 2)] sage: ip.upper_binary_tree() [[., .], [., [[., .], [., .]]]] sage: TamariIntervalPosets.initial_forest(ip.upper_binary_tree()) == ip.initial_forest() True sage: ip == TamariIntervalPosets.from_binary_trees(ip.lower_binary_tree(),ip.upper_binary_tree()) True """
r""" Return the highest Dyck word in the interval of the Tamari lattice represented by ``self``.
.. SEEALSO:: :meth:`upper_binary_tree`
EXAMPLES::
sage: ip = TamariIntervalPoset(6,[(3,2),(4,3),(5,2),(6,5),(1,2),(4,5)]); ip The Tamari interval of size 6 induced by relations [(1, 2), (4, 5), (6, 5), (5, 2), (4, 3), (3, 2)] sage: ip.upper_dyck_word() [1, 0, 1, 1, 1, 0, 1, 1, 0, 0, 0, 0] sage: TamariIntervalPosets.initial_forest(ip.upper_dyck_word()) == ip.initial_forest() True sage: ip == TamariIntervalPosets.from_dyck_words(ip.lower_dyck_word(),ip.upper_dyck_word()) True """
r""" Return the renormalized sub-poset of ``self`` consisting solely of integers from ``start`` (inclusive) to ``end`` (not inclusive).
"Renormalized" means that these integers are relabelled `1,2,\ldots,k` in the obvious way (i.e., by subtracting ``start - 1``).
INPUT:
- ``start`` -- an integer, the starting vertex (inclusive) - ``end`` -- an integer, the ending vertex (not inclusive)
EXAMPLES::
sage: ip = TamariIntervalPoset(6,[(3,2),(4,3),(5,2),(6,5),(1,2),(3,5),(4,5)]); ip The Tamari interval of size 6 induced by relations [(1, 2), (3, 5), (4, 5), (6, 5), (5, 2), (4, 3), (3, 2)] sage: ip.sub_poset(1,3) The Tamari interval of size 2 induced by relations [(1, 2)] sage: ip.sub_poset(1,4) The Tamari interval of size 3 induced by relations [(1, 2), (3, 2)] sage: ip.sub_poset(1,5) The Tamari interval of size 4 induced by relations [(1, 2), (4, 3), (3, 2)] sage: ip.sub_poset(1,7) == ip True sage: ip.sub_poset(1,1) The Tamari interval of size 0 induced by relations [] """ raise ValueError("Invalid starting or ending value, accepted: 1 <= start <= end <= size+1")
r""" Return the minimal permutation for the right weak order which is a linear extension of ``self``.
This is also the minimal permutation in the sylvester class of ``self.lower_binary_tree()`` and is a 312-avoiding permutation.
The right weak order is also known as the right permutohedron order. See :meth:`~sage.combinat.permutation.Permutation.permutohedron_lequal` for its definition.
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(1,2),(2,3),(4,3)]) sage: ip.min_linear_extension() [1, 2, 4, 3] sage: ip = TamariIntervalPoset(6,[(3,2),(4,3),(5,2),(6,5),(1,2),(4,5)]) sage: ip.min_linear_extension() [1, 4, 3, 6, 5, 2] sage: ip = TamariIntervalPoset(0,[]) sage: ip.min_linear_extension() [] sage: ip = TamariIntervalPoset(5, [(1, 4), (2, 4), (3, 4), (5, 4)]); ip The Tamari interval of size 5 induced by relations [(1, 4), (2, 4), (3, 4), (5, 4)] sage: ip.min_linear_extension() [1, 2, 3, 5, 4] """ # The min linear extension is built by postfix-reading the # final forest of ``self``. self.decreasing_cover_relations()], format='vertices_and_edges')
r""" Internal recursive method to compute the min linear extension. """
r""" Return the maximal permutation for the right weak order which is a linear extension of ``self``.
This is also the maximal permutation in the sylvester class of ``self.upper_binary_tree()`` and is a 132-avoiding permutation.
The right weak order is also known as the right permutohedron order. See :meth:`~sage.combinat.permutation.Permutation.permutohedron_lequal` for its definition.
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(1,2),(2,3),(4,3)]) sage: ip.max_linear_extension() [4, 1, 2, 3] sage: ip = TamariIntervalPoset(6,[(3,2),(4,3),(5,2),(6,5),(1,2),(4,5)]); ip The Tamari interval of size 6 induced by relations [(1, 2), (4, 5), (6, 5), (5, 2), (4, 3), (3, 2)] sage: ip.max_linear_extension() [6, 4, 5, 3, 1, 2] sage: ip = TamariIntervalPoset(0,[]); ip The Tamari interval of size 0 induced by relations [] sage: ip.max_linear_extension() [] sage: ip = TamariIntervalPoset(5, [(1, 4), (2, 4), (3, 4), (5, 4)]); ip The Tamari interval of size 5 induced by relations [(1, 4), (2, 4), (3, 4), (5, 4)] sage: ip.max_linear_extension() [5, 3, 2, 1, 4] """ # The max linear extension is built by right-to-left # postfix-reading the initial forest of ``self``. self.increasing_cover_relations()], format='vertices_and_edges')
r""" Internal recursive method to compute the max linear extension. """
r""" Return an iterator on the permutations which are linear extensions of ``self``.
They form an interval of the right weak order (also called the right permutohedron order -- see :meth:`~sage.combinat.permutation.Permutation.permutohedron_lequal` for a definition).
EXAMPLES::
sage: ip = TamariIntervalPoset(3,[(1,2),(3,2)]) sage: list(ip.linear_extensions()) [[3, 1, 2], [1, 3, 2]] sage: ip = TamariIntervalPoset(4,[(1,2),(2,3),(4,3)]) sage: list(ip.linear_extensions()) [[4, 1, 2, 3], [1, 4, 2, 3], [1, 2, 4, 3]] """
r""" If ``self`` represents the interval `[t_1, t_2]` of the Tamari lattice, return an iterator on all intervals `[t_1,t]` with `t \leq t_2` for the Tamari lattice.
In terms of interval-posets, it corresponds to adding all possible relations of the form `n` precedes `m` with `n<m`.
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(2,4),(3,4),(2,1),(3,1)]) sage: list(ip.lower_contained_intervals()) [The Tamari interval of size 4 induced by relations [(2, 4), (3, 4), (3, 1), (2, 1)], The Tamari interval of size 4 induced by relations [(1, 4), (2, 4), (3, 4), (3, 1), (2, 1)], The Tamari interval of size 4 induced by relations [(2, 3), (3, 4), (3, 1), (2, 1)], The Tamari interval of size 4 induced by relations [(1, 4), (2, 3), (3, 4), (3, 1), (2, 1)]] sage: ip = TamariIntervalPoset(4,[]) sage: len(list(ip.lower_contained_intervals())) 14 """ r""" we try to add links recursively in this order : 1 -> 2 2 -> 3 1 -> 3 3 -> 4 2 -> 4 1 -> 4 ... ("Link" means "relation of the poset".)
One useful feature of interval-posets is that if you add a single new relation -- say, `x` precedes `y` -- to an existing interval-poset and take the transitive closure, and if the axioms of an interval-poset are still satisfied for `(a,c) = (x,y)` and for `(a,c) = (y,x)`, then the transitive closure is an interval-poset (i.e., roughly speaking, the other new relations forced by `x` preceding `y` under transitive closure cannot invalidate the axioms). This is helpful when extending interval-posets, and is the reason why this and other iterators don't yield invalid interval-posets. """ r""" Internal recursive method to generate all possible intervals. At every step during the iteration, we have n < m and every i satisfying n < i < m satisfies that i precedes m in the poset ``poset`` (except when m > size). """ # if n<=0, then we go to the next m # if m>size, it's finished
# there is already a link n->m, so we go to the next n # there is an inverse link m->n, we know we won't be able # to create a link i->m with i<=n, so we go to the next m else: # there is no link n->m # first option : we don't create the link and go to the next m # (since the lack of a link n->m forbids any links i->m # with i<n) # second option : we create the link # (this is allowed because links i->m already exist for all # n<i<m, or else we wouldn't be here) # and then, we go to the next n
r""" Return the cardinality of the interval, i.e., the number of elements (binary trees or Dyck words) in the interval represented by ``self``.
Not to be confused with :meth:`size` which is the number of vertices.
.. SEEALSO:: :meth:`binary_trees`
EXAMPLES::
sage: TamariIntervalPoset(4,[(2,4),(3,4),(2,1),(3,1)]).interval_cardinality() 4 sage: TamariIntervalPoset(4,[]).interval_cardinality() 14 sage: TamariIntervalPoset(4,[(1,2),(2,3),(3,4)]).interval_cardinality() 1 """
r""" Return an iterator on all the binary trees in the interval represented by ``self``.
.. SEEALSO:: :meth:`interval_cardinality`
EXAMPLES::
sage: list(TamariIntervalPoset(4,[(2,4),(3,4),(2,1),(3,1)]).binary_trees()) [[., [[., [., .]], .]], [[., [., [., .]]], .], [., [[[., .], .], .]], [[., [[., .], .]], .]] sage: set(TamariIntervalPoset(4,[]).binary_trees()) == set(BinaryTrees(4)) True """
r""" Return an iterator on all the Dyck words in the interval represented by ``self``.
EXAMPLES::
sage: list(TamariIntervalPoset(4,[(2,4),(3,4),(2,1),(3,1)]).dyck_words()) [[1, 1, 1, 0, 0, 1, 0, 0], [1, 1, 1, 0, 0, 0, 1, 0], [1, 1, 0, 1, 0, 1, 0, 0], [1, 1, 0, 1, 0, 0, 1, 0]] sage: set(TamariIntervalPoset(4,[]).dyck_words()) == set(DyckWords(4)) True """
r""" Return an iterator on the upper contained intervals of one longest chain of the Tamari interval represented by ``self``.
If ``self`` represents the interval `[T_1,T_2]` of the Tamari lattice, this returns intervals `[T',T_2]` with `T'` following one longest chain between `T_1` and `T_2`.
To obtain a longest chain, we use the Tamari inversions of ``self``. The elements of the chain are obtained by adding one by one the relations `(b,a)` from each Tamari inversion `(a,b)` to ``self``, where the Tamari inversions are taken in lexicographic order.
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(2,4),(3,4),(2,1),(3,1)]) sage: list(ip.maximal_chain_tamari_intervals()) [The Tamari interval of size 4 induced by relations [(2, 4), (3, 4), (3, 1), (2, 1)], The Tamari interval of size 4 induced by relations [(2, 4), (3, 4), (4, 1), (3, 1), (2, 1)], The Tamari interval of size 4 induced by relations [(2, 4), (3, 4), (4, 1), (3, 2), (2, 1)]] sage: ip = TamariIntervalPoset(4,[]) sage: list(ip.maximal_chain_tamari_intervals()) [The Tamari interval of size 4 induced by relations [], The Tamari interval of size 4 induced by relations [(2, 1)], The Tamari interval of size 4 induced by relations [(3, 1), (2, 1)], The Tamari interval of size 4 induced by relations [(4, 1), (3, 1), (2, 1)], The Tamari interval of size 4 induced by relations [(4, 1), (3, 2), (2, 1)], The Tamari interval of size 4 induced by relations [(4, 2), (3, 2), (2, 1)], The Tamari interval of size 4 induced by relations [(4, 3), (3, 2), (2, 1)]] """
r""" Return an iterator on the binary trees forming a longest chain of ``self`` (regarding ``self`` as an interval of the Tamari lattice).
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(2,4),(3,4),(2,1),(3,1)]) sage: list(ip.maximal_chain_binary_trees()) [[[., [[., .], .]], .], [., [[[., .], .], .]], [., [[., [., .]], .]]] sage: ip = TamariIntervalPoset(4,[]) sage: list(ip.maximal_chain_binary_trees()) [[[[[., .], .], .], .], [[[., [., .]], .], .], [[., [[., .], .]], .], [., [[[., .], .], .]], [., [[., [., .]], .]], [., [., [[., .], .]]], [., [., [., [., .]]]]] """
r""" Return an iterator on the Dyck words forming a longest chain of ``self`` (regarding ``self`` as an interval of the Tamari lattice).
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(2,4),(3,4),(2,1),(3,1)]) sage: list(ip.maximal_chain_dyck_words()) [[1, 1, 0, 1, 0, 0, 1, 0], [1, 1, 0, 1, 0, 1, 0, 0], [1, 1, 1, 0, 0, 1, 0, 0]] sage: ip = TamariIntervalPoset(4,[]) sage: list(ip.maximal_chain_dyck_words()) [[1, 0, 1, 0, 1, 0, 1, 0], [1, 1, 0, 0, 1, 0, 1, 0], [1, 1, 0, 1, 0, 0, 1, 0], [1, 1, 0, 1, 0, 1, 0, 0], [1, 1, 1, 0, 0, 1, 0, 0], [1, 1, 1, 0, 1, 0, 0, 0], [1, 1, 1, 1, 0, 0, 0, 0]] """
r""" Return the Tamari inversions of ``self``.
A Tamari inversion is a pair of vertices `(a,b)` with `a < b` such that:
- the decreasing parent of `b` is strictly smaller than `a` (or does not exist), and - the increasing parent of `a` is strictly bigger than `b` (or does not exist).
"Smaller" and "bigger" refer to the numerical values of the elements, not to the poset order.
This method returns the list of all Tamari inversions in lexicographic order.
The number of Tamari inversions is the length of the longest chain of the Tamari interval represented by ``self``.
Indeed, when an interval consists of just one binary tree, it has no inversion. One can also prove that if a Tamari interval `I' = [T_1', T_2']` is a proper subset of a Tamari interval `I = [T_1, T_2]`, then the inversion number of `I'` is strictly lower than the inversion number of `I`. And finally, by adding the relation `(b,a)` to the interval-poset where `(a,b)` is the first inversion of `I` in lexicographic order, one reduces the inversion number by exactly `1`.
.. SEEALSO::
:meth:`tamari_inversions_iter`, :meth:`number_of_tamari_inversions`
EXAMPLES::
sage: ip = TamariIntervalPoset(3,[]) sage: ip.tamari_inversions() [(1, 2), (1, 3), (2, 3)] sage: ip = TamariIntervalPoset(3,[(2,1)]) sage: ip.tamari_inversions() [(1, 3), (2, 3)] sage: ip = TamariIntervalPoset(3,[(1,2)]) sage: ip.tamari_inversions() [(2, 3)] sage: ip = TamariIntervalPoset(3,[(1,2),(3,2)]) sage: ip.tamari_inversions() [] sage: ip = TamariIntervalPoset(4,[(2,4),(3,4),(2,1),(3,1)]) sage: ip.tamari_inversions() [(1, 4), (2, 3)] sage: ip = TamariIntervalPoset(4,[]) sage: ip.tamari_inversions() [(1, 2), (1, 3), (1, 4), (2, 3), (2, 4), (3, 4)] sage: all(len(TamariIntervalPosets.from_binary_trees(bt,bt).tamari_inversions())==0 for bt in BinaryTrees(3)) True sage: all(len(TamariIntervalPosets.from_binary_trees(bt,bt).tamari_inversions())==0 for bt in BinaryTrees(4)) True
"""
r""" Iterate over the Tamari inversions of ``self``, in lexicographic order.
See :meth:`tamari_inversions` for the definition of the terms involved.
EXAMPLES::
sage: T = TamariIntervalPoset(5, [[1,2],[3,4],[3,2],[5,2],[4,2]]) sage: list(T.tamari_inversions_iter()) [(4, 5)]
sage: T = TamariIntervalPoset(8, [(2, 7), (3, 7), (4, 7), (5, 7), (6, 7), (8, 7), (6, 4), (5, 4), (4, 3), (3, 2)]) sage: list(T.tamari_inversions_iter()) [(1, 2), (1, 7), (5, 6)]
sage: T = TamariIntervalPoset(1, []) sage: list(T.tamari_inversions_iter()) []
sage: T = TamariIntervalPoset(0, []) sage: list(T.tamari_inversions_iter()) [] """ self.decreasing_cover_relations()], format='vertices_and_edges') self.increasing_cover_relations()], format='vertices_and_edges')
r""" Return the number of Tamari inversions of ``self``.
This is also the length the longest chain of the Tamari interval represented by ``self``.
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(2,4),(3,4),(2,1),(3,1)]) sage: ip.number_of_tamari_inversions() 2 sage: ip = TamariIntervalPoset(4,[]) sage: ip.number_of_tamari_inversions() 6 sage: ip = TamariIntervalPoset(3,[]) sage: ip.number_of_tamari_inversions() 3 """
""" Return the number of terms in the decomposition in new interval-posets.
Every interval-poset has a unique decomposition as a planar tree of new interval-posets, as explained in [ChapTamari08]_. This function just computes the number of terms, not the planar tree nor the terms themselves.
.. SEEALSO:: :meth:`is_new`, :meth:`new_decomposition`
EXAMPLES::
sage: TIP4 = TamariIntervalPosets(4) sage: nb = [u.number_of_new_components() for u in TIP4] sage: [nb.count(i) for i in range(1, 5)] [12, 21, 21, 14] """
""" Return the decomposition of the interval-poset into new interval-posets.
Every interval-poset has a unique decomposition as a planar tree of new interval-posets, as explained in [ChapTamari08]_. This function computes the terms of this decomposition, but not the planar tree.
For the number of terms, you can use instead the method :meth:`number_of_new_components`.
OUTPUT:
a list of new interval-posets.
.. SEEALSO::
:meth:`number_of_new_components`, :meth:`is_new`
EXAMPLES::
sage: ex = TamariIntervalPosets(4)[11] sage: ex.number_of_new_components() 3 sage: ex.new_decomposition() [The Tamari interval of size 1 induced by relations [], The Tamari interval of size 2 induced by relations [], The Tamari interval of size 1 induced by relations []]
TESTS::
sage: ex = TamariIntervalPosets(4).random_element() sage: dec = ex.new_decomposition() sage: len(dec) == ex.number_of_new_components() True sage: all(u.is_new() for u in dec) True """
""" Extract a tree with root at position xy (recursive). """
extract_tree(cx, cy, t_up, common)) for cx, cy in common]
""" Decompose an interval-poset into a triple (``left``, ``right``, ``r``).
For the inverse method, see :meth:`TamariIntervalPosets.recomposition_from_triple`.
OUTPUT:
a triple (``left``, ``right``, ``r``) where ``left`` and ``right`` are interval-posets and ``r`` (an integer) is the parameter of the decomposition.
EXAMPLES::
sage: tip = TamariIntervalPoset(8, [(1,2), (2,4), (3,4), (6,7), (3,2), (5,4), (6,4), (8,7)]) sage: tip.decomposition_to_triple() (The Tamari interval of size 3 induced by relations [(1, 2), (3, 2)], The Tamari interval of size 4 induced by relations [(2, 3), (4, 3)], 2) sage: tip == TamariIntervalPosets.recomposition_from_triple(*tip.decomposition_to_triple()) True
REFERENCES:
- [ChP2015]_ """ return None
""" Return the grafting tree of the interval-poset.
For the inverse method, see :meth:`TamariIntervalPosets.from_grafting_tree`.
EXAMPLES::
sage: tip = TamariIntervalPoset(8, [(1,2), (2,4), (3,4), (6,7), (3,2), (5,4), (6,4), (8,7)]) sage: tip.grafting_tree() 2[1[0[., .], 0[., .]], 0[., 1[0[., .], 0[., .]]]] sage: tip == TamariIntervalPosets.from_grafting_tree(tip.grafting_tree()) True
REFERENCES:
- [Pons2018]_ """ right.grafting_tree()], label=r)
""" Return whether ``self`` is a new Tamari interval.
Here 'new' means that the interval is not contained in any facet of the associahedron. This condition is invariant under complementation.
They have been considered in section 9 of [ChapTamari08]_.
.. SEEALSO:: :meth:`is_modern`
EXAMPLES::
sage: TIP4 = TamariIntervalPosets(4) sage: len([u for u in TIP4 if u.is_new()]) 12
sage: TIP3 = TamariIntervalPosets(3) sage: len([u for u in TIP3 if u.is_new()]) 3 """
""" Return whether ``self`` is a simple Tamari interval.
Here 'simple' means that the interval contains a unique binary tree.
These intervals define the simple modules over the incidence algebras of the Tamari lattices.
.. SEEALSO:: :meth:`is_final_interval`, :meth:`is_initial_interval`
EXAMPLES::
sage: TIP4 = TamariIntervalPosets(4) sage: len([u for u in TIP4 if u.is_simple()]) 14
sage: TIP3 = TamariIntervalPosets(3) sage: len([u for u in TIP3 if u.is_simple()]) 5 """
""" Return whether ``self`` is a synchronized Tamari interval.
This means that the upper and lower binary trees have the same canopee. This condition is invariant under complementation.
This has been considered in [FPR15]_. The numbers of synchronized intervals are given by the sequence :oeis:`A000139`.
EXAMPLES::
sage: len([T for T in TamariIntervalPosets(3) ....: if T.is_synchronized()]) 6 """
r""" Return whether ``self`` is a modern Tamari interval.
This is defined by exclusion of a simple pattern in the Hasse diagram, namely there is no configuration `y \rightarrow x \leftarrow z` with `1 \leq y < x < z \leq n`.
This condition is invariant under complementation.
.. SEEALSO:: :meth:`is_new`, :meth:`is_infinitely_modern`
EXAMPLES::
sage: len([T for T in TamariIntervalPosets(3) if T.is_modern()]) 12
REFERENCES:
- [Rog2018]_ """
r""" Return whether ``self`` is an infinitely-modern Tamari interval.
This is defined by the exclusion of the configuration `i \rightarrow i + 1` and `j + 1 \rightarrow j` with `i < j`.
This condition is invariant under complementation.
.. SEEALSO:: :meth:`is_new`, :meth:`is_modern`
EXAMPLES::
sage: len([T for T in TamariIntervalPosets(3) ....: if T.is_infinitely_modern()]) 12
REFERENCES:
- [Rog2018]_ """
r""" Return whether ``self`` is an exceptional Tamari interval.
This is defined by exclusion of a simple pattern in the Hasse diagram, namely there is no configuration ``y <-- x --> z`` with `1 \leq y < x < z \leq n`.
This condition is invariant under complementation.
EXAMPLES::
sage: len([T for T in TamariIntervalPosets(3) ....: if T.is_exceptional()]) 12 """
r""" Return whether ``self`` is a dexter Tamari interval.
This is defined by an exclusion pattern in the Hasse diagram. See the code for the exact description.
This condition is not invariant under complementation.
EXAMPLES::
sage: len([T for T in TamariIntervalPosets(3) if T.is_dexter()]) 12 """
""" Return whether ``self`` is a connected Tamari interval.
This means that the Hasse diagram is connected.
EXAMPLES::
sage: len([T for T in TamariIntervalPosets(3) if T.is_connected()]) 8 """
# Abstract class to serve as a Factory ; no instances are created. r""" Factory for interval-posets.
INPUT:
- ``size`` -- (optional) an integer
OUTPUT:
- the set of all interval-posets (of the given ``size`` if specified)
EXAMPLES::
sage: TamariIntervalPosets() Interval-posets
sage: TamariIntervalPosets(2) Interval-posets of size 2
.. NOTE::
This is a factory class whose constructor returns instances of subclasses. """ r""" TESTS::
sage: from sage.combinat.interval_posets import TamariIntervalPosets_all, TamariIntervalPosets_size sage: isinstance(TamariIntervalPosets(2), TamariIntervalPosets_size) True sage: isinstance(TamariIntervalPosets(), TamariIntervalPosets_all) True sage: TamariIntervalPosets(2) is TamariIntervalPosets_size(2) True sage: TamariIntervalPosets() is TamariIntervalPosets_all() True """
raise ValueError("n must be a non negative integer")
# add options to class r""" Set and display the options for Tamari interval-posets.
If no parameters are set, then the function returns a copy of the options dictionary.
The ``options`` to Tamari interval-posets can be accessed as the method :meth:`TamariIntervalPosets.options` of :class:`TamariIntervalPosets` and related parent classes.
@OPTIONS@
EXAMPLES::
sage: TIP = TamariIntervalPosets sage: TIP.options.latex_color_decreasing red sage: TIP.options.latex_color_decreasing='green' sage: TIP.options.latex_color_decreasing green sage: TIP.options._reset() sage: TIP.options.latex_color_decreasing red """ description='the default value for the tikz scale when latexed', checker=lambda x: True) # More trouble than it's worth to check description='the default value for the line width as a' 'multiple of the tikz scale when latexed', checker=lambda x: True) # More trouble than it's worth to check description='the default color of decreasing relations when latexed', checker=lambda x: True) # More trouble than it's worth to check description='the default color of increasing relations when latexed', checker=lambda x: True) # More trouble than it's worth to check description='the default difference between horizontal' ' coordinates of vertices when latexed', checker=lambda x: True) # More trouble than it's worth to check description='the default difference between vertical' ' coordinates of vertices when latexed', checker=lambda x: True) # More trouble than it's worth to check
def check_poset(poset): r""" Check if the given poset ``poset`` is a interval-poset, that is, if it satisfies the following properties:
- Its labels are exactly `1, \ldots, n` where `n` is its size. - If `a < c` (as numbers) and `a` precedes `c`, then `b` precedes `c` for all `b` such that `a < b < c`. - If `a < c` (as numbers) and `c` precedes `a`, then `b` precedes `a` for all `b` such that `a < b < c`.
INPUT:
- ``poset`` -- a finite labeled poset
EXAMPLES::
sage: p = Poset(([1,2,3],[(1,2),(3,2)])) sage: TamariIntervalPosets.check_poset(p) True sage: p = Poset(([2,3],[(3,2)])) sage: TamariIntervalPosets.check_poset(p) False sage: p = Poset(([1,2,3],[(3,1)])) sage: TamariIntervalPosets.check_poset(p) False sage: p = Poset(([1,2,3],[(1,3)])) sage: TamariIntervalPosets.check_poset(p) False """
def final_forest(element): r""" Return the final forest of a binary tree, an interval-poset or a Dyck word.
A final forest is an interval-poset corresponding to a final interval of the Tamari lattice, i.e., containing only decreasing relations.
It can be constructed from a binary tree by its binary search tree labeling with the rule: `b` precedes `a` in the final forest iff `b` is in the right subtree of `a` in the binary search tree.
INPUT:
- ``element`` -- a binary tree, a Dyck word or an interval-poset
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(1,2),(2,3),(4,3)]) sage: TamariIntervalPosets.final_forest(ip) The Tamari interval of size 4 induced by relations [(1, 2), (2, 3)]
From binary trees::
sage: bt = BinaryTree(); bt . sage: TamariIntervalPosets.final_forest(bt) The Tamari interval of size 0 induced by relations [] sage: bt = BinaryTree([]); bt [., .] sage: TamariIntervalPosets.final_forest(bt) The Tamari interval of size 1 induced by relations [] sage: bt = BinaryTree([[],None]); bt [[., .], .] sage: TamariIntervalPosets.final_forest(bt) The Tamari interval of size 2 induced by relations [] sage: bt = BinaryTree([None,[]]); bt [., [., .]] sage: TamariIntervalPosets.final_forest(bt) The Tamari interval of size 2 induced by relations [(2, 1)] sage: bt = BinaryTree([[],[]]); bt [[., .], [., .]] sage: TamariIntervalPosets.final_forest(bt) The Tamari interval of size 3 induced by relations [(3, 2)] sage: bt = BinaryTree([[None,[[],None]],[]]); bt [[., [[., .], .]], [., .]] sage: TamariIntervalPosets.final_forest(bt) The Tamari interval of size 5 induced by relations [(5, 4), (3, 1), (2, 1)]
From Dyck words::
sage: dw = DyckWord([1,0]) sage: TamariIntervalPosets.final_forest(dw) The Tamari interval of size 1 induced by relations [] sage: dw = DyckWord([1,1,0,1,0,0,1,1,0,0]) sage: TamariIntervalPosets.final_forest(dw) The Tamari interval of size 5 induced by relations [(5, 4), (3, 1), (2, 1)] """ else: raise ValueError("Do not know how to construct the initial forest of {}".format(element))
r""" Recursive method to get the binary tree final forest relations with only one recursive reading of the tree.
The vertices are being labelled with integers starting with ``start``.
OUTPUT:
- the indexes of the nodes on the left border of the tree (these become the roots of the forest) - the relations of the final forest (as a list of tuples) - the next available index for a node (size of tree + ``start``) """
def initial_forest(element): r""" Return the initial forest of a binary tree, an interval-poset or a Dyck word.
An initial forest is an interval-poset corresponding to an initial interval of the Tamari lattice, i.e., containing only increasing relations.
It can be constructed from a binary tree by its binary search tree labeling with the rule: `a` precedes `b` in the initial forest iff `a` is in the left subtree of `b` in the binary search tree.
INPUT:
- ``element`` -- a binary tree, a Dyck word or an interval-poset
EXAMPLES::
sage: ip = TamariIntervalPoset(4,[(1,2),(2,3),(4,3)]) sage: TamariIntervalPosets.initial_forest(ip) The Tamari interval of size 4 induced by relations [(1, 2), (2, 3)]
with binary trees::
sage: bt = BinaryTree(); bt . sage: TamariIntervalPosets.initial_forest(bt) The Tamari interval of size 0 induced by relations [] sage: bt = BinaryTree([]); bt [., .] sage: TamariIntervalPosets.initial_forest(bt) The Tamari interval of size 1 induced by relations [] sage: bt = BinaryTree([[],None]); bt [[., .], .] sage: TamariIntervalPosets.initial_forest(bt) The Tamari interval of size 2 induced by relations [(1, 2)] sage: bt = BinaryTree([None,[]]); bt [., [., .]] sage: TamariIntervalPosets.initial_forest(bt) The Tamari interval of size 2 induced by relations [] sage: bt = BinaryTree([[],[]]); bt [[., .], [., .]] sage: TamariIntervalPosets.initial_forest(bt) The Tamari interval of size 3 induced by relations [(1, 2)] sage: bt = BinaryTree([[None,[[],None]],[]]); bt [[., [[., .], .]], [., .]] sage: TamariIntervalPosets.initial_forest(bt) The Tamari interval of size 5 induced by relations [(1, 4), (2, 3), (3, 4)]
from Dyck words::
sage: dw = DyckWord([1,0]) sage: TamariIntervalPosets.initial_forest(dw) The Tamari interval of size 1 induced by relations [] sage: dw = DyckWord([1,1,0,1,0,0,1,1,0,0]) sage: TamariIntervalPosets.initial_forest(dw) The Tamari interval of size 5 induced by relations [(1, 4), (2, 3), (3, 4)] """ else: raise ValueError("Do not know how to construct the initial forest of {}".format(element))
r""" Recursive method to get the binary tree initial forest relations with only one recursive reading of the tree.
The vertices are being labelled with integers starting with ``start``.
OUTPUT:
- the indexes of the nodes on the right border of the tree (these become the roots of the forest) - the relations of the initial forest (as a list of tuples) - the next available index for a node (size of tree + ``start``) """
def from_binary_trees(tree1, tree2): r""" Return the interval-poset corresponding to the interval [``tree1``, ``tree2``] of the Tamari lattice. Raise an exception if ``tree1`` is not `\leq` ``tree2`` in the Tamari lattice.
INPUT:
- ``tree1`` -- a binary tree - ``tree2`` -- a binary tree greater or equal than ``tree1`` for the Tamari lattice
EXAMPLES::
sage: tree1 = BinaryTree([[],None]) sage: tree2 = BinaryTree([None,[]]) sage: TamariIntervalPosets.from_binary_trees(tree1,tree2) The Tamari interval of size 2 induced by relations [] sage: TamariIntervalPosets.from_binary_trees(tree1,tree1) The Tamari interval of size 2 induced by relations [(1, 2)] sage: TamariIntervalPosets.from_binary_trees(tree2,tree2) The Tamari interval of size 2 induced by relations [(2, 1)]
sage: tree1 = BinaryTree([[],[[None,[]],[]]]) sage: tree2 = BinaryTree([None,[None,[None,[[],[]]]]]) sage: TamariIntervalPosets.from_binary_trees(tree1,tree2) The Tamari interval of size 6 induced by relations [(4, 5), (6, 5), (5, 2), (4, 3), (3, 2)]
sage: tree3 = BinaryTree([None,[None,[[],[None,[]]]]]) sage: TamariIntervalPosets.from_binary_trees(tree1,tree3) Traceback (most recent call last): ... ValueError: The two binary trees are not comparable on the Tamari lattice. sage: TamariIntervalPosets.from_binary_trees(tree1,BinaryTree()) Traceback (most recent call last): ... ValueError: The two binary trees are not comparable on the Tamari lattice. """
def from_dyck_words(dw1, dw2): r""" Return the interval-poset corresponding to the interval [``dw1``, ``dw2``] of the Tamari lattice. Raise an exception if the two Dyck words ``dw1`` and ``dw2`` do not satisfy ``dw1`` `\leq` ``dw2`` in the Tamari lattice.
INPUT:
- ``dw1`` -- a Dyck word - ``dw2`` -- a Dyck word greater or equal than ``dw1`` for the Tamari lattice
EXAMPLES::
sage: dw1 = DyckWord([1,0,1,0]) sage: dw2 = DyckWord([1,1,0,0]) sage: TamariIntervalPosets.from_dyck_words(dw1,dw2) The Tamari interval of size 2 induced by relations [] sage: TamariIntervalPosets.from_dyck_words(dw1,dw1) The Tamari interval of size 2 induced by relations [(1, 2)] sage: TamariIntervalPosets.from_dyck_words(dw2,dw2) The Tamari interval of size 2 induced by relations [(2, 1)]
sage: dw1 = DyckWord([1,0,1,1,1,0,0,1,1,0,0,0]) sage: dw2 = DyckWord([1,1,1,1,0,1,1,0,0,0,0,0]) sage: TamariIntervalPosets.from_dyck_words(dw1,dw2) The Tamari interval of size 6 induced by relations [(4, 5), (6, 5), (5, 2), (4, 3), (3, 2)]
sage: dw3 = DyckWord([1,1,1,0,1,1,1,0,0,0,0,0]) sage: TamariIntervalPosets.from_dyck_words(dw1,dw3) Traceback (most recent call last): ... ValueError: The two Dyck words are not comparable on the Tamari lattice. sage: TamariIntervalPosets.from_dyck_words(dw1,DyckWord([1,0])) Traceback (most recent call last): ... ValueError: The two Dyck words are not comparable on the Tamari lattice. """
def recomposition_from_triple(left, right, r): """ Recompose an interval-poset from a triple (``left``, ``right``, ``r``).
For the inverse method, see :meth:`TamariIntervalPoset.decomposition_to_triple`.
INPUT:
- ``left`` -- an interval-poset - ``right`` -- an interval-poset - ``r`` -- the parameter of the decomposition, an integer
OUTPUT: an interval-poset
EXAMPLES::
sage: T1 = TamariIntervalPoset(3, [(1, 2), (3, 2)]) sage: T2 = TamariIntervalPoset(4, [(2, 3), (4, 3)]) sage: TamariIntervalPosets.recomposition_from_triple(T1, T2, 2) The Tamari interval of size 8 induced by relations [(1, 2), (2, 4), (3, 4), (6, 7), (8, 7), (6, 4), (5, 4), (3, 2)]
REFERENCES:
- [Pons2018]_ """ for a, b in right.poset().cover_relations())
def from_grafting_tree(tree): """ Return an interval-poset from a grafting tree.
For the inverse method, see :meth:`TamariIntervalPoset.grafting_tree`.
EXAMPLES::
sage: tip = TamariIntervalPoset(8, [(1,2), (2,4), (3,4), (6,7), (3,2), (5,4), (6,4), (8,7)]) sage: t = tip.grafting_tree() sage: TamariIntervalPosets.from_grafting_tree(t) == tip True
REFERENCES:
- [Pons2018]_ """
def from_minimal_schnyder_wood(graph): """ Return a Tamari interval built from a minimal Schnyder wood.
This is an implementation of Bernardi and Bonichon's bijection [BerBon]_.
INPUT:
a minimal Schnyder wood, given as a graph with colored and oriented edges, without the three exterior unoriented edges
The three boundary vertices must be 'a', 'b' and 'c'.
One assumes moreover that the embedding around 'a' is the list of neighbors of 'a' and not just a cyclic permutation of that.
Beware that the embedding convention used here is the opposite of the one used by the plot method.
OUTPUT:
a Tamari interval-poset
EXAMPLES:
A small example::
sage: TIP = TamariIntervalPosets sage: G = DiGraph([(0,'a',0),(0,'b',1),(0,'c',2)], format='list_of_edges') sage: G.set_embedding({'a':[0],'b':[0],'c':[0],0:['a','b','c']}) sage: TIP.from_minimal_schnyder_wood(G) The Tamari interval of size 1 induced by relations []
An example from page 14 of [BerBon]_::
sage: c0 = [(0,'a'),(1,0),(2,0),(4,3),(3,'a'),(5,3)] sage: c1 = [(5,'b'),(3,'b'),(4,5),(1,3),(2,3),(0,3)] sage: c2 = [(0,'c'),(1,'c'),(3,'c'),(4,'c'),(5,'c'),(2,1)] sage: ed = [(u,v,0) for u,v in c0] sage: ed += [(u,v,1) for u,v in c1] sage: ed += [(u,v,2) for u,v in c2] sage: G = DiGraph(ed, format='list_of_edges') sage: embed = {'a':[3,0],'b':[5,3],'c':[0,1,3,4,5]} sage: data_emb = [[3,2,1,'c','a'],[2,3,'c',0],[3,1,0]] sage: data_emb += [['b',5,4,'c',1,2,0,'a'],[5,'c',3],['b','c',4,3]] sage: for k in range(6): ....: embed[k] = data_emb[k] sage: G.set_embedding(embed) sage: TIP.from_minimal_schnyder_wood(G) The Tamari interval of size 6 induced by relations [(1, 4), (2, 4), (3, 4), (5, 6), (6, 4), (5, 4), (3, 1), (2, 1)]
An example from page 18 of [BerBon]_::
sage: c0 = [(0,'a'),(1,0),(2,'a'),(3,2),(4,2),(5,'a')] sage: c1 = [(5,'b'),(2,'b'),(4,'b'),(3,4),(1,2),(0,2)] sage: c2 = [(0,'c'),(1,'c'),(3,'c'),(4,'c'),(2,'c'),(5,2)] sage: ed = [(u,v,0) for u,v in c0] sage: ed += [(u,v,1) for u,v in c1] sage: ed += [(u,v,2) for u,v in c2] sage: G = DiGraph(ed, format='list_of_edges') sage: embed = {'a':[5,2,0],'b':[4,2,5],'c':[0,1,2,3,4]} sage: data_emb = [[2,1,'c','a'],[2,'c',0],[3,'c',1,0,'a',5,'b',4]] sage: data_emb += [[4,'c',2],['b','c',3,2],['b',2,'a']] sage: for k in range(6): ....: embed[k] = data_emb[k] sage: G.set_embedding(embed) sage: TIP.from_minimal_schnyder_wood(G) The Tamari interval of size 6 induced by relations [(1, 3), (2, 3), (4, 5), (5, 3), (4, 3), (2, 1)]
Another small example::
sage: c0 = [(0,'a'),(2,'a'),(1,0)] sage: c1 = [(2,'b'),(1,'b'),(0,2)] sage: c2 = [(0,'c'),(1,'c'),(2,1)] sage: ed = [(u,v,0) for u,v in c0] sage: ed += [(u,v,1) for u,v in c1] sage: ed += [(u,v,2) for u,v in c2] sage: G = DiGraph(ed, format='list_of_edges') sage: embed = {'a':[2,0],'b':[1,2],'c':[0,1]} sage: data_emb = [[2,1,'c','a'],['c',0,2,'b'],['b',1,0,'a']] sage: for k in range(3): ....: embed[k] = data_emb[k] sage: G.set_embedding(embed) sage: TIP.from_minimal_schnyder_wood(G) The Tamari interval of size 3 induced by relations [(2, 3), (2, 1)]
REFERENCES:
.. [BerBon] Olivier Bernardi and Nicolas Bonichon, *Intervals in Catalan lattices and realizers of triangulations*, JCTA 116 (2009) """
if e[2] == color_a], format='list_of_edges') if v in graph0.neighbors_in(u) or v in graph0.neighbors_out(u)] for u in graph0}
else:
return [vertex] else:
return [] else:
# this is the profile of the planar graph graph0
if u[2] == color_b]) if u[2] == color_b])
r""" Allows for a poset to be directly transformed into an interval-poset.
It is some kind of coercion but cannot be made through the coercion system because posets do not have parents.
EXAMPLES::
sage: TIP = TamariIntervalPosets() sage: p = Poset( ([1,2,3], [(1,2)])) sage: TIP(p) The Tamari interval of size 3 induced by relations [(1, 2)] sage: TIP(TIP(p)) The Tamari interval of size 3 induced by relations [(1, 2)] sage: TIP(3,[(1,2)]) The Tamari interval of size 3 induced by relations [(1, 2)] sage: p = Poset(([1,2,3],[(1,3)])) sage: TIP(p) Traceback (most recent call last): ... ValueError: This does not satisfy the Tamari interval-poset condition. """
r""" Poset stucture on the set of interval-posets through interval containment.
Return whether the interval represented by ``el1`` is contained in the interval represented by ``el2``.
INPUT:
- ``el1`` -- an interval-poset - ``el2`` -- an interval-poset
EXAMPLES::
sage: ip1 = TamariIntervalPoset(4,[(1,2),(2,3),(4,3)]) sage: ip2 = TamariIntervalPoset(4,[(1,2),(2,3)]) sage: TamariIntervalPosets().le(ip1,ip2) True sage: TamariIntervalPosets().le(ip2,ip1) False """
################################################################# # Enumerated set of all Tamari Interval-posets #################################################################
r""" The enumerated set of all Tamari interval-posets. """ r""" TESTS::
sage: from sage.combinat.interval_posets import TamariIntervalPosets_all sage: S = TamariIntervalPosets_all() sage: S.cardinality() +Infinity
sage: it = iter(S) sage: [next(it) for i in range(5)] [The Tamari interval of size 0 induced by relations [], The Tamari interval of size 1 induced by relations [], The Tamari interval of size 2 induced by relations [], The Tamari interval of size 2 induced by relations [(2, 1)], The Tamari interval of size 2 induced by relations [(1, 2)]] sage: next(it).parent() Interval-posets sage: S(0,[]) The Tamari interval of size 0 induced by relations []
sage: S is TamariIntervalPosets_all() True sage: TestSuite(S).run() # long time (7s) """ self, Family(NonNegativeIntegers(), TamariIntervalPosets_size), facade=True, keepkey=False, category=(Posets(), EnumeratedSets()))
r""" TESTS::
sage: TamariIntervalPosets() Interval-posets """
r""" EXAMPLES::
sage: TIP = TamariIntervalPosets() sage: TIP(3,[(1,2)]) The Tamari interval of size 3 induced by relations [(1, 2)] """
r""" TESTS::
sage: S = TamariIntervalPosets() sage: 1 in S False sage: S(0,[]) in S True """
################################################################# # Enumerated set of Tamari interval-posets of a given size ################################################################# r""" The enumerated set of interval-posets of a given size. """ r""" TESTS::
sage: S = TamariIntervalPosets(3) sage: assert S is TamariIntervalPosets(3) sage: for i in range(5): TestSuite(TamariIntervalPosets(i)).run() """ # there is a natural order on interval-posets through inclusions # that is why we use the FinitePosets category
r""" TESTS::
sage: TamariIntervalPosets(3) Interval-posets of size 3 """
r""" TESTS::
sage: S = TamariIntervalPosets(3) sage: 1 in S False sage: S([]) in S True """
r""" The cardinality of ``self``. That is, the number of interval-posets of size `n`.
The formula was given in [ChapTamari08]_:
.. MATH::
\frac{2(4n+1)!}{(n+1)!(3n+2)!} = \frac{2}{n(n+1)} \binom{4n+1}{n-1}.
EXAMPLES::
sage: [TamariIntervalPosets(i).cardinality() for i in range(6)] [1, 1, 3, 13, 68, 399] """ # return Integer(2 * factorial(4*n+1)/(factorial(n+1)*factorial(3*n+2)))
r""" Recursive generation: we iterate through all interval-posets of size ``size - 1`` and add all possible relations to the last vertex.
This works thanks to the fact that the restriction of an interval-poset of size `n` to the subset `\{1, 2, \ldots, k\}` for a fixed `k \leq n` is an interval-poset.
TESTS::
sage: TIP1 = TamariIntervalPosets(1) sage: list(TIP1) [The Tamari interval of size 1 induced by relations []] sage: TIP2 = TamariIntervalPosets(2) sage: list(TIP2) [The Tamari interval of size 2 induced by relations [], The Tamari interval of size 2 induced by relations [(2, 1)], The Tamari interval of size 2 induced by relations [(1, 2)]] sage: TIP3 = TamariIntervalPosets(3) sage: list(TIP3) [The Tamari interval of size 3 induced by relations [], The Tamari interval of size 3 induced by relations [(3, 2)], The Tamari interval of size 3 induced by relations [(2, 3)], The Tamari interval of size 3 induced by relations [(1, 3), (2, 3)], The Tamari interval of size 3 induced by relations [(2, 1)], The Tamari interval of size 3 induced by relations [(3, 2), (2, 1)], The Tamari interval of size 3 induced by relations [(3, 1), (2, 1)], The Tamari interval of size 3 induced by relations [(2, 3), (2, 1)], The Tamari interval of size 3 induced by relations [(2, 3), (3, 1), (2, 1)], The Tamari interval of size 3 induced by relations [(1, 3), (2, 3), (2, 1)], The Tamari interval of size 3 induced by relations [(1, 2)], The Tamari interval of size 3 induced by relations [(1, 2), (3, 2)], The Tamari interval of size 3 induced by relations [(1, 2), (2, 3)]] sage: all(len(list(TamariIntervalPosets(i)))==TamariIntervalPosets(i).cardinality() for i in range(6)) True """
# adding a decreasing relation n>>m2 with m2<n and no # increasing relations
# adding an increasing relation m>>n else: continue
# further adding a decreasing relation n>>m2 with m2<m
""" Return a random Tamari interval of fixed size.
This is obtained by first creating a random rooted planar triangulation, then computing its unique minimal Schnyder wood, then applying a bijection of Bernardi and Bonichon [BerBon]_.
Because the random rooted planar triangulation is chosen uniformly at random, the Tamari interval is also chosen according to the uniform distribution.
EXAMPLES::
sage: T = TamariIntervalPosets(4).random_element() sage: T.parent() Interval-posets sage: u = T.lower_dyck_word(); u # random [1, 1, 0, 1, 0, 0, 1, 0] sage: v = T.lower_dyck_word(); v # random [1, 1, 0, 1, 0, 0, 1, 0] sage: len(u) 8 """ check=False)
def _parent_for(self): r""" The parent of the element generated by ``self``.
TESTS::
sage: TIP3 = TamariIntervalPosets(3) sage: TIP3._parent_for Interval-posets """
# This is needed because this is a facade parent via DisjointUnionEnumeratedSets def element_class(self): r""" TESTS::
sage: S = TamariIntervalPosets(3) sage: S.element_class <class 'sage.combinat.interval_posets.TamariIntervalPosets_all_with_category.element_class'> sage: S.first().__class__ == TamariIntervalPosets().first().__class__ True """
r""" EXAMPLES::
sage: TIP3 = TamariIntervalPosets(3) sage: TIP3([(1,2)]) The Tamari interval of size 3 induced by relations [(1, 2)] sage: TIP3([(3,4)]) Traceback (most recent call last): ... ValueError: The relations do not correspond to the size of the poset. """ |