src/Tools/jEdit/src/jedit/document_view.scala
author wenzelm
Tue, 07 Sep 2010 14:08:21 +0200
changeset 39168 e3ac771235f7
parent 39132 ba17ca3acdd3
child 39169 18cdf2833371
permissions -rw-r--r--
report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase; tuned color;
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
36760
b82a698ef6c9 tuned headers;
wenzelm
parents: 36015
diff changeset
     1
/*  Title:      Tools/jEdit/src/jedit/document_view.scala
b82a698ef6c9 tuned headers;
wenzelm
parents: 36015
diff changeset
     2
    Author:     Fabian Immler, TU Munich
b82a698ef6c9 tuned headers;
wenzelm
parents: 36015
diff changeset
     3
    Author:     Makarius
b82a698ef6c9 tuned headers;
wenzelm
parents: 36015
diff changeset
     4
b82a698ef6c9 tuned headers;
wenzelm
parents: 36015
diff changeset
     5
Document view connected to jEdit text area.
b82a698ef6c9 tuned headers;
wenzelm
parents: 36015
diff changeset
     6
*/
34408
ad7b6c4813c8 added some headers and comments;
wenzelm
parents: 34406
diff changeset
     7
34403
6c812a3cb170 information on command-phase left of scrollbar (with panel)
immler@in.tum.de
parents:
diff changeset
     8
package isabelle.jedit
6c812a3cb170 information on command-phase left of scrollbar (with panel)
immler@in.tum.de
parents:
diff changeset
     9
34760
dc7f5e0d9d27 misc modernization of names;
wenzelm
parents: 34759
diff changeset
    10
36015
6111de7c916a adapted to Scala 2.8.0 Beta 1;
wenzelm
parents: 34871
diff changeset
    11
import isabelle._
6111de7c916a adapted to Scala 2.8.0 Beta 1;
wenzelm
parents: 34871
diff changeset
    12
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    13
import scala.actors.Actor._
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    14
39131
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
    15
import java.awt.event.{MouseAdapter, MouseMotionAdapter, MouseEvent, FocusAdapter, FocusEvent}
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    16
import java.awt.{BorderLayout, Graphics, Dimension, Color, Graphics2D}
34734
wenzelm
parents: 34733
diff changeset
    17
import javax.swing.{JPanel, ToolTipManager}
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    18
import javax.swing.event.{CaretListener, CaretEvent}
34734
wenzelm
parents: 34733
diff changeset
    19
39131
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
    20
import org.gjt.sp.jedit.OperatingSystem
34709
2f0c18f9b6c7 minor tuning;
wenzelm
parents: 34679
diff changeset
    21
import org.gjt.sp.jedit.gui.RolloverButton
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    22
import org.gjt.sp.jedit.textarea.{JEditTextArea, TextArea, TextAreaExtension, TextAreaPainter}
37241
04d2521e79b0 basic support for sub/superscript token markup -- NB: need to maintain extended token types eagerly, since jEdit occasionally reinstalls a style array that is too short;
wenzelm
parents: 37201
diff changeset
    23
import org.gjt.sp.jedit.syntax.SyntaxStyle
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    24
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    25
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    26
object Document_View
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    27
{
39044
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
    28
  /* physical rendering */
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
    29
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
    30
  def status_color(snapshot: Document.Snapshot, command: Command): Color =
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    31
  {
38356
443fb83a21e8 consider command state as part of Snapshot, not Document;
wenzelm
parents: 38227
diff changeset
    32
    val state = snapshot.state(command)
38151
2837c952ca31 explicit Change.Snapshot and Document.Node;
wenzelm
parents: 38150
diff changeset
    33
    if (snapshot.is_outdated) new Color(240, 240, 240)
37186
349e9223c685 explicit markup for forked goals, as indicated by Goal.fork;
wenzelm
parents: 37132
diff changeset
    34
    else
38567
b670faa807c9 concentrate protocol message formats in Isar_Document;
wenzelm
parents: 38429
diff changeset
    35
      Isar_Document.command_status(state.status) match {
b670faa807c9 concentrate protocol message formats in Isar_Document;
wenzelm
parents: 38429
diff changeset
    36
        case Isar_Document.Forked(i) if i > 0 => new Color(255, 228, 225)
b670faa807c9 concentrate protocol message formats in Isar_Document;
wenzelm
parents: 38429
diff changeset
    37
        case Isar_Document.Finished => new Color(234, 248, 255)
b670faa807c9 concentrate protocol message formats in Isar_Document;
wenzelm
parents: 38429
diff changeset
    38
        case Isar_Document.Failed => new Color(255, 193, 193)
b670faa807c9 concentrate protocol message formats in Isar_Document;
wenzelm
parents: 38429
diff changeset
    39
        case Isar_Document.Unprocessed => new Color(255, 228, 225)
38429
9951852fae91 simplified command status: interpret stacked markup on demand;
wenzelm
parents: 38426
diff changeset
    40
        case _ => Color.red
37186
349e9223c685 explicit markup for forked goals, as indicated by Goal.fork;
wenzelm
parents: 37132
diff changeset
    41
      }
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    42
  }
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    43
39044
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
    44
  val message_markup: PartialFunction[Text.Info[Any], Color] =
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
    45
  {
39168
e3ac771235f7 report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase;
wenzelm
parents: 39132
diff changeset
    46
    case Text.Info(_, XML.Elem(Markup(Markup.WRITELN, _), _)) => new Color(192, 192, 192)
39044
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
    47
    case Text.Info(_, XML.Elem(Markup(Markup.WARNING, _), _)) => new Color(255, 165, 0)
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
    48
    case Text.Info(_, XML.Elem(Markup(Markup.ERROR, _), _)) => new Color(255, 106, 106)
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
    49
  }
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
    50
39168
e3ac771235f7 report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase;
wenzelm
parents: 39132
diff changeset
    51
  val box_markup: PartialFunction[Text.Info[Any], Color] =
e3ac771235f7 report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase;
wenzelm
parents: 39132
diff changeset
    52
  {
e3ac771235f7 report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase;
wenzelm
parents: 39132
diff changeset
    53
    case Text.Info(_, XML.Elem(Markup(Markup.TOKEN_RANGE, _), _)) => new Color(192, 192, 192)
e3ac771235f7 report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase;
wenzelm
parents: 39132
diff changeset
    54
  }
e3ac771235f7 report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase;
wenzelm
parents: 39132
diff changeset
    55
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    56
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    57
  /* document view of text area */
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    58
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    59
  private val key = new Object
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    60
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    61
  def init(model: Document_Model, text_area: TextArea): Document_View =
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    62
  {
38223
2a368e8e0a80 more explicit treatment of Swing thread context;
wenzelm
parents: 38158
diff changeset
    63
    Swing_Thread.require()
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    64
    val doc_view = new Document_View(model, text_area)
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    65
    text_area.putClientProperty(key, doc_view)
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    66
    doc_view.activate()
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    67
    doc_view
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    68
  }
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    69
34788
3779c54a2d21 direct apply for Document_Model and Document_View;
wenzelm
parents: 34784
diff changeset
    70
  def apply(text_area: TextArea): Option[Document_View] =
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    71
  {
38223
2a368e8e0a80 more explicit treatment of Swing thread context;
wenzelm
parents: 38158
diff changeset
    72
    Swing_Thread.require()
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    73
    text_area.getClientProperty(key) match {
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    74
      case doc_view: Document_View => Some(doc_view)
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    75
      case _ => None
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    76
    }
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    77
  }
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    78
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    79
  def exit(text_area: TextArea)
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    80
  {
38223
2a368e8e0a80 more explicit treatment of Swing thread context;
wenzelm
parents: 38158
diff changeset
    81
    Swing_Thread.require()
34788
3779c54a2d21 direct apply for Document_Model and Document_View;
wenzelm
parents: 34784
diff changeset
    82
    apply(text_area) match {
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    83
      case None => error("No document view for text area: " + text_area)
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    84
      case Some(doc_view) =>
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    85
        doc_view.deactivate()
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    86
        text_area.putClientProperty(key, null)
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    87
    }
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    88
  }
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    89
}
34403
6c812a3cb170 information on command-phase left of scrollbar (with panel)
immler@in.tum.de
parents:
diff changeset
    90
34733
a3ad6d51db1d misc tuning and unification;
wenzelm
parents: 34711
diff changeset
    91
37129
4c83696b340e Command.toString: include id for debugging;
wenzelm
parents: 36990
diff changeset
    92
class Document_View(val model: Document_Model, text_area: TextArea)
34654
30f588245884 arbitrary history
immler@in.tum.de
parents: 34653
diff changeset
    93
{
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    94
  private val session = model.session
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
    95
34403
6c812a3cb170 information on command-phase left of scrollbar (with panel)
immler@in.tum.de
parents:
diff changeset
    96
37241
04d2521e79b0 basic support for sub/superscript token markup -- NB: need to maintain extended token types eagerly, since jEdit occasionally reinstalls a style array that is too short;
wenzelm
parents: 37201
diff changeset
    97
  /* extended token styles */
04d2521e79b0 basic support for sub/superscript token markup -- NB: need to maintain extended token types eagerly, since jEdit occasionally reinstalls a style array that is too short;
wenzelm
parents: 37201
diff changeset
    98
04d2521e79b0 basic support for sub/superscript token markup -- NB: need to maintain extended token types eagerly, since jEdit occasionally reinstalls a style array that is too short;
wenzelm
parents: 37201
diff changeset
    99
  private var styles: Array[SyntaxStyle] = null  // owned by Swing thread
04d2521e79b0 basic support for sub/superscript token markup -- NB: need to maintain extended token types eagerly, since jEdit occasionally reinstalls a style array that is too short;
wenzelm
parents: 37201
diff changeset
   100
04d2521e79b0 basic support for sub/superscript token markup -- NB: need to maintain extended token types eagerly, since jEdit occasionally reinstalls a style array that is too short;
wenzelm
parents: 37201
diff changeset
   101
  def extend_styles()
04d2521e79b0 basic support for sub/superscript token markup -- NB: need to maintain extended token types eagerly, since jEdit occasionally reinstalls a style array that is too short;
wenzelm
parents: 37201
diff changeset
   102
  {
38223
2a368e8e0a80 more explicit treatment of Swing thread context;
wenzelm
parents: 38158
diff changeset
   103
    Swing_Thread.require()
38158
8aaa21db41f3 Document_Model: include token marker here;
wenzelm
parents: 38153
diff changeset
   104
    styles = Document_Model.Token_Markup.extend_styles(text_area.getPainter.getStyles)
37241
04d2521e79b0 basic support for sub/superscript token markup -- NB: need to maintain extended token types eagerly, since jEdit occasionally reinstalls a style array that is too short;
wenzelm
parents: 37201
diff changeset
   105
  }
04d2521e79b0 basic support for sub/superscript token markup -- NB: need to maintain extended token types eagerly, since jEdit occasionally reinstalls a style array that is too short;
wenzelm
parents: 37201
diff changeset
   106
  extend_styles()
04d2521e79b0 basic support for sub/superscript token markup -- NB: need to maintain extended token types eagerly, since jEdit occasionally reinstalls a style array that is too short;
wenzelm
parents: 37201
diff changeset
   107
04d2521e79b0 basic support for sub/superscript token markup -- NB: need to maintain extended token types eagerly, since jEdit occasionally reinstalls a style array that is too short;
wenzelm
parents: 37201
diff changeset
   108
  def set_styles()
04d2521e79b0 basic support for sub/superscript token markup -- NB: need to maintain extended token types eagerly, since jEdit occasionally reinstalls a style array that is too short;
wenzelm
parents: 37201
diff changeset
   109
  {
38223
2a368e8e0a80 more explicit treatment of Swing thread context;
wenzelm
parents: 38158
diff changeset
   110
    Swing_Thread.require()
37241
04d2521e79b0 basic support for sub/superscript token markup -- NB: need to maintain extended token types eagerly, since jEdit occasionally reinstalls a style array that is too short;
wenzelm
parents: 37201
diff changeset
   111
    text_area.getPainter.setStyles(styles)
04d2521e79b0 basic support for sub/superscript token markup -- NB: need to maintain extended token types eagerly, since jEdit occasionally reinstalls a style array that is too short;
wenzelm
parents: 37201
diff changeset
   112
  }
04d2521e79b0 basic support for sub/superscript token markup -- NB: need to maintain extended token types eagerly, since jEdit occasionally reinstalls a style array that is too short;
wenzelm
parents: 37201
diff changeset
   113
04d2521e79b0 basic support for sub/superscript token markup -- NB: need to maintain extended token types eagerly, since jEdit occasionally reinstalls a style array that is too short;
wenzelm
parents: 37201
diff changeset
   114
38881
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   115
  /* visible line ranges */
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   116
38883
0998a635684a refined proper_line_range (again), to make text_area_extension work with soft wwrap;
wenzelm
parents: 38881
diff changeset
   117
  // simplify slightly odd result of TextArea.getScreenLineEndOffset etc.
38881
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   118
  // NB: jEdit already normalizes \r\n and \r to \n
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   119
  def proper_line_range(start: Text.Offset, end: Text.Offset): Text.Range =
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   120
  {
38883
0998a635684a refined proper_line_range (again), to make text_area_extension work with soft wwrap;
wenzelm
parents: 38881
diff changeset
   121
    val stop = if (start < end) end - 1 else end min model.buffer.getLength
38881
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   122
    Text.Range(start, stop)
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   123
  }
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   124
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   125
  def screen_lines_range(): Text.Range =
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   126
  {
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   127
    val start = text_area.getScreenLineStartOffset(0)
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   128
    val raw_end = text_area.getScreenLineEndOffset(text_area.getVisibleLines - 1 max 0)
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   129
    proper_line_range(start, if (raw_end >= 0) raw_end else model.buffer.getLength)
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   130
  }
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   131
39132
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   132
  def invalidate_line_range(range: Text.Range)
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   133
  {
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   134
    text_area.invalidateLineRange(
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   135
      model.buffer.getLineOfOffset(range.start),
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   136
      model.buffer.getLineOfOffset(range.stop))
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   137
  }
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   138
38881
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   139
37129
4c83696b340e Command.toString: include id for debugging;
wenzelm
parents: 36990
diff changeset
   140
  /* commands_changed_actor */
34834
df9af932e418 slightly more uniform/robust handling of visible document;
wenzelm
parents: 34832
diff changeset
   141
37129
4c83696b340e Command.toString: include id for debugging;
wenzelm
parents: 36990
diff changeset
   142
  private val commands_changed_actor = actor {
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   143
    loop {
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   144
      react {
38360
53224a4d2f0e specific Session.Commands_Changed;
wenzelm
parents: 38356
diff changeset
   145
        case Session.Commands_Changed(changed) =>
38843
d95522496593 more careful locking of jEdit buffer;
wenzelm
parents: 38710
diff changeset
   146
          val buffer = model.buffer
d95522496593 more careful locking of jEdit buffer;
wenzelm
parents: 38710
diff changeset
   147
          Isabelle.swing_buffer_lock(buffer) {
38151
2837c952ca31 explicit Change.Snapshot and Document.Node;
wenzelm
parents: 38150
diff changeset
   148
            val snapshot = model.snapshot()
38881
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   149
38884
9ec5f6010d6e Document_View: repaint overview for any command change of this node (again);
wenzelm
parents: 38883
diff changeset
   150
            if (changed.exists(snapshot.node.commands.contains))
9ec5f6010d6e Document_View: repaint overview for any command change of this node (again);
wenzelm
parents: 38883
diff changeset
   151
              overview.repaint()
9ec5f6010d6e Document_View: repaint overview for any command change of this node (again);
wenzelm
parents: 38883
diff changeset
   152
38881
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   153
            val visible_range = screen_lines_range()
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   154
            val visible_cmds = snapshot.node.command_range(snapshot.revert(visible_range)).map(_._1)
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   155
            if (visible_cmds.exists(changed)) {
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   156
              for {
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   157
                line <- 0 until text_area.getVisibleLines
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   158
                val start = text_area.getScreenLineStartOffset(line) if start >= 0
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   159
                val end = text_area.getScreenLineEndOffset(line) if end >= 0
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   160
                val range = proper_line_range(start, end)
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   161
                val line_cmds = snapshot.node.command_range(snapshot.revert(range)).map(_._1)
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   162
                if line_cmds.exists(changed)
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   163
              } text_area.invalidateScreenLineRange(line, line)
38884
9ec5f6010d6e Document_View: repaint overview for any command change of this node (again);
wenzelm
parents: 38883
diff changeset
   164
38843
d95522496593 more careful locking of jEdit buffer;
wenzelm
parents: 38710
diff changeset
   165
              // FIXME danger of deadlock!?
38881
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   166
              // FIXME potentially slow!?
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   167
              model.buffer.propertiesChanged()
38640
105d1f112da5 sporadic locking of jEdit buffer;
wenzelm
parents: 38582
diff changeset
   168
            }
34834
df9af932e418 slightly more uniform/robust handling of visible document;
wenzelm
parents: 34832
diff changeset
   169
          }
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   170
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   171
        case bad => System.err.println("command_change_actor: ignoring bad message " + bad)
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   172
      }
34403
6c812a3cb170 information on command-phase left of scrollbar (with panel)
immler@in.tum.de
parents:
diff changeset
   173
    }
34678
acaac03ced00 tuned whitespace
immler@in.tum.de
parents: 34654
diff changeset
   174
  }
34403
6c812a3cb170 information on command-phase left of scrollbar (with panel)
immler@in.tum.de
parents:
diff changeset
   175
6c812a3cb170 information on command-phase left of scrollbar (with panel)
immler@in.tum.de
parents:
diff changeset
   176
39131
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   177
  /* subexpression highlighting */
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   178
39132
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   179
  private def subexp_range(snapshot: Document.Snapshot, x: Int, y: Int): Option[Text.Range] =
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   180
  {
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   181
    val subexp_markup: PartialFunction[Text.Info[Any], Option[Text.Range]] =
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   182
    {
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   183
      case Text.Info(range, XML.Elem(Markup(Markup.ML_TYPING, _), _)) =>
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   184
        Some(snapshot.convert(range))
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   185
    }
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   186
    val offset = text_area.xyToOffset(x, y)
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   187
    val markup = snapshot.select_markup(Text.Range(offset, offset + 1))(subexp_markup)(None)
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   188
    if (markup.hasNext) markup.next.info else None
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   189
  }
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   190
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   191
  private var highlight_range: Option[Text.Range] = None
39131
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   192
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   193
  private val focus_listener = new FocusAdapter {
39132
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   194
    override def focusLost(e: FocusEvent) { highlight_range = None }
39131
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   195
  }
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   196
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   197
  private val mouse_motion_listener = new MouseMotionAdapter {
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   198
    override def mouseMoved(e: MouseEvent) {
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   199
      val control = if (OperatingSystem.isMacOS()) e.isMetaDown else e.isControlDown
39132
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   200
      if (!model.buffer.isLoaded) highlight_range = None
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   201
      else
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   202
        Isabelle.swing_buffer_lock(model.buffer) {
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   203
          highlight_range.map(invalidate_line_range(_))
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   204
          highlight_range =
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   205
            if (control) subexp_range(model.snapshot(), e.getX(), e.getY()) else None
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   206
          highlight_range.map(invalidate_line_range(_))
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   207
        }
39131
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   208
    }
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   209
  }
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   210
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   211
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   212
  /* text_area_extension */
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   213
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   214
  private val text_area_extension = new TextAreaExtension
34678
acaac03ced00 tuned whitespace
immler@in.tum.de
parents: 34654
diff changeset
   215
  {
37685
305c326db33b more efficient document model/view -- avoid repeated iteration over commands from start, prefer bulk operations;
wenzelm
parents: 37555
diff changeset
   216
    override def paintScreenLineRange(gfx: Graphics2D,
305c326db33b more efficient document model/view -- avoid repeated iteration over commands from start, prefer bulk operations;
wenzelm
parents: 37555
diff changeset
   217
      first_line: Int, last_line: Int, physical_lines: Array[Int],
38886
wenzelm
parents: 38884
diff changeset
   218
      start: Array[Int], end: Array[Int], y: Int, line_height: Int)
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   219
    {
38843
d95522496593 more careful locking of jEdit buffer;
wenzelm
parents: 38710
diff changeset
   220
      Isabelle.swing_buffer_lock(model.buffer) {
d95522496593 more careful locking of jEdit buffer;
wenzelm
parents: 38710
diff changeset
   221
        val snapshot = model.snapshot()
d95522496593 more careful locking of jEdit buffer;
wenzelm
parents: 38710
diff changeset
   222
        val saved_color = gfx.getColor
39044
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   223
        val ascent = text_area.getPainter.getFontMetrics.getAscent
39131
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   224
38843
d95522496593 more careful locking of jEdit buffer;
wenzelm
parents: 38710
diff changeset
   225
        try {
d95522496593 more careful locking of jEdit buffer;
wenzelm
parents: 38710
diff changeset
   226
          for (i <- 0 until physical_lines.length) {
d95522496593 more careful locking of jEdit buffer;
wenzelm
parents: 38710
diff changeset
   227
            if (physical_lines(i) != -1) {
38881
c8123e77acc5 tuned commands_changed_actor: more precise/efficient handling of visible screen lines;
wenzelm
parents: 38880
diff changeset
   228
              val line_range = proper_line_range(start(i), end(i))
39044
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   229
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   230
              // background color
38880
5b4efe90c120 simplified/clarified Document_View.text_area_extension;
wenzelm
parents: 38855
diff changeset
   231
              val cmds = snapshot.node.command_range(snapshot.revert(line_range))
39044
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   232
              for {
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   233
                (command, command_start) <- cmds if !command.is_ignored
38880
5b4efe90c120 simplified/clarified Document_View.text_area_extension;
wenzelm
parents: 38855
diff changeset
   234
                val range = line_range.restrict(snapshot.convert(command.range + command_start))
39044
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   235
                r <- Isabelle.gfx_range(text_area, range)
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   236
              } {
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   237
                gfx.setColor(Document_View.status_color(snapshot, command))
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   238
                gfx.fillRect(r.x, y + i * line_height, r.length, line_height)
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   239
              }
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   240
39132
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   241
              // subexpression highlighting -- potentially from other snapshot
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   242
              if (highlight_range.isDefined) {
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   243
                if (line_range.overlaps(highlight_range.get)) {
ba17ca3acdd3 refined treatment of multi-line subexpressions;
wenzelm
parents: 39131
diff changeset
   244
                  Isabelle.gfx_range(text_area, line_range.restrict(highlight_range.get)) match {
39131
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   245
                    case None =>
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   246
                    case Some(r) =>
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   247
                      gfx.setColor(Color.black)
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   248
                      gfx.drawRect(r.x, y + i * line_height, r.length, line_height - 1)
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   249
                  }
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   250
                }
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   251
              }
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   252
39168
e3ac771235f7 report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase;
wenzelm
parents: 39132
diff changeset
   253
              // boxed text
e3ac771235f7 report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase;
wenzelm
parents: 39132
diff changeset
   254
              for {
e3ac771235f7 report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase;
wenzelm
parents: 39132
diff changeset
   255
                Text.Info(range, color) <-
e3ac771235f7 report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase;
wenzelm
parents: 39132
diff changeset
   256
                  snapshot.select_markup(line_range)(Document_View.box_markup)(null)
e3ac771235f7 report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase;
wenzelm
parents: 39132
diff changeset
   257
                if color != null
e3ac771235f7 report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase;
wenzelm
parents: 39132
diff changeset
   258
                r <- Isabelle.gfx_range(text_area, range)
e3ac771235f7 report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase;
wenzelm
parents: 39132
diff changeset
   259
              } {
e3ac771235f7 report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase;
wenzelm
parents: 39132
diff changeset
   260
                gfx.setColor(color)
e3ac771235f7 report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase;
wenzelm
parents: 39132
diff changeset
   261
                gfx.drawRect(r.x + 1, y + i * line_height + 1, r.length - 2, line_height - 3)
e3ac771235f7 report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase;
wenzelm
parents: 39132
diff changeset
   262
              }
e3ac771235f7 report token range after inner parse error -- often provides important clues about misunderstanding concerning lexical phase;
wenzelm
parents: 39132
diff changeset
   263
39044
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   264
              // squiggly underline
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   265
              for {
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   266
                Text.Info(range, color) <-
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   267
                  snapshot.select_markup(line_range)(Document_View.message_markup)(null)
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   268
                if color != null
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   269
                r <- Isabelle.gfx_range(text_area, range)
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   270
              } {
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   271
                gfx.setColor(color)
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   272
                val x0 = (r.x / 2) * 2
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   273
                val y0 = r.y + ascent + 1
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   274
                for (x1 <- Range(x0, x0 + r.length, 2)) {
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   275
                  val y1 = if (x1 % 4 < 2) y0 else y0 + 1
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   276
                  gfx.drawLine(x1, y1, x1 + 1, y1)
38883
0998a635684a refined proper_line_range (again), to make text_area_extension work with soft wwrap;
wenzelm
parents: 38881
diff changeset
   277
                }
38843
d95522496593 more careful locking of jEdit buffer;
wenzelm
parents: 38710
diff changeset
   278
              }
d95522496593 more careful locking of jEdit buffer;
wenzelm
parents: 38710
diff changeset
   279
            }
38223
2a368e8e0a80 more explicit treatment of Swing thread context;
wenzelm
parents: 38158
diff changeset
   280
          }
2a368e8e0a80 more explicit treatment of Swing thread context;
wenzelm
parents: 38158
diff changeset
   281
        }
38843
d95522496593 more careful locking of jEdit buffer;
wenzelm
parents: 38710
diff changeset
   282
        finally { gfx.setColor(saved_color) }
38223
2a368e8e0a80 more explicit treatment of Swing thread context;
wenzelm
parents: 38158
diff changeset
   283
      }
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   284
    }
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   285
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   286
    override def getToolTipText(x: Int, y: Int): String =
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   287
    {
38845
a9e37daf5bd0 added Document.Snapshot.select_markup, which includes command iteration, range conversion etc.;
wenzelm
parents: 38843
diff changeset
   288
      Isabelle.swing_buffer_lock(model.buffer) {
a9e37daf5bd0 added Document.Snapshot.select_markup, which includes command iteration, range conversion etc.;
wenzelm
parents: 38843
diff changeset
   289
        val snapshot = model.snapshot()
a9e37daf5bd0 added Document.Snapshot.select_markup, which includes command iteration, range conversion etc.;
wenzelm
parents: 38843
diff changeset
   290
        val offset = text_area.xyToOffset(x, y)
a9e37daf5bd0 added Document.Snapshot.select_markup, which includes command iteration, range conversion etc.;
wenzelm
parents: 38843
diff changeset
   291
        val markup =
a9e37daf5bd0 added Document.Snapshot.select_markup, which includes command iteration, range conversion etc.;
wenzelm
parents: 38843
diff changeset
   292
          snapshot.select_markup(Text.Range(offset, offset + 1)) {
38580
881c362d48e4 proper range for hyperlinks and tooltips, using original markup information;
wenzelm
parents: 38577
diff changeset
   293
            case Text.Info(range, XML.Elem(Markup(Markup.ML_TYPING, _), body)) =>
38845
a9e37daf5bd0 added Document.Snapshot.select_markup, which includes command iteration, range conversion etc.;
wenzelm
parents: 38843
diff changeset
   294
              Isabelle.tooltip(Pretty.string_of(List(Pretty.block(body)), margin = 40))
a9e37daf5bd0 added Document.Snapshot.select_markup, which includes command iteration, range conversion etc.;
wenzelm
parents: 38843
diff changeset
   295
          } { null }
a9e37daf5bd0 added Document.Snapshot.select_markup, which includes command iteration, range conversion etc.;
wenzelm
parents: 38843
diff changeset
   296
        if (markup.hasNext) markup.next.info else null
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   297
      }
34734
wenzelm
parents: 34733
diff changeset
   298
    }
34678
acaac03ced00 tuned whitespace
immler@in.tum.de
parents: 34654
diff changeset
   299
  }
34513
411017e76e98 respect current offsets
immler@in.tum.de
parents: 34503
diff changeset
   300
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   301
37129
4c83696b340e Command.toString: include id for debugging;
wenzelm
parents: 36990
diff changeset
   302
  /* caret handling */
34810
9ad3431a34a5 tuned caret_listener/selected_command;
wenzelm
parents: 34808
diff changeset
   303
37849
4f9de312cc23 Session: predefined real time parameters;
wenzelm
parents: 37685
diff changeset
   304
  def selected_command(): Option[Command] =
38223
2a368e8e0a80 more explicit treatment of Swing thread context;
wenzelm
parents: 38158
diff changeset
   305
  {
2a368e8e0a80 more explicit treatment of Swing thread context;
wenzelm
parents: 38158
diff changeset
   306
    Swing_Thread.require()
38151
2837c952ca31 explicit Change.Snapshot and Document.Node;
wenzelm
parents: 38150
diff changeset
   307
    model.snapshot().node.proper_command_at(text_area.getCaretPosition)
38223
2a368e8e0a80 more explicit treatment of Swing thread context;
wenzelm
parents: 38158
diff changeset
   308
  }
34810
9ad3431a34a5 tuned caret_listener/selected_command;
wenzelm
parents: 34808
diff changeset
   309
37849
4f9de312cc23 Session: predefined real time parameters;
wenzelm
parents: 37685
diff changeset
   310
  private val caret_listener = new CaretListener {
4f9de312cc23 Session: predefined real time parameters;
wenzelm
parents: 37685
diff changeset
   311
    private val delay = Swing_Thread.delay_last(session.input_delay) {
4f9de312cc23 Session: predefined real time parameters;
wenzelm
parents: 37685
diff changeset
   312
      session.perspective.event(Session.Perspective)
34810
9ad3431a34a5 tuned caret_listener/selected_command;
wenzelm
parents: 34808
diff changeset
   313
    }
37849
4f9de312cc23 Session: predefined real time parameters;
wenzelm
parents: 37685
diff changeset
   314
    override def caretUpdate(e: CaretEvent) { delay() }
34810
9ad3431a34a5 tuned caret_listener/selected_command;
wenzelm
parents: 34808
diff changeset
   315
  }
9ad3431a34a5 tuned caret_listener/selected_command;
wenzelm
parents: 34808
diff changeset
   316
9ad3431a34a5 tuned caret_listener/selected_command;
wenzelm
parents: 34808
diff changeset
   317
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   318
  /* overview of command status left of scrollbar */
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   319
34834
df9af932e418 slightly more uniform/robust handling of visible document;
wenzelm
parents: 34832
diff changeset
   320
  private val overview = new JPanel(new BorderLayout)
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   321
  {
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   322
    private val WIDTH = 10
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   323
    private val HEIGHT = 2
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   324
34806
wenzelm
parents: 34794
diff changeset
   325
    setPreferredSize(new Dimension(WIDTH, 0))
wenzelm
parents: 34794
diff changeset
   326
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   327
    setRequestFocusEnabled(false)
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   328
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   329
    addMouseListener(new MouseAdapter {
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   330
      override def mousePressed(event: MouseEvent) {
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   331
        val line = y_to_line(event.getY)
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   332
        if (line >= 0 && line < text_area.getLineCount)
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   333
          text_area.setCaretPosition(text_area.getLineStartOffset(line))
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   334
      }
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   335
    })
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   336
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   337
    override def addNotify() {
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   338
      super.addNotify()
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   339
      ToolTipManager.sharedInstance.registerComponent(this)
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   340
    }
34403
6c812a3cb170 information on command-phase left of scrollbar (with panel)
immler@in.tum.de
parents:
diff changeset
   341
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   342
    override def removeNotify() {
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   343
      ToolTipManager.sharedInstance.unregisterComponent(this)
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   344
      super.removeNotify
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   345
    }
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   346
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   347
    override def getToolTipText(event: MouseEvent): String =
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   348
    {
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   349
      val line = y_to_line(event.getY())
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   350
      if (line >= 0 && line < text_area.getLineCount) "<html><b>TODO:</b><br>Tooltip</html>"
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   351
      else ""
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   352
    }
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   353
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   354
    override def paintComponent(gfx: Graphics)
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   355
    {
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   356
      super.paintComponent(gfx)
38640
105d1f112da5 sporadic locking of jEdit buffer;
wenzelm
parents: 38582
diff changeset
   357
      Swing_Thread.assert()
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   358
      val buffer = model.buffer
38843
d95522496593 more careful locking of jEdit buffer;
wenzelm
parents: 38710
diff changeset
   359
      Isabelle.buffer_lock(buffer) {
38640
105d1f112da5 sporadic locking of jEdit buffer;
wenzelm
parents: 38582
diff changeset
   360
        val snapshot = model.snapshot()
105d1f112da5 sporadic locking of jEdit buffer;
wenzelm
parents: 38582
diff changeset
   361
        val saved_color = gfx.getColor  // FIXME needed!?
105d1f112da5 sporadic locking of jEdit buffer;
wenzelm
parents: 38582
diff changeset
   362
        try {
105d1f112da5 sporadic locking of jEdit buffer;
wenzelm
parents: 38582
diff changeset
   363
          for ((command, start) <- snapshot.node.command_starts if !command.is_ignored) {
105d1f112da5 sporadic locking of jEdit buffer;
wenzelm
parents: 38582
diff changeset
   364
            val line1 = buffer.getLineOfOffset(snapshot.convert(start))
105d1f112da5 sporadic locking of jEdit buffer;
wenzelm
parents: 38582
diff changeset
   365
            val line2 = buffer.getLineOfOffset(snapshot.convert(start + command.length)) + 1
105d1f112da5 sporadic locking of jEdit buffer;
wenzelm
parents: 38582
diff changeset
   366
            val y = line_to_y(line1)
105d1f112da5 sporadic locking of jEdit buffer;
wenzelm
parents: 38582
diff changeset
   367
            val height = HEIGHT * (line2 - line1)
39044
5c13736e81c7 Document_View: squiggly underline for messages;
wenzelm
parents: 38886
diff changeset
   368
            gfx.setColor(Document_View.status_color(snapshot, command))
38640
105d1f112da5 sporadic locking of jEdit buffer;
wenzelm
parents: 38582
diff changeset
   369
            gfx.fillRect(0, y, getWidth - 1, height)
105d1f112da5 sporadic locking of jEdit buffer;
wenzelm
parents: 38582
diff changeset
   370
          }
37188
b78ff6b4f4b3 do not highlight ignored command spans;
wenzelm
parents: 37187
diff changeset
   371
        }
38640
105d1f112da5 sporadic locking of jEdit buffer;
wenzelm
parents: 38582
diff changeset
   372
        finally { gfx.setColor(saved_color) }
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   373
      }
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   374
    }
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   375
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   376
    private def line_to_y(line: Int): Int =
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   377
      (line * getHeight) / (text_area.getBuffer.getLineCount max text_area.getVisibleLines)
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   378
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   379
    private def y_to_line(y: Int): Int =
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   380
      (y * (text_area.getBuffer.getLineCount max text_area.getVisibleLines)) / getHeight
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   381
  }
34403
6c812a3cb170 information on command-phase left of scrollbar (with panel)
immler@in.tum.de
parents:
diff changeset
   382
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   383
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   384
  /* activation */
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   385
34808
e462572536e9 eliminated global Session.document_0 -- did not work due to hardwired id;
wenzelm
parents: 34806
diff changeset
   386
  private def activate()
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   387
  {
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   388
    text_area.getPainter.
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   389
      addExtension(TextAreaPainter.LINE_BACKGROUND_LAYER + 1, text_area_extension)
39131
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   390
    text_area.addFocusListener(focus_listener)
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   391
    text_area.getPainter.addMouseMotionListener(mouse_motion_listener)
34810
9ad3431a34a5 tuned caret_listener/selected_command;
wenzelm
parents: 34808
diff changeset
   392
    text_area.addCaretListener(caret_listener)
9ad3431a34a5 tuned caret_listener/selected_command;
wenzelm
parents: 34808
diff changeset
   393
    text_area.addLeftOfScrollBar(overview)
37129
4c83696b340e Command.toString: include id for debugging;
wenzelm
parents: 36990
diff changeset
   394
    session.commands_changed += commands_changed_actor
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   395
  }
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   396
34808
e462572536e9 eliminated global Session.document_0 -- did not work due to hardwired id;
wenzelm
parents: 34806
diff changeset
   397
  private def deactivate()
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   398
  {
37129
4c83696b340e Command.toString: include id for debugging;
wenzelm
parents: 36990
diff changeset
   399
    session.commands_changed -= commands_changed_actor
39131
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   400
    text_area.removeFocusListener(focus_listener)
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   401
    text_area.getPainter.removeMouseMotionListener(mouse_motion_listener)
947c62440026 basic support for subexpression highlighting (see also gatchan.jedit.hyperlinks.HyperlinkManager/HyperlinkTextAreaPainter);
wenzelm
parents: 39044
diff changeset
   402
    text_area.removeCaretListener(caret_listener)
34810
9ad3431a34a5 tuned caret_listener/selected_command;
wenzelm
parents: 34808
diff changeset
   403
    text_area.removeLeftOfScrollBar(overview)
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   404
    text_area.getPainter.removeExtension(text_area_extension)
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34777
diff changeset
   405
  }
34403
6c812a3cb170 information on command-phase left of scrollbar (with panel)
immler@in.tum.de
parents:
diff changeset
   406
}