Merge master.
[dcpomatic.git] / src / wx / timecode.cc
index 460f7423b0d066feb9c8be54a125dec0925bc69f..1ab4b590b2b130b4f7d0c80cd21c766c6706ff41 100644 (file)
 */
 
 #include <boost/lexical_cast.hpp>
+#include "lib/util.h"
 #include "timecode.h"
 #include "wx_util.h"
 
 using std::string;
+using std::cout;
 using boost::lexical_cast;
 
 Timecode::Timecode (wxWindow* parent)
@@ -34,57 +36,108 @@ Timecode::Timecode (wxWindow* parent)
        wxTextValidator validator (wxFILTER_INCLUDE_CHAR_LIST);
        wxArrayString list;
 
-       string n = "0123456789";
-       for (size_t i = 0; i < n.length(); ++i) {
+       wxString n (wxT ("0123456789"));
+       for (size_t i = 0; i < n.Length(); ++i) {
                list.Add (n[i]);
        }
 
        validator.SetIncludes (list);
+
+       _sizer = new wxBoxSizer (wxHORIZONTAL);
        
-       wxBoxSizer* sizer = new wxBoxSizer (wxHORIZONTAL);
-       _hours = new wxTextCtrl (this, wxID_ANY, wxT(""), wxDefaultPosition, size, 0, validator);
+       _editable = new wxPanel (this);
+       wxSizer* editable_sizer = new wxBoxSizer (wxHORIZONTAL);
+       _hours = new wxTextCtrl (_editable, wxID_ANY, wxT(""), wxDefaultPosition, size, 0, validator);
        _hours->SetMaxLength (2);
-       sizer->Add (_hours);
-       add_label_to_sizer (sizer, this, ":");
-       _minutes = new wxTextCtrl (this, wxID_ANY, wxT(""), wxDefaultPosition, size);
+       editable_sizer->Add (_hours);
+       add_label_to_sizer (editable_sizer, _editable, wxT (":"), false);
+       _minutes = new wxTextCtrl (_editable, wxID_ANY, wxT(""), wxDefaultPosition, size, 0, validator);
        _minutes->SetMaxLength (2);
-       sizer->Add (_minutes);
-       add_label_to_sizer (sizer, this, ":");
-       _seconds = new wxTextCtrl (this, wxID_ANY, wxT(""), wxDefaultPosition, size);
+       editable_sizer->Add (_minutes);
+       add_label_to_sizer (editable_sizer, _editable, wxT (":"), false);
+       _seconds = new wxTextCtrl (_editable, wxID_ANY, wxT(""), wxDefaultPosition, size, 0, validator);
        _seconds->SetMaxLength (2);
-       sizer->Add (_seconds);
-       add_label_to_sizer (sizer, this, ".");
-       _frames = new wxTextCtrl (this, wxID_ANY, wxT(""), wxDefaultPosition, size);
+       editable_sizer->Add (_seconds);
+       add_label_to_sizer (editable_sizer, _editable, wxT (":"), false);
+       _frames = new wxTextCtrl (_editable, wxID_ANY, wxT(""), wxDefaultPosition, size, 0, validator);
        _frames->SetMaxLength (2);
-       sizer->Add (_frames);
+       editable_sizer->Add (_frames);
+       _set_button = new wxButton (_editable, wxID_ANY, _("Set"));
+       editable_sizer->Add (_set_button, 0, wxLEFT | wxRIGHT, 8);
+       _editable->SetSizerAndFit (editable_sizer);
+       _sizer->Add (_editable);
+
+       _fixed = add_label_to_sizer (_sizer, this, wxT ("42"), false);
+       
+       _hours->Bind      (wxEVT_COMMAND_TEXT_UPDATED,   boost::bind (&Timecode::changed, this));
+       _minutes->Bind    (wxEVT_COMMAND_TEXT_UPDATED,   boost::bind (&Timecode::changed, this));
+       _seconds->Bind    (wxEVT_COMMAND_TEXT_UPDATED,   boost::bind (&Timecode::changed, this));
+       _frames->Bind     (wxEVT_COMMAND_TEXT_UPDATED,   boost::bind (&Timecode::changed, this));
+       _set_button->Bind (wxEVT_COMMAND_BUTTON_CLICKED, boost::bind (&Timecode::set_clicked, this));
 
-       SetSizerAndFit (sizer);
+       _set_button->Enable (false);
+
+       set_editable (true);
+
+       SetSizerAndFit (_sizer);
 }
 
 void
-Timecode::set (Time t, int fps)
+Timecode::set (DCPTime t, int fps)
 {
-       int const h = t / (3600 * TIME_HZ);
-       t -= h * 3600 * TIME_HZ;
-       int const m = t / (60 * TIME_HZ);
-       t -= m * 60 * TIME_HZ;
-       int const s = t / TIME_HZ;
-       t -= s * TIME_HZ;
-       int const f = t * fps / TIME_HZ;
-
-       _hours->SetValue (wxString::Format ("%02d", h));
-       _minutes->SetValue (wxString::Format ("%02d", m));
-       _seconds->SetValue (wxString::Format ("%02d", s));
-       _frames->SetValue (wxString::Format ("%02d", f));
+       /* Do this calculation with frames so that we can round
+          to a frame boundary at the start rather than the end.
+       */
+       int64_t f = rint (t.seconds() * fps);
+       
+       int const h = f / (3600 * fps);
+       f -= h * 3600 * fps;
+       int const m = f / (60 * fps);
+       f -= m * 60 * fps;
+       int const s = f / fps;
+       f -= s * fps;
+
+       checked_set (_hours, lexical_cast<string> (h));
+       checked_set (_minutes, lexical_cast<string> (m));
+       checked_set (_seconds, lexical_cast<string> (s));
+       checked_set (_frames, lexical_cast<string> (f));
+
+       _fixed->SetLabel (wxString::Format ("%02d:%02d:%02d.%02" wxLongLongFmtSpec "d", h, m, s, f));
 }
 
-Time
+DCPTime
 Timecode::get (int fps) const
 {
-       Time t = 0;
-       t += lexical_cast<int> (wx_to_std (_hours->GetValue())) * 3600 * TIME_HZ;
-       t += lexical_cast<int> (wx_to_std (_minutes->GetValue())) * 60 * TIME_HZ;
-       t += lexical_cast<int> (wx_to_std (_seconds->GetValue())) * TIME_HZ;
-       t += lexical_cast<int> (wx_to_std (_frames->GetValue())) * TIME_HZ / fps;
+       DCPTime t;
+       string const h = wx_to_std (_hours->GetValue ());
+       t += DCPTime::from_seconds (lexical_cast<int> (h.empty() ? "0" : h) * 3600);
+       string const m = wx_to_std (_minutes->GetValue());
+       t += DCPTime::from_seconds (lexical_cast<int> (m.empty() ? "0" : m) * 60);
+       string const s = wx_to_std (_seconds->GetValue());
+       t += DCPTime::from_seconds (lexical_cast<int> (s.empty() ? "0" : s));
+       string const f = wx_to_std (_frames->GetValue());
+       t += DCPTime::from_seconds (lexical_cast<double> (f.empty() ? "0" : f) / fps);
+
        return t;
 }
+
+void
+Timecode::changed ()
+{
+       _set_button->Enable (true);
+}
+
+void
+Timecode::set_clicked ()
+{
+       Changed ();
+       _set_button->Enable (false);
+}
+
+void
+Timecode::set_editable (bool e)
+{
+       _editable->Show (e);
+       _fixed->Show (!e);
+       _sizer->Layout ();
+}