summaryrefslogtreecommitdiff
path: root/gtk2_ardour/tempo_lines.cc
blob: 083b7a5cca8b4db2049dc4b784d6acf490cea4b0 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
/*
    Copyright (C) 2002-2007 Paul Davis

    This program 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.

    This program 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 this program; if not, write to the Free Software
    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.

*/

#include "canvas/line.h"
#include "canvas/canvas.h"
#include "canvas/debug.h"
#include "tempo_lines.h"
#include "ardour_ui.h"

using namespace std;

TempoLines::TempoLines (ArdourCanvas::Canvas& canvas, ArdourCanvas::Group* group, double h)
	: _canvas (canvas)
	, _group (group)
	, _height (h)
{
}

void
TempoLines::tempo_map_changed()
{
	/* remove all lines from the group, put them in the cache (to avoid
	 * unnecessary object destruction+construction later), and clear _lines
	 */
	 
	_group->clear ();
	_cache.insert (_cache.end(), _lines.begin(), _lines.end());
	_lines.clear ();
}

void
TempoLines::show ()
{
	_group->show ();
}

void
TempoLines::hide ()
{
	_group->hide ();
}

void
TempoLines::draw (const ARDOUR::TempoMap::BBTPointList::const_iterator& begin, 
		  const ARDOUR::TempoMap::BBTPointList::const_iterator& end, 
		  double samples_per_pixel)
{
	ARDOUR::TempoMap::BBTPointList::const_iterator i;
	ArdourCanvas::Rect const visible = _canvas.visible_area ();
	double  beat_density;

	uint32_t beats = 0;
	uint32_t bars = 0;
	uint32_t color;

	/* get the first bar spacing */

	i = end;
	i--;
	bars = (*i).bar - (*begin).bar; 
	beats = distance (begin, end) - bars;

	beat_density = (beats * 10.0f) / visible.width ();

	if (beat_density > 4.0f) {
		/* if the lines are too close together, they become useless */
		tempo_map_changed();
		return;
	}

	tempo_map_changed ();

	for (i = begin; i != end; ++i) {

		if ((*i).is_bar()) {
			color = ARDOUR_UI::config()->get_canvasvar_MeasureLineBar();
		} else {
			if (beat_density > 2.0) {
				continue; /* only draw beat lines if the gaps between beats are large. */
			}
			color = ARDOUR_UI::config()->get_canvasvar_MeasureLineBeat();
		}

		ArdourCanvas::Coord xpos = rint(((framepos_t)(*i).frame) / (double)samples_per_pixel);

		ArdourCanvas::Line* line;

		if (!_cache.empty()) {
			line = _cache.back ();
			_cache.pop_back ();
			line->reparent (_group);
		} else {
			line = new ArdourCanvas::Line (_group);
			CANVAS_DEBUG_NAME (line, "tempo measure line");
			line->set_ignore_events (true);
		}

		/* move to 0.5 offset to ensure single pixel lines (see Cairo
		 * FAQ for info on why we do this).
		 */

		line->set_x0 (xpos + 0.5);
		line->set_x1 (xpos + 0.5);
		line->set_y0 (0.0);
		line->set_y1 (_height);
		line->set_outline_color (color);
		line->show ();
	}
}