src/Pure/System/system_dialog.scala
author wenzelm
Sat, 07 Sep 2013 16:33:10 +0200
changeset 53458 ddefd18d5ed0
parent 53457 b7c15885fd1e
child 53460 6015a663b889
permissions -rw-r--r--
tuned imports;
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
53453
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
     1
/*  Title:      Pure/Tools/system_dialog.scala
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
     2
    Author:     Makarius
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
     3
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
     4
Dialog for system processes, with optional output window.
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
     5
*/
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
     6
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
     7
package isabelle
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
     8
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
     9
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    10
import java.awt.{GraphicsEnvironment, Point, Font}
53455
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
    11
import javax.swing.WindowConstants
53453
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    12
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    13
import scala.swing.{ScrollPane, Button, CheckBox, FlowPanel,
53458
ddefd18d5ed0 tuned imports;
wenzelm
parents: 53457
diff changeset
    14
  BorderPanel, Frame, TextArea, Component, Label}
53453
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    15
import scala.swing.event.ButtonClicked
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    16
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    17
53454
1a0c39c728a1 clarified result;
wenzelm
parents: 53453
diff changeset
    18
class System_Dialog extends Build.Progress
53453
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    19
{
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    20
  /* GUI state -- owned by Swing thread */
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    21
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    22
  private var _title = "Isabelle"
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    23
  private var _window: Option[Window] = None
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    24
  private var _return_code: Option[Int] = None
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    25
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    26
  private def check_window(): Window =
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    27
  {
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    28
    Swing_Thread.require()
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    29
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    30
    _window match {
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    31
      case Some(window) => window
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    32
      case None =>
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    33
        val window = new Window
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    34
        _window = Some(window)
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    35
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    36
        window.pack()
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    37
        val point = GraphicsEnvironment.getLocalGraphicsEnvironment().getCenterPoint()
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    38
        window.location =
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    39
          new Point(point.x - window.size.width / 2, point.y - window.size.height / 2)
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    40
        window.visible = true
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    41
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    42
        window
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    43
      }
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    44
  }
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    45
53456
d12be8f62285 Build_Dialog based on System_Dialog;
wenzelm
parents: 53455
diff changeset
    46
  private val result = Future.promise[Int]
d12be8f62285 Build_Dialog based on System_Dialog;
wenzelm
parents: 53455
diff changeset
    47
53454
1a0c39c728a1 clarified result;
wenzelm
parents: 53453
diff changeset
    48
  private def conclude()
53453
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    49
  {
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    50
    Swing_Thread.require()
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    51
    require(_return_code.isDefined)
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    52
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    53
    _window match {
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    54
      case None =>
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    55
      case Some(window) =>
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    56
        window.visible = false
53457
b7c15885fd1e more robust exit;
wenzelm
parents: 53456
diff changeset
    57
        window.dispose
53453
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    58
        _window = None
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    59
    }
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    60
53456
d12be8f62285 Build_Dialog based on System_Dialog;
wenzelm
parents: 53455
diff changeset
    61
    try { result.fulfill(_return_code.get) }
d12be8f62285 Build_Dialog based on System_Dialog;
wenzelm
parents: 53455
diff changeset
    62
    catch { case ERROR(_) => }
53453
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    63
  }
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    64
53456
d12be8f62285 Build_Dialog based on System_Dialog;
wenzelm
parents: 53455
diff changeset
    65
  def join(): Int = result.join
d12be8f62285 Build_Dialog based on System_Dialog;
wenzelm
parents: 53455
diff changeset
    66
53453
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    67
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    68
  /* window */
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    69
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    70
  private class Window extends Frame
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    71
  {
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    72
    title = _title
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    73
    iconImage = GUI.isabelle_image()
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    74
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    75
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    76
    /* text */
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    77
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    78
    val text = new TextArea {
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    79
      font = new Font("SansSerif", Font.PLAIN, GUI.resolution_scale(10) max 14)
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    80
      editable = false
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    81
      columns = 50
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    82
      rows = 20
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    83
    }
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    84
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    85
    val scroll_text = new ScrollPane(text)
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    86
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    87
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    88
    /* layout panel with dynamic actions */
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    89
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    90
    val action_panel = new FlowPanel(FlowPanel.Alignment.Center)()
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    91
    val layout_panel = new BorderPanel
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    92
    layout_panel.layout(scroll_text) = BorderPanel.Position.Center
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    93
    layout_panel.layout(action_panel) = BorderPanel.Position.South
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    94
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    95
    contents = layout_panel
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    96
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    97
    def set_actions(cs: Component*)
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    98
    {
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
    99
      action_panel.contents.clear
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   100
      action_panel.contents ++= cs
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   101
      layout_panel.revalidate
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   102
      layout_panel.repaint
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   103
    }
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   104
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   105
53455
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   106
    /* close */
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   107
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   108
    peer.setDefaultCloseOperation(WindowConstants.DO_NOTHING_ON_CLOSE)
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   109
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   110
    override def closeOperation {
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   111
      if (_return_code.isDefined) conclude()
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   112
      else stopping()
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   113
    }
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   114
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   115
    def stopping()
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   116
    {
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   117
      is_stopped = true
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   118
      set_actions(new Label("Stopping ..."))
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   119
    }
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   120
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   121
    val stop_button = new Button("Stop") {
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   122
      reactions += { case ButtonClicked(_) => stopping() }
e9a3390217b3 clarified close operations;
wenzelm
parents: 53454
diff changeset
   123
    }
53453
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   124
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   125
    var do_auto_close = true
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   126
    def can_auto_close: Boolean = do_auto_close && _return_code == Some(0)
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   127
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   128
    val auto_close = new CheckBox("Auto close") {
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   129
      reactions += {
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   130
        case ButtonClicked(_) => do_auto_close = this.selected
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   131
        if (can_auto_close) conclude()
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   132
      }
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   133
    }
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   134
    auto_close.selected = do_auto_close
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   135
    auto_close.tooltip = "Automatically close dialog when finished"
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   136
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   137
    set_actions(stop_button, auto_close)
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   138
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   139
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   140
    /* exit */
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   141
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   142
    val delay_exit =
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   143
      Swing_Thread.delay_first(Time.seconds(1.0))
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   144
      {
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   145
        if (can_auto_close) conclude()
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   146
        else {
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   147
          val button =
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   148
            new Button(if (_return_code == Some(0)) "OK" else "Exit") {
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   149
              reactions += { case ButtonClicked(_) => conclude() }
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   150
            }
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   151
          set_actions(button)
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   152
          button.peer.getRootPane.setDefaultButton(button.peer)
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   153
        }
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   154
      }
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   155
  }
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   156
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   157
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   158
  /* progress operations */
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   159
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   160
  def title(txt: String): Unit =
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   161
    Swing_Thread.later {
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   162
      _title = txt
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   163
      _window.foreach(window => window.title = txt)
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   164
    }
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   165
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   166
  def return_code(rc: Int): Unit =
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   167
    Swing_Thread.later {
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   168
      _return_code = Some(rc)
53456
d12be8f62285 Build_Dialog based on System_Dialog;
wenzelm
parents: 53455
diff changeset
   169
      _window match {
d12be8f62285 Build_Dialog based on System_Dialog;
wenzelm
parents: 53455
diff changeset
   170
        case None => conclude()
d12be8f62285 Build_Dialog based on System_Dialog;
wenzelm
parents: 53455
diff changeset
   171
        case Some(window) => window.delay_exit.invoke
d12be8f62285 Build_Dialog based on System_Dialog;
wenzelm
parents: 53455
diff changeset
   172
      }
53453
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   173
    }
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   174
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   175
  override def echo(txt: String): Unit =
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   176
    Swing_Thread.later {
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   177
      val window = check_window()
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   178
      window.text.append(txt + "\n")
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   179
      val vertical = window.scroll_text.peer.getVerticalScrollBar
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   180
      vertical.setValue(vertical.getMaximum)
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   181
    }
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   182
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   183
  override def theory(session: String, theory: String): Unit =
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   184
    echo(session + ": theory " + theory)
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   185
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   186
  @volatile private var is_stopped = false
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   187
  override def stopped: Boolean = is_stopped
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   188
}
20ff79162ff3 dialog for system processes, with optional output window;
wenzelm
parents:
diff changeset
   189