| author | wenzelm | 
| Thu, 01 Dec 2022 11:36:45 +0100 | |
| changeset 76551 | c7996b073524 | 
| parent 76098 | bcca0fbb8a34 | 
| child 76965 | 922df6aa1607 | 
| permissions | -rw-r--r-- | 
| 55673 
0286219c1261
clarified module location (again, see 763d35697338);
 wenzelm parents: 
55666diff
changeset | 1 | /* Title: Pure/General/completion.scala | 
| 31763 | 2 | Author: Makarius | 
| 3 | ||
| 55687 | 4 | Semantic completion within the formal context (reported names). | 
| 55674 | 5 | Syntactic completion of keywords and symbols, with abbreviations | 
| 55992 | 6 | (based on language context). | 
| 7 | */ | |
| 31763 | 8 | |
| 9 | package isabelle | |
| 10 | ||
| 55618 | 11 | |
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 12 | import scala.collection.immutable.SortedMap | 
| 31763 | 13 | import scala.util.parsing.combinator.RegexParsers | 
| 57589 | 14 | import scala.util.matching.Regex | 
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 15 | import scala.math.Ordering | 
| 31763 | 16 | |
| 17 | ||
| 75393 | 18 | object Completion {
 | 
| 55694 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 19 | /** completion result **/ | 
| 53275 | 20 | |
| 53296 
65c60c782da5
less aggressive immediate completion, based on input and text;
 wenzelm parents: 
53295diff
changeset | 21 | sealed case class Item( | 
| 55692 
19e8b00684f7
more explicit Completion.Item.range, independently of caret;
 wenzelm parents: 
55687diff
changeset | 22 | range: Text.Range, | 
| 53296 
65c60c782da5
less aggressive immediate completion, based on input and text;
 wenzelm parents: 
53295diff
changeset | 23 | original: String, | 
| 55666 | 24 | name: String, | 
| 55978 | 25 | description: List[String], | 
| 53296 
65c60c782da5
less aggressive immediate completion, based on input and text;
 wenzelm parents: 
53295diff
changeset | 26 | replacement: String, | 
| 55666 | 27 | move: Int, | 
| 53296 
65c60c782da5
less aggressive immediate completion, based on input and text;
 wenzelm parents: 
53295diff
changeset | 28 | immediate: Boolean) | 
| 53275 | 29 | |
| 75393 | 30 |   object Result {
 | 
| 55914 
c5b752d549e3
clarified init_assignable: make double-sure that initial values are reset;
 wenzelm parents: 
55813diff
changeset | 31 | def empty(range: Text.Range): Result = Result(range, "", false, Nil) | 
| 66157 | 32 | def merge(history: History, results: Option[Result]*): Option[Result] = | 
| 73359 | 33 |       results.foldLeft(None: Option[Result]) {
 | 
| 66157 | 34 | case (result1, None) => result1 | 
| 35 | case (None, result2) => result2 | |
| 36 | case (result1 @ Some(res1), Some(res2)) => | |
| 56175 | 37 | if (res1.range != res2.range || res1.original != res2.original) result1 | 
| 38 |           else {
 | |
| 39 | val items = (res1.items ::: res2.items).sorted(history.ordering) | |
| 40 | Some(Result(res1.range, res1.original, false, items)) | |
| 41 | } | |
| 73359 | 42 | } | 
| 55914 
c5b752d549e3
clarified init_assignable: make double-sure that initial values are reset;
 wenzelm parents: 
55813diff
changeset | 43 | } | 
| 
c5b752d549e3
clarified init_assignable: make double-sure that initial values are reset;
 wenzelm parents: 
55813diff
changeset | 44 | |
| 55693 
93ba0085e888
try explicit semantic completion before syntax completion;
 wenzelm parents: 
55692diff
changeset | 45 | sealed case class Result( | 
| 
93ba0085e888
try explicit semantic completion before syntax completion;
 wenzelm parents: 
55692diff
changeset | 46 | range: Text.Range, | 
| 
93ba0085e888
try explicit semantic completion before syntax completion;
 wenzelm parents: 
55692diff
changeset | 47 | original: String, | 
| 
93ba0085e888
try explicit semantic completion before syntax completion;
 wenzelm parents: 
55692diff
changeset | 48 | unique: Boolean, | 
| 
93ba0085e888
try explicit semantic completion before syntax completion;
 wenzelm parents: 
55692diff
changeset | 49 | items: List[Item]) | 
| 53325 
ffabc0083786
more explicit indication of unique result (see also 45be26b98ca6, 3d654643cf56);
 wenzelm parents: 
53324diff
changeset | 50 | |
| 53275 | 51 | |
| 46624 | 52 | |
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 53 | /** persistent history **/ | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 54 | |
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 55 |   private val COMPLETION_HISTORY = Path.explode("$ISABELLE_HOME_USER/etc/completion_history")
 | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 56 | |
| 75393 | 57 |   object History {
 | 
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 58 | val empty: History = new History() | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 59 | |
| 75393 | 60 |     def load(): History = {
 | 
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 61 | def ignore_error(msg: String): Unit = | 
| 56782 
433cf57550fa
more systematic Isabelle output, like in classic Isabelle/ML (without markup);
 wenzelm parents: 
56661diff
changeset | 62 |         Output.warning("Ignoring bad content of file " + COMPLETION_HISTORY +
 | 
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 63 | (if (msg == "") "" else "\n" + msg)) | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 64 | |
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 65 | val content = | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 66 |         if (COMPLETION_HISTORY.is_file) {
 | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 67 |           try {
 | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 68 | import XML.Decode._ | 
| 65344 
b99283eed13c
clarified YXML vs. symbol encoding: operate on whole message;
 wenzelm parents: 
63887diff
changeset | 69 | list(pair(string, int))(Symbol.decode_yxml(File.read(COMPLETION_HISTORY))) | 
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 70 | } | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 71 |           catch {
 | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 72 | case ERROR(msg) => ignore_error(msg); Nil | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 73 |             case _: XML.Error => ignore_error(""); Nil
 | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 74 | } | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 75 | } | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 76 | else Nil | 
| 73359 | 77 | content.foldLeft(empty)(_ + _) | 
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 78 | } | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 79 | } | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 80 | |
| 75393 | 81 |   final class History private(rep: SortedMap[String, Int] = SortedMap.empty) {
 | 
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 82 |     override def toString: String = rep.mkString("Completion.History(", ",", ")")
 | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 83 | |
| 56878 
a5d082a85124
hardwired default_frequency to avoid fluctuation of popup content;
 wenzelm parents: 
56800diff
changeset | 84 | def frequency(name: String): Int = | 
| 
a5d082a85124
hardwired default_frequency to avoid fluctuation of popup content;
 wenzelm parents: 
56800diff
changeset | 85 | default_frequency(Symbol.encode(name)) getOrElse | 
| 
a5d082a85124
hardwired default_frequency to avoid fluctuation of popup content;
 wenzelm parents: 
56800diff
changeset | 86 | rep.getOrElse(name, 0) | 
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 87 | |
| 75393 | 88 |     def + (entry: (String, Int)): History = {
 | 
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 89 | val (name, freq) = entry | 
| 56564 
94c55cc73747
added spell-checker completion dialog, without counting frequency of items due to empty name;
 wenzelm parents: 
56278diff
changeset | 90 | if (name == "") this | 
| 
94c55cc73747
added spell-checker completion dialog, without counting frequency of items due to empty name;
 wenzelm parents: 
56278diff
changeset | 91 | else new History(rep + (name -> (frequency(name) + freq))) | 
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 92 | } | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 93 | |
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 94 | def ordering: Ordering[Item] = | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 95 |       new Ordering[Item] {
 | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 96 | def compare(item1: Item, item2: Item): Int = | 
| 56162 
ea6303e2261b
clarified completion ordering: prefer local names;
 wenzelm parents: 
56042diff
changeset | 97 | frequency(item2.name) compare frequency(item1.name) | 
| 
ea6303e2261b
clarified completion ordering: prefer local names;
 wenzelm parents: 
56042diff
changeset | 98 | } | 
| 
ea6303e2261b
clarified completion ordering: prefer local names;
 wenzelm parents: 
56042diff
changeset | 99 | |
| 75393 | 100 |     def save(): Unit = {
 | 
| 72375 | 101 | Isabelle_System.make_directory(COMPLETION_HISTORY.dir) | 
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 102 | File.write_backup(COMPLETION_HISTORY, | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 103 |         {
 | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 104 | import XML.Encode._ | 
| 65344 
b99283eed13c
clarified YXML vs. symbol encoding: operate on whole message;
 wenzelm parents: 
63887diff
changeset | 105 | Symbol.encode_yxml(list(pair(string, int))(rep.toList)) | 
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 106 | }) | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 107 | } | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 108 | } | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 109 | |
| 75393 | 110 |   class History_Variable {
 | 
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 111 | private var history = History.empty | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 112 |     def value: History = synchronized { history }
 | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 113 | |
| 75393 | 114 |     def load(): Unit = {
 | 
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 115 | val h = History.load() | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 116 |       synchronized { history = h }
 | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 117 | } | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 118 | |
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 119 |     def update(item: Item, freq: Int = 1): Unit = synchronized {
 | 
| 55666 | 120 | history = history + (item.name -> freq) | 
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 121 | } | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 122 | } | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 123 | |
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 124 | |
| 55694 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 125 | |
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 126 | /** semantic completion **/ | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 127 | |
| 61100 | 128 | def clean_name(s: String): Option[String] = | 
| 129 | if (s == "" || s == "_") None | |
| 130 | else Some(s.reverseIterator.dropWhile(_ == '_').toList.reverse.mkString) | |
| 131 | ||
| 66768 | 132 | def completed(input: String): String => Boolean = | 
| 133 |     clean_name(input) match {
 | |
| 134 | case None => (name: String) => false | |
| 135 | case Some(prefix) => (name: String) => name.startsWith(prefix) | |
| 136 | } | |
| 137 | ||
| 59717 
44a3ed0b8265
support for completion reports produced in Scala (inlined into messages);
 wenzelm parents: 
59319diff
changeset | 138 | def report_no_completion(pos: Position.T): String = | 
| 
44a3ed0b8265
support for completion reports produced in Scala (inlined into messages);
 wenzelm parents: 
59319diff
changeset | 139 | YXML.string_of_tree(Semantic.Info(pos, No_Completion)) | 
| 
44a3ed0b8265
support for completion reports produced in Scala (inlined into messages);
 wenzelm parents: 
59319diff
changeset | 140 | |
| 66766 | 141 | def report_names(pos: Position.T, names: List[(String, (String, String))], total: Int = 0): String = | 
| 66768 | 142 | if (names.isEmpty) "" | 
| 143 | else | |
| 144 | YXML.string_of_tree(Semantic.Info(pos, Names(if (total > 0) total else names.length, names))) | |
| 66766 | 145 | |
| 146 | def report_theories(pos: Position.T, thys: List[String], total: Int = 0): String = | |
| 72781 | 147 | report_names(pos, thys.map(name => (name, (Markup.THEORY, name))), total) | 
| 59717 
44a3ed0b8265
support for completion reports produced in Scala (inlined into messages);
 wenzelm parents: 
59319diff
changeset | 148 | |
| 75393 | 149 |   object Semantic {
 | 
| 150 |     object Info {
 | |
| 151 |       def apply(pos: Position.T, semantic: Semantic): XML.Elem = {
 | |
| 59717 
44a3ed0b8265
support for completion reports produced in Scala (inlined into messages);
 wenzelm parents: 
59319diff
changeset | 152 | val elem = | 
| 
44a3ed0b8265
support for completion reports produced in Scala (inlined into messages);
 wenzelm parents: 
59319diff
changeset | 153 |           semantic match {
 | 
| 
44a3ed0b8265
support for completion reports produced in Scala (inlined into messages);
 wenzelm parents: 
59319diff
changeset | 154 | case No_Completion => XML.Elem(Markup(Markup.NO_COMPLETION, pos), Nil) | 
| 
44a3ed0b8265
support for completion reports produced in Scala (inlined into messages);
 wenzelm parents: 
59319diff
changeset | 155 | case Names(total, names) => | 
| 
44a3ed0b8265
support for completion reports produced in Scala (inlined into messages);
 wenzelm parents: 
59319diff
changeset | 156 | XML.Elem(Markup(Markup.COMPLETION, pos), | 
| 
44a3ed0b8265
support for completion reports produced in Scala (inlined into messages);
 wenzelm parents: 
59319diff
changeset | 157 |                 {
 | 
| 
44a3ed0b8265
support for completion reports produced in Scala (inlined into messages);
 wenzelm parents: 
59319diff
changeset | 158 | import XML.Encode._ | 
| 
44a3ed0b8265
support for completion reports produced in Scala (inlined into messages);
 wenzelm parents: 
59319diff
changeset | 159 | pair(int, list(pair(string, pair(string, string))))(total, names) | 
| 
44a3ed0b8265
support for completion reports produced in Scala (inlined into messages);
 wenzelm parents: 
59319diff
changeset | 160 | }) | 
| 
44a3ed0b8265
support for completion reports produced in Scala (inlined into messages);
 wenzelm parents: 
59319diff
changeset | 161 | } | 
| 
44a3ed0b8265
support for completion reports produced in Scala (inlined into messages);
 wenzelm parents: 
59319diff
changeset | 162 | XML.Elem(Markup(Markup.REPORT, pos), List(elem)) | 
| 
44a3ed0b8265
support for completion reports produced in Scala (inlined into messages);
 wenzelm parents: 
59319diff
changeset | 163 | } | 
| 
44a3ed0b8265
support for completion reports produced in Scala (inlined into messages);
 wenzelm parents: 
59319diff
changeset | 164 | |
| 56173 | 165 | def unapply(info: Text.Markup): Option[Text.Info[Semantic]] = | 
| 55694 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 166 |         info.info match {
 | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 167 | case XML.Elem(Markup(Markup.COMPLETION, _), body) => | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 168 |             try {
 | 
| 75393 | 169 |               val (total, names) = {
 | 
| 55694 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 170 | import XML.Decode._ | 
| 55977 | 171 | pair(int, list(pair(string, pair(string, string))))(body) | 
| 55694 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 172 | } | 
| 56173 | 173 | Some(Text.Info(info.range, Names(total, names))) | 
| 55694 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 174 | } | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 175 |             catch { case _: XML.Error => None }
 | 
| 55914 
c5b752d549e3
clarified init_assignable: make double-sure that initial values are reset;
 wenzelm parents: 
55813diff
changeset | 176 | case XML.Elem(Markup(Markup.NO_COMPLETION, _), _) => | 
| 56173 | 177 | Some(Text.Info(info.range, No_Completion)) | 
| 55694 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 178 | case _ => None | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 179 | } | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 180 | } | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 181 | } | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 182 | |
| 56173 | 183 | sealed abstract class Semantic | 
| 56175 | 184 | case object No_Completion extends Semantic | 
| 75393 | 185 |   case class Names(total: Int, names: List[(String, (String, String))]) extends Semantic {
 | 
| 55694 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 186 | def complete( | 
| 56173 | 187 | range: Text.Range, | 
| 188 | history: Completion.History, | |
| 66055 | 189 | unicode: Boolean, | 
| 75393 | 190 | original: String | 
| 191 |     ): Option[Completion.Result] = {
 | |
| 66055 | 192 | def decode(s: String): String = if (unicode) Symbol.decode(s) else s | 
| 55694 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 193 | val items = | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 194 |         for {
 | 
| 55977 | 195 | (xname, (kind, name)) <- names | 
| 196 | xname1 = decode(xname) | |
| 55694 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 197 | if xname1 != original | 
| 55977 | 198 | (full_name, descr_name) = | 
| 199 | if (kind == "") (name, quote(decode(name))) | |
| 56600 | 200 | else | 
| 56800 | 201 | (Long_Name.qualify(kind, name), | 
| 60732 | 202 |               Word.implode(Word.explode('_', kind)) +
 | 
| 203 | (if (xname == name) "" else " " + quote(decode(name)))) | |
| 61622 
8bb7848b3d47
smart quoting of non-identifiers, e.g. jEdit actions;
 wenzelm parents: 
61613diff
changeset | 204 |         } yield {
 | 
| 
8bb7848b3d47
smart quoting of non-identifiers, e.g. jEdit actions;
 wenzelm parents: 
61613diff
changeset | 205 |           val description = List(xname1, "(" + descr_name + ")")
 | 
| 
8bb7848b3d47
smart quoting of non-identifiers, e.g. jEdit actions;
 wenzelm parents: 
61613diff
changeset | 206 | val replacement = | 
| 63761 
2ca536d0163e
more careful quoting, e.g. relevant for \<^control>cartouche;
 wenzelm parents: 
63587diff
changeset | 207 |             List(original, xname1).map(Token.explode(Keyword.Keywords.empty, _)) match {
 | 
| 
2ca536d0163e
more careful quoting, e.g. relevant for \<^control>cartouche;
 wenzelm parents: 
63587diff
changeset | 208 | case List(List(tok), _) if tok.kind == Token.Kind.CARTOUCHE => | 
| 67436 | 209 | Symbol.cartouche_decoded(xname1) | 
| 63761 
2ca536d0163e
more careful quoting, e.g. relevant for \<^control>cartouche;
 wenzelm parents: 
63587diff
changeset | 210 | case List(_, List(tok)) if tok.is_name => xname1 | 
| 61622 
8bb7848b3d47
smart quoting of non-identifiers, e.g. jEdit actions;
 wenzelm parents: 
61613diff
changeset | 211 | case _ => quote(xname1) | 
| 
8bb7848b3d47
smart quoting of non-identifiers, e.g. jEdit actions;
 wenzelm parents: 
61613diff
changeset | 212 | } | 
| 
8bb7848b3d47
smart quoting of non-identifiers, e.g. jEdit actions;
 wenzelm parents: 
61613diff
changeset | 213 | Item(range, original, full_name, description, replacement, 0, true) | 
| 
8bb7848b3d47
smart quoting of non-identifiers, e.g. jEdit actions;
 wenzelm parents: 
61613diff
changeset | 214 | } | 
| 55694 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 215 | |
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 216 | if (items.isEmpty) None | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 217 | else Some(Result(range, original, names.length == 1, items.sorted(history.ordering))) | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 218 | } | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 219 | } | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 220 | |
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 221 | |
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 222 | |
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 223 | /** syntactic completion **/ | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 224 | |
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 225 | /* language context */ | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 226 | |
| 75393 | 227 |   object Language_Context {
 | 
| 55749 | 228 |     val outer = Language_Context("", true, false)
 | 
| 229 | val inner = Language_Context(Markup.Language.UNKNOWN, true, false) | |
| 230 | val ML_outer = Language_Context(Markup.Language.ML, false, true) | |
| 231 | val ML_inner = Language_Context(Markup.Language.ML, true, false) | |
| 56278 
2576d3a40ed6
separate tokenization and language context for SML: no symbols, no antiquotes;
 wenzelm parents: 
56221diff
changeset | 232 | val SML_outer = Language_Context(Markup.Language.SML, false, false) | 
| 55694 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 233 | } | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 234 | |
| 75393 | 235 |   sealed case class Language_Context(language: String, symbols: Boolean, antiquotes: Boolean) {
 | 
| 55694 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 236 | def is_outer: Boolean = language == "" | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 237 | } | 
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 238 | |
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 239 | |
| 67004 
af72fa58f71b
clarified lazy Completion within Outer_Syntax: measurable speedup of Sessions.deps;
 wenzelm parents: 
66985diff
changeset | 240 | /* make */ | 
| 55694 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 241 | |
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 242 | val empty: Completion = new Completion() | 
| 66984 | 243 | |
| 67004 
af72fa58f71b
clarified lazy Completion within Outer_Syntax: measurable speedup of Sessions.deps;
 wenzelm parents: 
66985diff
changeset | 244 | def make(keywords: List[String], abbrevs: List[(String, String)]): Completion = | 
| 
af72fa58f71b
clarified lazy Completion within Outer_Syntax: measurable speedup of Sessions.deps;
 wenzelm parents: 
66985diff
changeset | 245 | empty.add_symbols.add_keywords(keywords).add_abbrevs( | 
| 
af72fa58f71b
clarified lazy Completion within Outer_Syntax: measurable speedup of Sessions.deps;
 wenzelm parents: 
66985diff
changeset | 246 | Completion.symbol_abbrevs ::: Completion.default_abbrevs ::: abbrevs) | 
| 55694 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 247 | |
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 248 | |
| 
a1184dfb8e00
clarified semantic completion: retain kind.full_name as official item name for history;
 wenzelm parents: 
55693diff
changeset | 249 | /* word parsers */ | 
| 31763 | 250 | |
| 75393 | 251 |   object Word_Parsers extends RegexParsers {
 | 
| 31763 | 252 | override val whiteSpace = "".r | 
| 253 | ||
| 61613 | 254 | private val symboloid_regex: Regex = """\\([A-Za-z0-9_']+|<\^?[A-Za-z0-9_']+>?)""".r | 
| 76098 | 255 | def is_symboloid(s: CharSequence): Boolean = symboloid_regex.matches(s) | 
| 57589 | 256 | |
| 55615 
bf4bbe72f740
completion of keywords and symbols based on language context;
 wenzelm parents: 
55497diff
changeset | 257 | private def reverse_symbol: Parser[String] = """>[A-Za-z0-9_']+\^?<\\""".r | 
| 
bf4bbe72f740
completion of keywords and symbols based on language context;
 wenzelm parents: 
55497diff
changeset | 258 |     private def reverse_symb: Parser[String] = """[A-Za-z0-9_']{2,}\^?<\\""".r
 | 
| 61599 | 259 | private def reverse_escape: Parser[String] = """[a-zA-Z0-9_']+\\""".r | 
| 55615 
bf4bbe72f740
completion of keywords and symbols based on language context;
 wenzelm parents: 
55497diff
changeset | 260 | |
| 56039 
5ff5208de089
include long identifiers in the reckoning of words (e.g. "integer.lifting" vs. 'lifting_update');
 wenzelm parents: 
56001diff
changeset | 261 | private val word_regex = "[a-zA-Z0-9_'.]+".r | 
| 63887 
2d9c12eba726
more uniform completion of short word: exclude single character prefix but include two chracter prefix (see also 31633e503c17);
 wenzelm parents: 
63873diff
changeset | 262 |     private def word2: Parser[String] = "[a-zA-Z0-9_'.]{2,}".r
 | 
| 55996 
13a7d9661ffc
allow suffix of underscores for words (notably keywords), similar to semantic completion;
 wenzelm parents: 
55994diff
changeset | 263 | private def underscores: Parser[String] = "_*".r | 
| 55615 
bf4bbe72f740
completion of keywords and symbols based on language context;
 wenzelm parents: 
55497diff
changeset | 264 | |
| 76098 | 265 | def is_word(s: CharSequence): Boolean = word_regex.matches(s) | 
| 56042 
d3c35a300433
proper Char comparison, despite weakly-typed Scala (cf. 5ff5208de089);
 wenzelm parents: 
56039diff
changeset | 266 | def is_word_char(c: Char): Boolean = Symbol.is_ascii_letdig(c) || c == '.' | 
| 31763 | 267 | |
| 75393 | 268 |     def read_symbol(in: CharSequence): Option[String] = {
 | 
| 55813 
08a1c860bc12
allow completion within a word (or symbol), like semantic completion;
 wenzelm parents: 
55749diff
changeset | 269 | val reverse_in = new Library.Reverse(in) | 
| 
08a1c860bc12
allow completion within a word (or symbol), like semantic completion;
 wenzelm parents: 
55749diff
changeset | 270 |       parse(reverse_symbol ^^ (_.reverse), reverse_in) match {
 | 
| 
08a1c860bc12
allow completion within a word (or symbol), like semantic completion;
 wenzelm parents: 
55749diff
changeset | 271 | case Success(result, _) => Some(result) | 
| 
08a1c860bc12
allow completion within a word (or symbol), like semantic completion;
 wenzelm parents: 
55749diff
changeset | 272 | case _ => None | 
| 
08a1c860bc12
allow completion within a word (or symbol), like semantic completion;
 wenzelm parents: 
55749diff
changeset | 273 | } | 
| 
08a1c860bc12
allow completion within a word (or symbol), like semantic completion;
 wenzelm parents: 
55749diff
changeset | 274 | } | 
| 
08a1c860bc12
allow completion within a word (or symbol), like semantic completion;
 wenzelm parents: 
55749diff
changeset | 275 | |
| 75393 | 276 |     def read_word(in: CharSequence): Option[(String, String)] = {
 | 
| 37072 
9105c8237c7a
renamed "rev" to "reverse" following usual Scala conventions;
 wenzelm parents: 
36012diff
changeset | 277 | val reverse_in = new Library.Reverse(in) | 
| 55996 
13a7d9661ffc
allow suffix of underscores for words (notably keywords), similar to semantic completion;
 wenzelm parents: 
55994diff
changeset | 278 | val parser = | 
| 61599 | 279 | (reverse_symbol | reverse_symb | reverse_escape) ^^ (x => (x.reverse, "")) | | 
| 63887 
2d9c12eba726
more uniform completion of short word: exclude single character prefix but include two chracter prefix (see also 31633e503c17);
 wenzelm parents: 
63873diff
changeset | 280 |         underscores ~ word2 ~ opt("?") ^^
 | 
| 56221 
a8dfbe9f25da
accomodate word as part of schematic variable name;
 wenzelm parents: 
56175diff
changeset | 281 |         { case x ~ y ~ z => (z.getOrElse("") + y.reverse, x) }
 | 
| 55996 
13a7d9661ffc
allow suffix of underscores for words (notably keywords), similar to semantic completion;
 wenzelm parents: 
55994diff
changeset | 282 |       parse(parser, reverse_in) match {
 | 
| 31763 | 283 | case Success(result, _) => Some(result) | 
| 284 | case _ => None | |
| 285 | } | |
| 286 | } | |
| 287 | } | |
| 55666 | 288 | |
| 289 | ||
| 63871 | 290 | /* templates */ | 
| 61960 | 291 | |
| 63869 | 292 | val caret_indicator = '\u0007' | 
| 63871 | 293 | |
| 63869 | 294 | def split_template(s: String): (String, String) = | 
| 295 |     space_explode(caret_indicator, s) match {
 | |
| 296 | case List(s1, s2) => (s1, s2) | |
| 297 | case _ => (s, "") | |
| 298 | } | |
| 299 | ||
| 63871 | 300 | |
| 301 | /* abbreviations */ | |
| 302 | ||
| 303 | private def symbol_abbrevs: Thy_Header.Abbrevs = | |
| 75199 | 304 | for (entry <- Symbol.symbols.entries; abbrev <- entry.abbrevs) | 
| 305 | yield (abbrev, entry.symbol) | |
| 63871 | 306 | |
| 55666 | 307 |   private val antiquote = "@{"
 | 
| 56878 
a5d082a85124
hardwired default_frequency to avoid fluctuation of popup content;
 wenzelm parents: 
56800diff
changeset | 308 | |
| 63578 | 309 | private val default_abbrevs = | 
| 56878 
a5d082a85124
hardwired default_frequency to avoid fluctuation of popup content;
 wenzelm parents: 
56800diff
changeset | 310 |     List("@{" -> "@{\u0007}",
 | 
| 
a5d082a85124
hardwired default_frequency to avoid fluctuation of popup content;
 wenzelm parents: 
56800diff
changeset | 311 | "`" -> "\\<close>", | 
| 
a5d082a85124
hardwired default_frequency to avoid fluctuation of popup content;
 wenzelm parents: 
56800diff
changeset | 312 | "`" -> "\\<open>", | 
| 63135 | 313 | "`" -> "\\<open>\u0007\\<close>", | 
| 314 | "\"" -> "\\<close>", | |
| 315 | "\"" -> "\\<open>", | |
| 316 | "\"" -> "\\<open>\u0007\\<close>") | |
| 56878 
a5d082a85124
hardwired default_frequency to avoid fluctuation of popup content;
 wenzelm parents: 
56800diff
changeset | 317 | |
| 
a5d082a85124
hardwired default_frequency to avoid fluctuation of popup content;
 wenzelm parents: 
56800diff
changeset | 318 | private def default_frequency(name: String): Option[Int] = | 
| 63578 | 319 | default_abbrevs.iterator.map(_._2).zipWithIndex.find(_._1 == name).map(_._2) | 
| 31763 | 320 | } | 
| 321 | ||
| 46712 | 322 | final class Completion private( | 
| 67004 
af72fa58f71b
clarified lazy Completion within Outer_Syntax: measurable speedup of Sessions.deps;
 wenzelm parents: 
66985diff
changeset | 323 | keywords: Set[String] = Set.empty, | 
| 
af72fa58f71b
clarified lazy Completion within Outer_Syntax: measurable speedup of Sessions.deps;
 wenzelm parents: 
66985diff
changeset | 324 | words_lex: Scan.Lexicon = Scan.Lexicon.empty, | 
| 
af72fa58f71b
clarified lazy Completion within Outer_Syntax: measurable speedup of Sessions.deps;
 wenzelm parents: 
66985diff
changeset | 325 | words_map: Multi_Map[String, String] = Multi_Map.empty, | 
| 
af72fa58f71b
clarified lazy Completion within Outer_Syntax: measurable speedup of Sessions.deps;
 wenzelm parents: 
66985diff
changeset | 326 | abbrevs_lex: Scan.Lexicon = Scan.Lexicon.empty, | 
| 75393 | 327 | abbrevs_map: Multi_Map[String, (String, String)] = Multi_Map.empty | 
| 328 | ) {
 | |
| 55983 | 329 | /* keywords */ | 
| 31763 | 330 | |
| 75197 | 331 | private def is_symbol(name: String): Boolean = Symbol.symbols.defined(name) | 
| 63579 | 332 | private def is_keyword(name: String): Boolean = !is_symbol(name) && keywords(name) | 
| 56001 
2df1e7bca361
do allow replacement of words where this is appropriate, notably symbol abbrevs and keyword templates (see also 1c42ebdb3a58);
 wenzelm parents: 
55996diff
changeset | 333 | private def is_keyword_template(name: String, template: Boolean): Boolean = | 
| 63579 | 334 | is_keyword(name) && (words_map.getOrElse(name, name) != name) == template | 
| 55984 | 335 | |
| 63579 | 336 | def add_keyword(keyword: String): Completion = | 
| 337 | new Completion(keywords + keyword, words_lex, words_map, abbrevs_lex, abbrevs_map) | |
| 40533 
e38e80686ce5
somewhat adhoc replacement for 'thus' and 'hence';
 wenzelm parents: 
37072diff
changeset | 338 | |
| 75393 | 339 |   def add_keywords(names: List[String]): Completion = {
 | 
| 73359 | 340 |     val keywords1 = names.foldLeft(keywords) { case (ks, k) => if (ks(k)) ks else ks + k }
 | 
| 67004 
af72fa58f71b
clarified lazy Completion within Outer_Syntax: measurable speedup of Sessions.deps;
 wenzelm parents: 
66985diff
changeset | 341 | if (keywords eq keywords1) this | 
| 
af72fa58f71b
clarified lazy Completion within Outer_Syntax: measurable speedup of Sessions.deps;
 wenzelm parents: 
66985diff
changeset | 342 | else new Completion(keywords1, words_lex, words_map, abbrevs_lex, abbrevs_map) | 
| 
af72fa58f71b
clarified lazy Completion within Outer_Syntax: measurable speedup of Sessions.deps;
 wenzelm parents: 
66985diff
changeset | 343 | } | 
| 
af72fa58f71b
clarified lazy Completion within Outer_Syntax: measurable speedup of Sessions.deps;
 wenzelm parents: 
66985diff
changeset | 344 | |
| 55983 | 345 | |
| 63578 | 346 | /* symbols and abbrevs */ | 
| 55983 | 347 | |
| 75393 | 348 |   def add_symbols: Completion = {
 | 
| 31763 | 349 | val words = | 
| 75394 | 350 |       Symbol.symbols.entries.flatMap { entry =>
 | 
| 75197 | 351 | val sym = entry.symbol | 
| 352 | val word = "\\" + entry.name | |
| 67311 | 353 | val seps = | 
| 75197 | 354 |           entry.argument match {
 | 
| 355 | case Symbol.Argument.none => Nil | |
| 356 |             case Symbol.Argument.cartouche => List("")
 | |
| 357 |             case Symbol.Argument.space_cartouche => List(" ")
 | |
| 358 | } | |
| 67311 | 359 | List(sym -> sym, word -> sym) ::: | 
| 360 | seps.map(sep => word -> (sym + sep + "\\<open>\u0007\\<close>")) | |
| 75394 | 361 | } | 
| 31763 | 362 | |
| 63578 | 363 | new Completion( | 
| 364 | keywords, | |
| 365 | words_lex ++ words.map(_._1), | |
| 366 | words_map ++ words, | |
| 367 | abbrevs_lex, | |
| 368 | abbrevs_map) | |
| 369 | } | |
| 55666 | 370 | |
| 63578 | 371 | def add_abbrevs(abbrevs: List[(String, String)]): Completion = | 
| 73359 | 372 | abbrevs.foldLeft(this)(_.add_abbrev(_)) | 
| 63808 | 373 | |
| 374 | private def add_abbrev(abbrev: (String, String)): Completion = | |
| 63873 
228a85f1d6af
clarified GUI representation of replacement texts with zero or more abbrevs;
 wenzelm parents: 
63871diff
changeset | 375 |     abbrev match {
 | 
| 
228a85f1d6af
clarified GUI representation of replacement texts with zero or more abbrevs;
 wenzelm parents: 
63871diff
changeset | 376 |       case ("", _) => this
 | 
| 
228a85f1d6af
clarified GUI representation of replacement texts with zero or more abbrevs;
 wenzelm parents: 
63871diff
changeset | 377 | case (abbr, text) => | 
| 
228a85f1d6af
clarified GUI representation of replacement texts with zero or more abbrevs;
 wenzelm parents: 
63871diff
changeset | 378 | val rev_abbr = abbr.reverse | 
| 
228a85f1d6af
clarified GUI representation of replacement texts with zero or more abbrevs;
 wenzelm parents: 
63871diff
changeset | 379 | val is_word = Completion.Word_Parsers.is_word(abbr) | 
| 31763 | 380 | |
| 63873 
228a85f1d6af
clarified GUI representation of replacement texts with zero or more abbrevs;
 wenzelm parents: 
63871diff
changeset | 381 | val (words_lex1, words_map1) = | 
| 
228a85f1d6af
clarified GUI representation of replacement texts with zero or more abbrevs;
 wenzelm parents: 
63871diff
changeset | 382 | if (!is_word) (words_lex, words_map) | 
| 
228a85f1d6af
clarified GUI representation of replacement texts with zero or more abbrevs;
 wenzelm parents: 
63871diff
changeset | 383 | else if (text != "") (words_lex + abbr, words_map + abbrev) | 
| 
228a85f1d6af
clarified GUI representation of replacement texts with zero or more abbrevs;
 wenzelm parents: 
63871diff
changeset | 384 | else (words_lex -- List(abbr), words_map - abbr) | 
| 63808 | 385 | |
| 63873 
228a85f1d6af
clarified GUI representation of replacement texts with zero or more abbrevs;
 wenzelm parents: 
63871diff
changeset | 386 | val (abbrevs_lex1, abbrevs_map1) = | 
| 
228a85f1d6af
clarified GUI representation of replacement texts with zero or more abbrevs;
 wenzelm parents: 
63871diff
changeset | 387 | if (is_word) (abbrevs_lex, abbrevs_map) | 
| 
228a85f1d6af
clarified GUI representation of replacement texts with zero or more abbrevs;
 wenzelm parents: 
63871diff
changeset | 388 | else if (text != "") (abbrevs_lex + rev_abbr, abbrevs_map + (rev_abbr -> abbrev)) | 
| 
228a85f1d6af
clarified GUI representation of replacement texts with zero or more abbrevs;
 wenzelm parents: 
63871diff
changeset | 389 | else (abbrevs_lex -- List(rev_abbr), abbrevs_map - rev_abbr) | 
| 63808 | 390 | |
| 63873 
228a85f1d6af
clarified GUI representation of replacement texts with zero or more abbrevs;
 wenzelm parents: 
63871diff
changeset | 391 | new Completion(keywords, words_lex1, words_map1, abbrevs_lex1, abbrevs_map1) | 
| 
228a85f1d6af
clarified GUI representation of replacement texts with zero or more abbrevs;
 wenzelm parents: 
63871diff
changeset | 392 | } | 
| 31763 | 393 | |
| 394 | ||
| 395 | /* complete */ | |
| 396 | ||
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 397 | def complete( | 
| 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 398 | history: Completion.History, | 
| 66055 | 399 | unicode: Boolean, | 
| 53337 
b3817a0e3211
sort items according to persistent history of frequency of use;
 wenzelm parents: 
53325diff
changeset | 400 | explicit: Boolean, | 
| 55813 
08a1c860bc12
allow completion within a word (or symbol), like semantic completion;
 wenzelm parents: 
55749diff
changeset | 401 | start: Text.Offset, | 
| 55615 
bf4bbe72f740
completion of keywords and symbols based on language context;
 wenzelm parents: 
55497diff
changeset | 402 | text: CharSequence, | 
| 55813 
08a1c860bc12
allow completion within a word (or symbol), like semantic completion;
 wenzelm parents: 
55749diff
changeset | 403 | caret: Int, | 
| 75393 | 404 | language_context: Completion.Language_Context | 
| 405 |   ): Option[Completion.Result] = {
 | |
| 55813 
08a1c860bc12
allow completion within a word (or symbol), like semantic completion;
 wenzelm parents: 
55749diff
changeset | 406 | val length = text.length | 
| 
08a1c860bc12
allow completion within a word (or symbol), like semantic completion;
 wenzelm parents: 
55749diff
changeset | 407 | |
| 75393 | 408 |     val abbrevs_result = {
 | 
| 55813 
08a1c860bc12
allow completion within a word (or symbol), like semantic completion;
 wenzelm parents: 
55749diff
changeset | 409 | val reverse_in = new Library.Reverse(text.subSequence(0, caret)) | 
| 
08a1c860bc12
allow completion within a word (or symbol), like semantic completion;
 wenzelm parents: 
55749diff
changeset | 410 |       Scan.Parsers.parse(Scan.Parsers.literal(abbrevs_lex), reverse_in) match {
 | 
| 61599 | 411 | case Scan.Parsers.Success(reverse_abbr, _) => | 
| 412 | val abbrevs = abbrevs_map.get_list(reverse_abbr) | |
| 55666 | 413 |           abbrevs match {
 | 
| 414 | case Nil => None | |
| 61599 | 415 | case (abbr, _) :: _ => | 
| 55666 | 416 | val ok = | 
| 61599 | 417 | if (abbr == Completion.antiquote) language_context.antiquotes | 
| 63578 | 418 | else language_context.symbols || Completion.default_abbrevs.exists(_._1 == abbr) | 
| 61599 | 419 | if (ok) Some((abbr, abbrevs)) | 
| 55813 
08a1c860bc12
allow completion within a word (or symbol), like semantic completion;
 wenzelm parents: 
55749diff
changeset | 420 | else None | 
| 55666 | 421 | } | 
| 422 | case _ => None | |
| 423 | } | |
| 55813 
08a1c860bc12
allow completion within a word (or symbol), like semantic completion;
 wenzelm parents: 
55749diff
changeset | 424 | } | 
| 55615 
bf4bbe72f740
completion of keywords and symbols based on language context;
 wenzelm parents: 
55497diff
changeset | 425 | |
| 
bf4bbe72f740
completion of keywords and symbols based on language context;
 wenzelm parents: 
55497diff
changeset | 426 | val words_result = | 
| 55982 | 427 | if (abbrevs_result.isDefined) None | 
| 428 |       else {
 | |
| 57602 
0f708666eb7c
no keyword completion within word context -- especially avoid its odd visual rendering;
 wenzelm parents: 
57589diff
changeset | 429 | val word_context = | 
| 
0f708666eb7c
no keyword completion within word context -- especially avoid its odd visual rendering;
 wenzelm parents: 
57589diff
changeset | 430 | caret < length && Completion.Word_Parsers.is_word_char(text.charAt(caret)) | 
| 55996 
13a7d9661ffc
allow suffix of underscores for words (notably keywords), similar to semantic completion;
 wenzelm parents: 
55994diff
changeset | 431 | val result = | 
| 57588 
ff31aad27661
discontinued unfinished attempts at syntactic word context (see 2e1398b484aa, 08a1c860bc12, 7f229b0212fe) -- back to more basic completion of Isabelle2013-2;
 wenzelm parents: 
56878diff
changeset | 432 |           Completion.Word_Parsers.read_symbol(text.subSequence(0, caret)) match {
 | 
| 55996 
13a7d9661ffc
allow suffix of underscores for words (notably keywords), similar to semantic completion;
 wenzelm parents: 
55994diff
changeset | 433 | case Some(symbol) => Some((symbol, "")) | 
| 63887 
2d9c12eba726
more uniform completion of short word: exclude single character prefix but include two chracter prefix (see also 31633e503c17);
 wenzelm parents: 
63873diff
changeset | 434 | case None => Completion.Word_Parsers.read_word(text.subSequence(0, caret)) | 
| 55992 | 435 | } | 
| 55996 
13a7d9661ffc
allow suffix of underscores for words (notably keywords), similar to semantic completion;
 wenzelm parents: 
55994diff
changeset | 436 | result.map( | 
| 55992 | 437 |           {
 | 
| 55996 
13a7d9661ffc
allow suffix of underscores for words (notably keywords), similar to semantic completion;
 wenzelm parents: 
55994diff
changeset | 438 | case (word, underscores) => | 
| 
13a7d9661ffc
allow suffix of underscores for words (notably keywords), similar to semantic completion;
 wenzelm parents: 
55994diff
changeset | 439 | val complete_words = words_lex.completions(word) | 
| 
13a7d9661ffc
allow suffix of underscores for words (notably keywords), similar to semantic completion;
 wenzelm parents: 
55994diff
changeset | 440 | val full_word = word + underscores | 
| 
13a7d9661ffc
allow suffix of underscores for words (notably keywords), similar to semantic completion;
 wenzelm parents: 
55994diff
changeset | 441 | val completions = | 
| 56001 
2df1e7bca361
do allow replacement of words where this is appropriate, notably symbol abbrevs and keyword templates (see also 1c42ebdb3a58);
 wenzelm parents: 
55996diff
changeset | 442 | if (complete_words.contains(full_word) && is_keyword_template(full_word, false)) Nil | 
| 55996 
13a7d9661ffc
allow suffix of underscores for words (notably keywords), similar to semantic completion;
 wenzelm parents: 
55994diff
changeset | 443 | else | 
| 
13a7d9661ffc
allow suffix of underscores for words (notably keywords), similar to semantic completion;
 wenzelm parents: 
55994diff
changeset | 444 |                   for {
 | 
| 
13a7d9661ffc
allow suffix of underscores for words (notably keywords), similar to semantic completion;
 wenzelm parents: 
55994diff
changeset | 445 | complete_word <- complete_words | 
| 
13a7d9661ffc
allow suffix of underscores for words (notably keywords), similar to semantic completion;
 wenzelm parents: 
55994diff
changeset | 446 | ok = | 
| 57602 
0f708666eb7c
no keyword completion within word context -- especially avoid its odd visual rendering;
 wenzelm parents: 
57589diff
changeset | 447 | if (is_keyword(complete_word)) !word_context && language_context.is_outer | 
| 61600 
1ca11ddfcc70
clarified completion of explicit symbols (see also f6bd97a587b7, e0e4ac981cf1);
 wenzelm parents: 
61599diff
changeset | 448 | else language_context.symbols || Completion.Word_Parsers.is_symboloid(word) | 
| 55996 
13a7d9661ffc
allow suffix of underscores for words (notably keywords), similar to semantic completion;
 wenzelm parents: 
55994diff
changeset | 449 | if ok | 
| 
13a7d9661ffc
allow suffix of underscores for words (notably keywords), similar to semantic completion;
 wenzelm parents: 
55994diff
changeset | 450 | completion <- words_map.get_list(complete_word) | 
| 
13a7d9661ffc
allow suffix of underscores for words (notably keywords), similar to semantic completion;
 wenzelm parents: 
55994diff
changeset | 451 | } yield (complete_word, completion) | 
| 60215 | 452 | (full_word, completions) | 
| 55992 | 453 | }) | 
| 53275 | 454 | } | 
| 55615 
bf4bbe72f740
completion of keywords and symbols based on language context;
 wenzelm parents: 
55497diff
changeset | 455 | |
| 55982 | 456 |     (abbrevs_result orElse words_result) match {
 | 
| 59319 | 457 | case Some((original, completions)) if completions.nonEmpty => | 
| 57588 
ff31aad27661
discontinued unfinished attempts at syntactic word context (see 2e1398b484aa, 08a1c860bc12, 7f229b0212fe) -- back to more basic completion of Isabelle2013-2;
 wenzelm parents: 
56878diff
changeset | 458 | val range = Text.Range(- original.length, 0) + caret + start | 
| 55985 | 459 | val immediate = | 
| 460 | explicit || | |
| 461 | (!Completion.Word_Parsers.is_word(original) && | |
| 61600 
1ca11ddfcc70
clarified completion of explicit symbols (see also f6bd97a587b7, e0e4ac981cf1);
 wenzelm parents: 
61599diff
changeset | 462 | !Completion.Word_Parsers.is_symboloid(original) && | 
| 55985 | 463 | Character.codePointCount(original, 0, original.length) > 1) | 
| 55993 | 464 | val unique = completions.length == 1 | 
| 55985 | 465 | |
| 66056 | 466 | def decode1(s: String): String = if (unicode) Symbol.decode(s) else s | 
| 467 | def decode2(s: String): String = if (unicode) s else Symbol.decode(s) | |
| 468 | ||
| 55985 | 469 | val items = | 
| 470 |           for {
 | |
| 55993 | 471 | (complete_word, name0) <- completions | 
| 66056 | 472 | name1 = decode1(name0) | 
| 473 | name2 = decode2(name0) | |
| 55985 | 474 | if name1 != original | 
| 63869 | 475 | (s1, s2) = Completion.split_template(name1) | 
| 55985 | 476 | move = - s2.length | 
| 477 | description = | |
| 478 |               if (is_symbol(name0)) {
 | |
| 66056 | 479 | if (name1 == name2) List(name1) | 
| 480 | else List(name1, "(symbol " + quote(name2) + ")") | |
| 55985 | 481 | } | 
| 56001 
2df1e7bca361
do allow replacement of words where this is appropriate, notably symbol abbrevs and keyword templates (see also 1c42ebdb3a58);
 wenzelm parents: 
55996diff
changeset | 482 | else if (is_keyword_template(complete_word, true)) | 
| 55993 | 483 | List(name1, "(template " + quote(complete_word) + ")") | 
| 484 | else if (move != 0) List(name1, "(template)") | |
| 485 | else if (is_keyword(complete_word)) List(name1, "(keyword)") | |
| 55985 | 486 | else List(name1) | 
| 487 | } | |
| 488 | yield Completion.Item(range, original, name1, description, s1 + s2, move, immediate) | |
| 489 | ||
| 490 | if (items.isEmpty) None | |
| 56174 | 491 | else | 
| 492 | Some(Completion.Result(range, original, unique, | |
| 493 | items.sortBy(_.name).sorted(history.ordering))) | |
| 55985 | 494 | |
| 55992 | 495 | case _ => None | 
| 31765 | 496 | } | 
| 31763 | 497 | } | 
| 498 | } |