90f387cdc5
We need to get more helpful structure into this message, and add a type=bg flag to handle background progress messages. Add unit test for merging progress: setvalue commands. Simplify ProtocolHandler sendTextMessage with a std::string wrapper. Android code needs manual testing. Signed-off-by: Michael Meeks <michael.meeks@collabora.com> Change-Id: I54ce807e2fc5de80118905e68557a95e637fbd18
298 lines
9.8 KiB
C++
298 lines
9.8 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4; fill-column: 100 -*- */
|
|
/*
|
|
* Copyright the Collabora Online contributors.
|
|
*
|
|
* SPDX-License-Identifier: MPL-2.0
|
|
*
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <unordered_map>
|
|
#include <queue>
|
|
|
|
#include <atomic>
|
|
|
|
#define LOK_USE_UNSTABLE_API
|
|
#include <LibreOfficeKit/LibreOfficeKit.hxx>
|
|
|
|
#include "Common.hpp"
|
|
#include "Kit.hpp"
|
|
#include "Session.hpp"
|
|
#include "Watermark.hpp"
|
|
#include "StateRecorder.hpp"
|
|
|
|
class Document;
|
|
class ChildSession;
|
|
|
|
enum class LokEventTargetEnum
|
|
{
|
|
Document,
|
|
Window
|
|
};
|
|
|
|
/// Represents a session to the WSD process, in a Kit process. Note that this is not a singleton.
|
|
class ChildSession final : public Session
|
|
{
|
|
public:
|
|
static bool NoCapsForKit;
|
|
|
|
/// Create a new ChildSession
|
|
/// jailId The JailID of the jail root directory,
|
|
// used by downloadas to construct jailed path.
|
|
ChildSession(
|
|
const std::shared_ptr<ProtocolHandlerInterface> &protocol,
|
|
const std::string& id,
|
|
const std::string& jailId,
|
|
const std::string& jailRoot,
|
|
Document& document);
|
|
virtual ~ChildSession();
|
|
|
|
bool getStatus();
|
|
int getViewId() const { return _viewId; }
|
|
void setViewId(const int viewId) { _viewId = viewId; }
|
|
const std::string& getViewUserId() const { return getUserId(); }
|
|
const std::string& getViewUserName() const { return getUserName(); }
|
|
const std::string& getViewUserExtraInfo() const { return getUserExtraInfo(); }
|
|
const std::string& getViewUserPrivateInfo() const { return getUserPrivateInfo(); }
|
|
void updateSpeed();
|
|
int getSpeed();
|
|
|
|
void loKitCallback(const int type, const std::string& payload);
|
|
|
|
/// Initializes the watermark support, if enabled and required.
|
|
/// Returns true if watermark is enabled and initialized.
|
|
bool initWatermark()
|
|
{
|
|
if (hasWatermark())
|
|
{
|
|
_docWatermark = std::make_shared<Watermark>(getLOKitDocument(), getWatermarkText(),
|
|
getWatermarkOpacity());
|
|
}
|
|
|
|
return _docWatermark != nullptr;
|
|
}
|
|
|
|
const std::shared_ptr<Watermark>& watermark() const { return _docWatermark; };
|
|
|
|
bool sendTextFrame(const char* buffer, int length) override
|
|
{
|
|
if (!_docManager)
|
|
{
|
|
LOG_TRC("ERR dropping - client-" + getId() + ' ' + std::string(buffer, length));
|
|
return false;
|
|
}
|
|
const auto msg = "client-" + getId() + ' ' + std::string(buffer, length);
|
|
return _docManager->sendFrame(msg.data(), msg.size(), WSOpCode::Text);
|
|
}
|
|
|
|
bool sendBinaryFrame(const char* buffer, int length) override
|
|
{
|
|
if (!_docManager)
|
|
{
|
|
LOG_TRC("ERR dropping binary - client-" + getId());
|
|
return false;
|
|
}
|
|
const auto msg = "client-" + getId() + ' ' + std::string(buffer, length);
|
|
return _docManager->sendFrame(msg.data(), msg.size(), WSOpCode::Binary);
|
|
}
|
|
|
|
bool sendProgressFrame(const char* id, const std::string &jsonProps);
|
|
|
|
using Session::sendTextFrame;
|
|
|
|
bool getClipboard(const StringVector& tokens);
|
|
|
|
void resetDocManager()
|
|
{
|
|
disconnect();
|
|
_docManager = nullptr;
|
|
}
|
|
|
|
// Only called by kit.
|
|
void setCanonicalViewId(int viewId) { _canonicalViewId = viewId; }
|
|
|
|
int getCanonicalViewId() { return _canonicalViewId; }
|
|
|
|
void setViewRenderState(const std::string& state) { _viewRenderState = state; }
|
|
|
|
bool getDumpTiles() { return _isDumpingTiles; }
|
|
|
|
void setDumpTiles(bool dumpTiles) { _isDumpingTiles = dumpTiles; }
|
|
|
|
std::string getViewRenderState() { return _viewRenderState; }
|
|
|
|
bool isTileInsideVisibleArea(const TileDesc& tile) const;
|
|
|
|
private:
|
|
bool loadDocument(const StringVector& tokens);
|
|
bool saveDocumentBackground(const StringVector &tokens);
|
|
|
|
bool sendFontRendering(const StringVector& tokens);
|
|
bool getCommandValues(const StringVector& tokens);
|
|
|
|
bool clientZoom(const StringVector& tokens);
|
|
bool clientVisibleArea(const StringVector& tokens);
|
|
bool outlineState(const StringVector& tokens);
|
|
bool downloadAs(const StringVector& tokens);
|
|
bool getChildId();
|
|
bool getTextSelection(const StringVector& tokens);
|
|
bool setClipboard(const char* buffer, int length, const StringVector& tokens);
|
|
std::string getTextSelectionInternal(const std::string& mimeType);
|
|
bool paste(const char* buffer, int length, const StringVector& tokens);
|
|
bool insertFile(const StringVector& tokens);
|
|
bool keyEvent(const StringVector& tokens, const LokEventTargetEnum target);
|
|
bool extTextInputEvent(const StringVector& tokens);
|
|
bool dialogKeyEvent(const char* buffer, int length, const std::vector<std::string>& tokens);
|
|
bool mouseEvent(const StringVector& tokens, const LokEventTargetEnum target);
|
|
bool gestureEvent(const StringVector& tokens);
|
|
bool dialogEvent(const StringVector& tokens);
|
|
bool completeFunction(const StringVector& tokens);
|
|
bool unoCommand(const StringVector& tokens);
|
|
bool selectText(const StringVector& tokens, const LokEventTargetEnum target);
|
|
bool selectGraphic(const StringVector& tokens);
|
|
bool renderWindow(const StringVector& tokens);
|
|
bool resizeWindow(const StringVector& tokens);
|
|
bool resetSelection(const StringVector& tokens);
|
|
bool saveAs(const StringVector& tokens);
|
|
bool exportAs(const StringVector& tokens);
|
|
bool setClientPart(const StringVector& tokens);
|
|
bool selectClientPart(const StringVector& tokens);
|
|
bool moveSelectedClientParts(const StringVector& tokens);
|
|
bool setPage(const StringVector& tokens);
|
|
bool sendWindowCommand(const StringVector& tokens);
|
|
bool askSignatureStatus(const char* buffer, int length, const StringVector& tokens);
|
|
bool renderShapeSelection(const StringVector& tokens);
|
|
bool removeTextContext(const StringVector& tokens);
|
|
#if ENABLE_FEATURE_LOCK || ENABLE_FEATURE_RESTRICTION
|
|
bool updateBlockingCommandStatus(const StringVector& tokens);
|
|
std::string getBlockedCommandType(std::string command);
|
|
#endif
|
|
bool handleZoteroMessage(const StringVector& tokens);
|
|
bool formFieldEvent(const char* buffer, int length, const StringVector& tokens);
|
|
bool contentControlEvent(const StringVector& tokens);
|
|
bool renderSearchResult(const char* buffer, int length, const StringVector& tokens);
|
|
bool setAccessibilityState(bool enable);
|
|
bool getA11yFocusedParagraph();
|
|
bool getA11yCaretPosition();
|
|
|
|
void rememberEventsForInactiveUser(const int type, const std::string& payload);
|
|
|
|
virtual void disconnect() override;
|
|
virtual bool _handleInput(const char* buffer, int length) override;
|
|
|
|
static void dumpRecordedUnoCommands();
|
|
|
|
std::shared_ptr<lok::Document> getLOKitDocument() const
|
|
{
|
|
return _docManager->getLOKitDocument();
|
|
}
|
|
|
|
std::shared_ptr<lok::Office> getLOKit() const
|
|
{
|
|
return _docManager->getLOKit();
|
|
}
|
|
|
|
std::string getLOKitLastError() const
|
|
{
|
|
char *lastErr = _docManager->getLOKit()->getError();
|
|
std::string ret;
|
|
if (lastErr)
|
|
{
|
|
ret = std::string(lastErr, strlen(lastErr));
|
|
free (lastErr);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
public:
|
|
// simple one line for priming
|
|
std::string getActivityState()
|
|
{
|
|
std::stringstream ss;
|
|
ss << "view: " << _viewId
|
|
<< ", session " << getId()
|
|
<< (isReadOnly() ? ", ro": ", rw")
|
|
<< ", user: '" << getUserNameAnonym() << "'"
|
|
<< ", load" << (_isDocLoaded ? "ed" : "ing")
|
|
<< ", type: " << _docType
|
|
<< ", lang: " << getLang();
|
|
return ss.str();
|
|
}
|
|
|
|
void dumpState(std::ostream& oss) override
|
|
{
|
|
Session::dumpState(oss);
|
|
|
|
oss << "\n\tviewId: " << _viewId
|
|
<< "\n\tcanonicalViewId: " << _canonicalViewId
|
|
<< "\n\tisDocLoaded: " << _isDocLoaded
|
|
<< "\n\tdocType: " << _docType
|
|
<< "\n\tcopyingToClipboard: " << _copyToClipboard
|
|
<< "\n\tdocType: " << _docType
|
|
// FIXME: _pixmapCache
|
|
<< "\n\texportAsWopiUrl: " << _exportAsWopiUrl
|
|
<< "\n\tviewRenderedState: " << _viewRenderState
|
|
<< "\n\tisDumpingTiles: " << _isDocLoaded
|
|
<< "\n\tclientVisibleArea: " << _clientVisibleArea.toString()
|
|
<< "\n\thasURP: " << _hasURP
|
|
<< "\n\tURPContext?: " << (_URPContext == nullptr)
|
|
<< '\n';
|
|
|
|
_stateRecorder.dumpState(oss);
|
|
}
|
|
|
|
private:
|
|
const std::string _jailId;
|
|
const std::string _jailRoot;
|
|
Document* _docManager;
|
|
|
|
std::shared_ptr<Watermark> _docWatermark;
|
|
|
|
std::queue<std::chrono::steady_clock::time_point> _cursorInvalidatedEvent;
|
|
const unsigned _eventStorageIntervalMs = 15*1000;
|
|
|
|
/// View ID, returned by createView() or 0 by default.
|
|
int _viewId;
|
|
|
|
/// Whether document has been opened successfully
|
|
bool _isDocLoaded;
|
|
|
|
std::string _docType;
|
|
|
|
StateRecorder _stateRecorder;
|
|
|
|
/// If we are copying to clipboard.
|
|
bool _copyToClipboard;
|
|
|
|
std::vector<uint64_t> _pixmapCache;
|
|
|
|
/// How many sessions / clients we have
|
|
static size_t NumSessions;
|
|
|
|
/// stores wopi url for export as operation
|
|
std::string _exportAsWopiUrl;
|
|
|
|
/// stores info about the view
|
|
std::string _viewRenderState;
|
|
|
|
/// the canonical id unique to the set of rendering properties of this session
|
|
int _canonicalViewId;
|
|
|
|
/// whether we are dumping tiles as they are being drawn
|
|
bool _isDumpingTiles;
|
|
|
|
Util::Rectangle _clientVisibleArea;
|
|
|
|
void* _URPContext;
|
|
|
|
/// whether there is a URP session created for this ChildSession
|
|
bool _hasURP;
|
|
|
|
// When state is added - please update dumpState above.
|
|
};
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|