src/Tools/jEdit/src/pretty_text_area.scala
author wenzelm
Tue, 03 Nov 2015 16:49:44 +0100
changeset 61561 f35786faee6c
parent 61556 0d4ee4168e41
child 61570 f26a4d5e82b5
permissions -rw-r--r--
prefer Isabelle/Scala Future;
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
     1
/*  Title:      Tools/jEdit/src/pretty_text_area.scala
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
     2
    Author:     Makarius
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
     3
50160
a29be9d067d2 tuned comment;
wenzelm
parents: 49700
diff changeset
     4
GUI component for pretty-printed text with markup, rendered like jEdit
a29be9d067d2 tuned comment;
wenzelm
parents: 49700
diff changeset
     5
text area.
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
     6
*/
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
     7
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
     8
package isabelle.jedit
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
     9
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    10
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    11
import isabelle._
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    12
58766
5bab56d3dcd4 tuned imports;
wenzelm
parents: 57612
diff changeset
    13
import java.awt.{Color, Font, Toolkit, Window}
53226
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
    14
import java.awt.event.KeyEvent
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
    15
import javax.swing.JTextField
56885
3020f6bbd119 clarified GUI events, e.g. relevant for insert via completion;
wenzelm
parents: 56883
diff changeset
    16
import javax.swing.event.{DocumentListener, DocumentEvent}
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
    17
56907
0f3c375fd27c enable "PIDE" docking framework by default, and rely on its "Detach" menu item;
wenzelm
parents: 56885
diff changeset
    18
import scala.swing.{Label, Component}
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
    19
import scala.util.matching.Regex
56881
15e18540df10 more uniform detach button;
wenzelm
parents: 56880
diff changeset
    20
49422
21f77309d93a minimal clipboard support (similar to org.lobobrowser.html.gui.HtmlBlockPanel);
wenzelm
parents: 49421
diff changeset
    21
import org.gjt.sp.jedit.{jEdit, View, Registers}
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    22
import org.gjt.sp.jedit.textarea.{AntiAlias, JEditEmbeddedTextArea}
50542
58bd88159f8f fold handling within Pretty_Text_Area, based on formal document content, which is static here;
wenzelm
parents: 50538
diff changeset
    23
import org.gjt.sp.jedit.syntax.SyntaxStyle
50640
b35bd8778754 more informative error;
wenzelm
parents: 50542
diff changeset
    24
import org.gjt.sp.util.{SyntaxUtilities, Log}
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    25
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    26
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    27
object Pretty_Text_Area
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    28
{
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
    29
  /* auxiliary */
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
    30
50501
6f41f1646617 more careful handling of Dialog_Result, with active area and color feedback;
wenzelm
parents: 50306
diff changeset
    31
  private def document_state(base_snapshot: Document.Snapshot, base_results: Command.Results,
6f41f1646617 more careful handling of Dialog_Result, with active area and color feedback;
wenzelm
parents: 50306
diff changeset
    32
    formatted_body: XML.Body): (String, Document.State) =
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    33
  {
52530
99dd8b4ef3fe explicit module Document_ID as source of globally unique identifiers across ML/Scala;
wenzelm
parents: 52527
diff changeset
    34
    val command = Command.rich_text(Document_ID.make(), base_results, formatted_body)
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    35
    val node_name = command.node_name
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    36
    val edits: List[Document.Edit_Text] =
49414
d7b5fb2e9ca2 some support for inital command markup;
wenzelm
parents: 49413
diff changeset
    37
      List(node_name -> Document.Node.Edits(List(Text.Edit.insert(0, command.source))))
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    38
49419
e2726211f834 pass base_snapshot to enable hyperlinks into other nodes;
wenzelm
parents: 49416
diff changeset
    39
    val state0 = base_snapshot.state.define_command(command)
e2726211f834 pass base_snapshot to enable hyperlinks into other nodes;
wenzelm
parents: 49416
diff changeset
    40
    val version0 = base_snapshot.version
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    41
    val nodes0 = version0.nodes
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    42
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    43
    val nodes1 = nodes0 + (node_name -> nodes0(node_name).update_commands(Linear_Set(command)))
59077
7e0d3da6e6d8 node-specific syntax, with base_syntax as default;
wenzelm
parents: 59076
diff changeset
    44
    val version1 = Document.Version.make(nodes1)
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    45
    val state1 =
56711
ef3d00153e3b tuned signature;
wenzelm
parents: 56662
diff changeset
    46
      state0.continue_history(Future.value(version0), edits, Future.value(version1))
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    47
        .define_version(version1, state0.the_assignment(version0))
52530
99dd8b4ef3fe explicit module Document_ID as source of globally unique identifiers across ML/Scala;
wenzelm
parents: 52527
diff changeset
    48
        .assign(version1.id, List(command.id -> List(Document_ID.make())))._2
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    49
49416
1053a564dd25 some actual rich text markup via XML.content_markup;
wenzelm
parents: 49414
diff changeset
    50
    (command.source, state1)
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    51
  }
51495
5944b20c41bf apply small result immediately, to avoid visible delay of text update after window move;
wenzelm
parents: 51493
diff changeset
    52
5944b20c41bf apply small result immediately, to avoid visible delay of text update after window move;
wenzelm
parents: 51493
diff changeset
    53
  private def text_rendering(base_snapshot: Document.Snapshot, base_results: Command.Results,
5944b20c41bf apply small result immediately, to avoid visible delay of text update after window move;
wenzelm
parents: 51493
diff changeset
    54
    formatted_body: XML.Body): (String, Rendering) =
5944b20c41bf apply small result immediately, to avoid visible delay of text update after window move;
wenzelm
parents: 51493
diff changeset
    55
  {
5944b20c41bf apply small result immediately, to avoid visible delay of text update after window move;
wenzelm
parents: 51493
diff changeset
    56
    val (text, state) = document_state(base_snapshot, base_results, formatted_body)
5944b20c41bf apply small result immediately, to avoid visible delay of text update after window move;
wenzelm
parents: 51493
diff changeset
    57
    val rendering = Rendering(state.snapshot(), PIDE.options.value)
5944b20c41bf apply small result immediately, to avoid visible delay of text update after window move;
wenzelm
parents: 51493
diff changeset
    58
    (text, rendering)
5944b20c41bf apply small result immediately, to avoid visible delay of text update after window move;
wenzelm
parents: 51493
diff changeset
    59
  }
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    60
}
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    61
50538
48cb76b785da init gutter according to view properties, which improves symmetry of windows and allows use of folds etc;
wenzelm
parents: 50507
diff changeset
    62
class Pretty_Text_Area(
48cb76b785da init gutter according to view properties, which improves symmetry of windows and allows use of folds etc;
wenzelm
parents: 50507
diff changeset
    63
  view: View,
50915
12de8ea66f54 close tooltip after Active.action, to make it look more interactive (notably due to lack of dynamic update);
wenzelm
parents: 50849
diff changeset
    64
  close_action: () => Unit = () => (),
50743
44571ac53fed propagate keys to enclosing view like org.gjt.sp.jedit.gui.CompletionPopup, but without its KeyEventInterceptor;
wenzelm
parents: 50726
diff changeset
    65
  propagate_keys: Boolean = false) extends JEditEmbeddedTextArea
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    66
{
49446
b8d8f738bf63 more direct GUI component;
wenzelm
parents: 49422
diff changeset
    67
  text_area =>
b8d8f738bf63 more direct GUI component;
wenzelm
parents: 49422
diff changeset
    68
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57042
diff changeset
    69
  GUI_Thread.require {}
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    70
55826
wenzelm
parents: 55825
diff changeset
    71
  private var current_font_info: Font_Info = Font_Info.main()
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    72
  private var current_body: XML.Body = Nil
52972
8fd8e1c14988 tuned signature;
wenzelm
parents: 52548
diff changeset
    73
  private var current_base_snapshot = Document.Snapshot.init
50507
9605b0d93d1e more formal class Command.Results;
wenzelm
parents: 50501
diff changeset
    74
  private var current_base_results = Command.Results.empty
50199
6d04e2422769 quasi-abstract module Rendering, with Isabelle-specific implementation;
wenzelm
parents: 50195
diff changeset
    75
  private var current_rendering: Rendering =
50501
6f41f1646617 more careful handling of Dialog_Result, with active area and color feedback;
wenzelm
parents: 50306
diff changeset
    76
    Pretty_Text_Area.text_rendering(current_base_snapshot, current_base_results, Nil)._2
61561
f35786faee6c prefer Isabelle/Scala Future;
wenzelm
parents: 61556
diff changeset
    77
  private var future_refresh: Option[Future[Unit]] = None
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    78
50306
b655d2d0406d updated to jedit-5.0.0;
wenzelm
parents: 50216
diff changeset
    79
  private val rich_text_area =
50915
12de8ea66f54 close tooltip after Active.action, to make it look more interactive (notably due to lack of dynamic update);
wenzelm
parents: 50849
diff changeset
    80
    new Rich_Text_Area(view, text_area, () => current_rendering, close_action,
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
    81
      get_search_pattern _, caret_visible = false, enable_hovering = true)
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
    82
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
    83
  private var current_search_pattern: Option[Regex] = None
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57042
diff changeset
    84
  def get_search_pattern(): Option[Regex] = GUI_Thread.require { current_search_pattern }
49416
1053a564dd25 some actual rich text markup via XML.content_markup;
wenzelm
parents: 49414
diff changeset
    85
51451
e4203ebfe750 recovered special background handling from 8d6e478934dc, particularly relevant for gutter border;
wenzelm
parents: 51449
diff changeset
    86
  def get_background(): Option[Color] = None
e4203ebfe750 recovered special background handling from 8d6e478934dc, particularly relevant for gutter border;
wenzelm
parents: 51449
diff changeset
    87
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    88
  def refresh()
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    89
  {
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57042
diff changeset
    90
    GUI_Thread.require {}
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    91
55825
694833e3e4a0 tuned signature -- separate module Font_Info;
wenzelm
parents: 53277
diff changeset
    92
    val font = current_font_info.font
50168
4a575ef46466 always refresh font metrics, to help window size calculation (amending 2585c81d840a);
wenzelm
parents: 50166
diff changeset
    93
    getPainter.setFont(font)
4a575ef46466 always refresh font metrics, to help window size calculation (amending 2585c81d840a);
wenzelm
parents: 50166
diff changeset
    94
    getPainter.setAntiAlias(new AntiAlias(jEdit.getProperty("view.antiAlias")))
51497
7e8968c9a549 no censorship of "view.fracFontMetrics", although it often degrades rendering quality;
wenzelm
parents: 51495
diff changeset
    95
    getPainter.setFractionalFontMetricsEnabled(jEdit.getBooleanProperty("view.fracFontMetrics"))
55825
694833e3e4a0 tuned signature -- separate module Font_Info;
wenzelm
parents: 53277
diff changeset
    96
    getPainter.setStyles(
694833e3e4a0 tuned signature -- separate module Font_Info;
wenzelm
parents: 53277
diff changeset
    97
      SyntaxUtilities.loadStyles(current_font_info.family, current_font_info.size.round))
56789
f377ddf1cc52 tuned whitespace;
wenzelm
parents: 56730
diff changeset
    98
    getPainter.setLineExtraSpacing(jEdit.getIntegerProperty("options.textarea.lineSpacing", 0))
50168
4a575ef46466 always refresh font metrics, to help window size calculation (amending 2585c81d840a);
wenzelm
parents: 50166
diff changeset
    99
50542
58bd88159f8f fold handling within Pretty_Text_Area, based on formal document content, which is static here;
wenzelm
parents: 50538
diff changeset
   100
    val fold_line_style = new Array[SyntaxStyle](4)
58bd88159f8f fold handling within Pretty_Text_Area, based on formal document content, which is static here;
wenzelm
parents: 50538
diff changeset
   101
    for (i <- 0 to 3) {
58bd88159f8f fold handling within Pretty_Text_Area, based on formal document content, which is static here;
wenzelm
parents: 50538
diff changeset
   102
      fold_line_style(i) =
58bd88159f8f fold handling within Pretty_Text_Area, based on formal document content, which is static here;
wenzelm
parents: 50538
diff changeset
   103
        SyntaxUtilities.parseStyle(
58bd88159f8f fold handling within Pretty_Text_Area, based on formal document content, which is static here;
wenzelm
parents: 50538
diff changeset
   104
          jEdit.getProperty("view.style.foldLine." + i),
55825
694833e3e4a0 tuned signature -- separate module Font_Info;
wenzelm
parents: 53277
diff changeset
   105
          current_font_info.family, current_font_info.size.round, true)
50542
58bd88159f8f fold handling within Pretty_Text_Area, based on formal document content, which is static here;
wenzelm
parents: 50538
diff changeset
   106
    }
58bd88159f8f fold handling within Pretty_Text_Area, based on formal document content, which is static here;
wenzelm
parents: 50538
diff changeset
   107
    getPainter.setFoldLineStyle(fold_line_style)
58bd88159f8f fold handling within Pretty_Text_Area, based on formal document content, which is static here;
wenzelm
parents: 50538
diff changeset
   108
51439
b10b64679c5b more precise tooltip window size (NB: dimensions are known after layout pack, before making content visible);
wenzelm
parents: 50915
diff changeset
   109
    getGutter.setForeground(jEdit.getColorProperty("view.gutter.fgColor"))
b10b64679c5b more precise tooltip window size (NB: dimensions are known after layout pack, before making content visible);
wenzelm
parents: 50915
diff changeset
   110
    getGutter.setBackground(jEdit.getColorProperty("view.gutter.bgColor"))
51451
e4203ebfe750 recovered special background handling from 8d6e478934dc, particularly relevant for gutter border;
wenzelm
parents: 51449
diff changeset
   111
    get_background().map(bg => { getPainter.setBackground(bg); getGutter.setBackground(bg) })
51439
b10b64679c5b more precise tooltip window size (NB: dimensions are known after layout pack, before making content visible);
wenzelm
parents: 50915
diff changeset
   112
    getGutter.setHighlightedForeground(jEdit.getColorProperty("view.gutter.highlightColor"))
b10b64679c5b more precise tooltip window size (NB: dimensions are known after layout pack, before making content visible);
wenzelm
parents: 50915
diff changeset
   113
    getGutter.setFoldColor(jEdit.getColorProperty("view.gutter.foldColor"))
b10b64679c5b more precise tooltip window size (NB: dimensions are known after layout pack, before making content visible);
wenzelm
parents: 50915
diff changeset
   114
    getGutter.setFont(jEdit.getFontProperty("view.gutter.font"))
b10b64679c5b more precise tooltip window size (NB: dimensions are known after layout pack, before making content visible);
wenzelm
parents: 50915
diff changeset
   115
    getGutter.setBorder(0,
b10b64679c5b more precise tooltip window size (NB: dimensions are known after layout pack, before making content visible);
wenzelm
parents: 50915
diff changeset
   116
      jEdit.getColorProperty("view.gutter.focusBorderColor"),
b10b64679c5b more precise tooltip window size (NB: dimensions are known after layout pack, before making content visible);
wenzelm
parents: 50915
diff changeset
   117
      jEdit.getColorProperty("view.gutter.noFocusBorderColor"),
b10b64679c5b more precise tooltip window size (NB: dimensions are known after layout pack, before making content visible);
wenzelm
parents: 50915
diff changeset
   118
      getPainter.getBackground)
60250
baf2c8fddaa4 proper fold painter according to jEdit options, not the hardwired default of JEditEmbeddedTextArea;
wenzelm
parents: 59286
diff changeset
   119
    getGutter.setFoldPainter(view.getTextArea.getFoldPainter)
51439
b10b64679c5b more precise tooltip window size (NB: dimensions are known after layout pack, before making content visible);
wenzelm
parents: 50915
diff changeset
   120
    getGutter.setGutterEnabled(jEdit.getBooleanProperty("view.gutter.enabled"))
b10b64679c5b more precise tooltip window size (NB: dimensions are known after layout pack, before making content visible);
wenzelm
parents: 50915
diff changeset
   121
50166
2585c81d840a take component width as indication if it is already visible/layed-out, to avoid multiple formatting with minimal margin;
wenzelm
parents: 50160
diff changeset
   122
    if (getWidth > 0) {
51492
eaa1c4cc1106 more explicit Pretty.Metric, with clear distinction of unit (space width) vs. average char width (for visual adjustments) -- NB: Pretty formatting works via full space characters (despite a981a5c8a505 and 70f7483df9cb);
wenzelm
parents: 51469
diff changeset
   123
      val metric = JEdit_Lib.pretty_metric(getPainter)
51493
59d8a1031c00 allow fractional pretty margin -- avoid premature rounding;
wenzelm
parents: 51492
diff changeset
   124
      val margin = (getPainter.getWidth.toDouble / metric.unit) max 20.0
49471
97964515a676 text_rendering as managed task, with cancellation;
wenzelm
parents: 49446
diff changeset
   125
50166
2585c81d840a take component width as indication if it is already visible/layed-out, to avoid multiple formatting with minimal margin;
wenzelm
parents: 50160
diff changeset
   126
      val base_snapshot = current_base_snapshot
50501
6f41f1646617 more careful handling of Dialog_Result, with active area and color feedback;
wenzelm
parents: 50306
diff changeset
   127
      val base_results = current_base_results
51492
eaa1c4cc1106 more explicit Pretty.Metric, with clear distinction of unit (space width) vs. average char width (for visual adjustments) -- NB: Pretty formatting works via full space characters (despite a981a5c8a505 and 70f7483df9cb);
wenzelm
parents: 51469
diff changeset
   128
      val formatted_body = Pretty.formatted(current_body, margin, metric)
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   129
61561
f35786faee6c prefer Isabelle/Scala Future;
wenzelm
parents: 61556
diff changeset
   130
      future_refresh.map(_.cancel)
61193
wenzelm
parents: 60250
diff changeset
   131
      future_refresh =
61561
f35786faee6c prefer Isabelle/Scala Future;
wenzelm
parents: 61556
diff changeset
   132
        Some(Future.fork {
51498
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   133
          val (text, rendering) =
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   134
            try { Pretty_Text_Area.text_rendering(base_snapshot, base_results, formatted_body) }
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   135
            catch { case exn: Throwable => Log.log(Log.ERROR, this, exn); throw exn }
56860
dc71c3d0e909 tuned signature;
wenzelm
parents: 56789
diff changeset
   136
          Exn.Interrupt.expose()
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   137
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57042
diff changeset
   138
          GUI_Thread.later {
51498
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   139
            current_rendering = rendering
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   140
            JEdit_Lib.buffer_edit(getBuffer) {
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   141
              rich_text_area.active_reset()
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   142
              getBuffer.setReadOnly(false)
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   143
              getBuffer.setFoldHandler(new Fold_Handling.Document_Fold_Handler(rendering))
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   144
              setText(text)
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   145
              setCaretPosition(0)
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   146
              getBuffer.setReadOnly(true)
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   147
            }
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   148
          }
56730
e723f041b6d0 tuned signature -- separate pool for JFuture tasks, which can be canceled;
wenzelm
parents: 56711
diff changeset
   149
        })
50166
2585c81d840a take component width as indication if it is already visible/layed-out, to avoid multiple formatting with minimal margin;
wenzelm
parents: 50160
diff changeset
   150
    }
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   151
  }
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   152
55825
694833e3e4a0 tuned signature -- separate module Font_Info;
wenzelm
parents: 53277
diff changeset
   153
  def resize(font_info: Font_Info)
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   154
  {
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57042
diff changeset
   155
    GUI_Thread.require {}
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   156
55825
694833e3e4a0 tuned signature -- separate module Font_Info;
wenzelm
parents: 53277
diff changeset
   157
    current_font_info = font_info
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   158
    refresh()
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   159
  }
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   160
50501
6f41f1646617 more careful handling of Dialog_Result, with active area and color feedback;
wenzelm
parents: 50306
diff changeset
   161
  def update(base_snapshot: Document.Snapshot, base_results: Command.Results, body: XML.Body)
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   162
  {
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57042
diff changeset
   163
    GUI_Thread.require {}
49419
e2726211f834 pass base_snapshot to enable hyperlinks into other nodes;
wenzelm
parents: 49416
diff changeset
   164
    require(!base_snapshot.is_outdated)
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   165
49419
e2726211f834 pass base_snapshot to enable hyperlinks into other nodes;
wenzelm
parents: 49416
diff changeset
   166
    current_base_snapshot = base_snapshot
50501
6f41f1646617 more careful handling of Dialog_Result, with active area and color feedback;
wenzelm
parents: 50306
diff changeset
   167
    current_base_results = base_results
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   168
    current_body = body
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   169
    refresh()
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   170
  }
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   171
56880
f8c1d2583699 tuned signature;
wenzelm
parents: 56860
diff changeset
   172
  def detach
f8c1d2583699 tuned signature;
wenzelm
parents: 56860
diff changeset
   173
  {
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57042
diff changeset
   174
    GUI_Thread.require {}
56880
f8c1d2583699 tuned signature;
wenzelm
parents: 56860
diff changeset
   175
    Info_Dockable(view, current_base_snapshot, current_base_results, current_body)
f8c1d2583699 tuned signature;
wenzelm
parents: 56860
diff changeset
   176
  }
f8c1d2583699 tuned signature;
wenzelm
parents: 56860
diff changeset
   177
56918
a442dc6d244d clarified detach_operation: ignore empty output;
wenzelm
parents: 56907
diff changeset
   178
  def detach_operation: Option[() => Unit] =
a442dc6d244d clarified detach_operation: ignore empty output;
wenzelm
parents: 56907
diff changeset
   179
    if (current_body.isEmpty) None else Some(() => detach)
a442dc6d244d clarified detach_operation: ignore empty output;
wenzelm
parents: 56907
diff changeset
   180
49422
21f77309d93a minimal clipboard support (similar to org.lobobrowser.html.gui.HtmlBlockPanel);
wenzelm
parents: 49421
diff changeset
   181
56881
15e18540df10 more uniform detach button;
wenzelm
parents: 56880
diff changeset
   182
  /* common GUI components */
15e18540df10 more uniform detach button;
wenzelm
parents: 56880
diff changeset
   183
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   184
  val search_label: Component = new Label("Search:") {
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   185
    tooltip = "Search and highlight output via regular expression"
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   186
  }
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   187
57042
5576d22abf3c tuned signature;
wenzelm
parents: 56918
diff changeset
   188
  val search_field: Component =
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   189
    Component.wrap(new Completion_Popup.History_Text_Field("isabelle-search")
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   190
      {
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   191
        private val input_delay =
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57042
diff changeset
   192
          GUI_Thread.delay_last(PIDE.options.seconds("editor_input_delay")) {
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   193
            search_action(this)
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   194
          }
56885
3020f6bbd119 clarified GUI events, e.g. relevant for insert via completion;
wenzelm
parents: 56883
diff changeset
   195
        getDocument.addDocumentListener(new DocumentListener {
3020f6bbd119 clarified GUI events, e.g. relevant for insert via completion;
wenzelm
parents: 56883
diff changeset
   196
          def changedUpdate(e: DocumentEvent) { input_delay.invoke() }
3020f6bbd119 clarified GUI events, e.g. relevant for insert via completion;
wenzelm
parents: 56883
diff changeset
   197
          def insertUpdate(e: DocumentEvent) { input_delay.invoke() }
3020f6bbd119 clarified GUI events, e.g. relevant for insert via completion;
wenzelm
parents: 56883
diff changeset
   198
          def removeUpdate(e: DocumentEvent) { input_delay.invoke() }
3020f6bbd119 clarified GUI events, e.g. relevant for insert via completion;
wenzelm
parents: 56883
diff changeset
   199
        })
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   200
        setColumns(20)
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   201
        setToolTipText(search_label.tooltip)
59286
ac74eedb910a GUI.imitate_font: more explicit result size, e.g. relevant for caching;
wenzelm
parents: 59224
diff changeset
   202
        setFont(GUI.imitate_font(getFont, Font_Info.main_family(), 1.2))
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   203
      })
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   204
57042
5576d22abf3c tuned signature;
wenzelm
parents: 56918
diff changeset
   205
  private val search_field_foreground = search_field.foreground
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   206
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   207
  private def search_action(text_field: JTextField)
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   208
  {
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   209
    val (pattern, ok) =
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   210
      text_field.getText match {
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   211
        case null | "" => (None, true)
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   212
        case s =>
59224
wenzelm
parents: 59077
diff changeset
   213
          val re = Library.make_regex(s)
wenzelm
parents: 59077
diff changeset
   214
          (re, re.isDefined)
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   215
      }
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   216
    if (current_search_pattern != pattern) {
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   217
      current_search_pattern = pattern
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   218
      text_area.getPainter.repaint()
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   219
    }
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   220
    text_field.setForeground(
57042
5576d22abf3c tuned signature;
wenzelm
parents: 56918
diff changeset
   221
      if (ok) search_field_foreground else current_rendering.error_color)
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   222
  }
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   223
56881
15e18540df10 more uniform detach button;
wenzelm
parents: 56880
diff changeset
   224
50726
27478c11f63c more elementary key handling: listen to low-level KEY_PRESSED events (without consuming);
wenzelm
parents: 50640
diff changeset
   225
  /* key handling */
49422
21f77309d93a minimal clipboard support (similar to org.lobobrowser.html.gui.HtmlBlockPanel);
wenzelm
parents: 49421
diff changeset
   226
53226
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   227
  addKeyListener(JEdit_Lib.key_listener(
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   228
    key_pressed = (evt: KeyEvent) =>
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   229
      {
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   230
        evt.getKeyCode match {
58835
462ec23aa92f hardwired imitation of copy.shortcut2 default;
wenzelm
parents: 58766
diff changeset
   231
          case KeyEvent.VK_C | KeyEvent.VK_INSERT
56323
e925118b1875 do not absorb vacuous copy operation, e.g. relevant when tooltip has focus but no selection, while the main text area has a selection but no focus;
wenzelm
parents: 55826
diff changeset
   232
          if (evt.getModifiers & Toolkit.getDefaultToolkit.getMenuShortcutKeyMask) != 0 &&
e925118b1875 do not absorb vacuous copy operation, e.g. relevant when tooltip has focus but no selection, while the main text area has a selection but no focus;
wenzelm
parents: 55826
diff changeset
   233
              text_area.getSelectionCount != 0 =>
53226
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   234
            Registers.copy(text_area, '$')
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   235
            evt.consume
53227
68cc55ceb7f6 more standard key handling according to jEdit (with workaround);
wenzelm
parents: 53226
diff changeset
   236
53226
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   237
          case KeyEvent.VK_A
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   238
          if (evt.getModifiers & Toolkit.getDefaultToolkit.getMenuShortcutKeyMask) != 0 =>
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   239
            text_area.selectAll
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   240
            evt.consume
53227
68cc55ceb7f6 more standard key handling according to jEdit (with workaround);
wenzelm
parents: 53226
diff changeset
   241
68cc55ceb7f6 more standard key handling according to jEdit (with workaround);
wenzelm
parents: 53226
diff changeset
   242
          case KeyEvent.VK_ESCAPE =>
53244
ec6011bf2362 dismiss popups more uniformly;
wenzelm
parents: 53231
diff changeset
   243
            if (PIDE.dismissed_popups(view)) evt.consume
53227
68cc55ceb7f6 more standard key handling according to jEdit (with workaround);
wenzelm
parents: 53226
diff changeset
   244
53226
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   245
          case _ =>
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   246
        }
53231
423e29f1f304 avoid complication and event duplication due to KeyEventInterceptor -- NB: popup has focus within root window, it is closed on loss of focus;
wenzelm
parents: 53227
diff changeset
   247
        if (propagate_keys) JEdit_Lib.propagate_key(view, evt)
53226
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   248
      },
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   249
    key_typed = (evt: KeyEvent) =>
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   250
      {
53231
423e29f1f304 avoid complication and event duplication due to KeyEventInterceptor -- NB: popup has focus within root window, it is closed on loss of focus;
wenzelm
parents: 53227
diff changeset
   251
        if (propagate_keys) JEdit_Lib.propagate_key(view, evt)
49422
21f77309d93a minimal clipboard support (similar to org.lobobrowser.html.gui.HtmlBlockPanel);
wenzelm
parents: 49421
diff changeset
   252
      }
53226
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   253
    )
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   254
  )
49422
21f77309d93a minimal clipboard support (similar to org.lobobrowser.html.gui.HtmlBlockPanel);
wenzelm
parents: 49421
diff changeset
   255
21f77309d93a minimal clipboard support (similar to org.lobobrowser.html.gui.HtmlBlockPanel);
wenzelm
parents: 49421
diff changeset
   256
21f77309d93a minimal clipboard support (similar to org.lobobrowser.html.gui.HtmlBlockPanel);
wenzelm
parents: 49421
diff changeset
   257
  /* init */
21f77309d93a minimal clipboard support (similar to org.lobobrowser.html.gui.HtmlBlockPanel);
wenzelm
parents: 49421
diff changeset
   258
49472
ba2c0d0cd429 no caret painting;
wenzelm
parents: 49471
diff changeset
   259
  getPainter.setStructureHighlightEnabled(false)
49475
8f3a3adadd5a tuned painter;
wenzelm
parents: 49472
diff changeset
   260
  getPainter.setLineHighlightEnabled(false)
8f3a3adadd5a tuned painter;
wenzelm
parents: 49472
diff changeset
   261
59076
65babcd8b0e6 clarified token marker / syntax for mode vs. buffer;
wenzelm
parents: 58835
diff changeset
   262
  getBuffer.setTokenMarker(Isabelle.mode_token_marker("isabelle-output").get)
49446
b8d8f738bf63 more direct GUI component;
wenzelm
parents: 49422
diff changeset
   263
  getBuffer.setReadOnly(true)
53002
9dd1a6dcebfd imitate "noWordSep" of isabelle mode, e.g. relevant for word selection via double-click;
wenzelm
parents: 53001
diff changeset
   264
  getBuffer.setStringProperty("noWordSep", "_'.?")
49475
8f3a3adadd5a tuned painter;
wenzelm
parents: 49472
diff changeset
   265
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
   266
  rich_text_area.activate()
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   267
}
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   268