From d91eb4992c8ab28a69db579d358592acfad2c432 Mon Sep 17 00:00:00 2001
From: Rainer Kartmann <rainer.kartmann@student.kit.edu>
Date: Tue, 2 Apr 2019 12:19:34 +0200
Subject: [PATCH] Added MJCF library files

---
 VirtualRobot/CMakeLists.txt               |  43 +++
 VirtualRobot/MJCF/Document.cpp            |  96 ++++++
 VirtualRobot/MJCF/Document.h              | 210 ++++++++++++
 VirtualRobot/MJCF/elements/Attribute.cpp  |   1 +
 VirtualRobot/MJCF/elements/Attribute.h    | 296 +++++++++++++++++
 VirtualRobot/MJCF/elements/Element.cpp    |   2 +
 VirtualRobot/MJCF/elements/Element.h      | 373 ++++++++++++++++++++++
 VirtualRobot/MJCF/elements/actuator.cpp   |  84 +++++
 VirtualRobot/MJCF/elements/actuator.h     | 143 +++++++++
 VirtualRobot/MJCF/elements/asset.cpp      |  55 ++++
 VirtualRobot/MJCF/elements/asset.h        |  93 ++++++
 VirtualRobot/MJCF/elements/body.cpp       | 155 +++++++++
 VirtualRobot/MJCF/elements/body.h         | 163 ++++++++++
 VirtualRobot/MJCF/elements/compiler.cpp   |   9 +
 VirtualRobot/MJCF/elements/compiler.h     |  14 +
 VirtualRobot/MJCF/elements/contact.cpp    |  49 +++
 VirtualRobot/MJCF/elements/contact.h      |  62 ++++
 VirtualRobot/MJCF/elements/custom.cpp     |   6 +
 VirtualRobot/MJCF/elements/custom.h       |  11 +
 VirtualRobot/MJCF/elements/default.cpp    |  31 ++
 VirtualRobot/MJCF/elements/default.h      |  57 ++++
 VirtualRobot/MJCF/elements/elements.cpp   |  11 +
 VirtualRobot/MJCF/elements/elements.h     |  88 +++++
 VirtualRobot/MJCF/elements/equality.cpp   |  48 +++
 VirtualRobot/MJCF/elements/equality.h     |  96 ++++++
 VirtualRobot/MJCF/elements/exceptions.cpp |  28 ++
 VirtualRobot/MJCF/elements/exceptions.h   |  42 +++
 VirtualRobot/MJCF/elements/has_member.hpp |  34 ++
 VirtualRobot/MJCF/elements/keyframe.cpp   |   9 +
 VirtualRobot/MJCF/elements/keyframe.h     |  15 +
 VirtualRobot/MJCF/elements/mjcf_utils.cpp |  82 +++++
 VirtualRobot/MJCF/elements/mjcf_utils.h   | 213 ++++++++++++
 VirtualRobot/MJCF/elements/option.cpp     |  16 +
 VirtualRobot/MJCF/elements/option.h       |  71 ++++
 VirtualRobot/MJCF/elements/sensor.cpp     |   9 +
 VirtualRobot/MJCF/elements/sensor.h       |  14 +
 VirtualRobot/MJCF/elements/size.cpp       |   8 +
 VirtualRobot/MJCF/elements/size.h         |  16 +
 VirtualRobot/MJCF/elements/statistic.cpp  |   9 +
 VirtualRobot/MJCF/elements/statistic.h    |  15 +
 VirtualRobot/MJCF/elements/tendon.cpp     |  10 +
 VirtualRobot/MJCF/elements/tendon.h       |  16 +
 VirtualRobot/MJCF/elements/visual.cpp     |   9 +
 VirtualRobot/MJCF/elements/visual.h       |  14 +
 44 files changed, 2826 insertions(+)
 create mode 100644 VirtualRobot/MJCF/Document.cpp
 create mode 100644 VirtualRobot/MJCF/Document.h
 create mode 100644 VirtualRobot/MJCF/elements/Attribute.cpp
 create mode 100644 VirtualRobot/MJCF/elements/Attribute.h
 create mode 100644 VirtualRobot/MJCF/elements/Element.cpp
 create mode 100644 VirtualRobot/MJCF/elements/Element.h
 create mode 100644 VirtualRobot/MJCF/elements/actuator.cpp
 create mode 100644 VirtualRobot/MJCF/elements/actuator.h
 create mode 100644 VirtualRobot/MJCF/elements/asset.cpp
 create mode 100644 VirtualRobot/MJCF/elements/asset.h
 create mode 100644 VirtualRobot/MJCF/elements/body.cpp
 create mode 100644 VirtualRobot/MJCF/elements/body.h
 create mode 100644 VirtualRobot/MJCF/elements/compiler.cpp
 create mode 100644 VirtualRobot/MJCF/elements/compiler.h
 create mode 100644 VirtualRobot/MJCF/elements/contact.cpp
 create mode 100644 VirtualRobot/MJCF/elements/contact.h
 create mode 100644 VirtualRobot/MJCF/elements/custom.cpp
 create mode 100644 VirtualRobot/MJCF/elements/custom.h
 create mode 100644 VirtualRobot/MJCF/elements/default.cpp
 create mode 100644 VirtualRobot/MJCF/elements/default.h
 create mode 100644 VirtualRobot/MJCF/elements/elements.cpp
 create mode 100644 VirtualRobot/MJCF/elements/elements.h
 create mode 100644 VirtualRobot/MJCF/elements/equality.cpp
 create mode 100644 VirtualRobot/MJCF/elements/equality.h
 create mode 100644 VirtualRobot/MJCF/elements/exceptions.cpp
 create mode 100644 VirtualRobot/MJCF/elements/exceptions.h
 create mode 100644 VirtualRobot/MJCF/elements/has_member.hpp
 create mode 100644 VirtualRobot/MJCF/elements/keyframe.cpp
 create mode 100644 VirtualRobot/MJCF/elements/keyframe.h
 create mode 100644 VirtualRobot/MJCF/elements/mjcf_utils.cpp
 create mode 100644 VirtualRobot/MJCF/elements/mjcf_utils.h
 create mode 100644 VirtualRobot/MJCF/elements/option.cpp
 create mode 100644 VirtualRobot/MJCF/elements/option.h
 create mode 100644 VirtualRobot/MJCF/elements/sensor.cpp
 create mode 100644 VirtualRobot/MJCF/elements/sensor.h
 create mode 100644 VirtualRobot/MJCF/elements/size.cpp
 create mode 100644 VirtualRobot/MJCF/elements/size.h
 create mode 100644 VirtualRobot/MJCF/elements/statistic.cpp
 create mode 100644 VirtualRobot/MJCF/elements/statistic.h
 create mode 100644 VirtualRobot/MJCF/elements/tendon.cpp
 create mode 100644 VirtualRobot/MJCF/elements/tendon.h
 create mode 100644 VirtualRobot/MJCF/elements/visual.cpp
 create mode 100644 VirtualRobot/MJCF/elements/visual.h

diff --git a/VirtualRobot/CMakeLists.txt b/VirtualRobot/CMakeLists.txt
index 52d4638a6..22ef15278 100644
--- a/VirtualRobot/CMakeLists.txt
+++ b/VirtualRobot/CMakeLists.txt
@@ -307,6 +307,27 @@ SET(SOURCES
     math/statistics/BoxPlot.cpp
     math/statistics/Histogram.cpp
     
+    MJCF/Document.cpp
+    MJCF/elements/Attribute.cpp
+    MJCF/elements/Element.cpp
+    MJCF/elements/actuator.cpp
+    MJCF/elements/asset.cpp
+    MJCF/elements/body.cpp
+    MJCF/elements/contact.cpp
+    MJCF/elements/custom.cpp
+    MJCF/elements/default.cpp
+    MJCF/elements/elements.cpp
+    MJCF/elements/equality.cpp
+    MJCF/elements/exceptions.cpp
+    MJCF/elements/keyframe.cpp
+    MJCF/elements/mjcf_utils.cpp
+    MJCF/elements/option.cpp
+    MJCF/elements/sensor.cpp
+    MJCF/elements/size.cpp
+    MJCF/elements/statistic.cpp
+    MJCF/elements/tendon.cpp
+    MJCF/elements/visual.cpp
+    
     Nodes/CameraSensor.cpp
     Nodes/CameraSensorFactory.cpp
     Nodes/ContactSensor.cpp
@@ -490,6 +511,28 @@ SET(INCLUDES
     math/statistics/BoxPlot.h
     math/statistics/Histogram.h
     
+    MJCF/Document.h
+    MJCF/elements/Attribute.h
+    MJCF/elements/Element.h
+    MJCF/elements/actuator.h
+    MJCF/elements/asset.h
+    MJCF/elements/body.h
+    MJCF/elements/contact.h
+    MJCF/elements/custom.h
+    MJCF/elements/default.h
+    MJCF/elements/elements.h
+    MJCF/elements/equality.h
+    MJCF/elements/exceptions.h
+    MJCF/elements/keyframe.h
+    MJCF/elements/mjcf_utils.h
+    MJCF/elements/option.h
+    MJCF/elements/sensor.h
+    MJCF/elements/size.h
+    MJCF/elements/statistic.h
+    MJCF/elements/tendon.h
+    MJCF/elements/visual.h
+    MJCF/elements/has_member.hpp
+    
     Nodes/CameraSensor.h
     Nodes/CameraSensorFactory.h
     Nodes/ConditionedLock.h
diff --git a/VirtualRobot/MJCF/Document.cpp b/VirtualRobot/MJCF/Document.cpp
new file mode 100644
index 000000000..e2616200f
--- /dev/null
+++ b/VirtualRobot/MJCF/Document.cpp
@@ -0,0 +1,96 @@
+#include "Document.h"
+
+#include "elements/exceptions.h"
+
+
+using namespace mjcf;
+
+
+Document::Document()
+{
+    // create root element
+    tinyxml2::XMLElement* xml = doc.NewElement(MujocoRoot::tag.c_str());
+    doc.InsertEndChild(xml);
+    root.reset(new MujocoRoot(this, xml));
+}
+
+void Document::loadFile(const std::string& fileName)
+{
+    tinyxml2::XMLError error = doc.LoadFile(fileName.c_str());
+    if (error != tinyxml2::XML_SUCCESS)
+    {
+        throw MjcfIOError(doc.ErrorStr());
+    }
+}
+
+void Document::saveFile(const std::string& fileName)
+{
+    tinyxml2::XMLError error = doc.SaveFile(fileName.c_str());
+    if (error != tinyxml2::XML_SUCCESS)
+    {
+        throw MjcfIOError(doc.ErrorStr());
+    }
+}
+
+void Document::deepCopyFrom(const Document& source)
+{
+    source.doc.DeepCopy(&this->doc);
+}
+
+
+
+std::string Document::getModelName() const
+{
+    if (root->isAttributeSet("model"))
+    {
+        return root->getAttribute("model");
+    }
+    else
+    {
+        return "";
+    }
+}
+void Document::setModelName(const std::string& name)
+{
+    root->setAttribute("model", name);
+}
+
+Include Document::addInclude(const std::string& filename)
+{
+    Include include = root->addChild<Include>();
+    include.file = filename;
+    return include;
+}
+
+void Document::setNewElementClass(const std::string& className, bool excludeBody)
+{
+    this->newElementClass = className;
+    this->newElementClassExcludeBody = excludeBody;
+    
+    if (!className.empty() && !default_().hasChild<DefaultClass>("class", className))
+    {
+        default_().addClass(className);
+    }
+}
+
+
+float Document::getFloatCompPrecision() const
+{
+    return floatCompPrecision;
+}
+
+void Document::setFloatCompPrecision(float value)
+{
+    floatCompPrecision = value;
+}
+
+float Document::getDummyMass() const
+{
+    return dummyMass;
+}
+
+void Document::setDummyMass(float value)
+{
+    dummyMass = value;
+}
+
diff --git a/VirtualRobot/MJCF/Document.h b/VirtualRobot/MJCF/Document.h
new file mode 100644
index 000000000..b27de93e9
--- /dev/null
+++ b/VirtualRobot/MJCF/Document.h
@@ -0,0 +1,210 @@
+#pragma once
+
+#include <memory>
+#include <set>
+
+#include <Eigen/Eigen>
+
+#include <VirtualRobot/Util/xml/tinyxml2.h>
+
+#include "elements/elements.h"
+
+
+namespace mjcf
+{
+
+    /**
+     * @brief A MJCF (Mujoco XML) document.
+     */
+    class Document
+    {
+        
+    public:
+
+        /// Constructor.
+        Document();
+        
+        /// Set the precision for float comparison.
+        float getFloatCompPrecision() const;
+        /// Set the precision for float comparison (used e.g. when comparing
+        /// to zero or identity).
+        void setFloatCompPrecision(float value);
+        /// Get the mass of dummy inertial elements.
+        float getDummyMass() const;
+        /// Set the mass of dummy inertial elements.
+        void setDummyMass(float value);
+        
+        /// Load from an MJCF file.
+        void loadFile(const std::string& fileName);
+        /// Save to an MJCF file.
+        void saveFile(const std::string& fileName);
+        
+        /// Make a deep copy of source to this.
+        void deepCopyFrom(const Document& source);
+        
+
+        std::string getModelName() const;
+        /// Set the name of the Mujoco model.
+        void setModelName(const std::string& name);
+
+        
+        // Section elements (children of top-level 'mujoco' element).
+        CompilerSection  compiler()  { return section<CompilerSection>();   }
+        OptionSection    option()    { return section<OptionSection>();     }
+        SizeSection      size()      { return section<SizeSection>();       }
+        VisualSection    visual()    { return section<VisualSection>();     }
+        StatisticSection statistic() { return section<StatisticSection>();  }
+        DefaultSection   default_()  { return section<DefaultSection>();    }
+        AssetSection     asset()     { return section<AssetSection>();      }
+        Worldbody        worldbody() { return section<Worldbody>();         }
+        ContactSection   contact()   { return section<ContactSection>();    }
+        EqualitySection  equality()  { return section<EqualitySection>();   }
+        TendonSection    tendon()    { return section<TendonSection>();     }
+        ActuatorSection  actuator()  { return section<ActuatorSection>();   }
+        SensorSection    sensor()    { return section<SensorSection>();     } 
+        KeyframeSection  keyframe()  { return section<KeyframeSection>();   }
+
+        Include addInclude(const std::string& filename);
+        
+        /**
+         * @brief Set the class attribute of all new applicable elements
+         * (after calling this method). Pass an empty string to disable class attributes.
+         * @param excludeBody If true (default), the class will not be set on
+         *  new body elements and its children (inertial, joint, ...).
+         *  They should be the childclass attribute of the robot root body.
+         */
+        void setNewElementClass(const std::string& className, bool excludeBody = true);
+        
+        
+        template <class Derived>
+        friend class Element;
+
+        
+    private:
+
+        // METHODS
+
+        /// Add a new element to a parent.
+        /// @param className If not empty, set the class attribute of the new element.
+        /// @param first If true, will be inserted as first, otherweise at end (default)
+        template <class ElementD, class ParentD>
+        ElementD createElement(Element<ParentD> parent, const std::string& className = "", bool first = false);
+        
+        /// Return the first child element of parent with the given element name.
+        /// If it does not exist, create it.
+        template <class ElementD, class ParentD>
+        ElementD getOrCreateElement(Element<ParentD>& parent);
+
+        /// Gets the section element (child of root element). If it does not exist, it is created.
+        template <class SectionD>
+        SectionD section();
+
+        
+    private:
+        
+        // ATTRIBUTES
+        
+        /// The document.
+        tinyxml2::XMLDocument doc;
+        
+        /// The "mujoco" root element.
+        std::unique_ptr<MujocoRoot> root;
+
+        
+        /// Precision when comparing floats (e.g. with zero).
+        float floatCompPrecision = 1e-6f;
+        /// Mass used for dummy inertial elements.
+        float dummyMass = 0.0001f;
+        
+        /// The class added to new elements, if applicable.
+        std::string newElementClass = "";
+        /// Indicate whether body elements and their children shall be
+        /// exluded from setting the class attribute.
+        bool newElementClassExcludeBody = true;
+
+        
+    };
+    
+    using DocumentPtr = std::unique_ptr<Document>;
+    
+
+    
+#include "elements/has_member.hpp"
+    
+    define_has_member(class_);
+    
+    template <class ElementD, class ParentD>
+    ElementD Document::createElement(Element<ParentD> parent, const std::string& className, bool first)
+    {
+        ElementD element(this, doc.NewElement(ElementD::Derived::tag.c_str()));
+
+        auto applyNewElementClass = [this]()
+        {
+            bool isSet = !newElementClass.empty();
+            bool hasClass = has_member(ElementD, class_);
+            bool excludeBecauseBody = std::is_same<ParentD, Body>() && newElementClassExcludeBody;
+                    // body itself does not have a class attribute
+            
+            bool inDefaultClass = std::is_same<ParentD, DefaultClass>();
+            
+            return isSet                     // must not be empty
+                    && hasClass              // must have class attribute
+                    && !excludeBecauseBody   // not excluded because of body exclusion
+                    && !inDefaultClass;      // not part of default class
+        };
+                
+        
+        if (!className.empty())
+        {
+            element.setAttribute("class", className);
+        }
+        else if (applyNewElementClass())
+        {
+            element.setAttribute("class", newElementClass);
+        }
+
+        if (first)
+        {
+            parent.insertFirstChild(element);
+        }
+        else
+        {
+            parent.insertEndChild(element);
+        }
+        return element;
+    }
+
+// cleanup macros by has_member.hpp
+#undef define_has_member
+#undef has_member
+    
+    
+    template <class ElementD, class ParentD>
+    ElementD Document::getOrCreateElement(Element<ParentD>& parent)
+    {
+        ElementD element = parent.template firstChild<ElementD>();
+        
+        if (!element)
+        {
+            element = createElement<ElementD>(parent);
+        }
+        
+        return element;
+    }
+    
+    
+    template<class SectionT>
+    SectionT Document::section()
+    {
+        return getOrCreateElement<SectionT>(*root);
+    }
+    
+    // Implementation of Element::createNewElement, which depends on the 
+    // definition of Document.
+    template <class D>
+    template <class ParentD, class ElementD>
+    ElementD Element<D>::createElement(Element<ParentD> parent, const std::string& className)
+    {
+        return _document->createElement<ElementD, ParentD>(parent, className);
+    }
+}
diff --git a/VirtualRobot/MJCF/elements/Attribute.cpp b/VirtualRobot/MJCF/elements/Attribute.cpp
new file mode 100644
index 000000000..0491e59f9
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/Attribute.cpp
@@ -0,0 +1 @@
+#include "Attribute.h"
diff --git a/VirtualRobot/MJCF/elements/Attribute.h b/VirtualRobot/MJCF/elements/Attribute.h
new file mode 100644
index 000000000..252684a48
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/Attribute.h
@@ -0,0 +1,296 @@
+#pragma once
+
+#include "Element.h"
+
+
+namespace mjcf
+{
+
+    /**
+     * @class Base class for an attribute of an Element<Derived>.
+     * Stores a pointer to the owning element.
+     */
+    template <class Derived>
+    class AttributeBase
+    {
+    public:
+        
+        AttributeBase() : _owner(nullptr) 
+        {}
+        
+        /// Constructor.
+        /// @param owner the owning element (Element instance of which this is a member).
+        AttributeBase(Element<Derived>& owner) : _owner(&owner) 
+        {}
+
+        
+        /* _owner must always point to the element this was constructed with.
+         * We therefore delete copy and move constructors (otherwise, the owner
+         * will be unavailable. In addition, _owner is constant, so it is not
+         * changed by copy or move operators. */
+        
+        /// Deleted copy constructor.
+        AttributeBase(const AttributeBase<Derived>& other) = delete;
+        /// Deleted move constructor.
+        AttributeBase(AttributeBase<Derived>&& other) = delete;
+        
+        
+    protected:
+        /// Get the owning element.
+        Element<Derived>& owner() { return *_owner; }
+        const Element<Derived>& owner() const { return *_owner; }
+        
+    private:
+        /// The owning element (Element instance of which this is a member).
+        /// It must not change after construction, and thus is const-qualified.
+        Element<Derived>* const _owner;
+        
+    };
+
+    
+    
+    /**
+     * @class An attribute of an Element<Derived> of type AttrT.
+     */
+    template <class Derived, typename AttrT>
+    class Attribute : public AttributeBase<Derived>
+    {
+    public:
+        
+        /// Constructor.
+        Attribute(Element<Derived>& element, const std::string& name)
+            : AttributeBase<Derived>(element), name(name)
+        {}
+        
+        /// Get the attribute value (or the default value if the attribute does not exist).
+        virtual AttrT get() const = 0;
+        /// Set the attribute value.
+        void set(const AttrT& value)
+        {
+            VR_ASSERT(this->owner());
+            this->owner().template setAttribute<AttrT>(name, value);
+        }
+        
+        /// Conversion operator to AttrT for reading.
+        operator AttrT()
+        {
+            return this->get();
+        }
+        
+        /// Copy assignment operator for writing an attribute value.
+        Attribute& operator=(const AttrT& value)
+        {
+            this->set(value);
+            return *this;
+        }
+        
+        /// Move assignment operator for writing an attribute value.
+        Attribute& operator=(AttrT&& value)
+        {
+            this->set(value);
+            return *this;
+        }
+        
+        /// Indicate whether the attribute is required (implies no default).
+        virtual bool isRequired() const = 0;
+        /// Indicate whether the attribute is optional or has a default.
+        bool isOptional() const { return !isRequired(); }
+        /// Indicate whether the attribute has a default (implies optional).
+        virtual bool hasDefault() const = 0;
+        
+        
+    protected:
+        
+        std::string name;   ///< The attribute name.
+        
+    };
+
+    
+    
+    /**
+     * @class An attribute of an Element<Derived> of type AttrT
+     * that is required or optional and has no default value.
+     */
+    template <class Derived, typename AttrT>
+    class AttributeNoDef : public Attribute<Derived, AttrT>
+    {
+        using Base = Attribute<Derived, AttrT>;
+    public:
+        
+        /// Construct an required or optional attribute without a default.
+        AttributeNoDef(Element<Derived>& element, const std::string& name, bool required)
+            : Base(element, name), required(required)
+        {}
+        
+        AttrT get() const override
+        {
+            VR_ASSERT(this->owner());
+            return this->owner().template getAttribute<AttrT>(this->name);
+        }
+        
+        bool isRequired() const override { return  required; }
+        bool hasDefault() const override { return  false; }
+        
+        using Base::operator=;
+        
+        
+    private:
+        
+        bool required;      ///< Whether the attribute is optional.
+    };
+    
+    
+    
+    /**
+     * @class An attribute of an Element<Derived> of type AttrT
+     * that is optional and has a default value.
+     */
+    template <class Derived, typename AttrT>
+    class AttributeDef : public Attribute<Derived, AttrT>
+    {
+        using Base = Attribute<Derived, AttrT>;
+        
+    public:
+        
+        /// Construct an optional attribute with a default.
+        AttributeDef(Element<Derived>& element, const std::string& name, const AttrT& defaultValue)
+            : Base(element, name), defaultValue(defaultValue)
+        {}
+        
+        AttrT get() const override
+        {
+            VR_ASSERT(this->owner());
+            return this->owner().template getAttribute<AttrT>(this->name, this->defaultValue);
+        }
+        
+        bool isRequired() const override { return  false; }
+        bool hasDefault() const override { return  true; }
+        
+        using Base::operator=;
+        
+        
+    private:
+        
+        AttrT defaultValue; ///< The default value.
+    };
+    
+
+// Macro for a requried or optional Attribute in an Element<Derived> of type AttrT
+// with potentially different memberName and attrName
+#define mjcf_AttributeFullNoDef(Derived, AttrT, memberName, attrName, required) \
+    AttributeNoDef<Derived, AttrT> memberName {*this, attrName, required}
+    
+// Macro for an optional Attribute in an Element<Derived> of type AttrT
+// with potentially different memberName and attrName.
+#define mjcf_AttributeFullDef(Derived, AttrT, memberName, attrName, defaultValue) \
+    AttributeDef<Derived, AttrT> memberName {*this, attrName, defaultValue}
+
+    
+// A requried Attribute in an Element<Derived> of type AttrT.
+#define mjcf_AttributeReq(Derived, AttrT, name) \
+    mjcf_AttributeFullNoDef(Derived, AttrT, name, #name, true)
+
+#define mjcf_AttributeOpt(Derived, AttrT, name) \
+    mjcf_AttributeFullNoDef(Derived, AttrT, name, #name, false)
+    
+// An optional Attribute in an Element<Derived> of type AttrT.
+#define mjcf_AttributeDef(Derived, AttrT, name, defaultValue) \
+    mjcf_AttributeFullDef(Derived, AttrT, name, #name, defaultValue)
+
+    
+// Attributes with a name that is a keyword (e.g. "class").
+#define mjcf_AttributeKwReq(Derived, AttrT, name) \
+    mjcf_AttributeFullNoDef(Derived, AttrT, name ## _, #name, true)
+
+#define mjcf_AttributeKwOpt(Derived, AttrT, name) \
+    mjcf_AttributeFullNoDef(Derived, AttrT, name ## _, #name, false)
+    
+#define mjcf_AttributeKwDef(Derived, AttrT, name, defaultValue) \
+    mjcf_AttributeFullDef(Derived, AttrT, name ## _, #name, defaultValue)
+
+
+// bool shorthands
+#define mjcf_BoolAttributeReq(Derived, name) \
+    mjcf_AttributeReq(Derived, bool, name)
+#define mjcf_BoolAttributeOpt(Derived, name) \
+    mjcf_AttributeOpt(Derived, bool, name)
+#define mjcf_BoolAttributeDef(Derived, name, defaultValue) \
+    mjcf_AttributeDef(Derived, bool, name, defaultValue)
+    
+// int shorthands
+#define mjcf_IntAttributeReq(Derived, name) \
+    mjcf_AttributeReq(Derived, int, name)
+#define mjcf_IntAttributeOpt(Derived, name) \
+    mjcf_AttributeOpt(Derived, int, name)
+#define mjcf_IntAttributeDef(Derived, name, defaultValue) \
+    mjcf_AttributeDef(Derived, int, name, defaultValue)
+    
+    
+// float shorthands
+#define mjcf_FloatAttributeReq(Derived, name) \
+    mjcf_AttributeReq(Derived, float, name)
+#define mjcf_FloatAttributeOpt(Derived, name) \
+    mjcf_AttributeOpt(Derived, float, name)
+#define mjcf_FloatAttributeDef(Derived, name, defaultValue) \
+    mjcf_AttributeDef(Derived, float, name, defaultValue)
+    
+// string shorthands
+#define mjcf_StringAttributeReq(Derived, memberName) \
+    mjcf_AttributeReq(Derived, std::string, memberName)
+#define mjcf_StringAttributeOpt(Derived, memberName) \
+    mjcf_AttributeOpt(Derived, std::string, memberName)
+#define mjcf_StringAttributeDef(Derived, memberName, defaultValue) \
+    mjcf_AttributeDef(Derived, std::string, memberName, defaultValue)
+    
+#define mjcf_StringAttributeDefEmpty(Derived, memberName, defaultValue) \
+    mjcf_StringAttributeDef(Derived, std::string, memberName, "")
+
+
+// vector shorthands
+    
+#define mjcf_Vector2fAttributeDef(Derived, name, defaultValue) \
+    mjcf_AttributeDef(Derived, Eigen::Vector2f, name, defaultValue)
+    
+#define mjcf_Vector3fAttributeReq(Derived, name) \
+    mjcf_AttributeReq(Derived, Eigen::Vector3f, name)
+#define mjcf_Vector3fAttributeOpt(Derived, name) \
+    mjcf_AttributeOpt(Derived, Eigen::Vector3f, name)
+#define mjcf_Vector3fAttributeDef(Derived, name, defaultValue) \
+    mjcf_AttributeDef(Derived, Eigen::Vector3f, name, defaultValue)
+    
+    
+// name
+#define mjcf_NameAttribute(Derived) \
+    mjcf_StringAttributeOpt(Derived, name)
+    
+// class
+#define mjcf_ClassAttribute(Derived) \
+    mjcf_AttributeKwReq(Derived, std::string, class)
+
+// group
+#define mjcf_GroupAttribute(Derived) \
+    mjcf_IntAttributeDef(Derived, group, 0)
+
+
+// RGB
+#define mjcf_RgbAttributeDef(Derived, name, defaultValue) \
+    mjcf_Vector3fAttributeDef(Derived, name, defaultValue)
+    
+#define mjcf_RgbaAttributeDef(Derived, name, defaultValue) \
+    mjcf_AttributeDef(Derived, Eigen::Vector4f, name, defaultValue)
+
+    
+// pos
+#define mjcf_PosAttribute(Derived) \
+    mjcf_Vector3fAttributeDef(Derived, pos, Eigen::Vector3f::Zero())
+
+// quat
+#define mjcf_QuatAttribute(Derived) \
+    mjcf_AttributeDef(Derived, Eigen::Quaternionf, quat, Eigen::Quaternionf::Identity())
+
+// pose = pos + quat
+#define mjcf_PoseAttributes(Derived) \
+    mjcf_PosAttribute(Derived);\
+    mjcf_QuatAttribute(Derived)
+
+} 
diff --git a/VirtualRobot/MJCF/elements/Element.cpp b/VirtualRobot/MJCF/elements/Element.cpp
new file mode 100644
index 000000000..114513eaf
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/Element.cpp
@@ -0,0 +1,2 @@
+#include "Element.h"
+
diff --git a/VirtualRobot/MJCF/elements/Element.h b/VirtualRobot/MJCF/elements/Element.h
new file mode 100644
index 000000000..355791e30
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/Element.h
@@ -0,0 +1,373 @@
+#pragma once
+
+#include <VirtualRobot.h>
+#include <VirtualRobot/Util/xml/tinyxml2.h>
+
+#include "exceptions.h"
+#include "mjcf_utils.h"
+
+
+namespace mjcf
+{
+
+    // Forward declaration.
+    class Document;
+    
+    
+    template <class _Derived>
+    class Element
+    {
+    public:
+        
+        using Derived = _Derived;
+        
+        template <class OtherDerived>
+        friend class Element;
+        
+        
+    public:
+
+        // CONSTRUCTORS
+        
+        /// Constructor.
+        Element(Document* _document, tinyxml2::XMLElement* elem);
+        
+        
+        /// Indicate whether this is an element of the given type.
+        template <class OtherT>
+        bool isElement() const { return std::is_same<Derived, typename OtherT::Derived>(); }
+
+        
+        // ATTRIBUTES
+        
+        /// Indicate whether this element has an attribute with the given name.
+        bool isAttributeSet(const std::string& attrName) const;
+
+        /// Get attribute value.
+        template <typename AttrT = std::string>
+        AttrT getAttribute(const std::string& name) const;
+        /// Get the attribute value, or the default value if it does not exist.
+        template <typename AttrT = std::string>
+        AttrT getAttribute(const std::string& name, const AttrT& defaultValue) const;
+        /// Set the attribute.
+        template <typename AttrT>
+        void setAttribute(const std::string& name, const AttrT& value);
+        
+        std::vector<std::string> getSetAttributeNames() const;
+        
+        
+        // CHILDREN
+        
+        /// Indicate whether there is a child of the given type.
+        template <class OtherDerived>
+        bool hasChild() const;
+        /// Indicate whether there is a child for which the given predicate is true.
+        template <class OtherDerived>
+        bool hasChild(std::function<bool(OtherDerived)> predicate) const;
+        /// Indicate whether there is a child with the given attribute value.
+        template <class OtherDerived>
+        bool hasChild(const std::string& attrName, const std::string& attrValue) const;
+        
+        /// Get the first child of the given type.
+        template <class OtherDerived>
+        OtherDerived firstChild() const;
+        /// Get the first child of the given type for which the given predicate is true.
+        template <class OtherDerived>
+        OtherDerived firstChild(std::function<bool(OtherDerived)> predicate) const;
+        /// Get the first child of the given type with the given attribute value.
+        template <class OtherDerived>
+        OtherDerived firstChild(const std::string& attrName, const std::string& attrValue) const;
+        
+        
+        // SIBLINGS
+        
+        /// Get the next sibling element of the given type.
+        template <class OtherDerived>
+        OtherDerived nextSiblingElement() const;
+        /// Get the next sibling element of the given type for which the predicate is true.
+        template <class OtherDerived>
+        OtherDerived nextSiblingElement(std::function<bool(OtherDerived)> predicate) const;
+        /// Get the next sibling element of the given type with the given attribute value.
+        template <class OtherDerived>
+        OtherDerived nextSiblingElement(const std::string& attrName, const std::string& attrValue) const;
+
+        
+        // INSERTION & DELETION
+
+        /// Add a new child of this with of given type.
+        template <class OtherDerived>
+        OtherDerived addChild(const std::string& className = "");
+        
+        /// Get the first child of type OtherDerived. If there is none, add one.
+        template <class OtherDerived>
+        OtherDerived getOrCreateChild();
+        
+        
+        /// Insert an element at the front.
+        template <class OtherDerived>
+        void insertFirstChild(const Element<OtherDerived>& element);
+        /// Insert an element at the end.
+        template <class OtherDerived>
+        void insertEndChild(const Element<OtherDerived>& element);
+
+        /// Delete given child from this.
+        template <class OtherDerived>
+        void deleteChild(const Element<OtherDerived>& element);
+        
+        
+        // OPERATORS
+        
+        /// Indicate whether this contains a valid element.
+        operator bool() const { return elem != nullptr; }
+        
+        template <typename Derived>
+        friend std::ostream& operator<<(std::ostream& os, const Element<Derived>& element);
+        
+        
+    protected:
+        
+        Document& document() { return *_document; }
+        const Document& document() const  { return *_document; }
+        
+        
+    private:
+
+        /// Use document to create a new element of type ElementD with given parent.
+        template <class ParentD, class ElementD>
+        ElementD createElement(Element<ParentD> parent, const std::string& className = "");
+        
+        void assertElemValueEqualsTag();
+        
+        template <class OtherDerived>
+        std::function<bool(OtherDerived)> predicateWithAttrib(
+                const std::string& attrName, const std::string& attrValue) const;
+        
+        Document* _document;
+        tinyxml2::XMLElement* elem;
+        
+    };
+    
+    template <class D>
+    Element<D>::Element(Document* document, tinyxml2::XMLElement* elem) : 
+        _document(document), elem(elem)
+    {
+        assertElemValueEqualsTag();
+    }
+    
+    template <class D>
+    bool Element<D>::isAttributeSet(const std::string& attrName) const
+    {
+        return elem->Attribute(attrName.c_str()) != nullptr;
+    }
+    
+    template <class D>
+    template <typename AttrT>    
+    AttrT Element<D>::getAttribute(const std::string& name) const
+    {
+        const char* attr = elem->Attribute(name.c_str());
+        if (!attr)
+        {
+            throw AttribNotSetError(name);
+        }
+        AttrT value;
+        fromAttr(attr, value);
+        return value;
+    }
+    
+    template <class D>
+    template <typename AttrT>
+    AttrT Element<D>::getAttribute(const std::string& name, const AttrT& defaultValue) const
+    {
+        return isAttributeSet(name) ? getAttribute<AttrT>(name) : defaultValue;
+    }
+    
+    template <class D>
+    template<typename AttrT>
+    void Element<D>::setAttribute(const std::string& name, const AttrT& value)
+    {
+        std::string attrStr = toAttr(value);
+        elem->SetAttribute(name.c_str(), attrStr.c_str());
+    }
+
+    template <class D>
+    std::vector<std::string> Element<D>::getSetAttributeNames() const
+    {
+        std::vector<std::string> names;
+        for (auto* attr = elem->FirstAttribute(); attr; attr = attr->Next())
+        {
+            names.emplace_back(attr->Name());
+        }
+        return names;
+    }
+    
+    template <class D>
+    template <class OtherD>
+    bool Element<D>::hasChild() const
+    {
+        return firstChild<OtherD>();
+    }
+    
+    template <class D>
+    template <class OtherD>
+    bool Element<D>::hasChild(std::function<bool(OtherD)> predicate) const
+    {
+        return firstChild<OtherD>(predicate);
+    }
+    
+    template <class D>
+    template <class OtherD>
+    bool Element<D>::hasChild(const std::string& attrName, const std::string& attrValue) const
+    {
+        return firstChild<OtherD>(attrName, attrValue);
+    }
+    
+    template <class D>
+    template <class OtherD>
+    OtherD Element<D>::firstChild() const
+    {
+        return OtherD(_document, /*may be null*/ elem->FirstChildElement(OtherD::tag.c_str()));
+    }
+    
+    template <class D>
+    template <class OtherD>
+    OtherD Element<D>::firstChild(std::function<bool(OtherD)> predicate) const
+    {
+        for (OtherD child = firstChild<OtherD>(); child;
+             child = child.template nextSiblingElement<OtherD>())
+        {
+            if (predicate(child))
+            {
+                return child;
+            }
+        }
+        return OtherD(_document, nullptr);
+    }
+    
+    template <class D>
+    template <class OtherD>
+    OtherD Element<D>::firstChild(const std::string& attrName, const std::string& attrValue) const
+    {
+        return firstChild<OtherD>(predicateWithAttrib<OtherD>(attrName, attrValue));
+    }
+
+
+    
+    template <class D>
+    template <class OtherD>
+    OtherD Element<D>::nextSiblingElement() const
+    {
+        return OtherD(_document, elem->NextSiblingElement(OtherD::tag.c_str()));
+    }
+    
+    template <class D>
+    template<class OtherD>
+    OtherD Element<D>::nextSiblingElement(std::function<bool (OtherD)> predicate) const
+    {
+        for (OtherD sibling = nextSiblingElement<OtherD>(); sibling;
+             sibling = sibling.template nextSiblingElement<OtherD>())
+        {
+            if (predicate(sibling))
+            {
+                return sibling;
+            }
+        }
+        return OtherD(_document, nullptr);
+    }
+    
+    template <class D>
+    template <class OtherD>
+    OtherD Element<D>::addChild(const std::string& className)
+    {
+        return createElement<D, OtherD>(*this, className);
+    }
+    
+    template <class D>
+    template <class OtherD>
+    OtherD Element<D>::getOrCreateChild()
+    {
+        OtherD child = firstChild<OtherD>();
+        if (!child)
+        {
+            child = addChild<OtherD>();
+        }
+        return child;
+    }
+    
+    
+    template <class D>
+    template <class OtherD>
+    void Element<D>::insertFirstChild(const Element<OtherD>& element)
+    {
+        elem->InsertFirstChild(element.elem);
+    }
+    
+    template <class D>
+    template <class OtherD>
+    void Element<D>::insertEndChild(const Element<OtherD>& element)
+    {
+        elem->InsertEndChild(element.elem);
+    }
+    
+    template <class D>
+    template <class OtherD>
+    void Element<D>::deleteChild(const Element<OtherD>& element)
+    {
+        elem->DeleteChild(element.elem);
+    }
+    
+    template <class D>
+    template<class OtherD>
+    OtherD Element<D>::nextSiblingElement(const std::string& attrName, const std::string& attrValue) const
+    {
+        return nextSiblingElement<OtherD>(predicateWithAttrib<OtherD>(attrName, attrValue));
+    }
+    
+#include <stdexcept>
+    
+    template <class D>
+    void Element<D>::assertElemValueEqualsTag()
+    {
+
+        if (elem)
+        {
+            VR_ASSERT_MESSAGE(elem->Value() == Derived::tag, 
+                              "Element tag '" + std::string(elem->Value()) + "' must be '" + Derived::tag + "'");
+        }
+    }
+    
+    template <class D>
+    template <class OtherDerived>
+    std::function<bool(OtherDerived)> Element<D>::predicateWithAttrib(
+            const std::string& attrName, const std::string& attrValue) const
+    {
+        return [&attrName, &attrValue](OtherDerived e)
+        { 
+            return e.isAttributeSet(attrName) && e.getAttribute(attrName) == attrValue;
+        };
+    }
+    
+    template <class D>
+    std::ostream& operator<<(std::ostream& os, const Element<D>& element)
+    {
+        os << "MJCF Element '" << D::tag << "' (-> " << element.elem << ")";
+        return os;
+    }
+    
+
+#define mjcf_ElementDerivedConstructorsBase(Base, Derived)  \
+    Derived() : Base<Derived>(nullptr, nullptr) {}                   \
+    Derived(Document* document, tinyxml2::XMLElement* elem) : Base<Derived>(document, elem) {} \
+    Derived(const Derived& other) : Base<Derived>(other) {} \
+    Derived(Derived&& other) : Base<Derived>(other) {}      \
+    Derived& operator=(const Derived& other)                \
+    { Base<Derived>::operator=(other); return *this; }      \
+    Derived& operator=(Derived&& other)                     \
+    { Base<Derived>::operator=(other); return *this; }
+    
+    
+    
+#define mjcf_ElementDerivedConstructors(Derived)            \
+    mjcf_ElementDerivedConstructorsBase(Element, Derived)
+
+
+}
diff --git a/VirtualRobot/MJCF/elements/actuator.cpp b/VirtualRobot/MJCF/elements/actuator.cpp
new file mode 100644
index 000000000..440cff2d6
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/actuator.cpp
@@ -0,0 +1,84 @@
+#include "actuator.h"
+
+#include "../Document.h"
+
+
+using namespace mjcf;
+
+
+const std::string ActuatorGeneral::tag  = "general";
+const std::string ActuatorMotor::tag    = "motor";
+const std::string ActuatorPosition::tag = "position";
+const std::string ActuatorVelocity::tag = "velocity";
+const std::string ActuatorCylinder::tag = "cylinder";
+const std::string ActuatorMuscle::tag   = "muscle";
+
+const std::string ActuatorSection::tag  = "actuator";
+
+
+Eigen::VectorXf ActuatorGeneral::dynprmDefault()
+{
+    Eigen::VectorXf v(10);
+    v.setZero();
+    v(0) = 1;
+    return v;
+}
+
+Eigen::VectorXf ActuatorGeneral::gainprmDefault()
+{
+    Eigen::VectorXf v(10);
+    v.setZero();
+    v(0) = 1;
+    return v;
+}
+
+Eigen::VectorXf ActuatorGeneral::biasprmDefault()
+{
+    Eigen::VectorXf v(10);
+    v.setZero();
+    return v;
+}
+
+Eigen::Vector2f ActuatorMuscle::timeconstDefault()
+{
+    return Eigen::Vector2f(0.01, 0.04);
+}
+
+Eigen::Vector2f ActuatorMuscle::rangeDefault()
+{
+    return Eigen::Vector2f(0.75, 1.05);
+}
+
+
+template <class ElementD>
+ElementD addActuator(ActuatorSection& as, const std::string& joint)
+{
+    ElementD motor = as.addChild<ElementD>();
+    motor.joint = joint;
+    return motor;
+}
+
+ActuatorMotor ActuatorSection::addMotor(const std::string& joint)
+{
+    return addActuator<ActuatorMotor>(*this, joint);
+}
+
+ActuatorPosition ActuatorSection::addPosition(const std::string& joint, float kp)
+{
+    ActuatorPosition position = addActuator<ActuatorPosition>(*this, joint);
+    if (kp >= 0)
+    {
+        position.kp = kp;
+    }
+    return position;
+}
+
+ActuatorVelocity ActuatorSection::addVelocity(const std::string& joint, float kv)
+{
+    ActuatorVelocity velocity = addActuator<ActuatorVelocity>(*this, joint);
+    if (kv >= 0)
+    {
+        velocity.kv = kv;
+    }
+    return velocity;
+}
diff --git a/VirtualRobot/MJCF/elements/actuator.h b/VirtualRobot/MJCF/elements/actuator.h
new file mode 100644
index 000000000..048290b48
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/actuator.h
@@ -0,0 +1,143 @@
+#pragma once
+
+#include "Attribute.h"
+
+
+namespace mjcf
+{
+
+static Eigen::Vector6f gearDefault()
+{
+    Eigen::Vector6f v;
+    v << 1, 0, 0, 0, 0, 0;
+    return v;
+}
+
+
+#define mjcf_ActuatorAttributes(Derived)    \
+    mjcf_NameAttribute(Derived);    \
+    mjcf_ClassAttribute(Derived);   \
+    \
+    mjcf_GroupAttribute(Derived);   \
+    \
+    mjcf_BoolAttributeDef(Derived, ctrllimited, false);     \
+    mjcf_BoolAttributeDef(Derived, forcelimited, false);    \
+    \
+    mjcf_Vector2fAttributeDef(Derived, ctrlrange, Eigen::Vector2f::Zero());     \
+    mjcf_Vector2fAttributeDef(Derived, forcerange, Eigen::Vector2f::Zero());    \
+    mjcf_Vector2fAttributeDef(Derived, lengthrange, Eigen::Vector2f::Zero());   \
+    \
+    mjcf_AttributeDef(Derived, Eigen::Vector6f, gear, gearDefault());   \
+    mjcf_FloatAttributeDef(Derived, cranklength, 0);    \
+    \
+    mjcf_StringAttributeOpt(Derived, joint);            \
+    mjcf_StringAttributeOpt(Derived, jointinparent);    \
+    mjcf_StringAttributeOpt(Derived, site);             \
+    mjcf_StringAttributeOpt(Derived, tendon);           \
+    mjcf_StringAttributeOpt(Derived, cranksite);        \
+    mjcf_StringAttributeOpt(Derived, slidersite)
+
+
+struct ActuatorGeneral : public Element<ActuatorGeneral>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(ActuatorGeneral)
+    
+    mjcf_ActuatorAttributes(ActuatorGeneral);
+    
+    // dyntype : [none, integrator, filter, muscle, user], 
+    mjcf_StringAttributeDef(ActuatorGeneral, dyntype, "none");
+    // gaintype : [fixed, muscle, user], "fixed"
+    mjcf_StringAttributeDef(ActuatorGeneral, gaintype, "fixed");
+    // biastype : [none, affine, muscle, user], "none"
+    mjcf_StringAttributeDef(ActuatorGeneral, biastype, "none");
+    
+    // dynprm : real(10), "1 0 ... 0"
+    mjcf_AttributeDef(ActuatorGeneral, Eigen::VectorXf, dynprm, dynprmDefault());
+    // gainprm : real(10), "1 0 ... 0"
+    mjcf_AttributeDef(ActuatorGeneral, Eigen::VectorXf, gainprm, gainprmDefault());
+    // biasprm : real(10), "1 0 ... 0"
+    mjcf_AttributeDef(ActuatorGeneral, Eigen::VectorXf, biasprm, biasprmDefault());
+    
+    
+private:
+    static Eigen::VectorXf dynprmDefault();
+    static Eigen::VectorXf gainprmDefault();
+    static Eigen::VectorXf biasprmDefault();
+    
+};
+
+struct ActuatorMotor : public Element<ActuatorMotor>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(ActuatorMotor)
+    
+    mjcf_ActuatorAttributes(ActuatorMotor);
+};
+
+struct ActuatorPosition : public Element<ActuatorPosition>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(ActuatorPosition)
+    
+    mjcf_ActuatorAttributes(ActuatorPosition);
+    mjcf_FloatAttributeDef(ActuatorPosition, kp, 1);
+};
+
+struct ActuatorVelocity : public Element<ActuatorVelocity>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(ActuatorVelocity)
+    
+    mjcf_ActuatorAttributes(ActuatorVelocity);
+    mjcf_FloatAttributeDef(ActuatorVelocity, kv, 1);
+};
+
+struct ActuatorCylinder : public Element<ActuatorCylinder>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(ActuatorCylinder)
+    
+    mjcf_ActuatorAttributes(ActuatorCylinder);
+    mjcf_FloatAttributeDef(ActuatorCylinder, timeconst, 1);
+    mjcf_FloatAttributeDef(ActuatorCylinder, area, 1);
+    mjcf_FloatAttributeOpt(ActuatorCylinder, diameter);
+    mjcf_Vector3fAttributeDef(ActuatorCylinder, bias, Eigen::Vector3f::Zero());
+};
+
+struct ActuatorMuscle : public Element<ActuatorMuscle>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(ActuatorMuscle)
+    
+    mjcf_ActuatorAttributes(ActuatorMuscle);
+    
+    mjcf_Vector2fAttributeDef(ActuatorMuscle, timeconst, timeconstDefault());
+    mjcf_Vector2fAttributeDef(ActuatorMuscle, range, rangeDefault());
+    
+    mjcf_FloatAttributeDef(ActuatorMuscle, force, -1);
+    mjcf_FloatAttributeDef(ActuatorMuscle, scale, 200);
+    mjcf_FloatAttributeDef(ActuatorMuscle, lmin, 0.5);
+    mjcf_FloatAttributeDef(ActuatorMuscle, lmax, 1.6f);
+    mjcf_FloatAttributeDef(ActuatorMuscle, vmax, 1.5);
+    mjcf_FloatAttributeDef(ActuatorMuscle, fpmax, 1.3f);
+    mjcf_FloatAttributeDef(ActuatorMuscle, vvmax, 1.2f);
+    
+private:
+    static Eigen::Vector2f timeconstDefault();
+    static Eigen::Vector2f rangeDefault();
+};
+
+
+struct ActuatorSection : public Element<ActuatorSection>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(ActuatorSection)
+    
+    ActuatorMotor addMotor(const std::string& jointName);
+    ActuatorPosition addPosition(const std::string& joint, float kp = -1);
+    ActuatorVelocity addVelocity(const std::string& joint, float kv = -1);
+    
+};
+
+}
diff --git a/VirtualRobot/MJCF/elements/asset.cpp b/VirtualRobot/MJCF/elements/asset.cpp
new file mode 100644
index 000000000..7774133ad
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/asset.cpp
@@ -0,0 +1,55 @@
+#include "asset.h"
+
+#include "../Document.h"
+
+
+using namespace mjcf;
+
+
+const std::string Texture::tag      = "texture";
+const std::string Material::tag     = "material";
+const std::string Mesh::tag         = "mesh";
+const std::string AssetSection::tag = "asset";
+
+
+Texture AssetSection::addSkyboxTexture(const Eigen::Vector3f& rgb1, const Eigen::Vector3f& rgb2)
+{
+    Texture texSkybox = addChild<Texture>();
+
+    texSkybox.type = "skybox";
+    texSkybox.builtin = "gradient";
+    texSkybox.width = 128;
+    texSkybox.height = 128;
+    texSkybox.rgb1 = rgb1;
+    texSkybox.rgb2 = rgb2;
+
+    return texSkybox;
+}
+
+Texture AssetSection::addTextureFile(const std::string& name, const std::string& file, 
+                                     bool vflip, bool hflip)
+{
+    Texture texture = addChild<Texture>();
+    texture.name = name;
+    texture.file = file;
+    texture.type = "2d";
+    texture.vflip = vflip;
+    texture.hflip = hflip;
+    return texture;
+}
+
+Mesh AssetSection::addMesh(const std::string& name, const std::string& file)
+{
+    Mesh mesh = addChild<Mesh>();
+    mesh.name = name;
+    mesh.file = file;
+    return mesh;
+}
+
+Material AssetSection::addMaterial(const std::string& name, const std::string& textureName)
+{
+    Material material = addChild<Material>();
+    material.name = name;
+    material.texture = textureName;
+    return material;
+}
diff --git a/VirtualRobot/MJCF/elements/asset.h b/VirtualRobot/MJCF/elements/asset.h
new file mode 100644
index 000000000..bd9d25bc2
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/asset.h
@@ -0,0 +1,93 @@
+#pragma once
+
+#include "Attribute.h"
+
+
+namespace mjcf
+{
+
+struct Texture : public Element<Texture>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(Texture)
+    
+    mjcf_NameAttribute(Texture);
+    mjcf_StringAttributeDef(Texture, type, "cube");
+    mjcf_StringAttributeOpt(Texture, file);
+    
+    mjcf_AttributeDef(Texture, Eigen::Vector2i, gridsize, Eigen::Vector2i::Ones());
+    mjcf_StringAttributeDef(Texture, gridlayout, "............");
+  
+    mjcf_StringAttributeDef(Texture, builtin, "none");
+    mjcf_StringAttributeDef(Texture, mark, "none");
+    mjcf_RgbAttributeDef(Texture, markrgb, Eigen::Vector3f::Zero());
+    mjcf_FloatAttributeDef(Texture, random, 0.01f);
+    
+    mjcf_IntAttributeDef(Texture, width, 0);
+    mjcf_IntAttributeDef(Texture, height, 0);
+  
+    mjcf_RgbAttributeDef(Texture, rgb1, 0.8f * Eigen::Vector3f::Ones());
+    mjcf_RgbAttributeDef(Texture, rgb2, 0.5f * Eigen::Vector3f::Ones());
+    
+    mjcf_BoolAttributeDef(Texture, hflip, false);
+    mjcf_BoolAttributeDef(Texture, vflip, true);
+};
+
+
+struct Mesh : public Element<Mesh>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(Mesh)
+    
+    mjcf_NameAttribute(Mesh);
+    mjcf_ClassAttribute(Mesh);
+    
+    mjcf_StringAttributeOpt(Mesh, file);
+    
+    mjcf_Vector3fAttributeDef(Mesh, scale,  Eigen::Vector3f::Ones());
+    mjcf_BoolAttributeDef(Mesh, smoothnormal, false);
+    
+    mjcf_Vector3fAttributeDef(Mesh, refpos,  Eigen::Vector3f::Ones());
+    mjcf_AttributeDef(Mesh, Eigen::Quaternionf, refquat, Eigen::Quaternionf::Identity());
+};
+
+
+struct Material : public Element<Material>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(Material)
+    
+    mjcf_NameAttribute(Material);
+    mjcf_ClassAttribute(Material);
+    
+    mjcf_StringAttributeOpt(Material, texture);
+    
+    mjcf_AttributeDef(Material, Eigen::Vector2f, texrepeat, Eigen::Vector2f::Ones());
+    mjcf_BoolAttributeDef(Material, texuniform, false);
+    
+    mjcf_FloatAttributeDef(Material, emission, 0);
+    mjcf_FloatAttributeDef(Material, specular, 0);
+    mjcf_FloatAttributeDef(Material, shininess, 0);
+    mjcf_FloatAttributeDef(Material, reflectance, 0);
+    mjcf_RgbaAttributeDef(Material, rgba, Eigen::Vector4f::Ones());
+};
+
+
+
+struct AssetSection : public Element<AssetSection>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(AssetSection)
+    
+    Texture addSkyboxTexture(const Eigen::Vector3f& rgb1, const Eigen::Vector3f& rgb2);    
+    
+    Texture addTextureFile(const std::string& name, const std::string& file,
+                           bool vflip = false, bool hflip = false);
+    
+    /// Add a mesh asset with a name and a file.
+    Mesh addMesh(const std::string& name, const std::string& file);
+    
+    Material addMaterial(const std::string& name, const std::string& textureName);
+};
+
+}
diff --git a/VirtualRobot/MJCF/elements/body.cpp b/VirtualRobot/MJCF/elements/body.cpp
new file mode 100644
index 000000000..e455072b9
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/body.cpp
@@ -0,0 +1,155 @@
+#include "body.h"
+
+#include "../Document.h"
+
+
+using namespace mjcf;
+
+
+const std::string Body::tag         = "body";
+const std::string Geom::tag         = "geom";
+const std::string Inertial::tag     = "inertial";
+const std::string Joint::tag        = "joint";
+const std::string FreeJoint::tag    = "freejoint";
+const std::string Worldbody::tag        = "worldbody";
+
+
+void Inertial::inertiaFromMatrix(const Eigen::Matrix3f& matrix)
+{
+    if (matrix.isDiagonal(1e-9f))
+    {
+        this->diaginertia = matrix.diagonal();
+    }
+    else
+    {
+        /* Full inertia matrix M. Since M is 3-by-3 and symmetric, 
+         * it is specified using only 6 numbers in the following order: 
+         * M(1,1), M(2,2), M(3,3), M(1,2), M(1,3), M(2,3).
+         * -- http://www.mujoco.org/book/XMLreference.html#inertial 
+         */
+        
+        Eigen::Vector6f inertia;
+        inertia << matrix(0, 0), matrix(1, 1), matrix(2, 2), 
+                matrix(0, 1), matrix(0, 2), matrix(1, 2);
+        this->fullinertia = inertia;
+    }
+}
+
+bool Body::hasMass()
+{
+    return hasChild<Geom>() || hasChild<Inertial>();
+}
+
+Body Body::addBody(const std::string& name)
+{
+    Body body = addChild<Body>();
+
+    if (!name.empty())
+    {
+        body.name = name;
+    }
+
+    return body;
+}
+
+Inertial Body::addInertial()
+{
+    return addChild<Inertial>();
+}
+
+Inertial Body::addInertial(const Eigen::Vector3f& pos, float mass, const Eigen::Matrix3f& matrix)
+{
+    Inertial inertial = addInertial();
+    
+    inertial.pos = pos;
+    inertial.mass = mass;
+
+    if (matrix.isDiagonal(document().getFloatCompPrecision()))
+    {
+        inertial.diaginertia = matrix.diagonal();
+    }
+    else
+    {
+        /* from mujoco xml reference:
+         * "Full inertia matrix M. Since M is 3-by-3 and symmetric, it is
+         * specified using only 6 numbers in the following order:
+         * M(1,1), M(2,2), M(3,3), M(1,2), M(1,3), M(2,3)."*/
+        Eigen::Vector6f inertia;
+        inertia << matrix(0, 0), matrix(1, 1), matrix(2, 2),
+                matrix(0, 1), matrix(0, 2), matrix(1, 2);
+        inertial.fullinertia = inertia;
+    }
+    
+    return inertial;
+}
+
+Inertial Body::addDummyInertial()
+{
+    Inertial inertial = addInertial();
+
+    inertial.pos = Eigen::Vector3f(0, 0, 0);
+    inertial.mass = document().getDummyMass();
+    inertial.diaginertia = Eigen::Vector3f::Ones();
+
+    return inertial;
+}
+
+Joint Body::addJoint()
+{
+    return addChild<Joint>();
+}
+
+FreeJoint Body::addFreeJoint()
+{
+    return addChild<FreeJoint>();
+    
+}
+
+Geom Body::addGeom(const std::string& type)
+{
+    Geom geom = addChild<Geom>();
+    
+    geom.type = type;
+    
+    return geom;
+}
+
+Geom Body::addGeomMesh(const std::string& meshName, const std::string& materialName)
+{
+    Geom geom = addGeom("mesh");
+    
+    geom.mesh = meshName;
+    if (!materialName.empty())
+    {
+        geom.material = materialName;
+    }
+    
+    return geom;
+}
+
+
+Body Worldbody::addMocapBody(const std::string& name, float geomSize)
+{
+    Body mocap = addChild<Body>();
+    mocap.name = name;
+    mocap.mocap = true;
+    // add geom for visualization
+
+    Geom geom = mocap.addChild<Geom>();
+    geom.type = "box";
+    geom.size = geomSize * Eigen::Vector3f::Ones();
+
+    return mocap;
+}
+
+Body Worldbody::addBody(const std::string& name)
+{
+    Body body = addChild<Body>();
+
+    if (!name.empty())
+    {
+        body.name = name;
+    }
+
+    return body;
+}
diff --git a/VirtualRobot/MJCF/elements/body.h b/VirtualRobot/MJCF/elements/body.h
new file mode 100644
index 000000000..04fe63286
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/body.h
@@ -0,0 +1,163 @@
+#pragma once
+
+#include "Attribute.h"
+
+
+namespace mjcf
+{
+
+
+struct Inertial : public Element<Inertial>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(Inertial)
+    
+    mjcf_PoseAttributes(Inertial);
+    
+    mjcf_AttributeReq(Inertial, float, mass);
+    
+    mjcf_AttributeOpt(Inertial, Eigen::Vector3f, diaginertia);
+    mjcf_AttributeOpt(Inertial, Eigen::Vector6f, fullinertia);
+    
+    /// Sets diaginertia or fullinertia (as adequate) from the given matrix.
+    void inertiaFromMatrix(const Eigen::Matrix3f& matrix);
+};
+
+
+struct Joint : public Element<Joint>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(Joint)
+    
+    void setAxis(const Eigen::Vector3f& axis);
+    
+    mjcf_NameAttribute(Joint);
+    mjcf_ClassAttribute(Joint);
+    
+    // free, ball, slide, hinge
+    mjcf_StringAttributeDef(Joint, type, "hinge");
+    mjcf_IntAttributeDef(Joint, group, 0);
+    
+    mjcf_PosAttribute(Joint);
+    mjcf_Vector3fAttributeDef(Joint, axis, Eigen::Vector3f(0, 0, 1));
+    
+    mjcf_AttributeDef(Joint, Eigen::Vector2f, springdamper, Eigen::Vector2f::Zero());
+    mjcf_FloatAttributeDef(Joint, stiffness, 0);
+    
+    mjcf_BoolAttributeDef(Joint, limited, false);
+    mjcf_AttributeDef(Joint, Eigen::Vector2f, range, Eigen::Vector2f::Zero());
+    
+    mjcf_FloatAttributeDef(Joint, margin, 0);
+    
+    mjcf_FloatAttributeDef(Joint, ref, 0);
+    mjcf_FloatAttributeDef(Joint, springref, 0);
+    
+    mjcf_FloatAttributeDef(Joint, armature, 0);
+    mjcf_FloatAttributeDef(Joint, damping, 0);
+    mjcf_FloatAttributeDef(Joint, frictionloss, 0);
+};
+
+
+struct FreeJoint : public Element<FreeJoint>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(FreeJoint)
+    
+    mjcf_NameAttribute(FreeJoint);
+    mjcf_IntAttributeDef(FreeJoint, group, 0);
+};
+
+
+struct Geom : public Element<Geom>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(Geom)
+    
+    mjcf_NameAttribute(Geom);
+    mjcf_ClassAttribute(Geom);
+    
+    // plane, hfield, sphere, capsule, ellipsoid, cylinder, box, mesh
+    mjcf_StringAttributeDef(Geom, type, "sphere");
+    
+    mjcf_IntAttributeDef(Geom, contype, 1);
+    mjcf_IntAttributeDef(Geom, conaffinity, 1);
+    mjcf_IntAttributeDef(Geom, condim, 3);
+    
+    mjcf_IntAttributeDef(Geom, group, 0);
+    
+    mjcf_Vector3fAttributeDef(Geom, size, Eigen::Vector3f::Zero());
+    mjcf_StringAttributeOpt(Geom, material);
+    mjcf_RgbaAttributeDef(Geom, rgba, Eigen::Vector4f(.5, .5, .5, 1));
+    
+    mjcf_Vector3fAttributeDef(Geom, friction, Eigen::Vector3f(1, 0.005f, 0.0001f));
+    
+    mjcf_FloatAttributeOpt(Geom, mass);
+    mjcf_FloatAttributeDef(Geom, density, 1000);
+    
+    mjcf_FloatAttributeDef(Geom, solmix, 1);
+    
+    mjcf_FloatAttributeDef(Geom, marin, 0);
+    mjcf_FloatAttributeDef(Geom, gap, 0);
+    
+    mjcf_AttributeOpt(Geom, Eigen::Vector6f, fromto);
+    
+    mjcf_PoseAttributes(Geom);
+    
+    mjcf_StringAttributeOpt(Geom, hfield);
+    mjcf_StringAttributeOpt(Geom, mesh);
+    
+    mjcf_FloatAttributeDef(Geom, fitscale, 1);
+};
+
+
+struct Body : public Element<Body>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(Body)
+    
+    bool hasMass();
+    
+    /// Add a child body element.
+    Body addBody(const std::string& name = "");
+    
+    /// Add an inertial element.
+    Inertial addInertial();
+    /// Add an inertial element with given settings.
+    Inertial addInertial(const Eigen::Vector3f& pos, float mass, const Eigen::Matrix3f& matrix);
+    /// Add a dummy inertial element with small mass and identity inertia matrix.
+    Inertial addDummyInertial();
+    
+    
+    /// Add a joint element to a body.
+    Joint addJoint();
+    /// Add a free joint element to a body.
+    FreeJoint addFreeJoint();
+
+    /// Add a geom to a body, referencing a mesh.
+    Geom addGeom(const std::string& type);
+    /// Add a mesh geom with optional material (for texture).
+    Geom addGeomMesh(const std::string& meshName, const std::string& materialName = "");
+    
+    
+    mjcf_NameAttribute(Body);
+    mjcf_StringAttributeOpt(Body, childclass);
+    
+    mjcf_BoolAttributeDef(Body, mocap, false);
+    
+    mjcf_PoseAttributes(Body);
+};
+
+struct Worldbody : public Element<Worldbody>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(Worldbody)
+    
+    /// Add a body element.
+    Body addBody(const std::string& name = "");
+    
+    /// Add a mocap body with the given name to the worldbody.
+    Body addMocapBody(const std::string& name, float geomSize);
+    
+};
+
+}
diff --git a/VirtualRobot/MJCF/elements/compiler.cpp b/VirtualRobot/MJCF/elements/compiler.cpp
new file mode 100644
index 000000000..cfa52011b
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/compiler.cpp
@@ -0,0 +1,9 @@
+#include "compiler.h"
+
+//#include "../Document.h"
+
+
+using namespace mjcf;
+
+
+const std::string CompilerSection::tag  = "compiler";
diff --git a/VirtualRobot/MJCF/elements/compiler.h b/VirtualRobot/MJCF/elements/compiler.h
new file mode 100644
index 000000000..054929963
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/compiler.h
@@ -0,0 +1,14 @@
+#pragma once
+
+#include "Attribute.h"
+
+namespace mjcf
+{
+
+struct CompilerSection : public Element<CompilerSection>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(CompilerSection)
+};
+
+}
diff --git a/VirtualRobot/MJCF/elements/contact.cpp b/VirtualRobot/MJCF/elements/contact.cpp
new file mode 100644
index 000000000..e60977b27
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/contact.cpp
@@ -0,0 +1,49 @@
+#include "contact.h"
+
+#include "../Document.h"
+
+using namespace mjcf;
+
+
+const std::string ContactPair::tag    = "pair";
+const std::string ContactExclude::tag = "exclude";
+const std::string ContactSection::tag = "contact";
+
+
+Eigen::Vector5f ContactPair::condimDefault() 
+{
+    Eigen::Vector5f v;
+    v << 1, 1, 0.005f, 0.0001f, 0.0001f;
+    return v;
+}
+
+ContactPair ContactSection::addPair(const Geom& geom1, const Geom& geom2)
+{
+    return addPair(geom1.name.get(), geom2.name.get());
+}
+
+ContactPair ContactSection::addPair(const std::string& geom1Name, const std::string& geom2Name)
+{
+    ContactPair pair = addChild<ContactPair>();
+
+    pair.geom1 = geom1Name;
+    pair.geom2 = geom2Name;
+
+    return pair;
+}
+
+
+ContactExclude ContactSection::addExclude(const Body& body1, const Body& body2)
+{
+    return addExclude(body1.name.get(), body2.name.get());
+}
+
+ContactExclude ContactSection::addExclude(const std::string& body1Name, const std::string& body2Name)
+{
+    ContactExclude exclude = addChild<ContactExclude>();
+
+    exclude.body1 = body1Name;
+    exclude.body2 = body2Name;
+
+    return exclude;
+}
diff --git a/VirtualRobot/MJCF/elements/contact.h b/VirtualRobot/MJCF/elements/contact.h
new file mode 100644
index 000000000..f5bbe7bfb
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/contact.h
@@ -0,0 +1,62 @@
+#pragma once
+
+#include "Attribute.h"
+#include "body.h"
+
+
+namespace mjcf
+{
+
+
+struct ContactPair : public Element<ContactPair>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(ContactPair)
+    
+    mjcf_NameAttribute(ContactPair);
+    mjcf_ClassAttribute(ContactPair);
+    
+    mjcf_StringAttributeReq(ContactPair, geom1);
+    mjcf_StringAttributeReq(ContactPair, geom2);
+    
+    mjcf_IntAttributeDef(ContactPair, condim, 3);
+    
+    mjcf_AttributeDef(ContactPair, Eigen::Vector5f, friction, condimDefault());
+    
+    mjcf_FloatAttributeDef(ContactPair, margin, 0);
+    mjcf_FloatAttributeDef(ContactPair, gap, 0);
+    
+private:
+    static Eigen::Vector5f condimDefault();
+};
+
+
+struct ContactExclude : public Element<ContactExclude>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(ContactExclude)
+    
+    mjcf_NameAttribute(ContactExclude);
+    mjcf_StringAttributeReq(ContactExclude, body1);
+    mjcf_StringAttributeReq(ContactExclude, body2);
+};
+
+
+struct ContactSection : public Element<ContactSection>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(ContactSection)
+    
+    /// Add a conact/pair element between the given geoms.
+    ContactPair addPair(const Geom& geom1, const Geom& geom2);
+    /// Add a conact/pair element between the given geoms.
+    ContactPair addPair(const std::string& geom1Name, const std::string& geom2Name);
+    
+    /// Add a conact/exclude element between the given bodies.
+    ContactExclude addExclude(const Body& body1, const Body& body2);
+    /// Add a conact/exclude element between the given bodies.
+    ContactExclude addExclude(const std::string& body1Name, const std::string& body2Name);
+};
+
+
+}
diff --git a/VirtualRobot/MJCF/elements/custom.cpp b/VirtualRobot/MJCF/elements/custom.cpp
new file mode 100644
index 000000000..4f2655738
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/custom.cpp
@@ -0,0 +1,6 @@
+#include "compiler.h"
+
+//#include "../Document.h"
+
+
+using namespace mjcf;
diff --git a/VirtualRobot/MJCF/elements/custom.h b/VirtualRobot/MJCF/elements/custom.h
new file mode 100644
index 000000000..56d302f89
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/custom.h
@@ -0,0 +1,11 @@
+#pragma once
+
+#include "Attribute.h"
+
+namespace mjcf
+{
+
+
+
+
+}
diff --git a/VirtualRobot/MJCF/elements/default.cpp b/VirtualRobot/MJCF/elements/default.cpp
new file mode 100644
index 000000000..cfc82ab53
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/default.cpp
@@ -0,0 +1,31 @@
+#include "default.h"
+
+#include "../Document.h"
+
+
+using namespace mjcf;
+
+
+const std::string DefaultClass::tag   = "default";
+const std::string DefaultSection::tag = "default";
+
+
+bool DefaultSection::hasClass(const std::string& className)
+{
+    return !hasChild<mjcf::DefaultClass>("class", className);
+}
+
+DefaultClass DefaultSection::getClass(const std::string& className)
+{
+    DefaultClass def = firstChild<DefaultClass>("class", className);
+    if (!def)
+    {
+        def = addClass(className);
+    }
+    return def ;
+}
+
+DefaultClass DefaultSection::addClass(const std::string& className)
+{
+    return addChild<DefaultClass>(className);
+}
diff --git a/VirtualRobot/MJCF/elements/default.h b/VirtualRobot/MJCF/elements/default.h
new file mode 100644
index 000000000..826c17017
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/default.h
@@ -0,0 +1,57 @@
+#pragma once
+
+#include "Attribute.h"
+
+
+namespace mjcf
+{
+
+
+struct DefaultClass : public Element<DefaultClass>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(DefaultClass)
+    
+    mjcf_ClassAttribute(DefaultClass);
+    
+    template <class ElementD>
+    ElementD getElement();
+    
+    template <class ElementD, typename AttrT>
+    ElementD setElementAttr(const std::string& attrName, const AttrT& attrValue);
+};
+
+
+struct DefaultSection : public Element<DefaultSection>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(DefaultSection)
+    
+    /// Indicate whether there is a default class with the given name.
+    bool hasClass(const std::string& className);
+    
+    /// Get the default class for given class name. If it does not exist, add it.
+    DefaultClass getClass(const std::string& className);
+    
+    /// Add a new default class with given class name.
+    DefaultClass addClass(const std::string& className);
+};
+
+
+
+template<class ElementD>
+ElementD DefaultClass::getElement()
+{
+    return getOrCreateChild<ElementD>();
+}
+
+template<class ElementD, typename AttrT>
+ElementD DefaultClass::setElementAttr(const std::string& attrName, const AttrT& attrValue)
+{
+    ElementD element = getElement<ElementD>();
+    element.setAttribute(attrName, attrValue);
+    return element;
+}
+
+
+}
diff --git a/VirtualRobot/MJCF/elements/elements.cpp b/VirtualRobot/MJCF/elements/elements.cpp
new file mode 100644
index 000000000..b87609cd3
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/elements.cpp
@@ -0,0 +1,11 @@
+#include "elements.h"
+
+using namespace mjcf;
+
+
+const std::string MujocoRoot::tag = "mujoco";
+const std::string Include::tag = "include";
+
+
+
+
diff --git a/VirtualRobot/MJCF/elements/elements.h b/VirtualRobot/MJCF/elements/elements.h
new file mode 100644
index 000000000..6a57c0c19
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/elements.h
@@ -0,0 +1,88 @@
+#pragma once
+
+#include "actuator.h"
+#include "asset.h"
+#include "body.h"
+#include "compiler.h"
+#include "contact.h"
+#include "default.h"
+#include "equality.h"
+#include "keyframe.h"
+#include "option.h"
+#include "sensor.h"
+#include "size.h"
+#include "statistic.h"
+#include "tendon.h"
+#include "visual.h"
+
+
+namespace mjcf
+{
+
+
+/// @see http://www.mujoco.org/book/XMLreference.html#mujoco
+struct MujocoRoot : public Element<MujocoRoot>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(MujocoRoot)
+    
+    mjcf_StringAttributeDef(MujocoRoot, model, "MuJoCo Model");
+};
+
+
+/// @see http://www.mujoco.org/book/XMLreference.html#include
+struct Include : public Element<Include>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(Include)
+    
+    mjcf_StringAttributeReq(Include, file);
+};
+
+
+}
+
+
+#undef mjcf_AttributeFullNoDef
+#undef mjcf_AttributeFullDef
+
+#undef mjcf_AttributeReq
+#undef mjcf_AttributeOpt
+#undef mjcf_AttributeDef
+
+#undef mjcf_AttributeKwReq
+#undef mjcf_AttributeKwOpt
+#undef mjcf_AttributeKwDef
+
+#undef mjcf_BoolAttributeReq
+#undef mjcf_BoolAttributeOpt
+#undef mjcf_BoolAttributeDef
+    
+#undef mjcf_IntAttributeReq
+#undef mjcf_IntAttributeOpt
+#undef mjcf_IntAttributeDef
+    
+#undef mjcf_FloatAttributeReq
+#undef mjcf_FloatAttributeOpt
+#undef mjcf_FloatAttributeDef
+
+#undef mjcf_StringAttributeReq
+#undef mjcf_StringAttributeOpt
+#undef mjcf_StringAttributeDef
+#undef mjcf_StringAttributeDefEmpty
+
+#undef mjcf_Vector2fAttributeDef
+#undef mjcf_Vector3fAttributeReq
+#undef mjcf_Vector3fAttributeOpt
+#undef mjcf_Vector3fAttributeDef
+    
+#undef mjcf_NameAttribute
+#undef mjcf_ClassAttribute
+#undef mjcf_GroupAttribute
+
+#undef mjcf_RgbAttributeDef
+#undef mjcf_RgbaAttributeDef
+    
+#undef mjcf_PosAttribute
+#undef mjcf_QuatAttribute
+#undef mjcf_PoseAttributes
diff --git a/VirtualRobot/MJCF/elements/equality.cpp b/VirtualRobot/MJCF/elements/equality.cpp
new file mode 100644
index 000000000..aedc14c37
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/equality.cpp
@@ -0,0 +1,48 @@
+#include "equality.h"
+
+#include "../Document.h"
+
+
+using namespace mjcf;
+
+
+const std::string EqualityConnect::tag = "connect";
+const std::string EqualityWeld::tag = "weld";
+const std::string EqualityJoint::tag = "joint";
+const std::string EqualityTendon::tag = "tendon";
+const std::string EqualityDistance::tag = "distance";
+const std::string EqualitySection::tag  = "equality";
+
+
+Eigen::Vector7f EqualityWeld::relposeDefault()
+{
+    Eigen::Vector7f v;
+    v << 0.f, 1.f, 0.f, 0.f, 0.f, 0.f, 0.f;
+    return v;
+}
+
+
+Eigen::Vector5f EqualityJoint::polycoefDefault()
+{
+    Eigen::Vector5f v;
+    v << 0, 1, 0, 0, 0;
+    return v;
+}
+
+Eigen::Vector5f EqualityTendon::polycoefDefault()
+{
+    Eigen::Vector5f v;
+    v << 0, 1, 0, 0, 0;
+    return v;
+}
+
+EqualityWeld EqualitySection::addWeld(const std::string& name, const std::string& body1, const std::string& body2)
+{
+    EqualityWeld weld = addChild<EqualityWeld>();
+
+    weld.name = name;
+    weld.body1 = body1;
+    weld.body2 = body2;
+
+    return weld;
+}
diff --git a/VirtualRobot/MJCF/elements/equality.h b/VirtualRobot/MJCF/elements/equality.h
new file mode 100644
index 000000000..1550a50cf
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/equality.h
@@ -0,0 +1,96 @@
+#pragma once
+
+#include "Attribute.h"
+
+
+namespace mjcf
+{
+
+#define mjcf_EqualityAttribs(Derived, entityName) \
+    mjcf_NameAttribute(Derived);    \
+    mjcf_ClassAttribute(Derived);   \
+    mjcf_BoolAttributeDef(Derived, active, true);   \
+    mjcf_StringAttributeReq(Derived, entityName ## 1);    \
+    mjcf_StringAttributeOpt(Derived, entityName ## 2)
+
+
+
+struct EqualityConnect : public Element<EqualityConnect>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(EqualityConnect)
+    
+    mjcf_EqualityAttribs(EqualityConnect, body);
+    
+    mjcf_Vector3fAttributeReq(EqualityConnect, anchor);
+};
+
+
+struct EqualityWeld : public Element<EqualityWeld>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(EqualityWeld)
+
+    mjcf_EqualityAttribs(EqualityWeld, body);
+    
+    mjcf_AttributeDef(EqualityWeld, Eigen::Vector7f, relpose, relposeDefault());
+    
+private:
+    static Eigen::Vector7f relposeDefault();
+    
+};
+
+
+struct EqualityJoint : public Element<EqualityJoint>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(EqualityJoint)
+
+    mjcf_EqualityAttribs(EqualityJoint, joint);
+    
+    mjcf_AttributeDef(EqualityJoint, Eigen::Vector5f, polycoef, polycoefDefault());
+    
+private:
+    static Eigen::Vector5f polycoefDefault();
+    
+};
+
+
+struct EqualityTendon : public Element<EqualityTendon>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(EqualityTendon)
+
+    mjcf_EqualityAttribs(EqualityTendon, tendon);
+    
+    mjcf_AttributeDef(EqualityTendon, Eigen::Vector5f, polycoef, polycoefDefault());
+    
+private:
+    static Eigen::Vector5f polycoefDefault();
+    
+};
+
+struct EqualityDistance : public Element<EqualityDistance>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(EqualityDistance)
+
+    mjcf_EqualityAttribs(EqualityDistance, geom);
+    
+    mjcf_FloatAttributeDef(EqualityDistance, distance, 0);
+};
+
+
+struct EqualitySection : public Element<EqualitySection>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(EqualitySection)
+    
+    EqualityWeld addWeld(const std::string& name, const std::string& body1, 
+                         const std::string& body2);
+    
+};
+
+
+#undef mjcf_EqualityAttribs
+}
diff --git a/VirtualRobot/MJCF/elements/exceptions.cpp b/VirtualRobot/MJCF/elements/exceptions.cpp
new file mode 100644
index 000000000..dc7ccbfdb
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/exceptions.cpp
@@ -0,0 +1,28 @@
+#include "exceptions.h"
+
+#include <typeinfo>
+
+
+namespace mjcf
+{
+
+MjcfError::MjcfError(const std::string& message) : std::runtime_error(message)
+{}
+
+MjcfIOError::MjcfIOError(const std::string& message) : MjcfError (message)
+{}
+
+AttribNotSetError::AttribNotSetError(const std::string& name) : 
+    MjcfError("Attrib '" + name + "' (without default) not set.")
+{}
+
+ParseAttributeError::ParseAttributeError(
+        const std::string& source, 
+        const std::type_info& targetType, 
+        const std::string& reason) :
+    MjcfError("Could not parse attribute string '" + source + " to " + targetType.name() + ".\n"
+              "Reason: " + reason)
+{}
+
+
+}
diff --git a/VirtualRobot/MJCF/elements/exceptions.h b/VirtualRobot/MJCF/elements/exceptions.h
new file mode 100644
index 000000000..93b207f3d
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/exceptions.h
@@ -0,0 +1,42 @@
+#pragma once
+
+#include <stdexcept>
+
+namespace mjcf
+{
+
+    class MjcfError : public std::runtime_error
+    {
+    public:
+        MjcfError(const std::string& message);
+    };
+
+
+    /// Indicates an error in load/saveFile.
+    class MjcfIOError : public MjcfError
+    {
+    public:
+        MjcfIOError(const std::string& message);
+    };
+
+    
+    /// Indicates that an attribute without default was not set when it was 
+    /// attempted to read.
+    class AttribNotSetError : public MjcfError
+    {
+    public:
+        AttribNotSetError(const std::string& name);
+    };
+    
+    /// Indicates that an attribute without default was not set when it was 
+    /// attempted to read.
+    class ParseAttributeError : public MjcfError
+    {
+    public:
+        ParseAttributeError(const std::string& source, const std::type_info& targetType,
+                            const std::string& reason);
+    };
+    
+
+
+}
diff --git a/VirtualRobot/MJCF/elements/has_member.hpp b/VirtualRobot/MJCF/elements/has_member.hpp
new file mode 100644
index 000000000..1f4f2239a
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/has_member.hpp
@@ -0,0 +1,34 @@
+// from: https://gist.github.com/maddouri/0da889b331d910f35e05ba3b7b9d869b
+
+// A compile-time method for checking the existence of a class member
+// @see https://general-purpose.io/2017/03/10/checking-the-existence-of-a-cpp-class-member-at-compile-time/
+
+// This code uses "decltype" which, according to http://en.cppreference.com/w/cpp/compiler_support
+// should be supported by Clang 2.9+, GCC 4.3+ and MSVC 2010+ (if you have an older compiler, please upgrade :)
+// As of "constexpr", if not supported by your compiler, you could try "const"
+// or use the value as an inner enum value e.g. enum { value = ... }
+
+// check "test_has_member.cpp" for a usage example
+
+/// Defines a "has_member_member_name" class template
+///
+/// This template can be used to check if its "T" argument
+/// has a data or function member called "member_name"
+#define define_has_member(member_name)                                         \
+    template <typename T>                                                      \
+    class has_member_##member_name                                             \
+    {                                                                          \
+        typedef char yes_type;                                                 \
+        typedef long no_type;                                                  \
+        template <typename U> static yes_type test(decltype(&U::member_name)); \
+        template <typename U> static no_type  test(...);                       \
+    public:                                                                    \
+        static constexpr bool value = sizeof(test<T>(0)) == sizeof(yes_type);  \
+    }
+
+/// Shorthand for testing if "class_" has a member called "member_name"
+///
+/// @note "define_has_member(member_name)" must be used
+///       before calling "has_member(class_, member_name)"
+#define has_member(class_, member_name)  has_member_##member_name<class_>::value
+
diff --git a/VirtualRobot/MJCF/elements/keyframe.cpp b/VirtualRobot/MJCF/elements/keyframe.cpp
new file mode 100644
index 000000000..5ba1e0678
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/keyframe.cpp
@@ -0,0 +1,9 @@
+#include "keyframe.h"
+
+//#include "../Document.h"
+
+
+using namespace mjcf;
+
+
+const std::string KeyframeSection::tag  = "keyframe";
diff --git a/VirtualRobot/MJCF/elements/keyframe.h b/VirtualRobot/MJCF/elements/keyframe.h
new file mode 100644
index 000000000..c4ff60b13
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/keyframe.h
@@ -0,0 +1,15 @@
+#pragma once
+
+#include "Attribute.h"
+
+namespace mjcf
+{
+
+struct KeyframeSection : public Element<KeyframeSection>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(KeyframeSection)
+};
+
+
+}
diff --git a/VirtualRobot/MJCF/elements/mjcf_utils.cpp b/VirtualRobot/MJCF/elements/mjcf_utils.cpp
new file mode 100644
index 000000000..5ce463791
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/mjcf_utils.cpp
@@ -0,0 +1,82 @@
+#include "mjcf_utils.h"
+
+#include <algorithm>
+#include <map>
+
+
+namespace mjcf
+{
+
+    ActuatorType toActuatorType(std::string str)
+    {
+        static const std::map<std::string, ActuatorType> map
+        {
+            {"motor",    ActuatorType::MOTOR},
+            {"position", ActuatorType::POSITION},
+            {"velocity", ActuatorType::VELOCITY}
+        };
+        std::transform(str.begin(), str.end(), str.begin(), ::tolower);
+        return map.at(str);
+    }
+
+    std::size_t commonPrefixLength(const std::string& a, const std::string& b)
+    {
+        const std::string* smaller = &a;
+        const std::string* bigger = &b;
+        if (b.size() < a.size())
+        {
+            std::swap(smaller, bigger);
+        }
+
+        auto mismatch = std::mismatch(smaller->begin(), smaller->end(),
+                                      bigger->begin()).first;
+        return std::size_t(std::distance(smaller->begin(), mismatch));
+    }
+    
+    template <>
+    std::string toAttr<bool>(const bool& b)
+    {
+        static const std::string strings[] = { "false", "true" };
+        return strings[int(b)];
+    }
+    
+
+    Eigen::Vector2f strToVec2(const char* string)
+    {
+        Eigen::Vector2f v;
+        sscanf(string, "%f %f", &v(0), &v(1));
+        return v;
+    }
+
+    Eigen::Vector3f strToVec3(const char* string)
+    {
+        Eigen::Vector3f v;
+        sscanf(string, "%f %f %f", &v(0), &v(1), &v(2));
+        return v;
+    }
+
+    Eigen::Quaternionf strToQuat(const char* string)
+    {
+        Eigen::Quaternionf q;
+        sscanf(string, "%f %f %f %f", &q.w(), &q.x(), &q.y(), &q.z());
+        return q;
+    }
+
+
+    
+    bool equals(const char* lhs, const char* rhs)
+    {
+        return strcmp(lhs, rhs) == 0;
+    }
+
+}
+
+bool std::operator==(const char* lhs, const std::string& rhs)
+{
+    return mjcf::equals(lhs, rhs.c_str());
+}
+
+bool std::operator==(const std::string& lhs, const char* rhs)
+{
+    return mjcf::equals(lhs.c_str(), rhs);
+}
diff --git a/VirtualRobot/MJCF/elements/mjcf_utils.h b/VirtualRobot/MJCF/elements/mjcf_utils.h
new file mode 100644
index 000000000..ceb155891
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/mjcf_utils.h
@@ -0,0 +1,213 @@
+#pragma once
+
+#include <string>
+#include <type_traits>
+
+#include <Eigen/Eigen>
+#include <boost/lexical_cast.hpp>
+
+#include "exceptions.h"
+
+
+namespace std
+{
+    bool operator==(const char* lhs, const string& rhs);
+    bool operator==(const string& lhs, const char* rhs);
+}
+
+namespace Eigen
+{
+    using Vector5f = Eigen::Matrix<float, 5, 1>;
+    using Vector6f = Eigen::Matrix<float, 6, 1>;
+    using Vector7f = Eigen::Matrix<float, 7, 1>;
+
+    template<typename Derived>
+    struct is_matrix_expression : 
+            std::is_base_of<Eigen::MatrixBase<typename std::decay<Derived>::type>, typename std::decay<Derived>::type>
+    {};
+    
+    template<typename Derived>
+    struct is_quaternion_expression : 
+            std::is_base_of<Eigen::QuaternionBase<typename std::decay<Derived>::type>, typename std::decay<Derived>::type>
+    {};
+    
+    template<typename Derived>
+    struct is_eigen_expression : 
+            is_matrix_expression<Derived>, is_quaternion_expression<Derived>
+    {};
+} 
+
+
+namespace mjcf
+{
+
+    enum class ActuatorType
+    {
+        MOTOR, POSITION, VELOCITY
+    };
+
+    ActuatorType toActuatorType(std::string str);
+
+    // Get lenght of common prefix of two strings (was used for mergin body names).
+    std::size_t commonPrefixLength(const std::string& a, const std::string& b);
+
+    bool equals(const char* lhs, const char* rhs);
+    
+    
+    // VALUE -> ATTRIBUTE
+    
+    // Convert to MJCF XML attribute format.
+    
+    template <typename AttrT,
+              typename std::enable_if<!Eigen::is_eigen_expression<AttrT>::value, AttrT>::type* = nullptr>
+    std::string toAttr(const AttrT& b);
+    template <>
+    std::string toAttr<bool>(const bool& b);
+    template <typename Derived>
+    std::string toAttr(const Eigen::MatrixBase<Derived>& mat);
+    template <typename Derived>
+    std::string toAttr(const Eigen::QuaternionBase<Derived>& quat);
+
+
+    // ATTRIBUTE -> VALUE
+    
+    // Convert from MJCF XML attribute.
+    Eigen::Vector2f strToVec2(const char* string);
+    Eigen::Vector3f strToVec3(const char* string);
+    Eigen::Quaternionf strToQuat(const char* string);
+    
+    
+    
+    /// Eigen Matrix and vectors
+    template <typename Derived> 
+    void fromAttr(const std::string& valueStr, Eigen::MatrixBase<Derived>& value);
+    /// Eigen Quaternions
+    template <typename Derived> 
+    void fromAttr(const std::string& valueStr, Eigen::QuaternionBase<Derived>& value);
+
+    /// Single values via boost::lexical cast. Only enabled for non-Eigen types.
+    template <typename AttrT,
+              typename std::enable_if<!Eigen::is_eigen_expression<AttrT>::value, AttrT>::type* = nullptr>
+    void fromAttr(const std::string& valueStr, AttrT& value);
+    
+    
+    template <typename Scalar>
+    std::vector<Scalar> parseCoeffs(const std::string& string, char delim = ' ');
+    
+    
+    // DEFINITIONS of templated methods
+
+    template <typename AttrT,
+              typename std::enable_if<!Eigen::is_eigen_expression<AttrT>::value, AttrT>::type*>
+    std::string toAttr(const AttrT& b)
+    {
+        static_assert (!Eigen::is_eigen_expression<AttrT>::value, "Resolved an Eigen type.");
+        static_assert (!std::is_same<bool, AttrT>::value, "Resolved bool.");
+        return boost::lexical_cast<std::string>(b);
+    }
+    
+    template<typename Derived>
+    std::string toAttr(const Eigen::MatrixBase<Derived>& mat)
+    {
+        static const Eigen::IOFormat iof
+        {
+            7, 0, "", " ", "", "", "", ""
+        };
+    
+        std::stringstream ss;
+        ss << mat.format(iof);
+        return ss.str();
+    }
+    
+    template<typename Derived>
+    std::string toAttr(const Eigen::QuaternionBase<Derived>& quat)
+    {
+        std::stringstream ss;
+        ss << quat.w() << " " << quat.x() << " " << quat.y() << " " << quat.z();
+        return ss.str();
+    }
+
+    
+    template <typename AttrT,
+              typename std::enable_if<!Eigen::is_eigen_expression<AttrT>::value, AttrT>::type*>
+    void fromAttr(const std::string& valueStr, AttrT& value)
+    {
+        static_assert (!Eigen::is_eigen_expression<AttrT>::value, "Resolved an Eigen type.");
+        value = boost::lexical_cast<AttrT>(valueStr);
+    }
+    
+    template <typename Derived> 
+    void fromAttr(const std::string& valueStr, Eigen::MatrixBase<Derived>& value)
+    {
+        using Matrix = Eigen::MatrixBase<Derived>;
+        using Scalar = typename Matrix::Scalar;
+        
+        std::vector<Scalar> coeffs;
+        try 
+        {
+            coeffs = parseCoeffs<Scalar>(valueStr);
+        }
+        catch (const std::bad_cast& e) 
+        {
+            throw mjcf::ParseAttributeError(valueStr, typeid(Matrix), e.what());
+        }
+        
+        if (value.size() >= 0 && static_cast<long>(coeffs.size()) != value.size())
+        {
+            throw mjcf::ParseAttributeError(valueStr, typeid(Matrix), "Number of coefficients does not match.");
+        }
+        
+        long i = 0;
+        for (auto& coeff : coeffs)
+        {
+            value(i) = coeff;
+        }
+    }
+    
+    template <typename Derived> 
+    void fromAttr(const std::string& valueStr, Eigen::QuaternionBase<Derived>& value)
+    {
+        using Quaternion = Eigen::QuaternionBase<Derived>;
+        using Scalar = typename Quaternion::Scalar;
+        std::vector<Scalar> coeffs;
+        
+        try 
+        {
+            coeffs = parseCoeffs<Scalar>(valueStr);
+        }
+        catch (const std::bad_cast& e) 
+        {
+            throw mjcf::ParseAttributeError(valueStr, typeid(Quaternion), e.what());
+        }
+        
+        
+        if (coeffs.size() != 4)
+        {
+            throw mjcf::ParseAttributeError(valueStr, typeid(Quaternion),
+                                            "Number of coefficients does not match.");
+        }
+        
+        value.w() = coeffs[0];
+        value.x() = coeffs[1];
+        value.y() = coeffs[2];
+        value.z() = coeffs[3];
+    }
+    
+    
+    template <typename Scalar>
+    std::vector<Scalar> parseCoeffs(const std::string& string, char delim)
+    {
+        std::vector<Scalar> coeffs;
+
+        std::stringstream ss(string);
+        for (std::string valueStr; std::getline(ss, valueStr, delim);)
+        {
+            // may throw boost::bad_lexical_cast (derives from std::bad_cast)
+            Scalar value = boost::lexical_cast<Scalar>(valueStr);
+            coeffs.push_back(value);
+        }
+
+        return coeffs;
+    }
+}
+
diff --git a/VirtualRobot/MJCF/elements/option.cpp b/VirtualRobot/MJCF/elements/option.cpp
new file mode 100644
index 000000000..a28581475
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/option.cpp
@@ -0,0 +1,16 @@
+#include "option.h"
+
+#include "../Document.h"
+
+
+using namespace mjcf;
+
+
+const std::string OptionFlag::tag    = "flag";
+const std::string OptionSection::tag = "option";
+
+
+OptionFlag OptionSection::flag()
+{
+    return getOrCreateChild<OptionFlag>();
+}
diff --git a/VirtualRobot/MJCF/elements/option.h b/VirtualRobot/MJCF/elements/option.h
new file mode 100644
index 000000000..b673dfcdd
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/option.h
@@ -0,0 +1,71 @@
+#pragma once
+
+#include "Attribute.h"
+#include "mjcf_utils.h"
+
+namespace mjcf
+{
+
+
+struct OptionFlag : public Element<OptionFlag>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(OptionFlag)
+};
+
+struct OptionSection : public Element<OptionSection>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(OptionSection)
+    
+    
+    /// Get (or create) the option/flag child.
+    OptionFlag flag();
+    
+    
+    mjcf_FloatAttributeDef(OptionSection, timestep, 0.002f);
+    mjcf_FloatAttributeDef(OptionSection, impratio, 1);
+    
+    mjcf_Vector3fAttributeDef(OptionSection, gravity, Eigen::Vector3f(0, 0, -9.81f));
+    
+    mjcf_Vector3fAttributeDef(OptionSection, wind, Eigen::Vector3f::Zero());
+    mjcf_Vector3fAttributeDef(OptionSection, magnetic, Eigen::Vector3f(0, -0.5, 0));
+    
+    mjcf_FloatAttributeDef(OptionSection, density, 0);
+    mjcf_FloatAttributeDef(OptionSection, viscosity, 0);
+    
+    mjcf_FloatAttributeDef(OptionSection,             o_margin, 0);
+    mjcf_Vector2fAttributeDef(OptionSection,          o_solref, Eigen::Vector2f(0.02, 1));
+    mjcf_AttributeDef(OptionSection, Eigen::Vector5f, o_solimp, o_solimp_default());
+    
+    // [Euler, RK4]
+    mjcf_StringAttributeDef(OptionSection, integrator, "Euler");
+    // [all, predefined, dynamic]
+    mjcf_StringAttributeDef(OptionSection, collision, "all");
+    // [pyramidal, elliptic]
+    mjcf_StringAttributeDef(OptionSection, cone, "pyramidal");
+    // [dense, sparse, auto]
+    mjcf_StringAttributeDef(OptionSection, jacobian, "auto");
+    // [PGS, CG, Newton]
+    mjcf_StringAttributeDef(OptionSection, solver, "Newton");
+    
+    mjcf_IntAttributeDef(OptionSection,   iterations, 100);
+    mjcf_FloatAttributeDef(OptionSection, tolerance,  1e-8f);
+    mjcf_IntAttributeDef(OptionSection,   noslip_iterations, 0);
+    mjcf_FloatAttributeDef(OptionSection, noslip_tolerance,  1e-6f);
+    mjcf_IntAttributeDef(OptionSection,   mpr_iterations, 50);
+    mjcf_FloatAttributeDef(OptionSection, mpr_tolerance,  1e-6f);
+    
+private:
+    
+    static Eigen::Vector5f o_solimp_default()
+    {
+        Eigen::Vector5f v;
+        v << 0.9f, 0.95f, 0.001f, 0.5, 2;
+        return v;
+    }
+};
+
+
+
+}
diff --git a/VirtualRobot/MJCF/elements/sensor.cpp b/VirtualRobot/MJCF/elements/sensor.cpp
new file mode 100644
index 000000000..4d9a02cef
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/sensor.cpp
@@ -0,0 +1,9 @@
+#include "sensor.h"
+
+//#include "../Document.h"
+
+
+using namespace mjcf;
+
+
+const std::string SensorSection::tag    = "sensor";
diff --git a/VirtualRobot/MJCF/elements/sensor.h b/VirtualRobot/MJCF/elements/sensor.h
new file mode 100644
index 000000000..f9122689c
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/sensor.h
@@ -0,0 +1,14 @@
+#pragma once
+
+#include "Attribute.h"
+
+namespace mjcf
+{
+
+struct SensorSection : public Element<SensorSection>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(SensorSection)
+};
+
+}
diff --git a/VirtualRobot/MJCF/elements/size.cpp b/VirtualRobot/MJCF/elements/size.cpp
new file mode 100644
index 000000000..79e6a61c5
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/size.cpp
@@ -0,0 +1,8 @@
+#include "size.h"
+
+//#include "../Document.h"
+
+
+using namespace mjcf;
+
+const std::string SizeSection::tag      = "size";
diff --git a/VirtualRobot/MJCF/elements/size.h b/VirtualRobot/MJCF/elements/size.h
new file mode 100644
index 000000000..1dffaa63e
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/size.h
@@ -0,0 +1,16 @@
+#pragma once
+
+#include "Attribute.h"
+
+namespace mjcf
+{
+
+struct SizeSection : public Element<SizeSection>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(SizeSection)
+};
+
+
+
+}
diff --git a/VirtualRobot/MJCF/elements/statistic.cpp b/VirtualRobot/MJCF/elements/statistic.cpp
new file mode 100644
index 000000000..5dfd99ff8
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/statistic.cpp
@@ -0,0 +1,9 @@
+#include "statistic.h"
+
+//#include "../Document.h"
+
+
+using namespace mjcf;
+
+
+const std::string StatisticSection::tag = "statistic";
diff --git a/VirtualRobot/MJCF/elements/statistic.h b/VirtualRobot/MJCF/elements/statistic.h
new file mode 100644
index 000000000..543b6d113
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/statistic.h
@@ -0,0 +1,15 @@
+#pragma once
+
+#include "Attribute.h"
+
+namespace mjcf
+{
+
+struct StatisticSection : public Element<StatisticSection>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(StatisticSection)
+};
+
+
+}
diff --git a/VirtualRobot/MJCF/elements/tendon.cpp b/VirtualRobot/MJCF/elements/tendon.cpp
new file mode 100644
index 000000000..a9148a1bb
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/tendon.cpp
@@ -0,0 +1,10 @@
+#include "tendon.h"
+
+//#include "../Document.h"
+
+
+using namespace mjcf;
+
+
+const std::string TendonSection::tag    = "tendon";
+
diff --git a/VirtualRobot/MJCF/elements/tendon.h b/VirtualRobot/MJCF/elements/tendon.h
new file mode 100644
index 000000000..671cfa636
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/tendon.h
@@ -0,0 +1,16 @@
+#pragma once
+
+#include "Attribute.h"
+
+namespace mjcf
+{
+
+
+struct TendonSection : public Element<TendonSection>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(TendonSection)
+};
+
+
+}
diff --git a/VirtualRobot/MJCF/elements/visual.cpp b/VirtualRobot/MJCF/elements/visual.cpp
new file mode 100644
index 000000000..72dd1bb81
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/visual.cpp
@@ -0,0 +1,9 @@
+#include "visual.h"
+
+//#include "../Document.h"
+
+
+using namespace mjcf;
+
+
+const std::string VisualSection::tag    = "visual";
diff --git a/VirtualRobot/MJCF/elements/visual.h b/VirtualRobot/MJCF/elements/visual.h
new file mode 100644
index 000000000..d52ffe2a6
--- /dev/null
+++ b/VirtualRobot/MJCF/elements/visual.h
@@ -0,0 +1,14 @@
+#pragma once
+
+#include "Attribute.h"
+
+namespace mjcf
+{
+
+struct VisualSection : public Element<VisualSection>
+{
+    static const std::string tag;
+    mjcf_ElementDerivedConstructors(VisualSection)
+};
+
+}
-- 
GitLab