#include "PendingNodeMonitorCache.h"
#include "PoseView.h"
const bigtime_t kDelayedNodeMonitorLifetime = 10000000;
PendingNodeMonitorEntry::PendingNodeMonitorEntry(const node_ref* node,
const BMessage* nodeMonitor)
:
fExpiresAfter(system_time() + kDelayedNodeMonitorLifetime),
fNodeMonitor(*nodeMonitor),
fNode(*node)
{
}
const BMessage*
PendingNodeMonitorEntry::NodeMonitor() const
{
return &fNodeMonitor;
}
bool
PendingNodeMonitorEntry::Match(const node_ref* node) const
{
return fNode == *node;
}
bool
PendingNodeMonitorEntry::TooOld(bigtime_t now) const
{
return now > fExpiresAfter;
}
PendingNodeMonitorCache::PendingNodeMonitorCache()
:
fList(10)
{
}
PendingNodeMonitorCache::~PendingNodeMonitorCache()
{
}
void
PendingNodeMonitorCache::Add(const BMessage* message)
{
#if xDEBUG
PRINT(("adding pending node monitor\n"));
message->PrintToStream();
#endif
node_ref node;
if (message->FindInt32("device", &node.device) != B_OK
|| message->FindInt64("node", (int64*)&node.node) != B_OK)
return;
fList.AddItem(new PendingNodeMonitorEntry(&node, message));
}
void
PendingNodeMonitorCache::RemoveEntries(const node_ref* nodeRef)
{
int32 count = fList.CountItems();
for (int32 index = count - 1; index >= 0; index--)
if (fList.ItemAt(index)->Match(nodeRef))
delete fList.RemoveItemAt(index);
}
void
PendingNodeMonitorCache::RemoveOldEntries()
{
bigtime_t now = system_time();
int32 count = fList.CountItems();
for (int32 index = count - 1; index >= 0; index--)
if (fList.ItemAt(index)->TooOld(now)) {
PRINT(("removing old entry from pending node monitor cache\n"));
delete fList.RemoveItemAt(index);
}
}
void
PendingNodeMonitorCache::PoseCreatedOrMoved(BPoseView* poseView,
const BPose* pose)
{
bigtime_t now = system_time();
for (int32 index = 0; index < fList.CountItems();) {
PendingNodeMonitorEntry* item = fList.ItemAt(index);
if (item->TooOld(now)) {
PRINT(("removing old entry from pending node monitor cache\n"));
delete fList.RemoveItemAt(index);
} else if (item->Match(pose->TargetModel()->NodeRef())) {
fList.RemoveItemAt(index);
#if DEBUG
PRINT(("reapplying node monitor for model:\n"));
pose->TargetModel()->PrintToStream();
item->NodeMonitor()->PrintToStream();
bool result =
#endif
poseView->FSNotification(item->NodeMonitor());
ASSERT(result);
delete item;
} else
index++;
}
}