src/Pure/Tools/build_dialog.scala
author wenzelm
Thu, 17 Jan 2013 12:04:05 +0100
changeset 50930 23601c59f347
parent 50854 2b15227b17e8
child 51253 ab4c296a1e60
permissions -rw-r--r--
delay to give users a chance to see what was happening, even with auto_close enabled;
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
50686
d703e3aafa8c moved files;
wenzelm
parents: 50546
diff changeset
     1
/*  Title:      Pure/Tools/build_dialog.scala
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
     2
    Author:     Makarius
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
     3
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
     4
Dialog for session build process.
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
     5
*/
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
     6
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
     7
package isabelle
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
     8
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
     9
50378
72367327bab2 evade ugly default font, notably on Windows laf;
wenzelm
parents: 50377
diff changeset
    10
import java.awt.{GraphicsEnvironment, Point, Font}
50379
a8b0d3729a69 added keyboard shortcut for button (canonical way to do that?);
wenzelm
parents: 50378
diff changeset
    11
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    12
import scala.swing.{ScrollPane, Button, CheckBox, FlowPanel,
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    13
  BorderPanel, MainFrame, TextArea, SwingApplication}
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    14
import scala.swing.event.ButtonClicked
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    15
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    16
50404
898cac1dad5e avoid startup within GUI thread -- it is only required later for dialog;
wenzelm
parents: 50403
diff changeset
    17
object Build_Dialog
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    18
{
50404
898cac1dad5e avoid startup within GUI thread -- it is only required later for dialog;
wenzelm
parents: 50403
diff changeset
    19
  def main(args: Array[String]) =
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    20
  {
50539
3b68e5760a2d tuned error dialog;
wenzelm
parents: 50532
diff changeset
    21
    Platform.init_laf()
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    22
    try {
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    23
      args.toList match {
50369
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
    24
        case
50403
87868964733c more uniform default logic, using settings, options, args etc.;
wenzelm
parents: 50381
diff changeset
    25
          logic_option ::
50546
1b01a57d2749 clarified build_dialog command line;
wenzelm
parents: 50539
diff changeset
    26
          logic ::
50369
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
    27
          Properties.Value.Boolean(system_mode) ::
50403
87868964733c more uniform default logic, using settings, options, args etc.;
wenzelm
parents: 50381
diff changeset
    28
          include_dirs =>
50405
366c4a602500 clarified build_dialog: regular up-to-date check (extra cost of approx. 5s startup for HOL);
wenzelm
parents: 50404
diff changeset
    29
            val more_dirs = include_dirs.map(s => ((false, Path.explode(s))))
50404
898cac1dad5e avoid startup within GUI thread -- it is only required later for dialog;
wenzelm
parents: 50403
diff changeset
    30
898cac1dad5e avoid startup within GUI thread -- it is only required later for dialog;
wenzelm
parents: 50403
diff changeset
    31
            val options = Options.init()
50403
87868964733c more uniform default logic, using settings, options, args etc.;
wenzelm
parents: 50381
diff changeset
    32
            val session =
50404
898cac1dad5e avoid startup within GUI thread -- it is only required later for dialog;
wenzelm
parents: 50403
diff changeset
    33
              Isabelle_System.default_logic(logic,
898cac1dad5e avoid startup within GUI thread -- it is only required later for dialog;
wenzelm
parents: 50403
diff changeset
    34
                if (logic_option != "") options.string(logic_option) else "")
50403
87868964733c more uniform default logic, using settings, options, args etc.;
wenzelm
parents: 50381
diff changeset
    35
50532
345b25cf2e4f actually request heap image in initial up-to-date check;
wenzelm
parents: 50405
diff changeset
    36
            if (Build.build(Build.Ignore_Progress, options, build_heap = true, no_build = true,
50405
366c4a602500 clarified build_dialog: regular up-to-date check (extra cost of approx. 5s startup for HOL);
wenzelm
parents: 50404
diff changeset
    37
                more_dirs = more_dirs, sessions = List(session)) == 0) sys.exit(0)
50404
898cac1dad5e avoid startup within GUI thread -- it is only required later for dialog;
wenzelm
parents: 50403
diff changeset
    38
            else
898cac1dad5e avoid startup within GUI thread -- it is only required later for dialog;
wenzelm
parents: 50403
diff changeset
    39
              Swing_Thread.later {
50405
366c4a602500 clarified build_dialog: regular up-to-date check (extra cost of approx. 5s startup for HOL);
wenzelm
parents: 50404
diff changeset
    40
                val top = build_dialog(options, system_mode, more_dirs, session)
50404
898cac1dad5e avoid startup within GUI thread -- it is only required later for dialog;
wenzelm
parents: 50403
diff changeset
    41
                top.pack()
898cac1dad5e avoid startup within GUI thread -- it is only required later for dialog;
wenzelm
parents: 50403
diff changeset
    42
898cac1dad5e avoid startup within GUI thread -- it is only required later for dialog;
wenzelm
parents: 50403
diff changeset
    43
                val point = GraphicsEnvironment.getLocalGraphicsEnvironment().getCenterPoint()
50532
345b25cf2e4f actually request heap image in initial up-to-date check;
wenzelm
parents: 50405
diff changeset
    44
                top.location =
345b25cf2e4f actually request heap image in initial up-to-date check;
wenzelm
parents: 50405
diff changeset
    45
                  new Point(point.x - top.size.width / 2, point.y - top.size.height / 2)
50404
898cac1dad5e avoid startup within GUI thread -- it is only required later for dialog;
wenzelm
parents: 50403
diff changeset
    46
898cac1dad5e avoid startup within GUI thread -- it is only required later for dialog;
wenzelm
parents: 50403
diff changeset
    47
                top.visible = true
898cac1dad5e avoid startup within GUI thread -- it is only required later for dialog;
wenzelm
parents: 50403
diff changeset
    48
              }
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    49
        case _ => error("Bad arguments:\n" + cat_lines(args))
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    50
      }
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    51
    }
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    52
    catch {
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    53
      case exn: Throwable =>
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    54
        Library.error_dialog(null, "Isabelle build failure",
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    55
          Library.scrollable_text(Exn.message(exn)))
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    56
        sys.exit(2)
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    57
    }
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    58
  }
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    59
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    60
50369
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
    61
  def build_dialog(
50404
898cac1dad5e avoid startup within GUI thread -- it is only required later for dialog;
wenzelm
parents: 50403
diff changeset
    62
    options: Options,
50369
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
    63
    system_mode: Boolean,
50405
366c4a602500 clarified build_dialog: regular up-to-date check (extra cost of approx. 5s startup for HOL);
wenzelm
parents: 50404
diff changeset
    64
    more_dirs: List[(Boolean, Path)],
50369
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
    65
    session: String): MainFrame = new MainFrame
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    66
  {
50854
2b15227b17e8 add icon for toplevel windows;
wenzelm
parents: 50853
diff changeset
    67
    iconImage = Isabelle_System.get_icon().getImage
2b15227b17e8 add icon for toplevel windows;
wenzelm
parents: 50853
diff changeset
    68
2b15227b17e8 add icon for toplevel windows;
wenzelm
parents: 50853
diff changeset
    69
50368
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
    70
    /* GUI state */
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    71
50368
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
    72
    private var is_stopped = false
50742
38114719a9bc premature window close means failure;
wenzelm
parents: 50740
diff changeset
    73
    private var return_code = 2
50368
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
    74
50740
21098a577294 proper return code on window close;
wenzelm
parents: 50686
diff changeset
    75
    override def closeOperation { sys.exit(return_code) }
21098a577294 proper return code on window close;
wenzelm
parents: 50686
diff changeset
    76
50368
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
    77
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    78
    /* text */
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    79
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    80
    val text = new TextArea {
50853
86389991636e lower bound to font size for the sake of Mac OS X (cf. 4cd2d090be8f);
wenzelm
parents: 50852
diff changeset
    81
      font = new Font("SansSerif", Font.PLAIN, Library.resolution_scale(10) max 14)
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    82
      editable = false
50850
4cd2d090be8f tuned font size, notably for current HD displays;
wenzelm
parents: 50848
diff changeset
    83
      columns = 50
4cd2d090be8f tuned font size, notably for current HD displays;
wenzelm
parents: 50848
diff changeset
    84
      rows = 20
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    85
    }
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    86
50852
a667ac8c7afe forced scroll to bottom, for improved cross-platform appearance;
wenzelm
parents: 50850
diff changeset
    87
    val scroll_text = new ScrollPane(text)
a667ac8c7afe forced scroll to bottom, for improved cross-platform appearance;
wenzelm
parents: 50850
diff changeset
    88
50368
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
    89
    val progress = new Build.Progress
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
    90
    {
50846
529e652d389d more uniform theory progress in build -v and build_dialog;
wenzelm
parents: 50845
diff changeset
    91
      override def echo(msg: String): Unit =
50852
a667ac8c7afe forced scroll to bottom, for improved cross-platform appearance;
wenzelm
parents: 50850
diff changeset
    92
        Swing_Thread.later {
a667ac8c7afe forced scroll to bottom, for improved cross-platform appearance;
wenzelm
parents: 50850
diff changeset
    93
          text.append(msg + "\n")
a667ac8c7afe forced scroll to bottom, for improved cross-platform appearance;
wenzelm
parents: 50850
diff changeset
    94
          val vertical = scroll_text.peer.getVerticalScrollBar
a667ac8c7afe forced scroll to bottom, for improved cross-platform appearance;
wenzelm
parents: 50850
diff changeset
    95
          vertical.setValue(vertical.getMaximum)
a667ac8c7afe forced scroll to bottom, for improved cross-platform appearance;
wenzelm
parents: 50850
diff changeset
    96
        }
50846
529e652d389d more uniform theory progress in build -v and build_dialog;
wenzelm
parents: 50845
diff changeset
    97
      override def theory(session: String, theory: String): Unit =
529e652d389d more uniform theory progress in build -v and build_dialog;
wenzelm
parents: 50845
diff changeset
    98
        echo(session + ": theory " + theory)
50368
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
    99
      override def stopped: Boolean =
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   100
        Swing_Thread.now { val b = is_stopped; is_stopped = false; b  }
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   101
    }
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   102
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
   103
50848
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   104
    /* action panel */
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   105
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   106
    var do_auto_close = true
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   107
    def check_auto_close(): Unit = if (do_auto_close && return_code == 0) sys.exit(return_code)
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   108
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   109
    val auto_close = new CheckBox("Auto close") {
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   110
      reactions += {
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   111
        case ButtonClicked(_) => do_auto_close = this.selected
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   112
        check_auto_close()
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   113
      }
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   114
    }
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   115
    auto_close.selected = do_auto_close
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   116
    auto_close.tooltip = "Automatically close dialog when finished"
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   117
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
   118
50376
82cbe4051d98 more direct dialog via existing GUI components;
wenzelm
parents: 50374
diff changeset
   119
    var button_action: () => Unit = (() => is_stopped = true)
82cbe4051d98 more direct dialog via existing GUI components;
wenzelm
parents: 50374
diff changeset
   120
    val button = new Button("Cancel") {
82cbe4051d98 more direct dialog via existing GUI components;
wenzelm
parents: 50374
diff changeset
   121
      reactions += { case ButtonClicked(_) => button_action() }
82cbe4051d98 more direct dialog via existing GUI components;
wenzelm
parents: 50374
diff changeset
   122
    }
50930
23601c59f347 delay to give users a chance to see what was happening, even with auto_close enabled;
wenzelm
parents: 50854
diff changeset
   123
23601c59f347 delay to give users a chance to see what was happening, even with auto_close enabled;
wenzelm
parents: 50854
diff changeset
   124
    val delay_button_exit =
23601c59f347 delay to give users a chance to see what was happening, even with auto_close enabled;
wenzelm
parents: 50854
diff changeset
   125
      Swing_Thread.delay_first(Time.seconds(1.0))
23601c59f347 delay to give users a chance to see what was happening, even with auto_close enabled;
wenzelm
parents: 50854
diff changeset
   126
      {
23601c59f347 delay to give users a chance to see what was happening, even with auto_close enabled;
wenzelm
parents: 50854
diff changeset
   127
        check_auto_close()
23601c59f347 delay to give users a chance to see what was happening, even with auto_close enabled;
wenzelm
parents: 50854
diff changeset
   128
        button.text = if (return_code == 0) "OK" else "Exit"
23601c59f347 delay to give users a chance to see what was happening, even with auto_close enabled;
wenzelm
parents: 50854
diff changeset
   129
        button_action = (() => sys.exit(return_code))
23601c59f347 delay to give users a chance to see what was happening, even with auto_close enabled;
wenzelm
parents: 50854
diff changeset
   130
        button.peer.getRootPane.setDefaultButton(button.peer)
23601c59f347 delay to give users a chance to see what was happening, even with auto_close enabled;
wenzelm
parents: 50854
diff changeset
   131
      }
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
   132
50848
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   133
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   134
    val action_panel = new FlowPanel(FlowPanel.Alignment.Center)(button, auto_close)
50368
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   135
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   136
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   137
    /* layout panel */
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   138
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   139
    val layout_panel = new BorderPanel
50852
a667ac8c7afe forced scroll to bottom, for improved cross-platform appearance;
wenzelm
parents: 50850
diff changeset
   140
    layout_panel.layout(scroll_text) = BorderPanel.Position.Center
50376
82cbe4051d98 more direct dialog via existing GUI components;
wenzelm
parents: 50374
diff changeset
   141
    layout_panel.layout(action_panel) = BorderPanel.Position.South
50368
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   142
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   143
    contents = layout_panel
50369
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   144
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   145
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   146
    /* main build */
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   147
50374
1a7cae0711d2 tuned message;
wenzelm
parents: 50372
diff changeset
   148
    title = "Isabelle build (" + Isabelle_System.getenv("ML_IDENTIFIER") + ")"
50372
11c96cac860d tuned message;
wenzelm
parents: 50371
diff changeset
   149
    progress.echo("Build started for Isabelle/" + session + " ...")
50369
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   150
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   151
    default_thread_pool.submit(() => {
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   152
      val (out, rc) =
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   153
        try {
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   154
          ("",
50405
366c4a602500 clarified build_dialog: regular up-to-date check (extra cost of approx. 5s startup for HOL);
wenzelm
parents: 50404
diff changeset
   155
            Build.build(progress, options, build_heap = true, more_dirs = more_dirs,
50370
d5dbb63df0c7 check for existing image (even if outdated);
wenzelm
parents: 50369
diff changeset
   156
              system_mode = system_mode, sessions = List(session)))
50369
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   157
        }
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   158
        catch { case exn: Throwable => (Exn.message(exn) + "\n", 2) }
50376
82cbe4051d98 more direct dialog via existing GUI components;
wenzelm
parents: 50374
diff changeset
   159
      Swing_Thread.now {
82cbe4051d98 more direct dialog via existing GUI components;
wenzelm
parents: 50374
diff changeset
   160
        progress.echo(out + (if (rc == 0) "OK\n" else "Return code: " + rc + "\n"))
50740
21098a577294 proper return code on window close;
wenzelm
parents: 50686
diff changeset
   161
        return_code = rc
50930
23601c59f347 delay to give users a chance to see what was happening, even with auto_close enabled;
wenzelm
parents: 50854
diff changeset
   162
        delay_button_exit.invoke()
50371
9b6f5f758c31 tuned OK feedback;
wenzelm
parents: 50370
diff changeset
   163
      }
50369
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   164
    })
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
   165
  }
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
   166
}
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
   167