author | wenzelm |
Mon, 01 Mar 2021 19:41:52 +0100 | |
changeset 73337 | 0af9e7e4476f |
parent 73055 | 3e4df2e689ff |
child 73340 | 0ffcad1f6130 |
permissions | -rw-r--r-- |
43282
5d294220ca43
moved sources -- eliminated Netbeans artifact of jedit package directory;
wenzelm
parents:
40851
diff
changeset
|
1 |
/* Title: Tools/jEdit/src/document_model.scala |
36760 | 2 |
Author: Fabian Immler, TU Munich |
3 |
Author: Makarius |
|
4 |
||
64817 | 5 |
Document model connected to jEdit buffer or external file: content of theory |
6 |
node or auxiliary file (blob). |
|
36760 | 7 |
*/ |
34407 | 8 |
|
34784
02959dcea756
split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents:
34783
diff
changeset
|
9 |
package isabelle.jedit |
34760 | 10 |
|
34318
c13e168a8ae6
original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff
changeset
|
11 |
|
36015 | 12 |
import isabelle._ |
13 |
||
64858 | 14 |
import java.io.{File => JFile} |
15 |
||
34693 | 16 |
import scala.collection.mutable |
66019 | 17 |
import scala.annotation.tailrec |
34446 | 18 |
|
72960 | 19 |
import org.gjt.sp.jedit.View |
34784
02959dcea756
split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents:
34783
diff
changeset
|
20 |
import org.gjt.sp.jedit.Buffer |
61192 | 21 |
import org.gjt.sp.jedit.buffer.{BufferAdapter, BufferListener, JEditBuffer} |
34318
c13e168a8ae6
original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff
changeset
|
22 |
|
34760 | 23 |
|
34784
02959dcea756
split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents:
34783
diff
changeset
|
24 |
object Document_Model |
34588 | 25 |
{ |
64817 | 26 |
/* document models */ |
27 |
||
28 |
sealed case class State( |
|
29 |
models: Map[Document.Node.Name, Document_Model] = Map.empty, |
|
66101
0f0f294e314f
maintain overlays within main state of document models;
wenzelm
parents:
66082
diff
changeset
|
30 |
buffer_models: Map[JEditBuffer, Buffer_Model] = Map.empty, |
0f0f294e314f
maintain overlays within main state of document models;
wenzelm
parents:
66082
diff
changeset
|
31 |
overlays: Document.Overlays = Document.Overlays.empty) |
64817 | 32 |
{ |
64868 | 33 |
def file_models_iterator: Iterator[(Document.Node.Name, File_Model)] = |
64817 | 34 |
for { |
64868 | 35 |
(node_name, model) <- models.iterator |
64817 | 36 |
if model.isInstanceOf[File_Model] |
64868 | 37 |
} yield (node_name, model.asInstanceOf[File_Model]) |
64813 | 38 |
|
68476 | 39 |
def document_blobs: Document.Blobs = |
40 |
Document.Blobs( |
|
41 |
(for { |
|
42 |
(node_name, model) <- models.iterator |
|
43 |
blob <- model.get_blob |
|
44 |
} yield (node_name -> blob)).toMap) |
|
45 |
||
64817 | 46 |
def open_buffer(session: Session, node_name: Document.Node.Name, buffer: Buffer) |
47 |
: (Buffer_Model, State) = |
|
48 |
{ |
|
49 |
val old_model = |
|
50 |
models.get(node_name) match { |
|
51 |
case Some(file_model: File_Model) => Some(file_model) |
|
52 |
case Some(buffer_model: Buffer_Model) => Some(buffer_model.exit()) |
|
53 |
case _ => None |
|
54 |
} |
|
55 |
val buffer_model = Buffer_Model(session, node_name, buffer).init(old_model) |
|
56 |
(buffer_model, |
|
57 |
copy(models = models + (node_name -> buffer_model), |
|
58 |
buffer_models = buffer_models + (buffer -> buffer_model))) |
|
59 |
} |
|
60 |
||
61 |
def close_buffer(buffer: JEditBuffer): State = |
|
62 |
{ |
|
63 |
buffer_models.get(buffer) match { |
|
64 |
case None => this |
|
65 |
case Some(buffer_model) => |
|
66 |
val file_model = buffer_model.exit() |
|
67 |
copy(models = models + (file_model.node_name -> file_model), |
|
68 |
buffer_models = buffer_models - buffer) |
|
69 |
} |
|
70 |
} |
|
64835
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
71 |
|
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
72 |
def provide_file(session: Session, node_name: Document.Node.Name, text: String): State = |
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
73 |
if (models.isDefinedAt(node_name)) this |
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
74 |
else { |
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
75 |
val edit = Text.Edit.insert(0, text) |
64863 | 76 |
val model = File_Model.init(session, node_name, text, pending_edits = List(edit)) |
64835
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
77 |
copy(models = models + (node_name -> model)) |
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
78 |
} |
64817 | 79 |
} |
34784
02959dcea756
split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents:
34783
diff
changeset
|
80 |
|
64817 | 81 |
private val state = Synchronized(State()) // owned by GUI thread |
82 |
||
64835
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
83 |
def get_models(): Map[Document.Node.Name, Document_Model] = state.value.models |
71601 | 84 |
def get(name: Document.Node.Name): Option[Document_Model] = get_models().get(name) |
64835
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
85 |
def get(buffer: JEditBuffer): Option[Buffer_Model] = state.value.buffer_models.get(buffer) |
64817 | 86 |
|
68476 | 87 |
def document_blobs(): Document.Blobs = state.value.document_blobs |
88 |
||
66152 | 89 |
|
90 |
/* bibtex */ |
|
91 |
||
64831 | 92 |
def bibtex_entries_iterator(): Iterator[Text.Info[(String, Document_Model)]] = |
66150 | 93 |
Bibtex.entries_iterator(state.value.models) |
64829
07f209e957bc
refer to bibtex entries via general Document_Model, instead of editor buffers;
wenzelm
parents:
64828
diff
changeset
|
94 |
|
66152 | 95 |
def bibtex_completion(history: Completion.History, rendering: Rendering, caret: Text.Offset) |
96 |
: Option[Completion.Result] = |
|
97 |
Bibtex.completion(history, rendering, caret, state.value.models) |
|
98 |
||
64817 | 99 |
|
66101
0f0f294e314f
maintain overlays within main state of document models;
wenzelm
parents:
66082
diff
changeset
|
100 |
/* overlays */ |
0f0f294e314f
maintain overlays within main state of document models;
wenzelm
parents:
66082
diff
changeset
|
101 |
|
0f0f294e314f
maintain overlays within main state of document models;
wenzelm
parents:
66082
diff
changeset
|
102 |
def node_overlays(name: Document.Node.Name): Document.Node.Overlays = |
0f0f294e314f
maintain overlays within main state of document models;
wenzelm
parents:
66082
diff
changeset
|
103 |
state.value.overlays(name) |
0f0f294e314f
maintain overlays within main state of document models;
wenzelm
parents:
66082
diff
changeset
|
104 |
|
0f0f294e314f
maintain overlays within main state of document models;
wenzelm
parents:
66082
diff
changeset
|
105 |
def insert_overlay(command: Command, fn: String, args: List[String]): Unit = |
0f0f294e314f
maintain overlays within main state of document models;
wenzelm
parents:
66082
diff
changeset
|
106 |
state.change(st => st.copy(overlays = st.overlays.insert(command, fn, args))) |
0f0f294e314f
maintain overlays within main state of document models;
wenzelm
parents:
66082
diff
changeset
|
107 |
|
0f0f294e314f
maintain overlays within main state of document models;
wenzelm
parents:
66082
diff
changeset
|
108 |
def remove_overlay(command: Command, fn: String, args: List[String]): Unit = |
0f0f294e314f
maintain overlays within main state of document models;
wenzelm
parents:
66082
diff
changeset
|
109 |
state.change(st => st.copy(overlays = st.overlays.remove(command, fn, args))) |
0f0f294e314f
maintain overlays within main state of document models;
wenzelm
parents:
66082
diff
changeset
|
110 |
|
0f0f294e314f
maintain overlays within main state of document models;
wenzelm
parents:
66082
diff
changeset
|
111 |
|
64858 | 112 |
/* sync external files */ |
113 |
||
114 |
def sync_files(changed_files: Set[JFile]): Boolean = |
|
115 |
{ |
|
116 |
state.change_result(st => |
|
117 |
{ |
|
118 |
val changed_models = |
|
119 |
(for { |
|
64868 | 120 |
(node_name, model) <- st.file_models_iterator |
64864
eec7ffef0be6
accomodate very big file_models and changed_files;
wenzelm
parents:
64863
diff
changeset
|
121 |
file <- model.file if changed_files(file) |
64868 | 122 |
text <- PIDE.resources.read_file_content(node_name) |
64858 | 123 |
if model.content.text != text |
124 |
} yield { |
|
125 |
val content = Document_Model.File_Content(text) |
|
126 |
val edits = Text.Edit.replace(0, model.content.text, text) |
|
64868 | 127 |
(node_name, model.copy(content = content, pending_edits = model.pending_edits ::: edits)) |
64858 | 128 |
}).toList |
129 |
if (changed_models.isEmpty) (false, st) |
|
130 |
else (true, st.copy(models = (st.models /: changed_models)(_ + _))) |
|
131 |
}) |
|
132 |
} |
|
133 |
||
134 |
||
64836 | 135 |
/* syntax */ |
136 |
||
137 |
def syntax_changed(names: List[Document.Node.Name]) |
|
138 |
{ |
|
139 |
GUI_Thread.require {} |
|
140 |
||
141 |
val models = state.value.models |
|
142 |
for (name <- names.iterator; model <- models.get(name)) { |
|
143 |
model match { case buffer_model: Buffer_Model => buffer_model.syntax_changed() case _ => } |
|
144 |
} |
|
145 |
} |
|
146 |
||
147 |
||
64817 | 148 |
/* init and exit */ |
149 |
||
150 |
def init(session: Session, node_name: Document.Node.Name, buffer: Buffer): Buffer_Model = |
|
151 |
{ |
|
152 |
GUI_Thread.require {} |
|
153 |
state.change_result(st => |
|
154 |
st.buffer_models.get(buffer) match { |
|
155 |
case Some(buffer_model) if buffer_model.node_name == node_name => |
|
72960 | 156 |
buffer_model.init_token_marker() |
64817 | 157 |
(buffer_model, st) |
158 |
case _ => |
|
159 |
val res = st.close_buffer(buffer).open_buffer(session, node_name, buffer) |
|
72960 | 160 |
buffer.propertiesChanged() |
64817 | 161 |
res |
162 |
}) |
|
163 |
} |
|
164 |
||
34784
02959dcea756
split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents:
34783
diff
changeset
|
165 |
def exit(buffer: Buffer) |
02959dcea756
split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents:
34783
diff
changeset
|
166 |
{ |
57612
990ffb84489b
clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents:
57610
diff
changeset
|
167 |
GUI_Thread.require {} |
64813 | 168 |
state.change(st => |
64817 | 169 |
if (st.buffer_models.isDefinedAt(buffer)) { |
170 |
val res = st.close_buffer(buffer) |
|
72960 | 171 |
buffer.propertiesChanged() |
64817 | 172 |
res |
173 |
} |
|
174 |
else st) |
|
175 |
} |
|
176 |
||
64835
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
177 |
def provide_files(session: Session, files: List[(Document.Node.Name, String)]) |
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
178 |
{ |
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
179 |
GUI_Thread.require {} |
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
180 |
state.change(st => |
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
181 |
(st /: files) { case (st1, (node_name, text)) => st1.provide_file(session, node_name, text) }) |
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
182 |
} |
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
183 |
|
64817 | 184 |
|
185 |
/* required nodes */ |
|
186 |
||
187 |
def required_nodes(): Set[Document.Node.Name] = |
|
188 |
(for { |
|
64868 | 189 |
(node_name, model) <- state.value.models.iterator |
64817 | 190 |
if model.node_required |
64868 | 191 |
} yield node_name).toSet |
64817 | 192 |
|
193 |
def node_required(name: Document.Node.Name, toggle: Boolean = false, set: Boolean = false) |
|
194 |
{ |
|
195 |
GUI_Thread.require {} |
|
196 |
||
197 |
val changed = |
|
198 |
state.change_result(st => |
|
199 |
st.models.get(name) match { |
|
200 |
case None => (false, st) |
|
201 |
case Some(model) => |
|
202 |
val required = if (toggle) !model.node_required else set |
|
203 |
model match { |
|
204 |
case model1: File_Model if required != model1.node_required => |
|
205 |
(true, st.copy(models = st.models + (name -> model1.copy(node_required = required)))) |
|
206 |
case model1: Buffer_Model if required != model1.node_required => |
|
207 |
model1.set_node_required(required); (true, st) |
|
208 |
case _ => (false, st) |
|
209 |
} |
|
210 |
}) |
|
211 |
if (changed) { |
|
65243 | 212 |
PIDE.plugin.options_changed() |
66082 | 213 |
PIDE.editor.flush() |
64817 | 214 |
} |
215 |
} |
|
216 |
||
217 |
def view_node_required(view: View, toggle: Boolean = false, set: Boolean = false): Unit = |
|
218 |
Document_Model.get(view.getBuffer).foreach(model => |
|
219 |
node_required(model.node_name, toggle = toggle, set = set)) |
|
220 |
||
221 |
||
222 |
/* flushed edits */ |
|
223 |
||
64867 | 224 |
def flush_edits(hidden: Boolean, purge: Boolean): (Document.Blobs, List[Document.Edit_Text]) = |
64817 | 225 |
{ |
226 |
GUI_Thread.require {} |
|
227 |
||
228 |
state.change_result(st => |
|
229 |
{ |
|
68476 | 230 |
val doc_blobs = st.document_blobs |
64817 | 231 |
|
232 |
val buffer_edits = |
|
233 |
(for { |
|
64868 | 234 |
(_, model) <- st.buffer_models.iterator |
64817 | 235 |
edit <- model.flush_edits(doc_blobs, hidden).iterator |
236 |
} yield edit).toList |
|
237 |
||
238 |
val file_edits = |
|
239 |
(for { |
|
64868 | 240 |
(node_name, model) <- st.file_models_iterator |
64867 | 241 |
(edits, model1) <- model.flush_edits(doc_blobs, hidden) |
64868 | 242 |
} yield (edits, node_name -> model1)).toList |
64867 | 243 |
|
244 |
val model_edits = buffer_edits ::: file_edits.flatMap(_._1) |
|
245 |
||
246 |
val purge_edits = |
|
247 |
if (purge) { |
|
248 |
val purged = |
|
64868 | 249 |
(for ((node_name, model) <- st.file_models_iterator) |
250 |
yield (node_name -> model.purge_edits(doc_blobs))).toList |
|
64817 | 251 |
|
64867 | 252 |
val imports = |
253 |
{ |
|
64868 | 254 |
val open_nodes = |
255 |
(for ((_, model) <- st.buffer_models.iterator) yield model.node_name).toList |
|
64867 | 256 |
val touched_nodes = model_edits.map(_._1) |
257 |
val pending_nodes = for ((node_name, None) <- purged) yield node_name |
|
258 |
(open_nodes ::: touched_nodes ::: pending_nodes).map((_, Position.none)) |
|
259 |
} |
|
67059 | 260 |
val retain = PIDE.resources.dependencies(imports).theories.toSet |
64817 | 261 |
|
67934 | 262 |
for ((node_name, Some(edits)) <- purged if !retain(node_name); edit <- edits) |
64867 | 263 |
yield edit |
264 |
} |
|
265 |
else Nil |
|
266 |
||
267 |
val st1 = st.copy(models = st.models ++ file_edits.map(_._2) -- purge_edits.map(_._1)) |
|
65245 | 268 |
PIDE.plugin.file_watcher.purge( |
64867 | 269 |
(for { |
64868 | 270 |
(_, model) <- st1.file_models_iterator |
64867 | 271 |
file <- model.file |
272 |
} yield file.getParentFile).toSet) |
|
273 |
||
274 |
((doc_blobs, model_edits ::: purge_edits), st1) |
|
64813 | 275 |
}) |
34318
c13e168a8ae6
original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff
changeset
|
276 |
} |
43397 | 277 |
|
64817 | 278 |
|
279 |
/* file content */ |
|
280 |
||
281 |
sealed case class File_Content(text: String) |
|
282 |
{ |
|
64835
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
283 |
lazy val bytes: Bytes = Bytes(Symbol.encode(text)) |
64817 | 284 |
lazy val chunk: Symbol.Text_Chunk = Symbol.Text_Chunk(text) |
64831 | 285 |
lazy val bibtex_entries: List[Text.Info[String]] = |
66150 | 286 |
try { Bibtex.entries(text) } |
67288
aa9d28034945
avoid spurious noise: Prover IDE is meant to check for errors;
wenzelm
parents:
67265
diff
changeset
|
287 |
catch { case ERROR(_) => Nil } |
64817 | 288 |
} |
66019 | 289 |
|
290 |
||
291 |
/* HTTP preview */ |
|
292 |
||
67262 | 293 |
private val plain_text_prefix = "plain_text=" |
294 |
||
295 |
def open_preview(view: View, plain_text: Boolean) |
|
66019 | 296 |
{ |
297 |
Document_Model.get(view.getBuffer) match { |
|
67246
4cedf44f2af1
isabelle.preview presents auxiliary text files as well;
wenzelm
parents:
67244
diff
changeset
|
298 |
case Some(model) => |
4cedf44f2af1
isabelle.preview presents auxiliary text files as well;
wenzelm
parents:
67244
diff
changeset
|
299 |
val name = model.node_name |
4cedf44f2af1
isabelle.preview presents auxiliary text files as well;
wenzelm
parents:
67244
diff
changeset
|
300 |
val url = |
72960 | 301 |
PIDE.plugin.http_server.url + PIDE.plugin.http_root + "/preview?" + |
67262 | 302 |
(if (plain_text) plain_text_prefix else "") + Url.encode(name.node) |
67246
4cedf44f2af1
isabelle.preview presents auxiliary text files as well;
wenzelm
parents:
67244
diff
changeset
|
303 |
PIDE.editor.hyperlink_url(url).follow(view) |
66036 | 304 |
case _ => |
66019 | 305 |
} |
306 |
} |
|
307 |
||
308 |
def http_handlers(http_root: String): List[HTTP.Handler] = |
|
309 |
{ |
|
67246
4cedf44f2af1
isabelle.preview presents auxiliary text files as well;
wenzelm
parents:
67244
diff
changeset
|
310 |
val fonts_root = http_root + "/fonts" |
66019 | 311 |
val preview_root = http_root + "/preview" |
67246
4cedf44f2af1
isabelle.preview presents auxiliary text files as well;
wenzelm
parents:
67244
diff
changeset
|
312 |
|
72957 | 313 |
val html = |
66207 | 314 |
HTTP.get(preview_root, arg => |
66019 | 315 |
for { |
71601 | 316 |
query <- Library.try_unprefix(preview_root + "?", arg.uri.toString).map(Url.decode) |
67262 | 317 |
name = Library.perhaps_unprefix(plain_text_prefix, query) |
318 |
model <- get(PIDE.resources.node_name(name)) |
|
66019 | 319 |
} |
67253 | 320 |
yield { |
321 |
val snapshot = model.await_stable_snapshot() |
|
72961 | 322 |
val html_context = Presentation.html_context(fonts_url = HTML.fonts_dir(fonts_root)) |
72957 | 323 |
val document = |
73036 | 324 |
Presentation.html_document( |
73055 | 325 |
PIDE.resources, snapshot, html_context, Presentation.elements2, |
67262 | 326 |
plain_text = query.startsWith(plain_text_prefix)) |
72957 | 327 |
HTTP.Response.html(document.content) |
67253 | 328 |
}) |
66019 | 329 |
|
72957 | 330 |
List(HTTP.fonts(fonts_root), html) |
66019 | 331 |
} |
64817 | 332 |
} |
333 |
||
64829
07f209e957bc
refer to bibtex entries via general Document_Model, instead of editor buffers;
wenzelm
parents:
64828
diff
changeset
|
334 |
sealed abstract class Document_Model extends Document.Model |
64817 | 335 |
{ |
336 |
/* perspective */ |
|
337 |
||
338 |
def document_view_ranges(snapshot: Document.Snapshot): List[Text.Range] = Nil |
|
339 |
||
340 |
def node_perspective( |
|
341 |
doc_blobs: Document.Blobs, hidden: Boolean): (Boolean, Document.Node.Perspective_Text) = |
|
43397 | 342 |
{ |
57612
990ffb84489b
clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents:
57610
diff
changeset
|
343 |
GUI_Thread.require {} |
64817 | 344 |
|
345 |
if (Isabelle.continuous_checking && is_theory) { |
|
346 |
val snapshot = this.snapshot() |
|
347 |
||
348 |
val reparse = snapshot.node.load_commands_changed(doc_blobs) |
|
349 |
val perspective = |
|
350 |
if (hidden) Text.Perspective.empty |
|
351 |
else { |
|
352 |
val view_ranges = document_view_ranges(snapshot) |
|
71601 | 353 |
val load_ranges = snapshot.commands_loading_ranges(PIDE.editor.visible_node) |
64817 | 354 |
Text.Perspective(view_ranges ::: load_ranges) |
355 |
} |
|
66082 | 356 |
val overlays = PIDE.editor.node_overlays(node_name) |
64817 | 357 |
|
358 |
(reparse, Document.Node.Perspective(node_required, perspective, overlays)) |
|
359 |
} |
|
360 |
else (false, Document.Node.no_perspective_text) |
|
43397 | 361 |
} |
66019 | 362 |
|
363 |
||
364 |
/* snapshot */ |
|
365 |
||
366 |
@tailrec final def await_stable_snapshot(): Document.Snapshot = |
|
367 |
{ |
|
368 |
val snapshot = this.snapshot() |
|
369 |
if (snapshot.is_outdated) { |
|
71684 | 370 |
PIDE.options.seconds("editor_output_delay").sleep |
66019 | 371 |
await_stable_snapshot() |
372 |
} |
|
373 |
else snapshot |
|
374 |
} |
|
34318
c13e168a8ae6
original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff
changeset
|
375 |
} |
c13e168a8ae6
original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff
changeset
|
376 |
|
64863 | 377 |
object File_Model |
378 |
{ |
|
66114 | 379 |
def empty(session: Session): File_Model = |
380 |
File_Model(session, Document.Node.Name.empty, None, Document_Model.File_Content(""), |
|
381 |
false, Document.Node.no_perspective_text, Nil) |
|
382 |
||
64863 | 383 |
def init(session: Session, |
384 |
node_name: Document.Node.Name, |
|
385 |
text: String, |
|
386 |
node_required: Boolean = false, |
|
387 |
last_perspective: Document.Node.Perspective_Text = Document.Node.no_perspective_text, |
|
388 |
pending_edits: List[Text.Edit] = Nil): File_Model = |
|
389 |
{ |
|
65469 | 390 |
val file = JEdit_Lib.check_file(node_name.node) |
65245 | 391 |
file.foreach(PIDE.plugin.file_watcher.register_parent(_)) |
64863 | 392 |
|
64864
eec7ffef0be6
accomodate very big file_models and changed_files;
wenzelm
parents:
64863
diff
changeset
|
393 |
val content = Document_Model.File_Content(text) |
71733 | 394 |
val node_required1 = node_required || File_Format.registry.is_theory(node_name) |
67291
1bd9a0142d7a
bibtex theory is always required and thus eventually consolidated;
wenzelm
parents:
67288
diff
changeset
|
395 |
File_Model(session, node_name, file, content, node_required1, last_perspective, pending_edits) |
64863 | 396 |
} |
397 |
} |
|
398 |
||
64817 | 399 |
case class File_Model( |
400 |
session: Session, |
|
401 |
node_name: Document.Node.Name, |
|
64864
eec7ffef0be6
accomodate very big file_models and changed_files;
wenzelm
parents:
64863
diff
changeset
|
402 |
file: Option[JFile], |
64817 | 403 |
content: Document_Model.File_Content, |
64863 | 404 |
node_required: Boolean, |
405 |
last_perspective: Document.Node.Perspective_Text, |
|
406 |
pending_edits: List[Text.Edit]) extends Document_Model |
|
34588 | 407 |
{ |
66114 | 408 |
/* text */ |
409 |
||
67014 | 410 |
def get_text(range: Text.Range): Option[String] = |
66114 | 411 |
range.try_substring(content.text) |
412 |
||
413 |
||
64817 | 414 |
/* header */ |
415 |
||
416 |
def node_header: Document.Node.Header = |
|
417 |
PIDE.resources.special_header(node_name) getOrElse |
|
72772 | 418 |
PIDE.resources.check_thy(node_name, Scan.char_reader(content.text), strict = false) |
64817 | 419 |
|
420 |
||
421 |
/* content */ |
|
422 |
||
64829
07f209e957bc
refer to bibtex entries via general Document_Model, instead of editor buffers;
wenzelm
parents:
64828
diff
changeset
|
423 |
def node_position(offset: Text.Offset): Line.Node_Position = |
07f209e957bc
refer to bibtex entries via general Document_Model, instead of editor buffers;
wenzelm
parents:
64828
diff
changeset
|
424 |
Line.Node_Position(node_name.node, |
65196
e8760a98db78
discontinued pointless Text.Length: Javascript and Java agree in old-fashioned UTF-16;
wenzelm
parents:
65132
diff
changeset
|
425 |
Line.Position.zero.advance(content.text.substring(0, offset))) |
64829
07f209e957bc
refer to bibtex entries via general Document_Model, instead of editor buffers;
wenzelm
parents:
64828
diff
changeset
|
426 |
|
64817 | 427 |
def get_blob: Option[Document.Blob] = |
428 |
if (is_theory) None |
|
67265
f32287c95432
store full blob source for the sake of markup_to_XML;
wenzelm
parents:
67262
diff
changeset
|
429 |
else Some(Document.Blob(content.bytes, content.text, content.chunk, pending_edits.nonEmpty)) |
64817 | 430 |
|
64831 | 431 |
def bibtex_entries: List[Text.Info[String]] = |
69255
800b1ce96fce
more general support for Isabelle/PIDE file formats -- less hardwired Bibtex operations;
wenzelm
parents:
68476
diff
changeset
|
432 |
if (Bibtex.is_bibtex(node_name.node)) content.bibtex_entries else Nil |
64680 | 433 |
|
434 |
||
64817 | 435 |
/* edits */ |
436 |
||
437 |
def update_text(text: String): Option[File_Model] = |
|
438 |
Text.Edit.replace(0, content.text, text) match { |
|
439 |
case Nil => None |
|
440 |
case edits => |
|
441 |
val content1 = Document_Model.File_Content(text) |
|
442 |
val pending_edits1 = pending_edits ::: edits |
|
443 |
Some(copy(content = content1, pending_edits = pending_edits1)) |
|
444 |
} |
|
38222
dac5fa0ac971
replaced individual Document_Model history by all-inclusive one in Session;
wenzelm
parents:
38158
diff
changeset
|
445 |
|
64817 | 446 |
def flush_edits(doc_blobs: Document.Blobs, hidden: Boolean) |
447 |
: Option[(List[Document.Edit_Text], File_Model)] = |
|
448 |
{ |
|
449 |
val (reparse, perspective) = node_perspective(doc_blobs, hidden) |
|
450 |
if (reparse || pending_edits.nonEmpty || last_perspective != perspective) { |
|
64867 | 451 |
val edits = node_edits(node_header, pending_edits, perspective) |
64817 | 452 |
Some((edits, copy(last_perspective = perspective, pending_edits = Nil))) |
453 |
} |
|
454 |
else None |
|
455 |
} |
|
456 |
||
64867 | 457 |
def purge_edits(doc_blobs: Document.Blobs): Option[List[Document.Edit_Text]] = |
67310 | 458 |
if (pending_edits.nonEmpty || |
71733 | 459 |
!File_Format.registry.is_theory(node_name) && |
67310 | 460 |
(node_required || !Document.Node.is_no_perspective_text(last_perspective))) None |
64867 | 461 |
else { |
462 |
val text_edits = List(Text.Edit.remove(0, content.text)) |
|
463 |
Some(node_edits(Document.Node.no_header, text_edits, Document.Node.no_perspective_text)) |
|
464 |
} |
|
465 |
||
64817 | 466 |
|
467 |
/* snapshot */ |
|
468 |
||
469 |
def is_stable: Boolean = pending_edits.isEmpty |
|
64818
67a0a563d2b3
clarified buffer events: exit model while loading;
wenzelm
parents:
64817
diff
changeset
|
470 |
def snapshot(): Document.Snapshot = session.snapshot(node_name, pending_edits) |
64817 | 471 |
} |
472 |
||
473 |
case class Buffer_Model(session: Session, node_name: Document.Node.Name, buffer: Buffer) |
|
474 |
extends Document_Model |
|
475 |
{ |
|
66114 | 476 |
/* text */ |
477 |
||
67014 | 478 |
def get_text(range: Text.Range): Option[String] = |
479 |
JEdit_Lib.get_text(buffer, range) |
|
66114 | 480 |
|
481 |
||
64817 | 482 |
/* header */ |
54509
1f77110c94ef
maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents:
54464
diff
changeset
|
483 |
|
48707
ba531af91148
simplified Document.Node.Header -- internalized errors;
wenzelm
parents:
47393
diff
changeset
|
484 |
def node_header(): Document.Node.Header = |
46920
5f44c8bea84e
more explicit indication of swing thread context;
wenzelm
parents:
46750
diff
changeset
|
485 |
{ |
57612
990ffb84489b
clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents:
57610
diff
changeset
|
486 |
GUI_Thread.require {} |
54509
1f77110c94ef
maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents:
54464
diff
changeset
|
487 |
|
64673
b5965890e54d
more uniform treatment of file name vs. theory name and special header;
wenzelm
parents:
63446
diff
changeset
|
488 |
PIDE.resources.special_header(node_name) getOrElse |
64826
c97296294f6d
clarified check_thy_reader: check node_name here;
wenzelm
parents:
64825
diff
changeset
|
489 |
JEdit_Lib.buffer_lock(buffer) { |
72772 | 490 |
PIDE.resources.check_thy(node_name, JEdit_Lib.buffer_reader(buffer), strict = false) |
46748
8f3ae4d04a2d
refined node_header -- more direct buffer access (again);
wenzelm
parents:
46740
diff
changeset
|
491 |
} |
46920
5f44c8bea84e
more explicit indication of swing thread context;
wenzelm
parents:
46750
diff
changeset
|
492 |
} |
44385
e7fdb008aa7d
propagate editor perspective through document model;
wenzelm
parents:
44379
diff
changeset
|
493 |
|
e7fdb008aa7d
propagate editor perspective through document model;
wenzelm
parents:
44379
diff
changeset
|
494 |
|
e7fdb008aa7d
propagate editor perspective through document model;
wenzelm
parents:
44379
diff
changeset
|
495 |
/* perspective */ |
e7fdb008aa7d
propagate editor perspective through document model;
wenzelm
parents:
44379
diff
changeset
|
496 |
|
57612
990ffb84489b
clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents:
57610
diff
changeset
|
497 |
// owned by GUI thread |
52816
c608e0ade554
home-grown mouse handling to pretend that the painted checkbox is actually a Swing component;
wenzelm
parents:
52815
diff
changeset
|
498 |
private var _node_required = false |
c608e0ade554
home-grown mouse handling to pretend that the painted checkbox is actually a Swing component;
wenzelm
parents:
52815
diff
changeset
|
499 |
def node_required: Boolean = _node_required |
64817 | 500 |
def set_node_required(b: Boolean) { GUI_Thread.require { _node_required = b } } |
52808
143f225e50f5
allow explicit indication of required node: full eval, no prints;
wenzelm
parents:
52807
diff
changeset
|
501 |
|
64883 | 502 |
def document_view_iterator: Iterator[Document_View] = |
503 |
for { |
|
504 |
text_area <- JEdit_Lib.jedit_text_areas(buffer) |
|
505 |
doc_view <- Document_View.get(text_area) |
|
506 |
} yield doc_view |
|
507 |
||
64817 | 508 |
override def document_view_ranges(snapshot: Document.Snapshot): List[Text.Range] = |
44385
e7fdb008aa7d
propagate editor perspective through document model;
wenzelm
parents:
44379
diff
changeset
|
509 |
{ |
57612
990ffb84489b
clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents:
57610
diff
changeset
|
510 |
GUI_Thread.require {} |
52759
a20631db9c8a
support declarative editor_execution_range, instead of old-style check/cancel buttons;
wenzelm
parents:
50565
diff
changeset
|
511 |
|
64817 | 512 |
(for { |
64883 | 513 |
doc_view <- document_view_iterator |
64817 | 514 |
range <- doc_view.perspective(snapshot).ranges.iterator |
515 |
} yield range).toList |
|
44385
e7fdb008aa7d
propagate editor perspective through document model;
wenzelm
parents:
44379
diff
changeset
|
516 |
} |
e7fdb008aa7d
propagate editor perspective through document model;
wenzelm
parents:
44379
diff
changeset
|
517 |
|
e7fdb008aa7d
propagate editor perspective through document model;
wenzelm
parents:
44379
diff
changeset
|
518 |
|
54509
1f77110c94ef
maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents:
54464
diff
changeset
|
519 |
/* blob */ |
1f77110c94ef
maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents:
54464
diff
changeset
|
520 |
|
64885 | 521 |
// owned by GUI thread |
67265
f32287c95432
store full blob source for the sake of markup_to_XML;
wenzelm
parents:
67262
diff
changeset
|
522 |
private var _blob: Option[(Bytes, String, Symbol.Text_Chunk)] = None |
54511 | 523 |
|
57612
990ffb84489b
clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents:
57610
diff
changeset
|
524 |
private def reset_blob(): Unit = GUI_Thread.require { _blob = None } |
54511 | 525 |
|
64817 | 526 |
def get_blob: Option[Document.Blob] = |
57612
990ffb84489b
clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents:
57610
diff
changeset
|
527 |
GUI_Thread.require { |
55783 | 528 |
if (is_theory) None |
529 |
else { |
|
67265
f32287c95432
store full blob source for the sake of markup_to_XML;
wenzelm
parents:
67262
diff
changeset
|
530 |
val (bytes, text, chunk) = |
55783 | 531 |
_blob match { |
532 |
case Some(x) => x |
|
533 |
case None => |
|
64835
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
534 |
val bytes = PIDE.resources.make_file_content(buffer) |
67265
f32287c95432
store full blob source for the sake of markup_to_XML;
wenzelm
parents:
67262
diff
changeset
|
535 |
val text = buffer.getText(0, buffer.getLength) |
f32287c95432
store full blob source for the sake of markup_to_XML;
wenzelm
parents:
67262
diff
changeset
|
536 |
val chunk = Symbol.Text_Chunk(text) |
f32287c95432
store full blob source for the sake of markup_to_XML;
wenzelm
parents:
67262
diff
changeset
|
537 |
val x = (bytes, text, chunk) |
f32287c95432
store full blob source for the sake of markup_to_XML;
wenzelm
parents:
67262
diff
changeset
|
538 |
_blob = Some(x) |
f32287c95432
store full blob source for the sake of markup_to_XML;
wenzelm
parents:
67262
diff
changeset
|
539 |
x |
55783 | 540 |
} |
64818
67a0a563d2b3
clarified buffer events: exit model while loading;
wenzelm
parents:
64817
diff
changeset
|
541 |
val changed = pending_edits.nonEmpty |
67265
f32287c95432
store full blob source for the sake of markup_to_XML;
wenzelm
parents:
67262
diff
changeset
|
542 |
Some(Document.Blob(bytes, text, chunk, changed)) |
54511 | 543 |
} |
544 |
} |
|
54509
1f77110c94ef
maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents:
54464
diff
changeset
|
545 |
|
1f77110c94ef
maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents:
54464
diff
changeset
|
546 |
|
58543 | 547 |
/* bibtex entries */ |
548 |
||
64885 | 549 |
// owned by GUI thread |
550 |
private var _bibtex_entries: Option[List[Text.Info[String]]] = None |
|
58543 | 551 |
|
64817 | 552 |
private def reset_bibtex_entries(): Unit = GUI_Thread.require { _bibtex_entries = None } |
58543 | 553 |
|
64831 | 554 |
def bibtex_entries: List[Text.Info[String]] = |
58543 | 555 |
GUI_Thread.require { |
69255
800b1ce96fce
more general support for Isabelle/PIDE file formats -- less hardwired Bibtex operations;
wenzelm
parents:
68476
diff
changeset
|
556 |
if (Bibtex.is_bibtex(node_name.node)) { |
64817 | 557 |
_bibtex_entries match { |
58543 | 558 |
case Some(entries) => entries |
559 |
case None => |
|
64817 | 560 |
val text = JEdit_Lib.buffer_text(buffer) |
64828 | 561 |
val entries = |
66150 | 562 |
try { Bibtex.entries(text) } |
64828 | 563 |
catch { case ERROR(msg) => Output.warning(msg); Nil } |
64817 | 564 |
_bibtex_entries = Some(entries) |
58543 | 565 |
entries |
566 |
} |
|
567 |
} |
|
568 |
else Nil |
|
569 |
} |
|
570 |
||
571 |
||
50344
608265769ce0
emit bulk edits on initialization of multiple buffers, which greatly improves performance when starting big sessions (e.g. JinjaThreads);
wenzelm
parents:
50207
diff
changeset
|
572 |
/* pending edits */ |
43648 | 573 |
|
60272
4f72b00d9952
no GUI_Thread for SideKick parsers (in contrast to 4c8205fe3644), to avoid danger of deadlock due to nested context switch;
wenzelm
parents:
59737
diff
changeset
|
574 |
private object pending_edits |
38224 | 575 |
{ |
38425 | 576 |
private val pending = new mutable.ListBuffer[Text.Edit] |
57615
df1b3452d71c
more explicit discrimination of empty nodes -- suppress from Theories panel;
wenzelm
parents:
57612
diff
changeset
|
577 |
private var last_perspective = Document.Node.no_perspective_text |
44438 | 578 |
|
64818
67a0a563d2b3
clarified buffer events: exit model while loading;
wenzelm
parents:
64817
diff
changeset
|
579 |
def nonEmpty: Boolean = synchronized { pending.nonEmpty } |
67a0a563d2b3
clarified buffer events: exit model while loading;
wenzelm
parents:
64817
diff
changeset
|
580 |
def get_edits: List[Text.Edit] = synchronized { pending.toList } |
64817 | 581 |
def get_last_perspective: Document.Node.Perspective_Text = synchronized { last_perspective } |
582 |
def set_last_perspective(perspective: Document.Node.Perspective_Text): Unit = |
|
583 |
synchronized { last_perspective = perspective } |
|
38224 | 584 |
|
64817 | 585 |
def flush_edits(doc_blobs: Document.Blobs, hidden: Boolean): List[Document.Edit_Text] = |
61728
5f5ff1eab407
double flush to ensure persistent "state" output is reset;
wenzelm
parents:
61538
diff
changeset
|
586 |
synchronized { |
5f5ff1eab407
double flush to ensure persistent "state" output is reset;
wenzelm
parents:
61538
diff
changeset
|
587 |
GUI_Thread.require {} |
60272
4f72b00d9952
no GUI_Thread for SideKick parsers (in contrast to 4c8205fe3644), to avoid danger of deadlock due to nested context switch;
wenzelm
parents:
59737
diff
changeset
|
588 |
|
64818
67a0a563d2b3
clarified buffer events: exit model while loading;
wenzelm
parents:
64817
diff
changeset
|
589 |
val edits = get_edits |
64817 | 590 |
val (reparse, perspective) = node_perspective(doc_blobs, hidden) |
64818
67a0a563d2b3
clarified buffer events: exit model while loading;
wenzelm
parents:
64817
diff
changeset
|
591 |
if (reparse || edits.nonEmpty || last_perspective != perspective) { |
61728
5f5ff1eab407
double flush to ensure persistent "state" output is reset;
wenzelm
parents:
61538
diff
changeset
|
592 |
pending.clear |
5f5ff1eab407
double flush to ensure persistent "state" output is reset;
wenzelm
parents:
61538
diff
changeset
|
593 |
last_perspective = perspective |
71601 | 594 |
node_edits(node_header(), edits, perspective) |
61728
5f5ff1eab407
double flush to ensure persistent "state" output is reset;
wenzelm
parents:
61538
diff
changeset
|
595 |
} |
5f5ff1eab407
double flush to ensure persistent "state" output is reset;
wenzelm
parents:
61538
diff
changeset
|
596 |
else Nil |
43648 | 597 |
} |
38224 | 598 |
|
64818
67a0a563d2b3
clarified buffer events: exit model while loading;
wenzelm
parents:
64817
diff
changeset
|
599 |
def edit(edits: List[Text.Edit]): Unit = synchronized |
38224 | 600 |
{ |
60272
4f72b00d9952
no GUI_Thread for SideKick parsers (in contrast to 4c8205fe3644), to avoid danger of deadlock due to nested context switch;
wenzelm
parents:
59737
diff
changeset
|
601 |
GUI_Thread.require {} |
4f72b00d9952
no GUI_Thread for SideKick parsers (in contrast to 4c8205fe3644), to avoid danger of deadlock due to nested context switch;
wenzelm
parents:
59737
diff
changeset
|
602 |
|
54511 | 603 |
reset_blob() |
64817 | 604 |
reset_bibtex_entries() |
54511 | 605 |
|
64883 | 606 |
for (doc_view <- document_view_iterator) |
61538
bf4969660913
avoid highlighted area getting "stuck" after edit;
wenzelm
parents:
61192
diff
changeset
|
607 |
doc_view.rich_text_area.active_reset() |
bf4969660913
avoid highlighted area getting "stuck" after edit;
wenzelm
parents:
61192
diff
changeset
|
608 |
|
64818
67a0a563d2b3
clarified buffer events: exit model while loading;
wenzelm
parents:
64817
diff
changeset
|
609 |
pending ++= edits |
66082 | 610 |
PIDE.editor.invoke() |
44436
546adfa8a6fc
update_perspective without actual edits, bypassing the full state assignment protocol;
wenzelm
parents:
44385
diff
changeset
|
611 |
} |
546adfa8a6fc
update_perspective without actual edits, bypassing the full state assignment protocol;
wenzelm
parents:
44385
diff
changeset
|
612 |
} |
546adfa8a6fc
update_perspective without actual edits, bypassing the full state assignment protocol;
wenzelm
parents:
44385
diff
changeset
|
613 |
|
64835
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64832
diff
changeset
|
614 |
def is_stable: Boolean = !pending_edits.nonEmpty |
64818
67a0a563d2b3
clarified buffer events: exit model while loading;
wenzelm
parents:
64817
diff
changeset
|
615 |
def snapshot(): Document.Snapshot = session.snapshot(node_name, pending_edits.get_edits) |
60933
6d03e05ef041
more robust access to stable tip version: take all pending edits into account, don't assume model for current buffer;
wenzelm
parents:
60274
diff
changeset
|
616 |
|
64817 | 617 |
def flush_edits(doc_blobs: Document.Blobs, hidden: Boolean): List[Document.Edit_Text] = |
618 |
pending_edits.flush_edits(doc_blobs, hidden) |
|
34828 | 619 |
|
620 |
||
621 |
/* buffer listener */ |
|
622 |
||
623 |
private val buffer_listener: BufferListener = new BufferAdapter |
|
624 |
{ |
|
625 |
override def contentInserted(buffer: JEditBuffer, |
|
626 |
start_line: Int, offset: Int, num_lines: Int, length: Int) |
|
627 |
{ |
|
64818
67a0a563d2b3
clarified buffer events: exit model while loading;
wenzelm
parents:
64817
diff
changeset
|
628 |
pending_edits.edit(List(Text.Edit.insert(offset, buffer.getText(offset, length)))) |
34828 | 629 |
} |
630 |
||
631 |
override def preContentRemoved(buffer: JEditBuffer, |
|
38426 | 632 |
start_line: Int, offset: Int, num_lines: Int, removed_length: Int) |
34828 | 633 |
{ |
64818
67a0a563d2b3
clarified buffer events: exit model while loading;
wenzelm
parents:
64817
diff
changeset
|
634 |
pending_edits.edit(List(Text.Edit.remove(offset, buffer.getText(offset, removed_length)))) |
34828 | 635 |
} |
636 |
} |
|
637 |
||
638 |
||
59078
cf255dc2b48f
more careful syntax_changed propagation -- avoid global jEdit.propertiesChanged;
wenzelm
parents:
59077
diff
changeset
|
639 |
/* syntax */ |
37557
1ae272fd4082
refresh Isabelle token marker after buffer properties changed, e.g. when fold mode is switched;
wenzelm
parents:
37555
diff
changeset
|
640 |
|
59078
cf255dc2b48f
more careful syntax_changed propagation -- avoid global jEdit.propertiesChanged;
wenzelm
parents:
59077
diff
changeset
|
641 |
def syntax_changed() |
cf255dc2b48f
more careful syntax_changed propagation -- avoid global jEdit.propertiesChanged;
wenzelm
parents:
59077
diff
changeset
|
642 |
{ |
61192 | 643 |
JEdit_Lib.buffer_line_manager(buffer).setFirstInvalidLineContext(0) |
59080
611914621edb
added Untyped.method convenience (for *this* class only);
wenzelm
parents:
59079
diff
changeset
|
644 |
for (text_area <- JEdit_Lib.jedit_text_areas(buffer)) |
611914621edb
added Untyped.method convenience (for *this* class only);
wenzelm
parents:
59079
diff
changeset
|
645 |
Untyped.method(Class.forName("org.gjt.sp.jedit.textarea.TextArea"), "foldStructureChanged"). |
611914621edb
added Untyped.method convenience (for *this* class only);
wenzelm
parents:
59079
diff
changeset
|
646 |
invoke(text_area) |
72960 | 647 |
buffer.invalidateCachedFoldLevels() |
59078
cf255dc2b48f
more careful syntax_changed propagation -- avoid global jEdit.propertiesChanged;
wenzelm
parents:
59077
diff
changeset
|
648 |
} |
cf255dc2b48f
more careful syntax_changed propagation -- avoid global jEdit.propertiesChanged;
wenzelm
parents:
59077
diff
changeset
|
649 |
|
64817 | 650 |
def init_token_marker() |
59077
7e0d3da6e6d8
node-specific syntax, with base_syntax as default;
wenzelm
parents:
59076
diff
changeset
|
651 |
{ |
7e0d3da6e6d8
node-specific syntax, with base_syntax as default;
wenzelm
parents:
59076
diff
changeset
|
652 |
Isabelle.buffer_token_marker(buffer) match { |
7e0d3da6e6d8
node-specific syntax, with base_syntax as default;
wenzelm
parents:
59076
diff
changeset
|
653 |
case Some(marker) if marker != buffer.getTokenMarker => |
7e0d3da6e6d8
node-specific syntax, with base_syntax as default;
wenzelm
parents:
59076
diff
changeset
|
654 |
buffer.setTokenMarker(marker) |
59078
cf255dc2b48f
more careful syntax_changed propagation -- avoid global jEdit.propertiesChanged;
wenzelm
parents:
59077
diff
changeset
|
655 |
syntax_changed() |
59077
7e0d3da6e6d8
node-specific syntax, with base_syntax as default;
wenzelm
parents:
59076
diff
changeset
|
656 |
case _ => |
7e0d3da6e6d8
node-specific syntax, with base_syntax as default;
wenzelm
parents:
59076
diff
changeset
|
657 |
} |
7e0d3da6e6d8
node-specific syntax, with base_syntax as default;
wenzelm
parents:
59076
diff
changeset
|
658 |
} |
7e0d3da6e6d8
node-specific syntax, with base_syntax as default;
wenzelm
parents:
59076
diff
changeset
|
659 |
|
59078
cf255dc2b48f
more careful syntax_changed propagation -- avoid global jEdit.propertiesChanged;
wenzelm
parents:
59077
diff
changeset
|
660 |
|
64817 | 661 |
/* init */ |
662 |
||
663 |
def init(old_model: Option[File_Model]): Buffer_Model = |
|
664 |
{ |
|
665 |
GUI_Thread.require {} |
|
59078
cf255dc2b48f
more careful syntax_changed propagation -- avoid global jEdit.propertiesChanged;
wenzelm
parents:
59077
diff
changeset
|
666 |
|
64817 | 667 |
old_model match { |
668 |
case None => |
|
64818
67a0a563d2b3
clarified buffer events: exit model while loading;
wenzelm
parents:
64817
diff
changeset
|
669 |
pending_edits.edit(List(Text.Edit.insert(0, JEdit_Lib.buffer_text(buffer)))) |
64817 | 670 |
case Some(file_model) => |
671 |
set_node_required(file_model.node_required) |
|
672 |
pending_edits.set_last_perspective(file_model.last_perspective) |
|
64818
67a0a563d2b3
clarified buffer events: exit model while loading;
wenzelm
parents:
64817
diff
changeset
|
673 |
pending_edits.edit( |
67a0a563d2b3
clarified buffer events: exit model while loading;
wenzelm
parents:
64817
diff
changeset
|
674 |
file_model.pending_edits ::: |
67a0a563d2b3
clarified buffer events: exit model while loading;
wenzelm
parents:
64817
diff
changeset
|
675 |
Text.Edit.replace(0, file_model.content.text, JEdit_Lib.buffer_text(buffer))) |
64817 | 676 |
} |
677 |
||
34784
02959dcea756
split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents:
34783
diff
changeset
|
678 |
buffer.addBufferListener(buffer_listener) |
59078
cf255dc2b48f
more careful syntax_changed propagation -- avoid global jEdit.propertiesChanged;
wenzelm
parents:
59077
diff
changeset
|
679 |
init_token_marker() |
64817 | 680 |
|
681 |
this |
|
34680 | 682 |
} |
683 |
||
64817 | 684 |
|
685 |
/* exit */ |
|
686 |
||
687 |
def exit(): File_Model = |
|
34680 | 688 |
{ |
64817 | 689 |
GUI_Thread.require {} |
690 |
||
34784
02959dcea756
split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents:
34783
diff
changeset
|
691 |
buffer.removeBufferListener(buffer_listener) |
59078
cf255dc2b48f
more careful syntax_changed propagation -- avoid global jEdit.propertiesChanged;
wenzelm
parents:
59077
diff
changeset
|
692 |
init_token_marker() |
64817 | 693 |
|
64863 | 694 |
File_Model.init(session, node_name, JEdit_Lib.buffer_text(buffer), node_required, |
695 |
pending_edits.get_last_perspective, pending_edits.get_edits) |
|
34680 | 696 |
} |
34447 | 697 |
} |