Browse Source

MouseCursor: Simplify implementation using shared_ptr instead of manual refcounting

v6.1.6
reuk 4 years ago
parent
commit
5c50ee4f94
2 changed files with 35 additions and 92 deletions
  1. +34
    -90
      modules/juce_gui_basics/mouse/juce_MouseCursor.cpp
  2. +1
    -2
      modules/juce_gui_basics/mouse/juce_MouseCursor.h

+ 34
- 90
modules/juce_gui_basics/mouse/juce_MouseCursor.cpp View File

@@ -45,96 +45,65 @@ class MouseCursor::SharedCursorHandle
{
public:
explicit SharedCursorHandle (const MouseCursor::StandardCursorType type)
: handle (createStandardMouseCursor (type)),
standardType (type),
isStandard (true)
: handle (createStandardMouseCursor (type), Deleter { true }),
standardType (type)
{
}
SharedCursorHandle (const Image& image, Point<int> hotSpot, float scaleFactor)
: info (new CustomMouseCursorInfo (image, hotSpot, scaleFactor)),
handle (info->create()),
standardType (MouseCursor::NormalCursor),
isStandard (false)
: info (std::make_unique<CustomMouseCursorInfo> (image, hotSpot, scaleFactor)),
handle (info->create(), Deleter { false }),
standardType (MouseCursor::NormalCursor)
{
// your hotspot needs to be within the bounds of the image!
jassert (image.getBounds().contains (hotSpot));
}
~SharedCursorHandle()
static std::shared_ptr<SharedCursorHandle> createStandard (const MouseCursor::StandardCursorType type)
{
deleteMouseCursor (handle, isStandard);
}
static SharedCursorHandle* createStandard (const MouseCursor::StandardCursorType type)
{
jassert (isPositiveAndBelow (type, MouseCursor::NumStandardCursorTypes));
if (! isPositiveAndBelow (type, MouseCursor::NumStandardCursorTypes))
return nullptr;
const SpinLock::ScopedLockType sl (lock);
auto& c = getSharedCursor (type);
static SpinLock mutex;
static std::array<std::weak_ptr<SharedCursorHandle>, MouseCursor::NumStandardCursorTypes> cursors;
if (c == nullptr)
c = new SharedCursorHandle (type);
else
c->retain();
const SpinLock::ScopedLockType sl (mutex);
return c;
}
auto& weak = cursors[type];
bool isStandardType (MouseCursor::StandardCursorType type) const noexcept
{
return type == standardType && isStandard;
}
if (auto strong = weak.lock())
return strong;
SharedCursorHandle* retain() noexcept
{
++refCount;
return this;
auto strong = std::make_shared<SharedCursorHandle> (type);
weak = strong;
return strong;
}
void release()
bool isStandardType (MouseCursor::StandardCursorType type) const noexcept
{
if (--refCount == 0)
{
if (isStandard)
{
const SpinLock::ScopedLockType sl (lock);
getSharedCursor (standardType) = nullptr;
}
delete this;
}
return type == standardType && handle.get_deleter().isStandard;
}
void* getHandle() const noexcept { return handle; }
void setHandle (void* newHandle) { handle = newHandle; }
void* getHandle() const noexcept { return handle.get(); }
MouseCursor::StandardCursorType getType() const noexcept { return standardType; }
CustomMouseCursorInfo* getCustomInfo() const noexcept { return info.get(); }
private:
struct Deleter
{
void operator() (void* ptr) const noexcept { deleteMouseCursor (ptr, isStandard); }
const bool isStandard;
};
std::unique_ptr<CustomMouseCursorInfo> info;
void* handle;
Atomic<int> refCount { 1 };
std::unique_ptr<void, Deleter> handle;
const MouseCursor::StandardCursorType standardType;
const bool isStandard;
static SpinLock lock;
static SharedCursorHandle*& getSharedCursor (const MouseCursor::StandardCursorType type)
{
static SharedCursorHandle* cursors[MouseCursor::NumStandardCursorTypes] = {};
return cursors[type];
}
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (SharedCursorHandle)
};
SpinLock MouseCursor::SharedCursorHandle::lock;
//==============================================================================
MouseCursor::MouseCursor() noexcept
{
}
MouseCursor::MouseCursor() noexcept = default;
MouseCursor::MouseCursor (const StandardCursorType type)
: cursorHandle (type != MouseCursor::NormalCursor ? SharedCursorHandle::createStandard (type) : nullptr)
@@ -147,44 +116,19 @@ MouseCursor::MouseCursor (const Image& image, int hotSpotX, int hotSpotY)
}
MouseCursor::MouseCursor (const Image& image, int hotSpotX, int hotSpotY, float scaleFactor)
: cursorHandle (new SharedCursorHandle (image, { hotSpotX, hotSpotY }, scaleFactor))
: cursorHandle (std::make_shared<SharedCursorHandle> (image, Point<int> { hotSpotX, hotSpotY }, scaleFactor))
{
}
MouseCursor::MouseCursor (const MouseCursor& other)
: cursorHandle (other.cursorHandle == nullptr ? nullptr : other.cursorHandle->retain())
{
}
MouseCursor::MouseCursor (const MouseCursor&) = default;
MouseCursor::~MouseCursor()
{
if (cursorHandle != nullptr)
cursorHandle->release();
}
MouseCursor::~MouseCursor() = default;
MouseCursor& MouseCursor::operator= (const MouseCursor& other)
{
if (other.cursorHandle != nullptr)
other.cursorHandle->retain();
MouseCursor& MouseCursor::operator= (const MouseCursor&) = default;
if (cursorHandle != nullptr)
cursorHandle->release();
MouseCursor::MouseCursor (MouseCursor&&) noexcept = default;
cursorHandle = other.cursorHandle;
return *this;
}
MouseCursor::MouseCursor (MouseCursor&& other) noexcept
: cursorHandle (other.cursorHandle)
{
other.cursorHandle = nullptr;
}
MouseCursor& MouseCursor::operator= (MouseCursor&& other) noexcept
{
std::swap (cursorHandle, other.cursorHandle);
return *this;
}
MouseCursor& MouseCursor::operator= (MouseCursor&&) noexcept = default;
bool MouseCursor::operator== (const MouseCursor& other) const noexcept
{


+ 1
- 2
modules/juce_gui_basics/mouse/juce_MouseCursor.h View File

@@ -166,8 +166,7 @@ public:
private:
//==============================================================================
class SharedCursorHandle;
friend class SharedCursorHandle;
SharedCursorHandle* cursorHandle = nullptr;
std::shared_ptr<SharedCursorHandle> cursorHandle;
friend class MouseInputSourceInternal;
void showInWindow (ComponentPeer*) const;


Loading…
Cancel
Save