src/Pure/Tools/build_dialog.scala
author wenzelm
Sat, 12 Jan 2013 20:13:34 +0100
changeset 50850 4cd2d090be8f
parent 50848 4e123d57c9b4
child 50852 a667ac8c7afe
permissions -rw-r--r--
tuned font size, notably for current HD displays;
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
  {
50368
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
    67
    /* GUI state */
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    68
50368
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
    69
    private var is_stopped = false
50742
38114719a9bc premature window close means failure;
wenzelm
parents: 50740
diff changeset
    70
    private var return_code = 2
50368
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
    71
50740
21098a577294 proper return code on window close;
wenzelm
parents: 50686
diff changeset
    72
    override def closeOperation { sys.exit(return_code) }
21098a577294 proper return code on window close;
wenzelm
parents: 50686
diff changeset
    73
50368
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
    74
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    75
    /* text */
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    76
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    77
    val text = new TextArea {
50850
4cd2d090be8f tuned font size, notably for current HD displays;
wenzelm
parents: 50848
diff changeset
    78
      font = new Font("SansSerif", Font.PLAIN, Library.resolution_scale(10))
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    79
      editable = false
50850
4cd2d090be8f tuned font size, notably for current HD displays;
wenzelm
parents: 50848
diff changeset
    80
      columns = 50
4cd2d090be8f tuned font size, notably for current HD displays;
wenzelm
parents: 50848
diff changeset
    81
      rows = 20
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    82
    }
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    83
50368
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
    84
    val progress = new Build.Progress
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
    85
    {
50846
529e652d389d more uniform theory progress in build -v and build_dialog;
wenzelm
parents: 50845
diff changeset
    86
      override def echo(msg: String): Unit =
529e652d389d more uniform theory progress in build -v and build_dialog;
wenzelm
parents: 50845
diff changeset
    87
        Swing_Thread.later { text.append(msg + "\n") }
529e652d389d more uniform theory progress in build -v and build_dialog;
wenzelm
parents: 50845
diff changeset
    88
      override def theory(session: String, theory: String): Unit =
529e652d389d more uniform theory progress in build -v and build_dialog;
wenzelm
parents: 50845
diff changeset
    89
        echo(session + ": theory " + theory)
50368
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
    90
      override def stopped: Boolean =
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
    91
        Swing_Thread.now { val b = is_stopped; is_stopped = false; b  }
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
    92
    }
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
    93
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
    94
50848
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
    95
    /* action panel */
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
    96
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
    97
    var do_auto_close = true
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
    98
    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
    99
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   100
    val auto_close = new CheckBox("Auto close") {
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   101
      reactions += {
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   102
        case ButtonClicked(_) => do_auto_close = this.selected
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   103
        check_auto_close()
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   104
      }
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
    auto_close.selected = do_auto_close
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   107
    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
   108
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
   109
50376
82cbe4051d98 more direct dialog via existing GUI components;
wenzelm
parents: 50374
diff changeset
   110
    var button_action: () => Unit = (() => is_stopped = true)
82cbe4051d98 more direct dialog via existing GUI components;
wenzelm
parents: 50374
diff changeset
   111
    val button = new Button("Cancel") {
82cbe4051d98 more direct dialog via existing GUI components;
wenzelm
parents: 50374
diff changeset
   112
      reactions += { case ButtonClicked(_) => button_action() }
82cbe4051d98 more direct dialog via existing GUI components;
wenzelm
parents: 50374
diff changeset
   113
    }
50848
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   114
    def button_exit()
50376
82cbe4051d98 more direct dialog via existing GUI components;
wenzelm
parents: 50374
diff changeset
   115
    {
50848
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   116
      check_auto_close()
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   117
      button.text = if (return_code == 0) "OK" else "Exit"
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   118
      button_action = (() => sys.exit(return_code))
50381
d9711842f1f9 clarified default button (cf. org/gjt/sp/jedit/gui/OptionsDialog.java);
wenzelm
parents: 50379
diff changeset
   119
      button.peer.getRootPane.setDefaultButton(button.peer)
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
   120
    }
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
   121
50848
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   122
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   123
    val action_panel = new FlowPanel(FlowPanel.Alignment.Center)(button, auto_close)
50368
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   124
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   125
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   126
    /* layout panel */
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   127
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   128
    val layout_panel = new BorderPanel
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   129
    layout_panel.layout(new ScrollPane(text)) = BorderPanel.Position.Center
50376
82cbe4051d98 more direct dialog via existing GUI components;
wenzelm
parents: 50374
diff changeset
   130
    layout_panel.layout(action_panel) = BorderPanel.Position.South
50368
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   131
e6c0720e4cef basic interaction with build process;
wenzelm
parents: 50365
diff changeset
   132
    contents = layout_panel
50369
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   133
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   134
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   135
    /* main build */
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   136
50374
1a7cae0711d2 tuned message;
wenzelm
parents: 50372
diff changeset
   137
    title = "Isabelle build (" + Isabelle_System.getenv("ML_IDENTIFIER") + ")"
50372
11c96cac860d tuned message;
wenzelm
parents: 50371
diff changeset
   138
    progress.echo("Build started for Isabelle/" + session + " ...")
50369
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   139
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   140
    default_thread_pool.submit(() => {
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   141
      val (out, rc) =
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   142
        try {
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   143
          ("",
50405
366c4a602500 clarified build_dialog: regular up-to-date check (extra cost of approx. 5s startup for HOL);
wenzelm
parents: 50404
diff changeset
   144
            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
   145
              system_mode = system_mode, sessions = List(session)))
50369
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   146
        }
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   147
        catch { case exn: Throwable => (Exn.message(exn) + "\n", 2) }
50376
82cbe4051d98 more direct dialog via existing GUI components;
wenzelm
parents: 50374
diff changeset
   148
      Swing_Thread.now {
82cbe4051d98 more direct dialog via existing GUI components;
wenzelm
parents: 50374
diff changeset
   149
        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
   150
        return_code = rc
50848
4e123d57c9b4 tuned build_dialog: auto_close checkbox avoids user sitting and waiting;
wenzelm
parents: 50846
diff changeset
   151
        button_exit()
50371
9b6f5f758c31 tuned OK feedback;
wenzelm
parents: 50370
diff changeset
   152
      }
50369
622002c702ad more elementary dialog, with less interaction;
wenzelm
parents: 50368
diff changeset
   153
    })
50365
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
   154
  }
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
   155
}
82f5aea343e7 basic wrapper for session build dialog;
wenzelm
parents:
diff changeset
   156