2 Copyright (C) 2013-2021 Carl Hetherington <cth@carlh.net>
4 This file is part of DCP-o-matic.
6 DCP-o-matic is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 2 of the License, or
9 (at your option) any later version.
11 DCP-o-matic is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with DCP-o-matic. If not, see <http://www.gnu.org/licenses/>.
22 #include "wx/about_dialog.h"
23 #include "wx/dcpomatic_button.h"
24 #include "wx/full_config_dialog.h"
25 #include "wx/job_manager_view.h"
26 #include "wx/servers_list_dialog.h"
27 #include "wx/wx_signal_manager.h"
28 #include "wx/wx_util.h"
29 #include "lib/compose.hpp"
30 #include "lib/config.h"
31 #include "lib/dcpomatic_socket.h"
34 #include "lib/job_manager.h"
35 #include "lib/make_dcp.h"
36 #include "lib/transcode_job.h"
38 #include "lib/version.h"
39 #include <dcp/warnings.h>
40 LIBDCP_DISABLE_WARNINGS
41 #include <wx/aboutdlg.h>
42 #include <wx/cmdline.h>
43 #include <wx/preferences.h>
44 #include <wx/splash.h>
45 #include <wx/stdpaths.h>
47 LIBDCP_ENABLE_WARNINGS
53 using std::dynamic_pointer_cast;
56 using std::make_shared;
58 using std::shared_ptr;
60 using boost::scoped_array;
62 #if BOOST_VERSION >= 106100
63 using namespace boost::placeholders;
67 static list<boost::filesystem::path> films_to_load;
72 ID_tools_encoding_servers,
78 setup_menu (wxMenuBar* m)
80 auto file = new wxMenu;
81 file->Append (ID_file_add_film, _("&Add Film...\tCtrl-A"));
83 file->Append (wxID_EXIT, _("&Exit"));
85 file->Append (wxID_EXIT, _("&Quit"));
89 file->Append (wxID_PREFERENCES, _("&Preferences...\tCtrl-P"));
91 auto edit = new wxMenu;
92 edit->Append (wxID_PREFERENCES, _("&Preferences...\tCtrl-P"));
95 auto tools = new wxMenu;
96 tools->Append (ID_tools_encoding_servers, _("Encoding servers..."));
98 auto help = new wxMenu;
99 help->Append (ID_help_about, _("About"));
101 m->Append (file, _("&File"));
102 #ifndef DCPOMATIC_OSX
103 m->Append (edit, _("&Edit"));
105 m->Append (tools, _("&Tools"));
106 m->Append (help, _("&Help"));
110 class DOMFrame : public wxFrame
113 explicit DOMFrame (wxString const & title)
114 : wxFrame (nullptr, -1, title)
115 , _sizer (new wxBoxSizer(wxVERTICAL))
117 auto bar = new wxMenuBar;
121 Config::instance()->Changed.connect (boost::bind (&DOMFrame::config_changed, this, _1));
123 Bind (wxEVT_MENU, boost::bind (&DOMFrame::file_add_film, this), ID_file_add_film);
124 Bind (wxEVT_MENU, boost::bind (&DOMFrame::file_quit, this), wxID_EXIT);
125 Bind (wxEVT_MENU, boost::bind (&DOMFrame::edit_preferences, this), wxID_PREFERENCES);
126 Bind (wxEVT_MENU, boost::bind (&DOMFrame::tools_encoding_servers, this), ID_tools_encoding_servers);
127 Bind (wxEVT_MENU, boost::bind (&DOMFrame::help_about, this), ID_help_about);
129 auto panel = new wxPanel (this);
130 auto s = new wxBoxSizer (wxHORIZONTAL);
131 s->Add (panel, 1, wxEXPAND);
134 auto job_manager_view = new JobManagerView (panel, true);
135 _sizer->Add (job_manager_view, 1, wxALL | wxEXPAND, 6);
137 auto buttons = new wxBoxSizer (wxHORIZONTAL);
138 auto add = new Button (panel, _("Add Film..."));
139 add->Bind (wxEVT_BUTTON, boost::bind(&DOMFrame::add_film, this));
140 buttons->Add (add, 1, wxALL, 6);
141 _pause = new Button (panel, _("Pause"));
142 _pause->Bind (wxEVT_BUTTON, boost::bind(&DOMFrame::pause, this));
143 buttons->Add (_pause, 1, wxALL, 6);
144 _resume = new Button (panel, _("Resume"));
145 _resume->Bind (wxEVT_BUTTON, boost::bind(&DOMFrame::resume, this));
146 buttons->Add (_resume, 1, wxALL, 6);
148 setup_sensitivity ();
150 _sizer->Add (buttons, 0, wxALL, 6);
152 panel->SetSizer (_sizer);
154 Bind (wxEVT_CLOSE_WINDOW, boost::bind(&DOMFrame::close, this, _1));
155 Bind (wxEVT_SIZE, boost::bind(&DOMFrame::sized, this, _1));
158 void setup_sensitivity ()
160 _pause->Enable (!JobManager::instance()->paused());
161 _resume->Enable (JobManager::instance()->paused());
166 JobManager::instance()->pause();
167 setup_sensitivity ();
172 JobManager::instance()->resume();
173 setup_sensitivity ();
176 void start_job (boost::filesystem::path path)
179 auto film = make_shared<Film>(path);
180 film->read_metadata ();
182 double total_required;
186 film->should_be_enough_disk_space (total_required, available, can_hard_link);
188 set<shared_ptr<const Film>> films;
190 for (auto i: JobManager::instance()->get()) {
191 films.insert (i->film());
194 for (auto i: films) {
196 for (auto j: JobManager::instance()->get()) {
197 if (i == j->film() && dynamic_pointer_cast<TranscodeJob>(j)) {
198 progress = j->progress().get_value_or(0);
203 i->should_be_enough_disk_space (required, available, can_hard_link);
204 total_required += (1 - progress) * required;
207 if ((total_required - available) > 1) {
208 if (!confirm_dialog (
211 _("The DCPs for this film and the films already in the queue will take up about %.1f GB. The "
212 "disks that you are using only have %.1f GB available. Do you want to add this film to the queue anyway?"),
213 total_required, available))) {
218 make_dcp (film, TranscodeJob::ChangedBehaviour::STOP);
219 } catch (std::exception& e) {
220 auto p = std_to_wx (path.string ());
221 auto b = p.ToUTF8 ();
222 error_dialog (this, wxString::Format(_("Could not open film at %s"), p.data()), std_to_wx(e.what()));
227 void sized (wxSizeEvent& ev)
235 if (!JobManager::instance()->work_to_do()) {
239 auto d = new wxMessageDialog (
241 _("There are unfinished jobs; are you sure you want to quit?"),
242 _("Unfinished jobs"),
243 wxYES_NO | wxYES_DEFAULT | wxICON_QUESTION
246 bool const r = d->ShowModal() == wxID_YES;
251 void close (wxCloseEvent& ev)
253 if (!should_close()) {
261 void file_add_film ()
268 if (should_close()) {
273 void edit_preferences ()
275 if (!_config_dialog) {
276 _config_dialog = create_full_config_dialog ();
278 _config_dialog->Show (this);
281 void tools_encoding_servers ()
283 if (!_servers_list_dialog) {
284 _servers_list_dialog = new ServersListDialog (this);
287 _servers_list_dialog->Show ();
292 auto d = new AboutDialog (this);
299 auto c = new wxDirDialog (this, _("Select film to open"), wxStandardPaths::Get().GetDocumentsDir(), wxDEFAULT_DIALOG_STYLE | wxDD_DIR_MUST_EXIST);
301 c->SetPath (std_to_wx(_last_parent.get().string()));
307 if (r == wxID_OK && c->GetPath() == wxStandardPaths::Get().GetDocumentsDir()) {
308 error_dialog (this, _("You did not select a folder. Make sure that you select a folder before clicking Open."));
315 start_job (wx_to_std (c->GetPath ()));
318 _last_parent = boost::filesystem::path (wx_to_std (c->GetPath ())).parent_path ();
323 void config_changed (Config::Property what)
325 /* Instantly save any config changes when using the DCP-o-matic GUI */
326 if (what == Config::CINEMAS) {
328 Config::instance()->write_cinemas();
329 } catch (exception& e) {
333 _("Could not write to cinemas file at %s. Your changes have not been saved."),
334 std_to_wx (Config::instance()->cinemas_file().string()).data()
340 Config::instance()->write_config();
341 } catch (exception& e) {
345 _("Could not write to config file at %s. Your changes have not been saved."),
346 std_to_wx (Config::instance()->cinemas_file().string()).data()
353 boost::optional<boost::filesystem::path> _last_parent;
355 wxPreferencesEditor* _config_dialog = nullptr;
356 ServersListDialog* _servers_list_dialog = nullptr;
362 static const wxCmdLineEntryDesc command_line_description[] = {
363 { wxCMD_LINE_PARAM, 0, 0, "film to load", wxCMD_LINE_VAL_STRING, wxCMD_LINE_PARAM_MULTIPLE | wxCMD_LINE_PARAM_OPTIONAL },
364 { wxCMD_LINE_NONE, "", "", "", wxCmdLineParamType (0), 0 }
368 class JobServer : public Server
371 explicit JobServer (DOMFrame* frame)
372 : Server (BATCH_JOB_PORT)
376 void handle (shared_ptr<Socket> socket) override
379 int const length = socket->read_uint32 ();
380 scoped_array<char> buffer(new char[length]);
381 socket->read (reinterpret_cast<uint8_t*>(buffer.get()), length);
382 string s (buffer.get());
383 _frame->start_job (s);
384 socket->write (reinterpret_cast<uint8_t const *>("OK"), 3);
395 class App : public wxApp
397 bool OnInit () override
399 wxInitAllImageHandlers ();
401 SetAppName (_("DCP-o-matic Batch Converter"));
402 is_batch_converter = true;
404 Config::FailedToLoad.connect(boost::bind(&App::config_failed_to_load, this, _1));
405 Config::Warning.connect (boost::bind(&App::config_warning, this, _1));
407 auto splash = maybe_show_splash ();
409 if (!wxApp::OnInit()) {
413 #ifdef DCPOMATIC_LINUX
414 unsetenv ("UBUNTU_MENUPROXY");
417 dcpomatic_setup_path_encoding ();
419 /* Enable i18n; this will create a Config object
420 to look for a force-configured language. This Config
421 object will be wrong, however, because dcpomatic_setup
422 hasn't yet been called and there aren't any filters etc.
425 dcpomatic_setup_i18n ();
427 /* Set things up, including filters etc.
428 which will now be internationalised correctly.
432 /* Force the configuration to be re-loaded correctly next
437 _frame = new DOMFrame (_("DCP-o-matic Batch Converter"));
438 SetTopWindow (_frame);
445 auto server = new JobServer (_frame);
446 new thread (boost::bind (&JobServer::run, server));
448 signal_manager = new wxSignalManager (this);
449 this->Bind (wxEVT_IDLE, boost::bind (&App::idle, this));
451 shared_ptr<Film> film;
452 for (auto i: films_to_load) {
453 if (boost::filesystem::is_directory(i)) {
455 film = make_shared<Film>(i);
456 film->read_metadata ();
457 make_dcp (film, TranscodeJob::ChangedBehaviour::EXAMINE_THEN_STOP);
458 } catch (exception& e) {
461 std_to_wx(String::compose(wx_to_std(_("Could not load film %1")), i.string())),
473 signal_manager->ui_idle ();
476 void OnInitCmdLine (wxCmdLineParser& parser) override
478 parser.SetDesc (command_line_description);
479 parser.SetSwitchChars (wxT ("-"));
482 bool OnCmdLineParsed (wxCmdLineParser& parser) override
484 for (size_t i = 0; i < parser.GetParamCount(); ++i) {
485 films_to_load.push_back (wx_to_std(parser.GetParam(i)));
491 void config_failed_to_load(Config::LoadFailure what)
493 report_config_load_failure(_frame, what);
496 void config_warning (string m)
498 message_dialog (_frame, std_to_wx(m));