diff --git a/python/mesa_pd.py b/python/mesa_pd.py
index d3e062a4f3603090763585531e537d990935e1bc..16d089ceecf24094e773f69c6e799258fe54f7f2 100755
--- a/python/mesa_pd.py
+++ b/python/mesa_pd.py
@@ -63,6 +63,10 @@ if __name__ == '__main__':
     ps.add_property("oldHydrodynamicTorque", "walberla::mesa_pd::Vec3", defValue="real_t(0)",
                     syncMode="ON_OWNERSHIP_CHANGE")
 
+    # properties for VBond model
+    ps.add_property("clusterID", "int64_t", defValue="-1", syncMode="ON_GHOST_CREATION")
+    ps.add_property("segmentID", "int64_t", defValue="-1", syncMode="ON_GHOST_CREATION")
+
     ch = mpd.add(data.ContactHistory())
     ch.add_property("tangentialSpringDisplacement", "walberla::mesa_pd::Vec3", defValue="real_t(0)")
     ch.add_property("isSticking", "bool", defValue="false")
diff --git a/src/mesa_pd/data/ParticleAccessor.h b/src/mesa_pd/data/ParticleAccessor.h
index 4da6f59ee87d1ab5a3370f7efe2aaf568da12cb0..4ebdcd135cc07827d47cb45e769b4f79b9909d07 100644
--- a/src/mesa_pd/data/ParticleAccessor.h
+++ b/src/mesa_pd/data/ParticleAccessor.h
@@ -164,6 +164,14 @@ public:
    walberla::mesa_pd::Vec3& getOldHydrodynamicTorqueRef(const size_t p_idx) {return ps_->getOldHydrodynamicTorqueRef(p_idx);}
    void setOldHydrodynamicTorque(const size_t p_idx, walberla::mesa_pd::Vec3 const & v) { ps_->setOldHydrodynamicTorque(p_idx, v);}
    
+   int64_t const & getClusterID(const size_t p_idx) const {return ps_->getClusterID(p_idx);}
+   int64_t& getClusterIDRef(const size_t p_idx) {return ps_->getClusterIDRef(p_idx);}
+   void setClusterID(const size_t p_idx, int64_t const & v) { ps_->setClusterID(p_idx, v);}
+   
+   int64_t const & getSegmentID(const size_t p_idx) const {return ps_->getSegmentID(p_idx);}
+   int64_t& getSegmentIDRef(const size_t p_idx) {return ps_->getSegmentIDRef(p_idx);}
+   void setSegmentID(const size_t p_idx, int64_t const & v) { ps_->setSegmentID(p_idx, v);}
+   
    std::unordered_set<walberla::mpi::MPIRank> const & getNeighborState(const size_t p_idx) const {return ps_->getNeighborState(p_idx);}
    std::unordered_set<walberla::mpi::MPIRank>& getNeighborStateRef(const size_t p_idx) {return ps_->getNeighborStateRef(p_idx);}
    void setNeighborState(const size_t p_idx, std::unordered_set<walberla::mpi::MPIRank> const & v) { ps_->setNeighborState(p_idx, v);}
@@ -329,6 +337,14 @@ public:
    void setOldHydrodynamicTorque(const size_t /*p_idx*/, walberla::mesa_pd::Vec3 const & v) { oldHydrodynamicTorque_ = v;}
    walberla::mesa_pd::Vec3& getOldHydrodynamicTorqueRef(const size_t /*p_idx*/) {return oldHydrodynamicTorque_;}
    
+   int64_t const & getClusterID(const size_t /*p_idx*/) const {return clusterID_;}
+   void setClusterID(const size_t /*p_idx*/, int64_t const & v) { clusterID_ = v;}
+   int64_t& getClusterIDRef(const size_t /*p_idx*/) {return clusterID_;}
+   
+   int64_t const & getSegmentID(const size_t /*p_idx*/) const {return segmentID_;}
+   void setSegmentID(const size_t /*p_idx*/, int64_t const & v) { segmentID_ = v;}
+   int64_t& getSegmentIDRef(const size_t /*p_idx*/) {return segmentID_;}
+   
    std::unordered_set<walberla::mpi::MPIRank> const & getNeighborState(const size_t /*p_idx*/) const {return neighborState_;}
    void setNeighborState(const size_t /*p_idx*/, std::unordered_set<walberla::mpi::MPIRank> const & v) { neighborState_ = v;}
    std::unordered_set<walberla::mpi::MPIRank>& getNeighborStateRef(const size_t /*p_idx*/) {return neighborState_;}
@@ -373,6 +389,8 @@ private:
    walberla::mesa_pd::Vec3 hydrodynamicTorque_;
    walberla::mesa_pd::Vec3 oldHydrodynamicForce_;
    walberla::mesa_pd::Vec3 oldHydrodynamicTorque_;
+   int64_t clusterID_;
+   int64_t segmentID_;
    std::unordered_set<walberla::mpi::MPIRank> neighborState_;
 };
 
diff --git a/src/mesa_pd/data/ParticleStorage.h b/src/mesa_pd/data/ParticleStorage.h
index 33def90f5af99df204d17c4a19ad6abc2aa4c090..81d628681345bacbc293eab25bbfc573bda53af2 100644
--- a/src/mesa_pd/data/ParticleStorage.h
+++ b/src/mesa_pd/data/ParticleStorage.h
@@ -100,6 +100,8 @@ public:
       using hydrodynamicTorque_type = walberla::mesa_pd::Vec3;
       using oldHydrodynamicForce_type = walberla::mesa_pd::Vec3;
       using oldHydrodynamicTorque_type = walberla::mesa_pd::Vec3;
+      using clusterID_type = int64_t;
+      using segmentID_type = int64_t;
       using neighborState_type = std::unordered_set<walberla::mpi::MPIRank>;
 
       
@@ -219,6 +221,14 @@ public:
       oldHydrodynamicTorque_type& getOldHydrodynamicTorqueRef() {return storage_.getOldHydrodynamicTorqueRef(i_);}
       void setOldHydrodynamicTorque(oldHydrodynamicTorque_type const & v) { storage_.setOldHydrodynamicTorque(i_, v);}
       
+      clusterID_type const & getClusterID() const {return storage_.getClusterID(i_);}
+      clusterID_type& getClusterIDRef() {return storage_.getClusterIDRef(i_);}
+      void setClusterID(clusterID_type const & v) { storage_.setClusterID(i_, v);}
+      
+      segmentID_type const & getSegmentID() const {return storage_.getSegmentID(i_);}
+      segmentID_type& getSegmentIDRef() {return storage_.getSegmentIDRef(i_);}
+      void setSegmentID(segmentID_type const & v) { storage_.setSegmentID(i_, v);}
+      
       neighborState_type const & getNeighborState() const {return storage_.getNeighborState(i_);}
       neighborState_type& getNeighborStateRef() {return storage_.getNeighborStateRef(i_);}
       void setNeighborState(neighborState_type const & v) { storage_.setNeighborState(i_, v);}
@@ -312,6 +322,8 @@ public:
    using hydrodynamicTorque_type = walberla::mesa_pd::Vec3;
    using oldHydrodynamicForce_type = walberla::mesa_pd::Vec3;
    using oldHydrodynamicTorque_type = walberla::mesa_pd::Vec3;
+   using clusterID_type = int64_t;
+   using segmentID_type = int64_t;
    using neighborState_type = std::unordered_set<walberla::mpi::MPIRank>;
 
    
@@ -431,6 +443,14 @@ public:
    oldHydrodynamicTorque_type& getOldHydrodynamicTorqueRef(const size_t idx) {return oldHydrodynamicTorque_[idx];}
    void setOldHydrodynamicTorque(const size_t idx, oldHydrodynamicTorque_type const & v) { oldHydrodynamicTorque_[idx] = v; }
    
+   clusterID_type const & getClusterID(const size_t idx) const {return clusterID_[idx];}
+   clusterID_type& getClusterIDRef(const size_t idx) {return clusterID_[idx];}
+   void setClusterID(const size_t idx, clusterID_type const & v) { clusterID_[idx] = v; }
+   
+   segmentID_type const & getSegmentID(const size_t idx) const {return segmentID_[idx];}
+   segmentID_type& getSegmentIDRef(const size_t idx) {return segmentID_[idx];}
+   void setSegmentID(const size_t idx, segmentID_type const & v) { segmentID_[idx] = v; }
+   
    neighborState_type const & getNeighborState(const size_t idx) const {return neighborState_[idx];}
    neighborState_type& getNeighborStateRef(const size_t idx) {return neighborState_[idx];}
    void setNeighborState(const size_t idx, neighborState_type const & v) { neighborState_[idx] = v; }
@@ -555,6 +575,8 @@ public:
    std::vector<hydrodynamicTorque_type> hydrodynamicTorque_ {};
    std::vector<oldHydrodynamicForce_type> oldHydrodynamicForce_ {};
    std::vector<oldHydrodynamicTorque_type> oldHydrodynamicTorque_ {};
+   std::vector<clusterID_type> clusterID_ {};
+   std::vector<segmentID_type> segmentID_ {};
    std::vector<neighborState_type> neighborState_ {};
    std::unordered_map<uid_type, size_t> uidToIdx_;
    static_assert(std::is_same<uid_type, id_t>::value,
@@ -594,6 +616,8 @@ ParticleStorage::Particle& ParticleStorage::Particle::operator=(const ParticleSt
    getHydrodynamicTorqueRef() = rhs.getHydrodynamicTorque();
    getOldHydrodynamicForceRef() = rhs.getOldHydrodynamicForce();
    getOldHydrodynamicTorqueRef() = rhs.getOldHydrodynamicTorque();
+   getClusterIDRef() = rhs.getClusterID();
+   getSegmentIDRef() = rhs.getSegmentID();
    getNeighborStateRef() = rhs.getNeighborState();
    return *this;
 }
@@ -630,6 +654,8 @@ ParticleStorage::Particle& ParticleStorage::Particle::operator=(ParticleStorage:
    getHydrodynamicTorqueRef() = std::move(rhs.getHydrodynamicTorqueRef());
    getOldHydrodynamicForceRef() = std::move(rhs.getOldHydrodynamicForceRef());
    getOldHydrodynamicTorqueRef() = std::move(rhs.getOldHydrodynamicTorqueRef());
+   getClusterIDRef() = std::move(rhs.getClusterIDRef());
+   getSegmentIDRef() = std::move(rhs.getSegmentIDRef());
    getNeighborStateRef() = std::move(rhs.getNeighborStateRef());
    return *this;
 }
@@ -667,6 +693,8 @@ void swap(ParticleStorage::Particle lhs, ParticleStorage::Particle rhs)
    std::swap(lhs.getHydrodynamicTorqueRef(), rhs.getHydrodynamicTorqueRef());
    std::swap(lhs.getOldHydrodynamicForceRef(), rhs.getOldHydrodynamicForceRef());
    std::swap(lhs.getOldHydrodynamicTorqueRef(), rhs.getOldHydrodynamicTorqueRef());
+   std::swap(lhs.getClusterIDRef(), rhs.getClusterIDRef());
+   std::swap(lhs.getSegmentIDRef(), rhs.getSegmentIDRef());
    std::swap(lhs.getNeighborStateRef(), rhs.getNeighborStateRef());
 }
 
@@ -704,6 +732,8 @@ std::ostream& operator<<( std::ostream& os, const ParticleStorage::Particle& p )
          "hydrodynamicTorque  : " << p.getHydrodynamicTorque() << "\n" <<
          "oldHydrodynamicForce: " << p.getOldHydrodynamicForce() << "\n" <<
          "oldHydrodynamicTorque: " << p.getOldHydrodynamicTorque() << "\n" <<
+         "clusterID           : " << p.getClusterID() << "\n" <<
+         "segmentID           : " << p.getSegmentID() << "\n" <<
          "neighborState       : " << p.getNeighborState() << "\n" <<
          "================================" << std::endl;
    return os;
@@ -811,6 +841,8 @@ inline ParticleStorage::iterator ParticleStorage::create(const id_t& uid)
    hydrodynamicTorque_.emplace_back(real_t(0));
    oldHydrodynamicForce_.emplace_back(real_t(0));
    oldHydrodynamicTorque_.emplace_back(real_t(0));
+   clusterID_.emplace_back(-1);
+   segmentID_.emplace_back(-1);
    neighborState_.emplace_back();
    uid_.back() = uid;
    uidToIdx_[uid] = uid_.size() - 1;
@@ -873,6 +905,8 @@ inline ParticleStorage::iterator ParticleStorage::erase(iterator& it)
    hydrodynamicTorque_.pop_back();
    oldHydrodynamicForce_.pop_back();
    oldHydrodynamicTorque_.pop_back();
+   clusterID_.pop_back();
+   segmentID_.pop_back();
    neighborState_.pop_back();
    return it;
 }
@@ -922,6 +956,8 @@ inline void ParticleStorage::reserve(const size_t size)
    hydrodynamicTorque_.reserve(size);
    oldHydrodynamicForce_.reserve(size);
    oldHydrodynamicTorque_.reserve(size);
+   clusterID_.reserve(size);
+   segmentID_.reserve(size);
    neighborState_.reserve(size);
 }
 
@@ -956,6 +992,8 @@ inline void ParticleStorage::clear()
    hydrodynamicTorque_.clear();
    oldHydrodynamicForce_.clear();
    oldHydrodynamicTorque_.clear();
+   clusterID_.clear();
+   segmentID_.clear();
    neighborState_.clear();
    uidToIdx_.clear();
 }
@@ -991,6 +1029,8 @@ inline size_t ParticleStorage::size() const
    //WALBERLA_ASSERT_EQUAL( uid_.size(), hydrodynamicTorque.size() );
    //WALBERLA_ASSERT_EQUAL( uid_.size(), oldHydrodynamicForce.size() );
    //WALBERLA_ASSERT_EQUAL( uid_.size(), oldHydrodynamicTorque.size() );
+   //WALBERLA_ASSERT_EQUAL( uid_.size(), clusterID.size() );
+   //WALBERLA_ASSERT_EQUAL( uid_.size(), segmentID.size() );
    //WALBERLA_ASSERT_EQUAL( uid_.size(), neighborState.size() );
    return uid_.size();
 }
@@ -1419,6 +1459,24 @@ public:
    walberla::mesa_pd::Vec3 const & operator()(const data::Particle& p) const {return p.getOldHydrodynamicTorque();}
 };
 ///Predicate that selects a certain property from a Particle
+class SelectParticleClusterID
+{
+public:
+   using return_type = int64_t;
+   int64_t& operator()(data::Particle& p) const {return p.getClusterIDRef();}
+   int64_t& operator()(data::Particle&& p) const {return p.getClusterIDRef();}
+   int64_t const & operator()(const data::Particle& p) const {return p.getClusterID();}
+};
+///Predicate that selects a certain property from a Particle
+class SelectParticleSegmentID
+{
+public:
+   using return_type = int64_t;
+   int64_t& operator()(data::Particle& p) const {return p.getSegmentIDRef();}
+   int64_t& operator()(data::Particle&& p) const {return p.getSegmentIDRef();}
+   int64_t const & operator()(const data::Particle& p) const {return p.getSegmentID();}
+};
+///Predicate that selects a certain property from a Particle
 class SelectParticleNeighborState
 {
 public:
diff --git a/src/mesa_pd/mpi/notifications/ParticleCopyNotification.h b/src/mesa_pd/mpi/notifications/ParticleCopyNotification.h
index c17b5ec796bba6af558dc21b0821313a3abf28bc..f0944ef6dc366541b335811870764b6473cc4a92 100644
--- a/src/mesa_pd/mpi/notifications/ParticleCopyNotification.h
+++ b/src/mesa_pd/mpi/notifications/ParticleCopyNotification.h
@@ -69,6 +69,8 @@ public:
       walberla::mesa_pd::Vec3 hydrodynamicTorque {real_t(0)};
       walberla::mesa_pd::Vec3 oldHydrodynamicForce {real_t(0)};
       walberla::mesa_pd::Vec3 oldHydrodynamicTorque {real_t(0)};
+      int64_t clusterID {-1};
+      int64_t segmentID {-1};
    };
 
    inline explicit ParticleCopyNotification( const data::Particle& particle ) : particle_(particle) {}
@@ -101,6 +103,8 @@ inline data::ParticleStorage::iterator createNewParticle(data::ParticleStorage&
    pIt->setHydrodynamicTorque(data.hydrodynamicTorque);
    pIt->setOldHydrodynamicForce(data.oldHydrodynamicForce);
    pIt->setOldHydrodynamicTorque(data.oldHydrodynamicTorque);
+   pIt->setClusterID(data.clusterID);
+   pIt->setSegmentID(data.segmentID);
    return pIt;
 }
 
@@ -148,6 +152,8 @@ mpi::GenericSendBuffer<T,G>& operator<<( mpi::GenericSendBuffer<T,G> & buf, cons
    buf << obj.particle_.getHydrodynamicTorque();
    buf << obj.particle_.getOldHydrodynamicForce();
    buf << obj.particle_.getOldHydrodynamicTorque();
+   buf << obj.particle_.getClusterID();
+   buf << obj.particle_.getSegmentID();
    return buf;
 }
 
@@ -176,6 +182,8 @@ mpi::GenericRecvBuffer<T>& operator>>( mpi::GenericRecvBuffer<T> & buf, mesa_pd:
    buf >> objparam.hydrodynamicTorque;
    buf >> objparam.oldHydrodynamicForce;
    buf >> objparam.oldHydrodynamicTorque;
+   buf >> objparam.clusterID;
+   buf >> objparam.segmentID;
    return buf;
 }
 
diff --git a/src/mesa_pd/mpi/notifications/ParticleGhostCopyNotification.h b/src/mesa_pd/mpi/notifications/ParticleGhostCopyNotification.h
index c9e3b02ba988c5f5decaa06f306da8f8eb5b9a6f..b9e6aa9663c96aa4b24d74c513c66be63fcea6e3 100644
--- a/src/mesa_pd/mpi/notifications/ParticleGhostCopyNotification.h
+++ b/src/mesa_pd/mpi/notifications/ParticleGhostCopyNotification.h
@@ -62,6 +62,8 @@ public:
       uint_t type {0};
       std::map<walberla::id_t, walberla::mesa_pd::data::ContactHistory> oldContactHistory {};
       walberla::real_t temperature {real_t(0)};
+      int64_t clusterID {-1};
+      int64_t segmentID {-1};
    };
 
    inline explicit ParticleGhostCopyNotification( const data::Particle& particle ) : particle_(particle) {}
@@ -87,6 +89,8 @@ inline data::ParticleStorage::iterator createNewParticle(data::ParticleStorage&
    pIt->setType(data.type);
    pIt->setOldContactHistory(data.oldContactHistory);
    pIt->setTemperature(data.temperature);
+   pIt->setClusterID(data.clusterID);
+   pIt->setSegmentID(data.segmentID);
    return pIt;
 }
 
@@ -127,6 +131,8 @@ mpi::GenericSendBuffer<T,G>& operator<<( mpi::GenericSendBuffer<T,G> & buf, cons
    buf << obj.particle_.getType();
    buf << obj.particle_.getOldContactHistory();
    buf << obj.particle_.getTemperature();
+   buf << obj.particle_.getClusterID();
+   buf << obj.particle_.getSegmentID();
    return buf;
 }
 
@@ -148,6 +154,8 @@ mpi::GenericRecvBuffer<T>& operator>>( mpi::GenericRecvBuffer<T> & buf, mesa_pd:
    buf >> objparam.type;
    buf >> objparam.oldContactHistory;
    buf >> objparam.temperature;
+   buf >> objparam.clusterID;
+   buf >> objparam.segmentID;
    return buf;
 }