src/Tools/jEdit/src/pretty_text_area.scala
author wenzelm
Wed, 21 May 2014 14:42:45 +0200
changeset 57042 5576d22abf3c
parent 56918 a442dc6d244d
child 57612 990ffb84489b
permissions -rw-r--r--
tuned signature;
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
50726
27478c11f63c more elementary key handling: listen to low-level KEY_PRESSED events (without consuming);
wenzelm
parents: 50640
diff changeset
    13
import java.awt.{Color, Font, FontMetrics, 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
56730
e723f041b6d0 tuned signature -- separate pool for JFuture tasks, which can be canceled;
wenzelm
parents: 56711
diff changeset
    18
import java.util.concurrent.{Future => JFuture}
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    19
56907
0f3c375fd27c enable "PIDE" docking framework by default, and rely on its "Detach" menu item;
wenzelm
parents: 56885
diff changeset
    20
import scala.swing.{Label, Component}
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
    21
import scala.util.matching.Regex
56881
15e18540df10 more uniform detach button;
wenzelm
parents: 56880
diff changeset
    22
49422
21f77309d93a minimal clipboard support (similar to org.lobobrowser.html.gui.HtmlBlockPanel);
wenzelm
parents: 49421
diff changeset
    23
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
    24
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
    25
import org.gjt.sp.jedit.syntax.SyntaxStyle
50640
b35bd8778754 more informative error;
wenzelm
parents: 50542
diff changeset
    26
import org.gjt.sp.util.{SyntaxUtilities, Log}
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    27
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    28
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    29
object Pretty_Text_Area
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    30
{
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
    31
  /* auxiliary */
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
    32
50501
6f41f1646617 more careful handling of Dialog_Result, with active area and color feedback;
wenzelm
parents: 50306
diff changeset
    33
  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
    34
    formatted_body: XML.Body): (String, Document.State) =
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    35
  {
52530
99dd8b4ef3fe explicit module Document_ID as source of globally unique identifiers across ML/Scala;
wenzelm
parents: 52527
diff changeset
    36
    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
    37
    val node_name = command.node_name
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    38
    val edits: List[Document.Edit_Text] =
49414
d7b5fb2e9ca2 some support for inital command markup;
wenzelm
parents: 49413
diff changeset
    39
      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
    40
49419
e2726211f834 pass base_snapshot to enable hyperlinks into other nodes;
wenzelm
parents: 49416
diff changeset
    41
    val state0 = base_snapshot.state.define_command(command)
e2726211f834 pass base_snapshot to enable hyperlinks into other nodes;
wenzelm
parents: 49416
diff changeset
    42
    val version0 = base_snapshot.version
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    43
    val nodes0 = version0.nodes
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    44
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    45
    val nodes1 = nodes0 + (node_name -> nodes0(node_name).update_commands(Linear_Set(command)))
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    46
    val version1 = Document.Version.make(version0.syntax, nodes1)
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    47
    val state1 =
56711
ef3d00153e3b tuned signature;
wenzelm
parents: 56662
diff changeset
    48
      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
    49
        .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
    50
        .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
    51
49416
1053a564dd25 some actual rich text markup via XML.content_markup;
wenzelm
parents: 49414
diff changeset
    52
    (command.source, state1)
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    53
  }
51495
5944b20c41bf apply small result immediately, to avoid visible delay of text update after window move;
wenzelm
parents: 51493
diff changeset
    54
5944b20c41bf apply small result immediately, to avoid visible delay of text update after window move;
wenzelm
parents: 51493
diff changeset
    55
  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
    56
    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
    57
  {
5944b20c41bf apply small result immediately, to avoid visible delay of text update after window move;
wenzelm
parents: 51493
diff changeset
    58
    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
    59
    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
    60
    (text, rendering)
5944b20c41bf apply small result immediately, to avoid visible delay of text update after window move;
wenzelm
parents: 51493
diff changeset
    61
  }
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    62
}
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
    63
50538
48cb76b785da init gutter according to view properties, which improves symmetry of windows and allows use of folds etc;
wenzelm
parents: 50507
diff changeset
    64
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
    65
  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
    66
  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
    67
  propagate_keys: Boolean = false) extends JEditEmbeddedTextArea
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    68
{
49446
b8d8f738bf63 more direct GUI component;
wenzelm
parents: 49422
diff changeset
    69
  text_area =>
b8d8f738bf63 more direct GUI component;
wenzelm
parents: 49422
diff changeset
    70
56662
f373fb77e0a4 avoid "Adaptation of argument list by inserting ()" -- deprecated in scala-2.11.0;
wenzelm
parents: 56360
diff changeset
    71
  Swing_Thread.require {}
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    72
55826
wenzelm
parents: 55825
diff changeset
    73
  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
    74
  private var current_body: XML.Body = Nil
52972
8fd8e1c14988 tuned signature;
wenzelm
parents: 52548
diff changeset
    75
  private var current_base_snapshot = Document.Snapshot.init
50507
9605b0d93d1e more formal class Command.Results;
wenzelm
parents: 50501
diff changeset
    76
  private var current_base_results = Command.Results.empty
50199
6d04e2422769 quasi-abstract module Rendering, with Isabelle-specific implementation;
wenzelm
parents: 50195
diff changeset
    77
  private var current_rendering: Rendering =
50501
6f41f1646617 more careful handling of Dialog_Result, with active area and color feedback;
wenzelm
parents: 50306
diff changeset
    78
    Pretty_Text_Area.text_rendering(current_base_snapshot, current_base_results, Nil)._2
56730
e723f041b6d0 tuned signature -- separate pool for JFuture tasks, which can be canceled;
wenzelm
parents: 56711
diff changeset
    79
  private var future_rendering: Option[JFuture[Unit]] = None
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    80
50306
b655d2d0406d updated to jedit-5.0.0;
wenzelm
parents: 50216
diff changeset
    81
  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
    82
    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
    83
      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
    84
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
    85
  private var current_search_pattern: Option[Regex] = None
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
    86
  def get_search_pattern(): Option[Regex] = Swing_Thread.require { current_search_pattern }
49416
1053a564dd25 some actual rich text markup via XML.content_markup;
wenzelm
parents: 49414
diff changeset
    87
51451
e4203ebfe750 recovered special background handling from 8d6e478934dc, particularly relevant for gutter border;
wenzelm
parents: 51449
diff changeset
    88
  def get_background(): Option[Color] = None
e4203ebfe750 recovered special background handling from 8d6e478934dc, particularly relevant for gutter border;
wenzelm
parents: 51449
diff changeset
    89
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    90
  def refresh()
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    91
  {
56662
f373fb77e0a4 avoid "Adaptation of argument list by inserting ()" -- deprecated in scala-2.11.0;
wenzelm
parents: 56360
diff changeset
    92
    Swing_Thread.require {}
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
    93
55825
694833e3e4a0 tuned signature -- separate module Font_Info;
wenzelm
parents: 53277
diff changeset
    94
    val font = current_font_info.font
50168
4a575ef46466 always refresh font metrics, to help window size calculation (amending 2585c81d840a);
wenzelm
parents: 50166
diff changeset
    95
    getPainter.setFont(font)
4a575ef46466 always refresh font metrics, to help window size calculation (amending 2585c81d840a);
wenzelm
parents: 50166
diff changeset
    96
    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
    97
    getPainter.setFractionalFontMetricsEnabled(jEdit.getBooleanProperty("view.fracFontMetrics"))
55825
694833e3e4a0 tuned signature -- separate module Font_Info;
wenzelm
parents: 53277
diff changeset
    98
    getPainter.setStyles(
694833e3e4a0 tuned signature -- separate module Font_Info;
wenzelm
parents: 53277
diff changeset
    99
      SyntaxUtilities.loadStyles(current_font_info.family, current_font_info.size.round))
56789
f377ddf1cc52 tuned whitespace;
wenzelm
parents: 56730
diff changeset
   100
    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
   101
50542
58bd88159f8f fold handling within Pretty_Text_Area, based on formal document content, which is static here;
wenzelm
parents: 50538
diff changeset
   102
    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
   103
    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
   104
      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
   105
        SyntaxUtilities.parseStyle(
58bd88159f8f fold handling within Pretty_Text_Area, based on formal document content, which is static here;
wenzelm
parents: 50538
diff changeset
   106
          jEdit.getProperty("view.style.foldLine." + i),
55825
694833e3e4a0 tuned signature -- separate module Font_Info;
wenzelm
parents: 53277
diff changeset
   107
          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
   108
    }
58bd88159f8f fold handling within Pretty_Text_Area, based on formal document content, which is static here;
wenzelm
parents: 50538
diff changeset
   109
    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
   110
51439
b10b64679c5b more precise tooltip window size (NB: dimensions are known after layout pack, before making content visible);
wenzelm
parents: 50915
diff changeset
   111
    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
   112
    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
   113
    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
   114
    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
   115
    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
   116
    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
   117
    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
   118
      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
   119
      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
   120
      getPainter.getBackground)
b10b64679c5b more precise tooltip window size (NB: dimensions are known after layout pack, before making content visible);
wenzelm
parents: 50915
diff changeset
   121
    getGutter.setFoldPainter(getFoldPainter)
b10b64679c5b more precise tooltip window size (NB: dimensions are known after layout pack, before making content visible);
wenzelm
parents: 50915
diff changeset
   122
    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
   123
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
   124
    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
   125
      val metric = JEdit_Lib.pretty_metric(getPainter)
51493
59d8a1031c00 allow fractional pretty margin -- avoid premature rounding;
wenzelm
parents: 51492
diff changeset
   126
      val margin = (getPainter.getWidth.toDouble / metric.unit) max 20.0
49471
97964515a676 text_rendering as managed task, with cancellation;
wenzelm
parents: 49446
diff changeset
   127
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
   128
      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
   129
      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
   130
      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
   131
51498
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   132
      future_rendering.map(_.cancel(true))
56730
e723f041b6d0 tuned signature -- separate pool for JFuture tasks, which can be canceled;
wenzelm
parents: 56711
diff changeset
   133
      future_rendering =
e723f041b6d0 tuned signature -- separate pool for JFuture tasks, which can be canceled;
wenzelm
parents: 56711
diff changeset
   134
        Some(Simple_Thread.submit_task {
51498
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   135
          val (text, rendering) =
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   136
            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
   137
            catch { case exn: Throwable => Log.log(Log.ERROR, this, exn); throw exn }
56860
dc71c3d0e909 tuned signature;
wenzelm
parents: 56789
diff changeset
   138
          Exn.Interrupt.expose()
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   139
51498
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   140
          Swing_Thread.later {
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   141
            current_rendering = rendering
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   142
            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
   143
              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
   144
              getBuffer.setReadOnly(false)
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   145
              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
   146
              setText(text)
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   147
              setCaretPosition(0)
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   148
              getBuffer.setReadOnly(true)
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   149
            }
979592b765f8 reverted most of 5944b20c41bf -- tends to cause race condition of synchronous vs. asynchronous version;
wenzelm
parents: 51497
diff changeset
   150
          }
56730
e723f041b6d0 tuned signature -- separate pool for JFuture tasks, which can be canceled;
wenzelm
parents: 56711
diff changeset
   151
        })
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
   152
    }
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   153
  }
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   154
55825
694833e3e4a0 tuned signature -- separate module Font_Info;
wenzelm
parents: 53277
diff changeset
   155
  def resize(font_info: Font_Info)
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   156
  {
56662
f373fb77e0a4 avoid "Adaptation of argument list by inserting ()" -- deprecated in scala-2.11.0;
wenzelm
parents: 56360
diff changeset
   157
    Swing_Thread.require {}
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   158
55825
694833e3e4a0 tuned signature -- separate module Font_Info;
wenzelm
parents: 53277
diff changeset
   159
    current_font_info = font_info
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   160
    refresh()
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   161
  }
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   162
50501
6f41f1646617 more careful handling of Dialog_Result, with active area and color feedback;
wenzelm
parents: 50306
diff changeset
   163
  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
   164
  {
56662
f373fb77e0a4 avoid "Adaptation of argument list by inserting ()" -- deprecated in scala-2.11.0;
wenzelm
parents: 56360
diff changeset
   165
    Swing_Thread.require {}
49419
e2726211f834 pass base_snapshot to enable hyperlinks into other nodes;
wenzelm
parents: 49416
diff changeset
   166
    require(!base_snapshot.is_outdated)
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   167
49419
e2726211f834 pass base_snapshot to enable hyperlinks into other nodes;
wenzelm
parents: 49416
diff changeset
   168
    current_base_snapshot = base_snapshot
50501
6f41f1646617 more careful handling of Dialog_Result, with active area and color feedback;
wenzelm
parents: 50306
diff changeset
   169
    current_base_results = base_results
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   170
    current_body = body
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   171
    refresh()
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   172
  }
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   173
56880
f8c1d2583699 tuned signature;
wenzelm
parents: 56860
diff changeset
   174
  def detach
f8c1d2583699 tuned signature;
wenzelm
parents: 56860
diff changeset
   175
  {
f8c1d2583699 tuned signature;
wenzelm
parents: 56860
diff changeset
   176
    Swing_Thread.require {}
f8c1d2583699 tuned signature;
wenzelm
parents: 56860
diff changeset
   177
    Info_Dockable(view, current_base_snapshot, current_base_results, current_body)
f8c1d2583699 tuned signature;
wenzelm
parents: 56860
diff changeset
   178
  }
f8c1d2583699 tuned signature;
wenzelm
parents: 56860
diff changeset
   179
56918
a442dc6d244d clarified detach_operation: ignore empty output;
wenzelm
parents: 56907
diff changeset
   180
  def detach_operation: Option[() => Unit] =
a442dc6d244d clarified detach_operation: ignore empty output;
wenzelm
parents: 56907
diff changeset
   181
    if (current_body.isEmpty) None else Some(() => detach)
a442dc6d244d clarified detach_operation: ignore empty output;
wenzelm
parents: 56907
diff changeset
   182
49422
21f77309d93a minimal clipboard support (similar to org.lobobrowser.html.gui.HtmlBlockPanel);
wenzelm
parents: 49421
diff changeset
   183
56881
15e18540df10 more uniform detach button;
wenzelm
parents: 56880
diff changeset
   184
  /* common GUI components */
15e18540df10 more uniform detach button;
wenzelm
parents: 56880
diff changeset
   185
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   186
  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
   187
    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
   188
  }
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   189
57042
5576d22abf3c tuned signature;
wenzelm
parents: 56918
diff changeset
   190
  val search_field: Component =
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   191
    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
   192
      {
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   193
        private val input_delay =
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   194
          Swing_Thread.delay_last(PIDE.options.seconds("editor_input_delay")) {
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   195
            search_action(this)
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   196
          }
56885
3020f6bbd119 clarified GUI events, e.g. relevant for insert via completion;
wenzelm
parents: 56883
diff changeset
   197
        getDocument.addDocumentListener(new DocumentListener {
3020f6bbd119 clarified GUI events, e.g. relevant for insert via completion;
wenzelm
parents: 56883
diff changeset
   198
          def changedUpdate(e: DocumentEvent) { input_delay.invoke() }
3020f6bbd119 clarified GUI events, e.g. relevant for insert via completion;
wenzelm
parents: 56883
diff changeset
   199
          def insertUpdate(e: DocumentEvent) { input_delay.invoke() }
3020f6bbd119 clarified GUI events, e.g. relevant for insert via completion;
wenzelm
parents: 56883
diff changeset
   200
          def removeUpdate(e: DocumentEvent) { input_delay.invoke() }
3020f6bbd119 clarified GUI events, e.g. relevant for insert via completion;
wenzelm
parents: 56883
diff changeset
   201
        })
56883
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   202
        setColumns(20)
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   203
        setToolTipText(search_label.tooltip)
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   204
        setFont(GUI.imitate_font(Font_Info.main_family(), getFont, 1.2))
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   205
      })
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   206
57042
5576d22abf3c tuned signature;
wenzelm
parents: 56918
diff changeset
   207
  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
   208
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   209
  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
   210
  {
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   211
    val (pattern, ok) =
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   212
      text_field.getText match {
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   213
        case null | "" => (None, true)
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   214
        case s =>
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   215
          try { (Some(new Regex(s)), true) }
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   216
          catch { case ERROR(_) => (None, false) }
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   217
      }
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   218
    if (current_search_pattern != pattern) {
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   219
      current_search_pattern = pattern
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   220
      text_area.getPainter.repaint()
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   221
    }
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   222
    text_field.setForeground(
57042
5576d22abf3c tuned signature;
wenzelm
parents: 56918
diff changeset
   223
      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
   224
  }
38c6b70e5e53 common support for search field, which is actually a light-weight Highlighter;
wenzelm
parents: 56881
diff changeset
   225
56881
15e18540df10 more uniform detach button;
wenzelm
parents: 56880
diff changeset
   226
50726
27478c11f63c more elementary key handling: listen to low-level KEY_PRESSED events (without consuming);
wenzelm
parents: 50640
diff changeset
   227
  /* key handling */
49422
21f77309d93a minimal clipboard support (similar to org.lobobrowser.html.gui.HtmlBlockPanel);
wenzelm
parents: 49421
diff changeset
   228
53226
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   229
  addKeyListener(JEdit_Lib.key_listener(
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   230
    key_pressed = (evt: KeyEvent) =>
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   231
      {
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   232
        evt.getKeyCode match {
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   233
          case KeyEvent.VK_C
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
   234
          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
   235
              text_area.getSelectionCount != 0 =>
53226
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   236
            Registers.copy(text_area, '$')
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   237
            evt.consume
53227
68cc55ceb7f6 more standard key handling according to jEdit (with workaround);
wenzelm
parents: 53226
diff changeset
   238
53226
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   239
          case KeyEvent.VK_A
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   240
          if (evt.getModifiers & Toolkit.getDefaultToolkit.getMenuShortcutKeyMask) != 0 =>
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   241
            text_area.selectAll
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   242
            evt.consume
53227
68cc55ceb7f6 more standard key handling according to jEdit (with workaround);
wenzelm
parents: 53226
diff changeset
   243
68cc55ceb7f6 more standard key handling according to jEdit (with workaround);
wenzelm
parents: 53226
diff changeset
   244
          case KeyEvent.VK_ESCAPE =>
53244
ec6011bf2362 dismiss popups more uniformly;
wenzelm
parents: 53231
diff changeset
   245
            if (PIDE.dismissed_popups(view)) evt.consume
53227
68cc55ceb7f6 more standard key handling according to jEdit (with workaround);
wenzelm
parents: 53226
diff changeset
   246
53226
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   247
          case _ =>
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   248
        }
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
   249
        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
   250
      },
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   251
    key_typed = (evt: KeyEvent) =>
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   252
      {
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
   253
        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
   254
      }
53226
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   255
    )
9cf8e2263ca7 more systematic JEdit_Lib.key_listener with optional KeyEventWorkaround;
wenzelm
parents: 53179
diff changeset
   256
  )
49422
21f77309d93a minimal clipboard support (similar to org.lobobrowser.html.gui.HtmlBlockPanel);
wenzelm
parents: 49421
diff changeset
   257
21f77309d93a minimal clipboard support (similar to org.lobobrowser.html.gui.HtmlBlockPanel);
wenzelm
parents: 49421
diff changeset
   258
21f77309d93a minimal clipboard support (similar to org.lobobrowser.html.gui.HtmlBlockPanel);
wenzelm
parents: 49421
diff changeset
   259
  /* init */
21f77309d93a minimal clipboard support (similar to org.lobobrowser.html.gui.HtmlBlockPanel);
wenzelm
parents: 49421
diff changeset
   260
49472
ba2c0d0cd429 no caret painting;
wenzelm
parents: 49471
diff changeset
   261
  getPainter.setStructureHighlightEnabled(false)
49475
8f3a3adadd5a tuned painter;
wenzelm
parents: 49472
diff changeset
   262
  getPainter.setLineHighlightEnabled(false)
8f3a3adadd5a tuned painter;
wenzelm
parents: 49472
diff changeset
   263
53277
6aa348237973 more uniform configuration of editor modes and token markers;
wenzelm
parents: 53244
diff changeset
   264
  getBuffer.setTokenMarker(Isabelle.token_marker("isabelle-output").get)
49446
b8d8f738bf63 more direct GUI component;
wenzelm
parents: 49422
diff changeset
   265
  getBuffer.setReadOnly(true)
53002
9dd1a6dcebfd imitate "noWordSep" of isabelle mode, e.g. relevant for word selection via double-click;
wenzelm
parents: 53001
diff changeset
   266
  getBuffer.setStringProperty("noWordSep", "_'.?")
49475
8f3a3adadd5a tuned painter;
wenzelm
parents: 49472
diff changeset
   267
49412
4cac648e0f85 Pretty_Text_Area is based on Rich_Text_Area;
wenzelm
parents: 49398
diff changeset
   268
  rich_text_area.activate()
49398
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   269
}
0fa4389c04f9 alternative output panel, based on Pretty_Text_Area, based on JEditEmbeddedTextArea;
wenzelm
parents:
diff changeset
   270