| /* |
| * Copyright (C) 2010 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. |
| */ |
| |
| #ifndef _UI_POINTER_CONTROLLER_H |
| #define _UI_POINTER_CONTROLLER_H |
| |
| #include <PointerControllerInterface.h> |
| #include <gui/DisplayEventReceiver.h> |
| #include <gui/WindowInfosUpdate.h> |
| #include <input/DisplayViewport.h> |
| #include <input/Input.h> |
| #include <utils/BitSet.h> |
| #include <utils/Looper.h> |
| #include <utils/RefBase.h> |
| |
| #include <map> |
| #include <memory> |
| #include <string> |
| #include <vector> |
| |
| #include "MouseCursorController.h" |
| #include "PointerControllerContext.h" |
| #include "SpriteController.h" |
| #include "TouchSpotController.h" |
| |
| namespace android { |
| |
| /* |
| * Tracks pointer movements and draws the pointer sprite to a surface. |
| * |
| * Handles pointer acceleration and animation. |
| */ |
| class PointerController : public PointerControllerInterface { |
| public: |
| static std::shared_ptr<PointerController> create( |
| const sp<PointerControllerPolicyInterface>& policy, const sp<Looper>& looper, |
| SpriteController& spriteController, bool enabled, |
| ControllerType type = ControllerType::LEGACY); |
| |
| ~PointerController() override; |
| |
| std::optional<FloatRect> getBounds() const override; |
| void move(float deltaX, float deltaY) override; |
| void setPosition(float x, float y) override; |
| FloatPoint getPosition() const override; |
| int32_t getDisplayId() const override; |
| void fade(Transition transition) override; |
| void unfade(Transition transition) override; |
| void setDisplayViewport(const DisplayViewport& viewport) override; |
| |
| void setPresentation(Presentation presentation) override; |
| void setSpots(const PointerCoords* spotCoords, const uint32_t* spotIdToIndex, |
| BitSet32 spotIdBits, int32_t displayId) override; |
| void clearSpots() override; |
| void updatePointerIcon(PointerIconStyle iconId) override; |
| void setCustomPointerIcon(const SpriteIcon& icon) override; |
| |
| virtual void setInactivityTimeout(InactivityTimeout inactivityTimeout); |
| void doInactivityTimeout(); |
| void reloadPointerResources(); |
| void onDisplayViewportsUpdated(const std::vector<DisplayViewport>& viewports); |
| |
| void onDisplayInfosChangedLocked(const std::vector<gui::DisplayInfo>& displayInfos) |
| REQUIRES(getLock()); |
| |
| std::string dump() override; |
| |
| protected: |
| using WindowListenerConsumer = |
| std::function<void(const sp<android::gui::WindowInfosListener>&)>; |
| |
| // Constructor used to test WindowInfosListener registration. |
| PointerController(const sp<PointerControllerPolicyInterface>& policy, const sp<Looper>& looper, |
| SpriteController& spriteController, bool enabled, |
| WindowListenerConsumer registerListener, |
| WindowListenerConsumer unregisterListener); |
| |
| PointerController(const sp<PointerControllerPolicyInterface>& policy, const sp<Looper>& looper, |
| SpriteController& spriteController, bool enabled); |
| |
| private: |
| friend PointerControllerContext::LooperCallback; |
| friend PointerControllerContext::MessageHandler; |
| |
| // PointerController's DisplayInfoListener can outlive the PointerController because when the |
| // listener is registered, a strong pointer to the listener (which can extend its lifecycle) |
| // is given away. To avoid the small overhead of using two separate locks in these two objects, |
| // we use the DisplayInfoListener's lock in PointerController. |
| std::mutex& getLock() const; |
| |
| const bool mEnabled; |
| |
| PointerControllerContext mContext; |
| |
| MouseCursorController mCursorController; |
| |
| struct Locked { |
| Presentation presentation; |
| int32_t pointerDisplayId = ADISPLAY_ID_NONE; |
| |
| std::vector<gui::DisplayInfo> mDisplayInfos; |
| std::unordered_map<int32_t /* displayId */, TouchSpotController> spotControllers; |
| } mLocked GUARDED_BY(getLock()); |
| |
| class DisplayInfoListener : public gui::WindowInfosListener { |
| public: |
| explicit DisplayInfoListener(PointerController* pc) : mPointerController(pc){}; |
| void onWindowInfosChanged(const gui::WindowInfosUpdate&) override; |
| void onPointerControllerDestroyed(); |
| |
| // This lock is also used by PointerController. See PointerController::getLock(). |
| std::mutex mLock; |
| |
| private: |
| PointerController* mPointerController GUARDED_BY(mLock); |
| }; |
| |
| sp<DisplayInfoListener> mDisplayInfoListener; |
| const WindowListenerConsumer mUnregisterWindowInfosListener; |
| |
| const ui::Transform& getTransformForDisplayLocked(int displayId) const REQUIRES(getLock()); |
| |
| void clearSpotsLocked() REQUIRES(getLock()); |
| }; |
| |
| class MousePointerController : public PointerController { |
| public: |
| /** A version of PointerController that controls one mouse pointer. */ |
| MousePointerController(const sp<PointerControllerPolicyInterface>& policy, |
| const sp<Looper>& looper, SpriteController& spriteController, |
| bool enabled); |
| |
| ~MousePointerController() override; |
| |
| void setPresentation(Presentation) override { |
| LOG_ALWAYS_FATAL("Should not be called"); |
| } |
| void setSpots(const PointerCoords*, const uint32_t*, BitSet32, int32_t) override { |
| LOG_ALWAYS_FATAL("Should not be called"); |
| } |
| void clearSpots() override { |
| LOG_ALWAYS_FATAL("Should not be called"); |
| } |
| }; |
| |
| class TouchPointerController : public PointerController { |
| public: |
| /** A version of PointerController that controls touch spots. */ |
| TouchPointerController(const sp<PointerControllerPolicyInterface>& policy, |
| const sp<Looper>& looper, SpriteController& spriteController, |
| bool enabled); |
| |
| ~TouchPointerController() override; |
| |
| std::optional<FloatRect> getBounds() const override { |
| LOG_ALWAYS_FATAL("Should not be called"); |
| } |
| void move(float, float) override { |
| LOG_ALWAYS_FATAL("Should not be called"); |
| } |
| void setPosition(float, float) override { |
| LOG_ALWAYS_FATAL("Should not be called"); |
| } |
| FloatPoint getPosition() const override { |
| LOG_ALWAYS_FATAL("Should not be called"); |
| } |
| int32_t getDisplayId() const override { |
| LOG_ALWAYS_FATAL("Should not be called"); |
| } |
| void fade(Transition) override { |
| LOG_ALWAYS_FATAL("Should not be called"); |
| } |
| void unfade(Transition) override { |
| LOG_ALWAYS_FATAL("Should not be called"); |
| } |
| void setDisplayViewport(const DisplayViewport&) override { |
| LOG_ALWAYS_FATAL("Should not be called"); |
| } |
| void setPresentation(Presentation) override { |
| LOG_ALWAYS_FATAL("Should not be called"); |
| } |
| void updatePointerIcon(PointerIconStyle) override { |
| LOG_ALWAYS_FATAL("Should not be called"); |
| } |
| void setCustomPointerIcon(const SpriteIcon&) override { |
| LOG_ALWAYS_FATAL("Should not be called"); |
| } |
| // fade() should not be called by inactivity timeout. Do nothing. |
| void setInactivityTimeout(InactivityTimeout) override {} |
| }; |
| |
| class StylusPointerController : public PointerController { |
| public: |
| /** A version of PointerController that controls one stylus pointer. */ |
| StylusPointerController(const sp<PointerControllerPolicyInterface>& policy, |
| const sp<Looper>& looper, SpriteController& spriteController, |
| bool enabled); |
| |
| ~StylusPointerController() override; |
| |
| void setPresentation(Presentation) override { |
| LOG_ALWAYS_FATAL("Should not be called"); |
| } |
| void setSpots(const PointerCoords*, const uint32_t*, BitSet32, int32_t) override { |
| LOG_ALWAYS_FATAL("Should not be called"); |
| } |
| void clearSpots() override { |
| LOG_ALWAYS_FATAL("Should not be called"); |
| } |
| }; |
| |
| } // namespace android |
| |
| #endif // _UI_POINTER_CONTROLLER_H |