push2: add follow action icons to display
This commit is contained in:
parent
6986c18163
commit
748fbf6fb1
@ -65,6 +65,7 @@
|
||||
#include "canvas.h"
|
||||
#include "cues.h"
|
||||
#include "knob.h"
|
||||
#include "follow_action.h"
|
||||
#include "level_meter.h"
|
||||
#include "push2.h"
|
||||
#include "utils.h"
|
||||
@ -141,6 +142,12 @@ CueLayout::CueLayout (Push2& p, Session & s, std::string const & name)
|
||||
_progress[n]->set_outline_width (10.);
|
||||
_progress[n]->set_outline (true);
|
||||
|
||||
follow_action_icon[n] = new FollowActionIcon (this);
|
||||
follow_action_icon[n]->set_font_description (fd);
|
||||
follow_action_icon[n]->set_size (25.);
|
||||
follow_action_icon[n]->set_fill_color (_p2.get_color (Push2::KnobArcBackground));
|
||||
follow_action_icon[n]->set_position (Duple (31 + (Push2Canvas::inter_button_spacing() * n), 67));
|
||||
|
||||
t = new Text (this);
|
||||
t->set_font_description (fd);
|
||||
t->set_color (_p2.get_color (Push2::ParameterName));
|
||||
@ -269,7 +276,6 @@ void
|
||||
CueLayout::button_lower (uint32_t n)
|
||||
{
|
||||
if (_p2.stop_down() || _long_stop) {
|
||||
std::cerr << "stop trigger in " << n + track_base << std::endl;
|
||||
_p2.unbang (n + track_base);
|
||||
} else {
|
||||
/* select track */
|
||||
@ -365,6 +371,7 @@ CueLayout::viewport_changed ()
|
||||
for (int n = 0; n < 8; ++n) {
|
||||
|
||||
_route[n] = _session.get_remote_nth_route (track_base+n);
|
||||
follow_action_icon[n]->reset_trigger ();
|
||||
|
||||
boost::shared_ptr<Route> r = _route[n];
|
||||
|
||||
@ -431,7 +438,6 @@ CueLayout::viewport_changed ()
|
||||
if (tp && tp->region()) {
|
||||
/* trigger in slot */
|
||||
pad->set_color (color);
|
||||
|
||||
tp->PropertyChanged.connect (_trig_connections[n * 8 + y], invalidator (*this), boost::bind (&CueLayout::trigger_property_change, this, _1, n, y), &_p2);
|
||||
|
||||
} else {
|
||||
@ -564,28 +570,22 @@ CueLayout::show_running_boxen (bool yn)
|
||||
void
|
||||
CueLayout::pad_press (int y, int x) /* fix coordinate order one day */
|
||||
{
|
||||
std::cerr << "bang on " << x + track_base << ", " << y + scene_base << std::endl;
|
||||
|
||||
if (!_route[x]) {
|
||||
std::cerr << "no route\n";
|
||||
return;
|
||||
}
|
||||
|
||||
boost::shared_ptr<TriggerBox> tb = _route[x]->triggerbox();
|
||||
|
||||
if (!tb) {
|
||||
std::cerr << "no TB\n";
|
||||
/* unpossible! */
|
||||
return;
|
||||
}
|
||||
|
||||
if (!tb->trigger (y + scene_base)->region()) {
|
||||
std::cerr << "empty slot, unbang TB\n";
|
||||
_p2.unbang (x + track_base);
|
||||
return;
|
||||
}
|
||||
|
||||
std::cerr << "bang TB\n";
|
||||
_p2.bang (x + track_base, y + scene_base);
|
||||
|
||||
|
||||
@ -671,11 +671,13 @@ CueLayout::trigger_property_change (PropertyChange const& what_changed, uint32_t
|
||||
{
|
||||
assert (_route[col]);
|
||||
|
||||
TriggerPtr trig;
|
||||
|
||||
if (what_changed.contains (Properties::running)) {
|
||||
boost::shared_ptr<TriggerBox> tb = _route[col]->triggerbox ();
|
||||
assert (tb);
|
||||
|
||||
TriggerPtr trig = tb->trigger (row);
|
||||
trig = tb->trigger (row);
|
||||
assert (trig);
|
||||
|
||||
boost::shared_ptr<Push2::Pad> pad = _p2.pad_by_xy (col, row);
|
||||
@ -684,13 +686,22 @@ CueLayout::trigger_property_change (PropertyChange const& what_changed, uint32_t
|
||||
set_pad_color_from_trigger_state (col, pad, trig);
|
||||
_p2.write (pad->state_msg());
|
||||
}
|
||||
|
||||
PropertyChange follow_stuff;
|
||||
follow_stuff.add (Properties::follow_action0);
|
||||
follow_stuff.add (Properties::follow_action1);
|
||||
follow_stuff.add (Properties::follow_action_probability);
|
||||
|
||||
if (what_changed.contains (follow_stuff)) {
|
||||
if (trig && trig->active()) {
|
||||
follow_action_icon[col]->redraw ();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
CueLayout::triggerbox_property_change (PropertyChange const& what_changed, uint32_t col)
|
||||
{
|
||||
std::cerr << "TB prop change "; what_changed.dump (std::cerr); std::cerr << std::endl;
|
||||
|
||||
assert (_route[col]);
|
||||
|
||||
if (what_changed.contains (Properties::currently_playing) || what_changed.contains (Properties::queued)) {
|
||||
@ -715,6 +726,18 @@ CueLayout::triggerbox_property_change (PropertyChange const& what_changed, uint3
|
||||
_p2.write (pad->state_msg());
|
||||
}
|
||||
|
||||
TriggerPtr playing = tb->currently_playing();
|
||||
|
||||
if (what_changed.contains (Properties::currently_playing)) {
|
||||
|
||||
if (playing) {
|
||||
follow_action_icon[col]->show ();
|
||||
follow_action_icon[col]->set_trigger (playing);
|
||||
} else {
|
||||
follow_action_icon[col]->hide ();
|
||||
follow_action_icon[col]->reset_trigger ();
|
||||
}
|
||||
}
|
||||
|
||||
if (!what_changed.contains (Properties::queued)) {
|
||||
|
||||
@ -722,9 +745,7 @@ CueLayout::triggerbox_property_change (PropertyChange const& what_changed, uint3
|
||||
* sure to disable blink on lower button
|
||||
*/
|
||||
|
||||
std::cerr << "running change on " << col << std::endl;
|
||||
|
||||
if (!tb->currently_playing()) {
|
||||
if (!playing) {
|
||||
boost::shared_ptr<Push2::Button> lower_button = _p2.lower_button_by_column (col);
|
||||
lower_button->set_color (_p2.get_color_index (_route[col]->presentation_info().color()));
|
||||
lower_button->set_state (Push2::LED::NoTransition);
|
||||
@ -755,7 +776,7 @@ CueLayout::set_pad_color_from_trigger_state (int col, boost::shared_ptr<Push2::P
|
||||
HSV hsv (_route[col]->presentation_info().color());
|
||||
hsv = hsv.shade (2.0);
|
||||
pad->set_color (_p2.get_color_index (hsv.color ()));
|
||||
pad->set_state (Push2::LED::Pulsing16th);
|
||||
pad->set_state (Push2::LED::Pulsing8th);
|
||||
|
||||
} else {
|
||||
|
||||
@ -772,3 +793,192 @@ CueLayout::set_pad_color_from_trigger_state (int col, boost::shared_ptr<Push2::P
|
||||
pad->set_state (Push2::LED::NoTransition);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
namespace ArdourCanvas {
|
||||
|
||||
FollowActionIcon::FollowActionIcon (Canvas* c)
|
||||
: Rectangle (c)
|
||||
, size (0)
|
||||
, scale (1.)
|
||||
{
|
||||
set_fill (false);
|
||||
set_outline (false);
|
||||
}
|
||||
|
||||
FollowActionIcon::FollowActionIcon (Item* i)
|
||||
: Rectangle (i)
|
||||
, size (0)
|
||||
, scale (1.)
|
||||
{
|
||||
set_fill (false);
|
||||
set_outline (false);
|
||||
}
|
||||
|
||||
void
|
||||
FollowActionIcon::set_trigger (boost::shared_ptr<Trigger> t)
|
||||
{
|
||||
begin_change ();
|
||||
trigger = t;
|
||||
set_bbox_dirty ();
|
||||
end_change ();
|
||||
}
|
||||
|
||||
void
|
||||
FollowActionIcon::reset_trigger ()
|
||||
{
|
||||
begin_change ();
|
||||
trigger.reset ();
|
||||
set_bbox_dirty ();
|
||||
end_change ();
|
||||
}
|
||||
|
||||
void
|
||||
FollowActionIcon::compute_bounding_box () const
|
||||
{
|
||||
/* a little crude, since we don't actually know how big the "?" might
|
||||
be if we use it to denote a random action.
|
||||
*/
|
||||
_bounding_box = _rect;
|
||||
set_bbox_clean ();
|
||||
}
|
||||
|
||||
void
|
||||
FollowActionIcon::set_size (double sz)
|
||||
{
|
||||
begin_change ();
|
||||
size = sz;
|
||||
set (Rect (0., 0., size * scale, size * scale));
|
||||
set_bbox_dirty ();
|
||||
end_change ();
|
||||
}
|
||||
|
||||
void
|
||||
FollowActionIcon::set_scale (double sc)
|
||||
{
|
||||
begin_change ();
|
||||
scale = sc;
|
||||
set (Rect (0., 0., size * scale, size * scale));
|
||||
set_bbox_dirty ();
|
||||
end_change ();
|
||||
}
|
||||
|
||||
void
|
||||
FollowActionIcon::set_font_description (Pango::FontDescription const & fd)
|
||||
{
|
||||
begin_change ();
|
||||
font_description = fd;
|
||||
set_bbox_dirty ();
|
||||
end_change ();
|
||||
}
|
||||
|
||||
void
|
||||
FollowActionIcon::render (ArdourCanvas::Rect const & area, Cairo::RefPtr<Cairo::Context> context) const
|
||||
{
|
||||
if (!trigger) {
|
||||
return;
|
||||
}
|
||||
|
||||
ArdourCanvas::Rect self (item_to_window (_rect));
|
||||
const ArdourCanvas::Rect draw = self.intersection (area);
|
||||
|
||||
if (!draw) {
|
||||
return;
|
||||
}
|
||||
|
||||
context->save ();
|
||||
context->translate (self.x0, self.y0);
|
||||
|
||||
/* in the case where there is a random follow-action, just put a "?" */
|
||||
if (trigger->follow_action_probability () > 0) {
|
||||
Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create (context);
|
||||
layout->set_font_description (font_description);
|
||||
layout->set_text ("?");
|
||||
int tw, th;
|
||||
layout->get_pixel_size (tw, th);
|
||||
context->move_to (size / 2, size / 2);
|
||||
context->rel_move_to (-tw / 2, -th / 2);
|
||||
layout->show_in_cairo_context (context);
|
||||
context->restore ();
|
||||
return;
|
||||
}
|
||||
|
||||
Gtkmm2ext::set_source_rgba (context, _fill_color);
|
||||
context->set_line_width (1 * scale);
|
||||
|
||||
switch (trigger->follow_action0().type) {
|
||||
case FollowAction::Stop:
|
||||
context->rectangle (6 * scale, 6 * scale, size - 12 * scale, size - 12 * scale);
|
||||
context->stroke ();
|
||||
break;
|
||||
case FollowAction::Again:
|
||||
context->arc (size / 2, size / 2, size * 0.20, 60. * (M_PI / 180.0), 2 * M_PI);
|
||||
context->stroke ();
|
||||
context->arc (size / 2 + size * 0.2, size / 2, 1.5 * scale, 0, 2 * M_PI); // arrow head
|
||||
context->fill ();
|
||||
break;
|
||||
case FollowAction::ForwardTrigger:
|
||||
context->move_to (size / 2, 3 * scale);
|
||||
context->line_to (size / 2, size - 5 * scale);
|
||||
context->stroke ();
|
||||
context->arc (size / 2, size - 5 * scale, 2 * scale, 0, 2 * M_PI); // arrow head
|
||||
context->fill ();
|
||||
break;
|
||||
case FollowAction::ReverseTrigger:
|
||||
context->move_to (size / 2, 5 * scale);
|
||||
context->line_to (size / 2, size - 3 * scale);
|
||||
context->stroke ();
|
||||
context->arc (size / 2, 5 * scale, 2 * scale, 0, 2 * M_PI); // arrow head
|
||||
context->fill ();
|
||||
break;
|
||||
case FollowAction::JumpTrigger:
|
||||
if (trigger->follow_action0().targets.count() == 1 ) { //jump to a specific row
|
||||
int cue_idx = -1;
|
||||
for (int i = 0; i < default_triggers_per_box; i++) {
|
||||
if (trigger->follow_action0().targets.test(i)) {
|
||||
cue_idx = i;
|
||||
break;
|
||||
}
|
||||
}
|
||||
Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create (context);
|
||||
layout->set_font_description (font_description);
|
||||
layout->set_text (cue_marker_name (cue_idx));
|
||||
int tw, th;
|
||||
layout->get_pixel_size (tw, th);
|
||||
context->move_to (size / 2, size / 2);
|
||||
context->rel_move_to (-tw / 2, -th / 2);
|
||||
layout->show_in_cairo_context (context);
|
||||
} else if (false) { // 'ANY' jump
|
||||
for (int i = 0; i < 6; i++) {
|
||||
Cairo::Matrix m = context->get_matrix ();
|
||||
context->translate (size / 2, size / 2);
|
||||
context->rotate (i * M_PI / 3);
|
||||
context->move_to (0, 0);
|
||||
context->line_to (0, (size / 2) - 4 * scale);
|
||||
context->stroke ();
|
||||
context->set_matrix (m);
|
||||
}
|
||||
} else { // 'OTHER' jump
|
||||
context->set_line_width (1.5 * scale);
|
||||
Gtkmm2ext::set_source_rgba (context, HSV (_fill_color).lighter (0.25).color ()); // needs to be brighter to maintain balance
|
||||
for (int i = 0; i < 6; i++) {
|
||||
Cairo::Matrix m = context->get_matrix ();
|
||||
context->translate (size / 2, size / 2);
|
||||
context->rotate (i * M_PI / 3);
|
||||
context->move_to (0, 2 * scale);
|
||||
context->line_to (0, (size / 2) - 4 * scale);
|
||||
context->stroke ();
|
||||
context->set_matrix (m);
|
||||
}
|
||||
}
|
||||
break;
|
||||
case FollowAction::None:
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
context->restore ();
|
||||
}
|
||||
|
||||
} /* namespace */
|
||||
|
||||
|
@ -21,6 +21,8 @@
|
||||
|
||||
#include <vector>
|
||||
|
||||
#include "ardour/triggerbox.h"
|
||||
|
||||
#include "layout.h"
|
||||
#include "push2.h"
|
||||
|
||||
@ -36,6 +38,7 @@ namespace ArdourCanvas {
|
||||
class Text;
|
||||
class Line;
|
||||
class Arc;
|
||||
class FollowActionIcon;
|
||||
}
|
||||
|
||||
namespace ArdourSurface {
|
||||
@ -108,6 +111,8 @@ class CueLayout : public Push2Layout
|
||||
void trigger_property_change (PBD::PropertyChange const& what_changed, uint32_t col, uint32_t row);
|
||||
|
||||
ArdourCanvas::Arc* _progress[8];
|
||||
ArdourCanvas::FollowActionIcon* follow_action_icon[8];
|
||||
|
||||
boost::shared_ptr<ARDOUR::AutomationControl> _controllables[8];
|
||||
|
||||
void viewport_changed ();
|
||||
@ -115,8 +120,9 @@ class CueLayout : public Push2Layout
|
||||
void show_state ();
|
||||
void update_clip_progress (int);
|
||||
void show_knob_function ();
|
||||
void set_pad_color_from_trigger_state (int col, boost::shared_ptr<Push2::Pad>, boost::shared_ptr<ARDOUR::Trigger>);
|
||||
void set_pad_color_from_trigger_state (int col, boost::shared_ptr<Push2::Pad>, ARDOUR::TriggerPtr);
|
||||
void show_running_boxen (bool);
|
||||
void draw_follow_icon (ARDOUR::TriggerPtr, Cairo::RefPtr<Cairo::Context> context, ARDOUR::FollowAction const & icon, float size) const;
|
||||
};
|
||||
|
||||
} /* namespace */
|
||||
|
56
libs/surfaces/push2/follow_action.h
Normal file
56
libs/surfaces/push2/follow_action.h
Normal file
@ -0,0 +1,56 @@
|
||||
/*
|
||||
* Copyright (C) 2022 Paul Davis <paul@linuxaudiosystems.com>
|
||||
* Copyright (C) 2022 Ben Loftis <ben@harrisonconsoles.com>
|
||||
*
|
||||
* 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.,
|
||||
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||
*/
|
||||
|
||||
#ifndef __ardour_push2_follow_action_h__
|
||||
#define __ardour_push2_follow_action_h__
|
||||
|
||||
/* This might be moved into libcanvas one day, but for now it is only used by
|
||||
* the push code. It has a dependency on ARDOUR::Trigger so it tricky to place
|
||||
* correctly.
|
||||
*/
|
||||
|
||||
#include "canvas/rectangle.h"
|
||||
|
||||
namespace ArdourCanvas {
|
||||
|
||||
class FollowActionIcon : public ArdourCanvas::Rectangle
|
||||
{
|
||||
public:
|
||||
FollowActionIcon (Canvas*);
|
||||
FollowActionIcon (Item*);
|
||||
|
||||
void render (Rect const &, Cairo::RefPtr<Cairo::Context>) const;
|
||||
void compute_bounding_box () const;
|
||||
|
||||
void set_font_description (Pango::FontDescription const &);
|
||||
void set_size (double size);
|
||||
void set_scale (double scale);
|
||||
void set_trigger (boost::shared_ptr<ARDOUR::Trigger> t);
|
||||
void reset_trigger ();
|
||||
|
||||
private:
|
||||
boost::shared_ptr<ARDOUR::Trigger> trigger;;
|
||||
Pango::FontDescription font_description;
|
||||
double size;
|
||||
double scale;
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
#endif /* __ardour_push2_follow_action_h__ */
|
Loading…
Reference in New Issue
Block a user