Overte C++ Documentation
EntityTree.h
1 //
2 // EntityTree.h
3 // libraries/entities/src
4 //
5 // Created by Brad Hefta-Gaub on 12/4/13.
6 // Copyright 2013 High Fidelity, Inc.
7 //
8 // Distributed under the Apache License, Version 2.0.
9 // See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html
10 //
11 
12 #ifndef hifi_EntityTree_h
13 #define hifi_EntityTree_h
14 
15 #include <QSet>
16 #include <QVector>
17 
18 #include <HelperScriptEngine.h>
19 #include <Octree.h>
20 #include <SpatialParentFinder.h>
21 
22 #include "AddEntityOperator.h"
23 #include "EntityTreeElement.h"
24 #include "DeleteEntityOperator.h"
25 #include "MovingEntitiesOperator.h"
26 
27 class EntityTree;
28 using EntityTreePointer = std::shared_ptr<EntityTree>;
29 
30 class EntitySimulation;
31 
32 namespace EntityQueryFilterSymbol {
33  static const QString NonDefault = "+";
34 }
35 
36 class NewlyCreatedEntityHook {
37 public:
38  virtual void entityCreated(const EntityItem& newEntity, const SharedNodePointer& senderNode) = 0;
39 };
40 
41 class SendEntitiesOperationArgs {
42 public:
43  glm::vec3 root;
44  QString entityHostType;
45  EntityTree* ourTree;
46  EntityTreePointer otherTree;
47  QHash<EntityItemID, EntityItemID>* map;
48 };
49 
50 class EntityTree : public Octree, public SpatialParentTree {
51  Q_OBJECT
52 public:
53  enum FilterType {
54  Add,
55  Edit,
56  Physics,
57  Delete
58  };
59  EntityTree(bool shouldReaverage = false);
60  virtual ~EntityTree();
61 
62  void createRootElement();
63 
64 
65  void setEntityMaxTmpLifetime(float maxTmpEntityLifetime) { _maxTmpEntityLifetime = maxTmpEntityLifetime; }
66  void setEntityScriptSourceWhitelist(const QString& entityScriptSourceWhitelist);
67 
69  virtual OctreeElementPointer createNewElement(unsigned char* octalCode = NULL) override;
70 
72  EntityTreeElementPointer getRoot() {
73  if (!_rootElement) {
74  createRootElement();
75  }
76  return std::static_pointer_cast<EntityTreeElement>(_rootElement);
77  }
78 
79 
80  virtual void eraseDomainAndNonOwnedEntities() override;
81  virtual void eraseAllOctreeElements(bool createNewRoot = true) override;
82 
83  virtual void readBitstreamToTree(const unsigned char* bitstream,
84  uint64_t bufferSizeBytes, ReadBitstreamToTreeParams& args) override;
85  int readEntityDataFromBuffer(const unsigned char* data, int bytesLeftToRead, ReadBitstreamToTreeParams& args);
86 
87  // These methods will allow the OctreeServer to send your tree inbound edit packets of your
88  // own definition. Implement these to allow your octree based server to support editing
89  virtual PacketType expectedDataPacketType() const override { return PacketType::EntityData; }
90  virtual bool handlesEditPacketType(PacketType packetType) const override;
91  void fixupTerseEditLogging(EntityItemProperties& properties, QList<QString>& changedProperties);
92  virtual int processEditPacketData(ReceivedMessage& message, const unsigned char* editData, int maxLength,
93  const SharedNodePointer& senderNode) override;
94 
95  virtual EntityItemID evalRayIntersection(const glm::vec3& origin, const glm::vec3& direction,
96  QVector<EntityItemID> entityIdsToInclude, QVector<EntityItemID> entityIdsToDiscard,
97  PickFilter searchFilter, OctreeElementPointer& element, float& distance,
98  BoxFace& face, glm::vec3& surfaceNormal, QVariantMap& extraInfo,
99  Octree::lockType lockType = Octree::TryLock, bool* accurateResult = NULL);
100 
101  virtual EntityItemID evalParabolaIntersection(const PickParabola& parabola,
102  QVector<EntityItemID> entityIdsToInclude, QVector<EntityItemID> entityIdsToDiscard,
103  PickFilter searchFilter, OctreeElementPointer& element, glm::vec3& intersection,
104  float& distance, float& parabolicDistance, BoxFace& face, glm::vec3& surfaceNormal, QVariantMap& extraInfo,
105  Octree::lockType lockType = Octree::TryLock, bool* accurateResult = NULL);
106 
107  virtual bool rootElementHasData() const override { return true; }
108 
109  virtual void releaseSceneEncodeData(OctreeElementExtraEncodeData* extraEncodeData) const override;
110 
111  // Why preUpdate() and update()?
112  // Because sometimes we need to do stuff between the two.
113  void preUpdate() override;
114  void update(bool simulate = true) override;
115 
116  // The newer API...
117  void postAddEntity(EntityItemPointer entityItem);
118 
119  EntityItemPointer addEntity(const EntityItemID& entityID, const EntityItemProperties& properties, bool isClone = false, const bool isImport = false);
120 
121  // use this method if you only know the entityID
122  bool updateEntity(const EntityItemID& entityID, const EntityItemProperties& properties, const SharedNodePointer& senderNode = SharedNodePointer(nullptr));
123 
124  // check if the avatar is a child of this entity, If so set the avatar parentID to null
125  void unhookChildAvatar(const EntityItemID entityID);
126  void cleanupCloneIDs(const EntityItemID& entityID);
127  void deleteEntity(const EntityItemID& entityID, bool force = false, bool ignoreWarnings = true);
128 
129  void deleteEntitiesByID(const std::vector<EntityItemID>& entityIDs, bool force = false, bool ignoreWarnings = true);
130  void deleteEntitiesByPointer(const std::vector<EntityItemPointer>& entities);
131 
132  EntityItemPointer findEntityByID(const QUuid& id) const;
133  EntityItemPointer findEntityByEntityItemID(const EntityItemID& entityID) const;
134  virtual SpatiallyNestablePointer findByID(const QUuid& id) const override { return findEntityByID(id); }
135 
136  EntityItemID assignEntityID(const EntityItemID& entityItemID);
137 
138  QUuid evalClosestEntity(const glm::vec3& position, float targetRadius, PickFilter searchFilter);
139  void evalEntitiesInSphere(const glm::vec3& center, float radius, PickFilter searchFilter, QVector<QUuid>& foundEntities);
140  void evalEntitiesInSphereWithType(const glm::vec3& center, float radius, EntityTypes::EntityType type, PickFilter searchFilter, QVector<QUuid>& foundEntities);
141  void evalEntitiesInSphereWithName(const glm::vec3& center, float radius, const QString& name, bool caseSensitive, PickFilter searchFilter, QVector<QUuid>& foundEntities);
142  void evalEntitiesInCube(const AACube& cube, PickFilter searchFilter, QVector<QUuid>& foundEntities);
143  void evalEntitiesInBox(const AABox& box, PickFilter searchFilter, QVector<QUuid>& foundEntities);
144  void evalEntitiesInFrustum(const ViewFrustum& frustum, PickFilter searchFilter, QVector<QUuid>& foundEntities);
145 
146  void addNewlyCreatedHook(NewlyCreatedEntityHook* hook);
147  void removeNewlyCreatedHook(NewlyCreatedEntityHook* hook);
148 
149  bool hasAnyDeletedEntities() const {
150  QReadLocker locker(&_recentlyDeletedEntitiesLock);
151  return _recentlyDeletedEntityItemIDs.size() > 0;
152  }
153 
154  bool hasEntitiesDeletedSince(quint64 sinceTime);
155  static quint64 getAdjustedConsiderSince(quint64 sinceTime);
156 
157  QMultiMap<quint64, QUuid> getRecentlyDeletedEntityIDs() const {
158  QReadLocker locker(&_recentlyDeletedEntitiesLock);
159  return _recentlyDeletedEntityItemIDs;
160  }
161 
162  void forgetEntitiesDeletedBefore(quint64 sinceTime);
163 
164  int processEraseMessage(ReceivedMessage& message, const SharedNodePointer& sourceNode);
165  int processEraseMessageDetails(const QByteArray& buffer, const SharedNodePointer& sourceNode);
166  bool shouldEraseEntity(EntityItemID entityID, const SharedNodePointer& sourceNode);
167 
168 
169  EntityTreeElementPointer getContainingElement(const EntityItemID& entityItemID) /*const*/;
170  void addEntityMapEntry(EntityItemPointer entity);
171  void clearEntityMapEntry(const EntityItemID& id);
172  void debugDumpMap();
173  virtual void dumpTree() override;
174  virtual void pruneTree() override;
175 
176  static QByteArray remapActionDataIDs(QByteArray actionData, QHash<EntityItemID, EntityItemID>& map);
177 
178  QVector<EntityItemID> sendEntities(EntityEditPacketSender* packetSender, EntityTreePointer localTree,
179  const QString& entityHostType, float x, float y, float z);
180 
181  void entityChanged(EntityItemPointer entity);
182 
183  void emitEntityScriptChanging(const EntityItemID& entityItemID, bool reload);
184  void emitEntityServerScriptChanging(const EntityItemID& entityItemID, bool reload);
185 
186  void setSimulation(EntitySimulationPointer simulation);
187  EntitySimulationPointer getSimulation() const { return _simulation; }
188 
189  bool wantEditLogging() const { return _wantEditLogging; }
190  void setWantEditLogging(bool value) { _wantEditLogging = value; }
191 
192  bool wantTerseEditLogging() const { return _wantTerseEditLogging; }
193  void setWantTerseEditLogging(bool value) { _wantTerseEditLogging = value; }
194 
195  virtual bool writeToMap(QVariantMap& entityDescription, OctreeElementPointer element, bool skipDefaultValues,
196  bool skipThoseWithBadParents) override;
197  virtual bool readFromMap(QVariantMap& entityDescription, const bool isImport = false) override;
198  virtual bool writeToJSON(QString& jsonString, const OctreeElementPointer& element) override;
199 
200 
201  glm::vec3 getContentsDimensions();
202  float getContentsLargestDimension();
203 
204  virtual void resetEditStats() override {
205  _totalEditMessages = 0;
206  _totalUpdates = 0;
207  _totalCreates = 0;
208  _totalDecodeTime = 0;
209  _totalLookupTime = 0;
210  _totalUpdateTime = 0;
211  _totalCreateTime = 0;
212  _totalLoggingTime = 0;
213  }
214 
215  virtual quint64 getAverageDecodeTime() const override { return _totalEditMessages == 0 ? 0 : _totalDecodeTime / _totalEditMessages; }
216  virtual quint64 getAverageLookupTime() const override { return _totalEditMessages == 0 ? 0 : _totalLookupTime / _totalEditMessages; }
217  virtual quint64 getAverageUpdateTime() const override { return _totalUpdates == 0 ? 0 : _totalUpdateTime / _totalUpdates; }
218  virtual quint64 getAverageCreateTime() const override { return _totalCreates == 0 ? 0 : _totalCreateTime / _totalCreates; }
219  virtual quint64 getAverageLoggingTime() const override { return _totalEditMessages == 0 ? 0 : _totalLoggingTime / _totalEditMessages; }
220  virtual quint64 getAverageFilterTime() const override { return _totalEditMessages == 0 ? 0 : _totalFilterTime / _totalEditMessages; }
221 
222  void trackIncomingEntityLastEdited(quint64 lastEditedTime, int bytesRead);
223  quint64 getAverageEditDeltas() const
224  { return _totalTrackedEdits == 0 ? 0 : _totalEditDeltas / _totalTrackedEdits; }
225  quint64 getAverageEditBytes() const
226  { return _totalTrackedEdits == 0 ? 0 : _totalEditBytes / _totalTrackedEdits; }
227  quint64 getMaxEditDelta() const { return _maxEditDelta; }
228  quint64 getTotalTrackedEdits() const { return _totalTrackedEdits; }
229 
230  EntityTreePointer getThisPointer() { return std::static_pointer_cast<EntityTree>(shared_from_this()); }
231 
232  bool isDeletedEntity(const QUuid& id) {
233  QReadLocker locker(&_deletedEntitiesLock);
234  return _deletedEntityItemIDs.contains(id);
235  }
236 
237  // these are used to call through to EntityItems
238  Q_INVOKABLE int getJointIndex(const QUuid& entityID, const QString& name) const;
239  Q_INVOKABLE QStringList getJointNames(const QUuid& entityID) const;
240 
241  void knowAvatarID(const QUuid& avatarID);
242  void forgetAvatarID(const QUuid& avatarID);
243  void deleteDescendantsOfAvatar(const QUuid& avatarID);
244  void removeFromChildrenOfAvatars(EntityItemPointer entity);
245 
246  void addToNeedsParentFixupList(EntityItemPointer entity);
247 
248  void notifyNewCollisionSoundURL(const QString& newCollisionSoundURL, const EntityItemID& entityID);
249 
250  static const float DEFAULT_MAX_TMP_ENTITY_LIFETIME;
251 
252  QUuid getMyAvatarSessionUUID() { return _myAvatar ? _myAvatar->getSessionUUID() : QUuid(); }
253  void setMyAvatar(std::shared_ptr<AvatarData> myAvatar) { _myAvatar = myAvatar; }
254 
255  void swapStaleProxies(std::vector<int>& proxies) { proxies.swap(_staleProxies); }
256 
257  void setIsServerlessMode(bool value) { _serverlessDomain = value; }
258  bool isServerlessMode() const { return _serverlessDomain; }
259 
260  static void setGetEntityObjectOperator(std::function<QObject*(const QUuid&)> getEntityObjectOperator) { _getEntityObjectOperator = getEntityObjectOperator; }
261  static QObject* getEntityObject(const QUuid& id);
262 
263  static void setTextSizeOperator(std::function<QSizeF(const QUuid&, const QString&)> textSizeOperator) { _textSizeOperator = textSizeOperator; }
264  static QSizeF textSize(const QUuid& id, const QString& text);
265 
266  static void setEntityClicksCapturedOperator(std::function<bool()> areEntityClicksCapturedOperator) { _areEntityClicksCapturedOperator = areEntityClicksCapturedOperator; }
267  static bool areEntityClicksCaptured();
268 
269  static void setEmitScriptEventOperator(std::function<void(const QUuid&, const QVariant&)> emitScriptEventOperator) { _emitScriptEventOperator = emitScriptEventOperator; }
270  static void emitScriptEvent(const QUuid& id, const QVariant& message);
271 
272  static void setGetUnscaledDimensionsForIDOperator(std::function<glm::vec3(const QUuid&)> getUnscaledDimensionsForIDOperator) { _getUnscaledDimensionsForIDOperator = getUnscaledDimensionsForIDOperator; }
273  static glm::vec3 getUnscaledDimensionsForID(const QUuid& id);
274 
275  std::map<QString, QString> getNamedPaths() const { return _namedPaths; }
276 
277  void updateEntityQueryAACube(SpatiallyNestablePointer object, EntityEditPacketSender* packetSender,
278  bool force, bool tellServer);
279 
280 signals:
281  void deletingEntity(const EntityItemID& entityID);
282  void deletingEntityPointer(EntityItem* entityID);
283  void addingEntity(const EntityItemID& entityID);
284  void addingEntityPointer(EntityItem* entityID);
285  void editingEntityPointer(const EntityItemPointer& entityID);
286  void entityScriptChanging(const EntityItemID& entityItemID, const bool reload);
287  void entityServerScriptChanging(const EntityItemID& entityItemID, const bool reload);
288  void newCollisionSoundURL(const QUrl& url, const EntityItemID& entityID);
289  void clearingEntities();
290  void killChallengeOwnershipTimeoutTimer(const EntityItemID& certID);
291 
292 protected:
293 
294  void recursivelyFilterAndCollectForDelete(const EntityItemPointer& entity, std::vector<EntityItemPointer>& entitiesToDelete, bool force) const;
295  void processRemovedEntities(const DeleteEntityOperator& theOperator);
296  bool updateEntity(EntityItemPointer entity, const EntityItemProperties& properties,
297  const SharedNodePointer& senderNode = SharedNodePointer(nullptr));
298  static bool sendEntitiesOperation(const OctreeElementPointer& element, void* extraData);
299  static void bumpTimestamp(EntityItemProperties& properties);
300 
301  void notifyNewlyCreatedEntity(const EntityItem& newEntity, const SharedNodePointer& senderNode);
302 
303  bool isScriptInWhitelist(const QString& scriptURL);
304 
305  QReadWriteLock _newlyCreatedHooksLock;
306  QVector<NewlyCreatedEntityHook*> _newlyCreatedHooks;
307 
308  mutable QReadWriteLock _recentlyDeletedEntitiesLock;
309  QMultiMap<quint64, QUuid> _recentlyDeletedEntityItemIDs;
310 
311  mutable QReadWriteLock _deletedEntitiesLock;
312  QSet<QUuid> _deletedEntityItemIDs;
313 
314  void clearDeletedEntities() {
315  QWriteLocker locker(&_deletedEntitiesLock);
316  _deletedEntityItemIDs.clear();
317  }
318 
319  void trackDeletedEntity(const QUuid& id) {
320  QWriteLocker locker(&_deletedEntitiesLock);
321  _deletedEntityItemIDs << id;
322  }
323 
324  mutable QReadWriteLock _entityMapLock;
325  QHash<EntityItemID, EntityItemPointer> _entityMap;
326 
327  EntitySimulationPointer _simulation;
328 
329  bool _wantEditLogging = false;
330  bool _wantTerseEditLogging = false;
331 
332 
333  // some performance tracking properties - only used in server trees
334  int _totalEditMessages = 0;
335  int _totalUpdates = 0;
336  int _totalCreates = 0;
337  mutable quint64 _totalDecodeTime = 0;
338  mutable quint64 _totalLookupTime = 0;
339  mutable quint64 _totalUpdateTime = 0;
340  mutable quint64 _totalCreateTime = 0;
341  mutable quint64 _totalLoggingTime = 0;
342  mutable quint64 _totalFilterTime = 0;
343 
344  // these performance statistics are only used in the client
345  void resetClientEditStats();
346  int _totalTrackedEdits = 0;
347  quint64 _totalEditBytes = 0;
348  quint64 _totalEditDeltas = 0;
349  quint64 _maxEditDelta = 0;
350  quint64 _treeResetTime = 0;
351 
352  void fixupNeedsParentFixups(); // try to hook members of _needsParentFixup to parent instances
353  QVector<EntityItemWeakPointer> _needsParentFixup; // entites with a parentID but no (yet) known parent instance
354  mutable QReadWriteLock _needsParentFixupLock;
355 
356  std::mutex _avatarIDsLock;
357  // we maintain a list of avatarIDs to notice when an entity is a child of one.
358  QSet<QUuid> _avatarIDs; // IDs of avatars connected to entity server
359  std::mutex _childrenOfAvatarsLock;
360  QHash<QUuid, QSet<EntityItemID>> _childrenOfAvatars; // which entities are children of which avatars
361 
362  float _maxTmpEntityLifetime { DEFAULT_MAX_TMP_ENTITY_LIFETIME };
363 
364  bool filterProperties(const EntityItemPointer& existingEntity, EntityItemProperties& propertiesIn, EntityItemProperties& propertiesOut, bool& wasChanged, FilterType filterType) const;
365  bool _hasEntityEditFilter{ false };
366  QStringList _entityScriptSourceWhitelist;
367 
368  MovingEntitiesOperator _entityMover;
369  QHash<EntityItemID, EntityItemPointer> _entitiesToAdd;
370 
371 private:
372  std::shared_ptr<AvatarData> _myAvatar{ nullptr };
373 
374  static std::function<QObject*(const QUuid&)> _getEntityObjectOperator;
375  static std::function<QSizeF(const QUuid&, const QString&)> _textSizeOperator;
376  static std::function<bool()> _areEntityClicksCapturedOperator;
377  static std::function<void(const QUuid&, const QVariant&)> _emitScriptEventOperator;
378  static std::function<glm::vec3(const QUuid&)> _getUnscaledDimensionsForIDOperator;
379 
380  std::vector<int32_t> _staleProxies;
381 
382  bool _serverlessDomain { false };
383 
384  std::map<QString, QString> _namedPaths;
385 
386  // Return an AACube containing object and all its entity descendants
387  AACube updateEntityQueryAACubeWorker(SpatiallyNestablePointer object, EntityEditPacketSender* packetSender,
388  MovingEntitiesOperator& moveOperator, bool force, bool tellServer);
389 
390  // Script engine for writing entity tree data to and from JSON
391  HelperScriptEngine _helperScriptEngine;
392 };
393 
394 void convertGrabUserDataToProperties(EntityItemProperties& properties);
395 
396 #endif // hifi_EntityTree_h
Utility for processing, packing, queueing and sending of outbound edit voxel messages.
Definition: EntityEditPacketSender.h:25
Definition: EntityItem.h:82
Abstract ID for editing model items. Used in EntityItem JS API.
Definition: EntityItemID.h:28
Definition: EntityItemProperties.h:106
Provides a wrapper around script engine that does not have ScriptManager.
Definition: HelperScriptEngine.h:31