/* This file is part of Ingen. Copyright 2007-2016 David Robillard Ingen is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation, either version 3 of the License, or any later version. Ingen 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 Affero General Public License for details. You should have received a copy of the GNU Affero General Public License along with Ingen. If not, see . */ #ifndef INGEN_RESOURCE_HPP #define INGEN_RESOURCE_HPP #include #include #include "ingen/Properties.hpp" #include "ingen/URIs.hpp" #include "ingen/ingen.h" #include "raul/Deletable.hpp" #include "raul/URI.hpp" namespace Ingen { class Atom; /** A resource with a URI described by properties. * * This is the base class for most things in Ingen, including graphs, blocks, * ports, and the engine itself. * * @ingroup Ingen */ class INGEN_API Resource : public Raul::Deletable { public: using Graph = Property::Graph; Resource(const URIs& uris, const Raul::URI& uri) : _uris(uris) , _uri(uri) {} Resource& operator=(const Resource& rhs) { assert(&rhs._uris == &_uris); if (&rhs != this) { _uri = rhs._uri; _properties = rhs._properties; } return *this; } static Raul::URI graph_to_uri(Graph g) { switch (g) { case Graph::EXTERNAL: return Raul::URI(INGEN_NS "externalContext"); case Graph::INTERNAL: return Raul::URI(INGEN_NS "internalContext"); default: return Raul::URI(INGEN_NS "defaultContext"); } } static Graph uri_to_graph(const Raul::URI& uri) { if (uri == INGEN_NS "externalContext") { return Graph::EXTERNAL; } else if (uri == INGEN_NS "internalContext") { return Graph::INTERNAL; } return Graph::DEFAULT; } ~Resource() override {} /** Get a single property value. * * This is only useful for properties with a single value. If the * requested property has several values, the first will be returned. */ virtual const Atom& get_property(const Raul::URI& uri) const; /** Set (replace) a property value. * * This will first erase any properties with the given `uri`, so after * this call exactly one property with predicate `uri` will be set. */ virtual const Atom& set_property( const Raul::URI& uri, const Atom& value, Graph ctx = Graph::DEFAULT); /** Set (replace) a property value. * * This will first erase any properties with the given `uri`, so after * this call exactly one property with predicate `uri` will be set. */ virtual const Atom& set_property( const Raul::URI& uri, const URIs::Quark& value, Graph ctx = Graph::DEFAULT); /** Add a property value. * * This will not remove any existing values, so if properties with * predicate `uri` and values other than `value` exist, this will result * in multiple values for the property. * * @return True iff a new property was added. */ virtual bool add_property(const Raul::URI& uri, const Atom& value, Graph ctx = Graph::DEFAULT); /** Remove a property. * * If `value` is patch:wildcard then any property with `uri` for a * predicate will be removed. */ virtual void remove_property(const Raul::URI& uri, const Atom& value); /** Remove a property. * * If `value` is patch:wildcard then any property with `uri` for a * predicate will be removed. */ virtual void remove_property(const Raul::URI& uri, const URIs::Quark& value); /** Return true iff a property is set with a specific value. */ virtual bool has_property(const Raul::URI& uri, const Atom& value) const; /** Return true iff a property is set with a specific value. */ virtual bool has_property(const Raul::URI& uri, const URIs::Quark& value) const; /** Set (replace) several properties at once. * * This will erase all properties with keys in `p`, though multiple values * for one property may exist in `p` will all be set (unlike simply * calling set_property in a loop which would only set one value). */ void set_properties(const Properties& p); /** Add several properties at once. */ void add_properties(const Properties& p); /** Remove several properties at once. * * This removes all matching properties (both key and value), or all * properties with a matching key if the value in `p` is patch:wildcard. */ void remove_properties(const Properties& p); /** Hook called whenever a property is added. * * This can be used by derived classes to implement special behaviour for * particular properties (e.g. ingen:value for ports). */ virtual void on_property(const Raul::URI& uri, const Atom& value) {} /** Hook called whenever a property value is removed. * * If all values for a key are removed, then value will be the wildcard. * * This can be used by derived classes to implement special behaviour for * particular properties (e.g. ingen:value for ports). */ virtual void on_property_removed(const Raul::URI& uri, const Atom& value) {} /** Get the ingen type from a set of Properties. * * If some coherent ingen type is found, true is returned and the appropriate * output parameter set to true. Otherwise false is returned. */ static bool type(const URIs& uris, const Properties& properties, bool& graph, bool& block, bool& port, bool& is_output); virtual void set_uri(const Raul::URI& uri) { _uri = uri; } /** Get all the properties with a given context. */ Properties properties(Resource::Graph ctx) const; const URIs& uris() const { return _uris; } const Raul::URI& uri() const { return _uri; } const Properties& properties() const { return _properties; } Properties& properties() { return _properties; } protected: const Atom& set_property(const Raul::URI& uri, const Atom& value) const; const URIs& _uris; private: Raul::URI _uri; mutable Properties _properties; }; } // namespace Ingen #endif // INGEN_RESOURCE_HPP