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"
26 #include "wx/job_manager_view.h"
27 #include "wx/servers_list_dialog.h"
28 #include "wx/wx_ptr.h"
29 #include "wx/wx_signal_manager.h"
30 #include "wx/wx_util.h"
31 #include "lib/compose.hpp"
32 #include "lib/config.h"
33 #include "lib/dcpomatic_socket.h"
36 #include "lib/grok/context.h"
39 #include "lib/job_manager.h"
40 #include "lib/make_dcp.h"
41 #include "lib/transcode_job.h"
43 #include "lib/version.h"
44 #include <dcp/filesystem.h>
45 #include <dcp/warnings.h>
46 LIBDCP_DISABLE_WARNINGS
47 #include <wx/aboutdlg.h>
48 #include <wx/cmdline.h>
50 #include <wx/preferences.h>
51 #include <wx/splash.h>
52 #include <wx/stdpaths.h>
54 LIBDCP_ENABLE_WARNINGS
60 using std::dynamic_pointer_cast;
63 using std::make_shared;
65 using std::shared_ptr;
67 using boost::scoped_array;
69 #if BOOST_VERSION >= 106100
70 using namespace boost::placeholders;
74 static list<boost::filesystem::path> films_to_load;
78 ID_file_add_film = DCPOMATIC_MAIN_MENU,
79 ID_tools_encoding_servers,
85 setup_menu (wxMenuBar* m)
87 auto file = new wxMenu;
88 file->Append (ID_file_add_film, _("&Add Film...\tCtrl-A"));
90 file->Append (wxID_EXIT, _("&Exit"));
92 file->Append (wxID_EXIT, _("&Quit"));
96 file->Append(wxID_PREFERENCES, _("&Preferences...\tCtrl-,"));
98 auto edit = new wxMenu;
99 edit->Append (wxID_PREFERENCES, _("&Preferences...\tCtrl-P"));
102 auto tools = new wxMenu;
103 tools->Append (ID_tools_encoding_servers, _("Encoding servers..."));
105 auto help = new wxMenu;
106 help->Append (ID_help_about, _("About"));
108 m->Append (file, _("&File"));
109 #ifndef DCPOMATIC_OSX
110 m->Append (edit, _("&Edit"));
112 m->Append (tools, _("&Tools"));
113 m->Append (help, _("&Help"));
117 class DOMFrame : public wxFrame
125 class DCPDropTarget : public wxFileDropTarget
128 DCPDropTarget(DOMFrame* owner)
132 bool OnDropFiles(wxCoord, wxCoord, wxArrayString const& filenames) override
134 if (filenames.GetCount() == 1) {
135 /* Try to load a directory */
136 auto path = boost::filesystem::path(wx_to_std(filenames[0]));
137 if (dcp::filesystem::is_directory(path)) {
138 _frame->start_job(wx_to_std(filenames[0]));
150 explicit DOMFrame (wxString const & title)
151 : wxFrame (nullptr, -1, title)
152 , _sizer (new wxBoxSizer(wxVERTICAL))
154 auto bar = new wxMenuBar;
158 Config::instance()->Changed.connect (boost::bind (&DOMFrame::config_changed, this, _1));
160 Bind (wxEVT_MENU, boost::bind (&DOMFrame::file_add_film, this), ID_file_add_film);
161 Bind (wxEVT_MENU, boost::bind (&DOMFrame::file_quit, this), wxID_EXIT);
162 Bind (wxEVT_MENU, boost::bind (&DOMFrame::edit_preferences, this), wxID_PREFERENCES);
163 Bind (wxEVT_MENU, boost::bind (&DOMFrame::tools_encoding_servers, this), ID_tools_encoding_servers);
164 Bind (wxEVT_MENU, boost::bind (&DOMFrame::help_about, this), ID_help_about);
166 auto panel = new wxPanel (this);
167 auto s = new wxBoxSizer (wxHORIZONTAL);
168 s->Add (panel, 1, wxEXPAND);
171 wxBitmap add(icon_path("add"), wxBITMAP_TYPE_PNG);
172 wxBitmap pause(icon_path("pause"), wxBITMAP_TYPE_PNG);
174 auto toolbar = new wxToolBar(panel, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTB_HORIZONTAL);
175 toolbar->SetMargins(4, 4);
176 toolbar->SetToolBitmapSize(wxSize(32, 32));
177 toolbar->AddTool(static_cast<int>(Tool::ADD), _("Add film"), add, _("Add film for conversion"));
178 toolbar->AddCheckTool(static_cast<int>(Tool::PAUSE), _("Pause/resume"), pause, wxNullBitmap, _("Pause or resume conversion"));
180 _sizer->Add(toolbar, 0, wxALL, 6);
182 toolbar->Bind(wxEVT_TOOL, bind(&DOMFrame::tool_clicked, this, _1));
184 auto job_manager_view = new JobManagerView (panel, true);
185 _sizer->Add (job_manager_view, 1, wxALL | wxEXPAND, 6);
187 panel->SetSizer (_sizer);
189 Bind (wxEVT_CLOSE_WINDOW, boost::bind(&DOMFrame::close, this, _1));
190 Bind (wxEVT_SIZE, boost::bind(&DOMFrame::sized, this, _1));
192 SetDropTarget(new DCPDropTarget(this));
195 void tool_clicked(wxCommandEvent& ev)
197 switch (static_cast<Tool>(ev.GetId())) {
203 auto jm = JobManager::instance();
214 void start_job (boost::filesystem::path path)
217 auto film = make_shared<Film>(path);
218 film->read_metadata ();
220 double total_required;
224 film->should_be_enough_disk_space (total_required, available, can_hard_link);
226 set<shared_ptr<const Film>> films;
228 for (auto i: JobManager::instance()->get()) {
229 films.insert (i->film());
232 for (auto i: films) {
234 for (auto j: JobManager::instance()->get()) {
235 if (i == j->film() && dynamic_pointer_cast<TranscodeJob>(j)) {
236 progress = j->progress().get_value_or(0);
241 i->should_be_enough_disk_space (required, available, can_hard_link);
242 total_required += (1 - progress) * required;
245 if ((total_required - available) > 1) {
246 if (!confirm_dialog (
249 _("The DCPs for this film and the films already in the queue will take up about %.1f GB. The "
250 "disks that you are using only have %.1f GB available. Do you want to add this film to the queue anyway?"),
251 total_required, available))) {
256 make_dcp (film, TranscodeJob::ChangedBehaviour::STOP);
257 } catch (std::exception& e) {
258 auto p = std_to_wx (path.string ());
259 auto b = p.ToUTF8 ();
260 error_dialog (this, wxString::Format(_("Could not open film at %s"), p.data()), std_to_wx(e.what()));
265 void sized (wxSizeEvent& ev)
273 if (!JobManager::instance()->work_to_do()) {
277 auto d = make_wx<wxMessageDialog>(
279 _("There are unfinished jobs; are you sure you want to quit?"),
280 _("Unfinished jobs"),
281 wxYES_NO | wxYES_DEFAULT | wxICON_QUESTION
284 return d->ShowModal() == wxID_YES;
287 void close (wxCloseEvent& ev)
289 if (!should_close()) {
298 void file_add_film ()
305 if (should_close()) {
310 void edit_preferences ()
312 if (!_config_dialog) {
313 _config_dialog = create_full_config_dialog ();
315 _config_dialog->Show (this);
318 void tools_encoding_servers ()
320 if (!_servers_list_dialog) {
321 _servers_list_dialog = new ServersListDialog (this);
324 _servers_list_dialog->Show ();
329 auto d = make_wx<AboutDialog>(this);
335 auto dialog = make_wx<wxDirDialog>(this, _("Select film to open"), wxStandardPaths::Get().GetDocumentsDir(), wxDEFAULT_DIALOG_STYLE | wxDD_DIR_MUST_EXIST);
337 dialog->SetPath(std_to_wx(_last_parent.get().string()));
342 r = dialog->ShowModal();
343 if (r == wxID_OK && dialog->GetPath() == wxStandardPaths::Get().GetDocumentsDir()) {
344 error_dialog (this, _("You did not select a folder. Make sure that you select a folder before clicking Open."));
351 start_job(wx_to_std(dialog->GetPath()));
354 _last_parent = boost::filesystem::path(wx_to_std(dialog->GetPath())).parent_path();
357 void config_changed (Config::Property what)
359 /* Instantly save any config changes when using the DCP-o-matic GUI */
360 if (what == Config::CINEMAS) {
362 Config::instance()->write_cinemas();
363 } catch (exception& e) {
367 _("Could not write to cinemas file at %s. Your changes have not been saved."),
368 std_to_wx (Config::instance()->cinemas_file().string()).data()
374 Config::instance()->write_config();
375 } catch (exception& e) {
379 _("Could not write to config file at %s. Your changes have not been saved."),
380 std_to_wx (Config::instance()->cinemas_file().string()).data()
386 #ifdef DCPOMATIC_GROK
387 if (what == Config::GROK) {
388 setup_grok_library_path();
393 boost::optional<boost::filesystem::path> _last_parent;
395 wxPreferencesEditor* _config_dialog = nullptr;
396 ServersListDialog* _servers_list_dialog = nullptr;
400 static const wxCmdLineEntryDesc command_line_description[] = {
401 { wxCMD_LINE_PARAM, 0, 0, "film to load", wxCMD_LINE_VAL_STRING, wxCMD_LINE_PARAM_MULTIPLE | wxCMD_LINE_PARAM_OPTIONAL },
402 { wxCMD_LINE_NONE, "", "", "", wxCmdLineParamType (0), 0 }
406 class JobServer : public Server, public Signaller
410 : Server (BATCH_JOB_PORT)
413 void handle (shared_ptr<Socket> socket) override
416 auto const length = socket->read_uint32();
417 if (length < 65536) {
418 scoped_array<char> buffer(new char[length]);
419 socket->read(reinterpret_cast<uint8_t*>(buffer.get()), length);
420 string s(buffer.get());
421 emit(boost::bind(boost::ref(StartJob), s));
422 socket->write (reinterpret_cast<uint8_t const *>("OK"), 3);
429 boost::signals2::signal<void(std::string)> StartJob;
433 class App : public wxApp
435 bool OnInit () override
437 wxInitAllImageHandlers ();
439 SetAppName (_("DCP-o-matic Batch Converter"));
440 is_batch_converter = true;
442 Config::FailedToLoad.connect(boost::bind(&App::config_failed_to_load, this, _1));
443 Config::Warning.connect (boost::bind(&App::config_warning, this, _1));
445 auto splash = maybe_show_splash ();
447 if (!wxApp::OnInit()) {
451 #ifdef DCPOMATIC_LINUX
452 unsetenv ("UBUNTU_MENUPROXY");
455 dcpomatic_setup_path_encoding ();
457 /* Enable i18n; this will create a Config object
458 to look for a force-configured language. This Config
459 object will be wrong, however, because dcpomatic_setup
460 hasn't yet been called and there aren't any filters etc.
463 dcpomatic_setup_i18n ();
465 /* Set things up, including filters etc.
466 which will now be internationalised correctly.
470 /* Force the configuration to be re-loaded correctly next
475 _frame = new DOMFrame (_("DCP-o-matic Batch Converter"));
476 SetTopWindow (_frame);
484 auto server = new JobServer();
485 server->StartJob.connect(bind(&DOMFrame::start_job, _frame, _1));
486 new thread (boost::bind (&JobServer::run, server));
487 } catch (boost::system::system_error& e) {
488 error_dialog(_frame, _("Could not listen for new batch jobs. Perhaps another instance of the DCP-o-matic Batch Converter is running."));
491 signal_manager = new wxSignalManager (this);
492 this->Bind (wxEVT_IDLE, boost::bind (&App::idle, this));
494 shared_ptr<Film> film;
495 for (auto i: films_to_load) {
496 if (dcp::filesystem::is_directory(i)) {
498 film = make_shared<Film>(i);
499 film->read_metadata ();
500 make_dcp (film, TranscodeJob::ChangedBehaviour::EXAMINE_THEN_STOP);
501 } catch (exception& e) {
504 std_to_wx(String::compose(wx_to_std(_("Could not load film %1")), i.string())),
511 #ifdef DCPOMATIC_GROK
512 grk_plugin::setMessengerLogger(new grk_plugin::GrokLogger("[GROK] "));
513 setup_grok_library_path();
521 signal_manager->ui_idle ();
524 void OnInitCmdLine (wxCmdLineParser& parser) override
526 parser.SetDesc (command_line_description);
527 parser.SetSwitchChars (wxT ("-"));
530 bool OnCmdLineParsed (wxCmdLineParser& parser) override
532 for (size_t i = 0; i < parser.GetParamCount(); ++i) {
533 films_to_load.push_back (wx_to_std(parser.GetParam(i)));
539 void config_failed_to_load(Config::LoadFailure what)
541 report_config_load_failure(_frame, what);
544 void config_warning (string m)
546 message_dialog (_frame, std_to_wx(m));