More file previews
This adds previews for downloaded pdfs and code files.  
This commit is contained in:
committed by
Tobias Fella
parent
33c55d1563
commit
1671e05d12
@@ -85,6 +85,7 @@ if(ANDROID)
|
|||||||
else()
|
else()
|
||||||
find_package(Qt6 ${QT_MIN_VERSION} COMPONENTS Widgets)
|
find_package(Qt6 ${QT_MIN_VERSION} COMPONENTS Widgets)
|
||||||
find_package(KF6 ${KF_MIN_VERSION} REQUIRED COMPONENTS QQC2DesktopStyle KIO WindowSystem StatusNotifierItem)
|
find_package(KF6 ${KF_MIN_VERSION} REQUIRED COMPONENTS QQC2DesktopStyle KIO WindowSystem StatusNotifierItem)
|
||||||
|
find_package(KF6SyntaxHighlighting ${KF_MIN_VERSION} REQUIRED)
|
||||||
set_package_properties(KF6QQC2DesktopStyle PROPERTIES
|
set_package_properties(KF6QQC2DesktopStyle PROPERTIES
|
||||||
TYPE RUNTIME
|
TYPE RUNTIME
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -532,7 +532,7 @@ if(ANDROID)
|
|||||||
)
|
)
|
||||||
ecm_add_android_apk(neochat-app ANDROID_DIR ${CMAKE_SOURCE_DIR}/android)
|
ecm_add_android_apk(neochat-app ANDROID_DIR ${CMAKE_SOURCE_DIR}/android)
|
||||||
else()
|
else()
|
||||||
target_link_libraries(neochat PUBLIC Qt::Widgets KF6::KIOWidgets)
|
target_link_libraries(neochat PUBLIC Qt::Widgets KF6::KIOWidgets KF6::SyntaxHighlighting)
|
||||||
install(FILES neochat.notifyrc DESTINATION ${KDE_INSTALL_KNOTIFYRCDIR})
|
install(FILES neochat.notifyrc DESTINATION ${KDE_INSTALL_KNOTIFYRCDIR})
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
|||||||
@@ -40,7 +40,8 @@ public:
|
|||||||
Code, /**< A code section. */
|
Code, /**< A code section. */
|
||||||
Quote, /**< A quote section. */
|
Quote, /**< A quote section. */
|
||||||
File, /**< A message that is a file. */
|
File, /**< A message that is a file. */
|
||||||
Itinerary, /**< A preview for a file that can integrate with KDE itinerary.. */
|
Itinerary, /**< A preview for a file that can integrate with KDE itinerary. */
|
||||||
|
Pdf, /**< A preview for a PDF file. */
|
||||||
Poll, /**< The initial event for a poll. */
|
Poll, /**< The initial event for a poll. */
|
||||||
Location, /**< A location event. */
|
Location, /**< A location event. */
|
||||||
LiveLocation, /**< The initial event of a shared live location (i.e., the place where this is supposed to be shown in the timeline). */
|
LiveLocation, /**< The initial event of a shared live location (i.e., the place where this is supposed to be shown in the timeline). */
|
||||||
|
|||||||
@@ -41,6 +41,12 @@ FileType::~FileType() noexcept
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
FileType &FileType::instance()
|
||||||
|
{
|
||||||
|
static FileType _instance;
|
||||||
|
return _instance;
|
||||||
|
}
|
||||||
|
|
||||||
QMimeType FileType::mimeTypeForName(const QString &nameOrAlias) const
|
QMimeType FileType::mimeTypeForName(const QString &nameOrAlias) const
|
||||||
{
|
{
|
||||||
Q_D(const FileType);
|
Q_D(const FileType);
|
||||||
@@ -113,4 +119,10 @@ QStringList FileType::supportedAnimatedImageFormats() const
|
|||||||
return d->supportedAnimatedImageFormats;
|
return d->supportedAnimatedImageFormats;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool FileType::fileHasImage(const QUrl &file) const
|
||||||
|
{
|
||||||
|
const auto mimeType = mimeTypeForFile(file.toString());
|
||||||
|
return mimeType.isValid() && supportedImageFormats().contains(mimeType.preferredSuffix());
|
||||||
|
}
|
||||||
|
|
||||||
#include "moc_filetype.cpp"
|
#include "moc_filetype.cpp"
|
||||||
|
|||||||
@@ -41,8 +41,13 @@ class FileType : public QObject
|
|||||||
Q_PROPERTY(QStringList supportedAnimatedImageFormats READ supportedAnimatedImageFormats CONSTANT FINAL)
|
Q_PROPERTY(QStringList supportedAnimatedImageFormats READ supportedAnimatedImageFormats CONSTANT FINAL)
|
||||||
|
|
||||||
public:
|
public:
|
||||||
explicit FileType(QObject *parent = nullptr);
|
|
||||||
~FileType();
|
~FileType();
|
||||||
|
static FileType &instance();
|
||||||
|
static FileType *create(QQmlEngine *engine, QJSEngine *)
|
||||||
|
{
|
||||||
|
engine->setObjectOwnership(&instance(), QQmlEngine::CppOwnership);
|
||||||
|
return &instance();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Returns a MIME type for nameOrAlias or an invalid one if none found.
|
* @brief Returns a MIME type for nameOrAlias or an invalid one if none found.
|
||||||
@@ -120,7 +125,11 @@ public:
|
|||||||
QStringList supportedImageFormats() const;
|
QStringList supportedImageFormats() const;
|
||||||
QStringList supportedAnimatedImageFormats() const;
|
QStringList supportedAnimatedImageFormats() const;
|
||||||
|
|
||||||
|
bool fileHasImage(const QUrl &file) const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
explicit FileType(QObject *parent = nullptr);
|
||||||
|
|
||||||
const QScopedPointer<FileTypePrivate> d_ptr;
|
const QScopedPointer<FileTypePrivate> d_ptr;
|
||||||
Q_DECLARE_PRIVATE(FileType)
|
Q_DECLARE_PRIVATE(FileType)
|
||||||
Q_DISABLE_COPY(FileType)
|
Q_DISABLE_COPY(FileType)
|
||||||
|
|||||||
@@ -133,6 +133,11 @@ void ItineraryModel::loadData()
|
|||||||
beginResetModel();
|
beginResetModel();
|
||||||
m_data = QJsonDocument::fromJson(data).array();
|
m_data = QJsonDocument::fromJson(data).array();
|
||||||
endResetModel();
|
endResetModel();
|
||||||
|
|
||||||
|
Q_EMIT loaded();
|
||||||
|
});
|
||||||
|
connect(process, &QProcess::errorOccurred, this, [this]() {
|
||||||
|
Q_EMIT loadErrorOccurred();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -44,6 +44,10 @@ public:
|
|||||||
|
|
||||||
Q_INVOKABLE void sendToItinerary();
|
Q_INVOKABLE void sendToItinerary();
|
||||||
|
|
||||||
|
Q_SIGNALS:
|
||||||
|
void loaded();
|
||||||
|
void loadErrorOccurred();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QJsonArray m_data;
|
QJsonArray m_data;
|
||||||
QString m_path;
|
QString m_path;
|
||||||
|
|||||||
@@ -3,15 +3,25 @@
|
|||||||
|
|
||||||
#include "messagecontentmodel.h"
|
#include "messagecontentmodel.h"
|
||||||
|
|
||||||
|
#include <QImageReader>
|
||||||
|
|
||||||
#include <Quotient/events/redactionevent.h>
|
#include <Quotient/events/redactionevent.h>
|
||||||
#include <Quotient/events/roommessageevent.h>
|
#include <Quotient/events/roommessageevent.h>
|
||||||
#include <Quotient/events/stickerevent.h>
|
#include <Quotient/events/stickerevent.h>
|
||||||
|
#include <Quotient/room.h>
|
||||||
|
|
||||||
#include <KLocalizedString>
|
#include <KLocalizedString>
|
||||||
|
|
||||||
|
#ifndef Q_OS_ANDROID
|
||||||
|
#include <KSyntaxHighlighting/Definition>
|
||||||
|
#include <KSyntaxHighlighting/Repository>
|
||||||
|
#endif
|
||||||
|
|
||||||
#include "chatbarcache.h"
|
#include "chatbarcache.h"
|
||||||
#include "enums/messagecomponenttype.h"
|
#include "enums/messagecomponenttype.h"
|
||||||
#include "eventhandler.h"
|
#include "eventhandler.h"
|
||||||
|
#include "filetype.h"
|
||||||
|
#include "itinerarymodel.h"
|
||||||
#include "linkpreviewer.h"
|
#include "linkpreviewer.h"
|
||||||
#include "neochatroom.h"
|
#include "neochatroom.h"
|
||||||
#include "texthandler.h"
|
#include "texthandler.h"
|
||||||
@@ -255,9 +265,38 @@ void MessageContentModel::updateComponents(bool isEditing)
|
|||||||
m_components.append(TextHandler().textComponents(body, EventHandler::messageBodyInputFormat(*event), m_room, event, event->isReplaced()));
|
m_components.append(TextHandler().textComponents(body, EventHandler::messageBodyInputFormat(*event), m_room, event, event->isReplaced()));
|
||||||
} else if (eventHandler.messageComponentType() == MessageComponentType::File) {
|
} else if (eventHandler.messageComponentType() == MessageComponentType::File) {
|
||||||
m_components += MessageComponent{MessageComponentType::File, QString(), {}};
|
m_components += MessageComponent{MessageComponentType::File, QString(), {}};
|
||||||
updateItineraryModel();
|
if (m_emptyItinerary) {
|
||||||
if (m_itineraryModel != nullptr) {
|
Quotient::FileTransferInfo fileTransferInfo;
|
||||||
m_components += MessageComponent{MessageComponentType::Itinerary, QString(), {}};
|
if (auto event = eventCast<const Quotient::RoomMessageEvent>(m_event)) {
|
||||||
|
if (event->hasFileContent()) {
|
||||||
|
fileTransferInfo = m_room->fileTransferInfo(event->id());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (auto event = eventCast<const Quotient::StickerEvent>(m_event)) {
|
||||||
|
fileTransferInfo = m_room->fileTransferInfo(event->id());
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifndef Q_OS_ANDROID
|
||||||
|
KSyntaxHighlighting::Repository repository;
|
||||||
|
const auto definitionForFile = repository.definitionForFileName(fileTransferInfo.localPath.toString());
|
||||||
|
if (definitionForFile.isValid() || QFileInfo(fileTransferInfo.localPath.path()).suffix() == QStringLiteral("txt")) {
|
||||||
|
QFile file(fileTransferInfo.localPath.path());
|
||||||
|
file.open(QIODevice::ReadOnly);
|
||||||
|
m_components += MessageComponent{MessageComponentType::Code,
|
||||||
|
QString::fromStdString(file.readAll().toStdString()),
|
||||||
|
{{QStringLiteral("class"), definitionForFile.name()}}};
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (FileType::instance().fileHasImage(fileTransferInfo.localPath)) {
|
||||||
|
QImageReader reader(fileTransferInfo.localPath.path());
|
||||||
|
m_components += MessageComponent{MessageComponentType::Pdf, QString(), {{QStringLiteral("size"), reader.size()}}};
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
updateItineraryModel();
|
||||||
|
if (m_itineraryModel != nullptr) {
|
||||||
|
m_components += MessageComponent{MessageComponentType::Itinerary, QString(), {}};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
m_components += MessageComponent{eventHandler.messageComponentType(), QString(), {}};
|
m_components += MessageComponent{eventHandler.messageComponentType(), QString(), {}};
|
||||||
@@ -290,6 +329,20 @@ void MessageContentModel::updateItineraryModel()
|
|||||||
} else if (!filePath.isEmpty()) {
|
} else if (!filePath.isEmpty()) {
|
||||||
if (m_itineraryModel == nullptr) {
|
if (m_itineraryModel == nullptr) {
|
||||||
m_itineraryModel = new ItineraryModel(this);
|
m_itineraryModel = new ItineraryModel(this);
|
||||||
|
connect(m_itineraryModel, &ItineraryModel::loaded, this, [this]() {
|
||||||
|
if (m_itineraryModel->rowCount() == 0) {
|
||||||
|
m_itineraryModel->deleteLater();
|
||||||
|
m_itineraryModel = nullptr;
|
||||||
|
m_emptyItinerary = true;
|
||||||
|
updateComponents();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
connect(m_itineraryModel, &ItineraryModel::loadErrorOccurred, this, [this]() {
|
||||||
|
m_itineraryModel->deleteLater();
|
||||||
|
m_itineraryModel = nullptr;
|
||||||
|
m_emptyItinerary = true;
|
||||||
|
updateComponents();
|
||||||
|
});
|
||||||
}
|
}
|
||||||
m_itineraryModel->setPath(filePath.toString());
|
m_itineraryModel->setPath(filePath.toString());
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -97,4 +97,5 @@ private:
|
|||||||
ItineraryModel *m_itineraryModel = nullptr;
|
ItineraryModel *m_itineraryModel = nullptr;
|
||||||
|
|
||||||
void updateItineraryModel();
|
void updateItineraryModel();
|
||||||
|
bool m_emptyItinerary = false;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -29,6 +29,7 @@ qt_add_qml_module(timeline
|
|||||||
LocationComponent.qml
|
LocationComponent.qml
|
||||||
MessageEditComponent.qml
|
MessageEditComponent.qml
|
||||||
MimeComponent.qml
|
MimeComponent.qml
|
||||||
|
PdfPreviewComponent.qml
|
||||||
PollComponent.qml
|
PollComponent.qml
|
||||||
QuoteComponent.qml
|
QuoteComponent.qml
|
||||||
ReplyComponent.qml
|
ReplyComponent.qml
|
||||||
|
|||||||
@@ -41,44 +41,25 @@ QQC2.Control {
|
|||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
Layout.maximumWidth: root.maxContentWidth
|
Layout.maximumWidth: root.maxContentWidth
|
||||||
|
Layout.maximumHeight: Kirigami.Units.gridUnit * 20
|
||||||
|
|
||||||
topPadding: 0
|
topPadding: 0
|
||||||
bottomPadding: 0
|
bottomPadding: 0
|
||||||
|
leftPadding: 0
|
||||||
|
rightPadding: 0
|
||||||
|
|
||||||
contentItem: RowLayout {
|
contentItem: QQC2.ScrollView {
|
||||||
spacing: Kirigami.Units.smallSpacing
|
id: codeScrollView
|
||||||
|
contentWidth: root.maxContentWidth
|
||||||
|
|
||||||
ColumnLayout {
|
// HACK: Hide unnecessary horizontal scrollbar (https://bugreports.qt.io/browse/QTBUG-83890)
|
||||||
id: lineNumberColumn
|
QQC2.ScrollBar.horizontal.policy: QQC2.ScrollBar.AlwaysOff
|
||||||
spacing: 0
|
|
||||||
Repeater {
|
|
||||||
id: repeater
|
|
||||||
model: LineModel {
|
|
||||||
id: lineModel
|
|
||||||
document: codeText.textDocument
|
|
||||||
}
|
|
||||||
delegate: QQC2.Label {
|
|
||||||
id: label
|
|
||||||
required property int index
|
|
||||||
required property int docLineHeight
|
|
||||||
Layout.fillWidth: true
|
|
||||||
Layout.preferredHeight: docLineHeight
|
|
||||||
horizontalAlignment: Text.AlignRight
|
|
||||||
text: index + 1
|
|
||||||
color: Kirigami.Theme.disabledTextColor
|
|
||||||
|
|
||||||
font.family: "monospace"
|
QQC2.TextArea {
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Kirigami.Separator {
|
|
||||||
Layout.fillHeight: true
|
|
||||||
}
|
|
||||||
TextEdit {
|
|
||||||
id: codeText
|
id: codeText
|
||||||
Layout.fillWidth: true
|
|
||||||
topPadding: Kirigami.Units.smallSpacing
|
topPadding: Kirigami.Units.smallSpacing
|
||||||
bottomPadding: Kirigami.Units.smallSpacing
|
bottomPadding: Kirigami.Units.smallSpacing
|
||||||
|
leftPadding: lineNumberColumn.width + lineNumberColumn.anchors.leftMargin + Kirigami.Units.smallSpacing * 2
|
||||||
|
|
||||||
text: root.display
|
text: root.display
|
||||||
readOnly: true
|
readOnly: true
|
||||||
@@ -100,11 +81,51 @@ QQC2.Control {
|
|||||||
textEdit: definitionName == "None" ? null : codeText
|
textEdit: definitionName == "None" ? null : codeText
|
||||||
definition: definitionName
|
definition: definitionName
|
||||||
}
|
}
|
||||||
|
ColumnLayout {
|
||||||
|
id: lineNumberColumn
|
||||||
|
anchors {
|
||||||
|
top: codeText.top
|
||||||
|
topMargin: codeText.topPadding
|
||||||
|
left: codeText.left
|
||||||
|
leftMargin: Kirigami.Units.smallSpacing
|
||||||
|
}
|
||||||
|
spacing: 0
|
||||||
|
Repeater {
|
||||||
|
id: repeater
|
||||||
|
model: LineModel {
|
||||||
|
id: lineModel
|
||||||
|
document: codeText.textDocument
|
||||||
|
}
|
||||||
|
delegate: QQC2.Label {
|
||||||
|
id: label
|
||||||
|
required property int index
|
||||||
|
required property int docLineHeight
|
||||||
|
Layout.fillWidth: true
|
||||||
|
Layout.preferredHeight: docLineHeight
|
||||||
|
horizontalAlignment: Text.AlignRight
|
||||||
|
text: index + 1
|
||||||
|
color: Kirigami.Theme.disabledTextColor
|
||||||
|
|
||||||
|
font.family: "monospace"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
TapHandler {
|
TapHandler {
|
||||||
acceptedButtons: Qt.LeftButton
|
acceptedButtons: Qt.LeftButton
|
||||||
onLongPressed: root.showMessageMenu()
|
onLongPressed: root.showMessageMenu()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
background: null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Kirigami.Separator {
|
||||||
|
anchors {
|
||||||
|
top: root.top
|
||||||
|
bottom: root.bottom
|
||||||
|
left: root.left
|
||||||
|
leftMargin: lineNumberColumn.width + lineNumberColumn.anchors.leftMargin + Kirigami.Units.smallSpacing
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -113,7 +134,7 @@ QQC2.Control {
|
|||||||
top: parent.top
|
top: parent.top
|
||||||
topMargin: Kirigami.Units.smallSpacing
|
topMargin: Kirigami.Units.smallSpacing
|
||||||
right: parent.right
|
right: parent.right
|
||||||
rightMargin: Kirigami.Units.smallSpacing
|
rightMargin: (codeScrollView.QQC2.ScrollBar.vertical.visible ? codeScrollView.QQC2.ScrollBar.vertical.width : 0) + Kirigami.Units.smallSpacing
|
||||||
}
|
}
|
||||||
visible: root.hovered
|
visible: root.hovered
|
||||||
icon.name: "edit-copy"
|
icon.name: "edit-copy"
|
||||||
|
|||||||
@@ -124,6 +124,13 @@ DelegateChooser {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
DelegateChoice {
|
||||||
|
roleValue: MessageComponentType.Pdf
|
||||||
|
delegate: PdfPreviewComponent {
|
||||||
|
maxContentWidth: root.maxContentWidth
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
DelegateChoice {
|
DelegateChoice {
|
||||||
roleValue: MessageComponentType.Poll
|
roleValue: MessageComponentType.Poll
|
||||||
delegate: PollComponent {
|
delegate: PollComponent {
|
||||||
|
|||||||
44
src/timeline/PdfPreviewComponent.qml
Normal file
44
src/timeline/PdfPreviewComponent.qml
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2024 Tobias Fella <tobias.fella@kde.org>
|
||||||
|
// SPDX-FileCopyrightText: 2024 James Graham <james.h.graham@protonmail.com>
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL
|
||||||
|
|
||||||
|
import QtQuick
|
||||||
|
import QtQuick.Layouts
|
||||||
|
|
||||||
|
import org.kde.neochat
|
||||||
|
|
||||||
|
Rectangle {
|
||||||
|
id: root
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief FileTransferInfo for any downloading files.
|
||||||
|
*/
|
||||||
|
required property var fileTransferInfo
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief The attributes of the component.
|
||||||
|
*/
|
||||||
|
required property var componentAttributes
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief The maximum width that the bubble's content can be.
|
||||||
|
*/
|
||||||
|
property real maxContentWidth: -1
|
||||||
|
|
||||||
|
Layout.preferredWidth: mediaSizeHelper.currentSize.width
|
||||||
|
Layout.preferredHeight: mediaSizeHelper.currentSize.height
|
||||||
|
|
||||||
|
color: "white"
|
||||||
|
|
||||||
|
Image {
|
||||||
|
anchors.fill: root
|
||||||
|
source: root?.fileTransferInfo.localPath ?? ""
|
||||||
|
|
||||||
|
MediaSizeHelper {
|
||||||
|
id: mediaSizeHelper
|
||||||
|
contentMaxWidth: root.maxContentWidth
|
||||||
|
mediaWidth: root.componentAttributes.size.width
|
||||||
|
mediaHeight: root.componentAttributes.size.height
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user