1 /* Title: Tools/jEdit/src/isabelle.scala
4 Global configuration and convenience operations for Isabelle/jEdit.
12 import scala.swing.CheckBox
13 import scala.swing.event.ButtonClicked
15 import org.gjt.sp.jedit.{jEdit, View, Buffer}
16 import org.gjt.sp.jedit.textarea.JEditTextArea
17 import org.gjt.sp.jedit.gui.{DockableWindowManager, CompleteWord}
26 "isabelle", // theory source
27 "isabelle-markup", // SideKick markup tree
28 "isabelle-news", // NEWS
29 "isabelle-options", // etc/options
30 "isabelle-output", // pretty text area output
31 "isabelle-root") // session ROOT
33 private lazy val news_syntax = Outer_Syntax.init().no_tokens
35 def mode_syntax(name: String): Option[Outer_Syntax] =
37 case "isabelle" | "isabelle-markup" =>
38 val syntax = PIDE.session.recent_syntax
39 if (syntax == Outer_Syntax.empty) None else Some(syntax)
40 case "isabelle-options" => Some(Options.options_syntax)
41 case "isabelle-root" => Some(Build.root_syntax)
42 case "isabelle-news" => Some(news_syntax)
43 case "isabelle-output" => None
51 Map(modes.map(name => (name, new Token_Markup.Marker(name))): _*)
53 def token_marker(name: String): Option[Token_Markup.Marker] = markers.get(name)
56 /* dockable windows */
58 private def wm(view: View): DockableWindowManager = view.getDockableWindowManager
60 def docked_theories(view: View): Option[Theories_Dockable] =
61 wm(view).getDockableWindow("isabelle-theories") match {
62 case dockable: Theories_Dockable => Some(dockable)
66 def docked_timing(view: View): Option[Timing_Dockable] =
67 wm(view).getDockableWindow("isabelle-timing") match {
68 case dockable: Timing_Dockable => Some(dockable)
72 def docked_output(view: View): Option[Output_Dockable] =
73 wm(view).getDockableWindow("isabelle-output") match {
74 case dockable: Output_Dockable => Some(dockable)
78 def docked_raw_output(view: View): Option[Raw_Output_Dockable] =
79 wm(view).getDockableWindow("isabelle-raw-output") match {
80 case dockable: Raw_Output_Dockable => Some(dockable)
84 def docked_protocol(view: View): Option[Protocol_Dockable] =
85 wm(view).getDockableWindow("isabelle-protocol") match {
86 case dockable: Protocol_Dockable => Some(dockable)
90 def docked_monitor(view: View): Option[Monitor_Dockable] =
91 wm(view).getDockableWindow("isabelle-monitor") match {
92 case dockable: Monitor_Dockable => Some(dockable)
97 /* continuous checking */
99 private val CONTINUOUS_CHECKING = "editor_continuous_checking"
101 def continuous_checking: Boolean = PIDE.options.bool(CONTINUOUS_CHECKING)
103 def continuous_checking_=(b: Boolean)
105 Swing_Thread.require()
107 if (continuous_checking != b) {
108 PIDE.options.bool(CONTINUOUS_CHECKING) = b
109 PIDE.options_changed()
114 def set_continuous_checking() { continuous_checking = true }
115 def reset_continuous_checking() { continuous_checking = false }
116 def toggle_continuous_checking() { continuous_checking = !continuous_checking }
118 class Continuous_Checking extends CheckBox("Continuous checking")
120 tooltip = "Continuous checking of proof document (visible and required parts)"
121 reactions += { case ButtonClicked(_) => continuous_checking = selected }
122 def load() { selected = continuous_checking }
127 /* required document nodes */
129 private def node_required_update(view: View, toggle: Boolean = false, set: Boolean = false)
131 Swing_Thread.require()
132 PIDE.document_model(view.getBuffer) match {
134 model.node_required = (if (toggle) !model.node_required else set)
139 def set_node_required(view: View) { node_required_update(view, set = true) }
140 def reset_node_required(view: View) { node_required_update(view, set = false) }
141 def toggle_node_required(view: View) { node_required_update(view, toggle = true) }
146 def reset_font_size(view: View): Unit =
147 Rendering.font_size_change(view, _ => PIDE.options.int("jedit_reset_font_size"))
149 def increase_font_size(view: View): Unit =
150 Rendering.font_size_change(view, i => i + ((i / 10) max 1))
152 def decrease_font_size(view: View): Unit =
153 Rendering.font_size_change(view, i => i - ((i / 10) max 1))
156 /* structured edits */
158 def insert_line_padding(text_area: JEditTextArea, text: String)
160 val buffer = text_area.getBuffer
161 JEdit_Lib.buffer_edit(buffer) {
163 if (text_area.getSelectionCount == 0) {
164 def pad(range: Text.Range): String =
165 if (JEdit_Lib.try_get_text(buffer, range) == Some("\n")) "" else "\n"
167 val caret = JEdit_Lib.point_range(buffer, text_area.getCaretPosition)
168 val before_caret = JEdit_Lib.point_range(buffer, caret.start - 1)
169 pad(before_caret) + text + pad(caret)
172 text_area.setSelectedText(text1)
177 snapshot: Document.Snapshot,
180 exec_id: Document_ID.Exec,
183 snapshot.state.execs.get(exec_id).map(_.command) match {
184 case Some(command) =>
185 snapshot.node.command_start(command) match {
187 JEdit_Lib.buffer_edit(buffer) {
188 val range = command.proper_range + start
191 JEdit_Lib.try_get_text(buffer, Text.Range(range.length - 1, range.length))
194 case Some(s) => if (Symbol.is_blank(s)) "" else " "
196 buffer.insert(start + range.length, pad + s)
199 buffer.remove(start, range.length)
200 buffer.insert(start, s)
212 def complete(view: View)
214 Completion_Popup.Text_Area(view.getTextArea) match {
215 case Some(text_area_completion) =>
216 text_area_completion.action(immediate = true, explicit = true)
217 case None => CompleteWord.completeWord(view)
224 def control_sub(text_area: JEditTextArea)
225 { Token_Markup.edit_control_style(text_area, Symbol.sub_decoded) }
227 def control_sup(text_area: JEditTextArea)
228 { Token_Markup.edit_control_style(text_area, Symbol.sup_decoded) }
230 def control_bold(text_area: JEditTextArea)
231 { Token_Markup.edit_control_style(text_area, Symbol.bold_decoded) }
233 def control_reset(text_area: JEditTextArea)
234 { Token_Markup.edit_control_style(text_area, "") }
239 private def enclose_input(text_area: JEditTextArea, s1: String, s2: String)
241 s1.foreach(text_area.userInput(_))
242 s2.foreach(text_area.userInput(_))
243 s2.foreach(_ => text_area.goToPrevCharacter(false))
246 def input_bsub(text_area: JEditTextArea)
247 { enclose_input(text_area, Symbol.bsub_decoded, Symbol.esub_decoded) }
249 def input_bsup(text_area: JEditTextArea)
250 { enclose_input(text_area, Symbol.bsup_decoded, Symbol.esup_decoded) }