author | haftmann |
Sat, 19 Jul 2025 18:41:55 +0200 | |
changeset 82886 | 8d1e295aab70 |
parent 82778 | 803731b62180 |
permissions | -rw-r--r-- |
73987
fc363a3b690a
build.props for isabelle.jar, including isabelle.jedit;
wenzelm
parents:
73947
diff
changeset
|
1 |
/* Title: Tools/jEdit/src/main_plugin.scala |
36760 | 2 |
Author: Makarius |
3 |
||
50362 | 4 |
Main plumbing for PIDE infrastructure as jEdit plugin. |
36760 | 5 |
*/ |
34407 | 6 |
|
34318
c13e168a8ae6
original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff
changeset
|
7 |
package isabelle.jedit |
c13e168a8ae6
original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff
changeset
|
8 |
|
34429 | 9 |
|
36015 | 10 |
import isabelle._ |
11 |
||
44573
51f8895b9ad9
some dialog for auto loading of required files (still inactive);
wenzelm
parents:
44434
diff
changeset
|
12 |
import javax.swing.JOptionPane |
34318
c13e168a8ae6
original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff
changeset
|
13 |
|
64858 | 14 |
import java.io.{File => JFile} |
15 |
||
66593 | 16 |
import org.gjt.sp.jedit.{jEdit, EBMessage, EBPlugin, Buffer, View, PerspectiveManager} |
65248 | 17 |
import org.gjt.sp.jedit.textarea.JEditTextArea |
50362 | 18 |
import org.gjt.sp.jedit.syntax.ModeProvider |
82410
4ca84abb16ef
support for navigation, independently of Navigator plugin;
wenzelm
parents:
82409
diff
changeset
|
19 |
import org.gjt.sp.jedit.msg.{EditorStarted, BufferUpdate, BufferChanging, PositionChanging, |
4ca84abb16ef
support for navigation, independently of Navigator plugin;
wenzelm
parents:
82409
diff
changeset
|
20 |
EditPaneUpdate, PropertiesChanged, ViewUpdate} |
62228
6dfe5b12c5b2
report error on internal channel as well: startup_failure dialog may be too late;
wenzelm
parents:
62227
diff
changeset
|
21 |
import org.gjt.sp.util.Log |
39241 | 22 |
|
34318
c13e168a8ae6
original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff
changeset
|
23 |
|
75393 | 24 |
object PIDE { |
65248 | 25 |
/* semantic document content */ |
55621
8d69c15b6fb9
added PIDE.snapshot, PIDE.rendering for convenience;
wenzelm
parents:
55618
diff
changeset
|
26 |
|
75393 | 27 |
def maybe_snapshot(view: View = null): Option[Document.Snapshot] = GUI_Thread.now { |
69636 | 28 |
val buffer = JEdit_Lib.jedit_view(view).getBuffer |
76765
c654103e9c9d
more robust Document.Pending_Edits: cover all nodes simulataneously, and thus support proper Snapshot.switch;
wenzelm
parents:
76716
diff
changeset
|
29 |
Document_Model.get_snapshot(buffer) |
55621
8d69c15b6fb9
added PIDE.snapshot, PIDE.rendering for convenience;
wenzelm
parents:
55618
diff
changeset
|
30 |
} |
8d69c15b6fb9
added PIDE.snapshot, PIDE.rendering for convenience;
wenzelm
parents:
55618
diff
changeset
|
31 |
|
75393 | 32 |
def maybe_rendering(view: View = null): Option[JEdit_Rendering] = GUI_Thread.now { |
69636 | 33 |
val text_area = JEdit_Lib.jedit_view(view).getTextArea |
76765
c654103e9c9d
more robust Document.Pending_Edits: cover all nodes simulataneously, and thus support proper Snapshot.switch;
wenzelm
parents:
76716
diff
changeset
|
34 |
Document_View.get_rendering(text_area) |
55621
8d69c15b6fb9
added PIDE.snapshot, PIDE.rendering for convenience;
wenzelm
parents:
55618
diff
changeset
|
35 |
} |
65277 | 36 |
|
69759 | 37 |
def snapshot(view: View = null): Document.Snapshot = |
38 |
maybe_snapshot(view) getOrElse error("No document model for current buffer") |
|
39 |
||
40 |
def rendering(view: View = null): JEdit_Rendering = |
|
41 |
maybe_rendering(view) getOrElse error("No document view for current text area") |
|
42 |
||
65277 | 43 |
|
44 |
/* plugin instance */ |
|
45 |
||
73987
fc363a3b690a
build.props for isabelle.jar, including isabelle.jedit;
wenzelm
parents:
73947
diff
changeset
|
46 |
@volatile var _plugin: Main_Plugin = null |
65277 | 47 |
|
73987
fc363a3b690a
build.props for isabelle.jar, including isabelle.jedit;
wenzelm
parents:
73947
diff
changeset
|
48 |
def plugin: Main_Plugin = |
65277 | 49 |
if (_plugin == null) error("Uninitialized Isabelle/jEdit plugin") |
50 |
else _plugin |
|
51 |
||
52 |
def options: JEdit_Options = plugin.options |
|
82744
0ca8b1861fa3
clarified signature: more explicit subtypes of Session, with corresponding subtypes of Resources;
wenzelm
parents:
82742
diff
changeset
|
53 |
def session: JEdit_Session = plugin.session |
0ca8b1861fa3
clarified signature: more explicit subtypes of Session, with corresponding subtypes of Resources;
wenzelm
parents:
82742
diff
changeset
|
54 |
def resources: JEdit_Resources = session.resources |
82764
da8ad1591985
more accurate ML_Settings from underlying Session;
wenzelm
parents:
82761
diff
changeset
|
55 |
def ml_settings: ML_Settings = session.store.ml_settings |
82720
956ecf2c07a0
more flexible ML_Settings in Isabelle/Scala, depending on system options and some default settings;
wenzelm
parents:
82436
diff
changeset
|
56 |
|
66083 | 57 |
object editor extends JEdit_Editor |
34318
c13e168a8ae6
original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff
changeset
|
58 |
} |
c13e168a8ae6
original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff
changeset
|
59 |
|
75393 | 60 |
class Main_Plugin extends EBPlugin { |
65262
0fe4ebab9fdf
resources are part of early/strict initialization, but session_base is permissive to avoid crash of "isabelle jedit -l BAD";
wenzelm
parents:
65261
diff
changeset
|
61 |
/* options */ |
65238 | 62 |
|
82321 | 63 |
private lazy val initial_options: Options = Options.init() |
64 |
||
65 |
private lazy val more_options: List[Options.Spec] = |
|
66 |
Library.space_explode('\u000b', Isabelle_System.getenv("JEDIT_ISABELLE_OPTIONS")) |
|
67 |
.map(Options.Spec.make) |
|
68 |
||
82720
956ecf2c07a0
more flexible ML_Settings in Isabelle/Scala, depending on system options and some default settings;
wenzelm
parents:
82436
diff
changeset
|
69 |
lazy val startup_options: Options = initial_options ++ more_options |
956ecf2c07a0
more flexible ML_Settings in Isabelle/Scala, depending on system options and some default settings;
wenzelm
parents:
82436
diff
changeset
|
70 |
|
65257
2307b91159bb
more explicit strict vs. non-strict initialization;
wenzelm
parents:
65256
diff
changeset
|
71 |
private var _options: JEdit_Options = null |
82321 | 72 |
private def init_options(): Unit = { |
82720
956ecf2c07a0
more flexible ML_Settings in Isabelle/Scala, depending on system options and some default settings;
wenzelm
parents:
82436
diff
changeset
|
73 |
_options = new JEdit_Options(startup_options) |
82321 | 74 |
} |
65257
2307b91159bb
more explicit strict vs. non-strict initialization;
wenzelm
parents:
65256
diff
changeset
|
75 |
def options: JEdit_Options = _options |
65239
509a9b0ad02e
avoid global variables with implicit initialization;
wenzelm
parents:
65238
diff
changeset
|
76 |
|
65262
0fe4ebab9fdf
resources are part of early/strict initialization, but session_base is permissive to avoid crash of "isabelle jedit -l BAD";
wenzelm
parents:
65261
diff
changeset
|
77 |
|
65263
c97abf0fa0c1
strict initialization of plugin.session: no user errors to be expected before session.start;
wenzelm
parents:
65262
diff
changeset
|
78 |
/* session */ |
c97abf0fa0c1
strict initialization of plugin.session: no user errors to be expected before session.start;
wenzelm
parents:
65262
diff
changeset
|
79 |
|
82744
0ca8b1861fa3
clarified signature: more explicit subtypes of Session, with corresponding subtypes of Resources;
wenzelm
parents:
82742
diff
changeset
|
80 |
private var _session: JEdit_Session = null |
82778 | 81 |
private def init_session(): Unit = { |
82 |
_session = |
|
83 |
new JEdit_Session(options.value) { |
|
84 |
override def deps_changed(): Unit = delay_load.invoke() |
|
85 |
} |
|
86 |
} |
|
82744
0ca8b1861fa3
clarified signature: more explicit subtypes of Session, with corresponding subtypes of Resources;
wenzelm
parents:
82742
diff
changeset
|
87 |
def session: JEdit_Session = _session |
65263
c97abf0fa0c1
strict initialization of plugin.session: no user errors to be expected before session.start;
wenzelm
parents:
65262
diff
changeset
|
88 |
|
c97abf0fa0c1
strict initialization of plugin.session: no user errors to be expected before session.start;
wenzelm
parents:
65262
diff
changeset
|
89 |
|
65262
0fe4ebab9fdf
resources are part of early/strict initialization, but session_base is permissive to avoid crash of "isabelle jedit -l BAD";
wenzelm
parents:
65261
diff
changeset
|
90 |
/* misc support */ |
0fe4ebab9fdf
resources are part of early/strict initialization, but session_base is permissive to avoid crash of "isabelle jedit -l BAD";
wenzelm
parents:
65261
diff
changeset
|
91 |
|
65239
509a9b0ad02e
avoid global variables with implicit initialization;
wenzelm
parents:
65238
diff
changeset
|
92 |
val completion_history = new Completion.History_Variable |
509a9b0ad02e
avoid global variables with implicit initialization;
wenzelm
parents:
65238
diff
changeset
|
93 |
val spell_checker = new Spell_Checker_Variable |
82410
4ca84abb16ef
support for navigation, independently of Navigator plugin;
wenzelm
parents:
82409
diff
changeset
|
94 |
val navigator = new Isabelle_Navigator |
65239
509a9b0ad02e
avoid global variables with implicit initialization;
wenzelm
parents:
65238
diff
changeset
|
95 |
|
65238 | 96 |
|
44577
96b6388d06c4
separate module for jEdit primitives for loading theory files;
wenzelm
parents:
44574
diff
changeset
|
97 |
/* theory files */ |
44574 | 98 |
|
76704 | 99 |
private lazy val delay_init: Delay = |
76610 | 100 |
Delay.last(PIDE.session.load_delay, gui = true) { init_models() } |
50344
608265769ce0
emit bulk edits on initialization of multiple buffers, which greatly improves performance when starting big sessions (e.g. JinjaThreads);
wenzelm
parents:
50209
diff
changeset
|
101 |
|
76704 | 102 |
private lazy val delay_load: Delay = |
103 |
Delay.last(session.load_delay, gui = true) { |
|
104 |
if (JEdit_Options.continuous_checking()) { |
|
105 |
if (!PerspectiveManager.isPerspectiveEnabled || |
|
106 |
JEdit_Lib.jedit_buffers().exists(_.isLoading)) delay_load.invoke() |
|
107 |
else if (delay_load_activated()) delay_load_body() |
|
108 |
else delay_load.invoke() |
|
109 |
} |
|
110 |
} |
|
111 |
||
57582 | 112 |
private val delay_load_active = Synchronized(false) |
76584 | 113 |
private def delay_load_finished(): Unit = delay_load_active.change(_ => false) |
57582 | 114 |
private def delay_load_activated(): Boolean = |
115 |
delay_load_active.guarded_access(a => Some((!a, true))) |
|
76584 | 116 |
|
76585
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
117 |
private def delay_load_body(): Unit = { |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
118 |
val required_files = { |
76777 | 119 |
val models = Document_Model.get_models_map() |
76585
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
120 |
|
76716
a7602257a825
clarified state document nodes for Theories_Status / Document_Dockable;
wenzelm
parents:
76706
diff
changeset
|
121 |
val thy_files = |
82744
0ca8b1861fa3
clarified signature: more explicit subtypes of Session, with corresponding subtypes of Resources;
wenzelm
parents:
82742
diff
changeset
|
122 |
session.resources.resolve_dependencies(models.values, PIDE.editor.document_required()) |
67290 | 123 |
|
76585
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
124 |
val aux_files = |
82778 | 125 |
if (session.auto_resolve) { |
76766 | 126 |
session.stable_tip_version(models.values) match { |
82744
0ca8b1861fa3
clarified signature: more explicit subtypes of Session, with corresponding subtypes of Resources;
wenzelm
parents:
82742
diff
changeset
|
127 |
case Some(version) => session.resources.undefined_blobs(version) |
76585
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
128 |
case None => delay_load.invoke(); Nil |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
129 |
} |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
130 |
} |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
131 |
else Nil |
46761 | 132 |
|
76777 | 133 |
(thy_files ::: aux_files).filterNot(models.keySet) |
76585
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
134 |
} |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
135 |
if (required_files.nonEmpty) { |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
136 |
try { |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
137 |
Isabelle_Thread.fork(name = "resolve_dependencies") { |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
138 |
val loaded_files = |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
139 |
for { |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
140 |
name <- required_files |
82744
0ca8b1861fa3
clarified signature: more explicit subtypes of Session, with corresponding subtypes of Resources;
wenzelm
parents:
82742
diff
changeset
|
141 |
text <- session.resources.read_file_content(name) |
76585
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
142 |
} yield (name, text) |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
143 |
|
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
144 |
GUI_Thread.later { |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
145 |
try { |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
146 |
Document_Model.provide_files(session, loaded_files) |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
147 |
delay_init.invoke() |
62068 | 148 |
} |
76585
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
149 |
finally { delay_load_finished() } |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
150 |
} |
64835
fd1efd6dd385
resolve dependencies implicitly via File_Model, without jEdit Buffer_Model;
wenzelm
parents:
64818
diff
changeset
|
151 |
} |
76585
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
152 |
} |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
153 |
catch { case _: Throwable => delay_load_finished() } |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
154 |
} |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
155 |
else delay_load_finished() |
1b7bb4f8c0f4
potentially more robust delay_load action: avoid loosing events due to guards;
wenzelm
parents:
76584
diff
changeset
|
156 |
} |
62068 | 157 |
|
64858 | 158 |
private def file_watcher_action(changed: Set[JFile]): Unit = |
66082 | 159 |
if (Document_Model.sync_files(changed)) PIDE.editor.invoke_generated() |
64858 | 160 |
|
161 |
lazy val file_watcher: File_Watcher = |
|
76610 | 162 |
File_Watcher(file_watcher_action, session.load_delay) |
64858 | 163 |
|
44574 | 164 |
|
56715
52125652e82a
clarified Session.Consumer, with Session.Outlet managed by dispatcher thread;
wenzelm
parents:
56662
diff
changeset
|
165 |
/* session phase */ |
44573
51f8895b9ad9
some dialog for auto loading of required files (still inactive);
wenzelm
parents:
44434
diff
changeset
|
166 |
|
75393 | 167 |
val session_phase_changed: Session.Consumer[Session.Phase] = Session.Consumer("Isabelle/jEdit") { |
65317 | 168 |
case Session.Terminated(result) if !result.ok => |
65226 | 169 |
GUI_Thread.later { |
65267 | 170 |
GUI.error_dialog(jEdit.getActiveView, "Prover process terminated with error", |
76488 | 171 |
"Isabelle Syslog", GUI.scrollable_text(session.syslog.content())) |
65226 | 172 |
} |
39735
969ede84aac0
more uniform init/exit model/view in session_manager, trading race wrt. session.phase for race wrt. global editor state;
wenzelm
parents:
39702
diff
changeset
|
173 |
|
71471
c06604896c3d
more robust shutdown while Isabelle_Process is starting up, e.g. quit after dialog Isabelle version for testing";
wenzelm
parents:
70775
diff
changeset
|
174 |
case Session.Ready if !shutting_down.value => |
65244 | 175 |
init_models() |
57640
0a28cf866d5d
reconfirm continuous checking on startup, to address common trap of disabling it accidentally;
wenzelm
parents:
57612
diff
changeset
|
176 |
|
75849 | 177 |
if (!JEdit_Options.continuous_checking()) { |
65226 | 178 |
GUI_Thread.later { |
179 |
val answer = |
|
180 |
GUI.confirm_dialog(jEdit.getActiveView, |
|
181 |
"Continuous checking of PIDE document", |
|
182 |
JOptionPane.YES_NO_OPTION, |
|
183 |
"Continuous checking is presently disabled:", |
|
184 |
"editor buffers will remain inactive!", |
|
185 |
"Enable continuous checking now?") |
|
75849 | 186 |
if (answer == 0) JEdit_Options.continuous_checking.set() |
57640
0a28cf866d5d
reconfirm continuous checking on startup, to address common trap of disabling it accidentally;
wenzelm
parents:
57612
diff
changeset
|
187 |
} |
65226 | 188 |
} |
57640
0a28cf866d5d
reconfirm continuous checking on startup, to address common trap of disabling it accidentally;
wenzelm
parents:
57612
diff
changeset
|
189 |
|
65226 | 190 |
delay_load.invoke() |
44573
51f8895b9ad9
some dialog for auto loading of required files (still inactive);
wenzelm
parents:
44434
diff
changeset
|
191 |
|
65226 | 192 |
case Session.Shutdown => |
193 |
GUI_Thread.later { |
|
194 |
delay_load.revoke() |
|
195 |
delay_init.revoke() |
|
66458
42d0d5c77c78
more robust shutdown, e.g. when plugin is stopped;
wenzelm
parents:
66351
diff
changeset
|
196 |
PIDE.editor.shutdown() |
65244 | 197 |
exit_models(JEdit_Lib.jedit_buffers().toList) |
65226 | 198 |
} |
46740
852baa599351
explicitly revoke delay, to avoid spurious timer events after deactivation of related components;
wenzelm
parents:
46204
diff
changeset
|
199 |
|
65226 | 200 |
case _ => |
201 |
} |
|
39630
44181423183a
explicit Session.Phase indication with associated event bus;
wenzelm
parents:
39628
diff
changeset
|
202 |
|
44181423183a
explicit Session.Phase indication with associated event bus;
wenzelm
parents:
39628
diff
changeset
|
203 |
|
65244 | 204 |
/* document model and view */ |
205 |
||
75393 | 206 |
def exit_models(buffers: List[Buffer]): Unit = { |
65244 | 207 |
GUI_Thread.now { |
208 |
buffers.foreach(buffer => |
|
209 |
JEdit_Lib.buffer_lock(buffer) { |
|
210 |
JEdit_Lib.jedit_text_areas(buffer).foreach(Document_View.exit) |
|
211 |
Document_Model.exit(buffer) |
|
212 |
}) |
|
213 |
} |
|
214 |
} |
|
215 |
||
75393 | 216 |
def init_models(): Unit = { |
65244 | 217 |
GUI_Thread.now { |
66082 | 218 |
PIDE.editor.flush() |
65244 | 219 |
|
220 |
for { |
|
221 |
buffer <- JEdit_Lib.jedit_buffers() |
|
222 |
if buffer != null && !buffer.getBooleanProperty(Buffer.GZIPPED) |
|
223 |
} { |
|
224 |
if (buffer.isLoaded) { |
|
225 |
JEdit_Lib.buffer_lock(buffer) { |
|
82744
0ca8b1861fa3
clarified signature: more explicit subtypes of Session, with corresponding subtypes of Resources;
wenzelm
parents:
82742
diff
changeset
|
226 |
val node_name = session.resources.node_name(buffer) |
65263
c97abf0fa0c1
strict initialization of plugin.session: no user errors to be expected before session.start;
wenzelm
parents:
65262
diff
changeset
|
227 |
val model = Document_Model.init(session, node_name, buffer) |
65244 | 228 |
for { |
229 |
text_area <- JEdit_Lib.jedit_text_areas(buffer) |
|
75751 | 230 |
if !Document_View.get(text_area).map(_.model).contains(model) |
65244 | 231 |
} Document_View.init(model, text_area) |
232 |
} |
|
233 |
} |
|
234 |
else delay_init.invoke() |
|
235 |
} |
|
236 |
||
66082 | 237 |
PIDE.editor.invoke_generated() |
65244 | 238 |
} |
239 |
} |
|
240 |
||
241 |
def init_view(buffer: Buffer, text_area: JEditTextArea): Unit = |
|
242 |
GUI_Thread.now { |
|
243 |
JEdit_Lib.buffer_lock(buffer) { |
|
76768
40c8275f0131
tuned signature: follow terminology of VSCode_Resources;
wenzelm
parents:
76767
diff
changeset
|
244 |
Document_Model.get_model(buffer) match { |
65244 | 245 |
case Some(model) => Document_View.init(model, text_area) |
246 |
case None => |
|
247 |
} |
|
248 |
} |
|
249 |
} |
|
250 |
||
251 |
def exit_view(buffer: Buffer, text_area: JEditTextArea): Unit = |
|
252 |
GUI_Thread.now { |
|
253 |
JEdit_Lib.buffer_lock(buffer) { |
|
254 |
Document_View.exit(text_area) |
|
255 |
} |
|
256 |
} |
|
257 |
||
258 |
||
34618 | 259 |
/* main plugin plumbing */ |
34433 | 260 |
|
65241 | 261 |
@volatile private var startup_failure: Option[Throwable] = None |
262 |
@volatile private var startup_notified = false |
|
263 |
||
75393 | 264 |
private def init_editor(view: View): Unit = { |
71725
c255ed582095
proper asynchronous GUI interaction for somewhat heavy JEdit_Sessions.session_build check;
wenzelm
parents:
71724
diff
changeset
|
265 |
Keymap_Merge.check_dialog(view) |
66459 | 266 |
Session_Build.check_dialog(view) |
267 |
} |
|
268 |
||
75393 | 269 |
private def init_title(view: View): Unit = { |
68080 | 270 |
val title = |
271 |
proper_string(Isabelle_System.getenv("ISABELLE_IDENTIFIER")).getOrElse("Isabelle") + |
|
75752 | 272 |
"/" + PIDE.resources.session_base.session_name |
68080 | 273 |
val marker = "\u200B" |
274 |
||
275 |
val old_title = view.getViewConfig.title |
|
276 |
if (old_title == null || old_title.startsWith(marker)) { |
|
277 |
view.setUserTitle(marker + title) |
|
278 |
} |
|
279 |
} |
|
280 |
||
75393 | 281 |
override def handleMessage(message: EBMessage): Unit = { |
57612
990ffb84489b
clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents:
57611
diff
changeset
|
282 |
GUI_Thread.assert {} |
39630
44181423183a
explicit Session.Phase indication with associated event bus;
wenzelm
parents:
39628
diff
changeset
|
283 |
|
65241 | 284 |
if (startup_failure.isDefined && !startup_notified) { |
49099
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
285 |
message match { |
75751 | 286 |
case _: EditorStarted => |
51616 | 287 |
GUI.error_dialog(null, "Isabelle plugin startup failure", |
65241 | 288 |
GUI.scrollable_text(Exn.message(startup_failure.get)), |
49099
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
289 |
"Prover IDE inactive!") |
65241 | 290 |
startup_notified = true |
49099
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
291 |
case _ => |
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
292 |
} |
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
293 |
} |
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
294 |
|
65241 | 295 |
if (startup_failure.isEmpty) { |
49099
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
296 |
message match { |
75751 | 297 |
case _: EditorStarted => |
82409 | 298 |
val view = jEdit.getActiveView |
299 |
||
82744
0ca8b1861fa3
clarified signature: more explicit subtypes of Session, with corresponding subtypes of Resources;
wenzelm
parents:
82742
diff
changeset
|
300 |
try { session.resources.session_background.check_errors } |
76666 | 301 |
catch { |
302 |
case ERROR(msg) => |
|
82409 | 303 |
GUI.warning_dialog(view, |
76666 | 304 |
"Bad session structure: may cause problems with theory imports", |
305 |
GUI.scrollable_text(msg)) |
|
66572
1e5ae735e026
tolerate errors in session structure, although this may lead to confusion about theory imports later on;
wenzelm
parents:
66459
diff
changeset
|
306 |
} |
1e5ae735e026
tolerate errors in session structure, although this may lead to confusion about theory imports later on;
wenzelm
parents:
66459
diff
changeset
|
307 |
|
66602
180b2e72601f
more thorough change of syntax style extender: jEdit.propertiesChanged invalidates buffer chunk cache;
wenzelm
parents:
66593
diff
changeset
|
308 |
jEdit.propertiesChanged() |
180b2e72601f
more thorough change of syntax style extender: jEdit.propertiesChanged invalidates buffer chunk cache;
wenzelm
parents:
66593
diff
changeset
|
309 |
|
82404 | 310 |
if (view != null) { |
311 |
init_editor(view) |
|
64602 | 312 |
|
82404 | 313 |
PIDE.editor.hyperlink_position(true, Document.Snapshot.init, |
82748 | 314 |
JEdit_Session.logic_root(options.value)).foreach(_.follow(view)) |
82404 | 315 |
} |
61277 | 316 |
|
82409 | 317 |
case msg: ViewUpdate => |
318 |
val what = msg.getWhat |
|
319 |
val view = msg.getView |
|
320 |
what match { |
|
321 |
case ViewUpdate.CREATED if view != null => init_title(view) |
|
322 |
case _ => |
|
64838 | 323 |
} |
57611
b6256ea3b7c5
proper change of perspective for removed nodes (stemming from closed buffers);
wenzelm
parents:
57582
diff
changeset
|
324 |
|
82409 | 325 |
case msg: BufferUpdate => |
326 |
val what = msg.getWhat |
|
327 |
val buffer = msg.getBuffer |
|
82410
4ca84abb16ef
support for navigation, independently of Navigator plugin;
wenzelm
parents:
82409
diff
changeset
|
328 |
val view = msg.getView |
4ca84abb16ef
support for navigation, independently of Navigator plugin;
wenzelm
parents:
82409
diff
changeset
|
329 |
val view_edit_pane = if (view == null) null else view.getEditPane |
4ca84abb16ef
support for navigation, independently of Navigator plugin;
wenzelm
parents:
82409
diff
changeset
|
330 |
|
82409 | 331 |
what match { |
332 |
case BufferUpdate.LOAD_STARTED | BufferUpdate.CLOSING if buffer != null => |
|
333 |
exit_models(List(buffer)) |
|
334 |
PIDE.editor.invoke_generated() |
|
335 |
case BufferUpdate.PROPERTIES_CHANGED | BufferUpdate.LOADED if session.is_ready => |
|
336 |
delay_init.invoke() |
|
337 |
delay_load.invoke() |
|
338 |
case _ => |
|
49099
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
339 |
} |
34784
02959dcea756
split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents:
34782
diff
changeset
|
340 |
|
82410
4ca84abb16ef
support for navigation, independently of Navigator plugin;
wenzelm
parents:
82409
diff
changeset
|
341 |
if (buffer != null && !buffer.isUntitled) { |
4ca84abb16ef
support for navigation, independently of Navigator plugin;
wenzelm
parents:
82409
diff
changeset
|
342 |
what match { |
4ca84abb16ef
support for navigation, independently of Navigator plugin;
wenzelm
parents:
82409
diff
changeset
|
343 |
case BufferUpdate.CREATED => navigator.init(Set(buffer)) |
4ca84abb16ef
support for navigation, independently of Navigator plugin;
wenzelm
parents:
82409
diff
changeset
|
344 |
case BufferUpdate.CLOSED => navigator.exit(Set(buffer)) |
4ca84abb16ef
support for navigation, independently of Navigator plugin;
wenzelm
parents:
82409
diff
changeset
|
345 |
case _ => |
4ca84abb16ef
support for navigation, independently of Navigator plugin;
wenzelm
parents:
82409
diff
changeset
|
346 |
} |
4ca84abb16ef
support for navigation, independently of Navigator plugin;
wenzelm
parents:
82409
diff
changeset
|
347 |
} |
4ca84abb16ef
support for navigation, independently of Navigator plugin;
wenzelm
parents:
82409
diff
changeset
|
348 |
|
82409 | 349 |
case msg: EditPaneUpdate => |
350 |
val what = msg.getWhat |
|
49099
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
351 |
val edit_pane = msg.getEditPane |
82409 | 352 |
val buffer = if (edit_pane == null) null else edit_pane.getBuffer |
353 |
val text_area = if (edit_pane == null) null else edit_pane.getTextArea |
|
49099
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
354 |
|
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
355 |
if (buffer != null && text_area != null) { |
82409 | 356 |
if (what == EditPaneUpdate.BUFFER_CHANGED || what == EditPaneUpdate.CREATED) { |
357 |
if (session.is_ready) init_view(buffer, text_area) |
|
49099
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
358 |
} |
82409 | 359 |
|
360 |
if (what == EditPaneUpdate.BUFFER_CHANGING || what == EditPaneUpdate.DESTROYED) { |
|
65240 | 361 |
Isabelle.dismissed_popups(text_area.getView) |
65244 | 362 |
exit_view(buffer, text_area) |
52867
8d8cb75ab20a
more central Pretty_Tooltip.dismissed_all -- avoid spurious crash of Rich_Text_Area.robust_body in asynchronous mouse_motion_listener;
wenzelm
parents:
52816
diff
changeset
|
363 |
} |
53274
1760c01f1c78
maintain Completion_Popup.Text_Area as client property like Document_View;
wenzelm
parents:
53272
diff
changeset
|
364 |
|
82409 | 365 |
if (what == EditPaneUpdate.CREATED) Completion_Popup.Text_Area.init(text_area) |
53274
1760c01f1c78
maintain Completion_Popup.Text_Area as client property like Document_View;
wenzelm
parents:
53272
diff
changeset
|
366 |
|
82409 | 367 |
if (what == EditPaneUpdate.DESTROYED) Completion_Popup.Text_Area.exit(text_area) |
43510
17d431c92575
init/exit model/view synchronously within the swing thread -- EditBus.send in jedit-4.4.1 always runs there;
wenzelm
parents:
43487
diff
changeset
|
368 |
} |
34784
02959dcea756
split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents:
34782
diff
changeset
|
369 |
|
82413 | 370 |
if (msg.isInstanceOf[PositionChanging]) navigator.record(edit_pane) |
82410
4ca84abb16ef
support for navigation, independently of Navigator plugin;
wenzelm
parents:
82409
diff
changeset
|
371 |
|
75751 | 372 |
case _: PropertiesChanged => |
62264
340f98836fd9
re-init document views for the sake of Text_Overview size;
wenzelm
parents:
62228
diff
changeset
|
373 |
for { |
73367 | 374 |
view <- JEdit_Lib.jedit_views() |
62264
340f98836fd9
re-init document views for the sake of Text_Overview size;
wenzelm
parents:
62228
diff
changeset
|
375 |
edit_pane <- JEdit_Lib.jedit_edit_panes(view) |
340f98836fd9
re-init document views for the sake of Text_Overview size;
wenzelm
parents:
62228
diff
changeset
|
376 |
} { |
340f98836fd9
re-init document views for the sake of Text_Overview size;
wenzelm
parents:
62228
diff
changeset
|
377 |
val buffer = edit_pane.getBuffer |
340f98836fd9
re-init document views for the sake of Text_Overview size;
wenzelm
parents:
62228
diff
changeset
|
378 |
val text_area = edit_pane.getTextArea |
65244 | 379 |
if (buffer != null && text_area != null) init_view(buffer, text_area) |
62264
340f98836fd9
re-init document views for the sake of Text_Overview size;
wenzelm
parents:
62228
diff
changeset
|
380 |
} |
340f98836fd9
re-init document views for the sake of Text_Overview size;
wenzelm
parents:
62228
diff
changeset
|
381 |
|
65249 | 382 |
spell_checker.update(options.value) |
65263
c97abf0fa0c1
strict initialization of plugin.session: no user errors to be expected before session.start;
wenzelm
parents:
65262
diff
changeset
|
383 |
session.update_options(options.value) |
34784
02959dcea756
split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents:
34782
diff
changeset
|
384 |
|
49099
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
385 |
case _ => |
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
386 |
} |
34318
c13e168a8ae6
original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff
changeset
|
387 |
} |
c13e168a8ae6
original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff
changeset
|
388 |
} |
c13e168a8ae6
original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff
changeset
|
389 |
|
65260 | 390 |
|
391 |
/* mode provider */ |
|
392 |
||
393 |
private var orig_mode_provider: ModeProvider = null |
|
394 |
private var pide_mode_provider: ModeProvider = null |
|
395 |
||
75393 | 396 |
def init_mode_provider(): Unit = { |
65260 | 397 |
orig_mode_provider = ModeProvider.instance |
398 |
if (orig_mode_provider.isInstanceOf[ModeProvider]) { |
|
399 |
pide_mode_provider = new Token_Markup.Mode_Provider(orig_mode_provider) |
|
400 |
ModeProvider.instance = pide_mode_provider |
|
401 |
} |
|
402 |
} |
|
403 |
||
75393 | 404 |
def exit_mode_provider(): Unit = { |
65260 | 405 |
if (ModeProvider.instance == pide_mode_provider) |
406 |
ModeProvider.instance = orig_mode_provider |
|
407 |
} |
|
408 |
||
409 |
||
66019 | 410 |
/* HTTP server */ |
411 |
||
79717 | 412 |
val http_root: String = "/" + UUID.random_string() |
66019 | 413 |
|
75107 | 414 |
val http_server: HTTP.Server = |
75145 | 415 |
HTTP.server(services = Document_Model.Preview_Service :: HTTP.isabelle_services) |
66019 | 416 |
|
417 |
||
65260 | 418 |
/* start and stop */ |
419 |
||
71471
c06604896c3d
more robust shutdown while Isabelle_Process is starting up, e.g. quit after dialog Isabelle version for testing";
wenzelm
parents:
70775
diff
changeset
|
420 |
private val shutting_down = Synchronized(false) |
c06604896c3d
more robust shutdown while Isabelle_Process is starting up, e.g. quit after dialog Isabelle version for testing";
wenzelm
parents:
70775
diff
changeset
|
421 |
|
75393 | 422 |
override def start(): Unit = { |
65257
2307b91159bb
more explicit strict vs. non-strict initialization;
wenzelm
parents:
65256
diff
changeset
|
423 |
/* strict initialization */ |
2307b91159bb
more explicit strict vs. non-strict initialization;
wenzelm
parents:
65256
diff
changeset
|
424 |
|
65262
0fe4ebab9fdf
resources are part of early/strict initialization, but session_base is permissive to avoid crash of "isabelle jedit -l BAD";
wenzelm
parents:
65261
diff
changeset
|
425 |
init_options() |
65263
c97abf0fa0c1
strict initialization of plugin.session: no user errors to be expected before session.start;
wenzelm
parents:
65262
diff
changeset
|
426 |
init_session() |
65262
0fe4ebab9fdf
resources are part of early/strict initialization, but session_base is permissive to avoid crash of "isabelle jedit -l BAD";
wenzelm
parents:
65261
diff
changeset
|
427 |
PIDE._plugin = this |
65257
2307b91159bb
more explicit strict vs. non-strict initialization;
wenzelm
parents:
65256
diff
changeset
|
428 |
|
2307b91159bb
more explicit strict vs. non-strict initialization;
wenzelm
parents:
65256
diff
changeset
|
429 |
|
2307b91159bb
more explicit strict vs. non-strict initialization;
wenzelm
parents:
65256
diff
changeset
|
430 |
/* non-strict initialization */ |
2307b91159bb
more explicit strict vs. non-strict initialization;
wenzelm
parents:
65256
diff
changeset
|
431 |
|
49099
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
432 |
try { |
65242 | 433 |
completion_history.load() |
65249 | 434 |
spell_checker.update(options.value) |
49245
cb70157293c0
manage Isabelle/jEdit options as Isabelle/Scala options (with persistent preferences);
wenzelm
parents:
49195
diff
changeset
|
435 |
|
73367 | 436 |
JEdit_Lib.jedit_views().foreach(init_title) |
82410
4ca84abb16ef
support for navigation, independently of Navigator plugin;
wenzelm
parents:
82409
diff
changeset
|
437 |
navigator.init(JEdit_Lib.jedit_buffers()) |
68080 | 438 |
|
73997 | 439 |
Syntax_Style.set_extender(Syntax_Style.Main_Extender) |
65260 | 440 |
init_mode_provider() |
73367 | 441 |
JEdit_Lib.jedit_text_areas().foreach(Completion_Popup.Text_Area.init) |
53274
1760c01f1c78
maintain Completion_Popup.Text_Area as client property like Document_View;
wenzelm
parents:
53272
diff
changeset
|
442 |
|
73367 | 443 |
http_server.start() |
66019 | 444 |
|
65241 | 445 |
startup_failure = None |
49099
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
446 |
} |
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
447 |
catch { |
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
448 |
case exn: Throwable => |
65241 | 449 |
startup_failure = Some(exn) |
450 |
startup_notified = false |
|
62228
6dfe5b12c5b2
report error on internal channel as well: startup_failure dialog may be too late;
wenzelm
parents:
62227
diff
changeset
|
451 |
Log.log(Log.ERROR, this, exn) |
49099
10e899bb6530
more permissive handling of plugin startup failure;
wenzelm
parents:
49098
diff
changeset
|
452 |
} |
66459 | 453 |
|
71471
c06604896c3d
more robust shutdown while Isabelle_Process is starting up, e.g. quit after dialog Isabelle version for testing";
wenzelm
parents:
70775
diff
changeset
|
454 |
shutting_down.change(_ => false) |
c06604896c3d
more robust shutdown while Isabelle_Process is starting up, e.g. quit after dialog Isabelle version for testing";
wenzelm
parents:
70775
diff
changeset
|
455 |
|
82403 | 456 |
val view = jEdit.getActiveView |
71724
522994a6c10e
tuned signature --- avoid confusion with init_view(buffer: Buffer, text_area: JEditTextArea);
wenzelm
parents:
71704
diff
changeset
|
457 |
if (view != null) init_editor(view) |
34318
c13e168a8ae6
original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff
changeset
|
458 |
} |
34618 | 459 |
|
75393 | 460 |
override def stop(): Unit = { |
73367 | 461 |
http_server.stop() |
66019 | 462 |
|
73987
fc363a3b690a
build.props for isabelle.jar, including isabelle.jedit;
wenzelm
parents:
73947
diff
changeset
|
463 |
Syntax_Style.set_extender(Syntax_Style.Base_Extender) |
fc363a3b690a
build.props for isabelle.jar, including isabelle.jedit;
wenzelm
parents:
73947
diff
changeset
|
464 |
|
65261 | 465 |
exit_mode_provider() |
73367 | 466 |
JEdit_Lib.jedit_text_areas().foreach(Completion_Popup.Text_Area.exit) |
53274
1760c01f1c78
maintain Completion_Popup.Text_Area as client property like Document_View;
wenzelm
parents:
53272
diff
changeset
|
467 |
|
65241 | 468 |
if (startup_failure.isEmpty) { |
82321 | 469 |
val save_options = |
470 |
more_options.foldLeft(options.value) { |
|
471 |
case (opts, opt) => opts + initial_options.spec(opt.name) |
|
472 |
} |
|
473 |
save_options.save_prefs() |
|
65242 | 474 |
completion_history.value.save() |
53337
b3817a0e3211
sort items according to persistent history of frequency of use;
wenzelm
parents:
53277
diff
changeset
|
475 |
} |
49245
cb70157293c0
manage Isabelle/jEdit options as Isabelle/Scala options (with persistent preferences);
wenzelm
parents:
49195
diff
changeset
|
476 |
|
65244 | 477 |
exit_models(JEdit_Lib.jedit_buffers().toList) |
71471
c06604896c3d
more robust shutdown while Isabelle_Process is starting up, e.g. quit after dialog Isabelle version for testing";
wenzelm
parents:
70775
diff
changeset
|
478 |
|
c06604896c3d
more robust shutdown while Isabelle_Process is starting up, e.g. quit after dialog Isabelle version for testing";
wenzelm
parents:
70775
diff
changeset
|
479 |
shutting_down.change(_ => true) |
65263
c97abf0fa0c1
strict initialization of plugin.session: no user errors to be expected before session.start;
wenzelm
parents:
65262
diff
changeset
|
480 |
session.stop() |
65245 | 481 |
file_watcher.shutdown() |
66458
42d0d5c77c78
more robust shutdown, e.g. when plugin is stopped;
wenzelm
parents:
66351
diff
changeset
|
482 |
PIDE.editor.shutdown() |
73999
6b213c0115f5
clarified global state: allow to deactivate main plugin;
wenzelm
parents:
73998
diff
changeset
|
483 |
|
6b213c0115f5
clarified global state: allow to deactivate main plugin;
wenzelm
parents:
73998
diff
changeset
|
484 |
Document_Model.reset() |
34318
c13e168a8ae6
original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff
changeset
|
485 |
} |
c13e168a8ae6
original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff
changeset
|
486 |
} |