/* * Copyright (C) 2006-2016 David Robillard * Copyright (C) 2007-2017 Paul Davis * Copyright (C) 2009-2012 Carl Hetherington * Copyright (C) 2016-2017 Nick Mainsbridge * Copyright (C) 2016-2017 Robin Gareus * * 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_midi_region_h__ #define __ardour_midi_region_h__ #include #include "temporal/beats.h" #include "evoral/Range.h" #include "pbd/string_convert.h" #include "ardour/ardour.h" #include "ardour/midi_cursor.h" #include "ardour/region.h" class XMLNode; namespace ARDOUR { namespace Properties { LIBARDOUR_API extern PBD::PropertyDescriptor start_beats; LIBARDOUR_API extern PBD::PropertyDescriptor length_beats; } } namespace Evoral { template class EventSink; } namespace ARDOUR { class MidiChannelFilter; class MidiFilter; class MidiModel; class MidiSource; class MidiStateTracker; class Playlist; class Route; class Session; template class MidiRingBuffer; class LIBARDOUR_API MidiRegion : public Region { public: static void make_property_quarks (); ~MidiRegion(); bool do_export (std::string path) const; boost::shared_ptr clone (std::string path = std::string()) const; boost::shared_ptr clone (boost::shared_ptr) const; boost::shared_ptr midi_source (uint32_t n=0) const; /* Stub Readable interface */ virtual samplecnt_t read (Sample*, samplepos_t /*pos*/, samplecnt_t /*cnt*/, int /*channel*/) const { return 0; } virtual samplecnt_t readable_length() const { return length(); } samplecnt_t read_at (Evoral::EventSink& dst, samplepos_t position, samplecnt_t dur, Evoral::Range* loop_range, MidiCursor& cursor, uint32_t chan_n = 0, NoteMode mode = Sustained, MidiStateTracker* tracker = 0, MidiChannelFilter* filter = 0) const; samplecnt_t master_read_at (MidiRingBuffer& dst, samplepos_t position, samplecnt_t dur, Evoral::Range* loop_range, MidiCursor& cursor, uint32_t chan_n = 0, NoteMode mode = Sustained) const; XMLNode& state (); int set_state (const XMLNode&, int version); int separate_by_channel (std::vector< boost::shared_ptr >&) const; /* automation */ boost::shared_ptr control(const Evoral::Parameter& id, bool create=false); virtual boost::shared_ptr control(const Evoral::Parameter& id) const; /* export */ boost::shared_ptr model(); boost::shared_ptr model() const; void fix_negative_start (); double start_beats () const {return _start_beats; } double length_beats () const {return _length_beats; } void clobber_sources (boost::shared_ptr source); int render (Evoral::EventSink& dst, uint32_t chan_n, NoteMode mode, MidiChannelFilter* filter) const; protected: virtual bool can_trim_start_before_source_start () const { return true; } private: friend class RegionFactory; PBD::Property _start_beats; PBD::Property _length_beats; MidiRegion (const SourceList&); MidiRegion (boost::shared_ptr); MidiRegion (boost::shared_ptr, ARDOUR::MusicSample offset); samplecnt_t _read_at (const SourceList&, Evoral::EventSink& dst, samplepos_t position, samplecnt_t dur, Evoral::Range* loop_range, MidiCursor& cursor, uint32_t chan_n = 0, NoteMode mode = Sustained, MidiStateTracker* tracker = 0, MidiChannelFilter* filter = 0) const; void register_properties (); void post_set (const PBD::PropertyChange&); void recompute_at_start (); void recompute_at_end (); bool set_name (const std::string & str); void set_position_internal (samplepos_t pos, bool allow_bbt_recompute, const int32_t sub_num); void set_position_music_internal (double qn); void set_length_internal (samplecnt_t len, const int32_t sub_num); void set_start_internal (samplecnt_t, const int32_t sub_num); void trim_to_internal (samplepos_t position, samplecnt_t length, const int32_t sub_num); void update_length_beats (const int32_t sub_num); void model_changed (); void model_contents_changed (); void model_shifted (double qn_distance); void model_automation_state_changed (Evoral::Parameter const &); void set_start_beats_from_start_samples (); void update_after_tempo_map_change (bool send_change = true); std::set _filtered_parameters; ///< parameters that we ask our source not to return when reading PBD::ScopedConnection _model_connection; PBD::ScopedConnection _model_shift_connection; PBD::ScopedConnection _model_changed_connection; PBD::ScopedConnection _source_connection; PBD::ScopedConnection _model_contents_connection; bool _ignore_shift; }; } /* namespace ARDOUR */ #endif /* __ardour_midi_region_h__ */