diff options
-rw-r--r-- | desktop/inc/lib/init.hxx | 201 | ||||
-rw-r--r-- | desktop/source/lib/init.cxx | 197 |
2 files changed, 207 insertions, 191 deletions
diff --git a/desktop/inc/lib/init.hxx b/desktop/inc/lib/init.hxx index 5d3297733edc..3a5c5b7333c9 100644 --- a/desktop/inc/lib/init.hxx +++ b/desktop/inc/lib/init.hxx @@ -27,202 +27,21 @@ class LOKInteractionHandler; namespace desktop { - class CallbackFlushHandler : public Idle + class DESKTOP_DLLPUBLIC CallbackFlushHandler : public Idle { public: - explicit CallbackFlushHandler(LibreOfficeKitCallback pCallback, void* pData) - : Idle( "lokit timer callback" ), - m_pCallback(pCallback), - m_pData(pData), - m_bPartTilePainting(false) - { - SetPriority(SchedulerPriority::POST_PAINT); - - // Add the states that are safe to skip duplicates on, - // even when not consequent. - m_states.emplace(LOK_CALLBACK_TEXT_SELECTION_START, "NIL"); - m_states.emplace(LOK_CALLBACK_TEXT_SELECTION_END, "NIL"); - m_states.emplace(LOK_CALLBACK_TEXT_SELECTION, "NIL"); - m_states.emplace(LOK_CALLBACK_GRAPHIC_SELECTION, "NIL"); - m_states.emplace(LOK_CALLBACK_INVALIDATE_VISIBLE_CURSOR, "NIL"); - m_states.emplace(LOK_CALLBACK_STATE_CHANGED, "NIL"); - m_states.emplace(LOK_CALLBACK_MOUSE_POINTER, "NIL"); - m_states.emplace(LOK_CALLBACK_CELL_CURSOR, "NIL"); - m_states.emplace(LOK_CALLBACK_CELL_FORMULA, "NIL"); - m_states.emplace(LOK_CALLBACK_CURSOR_VISIBLE, "NIL"); - m_states.emplace(LOK_CALLBACK_SET_PART, "NIL"); - - Start(); - } - - virtual ~CallbackFlushHandler() - { - Stop(); - - // We might have important notification (.uno:save?). - flush(); - } - - virtual void Invoke() override - { - flush(); - } - - static - void callback(const int type, const char* payload, void* data) - { - CallbackFlushHandler* self = static_cast<CallbackFlushHandler*>(data); - if (self) - { - self->queue(type, payload); - } - } - - void queue(const int type, const char* data) - { - const std::string payload(data ? data : "(nil)"); - if (m_bPartTilePainting) - { - // We drop notifications when this is set, except for important ones. - // When we issue a complex command (such as .uno:InsertAnnotation) - // there will be multiple notifications. On the first invalidation - // we will start painting, but other events will get fired - // while the complex command in question executes. - // We don't want to supress everything here on the wrong assumption - // that no new events are fired during painting. - if (type != LOK_CALLBACK_STATE_CHANGED && - type != LOK_CALLBACK_INVALIDATE_TILES && - type != LOK_CALLBACK_INVALIDATE_VISIBLE_CURSOR && - type != LOK_CALLBACK_CURSOR_VISIBLE && - type != LOK_CALLBACK_TEXT_SELECTION) - { - //SAL_WARN("lokevt", "Skipping while painting [" + std::to_string(type) + "]: [" + payload + "]."); - return; - } - } - - // Supress invalid payloads. - if (type == LOK_CALLBACK_INVALIDATE_VISIBLE_CURSOR && - payload.find(", 0, 0, ") != std::string::npos) - { - // The cursor position is often the relative coordinates of the widget - // issueing it, instead of the absolute one that we expect. - // This is temporary however, and, once the control is created and initialized - // correctly, it eventually emits the correct absolute coordinates. - //SAL_WARN("lokevt", "Skipping invalid event [" + std::to_string(type) + "]: [" + payload + "]."); - return; - } - - std::unique_lock<std::mutex> lock(m_mutex); - - const auto stateIt = m_states.find(type); - if (stateIt != m_states.end()) - { - // If the state didn't change, it's safe to ignore. - if (stateIt->second == payload) - { - //SAL_WARN("lokevt", "Skipping duplicate [" + std::to_string(type) + "]: [" + payload + "]."); - return; - } - - stateIt->second = payload; - } - - if (type == LOK_CALLBACK_TEXT_SELECTION && payload.empty()) - { - // Removing text selection invalidates the start and end as well. - m_states[LOK_CALLBACK_TEXT_SELECTION_START] = ""; - m_states[LOK_CALLBACK_TEXT_SELECTION_END] = ""; - } - - m_queue.emplace_back(type, payload); - - // These are safe to use the latest state and ignore previous - // ones (if any) since the last overrides previous ones. - switch (type) - { - case LOK_CALLBACK_TEXT_SELECTION_START: - case LOK_CALLBACK_TEXT_SELECTION_END: - case LOK_CALLBACK_TEXT_SELECTION: - case LOK_CALLBACK_GRAPHIC_SELECTION: - case LOK_CALLBACK_MOUSE_POINTER: - case LOK_CALLBACK_CELL_CURSOR: - case LOK_CALLBACK_CELL_FORMULA: - case LOK_CALLBACK_CURSOR_VISIBLE: - case LOK_CALLBACK_SET_PART: - case LOK_CALLBACK_STATUS_INDICATOR_SET_VALUE: - removeAllButLast(type, false); - break; - - // These come with rects, so drop earlier - // only when the latter includes former ones. - case LOK_CALLBACK_INVALIDATE_VISIBLE_CURSOR: - case LOK_CALLBACK_INVALIDATE_TILES: - if (payload.empty()) - { - // Invalidating everything means previously - // invalidated tiles can be dropped. - removeAllButLast(type, false); - } - else - { - removeAllButLast(type, true); - } - - break; - } - - lock.unlock(); - if (!IsActive()) - { - Start(); - } - } - - void setPartTilePainting(const bool bPartPainting) { m_bPartTilePainting = bPartPainting; } - bool isPartTilePainting() const { return m_bPartTilePainting; } + explicit CallbackFlushHandler(LibreOfficeKitCallback pCallback, void* pData); + virtual ~CallbackFlushHandler(); + virtual void Invoke() override; + static void callback(const int type, const char* payload, void* data); + void queue(const int type, const char* data); + void setPartTilePainting(const bool bPartPainting); + bool isPartTilePainting() const; private: - void flush() - { - if (m_pCallback) - { - std::unique_lock<std::mutex> lock(m_mutex); - for (auto& pair : m_queue) - { - m_pCallback(pair.first, pair.second.c_str(), m_pData); - } + void flush(); + void removeAllButLast(const int type, const bool identical); - m_queue.clear(); - } - } - - void removeAllButLast(const int type, const bool identical) - { - int i = m_queue.size() - 1; - std::string payload; - for (; i >= 0; --i) - { - if (m_queue[i].first == type) - { - payload = m_queue[i].second; - //SAL_WARN("lokevt", "Found [" + std::to_string(type) + "] at " + std::to_string(i) + ": [" + payload + "]."); - break; - } - } - - for (--i; i >= 0; --i) - { - if (m_queue[i].first == type && - (!identical || m_queue[i].second == payload)) - { - //SAL_WARN("lokevt", "Removing [" + std::to_string(type) + "] at " + std::to_string(i) + ": " + m_queue[i].second + "]."); - m_queue.erase(m_queue.begin() + i); - } - } - } - - private: std::vector<std::pair<int, std::string>> m_queue; std::map<int, std::string> m_states; LibreOfficeKitCallback m_pCallback; diff --git a/desktop/source/lib/init.cxx b/desktop/source/lib/init.cxx index ccb27c1bd398..caa6cd2fe62a 100644 --- a/desktop/source/lib/init.cxx +++ b/desktop/source/lib/init.cxx @@ -430,6 +430,203 @@ LibLODocument_Impl::~LibLODocument_Impl() mxComponent->dispose(); } +CallbackFlushHandler::CallbackFlushHandler(LibreOfficeKitCallback pCallback, void* pData) + : Idle( "lokit timer callback" ), + m_pCallback(pCallback), + m_pData(pData), + m_bPartTilePainting(false) +{ + SetPriority(SchedulerPriority::POST_PAINT); + + // Add the states that are safe to skip duplicates on, + // even when not consequent. + m_states.emplace(LOK_CALLBACK_TEXT_SELECTION_START, "NIL"); + m_states.emplace(LOK_CALLBACK_TEXT_SELECTION_END, "NIL"); + m_states.emplace(LOK_CALLBACK_TEXT_SELECTION, "NIL"); + m_states.emplace(LOK_CALLBACK_GRAPHIC_SELECTION, "NIL"); + m_states.emplace(LOK_CALLBACK_INVALIDATE_VISIBLE_CURSOR, "NIL"); + m_states.emplace(LOK_CALLBACK_STATE_CHANGED, "NIL"); + m_states.emplace(LOK_CALLBACK_MOUSE_POINTER, "NIL"); + m_states.emplace(LOK_CALLBACK_CELL_CURSOR, "NIL"); + m_states.emplace(LOK_CALLBACK_CELL_FORMULA, "NIL"); + m_states.emplace(LOK_CALLBACK_CURSOR_VISIBLE, "NIL"); + m_states.emplace(LOK_CALLBACK_SET_PART, "NIL"); + + Start(); +} + +CallbackFlushHandler::~CallbackFlushHandler() +{ + Stop(); + + // We might have important notification (.uno:save?). + flush(); +} + +void CallbackFlushHandler::Invoke() +{ + flush(); +} + +void CallbackFlushHandler::callback(const int type, const char* payload, void* data) +{ + CallbackFlushHandler* self = static_cast<CallbackFlushHandler*>(data); + if (self) + { + self->queue(type, payload); + } +} + +void CallbackFlushHandler::queue(const int type, const char* data) +{ + const std::string payload(data ? data : "(nil)"); + if (m_bPartTilePainting) + { + // We drop notifications when this is set, except for important ones. + // When we issue a complex command (such as .uno:InsertAnnotation) + // there will be multiple notifications. On the first invalidation + // we will start painting, but other events will get fired + // while the complex command in question executes. + // We don't want to supress everything here on the wrong assumption + // that no new events are fired during painting. + if (type != LOK_CALLBACK_STATE_CHANGED && + type != LOK_CALLBACK_INVALIDATE_TILES && + type != LOK_CALLBACK_INVALIDATE_VISIBLE_CURSOR && + type != LOK_CALLBACK_CURSOR_VISIBLE && + type != LOK_CALLBACK_TEXT_SELECTION) + { + //SAL_WARN("lokevt", "Skipping while painting [" + std::to_string(type) + "]: [" + payload + "]."); + return; + } + } + + // Supress invalid payloads. + if (type == LOK_CALLBACK_INVALIDATE_VISIBLE_CURSOR && + payload.find(", 0, 0, ") != std::string::npos) + { + // The cursor position is often the relative coordinates of the widget + // issueing it, instead of the absolute one that we expect. + // This is temporary however, and, once the control is created and initialized + // correctly, it eventually emits the correct absolute coordinates. + //SAL_WARN("lokevt", "Skipping invalid event [" + std::to_string(type) + "]: [" + payload + "]."); + return; + } + + std::unique_lock<std::mutex> lock(m_mutex); + + const auto stateIt = m_states.find(type); + if (stateIt != m_states.end()) + { + // If the state didn't change, it's safe to ignore. + if (stateIt->second == payload) + { + //SAL_WARN("lokevt", "Skipping duplicate [" + std::to_string(type) + "]: [" + payload + "]."); + return; + } + + stateIt->second = payload; + } + + if (type == LOK_CALLBACK_TEXT_SELECTION && payload.empty()) + { + // Removing text selection invalidates the start and end as well. + m_states[LOK_CALLBACK_TEXT_SELECTION_START] = ""; + m_states[LOK_CALLBACK_TEXT_SELECTION_END] = ""; + } + + m_queue.emplace_back(type, payload); + + // These are safe to use the latest state and ignore previous + // ones (if any) since the last overrides previous ones. + switch (type) + { + case LOK_CALLBACK_TEXT_SELECTION_START: + case LOK_CALLBACK_TEXT_SELECTION_END: + case LOK_CALLBACK_TEXT_SELECTION: + case LOK_CALLBACK_GRAPHIC_SELECTION: + case LOK_CALLBACK_MOUSE_POINTER: + case LOK_CALLBACK_CELL_CURSOR: + case LOK_CALLBACK_CELL_FORMULA: + case LOK_CALLBACK_CURSOR_VISIBLE: + case LOK_CALLBACK_SET_PART: + case LOK_CALLBACK_STATUS_INDICATOR_SET_VALUE: + removeAllButLast(type, false); + break; + + // These come with rects, so drop earlier + // only when the latter includes former ones. + case LOK_CALLBACK_INVALIDATE_VISIBLE_CURSOR: + case LOK_CALLBACK_INVALIDATE_TILES: + if (payload.empty()) + { + // Invalidating everything means previously + // invalidated tiles can be dropped. + removeAllButLast(type, false); + } + else + { + removeAllButLast(type, true); + } + + break; + } + + lock.unlock(); + if (!IsActive()) + { + Start(); + } +} + +void CallbackFlushHandler::setPartTilePainting(const bool bPartPainting) +{ + m_bPartTilePainting = bPartPainting; +} + +bool CallbackFlushHandler::isPartTilePainting() const +{ + return m_bPartTilePainting; +} + +void CallbackFlushHandler::flush() +{ + if (m_pCallback) + { + std::unique_lock<std::mutex> lock(m_mutex); + for (auto& pair : m_queue) + { + m_pCallback(pair.first, pair.second.c_str(), m_pData); + } + + m_queue.clear(); + } +} + +void CallbackFlushHandler::removeAllButLast(const int type, const bool identical) +{ + int i = m_queue.size() - 1; + std::string payload; + for (; i >= 0; --i) + { + if (m_queue[i].first == type) + { + payload = m_queue[i].second; + //SAL_WARN("lokevt", "Found [" + std::to_string(type) + "] at " + std::to_string(i) + ": [" + payload + "]."); + break; + } + } + + for (--i; i >= 0; --i) + { + if (m_queue[i].first == type && + (!identical || m_queue[i].second == payload)) + { + //SAL_WARN("lokevt", "Removing [" + std::to_string(type) + "] at " + std::to_string(i) + ": " + m_queue[i].second + "]."); + m_queue.erase(m_queue.begin() + i); + } + } +} + static void doc_destroy(LibreOfficeKitDocument *pThis) { LibLODocument_Impl *pDocument = static_cast<LibLODocument_Impl*>(pThis); |