src/Tools/VSCode/src/vscode_resources.scala
author wenzelm
Mon, 03 Apr 2017 17:00:36 +0200
changeset 65361 ecefb68dc21d
parent 65359 9ca34f0407a9
child 65441 9425e4d8bdb6
permissions -rw-r--r--
tuned signature;
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
64623
83f012ce2567 clarified module name;
wenzelm
parents: 64605
diff changeset
     1
/*  Title:      Tools/VSCode/src/vscode_resources.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
64729
4eccd9bc5fd9 clarified file URI operations;
wenzelm
parents: 64727
diff changeset
     4
Resources for VSCode Language Server: file-system access and global state.
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
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
    12
import java.io.{File => JFile}
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
    13
64824
330ec9bc4b75 tuned signature;
wenzelm
parents: 64821
diff changeset
    14
import scala.util.parsing.input.Reader
64727
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
    15
64605
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
    16
64623
83f012ce2567 clarified module name;
wenzelm
parents: 64605
diff changeset
    17
object VSCode_Resources
64605
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
    18
{
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
    19
  /* internal state */
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
    20
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
    21
  sealed case class State(
64777
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
    22
    models: Map[JFile, Document_Model] = Map.empty,
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
    23
    pending_input: Set[JFile] = Set.empty,
65189
41d2452845fc support for caret handling and dynamic output;
wenzelm
parents: 65160
diff changeset
    24
    pending_output: Set[JFile] = Set.empty,
41d2452845fc support for caret handling and dynamic output;
wenzelm
parents: 65160
diff changeset
    25
    caret: Option[(JFile, Line.Position)] = None)
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
    26
  {
65113
wenzelm
parents: 65112
diff changeset
    27
    def update_models(changed: Traversable[(JFile, Document_Model)]): State =
65111
3224a6f7bd6b more robust treatment of pending input/output: these are often correlated;
wenzelm
parents: 65107
diff changeset
    28
      copy(
65113
wenzelm
parents: 65112
diff changeset
    29
        models = models ++ changed,
wenzelm
parents: 65112
diff changeset
    30
        pending_input = (pending_input /: changed) { case (set, (file, _)) => set + file },
wenzelm
parents: 65112
diff changeset
    31
        pending_output = (pending_output /: changed) { case (set, (file, _)) => set + file })
65111
3224a6f7bd6b more robust treatment of pending input/output: these are often correlated;
wenzelm
parents: 65107
diff changeset
    32
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
    33
    lazy val document_blobs: Document.Blobs =
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
    34
      Document.Blobs(
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
    35
        (for {
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
    36
          (_, model) <- models.iterator
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
    37
          blob <- model.get_blob
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
    38
        } yield (model.node_name -> blob)).toMap)
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
    39
  }
64605
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
    40
}
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
    41
64623
83f012ce2567 clarified module name;
wenzelm
parents: 64605
diff changeset
    42
class VSCode_Resources(
65361
ecefb68dc21d tuned signature;
wenzelm
parents: 65359
diff changeset
    43
    val options: Options,
ecefb68dc21d tuned signature;
wenzelm
parents: 65359
diff changeset
    44
    session_base: Sessions.Base,
ecefb68dc21d tuned signature;
wenzelm
parents: 65359
diff changeset
    45
    log: Logger = No_Logger)
ecefb68dc21d tuned signature;
wenzelm
parents: 65359
diff changeset
    46
  extends Resources(session_name = "", session_base, log)
64605
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
    47
{
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
    48
  private val state = Synchronized(VSCode_Resources.State())
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
    49
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
    50
65137
812c35fbffa8 clarified options;
wenzelm
parents: 65132
diff changeset
    51
  /* options */
812c35fbffa8 clarified options;
wenzelm
parents: 65132
diff changeset
    52
812c35fbffa8 clarified options;
wenzelm
parents: 65132
diff changeset
    53
  def pide_extensions: Boolean = options.bool("vscode_pide_extensions")
812c35fbffa8 clarified options;
wenzelm
parents: 65132
diff changeset
    54
  def unicode_symbols: Boolean = options.bool("vscode_unicode_symbols")
812c35fbffa8 clarified options;
wenzelm
parents: 65132
diff changeset
    55
  def tooltip_margin: Int = options.int("vscode_tooltip_margin")
812c35fbffa8 clarified options;
wenzelm
parents: 65132
diff changeset
    56
  def message_margin: Int = options.int("vscode_message_margin")
812c35fbffa8 clarified options;
wenzelm
parents: 65132
diff changeset
    57
812c35fbffa8 clarified options;
wenzelm
parents: 65132
diff changeset
    58
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
    59
  /* document node name */
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
    60
64777
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
    61
  def node_file(name: Document.Node.Name): JFile = new JFile(name.node)
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
    62
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
    63
  def node_name(file: JFile): Document.Node.Name =
64605
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
    64
  {
64777
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
    65
    val node = file.getPath
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
    66
    val theory = Thy_Header.thy_name_bootstrap(node).getOrElse("")
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
    67
    val master_dir = if (theory == "") "" else file.getParent
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
    68
    Document.Node.Name(node, master_dir, theory)
64605
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
    69
  }
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
    70
64759
100941134718 clarified master_dir: file-URL;
wenzelm
parents: 64754
diff changeset
    71
  override def append(dir: String, source_path: Path): String =
64716
473793d52d97 proper import_name;
wenzelm
parents: 64710
diff changeset
    72
  {
64759
100941134718 clarified master_dir: file-URL;
wenzelm
parents: 64754
diff changeset
    73
    val path = source_path.expand
64777
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
    74
    if (dir == "" || path.is_absolute) File.platform_path(path)
64759
100941134718 clarified master_dir: file-URL;
wenzelm
parents: 64754
diff changeset
    75
    else if (path.is_current) dir
64777
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
    76
    else if (path.is_basic && !dir.endsWith("/") && !dir.endsWith(JFile.separator))
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
    77
      dir + JFile.separator + File.platform_path(path)
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
    78
    else if (path.is_basic) dir + File.platform_path(path)
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
    79
    else new JFile(dir + JFile.separator + File.platform_path(path)).getCanonicalPath
64716
473793d52d97 proper import_name;
wenzelm
parents: 64710
diff changeset
    80
  }
473793d52d97 proper import_name;
wenzelm
parents: 64710
diff changeset
    81
64834
wenzelm
parents: 64833
diff changeset
    82
  def get_model(file: JFile): Option[Document_Model] = state.value.models.get(file)
wenzelm
parents: 64833
diff changeset
    83
  def get_model(name: Document.Node.Name): Option[Document_Model] = get_model(node_file(name))
wenzelm
parents: 64833
diff changeset
    84
wenzelm
parents: 64833
diff changeset
    85
wenzelm
parents: 64833
diff changeset
    86
  /* file content */
wenzelm
parents: 64833
diff changeset
    87
wenzelm
parents: 64833
diff changeset
    88
  def read_file_content(file: JFile): Option[String] =
wenzelm
parents: 64833
diff changeset
    89
    try { Some(Line.normalize(File.read(file))) }
wenzelm
parents: 64833
diff changeset
    90
    catch { case ERROR(_) => None }
wenzelm
parents: 64833
diff changeset
    91
wenzelm
parents: 64833
diff changeset
    92
  def get_file_content(file: JFile): Option[String] =
wenzelm
parents: 64833
diff changeset
    93
    get_model(file) match {
wenzelm
parents: 64833
diff changeset
    94
      case Some(model) => Some(model.content.text)
wenzelm
parents: 64833
diff changeset
    95
      case None => read_file_content(file)
wenzelm
parents: 64833
diff changeset
    96
    }
wenzelm
parents: 64833
diff changeset
    97
wenzelm
parents: 64833
diff changeset
    98
  def bibtex_entries_iterator(): Iterator[Text.Info[(String, Document_Model)]] =
wenzelm
parents: 64833
diff changeset
    99
    for {
wenzelm
parents: 64833
diff changeset
   100
      (_, model) <- state.value.models.iterator
65132
wenzelm
parents: 65118
diff changeset
   101
      info <- model.content.bibtex_entries.iterator
wenzelm
parents: 65118
diff changeset
   102
    } yield info.map((_, model))
64834
wenzelm
parents: 64833
diff changeset
   103
64727
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   104
  override def with_thy_reader[A](name: Document.Node.Name, f: Reader[Char] => A): A =
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   105
  {
64777
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
   106
    val file = node_file(name)
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
   107
    get_model(file) match {
64830
9bc44bef99e6 more explocit Document_Model.Content;
wenzelm
parents: 64824
diff changeset
   108
      case Some(model) => f(Scan.char_reader(model.content.text))
64777
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
   109
      case None if file.isFile =>
64727
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   110
        val reader = Scan.byte_reader(file)
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   111
        try { f(reader) } finally { reader.close }
64777
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
   112
      case None =>
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
   113
        error("No such file: " + quote(file.toString))
64727
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   114
    }
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   115
  }
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   116
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   117
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   118
  /* document models */
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   119
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   120
  def visible_node(name: Document.Node.Name): Boolean =
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   121
    get_model(name) match {
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   122
      case Some(model) => model.node_visible
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   123
      case None => false
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   124
    }
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   125
65160
6e042537555d incremental document changes;
wenzelm
parents: 65142
diff changeset
   126
  def change_model(session: Session, file: JFile, text: String, range: Option[Line.Range] = None)
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   127
  {
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   128
    state.change(st =>
64709
5e6566ab78bf more explicit edits -- eliminated Clear;
wenzelm
parents: 64708
diff changeset
   129
      {
64777
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
   130
        val model = st.models.getOrElse(file, Document_Model.init(session, node_name(file)))
65160
6e042537555d incremental document changes;
wenzelm
parents: 65142
diff changeset
   131
        val model1 = (model.change_text(text, range) getOrElse model).external(false)
65114
200b787ceb51 potentially redundant pending_output, for the sake of uniformity and reactivity;
wenzelm
parents: 65113
diff changeset
   132
        st.update_models(Some(file -> model1))
64709
5e6566ab78bf more explicit edits -- eliminated Clear;
wenzelm
parents: 64708
diff changeset
   133
      })
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   134
  }
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   135
65111
3224a6f7bd6b more robust treatment of pending input/output: these are often correlated;
wenzelm
parents: 65107
diff changeset
   136
  def close_model(file: JFile): Boolean =
64710
72ca4e5f976e manage changes of external files;
wenzelm
parents: 64709
diff changeset
   137
    state.change_result(st =>
64777
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
   138
      st.models.get(file) match {
65111
3224a6f7bd6b more robust treatment of pending input/output: these are often correlated;
wenzelm
parents: 65107
diff changeset
   139
        case None => (false, st)
65113
wenzelm
parents: 65112
diff changeset
   140
        case Some(model) => (true, st.update_models(Some(file -> model.external(true))))
64710
72ca4e5f976e manage changes of external files;
wenzelm
parents: 64709
diff changeset
   141
      })
72ca4e5f976e manage changes of external files;
wenzelm
parents: 64709
diff changeset
   142
65116
06d9bcb66ef3 simplified;
wenzelm
parents: 65115
diff changeset
   143
  def sync_models(changed_files: Set[JFile]): Unit =
06d9bcb66ef3 simplified;
wenzelm
parents: 65115
diff changeset
   144
    state.change(st =>
64710
72ca4e5f976e manage changes of external files;
wenzelm
parents: 64709
diff changeset
   145
      {
64719
wenzelm
parents: 64718
diff changeset
   146
        val changed_models =
64710
72ca4e5f976e manage changes of external files;
wenzelm
parents: 64709
diff changeset
   147
          (for {
64777
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
   148
            (file, model) <- st.models.iterator
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
   149
            if changed_files(file) && model.external_file
64812
ddbb89e7621d tuned signature;
wenzelm
parents: 64806
diff changeset
   150
            text <- read_file_content(file)
65160
6e042537555d incremental document changes;
wenzelm
parents: 65142
diff changeset
   151
            model1 <- model.change_text(text)
65116
06d9bcb66ef3 simplified;
wenzelm
parents: 65115
diff changeset
   152
          } yield (file, model1)).toList
06d9bcb66ef3 simplified;
wenzelm
parents: 65115
diff changeset
   153
        st.update_models(changed_models)
64710
72ca4e5f976e manage changes of external files;
wenzelm
parents: 64709
diff changeset
   154
      })
72ca4e5f976e manage changes of external files;
wenzelm
parents: 64709
diff changeset
   155
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   156
64727
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   157
  /* resolve dependencies */
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   158
64857
wenzelm
parents: 64856
diff changeset
   159
  def resolve_dependencies(session: Session, file_watcher: File_Watcher): (Boolean, Boolean) =
64727
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   160
  {
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   161
    state.change_result(st =>
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   162
      {
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   163
        /* theory files */
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   164
64731
84192ecae582 just one synchronized access to global state: works recursively on JVM;
wenzelm
parents: 64729
diff changeset
   165
        val thys =
84192ecae582 just one synchronized access to global state: works recursively on JVM;
wenzelm
parents: 64729
diff changeset
   166
          (for ((_, model) <- st.models.iterator if model.is_theory)
84192ecae582 just one synchronized access to global state: works recursively on JVM;
wenzelm
parents: 64729
diff changeset
   167
           yield (model.node_name, Position.none)).toList
84192ecae582 just one synchronized access to global state: works recursively on JVM;
wenzelm
parents: 64729
diff changeset
   168
65359
9ca34f0407a9 provide session qualifier via resources;
wenzelm
parents: 65232
diff changeset
   169
        val thy_files = thy_info.dependencies(thys).deps.map(_.name)
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   170
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   171
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   172
        /* auxiliary files */
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   173
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   174
        val stable_tip_version =
64815
wenzelm
parents: 64812
diff changeset
   175
          if (st.models.forall(entry => entry._2.is_stable))
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   176
            session.current_state().stable_tip_version
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   177
          else None
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   178
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   179
        val aux_files =
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   180
          stable_tip_version match {
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   181
            case Some(version) => undefined_blobs(version.nodes)
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   182
            case None => Nil
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   183
          }
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   184
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   185
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   186
        /* loaded models */
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   187
64727
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   188
        val loaded_models =
64731
84192ecae582 just one synchronized access to global state: works recursively on JVM;
wenzelm
parents: 64729
diff changeset
   189
          (for {
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   190
            node_name <- thy_files.iterator ++ aux_files.iterator
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   191
            file = node_file(node_name)
64777
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
   192
            if !st.models.isDefinedAt(file)
64857
wenzelm
parents: 64856
diff changeset
   193
            text <- { file_watcher.register_parent(file); read_file_content(file) }
64727
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   194
          }
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   195
          yield {
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   196
            val model = Document_Model.init(session, node_name)
65160
6e042537555d incremental document changes;
wenzelm
parents: 65142
diff changeset
   197
            val model1 = (model.change_text(text) getOrElse model).external(true)
64777
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
   198
            (file, model1)
65117
wenzelm
parents: 65116
diff changeset
   199
          }).toList
64731
84192ecae582 just one synchronized access to global state: works recursively on JVM;
wenzelm
parents: 64729
diff changeset
   200
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   201
        val invoke_input = loaded_models.nonEmpty
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   202
        val invoke_load = stable_tip_version.isEmpty
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   203
65113
wenzelm
parents: 65112
diff changeset
   204
        ((invoke_input, invoke_load), st.update_models(loaded_models))
64727
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   205
      })
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   206
  }
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   207
13e37567a0d6 automatically resolve dependencies from document models and file-system;
wenzelm
parents: 64726
diff changeset
   208
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   209
  /* pending input */
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   210
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   211
  def flush_input(session: Session)
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   212
  {
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   213
    state.change(st =>
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   214
      {
64719
wenzelm
parents: 64718
diff changeset
   215
        val changed_models =
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   216
          (for {
64777
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
   217
            file <- st.pending_input.iterator
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
   218
            model <- st.models.get(file)
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   219
            (edits, model1) <- model.flush_edits(st.document_blobs)
64777
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
   220
          } yield (edits, (file, model1))).toList
64707
7157685b71e3 clarified Document_Model perspective and edits;
wenzelm
parents: 64706
diff changeset
   221
64800
415dafeb9669 manage document blobs as well;
wenzelm
parents: 64796
diff changeset
   222
        session.update(st.document_blobs, changed_models.flatMap(_._1))
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   223
        st.copy(
65112
wenzelm
parents: 65111
diff changeset
   224
          models = st.models ++ changed_models.iterator.map(_._2),
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   225
          pending_input = Set.empty)
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   226
      })
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   227
  }
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   228
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   229
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   230
  /* pending output */
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   231
65118
31fd8e41be02 tuned signature;
wenzelm
parents: 65117
diff changeset
   232
  def update_output(changed_nodes: Traversable[JFile]): Unit =
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   233
    state.change(st => st.copy(pending_output = st.pending_output ++ changed_nodes))
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   234
65232
ca571c8c0788 avoid race condition between current_state().stable_tip_version and model.rendering();
wenzelm
parents: 65198
diff changeset
   235
  def flush_output(channel: Channel): Boolean =
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   236
  {
65232
ca571c8c0788 avoid race condition between current_state().stable_tip_version and model.rendering();
wenzelm
parents: 65198
diff changeset
   237
    state.change_result(st =>
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   238
      {
65232
ca571c8c0788 avoid race condition between current_state().stable_tip_version and model.rendering();
wenzelm
parents: 65198
diff changeset
   239
        val (postponed, flushed) =
ca571c8c0788 avoid race condition between current_state().stable_tip_version and model.rendering();
wenzelm
parents: 65198
diff changeset
   240
          (for {
ca571c8c0788 avoid race condition between current_state().stable_tip_version and model.rendering();
wenzelm
parents: 65198
diff changeset
   241
            file <- st.pending_output.iterator
ca571c8c0788 avoid race condition between current_state().stable_tip_version and model.rendering();
wenzelm
parents: 65198
diff changeset
   242
            model <- st.models.get(file)
ca571c8c0788 avoid race condition between current_state().stable_tip_version and model.rendering();
wenzelm
parents: 65198
diff changeset
   243
          } yield (file, model, model.rendering())).toList.partition(_._3.snapshot.is_outdated)
ca571c8c0788 avoid race condition between current_state().stable_tip_version and model.rendering();
wenzelm
parents: 65198
diff changeset
   244
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   245
        val changed_iterator =
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   246
          for {
65232
ca571c8c0788 avoid race condition between current_state().stable_tip_version and model.rendering();
wenzelm
parents: 65198
diff changeset
   247
            (file, model, rendering) <- flushed.iterator
65115
93a0683e075a publish output more thoroughly;
wenzelm
parents: 65114
diff changeset
   248
            (changed_diags, changed_decos, model1) = model.publish(rendering)
93a0683e075a publish output more thoroughly;
wenzelm
parents: 65114
diff changeset
   249
            if changed_diags.isDefined || changed_decos.isDefined
65095
eb21a4f70b0e publish decorations like diagnostics;
wenzelm
parents: 64877
diff changeset
   250
          }
eb21a4f70b0e publish decorations like diagnostics;
wenzelm
parents: 64877
diff changeset
   251
          yield {
65115
93a0683e075a publish output more thoroughly;
wenzelm
parents: 65114
diff changeset
   252
            for (diags <- changed_diags)
93a0683e075a publish output more thoroughly;
wenzelm
parents: 65114
diff changeset
   253
              channel.write(Protocol.PublishDiagnostics(file, rendering.diagnostics_output(diags)))
65137
812c35fbffa8 clarified options;
wenzelm
parents: 65132
diff changeset
   254
            if (pide_extensions) {
812c35fbffa8 clarified options;
wenzelm
parents: 65132
diff changeset
   255
              for (decos <- changed_decos; deco <- decos)
812c35fbffa8 clarified options;
wenzelm
parents: 65132
diff changeset
   256
                channel.write(rendering.decoration_output(deco).json(file))
812c35fbffa8 clarified options;
wenzelm
parents: 65132
diff changeset
   257
            }
64777
ca09695eb43c clarified Document.Node.Name (again): canonical platform file;
wenzelm
parents: 64775
diff changeset
   258
            (file, model1)
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   259
          }
65232
ca571c8c0788 avoid race condition between current_state().stable_tip_version and model.rendering();
wenzelm
parents: 65198
diff changeset
   260
ca571c8c0788 avoid race condition between current_state().stable_tip_version and model.rendering();
wenzelm
parents: 65198
diff changeset
   261
        (postponed.nonEmpty,
ca571c8c0788 avoid race condition between current_state().stable_tip_version and model.rendering();
wenzelm
parents: 65198
diff changeset
   262
          st.copy(
ca571c8c0788 avoid race condition between current_state().stable_tip_version and model.rendering();
wenzelm
parents: 65198
diff changeset
   263
            models = st.models ++ changed_iterator,
ca571c8c0788 avoid race condition between current_state().stable_tip_version and model.rendering();
wenzelm
parents: 65198
diff changeset
   264
            pending_output = postponed.map(_._1).toSet))
64703
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   265
      }
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   266
    )
a115391494ed moved main state to VSCode_Resources;
wenzelm
parents: 64640
diff changeset
   267
  }
64870
41e2797af496 clarified text output wrt. symbols;
wenzelm
parents: 64857
diff changeset
   268
41e2797af496 clarified text output wrt. symbols;
wenzelm
parents: 64857
diff changeset
   269
41e2797af496 clarified text output wrt. symbols;
wenzelm
parents: 64857
diff changeset
   270
  /* output text */
41e2797af496 clarified text output wrt. symbols;
wenzelm
parents: 64857
diff changeset
   271
41e2797af496 clarified text output wrt. symbols;
wenzelm
parents: 64857
diff changeset
   272
  def output_text(s: String): String =
65137
812c35fbffa8 clarified options;
wenzelm
parents: 65132
diff changeset
   273
    if (unicode_symbols) Symbol.decode(s) else Symbol.encode(s)
64870
41e2797af496 clarified text output wrt. symbols;
wenzelm
parents: 64857
diff changeset
   274
65107
70b0113fa4ef clarified pretty margin;
wenzelm
parents: 65095
diff changeset
   275
  def output_xml(xml: XML.Tree): String =
70b0113fa4ef clarified pretty margin;
wenzelm
parents: 65095
diff changeset
   276
    output_text(XML.content(xml))
70b0113fa4ef clarified pretty margin;
wenzelm
parents: 65095
diff changeset
   277
64870
41e2797af496 clarified text output wrt. symbols;
wenzelm
parents: 64857
diff changeset
   278
  def output_pretty(body: XML.Body, margin: Int): String =
41e2797af496 clarified text output wrt. symbols;
wenzelm
parents: 64857
diff changeset
   279
    output_text(Pretty.string_of(body, margin))
65107
70b0113fa4ef clarified pretty margin;
wenzelm
parents: 65095
diff changeset
   280
  def output_pretty_tooltip(body: XML.Body): String = output_pretty(body, tooltip_margin)
70b0113fa4ef clarified pretty margin;
wenzelm
parents: 65095
diff changeset
   281
  def output_pretty_message(body: XML.Body): String = output_pretty(body, message_margin)
65142
368004bed323 decorations for spell-checker;
wenzelm
parents: 65137
diff changeset
   282
368004bed323 decorations for spell-checker;
wenzelm
parents: 65137
diff changeset
   283
65189
41d2452845fc support for caret handling and dynamic output;
wenzelm
parents: 65160
diff changeset
   284
  /* caret handling */
41d2452845fc support for caret handling and dynamic output;
wenzelm
parents: 65160
diff changeset
   285
65191
4c9c83311cad dynamic output, depending on caret focus (see also Tools/jEdit/src/output_dockable.scala);
wenzelm
parents: 65189
diff changeset
   286
  def update_caret(caret: Option[(JFile, Line.Position)])
4c9c83311cad dynamic output, depending on caret focus (see also Tools/jEdit/src/output_dockable.scala);
wenzelm
parents: 65189
diff changeset
   287
  { state.change(_.copy(caret = caret)) }
65189
41d2452845fc support for caret handling and dynamic output;
wenzelm
parents: 65160
diff changeset
   288
65198
76cef38242d2 clarified caret offset;
wenzelm
parents: 65197
diff changeset
   289
  def caret_offset(): Option[(Document_Model, Text.Offset)] =
65189
41d2452845fc support for caret handling and dynamic output;
wenzelm
parents: 65160
diff changeset
   290
  {
41d2452845fc support for caret handling and dynamic output;
wenzelm
parents: 65160
diff changeset
   291
    val st = state.value
41d2452845fc support for caret handling and dynamic output;
wenzelm
parents: 65160
diff changeset
   292
    for {
41d2452845fc support for caret handling and dynamic output;
wenzelm
parents: 65160
diff changeset
   293
      (file, pos) <- st.caret
41d2452845fc support for caret handling and dynamic output;
wenzelm
parents: 65160
diff changeset
   294
      model <- st.models.get(file)
65191
4c9c83311cad dynamic output, depending on caret focus (see also Tools/jEdit/src/output_dockable.scala);
wenzelm
parents: 65189
diff changeset
   295
      offset <- model.content.doc.offset(pos)
65189
41d2452845fc support for caret handling and dynamic output;
wenzelm
parents: 65160
diff changeset
   296
    }
65198
76cef38242d2 clarified caret offset;
wenzelm
parents: 65197
diff changeset
   297
    yield (model, offset)
65189
41d2452845fc support for caret handling and dynamic output;
wenzelm
parents: 65160
diff changeset
   298
  }
41d2452845fc support for caret handling and dynamic output;
wenzelm
parents: 65160
diff changeset
   299
41d2452845fc support for caret handling and dynamic output;
wenzelm
parents: 65160
diff changeset
   300
65142
368004bed323 decorations for spell-checker;
wenzelm
parents: 65137
diff changeset
   301
  /* spell checker */
368004bed323 decorations for spell-checker;
wenzelm
parents: 65137
diff changeset
   302
368004bed323 decorations for spell-checker;
wenzelm
parents: 65137
diff changeset
   303
  val spell_checker = new Spell_Checker_Variable
368004bed323 decorations for spell-checker;
wenzelm
parents: 65137
diff changeset
   304
  spell_checker.update(options)
64605
9c1173a7e4cb basic support for VSCode Language Server protocol;
wenzelm
parents:
diff changeset
   305
}