src/Tools/jEdit/src/document_model.scala
author wenzelm
Sun, 05 Oct 2014 18:21:39 +0200
changeset 58547 6080615b8b96
parent 58546 72e2b2a609c4
child 59076 65babcd8b0e6
permissions -rw-r--r--
clarified modules;
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
43282
5d294220ca43 moved sources -- eliminated Netbeans artifact of jedit package directory;
wenzelm
parents: 40851
diff changeset
     1
/*  Title:      Tools/jEdit/src/document_model.scala
36760
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
55778
e1fd8780f997 tuned comment;
wenzelm
parents: 55435
diff changeset
     5
Document model connected to jEdit buffer (node in theory graph or
e1fd8780f997 tuned comment;
wenzelm
parents: 55435
diff changeset
     6
auxiliary file).
36760
b82a698ef6c9 tuned headers;
wenzelm
parents: 36015
diff changeset
     7
*/
34407
aad6834ba380 added some headers and comments;
wenzelm
parents: 34406
diff changeset
     8
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
     9
package isabelle.jedit
34760
dc7f5e0d9d27 misc modernization of names;
wenzelm
parents: 34759
diff changeset
    10
34318
c13e168a8ae6 original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff changeset
    11
36015
6111de7c916a adapted to Scala 2.8.0 Beta 1;
wenzelm
parents: 34871
diff changeset
    12
import isabelle._
6111de7c916a adapted to Scala 2.8.0 Beta 1;
wenzelm
parents: 34871
diff changeset
    13
34693
3e995f100ad2 sealed Edit;
wenzelm
parents: 34685
diff changeset
    14
import scala.collection.mutable
34446
5c79f97ec1d1 superficial tuning;
wenzelm
parents: 34445
diff changeset
    15
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
    16
import org.gjt.sp.jedit.Buffer
34783
cb95d6bbf5f1 clarified BufferListener: use adapter, listen to contentInserted instead of preContentInserted;
wenzelm
parents: 34778
diff changeset
    17
import org.gjt.sp.jedit.buffer.{BufferAdapter, BufferListener, JEditBuffer}
34318
c13e168a8ae6 original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff changeset
    18
34760
dc7f5e0d9d27 misc modernization of names;
wenzelm
parents: 34759
diff changeset
    19
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
    20
object Document_Model
34588
e8ac8794971f superficial tuning;
wenzelm
parents: 34583
diff changeset
    21
{
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
    22
  /* document model of buffer */
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
    23
50565
b00ea974613c tuned property name;
wenzelm
parents: 50363
diff changeset
    24
  private val key = "PIDE.document_model"
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
    25
34788
3779c54a2d21 direct apply for Document_Model and Document_View;
wenzelm
parents: 34784
diff changeset
    26
  def apply(buffer: Buffer): Option[Document_Model] =
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
    27
  {
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57610
diff changeset
    28
    GUI_Thread.require {}
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
    29
    buffer.getProperty(key) match {
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
    30
      case model: Document_Model => Some(model)
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
    31
      case _ => None
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
    32
    }
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
    33
  }
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
    34
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
    35
  def exit(buffer: Buffer)
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
    36
  {
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57610
diff changeset
    37
    GUI_Thread.require {}
34788
3779c54a2d21 direct apply for Document_Model and Document_View;
wenzelm
parents: 34784
diff changeset
    38
    apply(buffer) match {
39636
610dc743932c permissive exit;
wenzelm
parents: 39179
diff changeset
    39
      case None =>
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
    40
      case Some(model) =>
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
    41
        model.deactivate()
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
    42
        buffer.unsetProperty(key)
47058
34761733526c refined init_model: allow change of buffer name as caused by "Save as", for example;
wenzelm
parents: 46920
diff changeset
    43
        buffer.propertiesChanged
34653
2e033aaf128e commands carrying state-information
immler@in.tum.de
parents: 34652
diff changeset
    44
    }
34318
c13e168a8ae6 original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff changeset
    45
  }
43397
dba359c0ae3b more robust init;
wenzelm
parents: 43394
diff changeset
    46
52973
d5f7fa1498b7 tuned signature;
wenzelm
parents: 52887
diff changeset
    47
  def init(session: Session, buffer: Buffer, node_name: Document.Node.Name): Document_Model =
43397
dba359c0ae3b more robust init;
wenzelm
parents: 43394
diff changeset
    48
  {
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57610
diff changeset
    49
    GUI_Thread.require {}
47058
34761733526c refined init_model: allow change of buffer name as caused by "Save as", for example;
wenzelm
parents: 46920
diff changeset
    50
    apply(buffer).map(_.deactivate)
52973
d5f7fa1498b7 tuned signature;
wenzelm
parents: 52887
diff changeset
    51
    val model = new Document_Model(session, buffer, node_name)
43397
dba359c0ae3b more robust init;
wenzelm
parents: 43394
diff changeset
    52
    buffer.setProperty(key, model)
dba359c0ae3b more robust init;
wenzelm
parents: 43394
diff changeset
    53
    model.activate()
47058
34761733526c refined init_model: allow change of buffer name as caused by "Save as", for example;
wenzelm
parents: 46920
diff changeset
    54
    buffer.propertiesChanged
43397
dba359c0ae3b more robust init;
wenzelm
parents: 43394
diff changeset
    55
    model
dba359c0ae3b more robust init;
wenzelm
parents: 43394
diff changeset
    56
  }
34318
c13e168a8ae6 original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff changeset
    57
}
c13e168a8ae6 original sources from Johannes Hölzl a48e0c6ab1aea77c52d596f7efc007a543d3d10c with minor modifications of directory layout;
wenzelm
parents:
diff changeset
    58
38151
2837c952ca31 explicit Change.Snapshot and Document.Node;
wenzelm
parents: 38150
diff changeset
    59
52973
d5f7fa1498b7 tuned signature;
wenzelm
parents: 52887
diff changeset
    60
class Document_Model(val session: Session, val buffer: Buffer, val node_name: Document.Node.Name)
34588
e8ac8794971f superficial tuning;
wenzelm
parents: 34583
diff changeset
    61
{
44385
e7fdb008aa7d propagate editor perspective through document model;
wenzelm
parents: 44379
diff changeset
    62
  /* header */
38222
dac5fa0ac971 replaced individual Document_Model history by all-inclusive one in Session;
wenzelm
parents: 38158
diff changeset
    63
54509
1f77110c94ef maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents: 54464
diff changeset
    64
  def is_theory: Boolean = node_name.is_theory
1f77110c94ef maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents: 54464
diff changeset
    65
48707
ba531af91148 simplified Document.Node.Header -- internalized errors;
wenzelm
parents: 47393
diff changeset
    66
  def node_header(): Document.Node.Header =
46920
5f44c8bea84e more explicit indication of swing thread context;
wenzelm
parents: 46750
diff changeset
    67
  {
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57610
diff changeset
    68
    GUI_Thread.require {}
54509
1f77110c94ef maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents: 54464
diff changeset
    69
1f77110c94ef maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents: 54464
diff changeset
    70
    if (is_theory) {
1f77110c94ef maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents: 54464
diff changeset
    71
      JEdit_Lib.buffer_lock(buffer) {
1f77110c94ef maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents: 54464
diff changeset
    72
        Exn.capture {
56823
37be55461dbe more frugal access to theory text via Reader, reduced costs for I/O text decoding;
wenzelm
parents: 56801
diff changeset
    73
          PIDE.resources.check_thy_reader("", node_name, JEdit_Lib.buffer_reader(buffer))
54509
1f77110c94ef maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents: 54464
diff changeset
    74
        } match {
1f77110c94ef maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents: 54464
diff changeset
    75
          case Exn.Res(header) => header
1f77110c94ef maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents: 54464
diff changeset
    76
          case Exn.Exn(exn) => Document.Node.bad_header(Exn.message(exn))
1f77110c94ef maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents: 54464
diff changeset
    77
        }
46748
8f3ae4d04a2d refined node_header -- more direct buffer access (again);
wenzelm
parents: 46740
diff changeset
    78
      }
8f3ae4d04a2d refined node_header -- more direct buffer access (again);
wenzelm
parents: 46740
diff changeset
    79
    }
54509
1f77110c94ef maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents: 54464
diff changeset
    80
    else Document.Node.no_header
46920
5f44c8bea84e more explicit indication of swing thread context;
wenzelm
parents: 46750
diff changeset
    81
  }
44385
e7fdb008aa7d propagate editor perspective through document model;
wenzelm
parents: 44379
diff changeset
    82
e7fdb008aa7d propagate editor perspective through document model;
wenzelm
parents: 44379
diff changeset
    83
e7fdb008aa7d propagate editor perspective through document model;
wenzelm
parents: 44379
diff changeset
    84
  /* perspective */
e7fdb008aa7d propagate editor perspective through document model;
wenzelm
parents: 44379
diff changeset
    85
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57610
diff changeset
    86
  // owned by GUI thread
52816
c608e0ade554 home-grown mouse handling to pretend that the painted checkbox is actually a Swing component;
wenzelm
parents: 52815
diff changeset
    87
  private var _node_required = false
c608e0ade554 home-grown mouse handling to pretend that the painted checkbox is actually a Swing component;
wenzelm
parents: 52815
diff changeset
    88
  def node_required: Boolean = _node_required
c608e0ade554 home-grown mouse handling to pretend that the painted checkbox is actually a Swing component;
wenzelm
parents: 52815
diff changeset
    89
  def node_required_=(b: Boolean)
c608e0ade554 home-grown mouse handling to pretend that the painted checkbox is actually a Swing component;
wenzelm
parents: 52815
diff changeset
    90
  {
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57610
diff changeset
    91
    GUI_Thread.require {}
54531
8330faaeebd5 restrict node_required status and Theories panel to actual theories;
wenzelm
parents: 54530
diff changeset
    92
    if (_node_required != b && is_theory) {
52816
c608e0ade554 home-grown mouse handling to pretend that the painted checkbox is actually a Swing component;
wenzelm
parents: 52815
diff changeset
    93
      _node_required = b
c608e0ade554 home-grown mouse handling to pretend that the painted checkbox is actually a Swing component;
wenzelm
parents: 52815
diff changeset
    94
      PIDE.options_changed()
52974
908e8a36e975 tuned signature;
wenzelm
parents: 52973
diff changeset
    95
      PIDE.editor.flush()
52816
c608e0ade554 home-grown mouse handling to pretend that the painted checkbox is actually a Swing component;
wenzelm
parents: 52815
diff changeset
    96
    }
c608e0ade554 home-grown mouse handling to pretend that the painted checkbox is actually a Swing component;
wenzelm
parents: 52815
diff changeset
    97
  }
52808
143f225e50f5 allow explicit indication of required node: full eval, no prints;
wenzelm
parents: 52807
diff changeset
    98
55783
da0513d95155 more formal Document.Blobs;
wenzelm
parents: 55781
diff changeset
    99
  def node_perspective(doc_blobs: Document.Blobs): (Boolean, Document.Node.Perspective_Text) =
44385
e7fdb008aa7d propagate editor perspective through document model;
wenzelm
parents: 44379
diff changeset
   100
  {
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57610
diff changeset
   101
    GUI_Thread.require {}
52759
a20631db9c8a support declarative editor_execution_range, instead of old-style check/cancel buttons;
wenzelm
parents: 50565
diff changeset
   102
54509
1f77110c94ef maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents: 54464
diff changeset
   103
    if (Isabelle.continuous_checking && is_theory) {
54325
2c4155003352 clarified Editor.current_command: allow outdated snapshot;
wenzelm
parents: 52977
diff changeset
   104
      val snapshot = this.snapshot()
54530
2c1440f70028 ranges of thy_load commands count as visible within perspective;
wenzelm
parents: 54521
diff changeset
   105
2c1440f70028 ranges of thy_load commands count as visible within perspective;
wenzelm
parents: 54521
diff changeset
   106
      val document_view_ranges =
57883
d50aeb916a4b tuned -- eliminated redundant check (see 1f77110c94ef);
wenzelm
parents: 57621
diff changeset
   107
        for {
d50aeb916a4b tuned -- eliminated redundant check (see 1f77110c94ef);
wenzelm
parents: 57621
diff changeset
   108
          doc_view <- PIDE.document_views(buffer)
d50aeb916a4b tuned -- eliminated redundant check (see 1f77110c94ef);
wenzelm
parents: 57621
diff changeset
   109
          range <- doc_view.perspective(snapshot).ranges
d50aeb916a4b tuned -- eliminated redundant check (see 1f77110c94ef);
wenzelm
parents: 57621
diff changeset
   110
        } yield range
54530
2c1440f70028 ranges of thy_load commands count as visible within perspective;
wenzelm
parents: 54521
diff changeset
   111
56314
9a513737a0b2 tuned signature;
wenzelm
parents: 56208
diff changeset
   112
      val load_ranges =
54530
2c1440f70028 ranges of thy_load commands count as visible within perspective;
wenzelm
parents: 54521
diff changeset
   113
        for {
56314
9a513737a0b2 tuned signature;
wenzelm
parents: 56208
diff changeset
   114
          cmd <- snapshot.node.load_commands
54530
2c1440f70028 ranges of thy_load commands count as visible within perspective;
wenzelm
parents: 54521
diff changeset
   115
          blob_name <- cmd.blobs_names
56457
eea4bbe15745 tuned signature -- prefer static type Document.Node.Name;
wenzelm
parents: 56335
diff changeset
   116
          blob_buffer <- JEdit_Lib.jedit_buffer(blob_name)
54530
2c1440f70028 ranges of thy_load commands count as visible within perspective;
wenzelm
parents: 54521
diff changeset
   117
          if !JEdit_Lib.jedit_text_areas(blob_buffer).isEmpty
2c1440f70028 ranges of thy_load commands count as visible within perspective;
wenzelm
parents: 54521
diff changeset
   118
          start <- snapshot.node.command_start(cmd)
2c1440f70028 ranges of thy_load commands count as visible within perspective;
wenzelm
parents: 54521
diff changeset
   119
          range = snapshot.convert(cmd.proper_range + start)
2c1440f70028 ranges of thy_load commands count as visible within perspective;
wenzelm
parents: 54521
diff changeset
   120
        } yield range
2c1440f70028 ranges of thy_load commands count as visible within perspective;
wenzelm
parents: 54521
diff changeset
   121
56314
9a513737a0b2 tuned signature;
wenzelm
parents: 56208
diff changeset
   122
      val reparse = snapshot.node.load_commands.exists(_.blobs_changed(doc_blobs))
55781
b3a4207fb9a6 proper update of text perspective for nodes with changed blobs, which is important to refresh the corresponding command perspective (otherwise it might refer to invalid thy_load commands and cause full execution of the node by the prover);
wenzelm
parents: 55778
diff changeset
   123
b3a4207fb9a6 proper update of text perspective for nodes with changed blobs, which is important to refresh the corresponding command perspective (otherwise it might refer to invalid thy_load commands and cause full execution of the node by the prover);
wenzelm
parents: 55778
diff changeset
   124
      (reparse,
b3a4207fb9a6 proper update of text perspective for nodes with changed blobs, which is important to refresh the corresponding command perspective (otherwise it might refer to invalid thy_load commands and cause full execution of the node by the prover);
wenzelm
parents: 55778
diff changeset
   125
        Document.Node.Perspective(node_required,
56314
9a513737a0b2 tuned signature;
wenzelm
parents: 56208
diff changeset
   126
          Text.Perspective(document_view_ranges ::: load_ranges),
55781
b3a4207fb9a6 proper update of text perspective for nodes with changed blobs, which is important to refresh the corresponding command perspective (otherwise it might refer to invalid thy_load commands and cause full execution of the node by the prover);
wenzelm
parents: 55778
diff changeset
   127
          PIDE.editor.node_overlays(node_name)))
52849
199e9fa5a5c2 maintain overlays within node perspective;
wenzelm
parents: 52816
diff changeset
   128
    }
57615
df1b3452d71c more explicit discrimination of empty nodes -- suppress from Theories panel;
wenzelm
parents: 57612
diff changeset
   129
    else (false, Document.Node.no_perspective_text)
44385
e7fdb008aa7d propagate editor perspective through document model;
wenzelm
parents: 44379
diff changeset
   130
  }
e7fdb008aa7d propagate editor perspective through document model;
wenzelm
parents: 44379
diff changeset
   131
e7fdb008aa7d propagate editor perspective through document model;
wenzelm
parents: 44379
diff changeset
   132
54509
1f77110c94ef maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents: 54464
diff changeset
   133
  /* blob */
1f77110c94ef maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents: 54464
diff changeset
   134
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57610
diff changeset
   135
  private var _blob: Option[(Bytes, Symbol.Text_Chunk)] = None  // owned by GUI thread
54511
1fd24c96ce9b caching of blob;
wenzelm
parents: 54509
diff changeset
   136
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57610
diff changeset
   137
  private def reset_blob(): Unit = GUI_Thread.require { _blob = None }
54511
1fd24c96ce9b caching of blob;
wenzelm
parents: 54509
diff changeset
   138
55783
da0513d95155 more formal Document.Blobs;
wenzelm
parents: 55781
diff changeset
   139
  def get_blob(): Option[Document.Blob] =
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57610
diff changeset
   140
    GUI_Thread.require {
55783
da0513d95155 more formal Document.Blobs;
wenzelm
parents: 55781
diff changeset
   141
      if (is_theory) None
da0513d95155 more formal Document.Blobs;
wenzelm
parents: 55781
diff changeset
   142
      else {
56473
5b5c750e9763 simplified Text.Chunk -- eliminated ooddities;
wenzelm
parents: 56469
diff changeset
   143
        val (bytes, chunk) =
55783
da0513d95155 more formal Document.Blobs;
wenzelm
parents: 55781
diff changeset
   144
          _blob match {
da0513d95155 more formal Document.Blobs;
wenzelm
parents: 55781
diff changeset
   145
            case Some(x) => x
da0513d95155 more formal Document.Blobs;
wenzelm
parents: 55781
diff changeset
   146
            case None =>
56208
06cc31dff138 clarifed module name;
wenzelm
parents: 55791
diff changeset
   147
              val bytes = PIDE.resources.file_content(buffer)
56746
d37a5d09a277 tuned signature;
wenzelm
parents: 56662
diff changeset
   148
              val chunk = Symbol.Text_Chunk(buffer.getSegment(0, buffer.getLength))
56473
5b5c750e9763 simplified Text.Chunk -- eliminated ooddities;
wenzelm
parents: 56469
diff changeset
   149
              _blob = Some((bytes, chunk))
5b5c750e9763 simplified Text.Chunk -- eliminated ooddities;
wenzelm
parents: 56469
diff changeset
   150
              (bytes, chunk)
55783
da0513d95155 more formal Document.Blobs;
wenzelm
parents: 55781
diff changeset
   151
          }
da0513d95155 more formal Document.Blobs;
wenzelm
parents: 55781
diff changeset
   152
        val changed = pending_edits.is_pending()
56473
5b5c750e9763 simplified Text.Chunk -- eliminated ooddities;
wenzelm
parents: 56469
diff changeset
   153
        Some(Document.Blob(bytes, chunk, changed))
54511
1fd24c96ce9b caching of blob;
wenzelm
parents: 54509
diff changeset
   154
      }
1fd24c96ce9b caching of blob;
wenzelm
parents: 54509
diff changeset
   155
    }
54509
1f77110c94ef maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents: 54464
diff changeset
   156
1f77110c94ef maintain document model for all files, with document view for theory only, and special blob for non-theory files;
wenzelm
parents: 54464
diff changeset
   157
58543
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   158
  /* bibtex entries */
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   159
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   160
  private var _bibtex: Option[List[(String, Text.Offset)]] = None  // owned by GUI thread
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   161
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   162
  private def reset_bibtex(): Unit = GUI_Thread.require { _bibtex = None }
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   163
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   164
  def bibtex_entries(): List[(String, Text.Offset)] =
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   165
    GUI_Thread.require {
58546
72e2b2a609c4 clarified modules;
wenzelm
parents: 58543
diff changeset
   166
      if (Bibtex_JEdit.check(buffer)) {
58543
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   167
        _bibtex match {
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   168
          case Some(entries) => entries
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   169
          case None =>
58547
6080615b8b96 clarified modules;
wenzelm
parents: 58546
diff changeset
   170
            val entries = Bibtex_JEdit.parse_buffer_entries(buffer)
58543
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   171
            _bibtex = Some(entries)
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   172
            entries
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   173
        }
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   174
      }
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   175
      else Nil
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   176
    }
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   177
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   178
50363
2f8dc9e65401 tuned signature in accordance to document operations;
wenzelm
parents: 50344
diff changeset
   179
  /* edits */
50344
608265769ce0 emit bulk edits on initialization of multiple buffers, which greatly improves performance when starting big sessions (e.g. JinjaThreads);
wenzelm
parents: 50207
diff changeset
   180
54461
6c360a6ade0e centralized management of pending buffer edits;
wenzelm
parents: 54441
diff changeset
   181
  def node_edits(
56335
8953d4cc060a store blob content within document node: aux. files that were once open are made persistent;
wenzelm
parents: 56314
diff changeset
   182
      clear: Boolean,
8953d4cc060a store blob content within document node: aux. files that were once open are made persistent;
wenzelm
parents: 56314
diff changeset
   183
      text_edits: List[Text.Edit],
8953d4cc060a store blob content within document node: aux. files that were once open are made persistent;
wenzelm
parents: 56314
diff changeset
   184
      perspective: Document.Node.Perspective_Text): List[Document.Edit_Text] =
57621
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   185
  {
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   186
    val edits: List[Document.Edit_Text] =
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   187
      get_blob() match {
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   188
        case None =>
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   189
          val header_edit = session.header_edit(node_name, node_header())
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   190
          if (clear)
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   191
            List(header_edit,
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   192
              node_name -> Document.Node.Clear(),
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   193
              node_name -> Document.Node.Edits(text_edits),
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   194
              node_name -> perspective)
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   195
          else
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   196
            List(header_edit,
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   197
              node_name -> Document.Node.Edits(text_edits),
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   198
              node_name -> perspective)
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   199
        case Some(blob) =>
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   200
          List(node_name -> Document.Node.Blob(blob),
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   201
            node_name -> Document.Node.Edits(text_edits))
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   202
      }
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   203
    edits.filterNot(_._2.is_void)
caa37976801f more frugal edits;
wenzelm
parents: 57615
diff changeset
   204
  }
50363
2f8dc9e65401 tuned signature in accordance to document operations;
wenzelm
parents: 50344
diff changeset
   205
50344
608265769ce0 emit bulk edits on initialization of multiple buffers, which greatly improves performance when starting big sessions (e.g. JinjaThreads);
wenzelm
parents: 50207
diff changeset
   206
608265769ce0 emit bulk edits on initialization of multiple buffers, which greatly improves performance when starting big sessions (e.g. JinjaThreads);
wenzelm
parents: 50207
diff changeset
   207
  /* pending edits */
43648
e32de528b5ef more explicit edit_node vs. init_node;
wenzelm
parents: 43644
diff changeset
   208
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57610
diff changeset
   209
  private object pending_edits  // owned by GUI thread
38224
809578d7f6af more explicit model of pending text edits;
wenzelm
parents: 38223
diff changeset
   210
  {
54461
6c360a6ade0e centralized management of pending buffer edits;
wenzelm
parents: 54441
diff changeset
   211
    private var pending_clear = false
38425
e467db701d78 moved Text_Edit to Text.Edit;
wenzelm
parents: 38417
diff changeset
   212
    private val pending = new mutable.ListBuffer[Text.Edit]
57615
df1b3452d71c more explicit discrimination of empty nodes -- suppress from Theories panel;
wenzelm
parents: 57612
diff changeset
   213
    private var last_perspective = Document.Node.no_perspective_text
44438
0fc38897248a early filtering of unchanged perspective;
wenzelm
parents: 44436
diff changeset
   214
55781
b3a4207fb9a6 proper update of text perspective for nodes with changed blobs, which is important to refresh the corresponding command perspective (otherwise it might refer to invalid thy_load commands and cause full execution of the node by the prover);
wenzelm
parents: 55778
diff changeset
   215
    def is_pending(): Boolean = pending_clear || !pending.isEmpty
38425
e467db701d78 moved Text_Edit to Text.Edit;
wenzelm
parents: 38417
diff changeset
   216
    def snapshot(): List[Text.Edit] = pending.toList
38224
809578d7f6af more explicit model of pending text edits;
wenzelm
parents: 38223
diff changeset
   217
55783
da0513d95155 more formal Document.Blobs;
wenzelm
parents: 55781
diff changeset
   218
    def flushed_edits(doc_blobs: Document.Blobs): List[Document.Edit_Text] =
38224
809578d7f6af more explicit model of pending text edits;
wenzelm
parents: 38223
diff changeset
   219
    {
54461
6c360a6ade0e centralized management of pending buffer edits;
wenzelm
parents: 54441
diff changeset
   220
      val clear = pending_clear
47393
d760049b2d18 more robust update_perspective, e.g. required after reload of buffer that is not at start position;
wenzelm
parents: 47058
diff changeset
   221
      val edits = snapshot()
55783
da0513d95155 more formal Document.Blobs;
wenzelm
parents: 55781
diff changeset
   222
      val (reparse, perspective) = node_perspective(doc_blobs)
55781
b3a4207fb9a6 proper update of text perspective for nodes with changed blobs, which is important to refresh the corresponding command perspective (otherwise it might refer to invalid thy_load commands and cause full execution of the node by the prover);
wenzelm
parents: 55778
diff changeset
   223
      if (clear || reparse || !edits.isEmpty || last_perspective != perspective) {
54461
6c360a6ade0e centralized management of pending buffer edits;
wenzelm
parents: 54441
diff changeset
   224
        pending_clear = false
47393
d760049b2d18 more robust update_perspective, e.g. required after reload of buffer that is not at start position;
wenzelm
parents: 47058
diff changeset
   225
        pending.clear
54461
6c360a6ade0e centralized management of pending buffer edits;
wenzelm
parents: 54441
diff changeset
   226
        last_perspective = perspective
6c360a6ade0e centralized management of pending buffer edits;
wenzelm
parents: 54441
diff changeset
   227
        node_edits(clear, edits, perspective)
43648
e32de528b5ef more explicit edit_node vs. init_node;
wenzelm
parents: 43644
diff changeset
   228
      }
52759
a20631db9c8a support declarative editor_execution_range, instead of old-style check/cancel buttons;
wenzelm
parents: 50565
diff changeset
   229
      else Nil
38224
809578d7f6af more explicit model of pending text edits;
wenzelm
parents: 38223
diff changeset
   230
    }
809578d7f6af more explicit model of pending text edits;
wenzelm
parents: 38223
diff changeset
   231
54461
6c360a6ade0e centralized management of pending buffer edits;
wenzelm
parents: 54441
diff changeset
   232
    def edit(clear: Boolean, e: Text.Edit)
38224
809578d7f6af more explicit model of pending text edits;
wenzelm
parents: 38223
diff changeset
   233
    {
54511
1fd24c96ce9b caching of blob;
wenzelm
parents: 54509
diff changeset
   234
      reset_blob()
58543
9c1389befa56 maintain Document_Model.bibtex_entries;
wenzelm
parents: 57883
diff changeset
   235
      reset_bibtex()
54511
1fd24c96ce9b caching of blob;
wenzelm
parents: 54509
diff changeset
   236
54461
6c360a6ade0e centralized management of pending buffer edits;
wenzelm
parents: 54441
diff changeset
   237
      if (clear) {
6c360a6ade0e centralized management of pending buffer edits;
wenzelm
parents: 54441
diff changeset
   238
        pending_clear = true
6c360a6ade0e centralized management of pending buffer edits;
wenzelm
parents: 54441
diff changeset
   239
        pending.clear
6c360a6ade0e centralized management of pending buffer edits;
wenzelm
parents: 54441
diff changeset
   240
      }
6c360a6ade0e centralized management of pending buffer edits;
wenzelm
parents: 54441
diff changeset
   241
      pending += e
6c360a6ade0e centralized management of pending buffer edits;
wenzelm
parents: 54441
diff changeset
   242
      PIDE.editor.invoke()
44436
546adfa8a6fc update_perspective without actual edits, bypassing the full state assignment protocol;
wenzelm
parents: 44385
diff changeset
   243
    }
546adfa8a6fc update_perspective without actual edits, bypassing the full state assignment protocol;
wenzelm
parents: 44385
diff changeset
   244
  }
546adfa8a6fc update_perspective without actual edits, bypassing the full state assignment protocol;
wenzelm
parents: 44385
diff changeset
   245
54464
wenzelm
parents: 54461
diff changeset
   246
  def snapshot(): Document.Snapshot =
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57610
diff changeset
   247
    GUI_Thread.require { session.snapshot(node_name, pending_edits.snapshot()) }
34731
c0cb6bd10eec keep BufferListener and TextAreaExtension private;
wenzelm
parents: 34724
diff changeset
   248
55783
da0513d95155 more formal Document.Blobs;
wenzelm
parents: 55781
diff changeset
   249
  def flushed_edits(doc_blobs: Document.Blobs): List[Document.Edit_Text] =
57612
990ffb84489b clarified module name: facilitate alternative GUI frameworks;
wenzelm
parents: 57610
diff changeset
   250
    GUI_Thread.require { pending_edits.flushed_edits(doc_blobs) }
34828
c2308b317244 misc tuning;
wenzelm
parents: 34827
diff changeset
   251
c2308b317244 misc tuning;
wenzelm
parents: 34827
diff changeset
   252
c2308b317244 misc tuning;
wenzelm
parents: 34827
diff changeset
   253
  /* buffer listener */
c2308b317244 misc tuning;
wenzelm
parents: 34827
diff changeset
   254
c2308b317244 misc tuning;
wenzelm
parents: 34827
diff changeset
   255
  private val buffer_listener: BufferListener = new BufferAdapter
c2308b317244 misc tuning;
wenzelm
parents: 34827
diff changeset
   256
  {
40478
4bae781b8f7c replaced Document.Node_Text_Edit by Document.Text_Edit, with treatment of deleted nodes;
wenzelm
parents: 40474
diff changeset
   257
    override def bufferLoaded(buffer: JEditBuffer)
4bae781b8f7c replaced Document.Node_Text_Edit by Document.Text_Edit, with treatment of deleted nodes;
wenzelm
parents: 40474
diff changeset
   258
    {
55791
5821b1937fa5 clarified init_models: simultaneous initialization of all document models, before flushing edits by regular means (via PIDE.editor.invoke) -- important for consolidated doc_blobs when determining initial edits;
wenzelm
parents: 55785
diff changeset
   259
      pending_edits.edit(true, Text.Edit.insert(0, JEdit_Lib.buffer_text(buffer)))
40478
4bae781b8f7c replaced Document.Node_Text_Edit by Document.Text_Edit, with treatment of deleted nodes;
wenzelm
parents: 40474
diff changeset
   260
    }
4bae781b8f7c replaced Document.Node_Text_Edit by Document.Text_Edit, with treatment of deleted nodes;
wenzelm
parents: 40474
diff changeset
   261
34828
c2308b317244 misc tuning;
wenzelm
parents: 34827
diff changeset
   262
    override def contentInserted(buffer: JEditBuffer,
c2308b317244 misc tuning;
wenzelm
parents: 34827
diff changeset
   263
      start_line: Int, offset: Int, num_lines: Int, length: Int)
c2308b317244 misc tuning;
wenzelm
parents: 34827
diff changeset
   264
    {
40478
4bae781b8f7c replaced Document.Node_Text_Edit by Document.Text_Edit, with treatment of deleted nodes;
wenzelm
parents: 40474
diff changeset
   265
      if (!buffer.isLoading)
54461
6c360a6ade0e centralized management of pending buffer edits;
wenzelm
parents: 54441
diff changeset
   266
        pending_edits.edit(false, Text.Edit.insert(offset, buffer.getText(offset, length)))
34828
c2308b317244 misc tuning;
wenzelm
parents: 34827
diff changeset
   267
    }
c2308b317244 misc tuning;
wenzelm
parents: 34827
diff changeset
   268
c2308b317244 misc tuning;
wenzelm
parents: 34827
diff changeset
   269
    override def preContentRemoved(buffer: JEditBuffer,
38426
2858ec7b6dd8 specific types Text.Offset and Text.Range;
wenzelm
parents: 38425
diff changeset
   270
      start_line: Int, offset: Int, num_lines: Int, removed_length: Int)
34828
c2308b317244 misc tuning;
wenzelm
parents: 34827
diff changeset
   271
    {
40478
4bae781b8f7c replaced Document.Node_Text_Edit by Document.Text_Edit, with treatment of deleted nodes;
wenzelm
parents: 40474
diff changeset
   272
      if (!buffer.isLoading)
54461
6c360a6ade0e centralized management of pending buffer edits;
wenzelm
parents: 54441
diff changeset
   273
        pending_edits.edit(false, Text.Edit.remove(offset, buffer.getText(offset, removed_length)))
34828
c2308b317244 misc tuning;
wenzelm
parents: 34827
diff changeset
   274
    }
c2308b317244 misc tuning;
wenzelm
parents: 34827
diff changeset
   275
  }
c2308b317244 misc tuning;
wenzelm
parents: 34827
diff changeset
   276
c2308b317244 misc tuning;
wenzelm
parents: 34827
diff changeset
   277
38158
8aaa21db41f3 Document_Model: include token marker here;
wenzelm
parents: 38152
diff changeset
   278
  /* activation */
37557
1ae272fd4082 refresh Isabelle token marker after buffer properties changed, e.g. when fold mode is switched;
wenzelm
parents: 37555
diff changeset
   279
43512
270ce5ff2086 clarified init/exit procedure;
wenzelm
parents: 43452
diff changeset
   280
  private def activate()
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
   281
  {
55791
5821b1937fa5 clarified init_models: simultaneous initialization of all document models, before flushing edits by regular means (via PIDE.editor.invoke) -- important for consolidated doc_blobs when determining initial edits;
wenzelm
parents: 55785
diff changeset
   282
    pending_edits.edit(true, Text.Edit.insert(0, JEdit_Lib.buffer_text(buffer)))
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
   283
    buffer.addBufferListener(buffer_listener)
44358
2a2df4de1bbe more robust initialization of token marker and line context wrt. session startup;
wenzelm
parents: 44182
diff changeset
   284
    Token_Markup.refresh_buffer(buffer)
34680
1f1f6c95de64 clarified structure of TheoryView
immler@in.tum.de
parents: 34679
diff changeset
   285
  }
1f1f6c95de64 clarified structure of TheoryView
immler@in.tum.de
parents: 34679
diff changeset
   286
43512
270ce5ff2086 clarified init/exit procedure;
wenzelm
parents: 43452
diff changeset
   287
  private def deactivate()
34680
1f1f6c95de64 clarified structure of TheoryView
immler@in.tum.de
parents: 34679
diff changeset
   288
  {
34784
02959dcea756 split Theory_View into Document_Model (connected to Buffer) and Document_View (connected to JEditTextArea);
wenzelm
parents: 34783
diff changeset
   289
    buffer.removeBufferListener(buffer_listener)
44358
2a2df4de1bbe more robust initialization of token marker and line context wrt. session startup;
wenzelm
parents: 44182
diff changeset
   290
    Token_Markup.refresh_buffer(buffer)
34680
1f1f6c95de64 clarified structure of TheoryView
immler@in.tum.de
parents: 34679
diff changeset
   291
  }
34447
wenzelm
parents: 34446
diff changeset
   292
}