| author | blanchet | 
| Sun, 15 Sep 2013 23:02:23 +0200 | |
| changeset 53646 | ac6e0a28489f | 
| parent 52900 | d29bf6db8a2d | 
| child 55618 | 995162143ef4 | 
| permissions | -rw-r--r-- | 
| 38479 
e628da370072
more efficient Markup_Tree, based on branches sorted by quasi-order;
 wenzelm parents: 
38478diff
changeset | 1 | /* Title: Pure/PIDE/markup_tree.scala | 
| 45673 
cd41e3903fbf
separate compilation of PIDE vs. Pure sources, which enables independent Scala library;
 wenzelm parents: 
45667diff
changeset | 2 | Module: PIDE | 
| 36676 | 3 | Author: Fabian Immler, TU Munich | 
| 4 | Author: Makarius | |
| 5 | ||
| 38479 
e628da370072
more efficient Markup_Tree, based on branches sorted by quasi-order;
 wenzelm parents: 
38478diff
changeset | 6 | Markup trees over nested / non-overlapping text ranges. | 
| 36676 | 7 | */ | 
| 34393 | 8 | |
| 34871 
e596a0b71f3c
incorporate "proofdocument" part into main Isabelle/Pure.jar -- except for html_panel.scala, which depends on external library (Lobo/Cobra browser);
 wenzelm parents: 
34760diff
changeset | 9 | package isabelle | 
| 34393 | 10 | |
| 34701 
80b0add08eef
IsabelleSideKickParser: incorporate former MarkupNode.markup2default_node, observe stopped flag;
 wenzelm parents: 
34676diff
changeset | 11 | |
| 38479 
e628da370072
more efficient Markup_Tree, based on branches sorted by quasi-order;
 wenzelm parents: 
38478diff
changeset | 12 | import scala.collection.immutable.SortedMap | 
| 49614 | 13 | import scala.collection.mutable | 
| 49467 | 14 | import scala.annotation.tailrec | 
| 38479 
e628da370072
more efficient Markup_Tree, based on branches sorted by quasi-order;
 wenzelm parents: 
38478diff
changeset | 15 | |
| 
e628da370072
more efficient Markup_Tree, based on branches sorted by quasi-order;
 wenzelm parents: 
38478diff
changeset | 16 | |
| 
e628da370072
more efficient Markup_Tree, based on branches sorted by quasi-order;
 wenzelm parents: 
38478diff
changeset | 17 | object Markup_Tree | 
| 
e628da370072
more efficient Markup_Tree, based on branches sorted by quasi-order;
 wenzelm parents: 
38478diff
changeset | 18 | {
 | 
| 50551 | 19 | /* construct trees */ | 
| 20 | ||
| 45456 | 21 | val empty: Markup_Tree = new Markup_Tree(Branches.empty) | 
| 38479 
e628da370072
more efficient Markup_Tree, based on branches sorted by quasi-order;
 wenzelm parents: 
38478diff
changeset | 22 | |
| 49467 | 23 | def merge_disjoint(trees: List[Markup_Tree]): Markup_Tree = | 
| 24 |     trees match {
 | |
| 25 | case Nil => empty | |
| 26 | case head :: tail => | |
| 27 | new Markup_Tree( | |
| 28 |           (head.branches /: tail) {
 | |
| 29 | case (branches, tree) => | |
| 30 |               (branches /: tree.branches) {
 | |
| 31 | case (bs, (r, entry)) => | |
| 32 | require(!bs.isDefinedAt(r)) | |
| 33 | bs + (r -> entry) | |
| 34 | } | |
| 35 | }) | |
| 36 | } | |
| 37 | ||
| 50551 | 38 | |
| 39 | /* tree building blocks */ | |
| 40 | ||
| 41 | object Elements | |
| 42 |   {
 | |
| 43 | val empty = new Elements(Set.empty) | |
| 44 | } | |
| 45 | ||
| 46 | final class Elements private(private val rep: Set[String]) | |
| 47 |   {
 | |
| 48 | def contains(name: String): Boolean = rep.contains(name) | |
| 49 | ||
| 50 | def + (name: String): Elements = | |
| 51 | if (contains(name)) this | |
| 52 | else new Elements(rep + name) | |
| 53 | ||
| 54 | def + (elem: XML.Elem): Elements = this + elem.markup.name | |
| 55 | def ++ (elems: Iterable[XML.Elem]): Elements = (this /: elems.iterator)(_ + _) | |
| 56 | ||
| 57 | def ++ (other: Elements): Elements = | |
| 58 | if (this eq other) this | |
| 59 | else if (rep.isEmpty) other | |
| 60 | else (this /: other.rep)(_ + _) | |
| 61 | } | |
| 62 | ||
| 45473 | 63 | object Entry | 
| 64 |   {
 | |
| 65 | def apply(markup: Text.Markup, subtree: Markup_Tree): Entry = | |
| 50552 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 66 | Entry(markup.range, List(markup.info), Elements.empty + markup.info, | 
| 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 67 | subtree, subtree.make_elements) | 
| 49469 | 68 | |
| 69 | def apply(range: Text.Range, rev_markups: List[XML.Elem], subtree: Markup_Tree): Entry = | |
| 50552 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 70 | Entry(range, rev_markups, Elements.empty ++ rev_markups, | 
| 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 71 | subtree, subtree.make_elements) | 
| 45473 | 72 | } | 
| 73 | ||
| 45474 
f793dd5d84b2
index markup elements for more efficient cumulate/select operations;
 wenzelm parents: 
45473diff
changeset | 74 | sealed case class Entry( | 
| 
f793dd5d84b2
index markup elements for more efficient cumulate/select operations;
 wenzelm parents: 
45473diff
changeset | 75 | range: Text.Range, | 
| 
f793dd5d84b2
index markup elements for more efficient cumulate/select operations;
 wenzelm parents: 
45473diff
changeset | 76 | rev_markup: List[XML.Elem], | 
| 50551 | 77 | elements: Elements, | 
| 50552 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 78 | subtree: Markup_Tree, | 
| 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 79 | subtree_elements: Elements) | 
| 45469 
25306d92f4ad
refined Markup_Tree implementation: stacked markup within each entry;
 wenzelm parents: 
45468diff
changeset | 80 |   {
 | 
| 50552 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 81 | def markup: List[XML.Elem] = rev_markup.reverse | 
| 45474 
f793dd5d84b2
index markup elements for more efficient cumulate/select operations;
 wenzelm parents: 
45473diff
changeset | 82 | |
| 50552 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 83 | def + (markup: Text.Markup): Entry = | 
| 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 84 | copy(rev_markup = markup.info :: rev_markup, elements = elements + markup.info) | 
| 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 85 | |
| 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 86 | def \ (markup: Text.Markup): Entry = | 
| 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 87 | copy(subtree = subtree + markup, subtree_elements = subtree_elements + markup.info) | 
| 45469 
25306d92f4ad
refined Markup_Tree implementation: stacked markup within each entry;
 wenzelm parents: 
45468diff
changeset | 88 | } | 
| 
25306d92f4ad
refined Markup_Tree implementation: stacked markup within each entry;
 wenzelm parents: 
45468diff
changeset | 89 | |
| 38479 
e628da370072
more efficient Markup_Tree, based on branches sorted by quasi-order;
 wenzelm parents: 
38478diff
changeset | 90 | object Branches | 
| 
e628da370072
more efficient Markup_Tree, based on branches sorted by quasi-order;
 wenzelm parents: 
38478diff
changeset | 91 |   {
 | 
| 38578 | 92 | type T = SortedMap[Text.Range, Entry] | 
| 45456 | 93 | val empty: T = SortedMap.empty(Text.Range.Ordering) | 
| 38479 
e628da370072
more efficient Markup_Tree, based on branches sorted by quasi-order;
 wenzelm parents: 
38478diff
changeset | 94 | } | 
| 49466 | 95 | |
| 96 | ||
| 97 | /* XML representation */ | |
| 98 | ||
| 49650 
9fad6480300d
support for wrapped XML elements, which allows to preserve full markup tree information in to_XML/from_XML conversion;
 wenzelm parents: 
49614diff
changeset | 99 | @tailrec private def strip_elems( | 
| 
9fad6480300d
support for wrapped XML elements, which allows to preserve full markup tree information in to_XML/from_XML conversion;
 wenzelm parents: 
49614diff
changeset | 100 | elems: List[XML.Elem], body: XML.Body): (List[XML.Elem], XML.Body) = | 
| 49467 | 101 |     body match {
 | 
| 49650 
9fad6480300d
support for wrapped XML elements, which allows to preserve full markup tree information in to_XML/from_XML conversion;
 wenzelm parents: 
49614diff
changeset | 102 | case List(XML.Wrapped_Elem(markup1, body1, body2)) => | 
| 
9fad6480300d
support for wrapped XML elements, which allows to preserve full markup tree information in to_XML/from_XML conversion;
 wenzelm parents: 
49614diff
changeset | 103 | strip_elems(XML.Elem(markup1, body1) :: elems, body2) | 
| 
9fad6480300d
support for wrapped XML elements, which allows to preserve full markup tree information in to_XML/from_XML conversion;
 wenzelm parents: 
49614diff
changeset | 104 | case List(XML.Elem(markup1, body1)) => | 
| 
9fad6480300d
support for wrapped XML elements, which allows to preserve full markup tree information in to_XML/from_XML conversion;
 wenzelm parents: 
49614diff
changeset | 105 | strip_elems(XML.Elem(markup1, Nil) :: elems, body1) | 
| 
9fad6480300d
support for wrapped XML elements, which allows to preserve full markup tree information in to_XML/from_XML conversion;
 wenzelm parents: 
49614diff
changeset | 106 | case _ => (elems, body) | 
| 49467 | 107 | } | 
| 108 | ||
| 109 | private def make_trees(acc: (Int, List[Markup_Tree]), tree: XML.Tree): (Int, List[Markup_Tree]) = | |
| 110 |     {
 | |
| 111 | val (offset, markup_trees) = acc | |
| 112 | ||
| 113 |       strip_elems(Nil, List(tree)) match {
 | |
| 114 | case (Nil, body) => | |
| 115 | (offset + XML.text_length(body), markup_trees) | |
| 49466 | 116 | |
| 49469 | 117 | case (elems, body) => | 
| 118 | val (end_offset, subtrees) = ((offset, Nil: List[Markup_Tree]) /: body)(make_trees) | |
| 50642 
aca12f646772
ignore markup elements over empty body, which are not well-defined within markup tree and fail to work with merge_disjoint (e.g. multiple inlined positions);
 wenzelm parents: 
50552diff
changeset | 119 | if (offset == end_offset) acc | 
| 
aca12f646772
ignore markup elements over empty body, which are not well-defined within markup tree and fail to work with merge_disjoint (e.g. multiple inlined positions);
 wenzelm parents: 
50552diff
changeset | 120 |           else {
 | 
| 
aca12f646772
ignore markup elements over empty body, which are not well-defined within markup tree and fail to work with merge_disjoint (e.g. multiple inlined positions);
 wenzelm parents: 
50552diff
changeset | 121 | val range = Text.Range(offset, end_offset) | 
| 
aca12f646772
ignore markup elements over empty body, which are not well-defined within markup tree and fail to work with merge_disjoint (e.g. multiple inlined positions);
 wenzelm parents: 
50552diff
changeset | 122 | val entry = Entry(range, elems, merge_disjoint(subtrees)) | 
| 
aca12f646772
ignore markup elements over empty body, which are not well-defined within markup tree and fail to work with merge_disjoint (e.g. multiple inlined positions);
 wenzelm parents: 
50552diff
changeset | 123 | (end_offset, new Markup_Tree(Branches.empty, entry) :: markup_trees) | 
| 
aca12f646772
ignore markup elements over empty body, which are not well-defined within markup tree and fail to work with merge_disjoint (e.g. multiple inlined positions);
 wenzelm parents: 
50552diff
changeset | 124 | } | 
| 49466 | 125 | } | 
| 49467 | 126 | } | 
| 49466 | 127 | |
| 49467 | 128 | def from_XML(body: XML.Body): Markup_Tree = | 
| 129 | merge_disjoint(((0, Nil: List[Markup_Tree]) /: body)(make_trees)._2) | |
| 38479 
e628da370072
more efficient Markup_Tree, based on branches sorted by quasi-order;
 wenzelm parents: 
38478diff
changeset | 130 | } | 
| 34554 
7dc6c231da40
abs. stops, markup nodes depend on doc-version;
 immler@in.tum.de parents: 
34517diff
changeset | 131 | |
| 34393 | 132 | |
| 51618 
a3577cd80c41
tuned signature -- avoid intrusion of slightly odd Swing structures into pure Markup_Tree;
 wenzelm parents: 
50642diff
changeset | 133 | final class Markup_Tree private(val branches: Markup_Tree.Branches.T) | 
| 34717 
3f32e08bbb6c
sidekick root data: set buffer length to avoid crash of initial caret move;
 wenzelm parents: 
34708diff
changeset | 134 | {
 | 
| 49417 | 135 | import Markup_Tree._ | 
| 136 | ||
| 45469 
25306d92f4ad
refined Markup_Tree implementation: stacked markup within each entry;
 wenzelm parents: 
45468diff
changeset | 137 | private def this(branches: Markup_Tree.Branches.T, entry: Markup_Tree.Entry) = | 
| 
25306d92f4ad
refined Markup_Tree implementation: stacked markup within each entry;
 wenzelm parents: 
45468diff
changeset | 138 | this(branches + (entry.range -> entry)) | 
| 
25306d92f4ad
refined Markup_Tree implementation: stacked markup within each entry;
 wenzelm parents: 
45468diff
changeset | 139 | |
| 38563 | 140 | override def toString = | 
| 141 |     branches.toList.map(_._2) match {
 | |
| 142 | case Nil => "Empty" | |
| 143 |       case list => list.mkString("Tree(", ",", ")")
 | |
| 144 | } | |
| 145 | ||
| 45469 
25306d92f4ad
refined Markup_Tree implementation: stacked markup within each entry;
 wenzelm parents: 
45468diff
changeset | 146 | private def overlapping(range: Text.Range): Branches.T = | 
| 45457 | 147 |   {
 | 
| 148 | val start = Text.Range(range.start) | |
| 149 | val stop = Text.Range(range.stop) | |
| 150 | val bs = branches.range(start, stop) | |
| 151 |     branches.get(stop) match {
 | |
| 45469 
25306d92f4ad
refined Markup_Tree implementation: stacked markup within each entry;
 wenzelm parents: 
45468diff
changeset | 152 | case Some(end) if range overlaps end.range => bs + (end.range -> end) | 
| 45457 | 153 | case _ => bs | 
| 154 | } | |
| 155 | } | |
| 156 | ||
| 50552 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 157 | def make_elements: Elements = | 
| 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 158 | (Elements.empty /: branches)( | 
| 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 159 |       { case (elements, (_, entry)) => elements ++ entry.subtree_elements ++ entry.elements })
 | 
| 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 160 | |
| 45469 
25306d92f4ad
refined Markup_Tree implementation: stacked markup within each entry;
 wenzelm parents: 
45468diff
changeset | 161 | def + (new_markup: Text.Markup): Markup_Tree = | 
| 34703 | 162 |   {
 | 
| 45469 
25306d92f4ad
refined Markup_Tree implementation: stacked markup within each entry;
 wenzelm parents: 
45468diff
changeset | 163 | val new_range = new_markup.range | 
| 
25306d92f4ad
refined Markup_Tree implementation: stacked markup within each entry;
 wenzelm parents: 
45468diff
changeset | 164 | |
| 38578 | 165 |     branches.get(new_range) match {
 | 
| 45473 | 166 | case None => new Markup_Tree(branches, Entry(new_markup, empty)) | 
| 45469 
25306d92f4ad
refined Markup_Tree implementation: stacked markup within each entry;
 wenzelm parents: 
45468diff
changeset | 167 | case Some(entry) => | 
| 
25306d92f4ad
refined Markup_Tree implementation: stacked markup within each entry;
 wenzelm parents: 
45468diff
changeset | 168 | if (entry.range == new_range) | 
| 50552 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 169 | new Markup_Tree(branches, entry + new_markup) | 
| 45469 
25306d92f4ad
refined Markup_Tree implementation: stacked markup within each entry;
 wenzelm parents: 
45468diff
changeset | 170 | else if (entry.range.contains(new_range)) | 
| 50552 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 171 | new Markup_Tree(branches, entry \ new_markup) | 
| 38578 | 172 | else if (new_range.contains(branches.head._1) && new_range.contains(branches.last._1)) | 
| 45473 | 173 | new Markup_Tree(Branches.empty, Entry(new_markup, this)) | 
| 38482 
7b6ee937b75f
tuned Markup_Tree, using SortedMap more carefully;
 wenzelm parents: 
38479diff
changeset | 174 |         else {
 | 
| 45457 | 175 | val body = overlapping(new_range) | 
| 49607 
b610c0d52bfa
updated to consolidated SortedMap in scala-2.9.x;
 wenzelm parents: 
49469diff
changeset | 176 | if (body.forall(e => new_range.contains(e._1))) | 
| 
b610c0d52bfa
updated to consolidated SortedMap in scala-2.9.x;
 wenzelm parents: 
49469diff
changeset | 177 | new Markup_Tree(branches -- body.keys, Entry(new_markup, new Markup_Tree(body))) | 
| 49608 | 178 |           else {
 | 
| 179 |             java.lang.System.err.println("Ignored overlapping markup information: " + new_markup +
 | |
| 48762 | 180 |               body.filter(e => !new_range.contains(e._1)).mkString("\n"))
 | 
| 38482 
7b6ee937b75f
tuned Markup_Tree, using SortedMap more carefully;
 wenzelm parents: 
38479diff
changeset | 181 | this | 
| 
7b6ee937b75f
tuned Markup_Tree, using SortedMap more carefully;
 wenzelm parents: 
38479diff
changeset | 182 | } | 
| 
7b6ee937b75f
tuned Markup_Tree, using SortedMap more carefully;
 wenzelm parents: 
38479diff
changeset | 183 | } | 
| 34703 | 184 | } | 
| 185 | } | |
| 186 | ||
| 52642 
84eb792224a8
full merge of Command.State, which enables Command.prints to augment markup as well (assuming that these dynamic overlays are relatively few);
 wenzelm parents: 
51618diff
changeset | 187 | def ++ (other: Markup_Tree): Markup_Tree = | 
| 
84eb792224a8
full merge of Command.State, which enables Command.prints to augment markup as well (assuming that these dynamic overlays are relatively few);
 wenzelm parents: 
51618diff
changeset | 188 |     (this /: other.branches)({ case (tree, (range, entry)) =>
 | 
| 
84eb792224a8
full merge of Command.State, which enables Command.prints to augment markup as well (assuming that these dynamic overlays are relatively few);
 wenzelm parents: 
51618diff
changeset | 189 |       ((tree ++ entry.subtree) /: entry.markup)({ case (t, elem) => t + Text.Info(range, elem) }) })
 | 
| 
84eb792224a8
full merge of Command.State, which enables Command.prints to augment markup as well (assuming that these dynamic overlays are relatively few);
 wenzelm parents: 
51618diff
changeset | 190 | |
| 49614 | 191 | def to_XML(root_range: Text.Range, text: CharSequence, filter: XML.Elem => Boolean): XML.Body = | 
| 192 |   {
 | |
| 193 | def make_text(start: Text.Offset, stop: Text.Offset): XML.Body = | |
| 194 | if (start == stop) Nil | |
| 195 | else List(XML.Text(text.subSequence(start, stop).toString)) | |
| 196 | ||
| 197 | def make_elems(rev_markups: List[XML.Elem], body: XML.Body): XML.Body = | |
| 198 |       (body /: rev_markups) {
 | |
| 49650 
9fad6480300d
support for wrapped XML elements, which allows to preserve full markup tree information in to_XML/from_XML conversion;
 wenzelm parents: 
49614diff
changeset | 199 | case (b, elem) => | 
| 
9fad6480300d
support for wrapped XML elements, which allows to preserve full markup tree information in to_XML/from_XML conversion;
 wenzelm parents: 
49614diff
changeset | 200 | if (!filter(elem)) b | 
| 
9fad6480300d
support for wrapped XML elements, which allows to preserve full markup tree information in to_XML/from_XML conversion;
 wenzelm parents: 
49614diff
changeset | 201 | else if (elem.body.isEmpty) List(XML.Elem(elem.markup, b)) | 
| 
9fad6480300d
support for wrapped XML elements, which allows to preserve full markup tree information in to_XML/from_XML conversion;
 wenzelm parents: 
49614diff
changeset | 202 | else List(XML.Wrapped_Elem(elem.markup, elem.body, b)) | 
| 49614 | 203 | } | 
| 204 | ||
| 205 | def make_body(elem_range: Text.Range, elem_markup: List[XML.Elem], entries: Branches.T) | |
| 206 | : XML.Body = | |
| 207 |     {
 | |
| 208 | val body = new mutable.ListBuffer[XML.Tree] | |
| 209 | var last = elem_range.start | |
| 210 |       for ((range, entry) <- entries) {
 | |
| 211 | val subrange = range.restrict(elem_range) | |
| 212 | body ++= make_text(last, subrange.start) | |
| 213 | body ++= make_body(subrange, entry.rev_markup, entry.subtree.overlapping(subrange)) | |
| 214 | last = subrange.stop | |
| 215 | } | |
| 216 | body ++= make_text(last, elem_range.stop) | |
| 217 | make_elems(elem_markup, body.toList) | |
| 218 | } | |
| 219 | make_body(root_range, Nil, overlapping(root_range)) | |
| 220 | } | |
| 221 | ||
| 222 | def to_XML(text: CharSequence): XML.Body = | |
| 223 | to_XML(Text.Range(0, text.length), text, (_: XML.Elem) => true) | |
| 224 | ||
| 46178 
1c5c88f6feb5
clarified Isabelle_Rendering vs. physical painting;
 wenzelm parents: 
46165diff
changeset | 225 | def cumulate[A](root_range: Text.Range, root_info: A, result_elements: Option[Set[String]], | 
| 52900 
d29bf6db8a2d
more elementary list structures for markup tree traversal;
 wenzelm parents: 
52889diff
changeset | 226 | result: (A, Text.Markup) => Option[A]): List[Text.Info[A]] = | 
| 45459 | 227 |   {
 | 
| 50551 | 228 | val notable: Elements => Boolean = | 
| 229 |       result_elements match {
 | |
| 230 | case Some(res) => (elements: Elements) => res.exists(elements.contains) | |
| 231 | case None => (elements: Elements) => true | |
| 232 | } | |
| 233 | ||
| 46178 
1c5c88f6feb5
clarified Isabelle_Rendering vs. physical painting;
 wenzelm parents: 
46165diff
changeset | 234 | def results(x: A, entry: Entry): Option[A] = | 
| 50552 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 235 |     {
 | 
| 52889 
ea3338812e67
more tight interface for markup cumulate/select: avoid duplicate application, allow to defer decision about definedness;
 wenzelm parents: 
52642diff
changeset | 236 | var y = x | 
| 
ea3338812e67
more tight interface for markup cumulate/select: avoid duplicate application, allow to defer decision about definedness;
 wenzelm parents: 
52642diff
changeset | 237 | var changed = false | 
| 
ea3338812e67
more tight interface for markup cumulate/select: avoid duplicate application, allow to defer decision about definedness;
 wenzelm parents: 
52642diff
changeset | 238 |       for {
 | 
| 
ea3338812e67
more tight interface for markup cumulate/select: avoid duplicate application, allow to defer decision about definedness;
 wenzelm parents: 
52642diff
changeset | 239 | info <- entry.rev_markup // FIXME proper cumulation order (including status markup) (!?) | 
| 
ea3338812e67
more tight interface for markup cumulate/select: avoid duplicate application, allow to defer decision about definedness;
 wenzelm parents: 
52642diff
changeset | 240 | y1 <- result(y, Text.Info(entry.range, info)) | 
| 
ea3338812e67
more tight interface for markup cumulate/select: avoid duplicate application, allow to defer decision about definedness;
 wenzelm parents: 
52642diff
changeset | 241 |       } { y = y1; changed = true }
 | 
| 50552 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 242 | if (changed) Some(y) else None | 
| 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 243 | } | 
| 45467 | 244 | |
| 52900 
d29bf6db8a2d
more elementary list structures for markup tree traversal;
 wenzelm parents: 
52889diff
changeset | 245 | def traverse( | 
| 45459 | 246 | last: Text.Offset, | 
| 52900 
d29bf6db8a2d
more elementary list structures for markup tree traversal;
 wenzelm parents: 
52889diff
changeset | 247 | stack: List[(Text.Info[A], List[(Text.Range, Entry)])]): List[Text.Info[A]] = | 
| 45459 | 248 |     {
 | 
| 249 |       stack match {
 | |
| 52900 
d29bf6db8a2d
more elementary list structures for markup tree traversal;
 wenzelm parents: 
52889diff
changeset | 250 | case (parent, (range, entry) :: more) :: rest => | 
| 45467 | 251 | val subrange = range.restrict(root_range) | 
| 50552 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 252 | val subtree = | 
| 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 253 | if (notable(entry.subtree_elements)) | 
| 52900 
d29bf6db8a2d
more elementary list structures for markup tree traversal;
 wenzelm parents: 
52889diff
changeset | 254 | entry.subtree.overlapping(subrange).toList | 
| 
d29bf6db8a2d
more elementary list structures for markup tree traversal;
 wenzelm parents: 
52889diff
changeset | 255 | else Nil | 
| 45459 | 256 | val start = subrange.start | 
| 257 | ||
| 50552 
2b7fd8c9c4ac
maintain subtree_elements for improved performance of cumulate operator;
 wenzelm parents: 
50551diff
changeset | 258 |           (if (notable(entry.elements)) results(parent.info, entry) else None) match {
 | 
| 45469 
25306d92f4ad
refined Markup_Tree implementation: stacked markup within each entry;
 wenzelm parents: 
45468diff
changeset | 259 | case Some(res) => | 
| 
25306d92f4ad
refined Markup_Tree implementation: stacked markup within each entry;
 wenzelm parents: 
45468diff
changeset | 260 | val next = Text.Info(subrange, res) | 
| 52900 
d29bf6db8a2d
more elementary list structures for markup tree traversal;
 wenzelm parents: 
52889diff
changeset | 261 | val nexts = traverse(start, (next, subtree) :: (parent, more) :: rest) | 
| 
d29bf6db8a2d
more elementary list structures for markup tree traversal;
 wenzelm parents: 
52889diff
changeset | 262 | if (last < start) parent.restrict(Text.Range(last, start)) :: nexts | 
| 45469 
25306d92f4ad
refined Markup_Tree implementation: stacked markup within each entry;
 wenzelm parents: 
45468diff
changeset | 263 | else nexts | 
| 52900 
d29bf6db8a2d
more elementary list structures for markup tree traversal;
 wenzelm parents: 
52889diff
changeset | 264 | case None => traverse(last, (parent, subtree ::: more) :: rest) | 
| 45459 | 265 | } | 
| 266 | ||
| 52900 
d29bf6db8a2d
more elementary list structures for markup tree traversal;
 wenzelm parents: 
52889diff
changeset | 267 | case (parent, Nil) :: rest => | 
| 45459 | 268 | val stop = parent.range.stop | 
| 52900 
d29bf6db8a2d
more elementary list structures for markup tree traversal;
 wenzelm parents: 
52889diff
changeset | 269 | val nexts = traverse(stop, rest) | 
| 
d29bf6db8a2d
more elementary list structures for markup tree traversal;
 wenzelm parents: 
52889diff
changeset | 270 | if (last < stop) parent.restrict(Text.Range(last, stop)) :: nexts | 
| 45459 | 271 | else nexts | 
| 272 | ||
| 273 | case Nil => | |
| 45467 | 274 | val stop = root_range.stop | 
| 52900 
d29bf6db8a2d
more elementary list structures for markup tree traversal;
 wenzelm parents: 
52889diff
changeset | 275 | if (last < stop) List(Text.Info(Text.Range(last, stop), root_info)) | 
| 
d29bf6db8a2d
more elementary list structures for markup tree traversal;
 wenzelm parents: 
52889diff
changeset | 276 | else Nil | 
| 45459 | 277 | } | 
| 278 | } | |
| 52900 
d29bf6db8a2d
more elementary list structures for markup tree traversal;
 wenzelm parents: 
52889diff
changeset | 279 | traverse(root_range.start, | 
| 
d29bf6db8a2d
more elementary list structures for markup tree traversal;
 wenzelm parents: 
52889diff
changeset | 280 | List((Text.Info(root_range, root_info), overlapping(root_range).toList))) | 
| 45459 | 281 | } | 
| 34717 
3f32e08bbb6c
sidekick root data: set buffer length to avoid crash of initial caret move;
 wenzelm parents: 
34708diff
changeset | 282 | } | 
| 34514 
2104a836b415
renamed fields of MarkupNode; implemented flatten and leafs
 immler@in.tum.de parents: 
34503diff
changeset | 283 |