src/Tools/VSCode/src/vscode_model.scala
author Thomas Lindae <thomas.lindae@in.tum.de>
Mon, 27 May 2024 13:17:09 +0200 (8 months ago)
changeset 81043 2174ec5f0d0c
parent 76904 e27d097d7d15
child 81061 fe9ae6b67c29
permissions -rw-r--r--
lsp: added decoration_request notification;
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
72761
4519eeefe3b5 avoid conflicting base names;
wenzelm
parents: 71774
diff changeset
     1
/*  Title:      Tools/VSCode/src/vscode_model.scala
64605
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
     2
    Author:     Makarius
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
     3
72761
4519eeefe3b5 avoid conflicting base names;
wenzelm
parents: 71774
diff changeset
     4
VSCode document model for line-oriented text.
64605
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
     5
*/
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
     6
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
     7
package isabelle.vscode
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
     8
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
     9
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
    10
import isabelle._
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
    11
64710
72ca4e5f976e manage changes of external files;
wenzelm
parents: 64709
diff changeset
    12
import java.io.{File => JFile}
72ca4e5f976e manage changes of external files;
wenzelm
parents: 64709
diff changeset
    13
64605
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
    14
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
    15
object VSCode_Model {
65095
eb21a4f70b0e publish decorations like diagnostics;
wenzelm
parents: 64877
diff changeset
    16
  /* decorations */
eb21a4f70b0e publish decorations like diagnostics;
wenzelm
parents: 64877
diff changeset
    17
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
    18
  object Decoration {
65140
1191df79aa1c tuned signature;
wenzelm
parents: 65120
diff changeset
    19
    def empty(typ: String): Decoration = Decoration(typ, Nil)
1191df79aa1c tuned signature;
wenzelm
parents: 65120
diff changeset
    20
1191df79aa1c tuned signature;
wenzelm
parents: 65120
diff changeset
    21
    def ranges(typ: String, ranges: List[Text.Range]): Decoration =
1191df79aa1c tuned signature;
wenzelm
parents: 65120
diff changeset
    22
      Decoration(typ, ranges.map(Text.Info(_, List.empty[XML.Body])))
1191df79aa1c tuned signature;
wenzelm
parents: 65120
diff changeset
    23
  }
65105
1f47b92021de clarified signature;
wenzelm
parents: 65095
diff changeset
    24
  sealed case class Decoration(typ: String, content: List[Text.Info[List[XML.Body]]])
65095
eb21a4f70b0e publish decorations like diagnostics;
wenzelm
parents: 64877
diff changeset
    25
eb21a4f70b0e publish decorations like diagnostics;
wenzelm
parents: 64877
diff changeset
    26
eb21a4f70b0e publish decorations like diagnostics;
wenzelm
parents: 64877
diff changeset
    27
  /* content */
eb21a4f70b0e publish decorations like diagnostics;
wenzelm
parents: 64877
diff changeset
    28
76781
d9f48960bf23 clarified signature: more position information via node_name;
wenzelm
parents: 76778
diff changeset
    29
  sealed case class Content(node_name: Document.Node.Name, doc: Line.Document) {
65161
6af056380d0b clarified;
wenzelm
parents: 65160
diff changeset
    30
    override def toString: String = doc.toString
65197
8fada74d82be tuned signature;
wenzelm
parents: 65196
diff changeset
    31
    def text_length: Text.Offset = doc.text_length
64830
9bc44bef99e6 more explocit Document_Model.Content;
wenzelm
parents: 64827
diff changeset
    32
    def text_range: Text.Range = doc.text_range
9bc44bef99e6 more explocit Document_Model.Content;
wenzelm
parents: 64827
diff changeset
    33
    def text: String = doc.text
64833
0f410e3b1d20 support for bibtex entries;
wenzelm
parents: 64830
diff changeset
    34
76307
072e6c0a2373 proper Symbol.encode (following fd1efd6dd385), e.g. relevant for 'ML_file' with symbols like \<^here>;
wenzelm
parents: 75393
diff changeset
    35
    lazy val bytes: Bytes = Bytes(Symbol.encode(text))
64830
9bc44bef99e6 more explocit Document_Model.Content;
wenzelm
parents: 64827
diff changeset
    36
    lazy val chunk: Symbol.Text_Chunk = Symbol.Text_Chunk(text)
76792
23f433294173 support for generic File_Format.parse_data, with persistent result in document model;
wenzelm
parents: 76781
diff changeset
    37
    lazy val data: AnyRef = File_Format.registry.parse_data(node_name, text)
66676
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
    38
72761
4519eeefe3b5 avoid conflicting base names;
wenzelm
parents: 71774
diff changeset
    39
    def recode_symbols: List[LSP.TextEdit] =
66676
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
    40
      (for {
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
    41
        (line, l) <- doc.lines.iterator.zipWithIndex
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
    42
        text1 = Symbol.encode(line.text)
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
    43
        if (line.text != text1)
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
    44
      } yield {
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
    45
        val range = Line.Range(Line.Position(l), Line.Position(l, line.text.length))
72761
4519eeefe3b5 avoid conflicting base names;
wenzelm
parents: 71774
diff changeset
    46
        LSP.TextEdit(range, text1)
66676
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
    47
      }).toList
64830
9bc44bef99e6 more explocit Document_Model.Content;
wenzelm
parents: 64827
diff changeset
    48
  }
9bc44bef99e6 more explocit Document_Model.Content;
wenzelm
parents: 64827
diff changeset
    49
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
    50
  def init(
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
    51
    session: Session,
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
    52
    editor: Language_Server.Editor,
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
    53
    node_name: Document.Node.Name
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
    54
  ): VSCode_Model = {
76781
d9f48960bf23 clarified signature: more position information via node_name;
wenzelm
parents: 76778
diff changeset
    55
    val content = Content(node_name, Line.Document.empty)
d9f48960bf23 clarified signature: more position information via node_name;
wenzelm
parents: 76778
diff changeset
    56
    val is_theory = File_Format.registry.is_theory(node_name)
d9f48960bf23 clarified signature: more position information via node_name;
wenzelm
parents: 76778
diff changeset
    57
    VSCode_Model(session, editor, content, node_required = is_theory)
72761
4519eeefe3b5 avoid conflicting base names;
wenzelm
parents: 71774
diff changeset
    58
  }
64707
7157685b71e3 clarified Document_Model perspective and edits;
wenzelm
parents: 64704
diff changeset
    59
}
7157685b71e3 clarified Document_Model perspective and edits;
wenzelm
parents: 64704
diff changeset
    60
72761
4519eeefe3b5 avoid conflicting base names;
wenzelm
parents: 71774
diff changeset
    61
sealed case class VSCode_Model(
64707
7157685b71e3 clarified Document_Model perspective and edits;
wenzelm
parents: 64704
diff changeset
    62
  session: Session,
72761
4519eeefe3b5 avoid conflicting base names;
wenzelm
parents: 71774
diff changeset
    63
  editor: Language_Server.Editor,
4519eeefe3b5 avoid conflicting base names;
wenzelm
parents: 71774
diff changeset
    64
  content: VSCode_Model.Content,
66674
30d5195299e2 store document version;
wenzelm
parents: 66150
diff changeset
    65
  version: Option[Long] = None,
64726
c534a2ac537d clarified;
wenzelm
parents: 64725
diff changeset
    66
  external_file: Boolean = false,
76715
bf5ff407f32f clarified state of document model vs. document editor selection (again, see also a9d52d02bd83);
wenzelm
parents: 76703
diff changeset
    67
  node_required: Boolean = false,
76702
94cdf6513f01 clarified signature;
wenzelm
parents: 76481
diff changeset
    68
  last_perspective: Document.Node.Perspective_Text.T = Document.Node.Perspective_Text.empty,
64816
wenzelm
parents: 64815
diff changeset
    69
  pending_edits: List[Text.Edit] = Nil,
65095
eb21a4f70b0e publish decorations like diagnostics;
wenzelm
parents: 64877
diff changeset
    70
  published_diagnostics: List[Text.Info[Command.Results]] = Nil,
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
    71
  published_decorations: List[VSCode_Model.Decoration] = Nil
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
    72
) extends Document.Model {
66114
c137a9f038a6 clarified signature;
wenzelm
parents: 66100
diff changeset
    73
  model =>
c137a9f038a6 clarified signature;
wenzelm
parents: 66100
diff changeset
    74
76765
c654103e9c9d more robust Document.Pending_Edits: cover all nodes simulataneously, and thus support proper Snapshot.switch;
wenzelm
parents: 76761
diff changeset
    75
66674
30d5195299e2 store document version;
wenzelm
parents: 66150
diff changeset
    76
  /* content */
66114
c137a9f038a6 clarified signature;
wenzelm
parents: 66100
diff changeset
    77
76781
d9f48960bf23 clarified signature: more position information via node_name;
wenzelm
parents: 76778
diff changeset
    78
  def node_name: Document.Node.Name = content.node_name
d9f48960bf23 clarified signature: more position information via node_name;
wenzelm
parents: 76778
diff changeset
    79
67014
e6a695d6a6b2 tuned signature;
wenzelm
parents: 67004
diff changeset
    80
  def get_text(range: Text.Range): Option[String] = content.doc.get_text(range)
66114
c137a9f038a6 clarified signature;
wenzelm
parents: 66100
diff changeset
    81
72761
4519eeefe3b5 avoid conflicting base names;
wenzelm
parents: 71774
diff changeset
    82
  def set_version(new_version: Long): VSCode_Model = copy(version = Some(new_version))
66674
30d5195299e2 store document version;
wenzelm
parents: 66150
diff changeset
    83
66114
c137a9f038a6 clarified signature;
wenzelm
parents: 66100
diff changeset
    84
64710
72ca4e5f976e manage changes of external files;
wenzelm
parents: 64709
diff changeset
    85
  /* external file */
72ca4e5f976e manage changes of external files;
wenzelm
parents: 64709
diff changeset
    86
72761
4519eeefe3b5 avoid conflicting base names;
wenzelm
parents: 71774
diff changeset
    87
  def external(b: Boolean): VSCode_Model = copy(external_file = b)
64727
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
    88
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64777
diff changeset
    89
  def node_visible: Boolean = !external_file
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64777
diff changeset
    90
64710
72ca4e5f976e manage changes of external files;
wenzelm
parents: 64709
diff changeset
    91
64690
599873de8b01 more uniform pending_input / pending_output;
wenzelm
parents: 64683
diff changeset
    92
  /* header */
599873de8b01 more uniform pending_input / pending_output;
wenzelm
parents: 64683
diff changeset
    93
64702
d95b9117cb5b re-use resources from session;
wenzelm
parents: 64690
diff changeset
    94
  def node_header: Document.Node.Header =
64673
b5965890e54d more uniform treatment of file name vs. theory name and special header;
wenzelm
parents: 64672
diff changeset
    95
    resources.special_header(node_name) getOrElse
72772
a9ef39041114 clarified signature;
wenzelm
parents: 72761
diff changeset
    96
      resources.check_thy(node_name, Scan.char_reader(content.text))
64605
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
    97
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
    98
64707
7157685b71e3 clarified Document_Model perspective and edits;
wenzelm
parents: 64704
diff changeset
    99
  /* perspective */
7157685b71e3 clarified Document_Model perspective and edits;
wenzelm
parents: 64704
diff changeset
   100
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
   101
  def node_perspective(
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
   102
    doc_blobs: Document.Blobs,
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
   103
    caret: Option[Line.Position]
76702
94cdf6513f01 clarified signature;
wenzelm
parents: 76481
diff changeset
   104
  ): (Boolean, Document.Node.Perspective_Text.T) = {
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64777
diff changeset
   105
    if (is_theory) {
76765
c654103e9c9d more robust Document.Pending_Edits: cover all nodes simulataneously, and thus support proper Snapshot.switch;
wenzelm
parents: 76761
diff changeset
   106
      val snapshot = resources.snapshot(model)
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64777
diff changeset
   107
76715
bf5ff407f32f clarified state of document model vs. document editor selection (again, see also a9d52d02bd83);
wenzelm
parents: 76703
diff changeset
   108
      val required = node_required || editor.document_node_required(node_name)
bf5ff407f32f clarified state of document model vs. document editor selection (again, see also a9d52d02bd83);
wenzelm
parents: 76703
diff changeset
   109
65926
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   110
      val caret_perspective = resources.options.int("vscode_caret_perspective") max 0
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   111
      val caret_range =
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   112
        if (caret_perspective != 0) {
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   113
          caret match {
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   114
            case Some(pos) =>
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   115
              val doc = content.doc
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   116
              val pos1 = Line.Position((pos.line - caret_perspective) max 0)
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   117
              val pos2 = Line.Position((pos.line + caret_perspective + 1) min doc.lines.length)
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   118
              Text.Range(doc.offset(pos1).get, doc.offset(pos2).get)
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   119
            case None => Text.Range.offside
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   120
          }
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   121
        }
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   122
        else if (node_visible) content.text_range
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   123
        else Text.Range.offside
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   124
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64777
diff changeset
   125
      val text_perspective =
65926
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   126
        if (snapshot.commands_loading_ranges(resources.visible_node(_)).nonEmpty)
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64777
diff changeset
   127
          Text.Perspective.full
65926
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   128
        else
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   129
          content.text_range.try_restrict(caret_range) match {
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   130
            case Some(range) => Text.Perspective(List(range))
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   131
            case None => Text.Perspective.empty
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   132
          }
64723
65bcb1fbaa73 clarified node_visible;
wenzelm
parents: 64710
diff changeset
   133
66100
d1ad5a7458c2 proper treatment of editor overlays;
wenzelm
parents: 66054
diff changeset
   134
      val overlays = editor.node_overlays(node_name)
d1ad5a7458c2 proper treatment of editor overlays;
wenzelm
parents: 66054
diff changeset
   135
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64777
diff changeset
   136
      (snapshot.node.load_commands_changed(doc_blobs),
76715
bf5ff407f32f clarified state of document model vs. document editor selection (again, see also a9d52d02bd83);
wenzelm
parents: 76703
diff changeset
   137
        Document.Node.Perspective(required, text_perspective, overlays))
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64777
diff changeset
   138
    }
76702
94cdf6513f01 clarified signature;
wenzelm
parents: 76481
diff changeset
   139
    else (false, Document.Node.Perspective_Text.empty)
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64777
diff changeset
   140
  }
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64777
diff changeset
   141
64707
7157685b71e3 clarified Document_Model perspective and edits;
wenzelm
parents: 64704
diff changeset
   142
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64777
diff changeset
   143
  /* blob */
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64777
diff changeset
   144
76904
e27d097d7d15 tuned signature: avoid confusion with Document.Node.Blob and Command.Blob;
wenzelm
parents: 76792
diff changeset
   145
  def get_blob: Option[Document.Blobs.Item] =
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64777
diff changeset
   146
    if (is_theory) None
76904
e27d097d7d15 tuned signature: avoid confusion with Document.Node.Blob and Command.Blob;
wenzelm
parents: 76792
diff changeset
   147
    else Some(Document.Blobs.Item(content.bytes, content.text, content.chunk, pending_edits.nonEmpty))
64707
7157685b71e3 clarified Document_Model perspective and edits;
wenzelm
parents: 64704
diff changeset
   148
7157685b71e3 clarified Document_Model perspective and edits;
wenzelm
parents: 64704
diff changeset
   149
76792
23f433294173 support for generic File_Format.parse_data, with persistent result in document model;
wenzelm
parents: 76781
diff changeset
   150
  /* data */
66150
c2e19b9e1398 tuned signature;
wenzelm
parents: 66114
diff changeset
   151
76792
23f433294173 support for generic File_Format.parse_data, with persistent result in document model;
wenzelm
parents: 76781
diff changeset
   152
  def untyped_data: AnyRef = model.content.data
66150
c2e19b9e1398 tuned signature;
wenzelm
parents: 66114
diff changeset
   153
c2e19b9e1398 tuned signature;
wenzelm
parents: 66114
diff changeset
   154
64605
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
   155
  /* edits */
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
   156
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
   157
  def change_text(text: String, range: Option[Line.Range] = None): Option[VSCode_Model] = {
65160
6e042537555d incremental document changes;
wenzelm
parents: 65140
diff changeset
   158
    val insert = Line.normalize(text)
6e042537555d incremental document changes;
wenzelm
parents: 65140
diff changeset
   159
    range match {
6e042537555d incremental document changes;
wenzelm
parents: 65140
diff changeset
   160
      case None =>
6e042537555d incremental document changes;
wenzelm
parents: 65140
diff changeset
   161
        Text.Edit.replace(0, content.text, insert) match {
6e042537555d incremental document changes;
wenzelm
parents: 65140
diff changeset
   162
          case Nil => None
6e042537555d incremental document changes;
wenzelm
parents: 65140
diff changeset
   163
          case edits =>
76781
d9f48960bf23 clarified signature: more position information via node_name;
wenzelm
parents: 76778
diff changeset
   164
            val content1 = VSCode_Model.Content(node_name, Line.Document(insert))
65160
6e042537555d incremental document changes;
wenzelm
parents: 65140
diff changeset
   165
            Some(copy(content = content1, pending_edits = pending_edits ::: edits))
6e042537555d incremental document changes;
wenzelm
parents: 65140
diff changeset
   166
        }
6e042537555d incremental document changes;
wenzelm
parents: 65140
diff changeset
   167
      case Some(remove) =>
6e042537555d incremental document changes;
wenzelm
parents: 65140
diff changeset
   168
        content.doc.change(remove, insert) match {
6e042537555d incremental document changes;
wenzelm
parents: 65140
diff changeset
   169
          case None => error("Failed to apply document change: " + remove)
6e042537555d incremental document changes;
wenzelm
parents: 65140
diff changeset
   170
          case Some((Nil, _)) => None
6e042537555d incremental document changes;
wenzelm
parents: 65140
diff changeset
   171
          case Some((edits, doc1)) =>
76781
d9f48960bf23 clarified signature: more position information via node_name;
wenzelm
parents: 76778
diff changeset
   172
            val content1 = VSCode_Model.Content(node_name, doc1)
65160
6e042537555d incremental document changes;
wenzelm
parents: 65140
diff changeset
   173
            Some(copy(content = content1, pending_edits = pending_edits ::: edits))
6e042537555d incremental document changes;
wenzelm
parents: 65140
diff changeset
   174
        }
64709
5e6566ab78bf more explicit edits -- eliminated Clear;
wenzelm
parents: 64707
diff changeset
   175
    }
5e6566ab78bf more explicit edits -- eliminated Clear;
wenzelm
parents: 64707
diff changeset
   176
  }
5e6566ab78bf more explicit edits -- eliminated Clear;
wenzelm
parents: 64707
diff changeset
   177
66676
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
   178
  def flush_edits(
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
   179
      unicode_symbols: Boolean,
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
   180
      doc_blobs: Document.Blobs,
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
   181
      file: JFile,
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
   182
      caret: Option[Line.Position]
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
   183
  ): Option[((List[LSP.TextDocumentEdit], List[Document.Edit_Text]), VSCode_Model)] = {
66676
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
   184
    val workspace_edits =
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
   185
      if (unicode_symbols && version.isDefined) {
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
   186
        val edits = content.recode_symbols
72761
4519eeefe3b5 avoid conflicting base names;
wenzelm
parents: 71774
diff changeset
   187
        if (edits.nonEmpty) List(LSP.TextDocumentEdit(file, version.get, edits))
66676
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
   188
        else Nil
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
   189
      }
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
   190
      else Nil
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
   191
65926
0f7821a07aa9 restricted perspective depending on the caret -- important for reactivity when editing big files;
wenzelm
parents: 65923
diff changeset
   192
    val (reparse, perspective) = node_perspective(doc_blobs, caret)
66676
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
   193
    if (reparse || pending_edits.nonEmpty || last_perspective != perspective ||
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
   194
        workspace_edits.nonEmpty) {
66676
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
   195
      val prover_edits = node_edits(node_header, pending_edits, perspective)
39db5bb7eb0a recode Unicode text on the spot, e.g. from copy-paste of output;
wenzelm
parents: 66674
diff changeset
   196
      val edits = (workspace_edits, prover_edits)
64816
wenzelm
parents: 64815
diff changeset
   197
      Some((edits, copy(pending_edits = Nil, last_perspective = perspective)))
64605
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
   198
    }
64707
7157685b71e3 clarified Document_Model perspective and edits;
wenzelm
parents: 64704
diff changeset
   199
    else None
7157685b71e3 clarified Document_Model perspective and edits;
wenzelm
parents: 64704
diff changeset
   200
  }
64605
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
   201
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
   202
65095
eb21a4f70b0e publish decorations like diagnostics;
wenzelm
parents: 64877
diff changeset
   203
  /* publish annotations */
64679
b2bf280b7e13 more uniform treatment of input/output wrt. client;
wenzelm
parents: 64673
diff changeset
   204
75393
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
   205
  def publish(
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
   206
    rendering: VSCode_Rendering
87ebf5a50283 clarified formatting, for the sake of scala3;
wenzelm
parents: 72772
diff changeset
   207
  ): (Option[List[Text.Info[Command.Results]]], Option[List[VSCode_Model.Decoration]], VSCode_Model) = {
81043
2174ec5f0d0c lsp: added decoration_request notification;
Thomas Lindae <thomas.lindae@in.tum.de>
parents: 76904
diff changeset
   208
    val (diagnostics, decorations, model) = publish_full(rendering)
2174ec5f0d0c lsp: added decoration_request notification;
Thomas Lindae <thomas.lindae@in.tum.de>
parents: 76904
diff changeset
   209
    
65115
93a0683e075a publish output more thoroughly;
wenzelm
parents: 65111
diff changeset
   210
    val changed_diagnostics =
93a0683e075a publish output more thoroughly;
wenzelm
parents: 65111
diff changeset
   211
      if (diagnostics == published_diagnostics) None else Some(diagnostics)
93a0683e075a publish output more thoroughly;
wenzelm
parents: 65111
diff changeset
   212
    val changed_decorations =
93a0683e075a publish output more thoroughly;
wenzelm
parents: 65111
diff changeset
   213
      if (decorations == published_decorations) None
65120
db6cc23fcf33 proper reset of published decorations: initial value is Nil, afterwards it is a list of canonical length and order;
wenzelm
parents: 65115
diff changeset
   214
      else if (published_decorations.isEmpty) Some(decorations)
65115
93a0683e075a publish output more thoroughly;
wenzelm
parents: 65111
diff changeset
   215
      else Some(for { (a, b) <- decorations zip published_decorations if a != b } yield a)
93a0683e075a publish output more thoroughly;
wenzelm
parents: 65111
diff changeset
   216
81043
2174ec5f0d0c lsp: added decoration_request notification;
Thomas Lindae <thomas.lindae@in.tum.de>
parents: 76904
diff changeset
   217
    (changed_diagnostics, changed_decorations, model)
2174ec5f0d0c lsp: added decoration_request notification;
Thomas Lindae <thomas.lindae@in.tum.de>
parents: 76904
diff changeset
   218
  }
2174ec5f0d0c lsp: added decoration_request notification;
Thomas Lindae <thomas.lindae@in.tum.de>
parents: 76904
diff changeset
   219
2174ec5f0d0c lsp: added decoration_request notification;
Thomas Lindae <thomas.lindae@in.tum.de>
parents: 76904
diff changeset
   220
  def publish_full(
2174ec5f0d0c lsp: added decoration_request notification;
Thomas Lindae <thomas.lindae@in.tum.de>
parents: 76904
diff changeset
   221
    rendering: VSCode_Rendering
2174ec5f0d0c lsp: added decoration_request notification;
Thomas Lindae <thomas.lindae@in.tum.de>
parents: 76904
diff changeset
   222
  ): (List[Text.Info[Command.Results]],List[VSCode_Model.Decoration], VSCode_Model) = {
2174ec5f0d0c lsp: added decoration_request notification;
Thomas Lindae <thomas.lindae@in.tum.de>
parents: 76904
diff changeset
   223
    val diagnostics = rendering.diagnostics
2174ec5f0d0c lsp: added decoration_request notification;
Thomas Lindae <thomas.lindae@in.tum.de>
parents: 76904
diff changeset
   224
    val decorations =
2174ec5f0d0c lsp: added decoration_request notification;
Thomas Lindae <thomas.lindae@in.tum.de>
parents: 76904
diff changeset
   225
      if (node_visible) rendering.decorations
2174ec5f0d0c lsp: added decoration_request notification;
Thomas Lindae <thomas.lindae@in.tum.de>
parents: 76904
diff changeset
   226
      else { for (deco <- published_decorations) yield VSCode_Model.Decoration.empty(deco.typ) }
2174ec5f0d0c lsp: added decoration_request notification;
Thomas Lindae <thomas.lindae@in.tum.de>
parents: 76904
diff changeset
   227
2174ec5f0d0c lsp: added decoration_request notification;
Thomas Lindae <thomas.lindae@in.tum.de>
parents: 76904
diff changeset
   228
    (diagnostics, decorations,
65115
93a0683e075a publish output more thoroughly;
wenzelm
parents: 65111
diff changeset
   229
      copy(published_diagnostics = diagnostics, published_decorations = decorations))
64679
b2bf280b7e13 more uniform treatment of input/output wrt. client;
wenzelm
parents: 64673
diff changeset
   230
  }
b2bf280b7e13 more uniform treatment of input/output wrt. client;
wenzelm
parents: 64673
diff changeset
   231
b2bf280b7e13 more uniform treatment of input/output wrt. client;
wenzelm
parents: 64673
diff changeset
   232
64725
wenzelm
parents: 64723
diff changeset
   233
  /* prover session */
64702
d95b9117cb5b re-use resources from session;
wenzelm
parents: 64690
diff changeset
   234
d95b9117cb5b re-use resources from session;
wenzelm
parents: 64690
diff changeset
   235
  def resources: VSCode_Resources = session.resources.asInstanceOf[VSCode_Resources]
64605
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
   236
64816
wenzelm
parents: 64815
diff changeset
   237
  def is_stable: Boolean = pending_edits.isEmpty
66054
fb0eea226a4d more uniform syntax_completion + semantic_completion;
wenzelm
parents: 65926
diff changeset
   238
fb0eea226a4d more uniform syntax_completion + semantic_completion;
wenzelm
parents: 65926
diff changeset
   239
fb0eea226a4d more uniform syntax_completion + semantic_completion;
wenzelm
parents: 65926
diff changeset
   240
  /* syntax */
fb0eea226a4d more uniform syntax_completion + semantic_completion;
wenzelm
parents: 65926
diff changeset
   241
fb0eea226a4d more uniform syntax_completion + semantic_completion;
wenzelm
parents: 65926
diff changeset
   242
  def syntax(): Outer_Syntax =
67004
af72fa58f71b clarified lazy Completion within Outer_Syntax: measurable speedup of Sessions.deps;
wenzelm
parents: 66984
diff changeset
   243
    if (is_theory) session.recent_syntax(node_name) else Outer_Syntax.empty
64605
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
   244
}