| /* | 
 |  * Copyright (C) 2019 The Android Open Source Project | 
 |  * | 
 |  * Licensed under the Apache License, Version 2.0 (the "License"); | 
 |  * you may not use this file except in compliance with the License. | 
 |  * You may obtain a copy of the License at | 
 |  * | 
 |  *      http://www.apache.org/licenses/LICENSE-2.0 | 
 |  * | 
 |  * Unless required by applicable law or agreed to in writing, software | 
 |  * distributed under the License is distributed on an "AS IS" BASIS, | 
 |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 |  * See the License for the specific language governing permissions and | 
 |  * limitations under the License. | 
 |  */ | 
 |  | 
 | #include "mocks/MockSprite.h" | 
 | #include "mocks/MockSpriteController.h" | 
 |  | 
 | #include <input/PointerController.h> | 
 | #include <input/SpriteController.h> | 
 |  | 
 | #include <atomic> | 
 | #include <gmock/gmock.h> | 
 | #include <gtest/gtest.h> | 
 | #include <thread> | 
 |  | 
 | namespace android { | 
 |  | 
 | enum TestCursorType { | 
 |     CURSOR_TYPE_DEFAULT = 0, | 
 |     CURSOR_TYPE_HOVER, | 
 |     CURSOR_TYPE_TOUCH, | 
 |     CURSOR_TYPE_ANCHOR, | 
 |     CURSOR_TYPE_ADDITIONAL, | 
 |     CURSOR_TYPE_ADDITIONAL_ANIM, | 
 |     CURSOR_TYPE_CUSTOM = -1, | 
 | }; | 
 |  | 
 | using ::testing::AllOf; | 
 | using ::testing::Field; | 
 | using ::testing::Mock; | 
 | using ::testing::NiceMock; | 
 | using ::testing::Return; | 
 | using ::testing::Test; | 
 |  | 
 | std::pair<float, float> getHotSpotCoordinatesForType(int32_t type) { | 
 |     return std::make_pair(type * 10, type * 10 + 5); | 
 | } | 
 |  | 
 | class MockPointerControllerPolicyInterface : public PointerControllerPolicyInterface { | 
 | public: | 
 |     virtual void loadPointerIcon(SpriteIcon* icon, int32_t displayId) override; | 
 |     virtual void loadPointerResources(PointerResources* outResources, int32_t displayId) override; | 
 |     virtual void loadAdditionalMouseResources(std::map<int32_t, SpriteIcon>* outResources, | 
 |             std::map<int32_t, PointerAnimation>* outAnimationResources, int32_t displayId) override; | 
 |     virtual int32_t getDefaultPointerIconId() override; | 
 |     virtual int32_t getCustomPointerIconId() override; | 
 |  | 
 |     bool allResourcesAreLoaded(); | 
 |     bool noResourcesAreLoaded(); | 
 |  | 
 | private: | 
 |     void loadPointerIconForType(SpriteIcon* icon, int32_t cursorType); | 
 |  | 
 |     bool pointerIconLoaded{false}; | 
 |     bool pointerResourcesLoaded{false}; | 
 |     bool additionalMouseResourcesLoaded{false}; | 
 | }; | 
 |  | 
 | void MockPointerControllerPolicyInterface::loadPointerIcon(SpriteIcon* icon, int32_t) { | 
 |     loadPointerIconForType(icon, CURSOR_TYPE_DEFAULT); | 
 |     pointerIconLoaded = true; | 
 | } | 
 |  | 
 | void MockPointerControllerPolicyInterface::loadPointerResources(PointerResources* outResources, | 
 |         int32_t) { | 
 |     loadPointerIconForType(&outResources->spotHover, CURSOR_TYPE_HOVER); | 
 |     loadPointerIconForType(&outResources->spotTouch, CURSOR_TYPE_TOUCH); | 
 |     loadPointerIconForType(&outResources->spotAnchor, CURSOR_TYPE_ANCHOR); | 
 |     pointerResourcesLoaded = true; | 
 | } | 
 |  | 
 | void MockPointerControllerPolicyInterface::loadAdditionalMouseResources( | 
 |         std::map<int32_t, SpriteIcon>* outResources, | 
 |         std::map<int32_t, PointerAnimation>* outAnimationResources, | 
 |         int32_t) { | 
 |     SpriteIcon icon; | 
 |     PointerAnimation anim; | 
 |  | 
 |     // CURSOR_TYPE_ADDITIONAL doesn't have animation resource. | 
 |     int32_t cursorType = CURSOR_TYPE_ADDITIONAL; | 
 |     loadPointerIconForType(&icon, cursorType); | 
 |     (*outResources)[cursorType] = icon; | 
 |  | 
 |     // CURSOR_TYPE_ADDITIONAL_ANIM has animation resource. | 
 |     cursorType = CURSOR_TYPE_ADDITIONAL_ANIM; | 
 |     loadPointerIconForType(&icon, cursorType); | 
 |     anim.animationFrames.push_back(icon); | 
 |     anim.durationPerFrame = 10; | 
 |     (*outResources)[cursorType] = icon; | 
 |     (*outAnimationResources)[cursorType] = anim; | 
 |  | 
 |     additionalMouseResourcesLoaded = true; | 
 | } | 
 |  | 
 | int32_t MockPointerControllerPolicyInterface::getDefaultPointerIconId() { | 
 |     return CURSOR_TYPE_DEFAULT; | 
 | } | 
 |  | 
 | int32_t MockPointerControllerPolicyInterface::getCustomPointerIconId() { | 
 |     return CURSOR_TYPE_CUSTOM; | 
 | } | 
 |  | 
 | bool MockPointerControllerPolicyInterface::allResourcesAreLoaded() { | 
 |     return pointerIconLoaded && pointerResourcesLoaded && additionalMouseResourcesLoaded; | 
 | } | 
 |  | 
 | bool MockPointerControllerPolicyInterface::noResourcesAreLoaded() { | 
 |     return !(pointerIconLoaded || pointerResourcesLoaded || additionalMouseResourcesLoaded); | 
 | } | 
 |  | 
 | void MockPointerControllerPolicyInterface::loadPointerIconForType(SpriteIcon* icon, int32_t type) { | 
 |     icon->style = type; | 
 |     std::pair<float, float> hotSpot = getHotSpotCoordinatesForType(type); | 
 |     icon->hotSpotX = hotSpot.first; | 
 |     icon->hotSpotY = hotSpot.second; | 
 | } | 
 | class PointerControllerTest : public Test { | 
 | protected: | 
 |     PointerControllerTest(); | 
 |     ~PointerControllerTest(); | 
 |  | 
 |     void ensureDisplayViewportIsSet(); | 
 |  | 
 |     sp<MockSprite> mPointerSprite; | 
 |     sp<MockPointerControllerPolicyInterface> mPolicy; | 
 |     sp<MockSpriteController> mSpriteController; | 
 |     std::shared_ptr<PointerController> mPointerController; | 
 |  | 
 | private: | 
 |     void loopThread(); | 
 |  | 
 |     std::atomic<bool> mRunning = true; | 
 |     class MyLooper : public Looper { | 
 |     public: | 
 |         MyLooper() : Looper(false) {} | 
 |         ~MyLooper() = default; | 
 |     }; | 
 |     sp<MyLooper> mLooper; | 
 |     std::thread mThread; | 
 | }; | 
 |  | 
 | PointerControllerTest::PointerControllerTest() : mPointerSprite(new NiceMock<MockSprite>), | 
 |         mLooper(new MyLooper), mThread(&PointerControllerTest::loopThread, this) { | 
 |  | 
 |     mSpriteController = new NiceMock<MockSpriteController>(mLooper); | 
 |     mPolicy = new MockPointerControllerPolicyInterface(); | 
 |  | 
 |     EXPECT_CALL(*mSpriteController, createSprite()) | 
 |             .WillOnce(Return(mPointerSprite)); | 
 |  | 
 |     mPointerController = PointerController::create(mPolicy, mLooper, mSpriteController); | 
 | } | 
 |  | 
 | PointerControllerTest::~PointerControllerTest() { | 
 |     mRunning.store(false, std::memory_order_relaxed); | 
 |     mThread.join(); | 
 | } | 
 |  | 
 | void PointerControllerTest::ensureDisplayViewportIsSet() { | 
 |     DisplayViewport viewport; | 
 |     viewport.displayId = ADISPLAY_ID_DEFAULT; | 
 |     viewport.logicalRight = 1600; | 
 |     viewport.logicalBottom = 1200; | 
 |     viewport.physicalRight = 800; | 
 |     viewport.physicalBottom = 600; | 
 |     viewport.deviceWidth = 400; | 
 |     viewport.deviceHeight = 300; | 
 |     mPointerController->setDisplayViewport(viewport); | 
 | } | 
 |  | 
 | void PointerControllerTest::loopThread() { | 
 |     Looper::setForThread(mLooper); | 
 |  | 
 |     while (mRunning.load(std::memory_order_relaxed)) { | 
 |         mLooper->pollOnce(100); | 
 |     } | 
 | } | 
 |  | 
 | TEST_F(PointerControllerTest, useDefaultCursorTypeByDefault) { | 
 |     ensureDisplayViewportIsSet(); | 
 |     mPointerController->unfade(PointerController::Transition::IMMEDIATE); | 
 |  | 
 |     std::pair<float, float> hotspot = getHotSpotCoordinatesForType(CURSOR_TYPE_DEFAULT); | 
 |     EXPECT_CALL(*mPointerSprite, setVisible(true)); | 
 |     EXPECT_CALL(*mPointerSprite, setAlpha(1.0f)); | 
 |     EXPECT_CALL(*mPointerSprite, setIcon( | 
 |             AllOf( | 
 |                     Field(&SpriteIcon::style, CURSOR_TYPE_DEFAULT), | 
 |                     Field(&SpriteIcon::hotSpotX, hotspot.first), | 
 |                     Field(&SpriteIcon::hotSpotY, hotspot.second)))); | 
 |     mPointerController->reloadPointerResources(); | 
 | } | 
 |  | 
 | TEST_F(PointerControllerTest, updatePointerIcon) { | 
 |     ensureDisplayViewportIsSet(); | 
 |     mPointerController->setPresentation(PointerController::Presentation::POINTER); | 
 |     mPointerController->unfade(PointerController::Transition::IMMEDIATE); | 
 |  | 
 |     int32_t type = CURSOR_TYPE_ADDITIONAL; | 
 |     std::pair<float, float> hotspot = getHotSpotCoordinatesForType(type); | 
 |     EXPECT_CALL(*mPointerSprite, setVisible(true)); | 
 |     EXPECT_CALL(*mPointerSprite, setAlpha(1.0f)); | 
 |     EXPECT_CALL(*mPointerSprite, setIcon( | 
 |             AllOf( | 
 |                     Field(&SpriteIcon::style, type), | 
 |                     Field(&SpriteIcon::hotSpotX, hotspot.first), | 
 |                     Field(&SpriteIcon::hotSpotY, hotspot.second)))); | 
 |     mPointerController->updatePointerIcon(type); | 
 | } | 
 |  | 
 | TEST_F(PointerControllerTest, setCustomPointerIcon) { | 
 |     ensureDisplayViewportIsSet(); | 
 |     mPointerController->unfade(PointerController::Transition::IMMEDIATE); | 
 |  | 
 |     int32_t style = CURSOR_TYPE_CUSTOM; | 
 |     float hotSpotX = 15; | 
 |     float hotSpotY = 20; | 
 |  | 
 |     SpriteIcon icon; | 
 |     icon.style = style; | 
 |     icon.hotSpotX = hotSpotX; | 
 |     icon.hotSpotY = hotSpotY; | 
 |  | 
 |     EXPECT_CALL(*mPointerSprite, setVisible(true)); | 
 |     EXPECT_CALL(*mPointerSprite, setAlpha(1.0f)); | 
 |     EXPECT_CALL(*mPointerSprite, setIcon( | 
 |             AllOf( | 
 |                     Field(&SpriteIcon::style, style), | 
 |                     Field(&SpriteIcon::hotSpotX, hotSpotX), | 
 |                     Field(&SpriteIcon::hotSpotY, hotSpotY)))); | 
 |     mPointerController->setCustomPointerIcon(icon); | 
 | } | 
 |  | 
 | TEST_F(PointerControllerTest, doesNotGetResourcesBeforeSettingViewport) { | 
 |     mPointerController->setPresentation(PointerController::Presentation::POINTER); | 
 |     mPointerController->setPosition(1.0f, 1.0f); | 
 |     mPointerController->move(1.0f, 1.0f); | 
 |     mPointerController->unfade(PointerController::Transition::IMMEDIATE); | 
 |     mPointerController->fade(PointerController::Transition::IMMEDIATE); | 
 |  | 
 |     EXPECT_TRUE(mPolicy->noResourcesAreLoaded()); | 
 |  | 
 |     ensureDisplayViewportIsSet(); | 
 | } | 
 |  | 
 | }  // namespace android |