Extract common code out into kdm_for_screen()
[dcpomatic.git] / src / wx / timeline_content_view.cc
1 /*
2     Copyright (C) 2013-2016 Carl Hetherington <cth@carlh.net>
3
4     This file is part of DCP-o-matic.
5
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.
10
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.
15
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/>.
18
19 */
20
21 #include "timeline_content_view.h"
22 #include "timeline.h"
23 #include "wx_util.h"
24 #include "lib/content.h"
25 #include <wx/graphics.h>
26 #include <boost/foreach.hpp>
27
28 using std::list;
29 using boost::shared_ptr;
30 using namespace dcpomatic;
31
32 TimelineContentView::TimelineContentView (Timeline& tl, shared_ptr<Content> c)
33         : TimelineView (tl)
34         , _content (c)
35         , _selected (false)
36 {
37         _content_connection = c->Change.connect (bind (&TimelineContentView::content_change, this, _1, _3));
38 }
39
40 dcpomatic::Rect<int>
41 TimelineContentView::bbox () const
42 {
43         DCPOMATIC_ASSERT (_track);
44
45         shared_ptr<const Film> film = _timeline.film ();
46         shared_ptr<const Content> content = _content.lock ();
47         if (!film || !content) {
48                 return dcpomatic::Rect<int> ();
49         }
50
51         return dcpomatic::Rect<int> (
52                 time_x (content->position ()),
53                 y_pos (_track.get()),
54                 content->length_after_trim(film).seconds() * _timeline.pixels_per_second().get_value_or(0),
55                 _timeline.pixels_per_track()
56                 );
57 }
58
59 void
60 TimelineContentView::set_selected (bool s)
61 {
62         _selected = s;
63         force_redraw ();
64 }
65
66 bool
67 TimelineContentView::selected () const
68 {
69         return _selected;
70 }
71
72 shared_ptr<Content>
73 TimelineContentView::content () const
74 {
75         return _content.lock ();
76 }
77
78 void
79 TimelineContentView::set_track (int t)
80 {
81         _track = t;
82 }
83
84 void
85 TimelineContentView::unset_track ()
86 {
87         _track = boost::optional<int> ();
88 }
89
90 boost::optional<int>
91 TimelineContentView::track () const
92 {
93         return _track;
94 }
95
96 void
97 TimelineContentView::do_paint (wxGraphicsContext* gc, list<dcpomatic::Rect<int> > overlaps)
98 {
99         DCPOMATIC_ASSERT (_track);
100
101         shared_ptr<const Film> film = _timeline.film ();
102         shared_ptr<const Content> cont = content ();
103         if (!film || !cont) {
104                 return;
105         }
106
107         DCPTime const position = cont->position ();
108         DCPTime const len = cont->length_after_trim (film);
109
110         wxColour selected (background_colour().Red() / 2, background_colour().Green() / 2, background_colour().Blue() / 2);
111
112         gc->SetPen (*wxThePenList->FindOrCreatePen (foreground_colour(), 4, wxPENSTYLE_SOLID));
113         if (_selected) {
114                 gc->SetBrush (*wxTheBrushList->FindOrCreateBrush (selected, wxBRUSHSTYLE_SOLID));
115         } else {
116                 gc->SetBrush (*wxTheBrushList->FindOrCreateBrush (background_colour(), wxBRUSHSTYLE_SOLID));
117         }
118
119         /* Outline */
120         wxGraphicsPath path = gc->CreatePath ();
121         path.MoveToPoint    (time_x (position) + 2,           y_pos (_track.get()) + 4);
122         path.AddLineToPoint (time_x (position + len) - 1,     y_pos (_track.get()) + 4);
123         path.AddLineToPoint (time_x (position + len) - 1,     y_pos (_track.get() + 1) - 4);
124         path.AddLineToPoint (time_x (position) + 2,           y_pos (_track.get() + 1) - 4);
125         path.AddLineToPoint (time_x (position) + 2,           y_pos (_track.get()) + 4);
126         gc->StrokePath (path);
127         gc->FillPath (path);
128
129         /* Reel split points */
130         gc->SetPen (*wxThePenList->FindOrCreatePen (foreground_colour(), 1, wxPENSTYLE_DOT));
131         BOOST_FOREACH (DCPTime i, cont->reel_split_points(film)) {
132                 path = gc->CreatePath ();
133                 path.MoveToPoint (time_x (i), y_pos (_track.get()) + 4);
134                 path.AddLineToPoint (time_x (i), y_pos (_track.get() + 1) - 4);
135                 gc->StrokePath (path);
136         }
137
138         /* Overlaps */
139         gc->SetBrush (*wxTheBrushList->FindOrCreateBrush (foreground_colour(), wxBRUSHSTYLE_CROSSDIAG_HATCH));
140         for (list<dcpomatic::Rect<int> >::const_iterator i = overlaps.begin(); i != overlaps.end(); ++i) {
141                 gc->DrawRectangle (i->x, i->y + 4, i->width, i->height - 8);
142         }
143
144         /* Label text */
145         wxString lab = label ();
146         wxDouble lab_width;
147         wxDouble lab_height;
148         wxDouble lab_descent;
149         wxDouble lab_leading;
150         gc->SetFont (gc->CreateFont (*wxNORMAL_FONT, foreground_colour ()));
151         gc->GetTextExtent (lab, &lab_width, &lab_height, &lab_descent, &lab_leading);
152         gc->Clip (wxRegion (time_x (position), y_pos (_track.get()), len.seconds() * _timeline.pixels_per_second().get_value_or(0), _timeline.pixels_per_track()));
153         gc->DrawText (lab, time_x (position) + 12, y_pos (_track.get() + 1) - lab_height - 4);
154         gc->ResetClip ();
155 }
156
157 int
158 TimelineContentView::y_pos (int t) const
159 {
160         return t * _timeline.pixels_per_track() + _timeline.tracks_y_offset();
161 }
162
163 void
164 TimelineContentView::content_change (ChangeType type, int p)
165 {
166         if (type != CHANGE_TYPE_DONE) {
167                 return;
168         }
169
170         ensure_ui_thread ();
171
172         if (p == ContentProperty::POSITION || p == ContentProperty::LENGTH) {
173                 force_redraw ();
174         }
175 }
176
177 wxString
178 TimelineContentView::label () const
179 {
180         return std_to_wx(content()->summary());
181 }