midway snapshot of work done on managing Region & Source lifetimes correctly. may...
[ardour.git] / libs / pbd / pbd / stateful.h
index 5fbac11e5c7abcda4ec8ebfa9a721ae0cb68c7fa..1272a735f93d7487650ecf46795ff71a78dcc555 100644 (file)
@@ -1,5 +1,5 @@
 /*
-    Copyright (C) 2000 Paul Davis 
+    Copyright (C) 2000-2010 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
     along with this program; if not, write to the Free Software
     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
 
-    $Id: stateful.h 17 2005-09-24 19:13:41Z taybin $
 */
 
 #ifndef __pbd_stateful_h__
 #define __pbd_stateful_h__
 
 #include <string>
-#include <pbd/id.h>
+#include <list>
+#include <cassert>
+
+#include "pbd/id.h"
+#include "pbd/xml++.h"
+#include "pbd/property_basics.h"
+#include "pbd/signals.h"
 
 class XMLNode;
 
+namespace PBD {
+
+namespace sys {
+       class path;
+}
+
+class PropertyList;
+class OwnedPropertyList;
+
+/** Base class for objects with saveable and undoable state */
 class Stateful {
   public:
-       Stateful();
+       Stateful ();
        virtual ~Stateful();
 
        virtual XMLNode& get_state (void) = 0;
+       virtual int set_state (const XMLNode&, int version) = 0;
+       virtual bool set_property (const PropertyBase&);
 
-       virtual int set_state (const XMLNode&) = 0;
+       PropertyChange set_properties (const PropertyList&);
+        const OwnedPropertyList& properties() const { return *_properties; }
 
-       /* Extra XML nodes */
+       void add_property (PropertyBase& s);
+
+       /* Extra XML node: so that 3rd parties can attach state to the XMLNode
+          representing the state of this object.
+        */
 
        void add_extra_xml (XMLNode&);
        XMLNode *extra_xml (const std::string& str);
 
-       virtual void add_instant_xml (XMLNode&, const std::string& dir);
-       XMLNode *instant_xml (const std::string& str, const std::string& dir);
-        const PBD::ID& id() const { return _id; }
+       const PBD::ID& id() const { return _id; }
+
+        /* history management */
+
+       void clear_history ();
+        void diff (PropertyList&, PropertyList&, Command*) const;
+        bool changed() const;
 
+        /* create a property list from an XMLNode
+         */
+        virtual PropertyList* property_factory(const XMLNode&) const { return 0; }
+
+       /* How stateful's notify of changes to their properties
+        */
+       PBD::Signal1<void,const PropertyChange&> PropertyChanged;
+
+       static int current_state_version;
+       static int loading_state_version;
+
+       virtual void suspend_property_changes ();
+       virtual void resume_property_changes ();
+
+       void unlock_property_changes () { _no_property_changes = false; }
+       void block_property_changes () { _no_property_changes = true; }
+       
   protected:
+
+       void add_instant_xml (XMLNode&, const sys::path& directory_path);
+       XMLNode *instant_xml (const std::string& str, const sys::path& directory_path);
+       void add_properties (XMLNode &);
+       /* derived types can call this from ::set_state() (or elsewhere)
+          to get basic property setting done.
+       */
+       PropertyChange set_properties (XMLNode const &);
+       
+       /* derived classes can implement this to do cross-checking
+          of property values after either a PropertyList or XML 
+          driven property change.
+       */
+       virtual void post_set () { };
+
        XMLNode *_extra_xml;
        XMLNode *_instant_xml;
-        PBD::ID _id;
+       PBD::ID  _id;
+        int32_t  _frozen;
+        bool     _no_property_changes;
+       PBD::PropertyChange     _pending_changed;
+        Glib::Mutex _lock;
+
+       std::string _xml_node_name; ///< name of node to use for this object in XML
+       OwnedPropertyList* _properties;
+
+        virtual void send_change (const PropertyChange&);
+        /** derived classes can implement this in order to process a property change
+            within thaw() just before send_change() is called.
+        */
+        virtual void mid_thaw (const PropertyChange&) { }
+        bool property_changes_suspended() const { return g_atomic_int_get (&_frozen) > 0; }
 };
 
+} // namespace PBD
+
 #endif /* __pbd_stateful_h__ */