use notification of what was changed in MidiRegion::post_set() call
[ardour.git] / libs / ardour / ardour / midi_region.h
index f8f663bc78e78fab5b383e28345e65c36fdd78a1..650c7c9b2e1329b809eece02ea30ad976a28ab37 100644 (file)
 
 class XMLNode;
 
+namespace ARDOUR {
+        namespace Properties {
+                /* this is pseudo-property: nothing has this as an actual
+                   property, but it allows us to signal changes to the
+                   MidiModel used by the MidiRegion 
+                */
+                extern PBD::PropertyDescriptor<void*> midi_data; 
+        }
+}
+
 namespace ARDOUR {
 
 class Route;
@@ -48,42 +58,46 @@ template<typename T> class MidiRingBuffer;
 class MidiRegion : public Region
 {
   public:
+       static void make_property_quarks ();
+
        ~MidiRegion();
 
+        boost::shared_ptr<MidiRegion> clone ();
+        
        boost::shared_ptr<MidiSource> midi_source (uint32_t n=0) const;
 
        /* Stub Readable interface */
-       virtual nframes_t read (Sample*, sframes_t /*pos*/, nframes_t /*cnt*/, int /*channel*/) const { return 0; }
-       virtual sframes_t readable_length() const { return length(); }
-
-       nframes_t read_at (MidiRingBuffer<nframes_t>& dst,
-                          sframes_t position,
-                          nframes_t dur,
-                          uint32_t  chan_n = 0,
-                          NoteMode  mode = Sustained,
-                          MidiStateTracker* tracker = 0) const;
-
-       nframes_t master_read_at (MidiRingBuffer<nframes_t>& dst,
-                       sframes_t position,
-                       nframes_t dur,
-                       uint32_t  chan_n = 0,
-                       NoteMode  mode = Sustained) const;
-
-       XMLNode& state (bool);
+       virtual framecnt_t read (Sample*, framepos_t /*pos*/, framecnt_t /*cnt*/, int /*channel*/) const { return 0; }
+       virtual framecnt_t readable_length() const { return length(); }
+
+       framecnt_t read_at (Evoral::EventSink<framepos_t>& dst,
+                           framepos_t position,
+                           framecnt_t dur,
+                           uint32_t  chan_n = 0,
+                           NoteMode  mode = Sustained,
+                           MidiStateTracker* tracker = 0) const;
+       
+       framepos_t master_read_at (MidiRingBuffer<framepos_t>& dst,
+                                  framepos_t position,
+                                  framecnt_t dur,
+                                  uint32_t  chan_n = 0,
+                                  NoteMode  mode = Sustained) const;
+
+       XMLNode& state ();
        int      set_state (const XMLNode&, int version);
-
+       
        int separate_by_channel (ARDOUR::Session&, std::vector< boost::shared_ptr<Region> >&) const;
 
        /* automation */
-
+       
        boost::shared_ptr<Evoral::Control>
        control(const Evoral::Parameter& id, bool create=false) {
-               return model()->data().control(id, create);
+               return model()->control(id, create);
        }
 
        virtual boost::shared_ptr<const Evoral::Control>
        control(const Evoral::Parameter& id) const {
-               return model()->data().control(id);
+               return model()->control(id);
        }
 
        /* export */
@@ -93,35 +107,48 @@ class MidiRegion : public Region
        boost::shared_ptr<MidiModel> model()             { return midi_source()->model(); }
        boost::shared_ptr<const MidiModel> model() const { return midi_source()->model(); }
 
+       void fix_negative_start ();
+
+  protected:
+       
+       virtual bool can_trim_start_before_source_start () const {
+               return true;
+       }
+
   private:
        friend class RegionFactory;
+        PBD::Property<Evoral::MusicalTime> _length_beats;
 
-       MidiRegion (boost::shared_ptr<MidiSource>, nframes_t start, nframes_t length);
-       MidiRegion (boost::shared_ptr<MidiSource>, nframes_t start, nframes_t length, const string& name, layer_t = 0, Region::Flag flags = Region::DefaultFlags);
-       MidiRegion (const SourceList &, nframes_t start, nframes_t length, const string& name, layer_t = 0, Region::Flag flags = Region::DefaultFlags);
-       MidiRegion (boost::shared_ptr<const MidiRegion>, nframes_t start, nframes_t length, const string& name, layer_t = 0, Region::Flag flags = Region::DefaultFlags);
-       MidiRegion (boost::shared_ptr<const MidiRegion>);
-       MidiRegion (boost::shared_ptr<MidiSource>, const XMLNode&);
-       MidiRegion (const SourceList &, const XMLNode&);
+       MidiRegion (const SourceList&);
+       MidiRegion (boost::shared_ptr<const MidiRegion>, frameoffset_t offset = 0, bool offset_relative = true);
 
-  private:
-       nframes_t _read_at (const SourceList&, MidiRingBuffer<nframes_t>& dst,
-                           sframes_t position,
-                           nframes_t dur,
-                           uint32_t chan_n = 0,
-                           NoteMode mode = Sustained, 
-                           MidiStateTracker* tracker = 0) const;
+       framecnt_t _read_at (const SourceList&, Evoral::EventSink<framepos_t>& dst,
+                            framepos_t position,
+                            framecnt_t dur,
+                            uint32_t chan_n = 0,
+                            NoteMode mode = Sustained, 
+                            MidiStateTracker* tracker = 0) const;
+
+       void register_properties ();
+        void post_set (const PBD::PropertyChange&);
 
        void recompute_at_start ();
        void recompute_at_end ();
 
-       void set_position_internal (nframes_t pos, bool allow_bbt_recompute);
+       void set_position_internal (framepos_t pos, bool allow_bbt_recompute);
+        void set_length_internal (framecnt_t len);
+        void update_length_beats ();
 
-       void switch_source(boost::shared_ptr<Source> source);
+       void model_changed ();
+       void model_automation_state_changed (Evoral::Parameter const &);
+        void model_contents_changed ();
 
-  protected:
+       std::set<Evoral::Parameter> _filtered_parameters; ///< parameters that we ask our source not to return when reading
+       PBD::ScopedConnection _model_connection;
+       PBD::ScopedConnection _source_connection;
+        PBD::ScopedConnection _model_contents_connection;
 
-       int set_live_state (const XMLNode&, int version, Change&, bool send);
+        double _last_length_beats;
 };
 
 } /* namespace ARDOUR */