summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorCarl Hetherington <cth@carlh.net>2020-02-28 00:03:01 +0100
committerCarl Hetherington <cth@carlh.net>2020-02-29 21:51:47 +0100
commitd3f97ca7ca2877689f4ed59482e935431d43f027 (patch)
treed4d43304982d6122a009b747eb3eaedc28727517 /src
parentadeab682b76138d7a05bab3943af23cd57e2bc57 (diff)
Copy swaroop_ playlist editor stuff back to main DoM.
Diffstat (limited to 'src')
-rw-r--r--src/lib/spl.cc71
-rw-r--r--src/lib/spl.h112
-rw-r--r--src/lib/spl_entry.cc72
-rw-r--r--src/lib/spl_entry.h63
-rw-r--r--src/lib/wscript2
-rw-r--r--src/tools/dcpomatic_playlist.cc478
-rw-r--r--src/tools/wscript2
7 files changed, 799 insertions, 1 deletions
diff --git a/src/lib/spl.cc b/src/lib/spl.cc
new file mode 100644
index 000000000..02ed966a4
--- /dev/null
+++ b/src/lib/spl.cc
@@ -0,0 +1,71 @@
+/*
+ Copyright (C) 2018-2019 Carl Hetherington <cth@carlh.net>
+
+ This file is part of DCP-o-matic.
+
+ DCP-o-matic is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ DCP-o-matic is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with DCP-o-matic. If not, see <http://www.gnu.org/licenses/>.
+
+*/
+
+#include "swaroop_spl.h"
+#include "content_store.h"
+#include <libcxml/cxml.h>
+#include <dcp/raw_convert.h>
+#include <libxml++/libxml++.h>
+#include <boost/foreach.hpp>
+#include <iostream>
+
+using std::cout;
+using std::string;
+using boost::shared_ptr;
+using dcp::raw_convert;
+
+void
+SPL::read (boost::filesystem::path path, ContentStore* store)
+{
+ _path = path;
+
+ _spl.clear ();
+ _missing = false;
+ cxml::Document doc ("SPL");
+ doc.read_file (path);
+ _id = doc.string_child("Id");
+ BOOST_FOREACH (cxml::ConstNodePtr i, doc.node_children("Entry")) {
+ shared_ptr<Content> c = store->get(i->string_child("Digest"));
+ if (c) {
+ add (SPLEntry(c, i));
+ } else {
+ _missing = true;
+ }
+ }
+
+ _allowed_shows = doc.optional_number_child<int>("AllowedShows");
+}
+
+void
+SPL::write (boost::filesystem::path path) const
+{
+ _path = path;
+
+ xmlpp::Document doc;
+ xmlpp::Element* root = doc.create_root_node ("SPL");
+ root->add_child("Id")->add_child_text (_id);
+ BOOST_FOREACH (SPLEntry i, _spl) {
+ i.as_xml (root->add_child("Entry"));
+ }
+ if (_allowed_shows) {
+ root->add_child("AllowedShows")->add_child_text(raw_convert<string>(*_allowed_shows));
+ }
+ doc.write_to_file_formatted (path.string());
+}
diff --git a/src/lib/spl.h b/src/lib/spl.h
new file mode 100644
index 000000000..308f5286d
--- /dev/null
+++ b/src/lib/spl.h
@@ -0,0 +1,112 @@
+/*
+ Copyright (C) 2018 Carl Hetherington <cth@carlh.net>
+
+ This file is part of DCP-o-matic.
+
+ DCP-o-matic is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ DCP-o-matic is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with DCP-o-matic. If not, see <http://www.gnu.org/licenses/>.
+
+*/
+
+#ifndef DCPOMATIC_SPL_H
+#define DCPOMATIC_SPL_H
+
+#include "swaroop_spl_entry.h"
+#include <dcp/util.h>
+
+class ContentStore;
+
+class SPL
+{
+public:
+ SPL ()
+ : _id (dcp::make_uuid())
+ , _missing (false)
+ {}
+
+ void add (SPLEntry e) {
+ _spl.push_back (e);
+ }
+
+ void remove (std::size_t index) {
+ _spl.erase (_spl.begin() + index);
+ }
+
+ std::vector<SPLEntry> const & get () const {
+ return _spl;
+ }
+
+ SPLEntry & operator[] (std::size_t index) {
+ return _spl[index];
+ }
+
+ SPLEntry const & operator[] (std::size_t index) const {
+ return _spl[index];
+ }
+
+ void read (boost::filesystem::path path, ContentStore* store);
+ void write (boost::filesystem::path path) const;
+
+ std::string id () const {
+ return _id;
+ }
+
+ boost::optional<boost::filesystem::path> path () const {
+ return _path;
+ }
+
+ std::string name () const {
+ if (!_path) {
+ return "";
+ }
+ return _path->filename().string();
+ }
+
+ bool missing () const {
+ return _missing;
+ }
+
+ boost::optional<int> allowed_shows () const {
+ return _allowed_shows;
+ }
+
+ bool have_allowed_shows () const {
+ return !_allowed_shows || *_allowed_shows > 0;
+ }
+
+ void set_allowed_shows (int s) {
+ _allowed_shows = s;
+ }
+
+ void unset_allowed_shows () {
+ _allowed_shows = boost::optional<int>();
+ }
+
+ void decrement_allowed_shows () {
+ if (_allowed_shows) {
+ (*_allowed_shows)--;
+ }
+
+ }
+
+private:
+ std::string _id;
+ mutable boost::optional<boost::filesystem::path> _path;
+ std::vector<SPLEntry> _spl;
+ /** true if any content was missing when read() was last called on this SPL */
+ bool _missing;
+ /** number of times left that the player will allow this playlist to be played (unset means infinite shows) */
+ boost::optional<int> _allowed_shows;
+};
+
+#endif
diff --git a/src/lib/spl_entry.cc b/src/lib/spl_entry.cc
new file mode 100644
index 000000000..ed5a469ac
--- /dev/null
+++ b/src/lib/spl_entry.cc
@@ -0,0 +1,72 @@
+/*
+ Copyright (C) 2018 Carl Hetherington <cth@carlh.net>
+
+ This file is part of DCP-o-matic.
+
+ DCP-o-matic is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ DCP-o-matic is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with DCP-o-matic. If not, see <http://www.gnu.org/licenses/>.
+
+*/
+
+#include "swaroop_spl_entry.h"
+#include "dcp_content.h"
+#include "dcpomatic_assert.h"
+#include <libxml++/libxml++.h>
+
+using boost::shared_ptr;
+using boost::dynamic_pointer_cast;
+
+SPLEntry::SPLEntry (shared_ptr<Content> content)
+ : skippable (false)
+ , disable_timeline (false)
+ , stop_after_play (false)
+{
+ construct (content);
+}
+
+SPLEntry::SPLEntry (shared_ptr<Content> content, cxml::ConstNodePtr node)
+ : skippable (node->bool_child("Skippable"))
+ , disable_timeline (node->bool_child("DisableTimeline"))
+ , stop_after_play (node->bool_child("StopAfterPlay"))
+{
+ construct (content);
+}
+
+void
+SPLEntry::construct (shared_ptr<Content> c)
+{
+ content = c;
+ shared_ptr<DCPContent> dcp = dynamic_pointer_cast<DCPContent> (content);
+ digest = content->digest ();
+ if (dcp) {
+ name = dcp->name ();
+ DCPOMATIC_ASSERT (dcp->cpl());
+ id = *dcp->cpl();
+ kind = dcp->content_kind().get_value_or(dcp::FEATURE);
+ type = DCP;
+ encrypted = dcp->encrypted ();
+ } else {
+ name = content->path(0).filename().string();
+ type = ECINEMA;
+ kind = dcp::FEATURE;
+ }
+}
+
+void
+SPLEntry::as_xml (xmlpp::Element* e)
+{
+ e->add_child("Digest")->add_child_text(digest);
+ e->add_child("Skippable")->add_child_text(skippable ? "1" : "0");
+ e->add_child("DisableTimeline")->add_child_text(disable_timeline ? "1" : "0");
+ e->add_child("StopAfterPlay")->add_child_text(stop_after_play ? "1" : "0");
+}
diff --git a/src/lib/spl_entry.h b/src/lib/spl_entry.h
new file mode 100644
index 000000000..75e0b5223
--- /dev/null
+++ b/src/lib/spl_entry.h
@@ -0,0 +1,63 @@
+/*
+ Copyright (C) 2018 Carl Hetherington <cth@carlh.net>
+
+ This file is part of DCP-o-matic.
+
+ DCP-o-matic is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ DCP-o-matic is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with DCP-o-matic. If not, see <http://www.gnu.org/licenses/>.
+
+*/
+
+#ifndef DCPOMATIC_SWAROOP_SPL_ENTRY_H
+#define DCPOMATIC_SWAROOP_SPL_ENTRY_H
+
+#include <libcxml/cxml.h>
+#include <dcp/types.h>
+#include <boost/shared_ptr.hpp>
+
+namespace xmlpp {
+ class Element;
+}
+
+class Content;
+
+class SPLEntry
+{
+public:
+ SPLEntry (boost::shared_ptr<Content> content);
+ SPLEntry (boost::shared_ptr<Content> content, cxml::ConstNodePtr node);
+
+ void as_xml (xmlpp::Element* e);
+
+ boost::shared_ptr<Content> content;
+ std::string name;
+ /** Digest of this content */
+ std::string digest;
+ /** CPL ID or something else for MP4 (?) */
+ std::string id;
+ dcp::ContentKind kind;
+ enum Type {
+ DCP,
+ ECINEMA
+ };
+ Type type;
+ bool encrypted;
+ bool skippable;
+ bool disable_timeline;
+ bool stop_after_play;
+
+private:
+ void construct (boost::shared_ptr<Content> content);
+};
+
+#endif
diff --git a/src/lib/wscript b/src/lib/wscript
index 6ef41e91e..a37d873a8 100644
--- a/src/lib/wscript
+++ b/src/lib/wscript
@@ -150,6 +150,8 @@ sources = """
server.cc
shuffler.cc
state.cc
+ spl.cc
+ spl_entry.cc
string_log_entry.cc
string_text_file.cc
string_text_file_content.cc
diff --git a/src/tools/dcpomatic_playlist.cc b/src/tools/dcpomatic_playlist.cc
new file mode 100644
index 000000000..73fcbf802
--- /dev/null
+++ b/src/tools/dcpomatic_playlist.cc
@@ -0,0 +1,478 @@
+/*
+ Copyright (C) 2018 Carl Hetherington <cth@carlh.net>
+
+ This file is part of DCP-o-matic.
+
+ DCP-o-matic is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ DCP-o-matic is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with DCP-o-matic. If not, see <http://www.gnu.org/licenses/>.
+
+*/
+
+#include "../wx/wx_util.h"
+#include "../wx/wx_signal_manager.h"
+#include "../wx/content_view.h"
+#include "../wx/dcpomatic_button.h"
+#include "../lib/util.h"
+#include "../lib/config.h"
+#include "../lib/cross.h"
+#include "../lib/film.h"
+#include "../lib/dcp_content.h"
+#include "../lib/swaroop_spl_entry.h"
+#include "../lib/swaroop_spl.h"
+#include <wx/wx.h>
+#include <wx/listctrl.h>
+#include <wx/imaglist.h>
+#include <wx/spinctrl.h>
+#ifdef __WXOSX__
+#include <ApplicationServices/ApplicationServices.h>
+#endif
+
+using std::exception;
+using std::cout;
+using std::string;
+using boost::optional;
+using boost::shared_ptr;
+using boost::weak_ptr;
+using boost::bind;
+using boost::dynamic_pointer_cast;
+
+class ContentDialog : public wxDialog, public ContentStore
+{
+public:
+ ContentDialog (wxWindow* parent)
+ : wxDialog (parent, wxID_ANY, _("Add content"), wxDefaultPosition, wxSize(800, 640))
+ , _content_view (new ContentView(this))
+ {
+ _content_view->update ();
+
+ wxBoxSizer* overall_sizer = new wxBoxSizer (wxVERTICAL);
+ SetSizer (overall_sizer);
+
+ overall_sizer->Add (_content_view, 1, wxEXPAND | wxALL, DCPOMATIC_DIALOG_BORDER);
+
+ wxSizer* buttons = CreateSeparatedButtonSizer (wxOK | wxCANCEL);
+ if (buttons) {
+ overall_sizer->Add (buttons, wxSizerFlags().Expand().DoubleBorder());
+ }
+
+ overall_sizer->Layout ();
+ }
+
+ shared_ptr<Content> selected () const
+ {
+ return _content_view->selected ();
+ }
+
+ shared_ptr<Content> get (string digest) const
+ {
+ return _content_view->get (digest);
+ }
+
+private:
+ ContentView* _content_view;
+};
+
+class DOMFrame : public wxFrame
+{
+public:
+ explicit DOMFrame (wxString const & title)
+ : wxFrame (0, -1, title)
+ , _content_dialog (new ContentDialog(this))
+ {
+ /* Use a panel as the only child of the Frame so that we avoid
+ the dark-grey background on Windows.
+ */
+ wxPanel* overall_panel = new wxPanel (this, wxID_ANY);
+ wxBoxSizer* h_sizer = new wxBoxSizer (wxHORIZONTAL);
+
+ _list = new wxListCtrl (
+ overall_panel, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_REPORT | wxLC_SINGLE_SEL
+ );
+
+ _list->AppendColumn (_("Name"), wxLIST_FORMAT_LEFT, 400);
+ _list->AppendColumn (_("CPL"), wxLIST_FORMAT_LEFT, 350);
+ _list->AppendColumn (_("Type"), wxLIST_FORMAT_CENTRE, 100);
+ _list->AppendColumn (_("Format"), wxLIST_FORMAT_CENTRE, 75);
+ _list->AppendColumn (_("Encrypted"), wxLIST_FORMAT_CENTRE, 90);
+ _list->AppendColumn (_("Skippable"), wxLIST_FORMAT_CENTRE, 90);
+ _list->AppendColumn (_("Disable timeline"), wxLIST_FORMAT_CENTRE, 125);
+ _list->AppendColumn (_("Stop after play"), wxLIST_FORMAT_CENTRE, 125);
+
+ wxImageList* images = new wxImageList (16, 16);
+ wxIcon tick_icon;
+ wxIcon no_tick_icon;
+#ifdef DCPOMATIX_OSX
+ tick_icon.LoadFile ("tick.png", wxBITMAP_TYPE_PNG_RESOURCE);
+ no_tick_icon.LoadFile ("no_tick.png", wxBITMAP_TYPE_PNG_RESOURCE);
+#else
+ boost::filesystem::path tick_path = shared_path() / "tick.png";
+ tick_icon.LoadFile (std_to_wx(tick_path.string()));
+ boost::filesystem::path no_tick_path = shared_path() / "no_tick.png";
+ no_tick_icon.LoadFile (std_to_wx(no_tick_path.string()));
+#endif
+ images->Add (tick_icon);
+ images->Add (no_tick_icon);
+
+ _list->SetImageList (images, wxIMAGE_LIST_SMALL);
+
+ h_sizer->Add (_list, 1, wxEXPAND | wxALL, DCPOMATIC_SIZER_GAP);
+
+ wxBoxSizer* button_sizer = new wxBoxSizer (wxVERTICAL);
+ _up = new Button (overall_panel, _("Up"));
+ _down = new Button (overall_panel, _("Down"));
+ _add = new Button (overall_panel, _("Add"));
+ _remove = new Button (overall_panel, _("Remove"));
+ _save = new Button (overall_panel, _("Save playlist"));
+ _load = new Button (overall_panel, _("Load playlist"));
+ button_sizer->Add (_up, 0, wxEXPAND | wxBOTTOM, DCPOMATIC_BUTTON_STACK_GAP);
+ button_sizer->Add (_down, 0, wxEXPAND | wxBOTTOM, DCPOMATIC_BUTTON_STACK_GAP);
+ button_sizer->Add (_add, 0, wxEXPAND | wxBOTTOM, DCPOMATIC_BUTTON_STACK_GAP);
+ button_sizer->Add (_remove, 0, wxEXPAND | wxBOTTOM, DCPOMATIC_BUTTON_STACK_GAP);
+ button_sizer->Add (_save, 0, wxEXPAND | wxBOTTOM, DCPOMATIC_BUTTON_STACK_GAP);
+ button_sizer->Add (_load, 0, wxEXPAND | wxBOTTOM, DCPOMATIC_BUTTON_STACK_GAP);
+
+ h_sizer->Add (button_sizer, 0, wxALL, DCPOMATIC_SIZER_GAP);
+
+ wxBoxSizer* v_sizer = new wxBoxSizer (wxVERTICAL);
+
+ wxBoxSizer* allowed_shows_sizer = new wxBoxSizer (wxHORIZONTAL);
+ _allowed_shows_enable = new wxCheckBox (overall_panel, wxID_ANY, _("Limit number of shows with this playlist to"));
+ allowed_shows_sizer->Add (_allowed_shows_enable, 0, wxRIGHT, DCPOMATIC_SIZER_GAP);
+ _allowed_shows = new wxSpinCtrl (overall_panel, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxSP_ARROW_KEYS, 1, 65536, 100);
+ allowed_shows_sizer->Add (_allowed_shows);
+
+ v_sizer->Add (allowed_shows_sizer, 0, wxALL, DCPOMATIC_SIZER_GAP);
+ v_sizer->Add (h_sizer);
+
+ overall_panel->SetSizer (v_sizer);
+
+ _list->Bind (wxEVT_LEFT_DOWN, bind(&DOMFrame::list_left_click, this, _1));
+ _list->Bind (wxEVT_COMMAND_LIST_ITEM_SELECTED, boost::bind (&DOMFrame::selection_changed, this));
+ _list->Bind (wxEVT_COMMAND_LIST_ITEM_DESELECTED, boost::bind (&DOMFrame::selection_changed, this));
+ _up->Bind (wxEVT_BUTTON, bind(&DOMFrame::up_clicked, this));
+ _down->Bind (wxEVT_BUTTON, bind(&DOMFrame::down_clicked, this));
+ _add->Bind (wxEVT_BUTTON, bind(&DOMFrame::add_clicked, this));
+ _remove->Bind (wxEVT_BUTTON, bind(&DOMFrame::remove_clicked, this));
+ _save->Bind (wxEVT_BUTTON, bind(&DOMFrame::save_clicked, this));
+ _load->Bind (wxEVT_BUTTON, bind(&DOMFrame::load_clicked, this));
+ _allowed_shows_enable->Bind (wxEVT_CHECKBOX, bind(&DOMFrame::allowed_shows_changed, this));
+ _allowed_shows->Bind (wxEVT_SPINCTRL, bind(&DOMFrame::allowed_shows_changed, this));
+
+ setup_sensitivity ();
+ }
+
+private:
+
+ void allowed_shows_changed ()
+ {
+ if (_allowed_shows_enable->GetValue()) {
+ _playlist.set_allowed_shows (_allowed_shows->GetValue());
+ } else {
+ _playlist.unset_allowed_shows ();
+ }
+ setup_sensitivity ();
+ }
+
+ void add (SPLEntry e)
+ {
+ wxListItem item;
+ item.SetId (_list->GetItemCount());
+ long const N = _list->InsertItem (item);
+ set_item (N, e);
+ }
+
+ void selection_changed ()
+ {
+ setup_sensitivity ();
+ }
+
+ void set_item (long N, SPLEntry e)
+ {
+ _list->SetItem (N, 0, std_to_wx(e.name));
+ _list->SetItem (N, 1, std_to_wx(e.id));
+ _list->SetItem (N, 2, std_to_wx(dcp::content_kind_to_string(e.kind)));
+ _list->SetItem (N, 3, e.type == SPLEntry::DCP ? _("DCP") : _("E-cinema"));
+ _list->SetItem (N, 4, e.encrypted ? S_("Question|Y") : S_("Question|N"));
+ _list->SetItem (N, COLUMN_SKIPPABLE, wxEmptyString, e.skippable ? 0 : 1);
+ _list->SetItem (N, COLUMN_DISABLE_TIMELINE, wxEmptyString, e.disable_timeline ? 0 : 1);
+ _list->SetItem (N, COLUMN_STOP_AFTER_PLAY, wxEmptyString, e.stop_after_play ? 0 : 1);
+ }
+
+ void setup_sensitivity ()
+ {
+ int const num_selected = _list->GetSelectedItemCount ();
+ long int selected = _list->GetNextItem (-1, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
+ _up->Enable (selected > 0);
+ _down->Enable (selected != -1 && selected < (_list->GetItemCount() - 1));
+ _remove->Enable (num_selected > 0);
+ _allowed_shows->Enable (_allowed_shows_enable->GetValue());
+ }
+
+ void list_left_click (wxMouseEvent& ev)
+ {
+ int flags;
+ long item = _list->HitTest (ev.GetPosition(), flags, 0);
+ int x = ev.GetPosition().x;
+ optional<int> column;
+ for (int i = 0; i < _list->GetColumnCount(); ++i) {
+ x -= _list->GetColumnWidth (i);
+ if (x < 0) {
+ column = i;
+ break;
+ }
+ }
+
+ if (item != -1 && column) {
+ switch (*column) {
+ case COLUMN_SKIPPABLE:
+ _playlist[item].skippable = !_playlist[item].skippable;
+ break;
+ case COLUMN_DISABLE_TIMELINE:
+ _playlist[item].disable_timeline = !_playlist[item].disable_timeline;
+ break;
+ case COLUMN_STOP_AFTER_PLAY:
+ _playlist[item].stop_after_play = !_playlist[item].stop_after_play;
+ break;
+ default:
+ ev.Skip ();
+ }
+ set_item (item, _playlist[item]);
+ } else {
+ ev.Skip ();
+ }
+ }
+
+ void add_clicked ()
+ {
+ int const r = _content_dialog->ShowModal ();
+ if (r == wxID_OK) {
+ shared_ptr<Content> content = _content_dialog->selected ();
+ if (content) {
+ SPLEntry e (content);
+ add (e);
+ _playlist.add (e);
+ }
+ }
+ }
+
+ void up_clicked ()
+ {
+ long int s = _list->GetNextItem (-1, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
+ if (s < 1) {
+ return;
+ }
+
+ SPLEntry tmp = _playlist[s];
+ _playlist[s] = _playlist[s-1];
+ _playlist[s-1] = tmp;
+
+ set_item (s - 1, _playlist[s-1]);
+ set_item (s, _playlist[s]);
+ }
+
+ void down_clicked ()
+ {
+ long int s = _list->GetNextItem (-1, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
+ if (s > (_list->GetItemCount() - 1)) {
+ return;
+ }
+
+ SPLEntry tmp = _playlist[s];
+ _playlist[s] = _playlist[s+1];
+ _playlist[s+1] = tmp;
+
+ set_item (s + 1, _playlist[s+1]);
+ set_item (s, _playlist[s]);
+ }
+
+ void remove_clicked ()
+ {
+ long int s = _list->GetNextItem (-1, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
+ if (s == -1) {
+ return;
+ }
+
+ _playlist.remove (s);
+ _list->DeleteItem (s);
+ }
+
+ void save_clicked ()
+ {
+ Config* c = Config::instance ();
+ wxString default_dir = c->player_playlist_directory() ? std_to_wx(c->player_playlist_directory()->string()) : wxString(wxEmptyString);
+ wxFileDialog* d = new wxFileDialog (this, _("Select playlist file"), default_dir, wxEmptyString, wxT("XML files (*.xml)|*.xml"), wxFD_SAVE | wxFD_OVERWRITE_PROMPT);
+ if (d->ShowModal() == wxID_OK) {
+ boost::filesystem::path file = wx_to_std (d->GetPath());
+ file.replace_extension (".xml");
+ _playlist.write (file);
+ }
+ }
+
+ void load_clicked ()
+ {
+ Config* c = Config::instance ();
+ wxString default_dir = c->player_playlist_directory() ? std_to_wx(c->player_playlist_directory()->string()) : wxString(wxEmptyString);
+ wxFileDialog* d = new wxFileDialog (this, _("Select playlist file"), default_dir, wxEmptyString, wxT("XML files (*.xml)|*.xml"));
+ if (d->ShowModal() == wxID_OK) {
+ _list->DeleteAllItems ();
+ _playlist.read (wx_to_std(d->GetPath()), _content_dialog);
+ if (!_playlist.missing()) {
+ _list->DeleteAllItems ();
+ BOOST_FOREACH (SPLEntry i, _playlist.get()) {
+ add (i);
+ }
+ } else {
+ error_dialog (this, _("Some content in this playlist was not found."));
+ }
+ optional<int> allowed_shows = _playlist.allowed_shows ();
+ _allowed_shows_enable->SetValue (static_cast<bool>(allowed_shows));
+ if (allowed_shows) {
+ _allowed_shows->SetValue (*allowed_shows);
+ } else {
+ _allowed_shows->SetValue (65536);
+ }
+ setup_sensitivity ();
+ }
+ }
+
+ wxListCtrl* _list;
+ wxButton* _up;
+ wxButton* _down;
+ wxButton* _add;
+ wxButton* _remove;
+ wxButton* _save;
+ wxButton* _load;
+ wxCheckBox* _allowed_shows_enable;
+ wxSpinCtrl* _allowed_shows;
+ SPL _playlist;
+ ContentDialog* _content_dialog;
+
+ enum {
+ COLUMN_SKIPPABLE = 5,
+ COLUMN_DISABLE_TIMELINE = 6,
+ COLUMN_STOP_AFTER_PLAY = 7
+ };
+};
+
+/** @class App
+ * @brief The magic App class for wxWidgets.
+ */
+class App : public wxApp
+{
+public:
+ App ()
+ : wxApp ()
+ , _frame (0)
+ {}
+
+private:
+
+ bool OnInit ()
+ try
+ {
+ SetAppName (_("DCP-o-matic KDM Creator"));
+
+ if (!wxApp::OnInit()) {
+ return false;
+ }
+
+#ifdef DCPOMATIC_LINUX
+ unsetenv ("UBUNTU_MENUPROXY");
+#endif
+
+#ifdef __WXOSX__
+ ProcessSerialNumber serial;
+ GetCurrentProcess (&serial);
+ TransformProcessType (&serial, kProcessTransformToForegroundApplication);
+#endif
+
+ dcpomatic_setup_path_encoding ();
+
+ /* Enable i18n; this will create a Config object
+ to look for a force-configured language. This Config
+ object will be wrong, however, because dcpomatic_setup
+ hasn't yet been called and there aren't any filters etc.
+ set up yet.
+ */
+ dcpomatic_setup_i18n ();
+
+ /* Set things up, including filters etc.
+ which will now be internationalised correctly.
+ */
+ dcpomatic_setup ();
+
+ /* Force the configuration to be re-loaded correctly next
+ time it is needed.
+ */
+ Config::drop ();
+
+ _frame = new DOMFrame (_("DCP-o-matic Playlist Editor"));
+ SetTopWindow (_frame);
+ _frame->Maximize ();
+ _frame->Show ();
+
+ signal_manager = new wxSignalManager (this);
+ Bind (wxEVT_IDLE, boost::bind (&App::idle, this));
+
+ return true;
+ }
+ catch (exception& e)
+ {
+ error_dialog (0, _("DCP-o-matic could not start"), std_to_wx(e.what()));
+ return true;
+ }
+
+ /* An unhandled exception has occurred inside the main event loop */
+ bool OnExceptionInMainLoop ()
+ {
+ try {
+ throw;
+ } catch (FileError& e) {
+ error_dialog (
+ 0,
+ wxString::Format (
+ _("An exception occurred: %s (%s)\n\n") + REPORT_PROBLEM,
+ std_to_wx (e.what()),
+ std_to_wx (e.file().string().c_str ())
+ )
+ );
+ } catch (exception& e) {
+ error_dialog (
+ 0,
+ wxString::Format (
+ _("An exception occurred: %s.\n\n") + " " + REPORT_PROBLEM,
+ std_to_wx (e.what ())
+ )
+ );
+ } catch (...) {
+ error_dialog (0, _("An unknown exception occurred.") + " " + REPORT_PROBLEM);
+ }
+
+ /* This will terminate the program */
+ return false;
+ }
+
+ void OnUnhandledException ()
+ {
+ error_dialog (0, _("An unknown exception occurred.") + " " + REPORT_PROBLEM);
+ }
+
+ void idle ()
+ {
+ signal_manager->ui_idle ();
+ }
+
+ DOMFrame* _frame;
+};
+
+IMPLEMENT_APP (App)
diff --git a/src/tools/wscript b/src/tools/wscript
index 710f49eeb..8af9e7589 100644
--- a/src/tools/wscript
+++ b/src/tools/wscript
@@ -60,7 +60,7 @@ def build(bld):
elif bld.env.VARIANT == 'swaroop-studio':
gui_tools = ['dcpomatic', 'dcpomatic_batch', 'dcpomatic_server', 'dcpomatic_kdm', 'dcpomatic_player', 'swaroop_dcpomatic_playlist']
else:
- gui_tools = ['dcpomatic', 'dcpomatic_batch', 'dcpomatic_server', 'dcpomatic_kdm', 'dcpomatic_player']
+ gui_tools = ['dcpomatic', 'dcpomatic_batch', 'dcpomatic_server', 'dcpomatic_kdm', 'dcpomatic_player', 'dcpomatic_playlist']
for t in gui_tools:
obj = bld(features='cxx cxxprogram')