2022-03-20 20:01:08 +01:00
|
|
|
// SPDX-License-Identifier: GPL-3.0-only
|
|
|
|
/*
|
2022-12-14 15:02:04 +00:00
|
|
|
* Prism Launcher - Minecraft Launcher
|
2022-03-20 20:01:08 +01:00
|
|
|
* Copyright (c) 2022 Jamie Mansfield <jmansfield@cadixdev.org>
|
2022-06-12 13:50:58 +02:00
|
|
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
2022-12-14 15:02:04 +00:00
|
|
|
* Copyright (C) 2022 TheKodeToad <TheKodeToad@proton.me>
|
2022-03-20 20:01:08 +01:00
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, version 3.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* This file incorporates work covered by the following copyright and
|
|
|
|
* permission notice:
|
|
|
|
*
|
|
|
|
* Copyright 2013-2021 MultiMC Contributors
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2014-06-28 17:09:23 +02:00
|
|
|
#include "ScreenshotsPage.h"
|
2023-06-27 19:31:36 +03:00
|
|
|
#include "BuildConfig.h"
|
2014-06-28 17:09:23 +02:00
|
|
|
#include "ui_ScreenshotsPage.h"
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
#include <QClipboard>
|
|
|
|
#include <QEvent>
|
2014-06-28 17:09:23 +02:00
|
|
|
#include <QFileIconProvider>
|
|
|
|
#include <QFileSystemModel>
|
2014-07-12 23:02:52 +02:00
|
|
|
#include <QKeyEvent>
|
2023-08-02 18:35:35 +02:00
|
|
|
#include <QLineEdit>
|
|
|
|
#include <QMap>
|
2019-07-17 02:01:29 +02:00
|
|
|
#include <QMenu>
|
2023-08-02 18:35:35 +02:00
|
|
|
#include <QModelIndex>
|
|
|
|
#include <QMutableListIterator>
|
|
|
|
#include <QPainter>
|
2022-05-02 19:48:37 +02:00
|
|
|
#include <QRegularExpression>
|
2023-08-02 18:35:35 +02:00
|
|
|
#include <QSet>
|
|
|
|
#include <QStyledItemDelegate>
|
2014-06-28 17:09:23 +02:00
|
|
|
|
2021-11-20 16:22:22 +01:00
|
|
|
#include <Application.h>
|
2014-06-28 17:09:23 +02:00
|
|
|
|
2021-11-22 03:55:16 +01:00
|
|
|
#include "ui/dialogs/CustomMessageBox.h"
|
2023-08-02 18:35:35 +02:00
|
|
|
#include "ui/dialogs/ProgressDialog.h"
|
2021-11-22 03:55:16 +01:00
|
|
|
|
2015-02-09 01:51:14 +01:00
|
|
|
#include "net/NetJob.h"
|
|
|
|
#include "screenshots/ImgurAlbumCreation.h"
|
2023-08-02 18:35:35 +02:00
|
|
|
#include "screenshots/ImgurUpload.h"
|
2015-02-09 01:51:14 +01:00
|
|
|
#include "tasks/SequentialTask.h"
|
2014-06-28 17:09:23 +02:00
|
|
|
|
2016-01-05 07:32:52 +01:00
|
|
|
#include <DesktopServices.h>
|
2023-08-02 18:35:35 +02:00
|
|
|
#include <FileSystem.h>
|
|
|
|
#include "RWStorage.h"
|
2014-07-12 23:02:52 +02:00
|
|
|
|
2014-07-05 13:26:37 +02:00
|
|
|
typedef RWStorage<QString, QIcon> SharedIconCache;
|
|
|
|
typedef std::shared_ptr<SharedIconCache> SharedIconCachePtr;
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
class ThumbnailingResult : public QObject {
|
2014-07-05 13:26:37 +02:00
|
|
|
Q_OBJECT
|
2023-08-02 18:35:35 +02:00
|
|
|
public slots:
|
|
|
|
inline void emitResultsReady(const QString& path) { emit resultsReady(path); }
|
|
|
|
inline void emitResultsFailed(const QString& path) { emit resultsFailed(path); }
|
|
|
|
signals:
|
|
|
|
void resultsReady(const QString& path);
|
|
|
|
void resultsFailed(const QString& path);
|
2014-07-05 13:26:37 +02:00
|
|
|
};
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
class ThumbnailRunnable : public QRunnable {
|
|
|
|
public:
|
2014-07-12 23:02:52 +02:00
|
|
|
ThumbnailRunnable(QString path, SharedIconCachePtr cache)
|
2014-07-05 13:26:37 +02:00
|
|
|
{
|
|
|
|
m_path = path;
|
|
|
|
m_cache = cache;
|
|
|
|
}
|
|
|
|
void run()
|
|
|
|
{
|
|
|
|
QFileInfo info(m_path);
|
2014-07-12 23:02:52 +02:00
|
|
|
if (info.isDir())
|
2014-07-05 13:26:37 +02:00
|
|
|
return;
|
2014-07-12 23:02:52 +02:00
|
|
|
if ((info.suffix().compare("png", Qt::CaseInsensitive) != 0))
|
2014-07-05 13:26:37 +02:00
|
|
|
return;
|
2023-06-20 16:59:59 +12:00
|
|
|
if (!m_cache->stale(m_path))
|
|
|
|
return;
|
|
|
|
QImage image(m_path);
|
|
|
|
if (image.isNull()) {
|
|
|
|
m_resultEmitter.emitResultsFailed(m_path);
|
|
|
|
qDebug() << "Error loading screenshot: " + m_path + ". Perhaps too large?";
|
2014-07-05 13:26:37 +02:00
|
|
|
return;
|
|
|
|
}
|
2023-06-20 16:59:59 +12:00
|
|
|
QImage small;
|
|
|
|
if (image.width() > image.height())
|
|
|
|
small = image.scaledToWidth(512).scaledToWidth(256, Qt::SmoothTransformation);
|
|
|
|
else
|
|
|
|
small = image.scaledToHeight(512).scaledToHeight(256, Qt::SmoothTransformation);
|
|
|
|
QPoint offset((256 - small.width()) / 2, (256 - small.height()) / 2);
|
|
|
|
QImage square(QSize(256, 256), QImage::Format_ARGB32);
|
|
|
|
square.fill(Qt::transparent);
|
|
|
|
|
|
|
|
QPainter painter(&square);
|
|
|
|
painter.drawImage(offset, small);
|
|
|
|
painter.end();
|
|
|
|
|
|
|
|
QIcon icon(QPixmap::fromImage(square));
|
|
|
|
m_cache->add(m_path, icon);
|
|
|
|
m_resultEmitter.emitResultsReady(m_path);
|
2014-07-05 13:26:37 +02:00
|
|
|
}
|
|
|
|
QString m_path;
|
|
|
|
SharedIconCachePtr m_cache;
|
|
|
|
ThumbnailingResult m_resultEmitter;
|
|
|
|
};
|
|
|
|
|
2014-07-12 23:02:52 +02:00
|
|
|
// this is about as elegant and well written as a bag of bricks with scribbles done by insane
|
|
|
|
// asylum patients.
|
2023-08-02 18:35:35 +02:00
|
|
|
class FilterModel : public QIdentityProxyModel {
|
2014-07-05 13:26:37 +02:00
|
|
|
Q_OBJECT
|
2023-08-02 18:35:35 +02:00
|
|
|
public:
|
|
|
|
explicit FilterModel(QObject* parent = 0) : QIdentityProxyModel(parent)
|
2014-07-05 13:26:37 +02:00
|
|
|
{
|
|
|
|
m_thumbnailingPool.setMaxThreadCount(4);
|
|
|
|
m_thumbnailCache = std::make_shared<SharedIconCache>();
|
2021-11-20 16:22:22 +01:00
|
|
|
m_thumbnailCache->add("placeholder", APPLICATION->getThemedIcon("screenshot-placeholder"));
|
2014-07-05 13:26:37 +02:00
|
|
|
connect(&watcher, SIGNAL(fileChanged(QString)), SLOT(fileChanged(QString)));
|
|
|
|
}
|
2023-08-02 18:35:35 +02:00
|
|
|
virtual ~FilterModel()
|
|
|
|
{
|
2023-06-20 16:54:15 +12:00
|
|
|
m_thumbnailingPool.clear();
|
|
|
|
if (!m_thumbnailingPool.waitForDone(500))
|
|
|
|
qDebug() << "Thumbnail pool took longer than 500ms to finish";
|
|
|
|
}
|
2023-08-02 18:35:35 +02:00
|
|
|
virtual QVariant data(const QModelIndex& proxyIndex, int role = Qt::DisplayRole) const
|
2014-06-28 17:09:23 +02:00
|
|
|
{
|
|
|
|
auto model = sourceModel();
|
2014-06-30 02:02:57 +02:00
|
|
|
if (!model)
|
2014-06-28 17:09:23 +02:00
|
|
|
return QVariant();
|
2023-08-02 18:35:35 +02:00
|
|
|
if (role == Qt::DisplayRole || role == Qt::EditRole) {
|
2014-06-30 02:02:57 +02:00
|
|
|
QVariant result = sourceModel()->data(mapToSource(proxyIndex), role);
|
2022-05-02 19:48:37 +02:00
|
|
|
return result.toString().remove(QRegularExpression("\\.png$"));
|
2014-06-30 02:02:57 +02:00
|
|
|
}
|
2023-08-02 18:35:35 +02:00
|
|
|
if (role == Qt::DecorationRole) {
|
|
|
|
QVariant result = sourceModel()->data(mapToSource(proxyIndex), QFileSystemModel::FilePathRole);
|
2014-06-30 02:02:57 +02:00
|
|
|
QString filePath = result.toString();
|
2014-07-05 13:26:37 +02:00
|
|
|
QIcon temp;
|
2023-08-02 18:35:35 +02:00
|
|
|
if (!watched.contains(filePath)) {
|
|
|
|
((QFileSystemWatcher&)watcher).addPath(filePath);
|
|
|
|
((QSet<QString>&)watched).insert(filePath);
|
2014-06-30 02:02:57 +02:00
|
|
|
}
|
2023-08-02 18:35:35 +02:00
|
|
|
if (m_thumbnailCache->get(filePath, temp)) {
|
2014-07-05 13:26:37 +02:00
|
|
|
return temp;
|
2014-06-30 02:02:57 +02:00
|
|
|
}
|
2023-08-02 18:35:35 +02:00
|
|
|
if (!m_failed.contains(filePath)) {
|
|
|
|
((FilterModel*)this)->thumbnailImage(filePath);
|
2014-07-05 13:26:37 +02:00
|
|
|
}
|
2014-07-12 23:02:52 +02:00
|
|
|
return (m_thumbnailCache->get("placeholder"));
|
2014-06-28 17:09:23 +02:00
|
|
|
}
|
2014-07-05 13:26:37 +02:00
|
|
|
return sourceModel()->data(mapToSource(proxyIndex), role);
|
2014-06-28 17:09:23 +02:00
|
|
|
}
|
2023-08-02 18:35:35 +02:00
|
|
|
virtual bool setData(const QModelIndex& index, const QVariant& value, int role = Qt::EditRole)
|
2014-06-28 17:09:23 +02:00
|
|
|
{
|
|
|
|
auto model = sourceModel();
|
2014-06-30 02:02:57 +02:00
|
|
|
if (!model)
|
2014-06-28 17:09:23 +02:00
|
|
|
return false;
|
2014-06-30 02:02:57 +02:00
|
|
|
if (role != Qt::EditRole)
|
2014-06-28 17:09:23 +02:00
|
|
|
return false;
|
|
|
|
// FIXME: this is a workaround for a bug in QFileSystemModel, where it doesn't
|
|
|
|
// sort after renames
|
|
|
|
{
|
2023-08-02 18:35:35 +02:00
|
|
|
((QFileSystemModel*)model)->setNameFilterDisables(true);
|
|
|
|
((QFileSystemModel*)model)->setNameFilterDisables(false);
|
2014-06-28 17:09:23 +02:00
|
|
|
}
|
|
|
|
return model->setData(mapToSource(index), value.toString() + ".png", role);
|
|
|
|
}
|
2014-07-12 23:02:52 +02:00
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
private:
|
2014-07-05 13:26:37 +02:00
|
|
|
void thumbnailImage(QString path)
|
|
|
|
{
|
|
|
|
auto runnable = new ThumbnailRunnable(path, m_thumbnailCache);
|
2023-08-02 18:35:35 +02:00
|
|
|
connect(&(runnable->m_resultEmitter), SIGNAL(resultsReady(QString)), SLOT(thumbnailReady(QString)));
|
|
|
|
connect(&(runnable->m_resultEmitter), SIGNAL(resultsFailed(QString)), SLOT(thumbnailFailed(QString)));
|
|
|
|
((QThreadPool&)m_thumbnailingPool).start(runnable);
|
2014-07-05 13:26:37 +02:00
|
|
|
}
|
2023-08-02 18:35:35 +02:00
|
|
|
private slots:
|
2014-07-12 23:02:52 +02:00
|
|
|
void thumbnailReady(QString path) { emit layoutChanged(); }
|
|
|
|
void thumbnailFailed(QString path) { m_failed.insert(path); }
|
2014-07-05 13:26:37 +02:00
|
|
|
void fileChanged(QString filepath)
|
|
|
|
{
|
|
|
|
m_thumbnailCache->setStale(filepath);
|
|
|
|
// reinsert the path...
|
|
|
|
watcher.removePath(filepath);
|
2023-06-20 16:52:57 +12:00
|
|
|
if (QFile::exists(filepath)) {
|
|
|
|
watcher.addPath(filepath);
|
|
|
|
thumbnailImage(filepath);
|
|
|
|
}
|
2014-07-05 13:26:37 +02:00
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
private:
|
2014-07-05 13:26:37 +02:00
|
|
|
SharedIconCachePtr m_thumbnailCache;
|
|
|
|
QThreadPool m_thumbnailingPool;
|
|
|
|
QSet<QString> m_failed;
|
|
|
|
QSet<QString> watched;
|
|
|
|
QFileSystemWatcher watcher;
|
2014-06-28 17:09:23 +02:00
|
|
|
};
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
class CenteredEditingDelegate : public QStyledItemDelegate {
|
|
|
|
public:
|
|
|
|
explicit CenteredEditingDelegate(QObject* parent = 0) : QStyledItemDelegate(parent) {}
|
2014-07-12 23:02:52 +02:00
|
|
|
virtual ~CenteredEditingDelegate() {}
|
2023-08-02 18:35:35 +02:00
|
|
|
virtual QWidget* createEditor(QWidget* parent, const QStyleOptionViewItem& option, const QModelIndex& index) const
|
2014-06-28 17:09:23 +02:00
|
|
|
{
|
|
|
|
auto widget = QStyledItemDelegate::createEditor(parent, option, index);
|
2023-08-02 18:35:35 +02:00
|
|
|
auto foo = dynamic_cast<QLineEdit*>(widget);
|
|
|
|
if (foo) {
|
2014-06-28 17:09:23 +02:00
|
|
|
foo->setAlignment(Qt::AlignHCenter);
|
|
|
|
foo->setFrame(true);
|
|
|
|
foo->setMaximumWidth(192);
|
|
|
|
}
|
|
|
|
return widget;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
ScreenshotsPage::ScreenshotsPage(QString path, QWidget* parent) : QMainWindow(parent), ui(new Ui::ScreenshotsPage)
|
2014-06-28 17:09:23 +02:00
|
|
|
{
|
|
|
|
m_model.reset(new QFileSystemModel());
|
|
|
|
m_filterModel.reset(new FilterModel());
|
|
|
|
m_filterModel->setSourceModel(m_model.get());
|
2022-05-09 00:54:47 -04:00
|
|
|
m_model->setFilter(QDir::Files);
|
2014-06-28 17:09:23 +02:00
|
|
|
m_model->setReadOnly(false);
|
2023-08-02 18:35:35 +02:00
|
|
|
m_model->setNameFilters({ "*.png" });
|
2014-07-05 13:26:37 +02:00
|
|
|
m_model->setNameFilterDisables(false);
|
2015-01-27 22:31:07 +01:00
|
|
|
m_folder = path;
|
2015-10-05 01:47:27 +02:00
|
|
|
m_valid = FS::ensureFolderPathExists(m_folder);
|
2014-06-28 17:09:23 +02:00
|
|
|
|
|
|
|
ui->setupUi(this);
|
2019-07-23 00:48:14 +02:00
|
|
|
ui->toolBar->insertSpacer(ui->actionView_Folder);
|
2019-07-17 02:09:54 +02:00
|
|
|
|
2014-06-28 17:09:23 +02:00
|
|
|
ui->listView->setIconSize(QSize(128, 128));
|
2014-07-05 13:26:37 +02:00
|
|
|
ui->listView->setGridSize(QSize(192, 160));
|
2014-06-28 17:09:23 +02:00
|
|
|
ui->listView->setSpacing(9);
|
|
|
|
// ui->listView->setUniformItemSizes(true);
|
|
|
|
ui->listView->setLayoutMode(QListView::Batched);
|
|
|
|
ui->listView->setViewMode(QListView::IconMode);
|
|
|
|
ui->listView->setResizeMode(QListView::Adjust);
|
|
|
|
ui->listView->installEventFilter(this);
|
2022-05-02 19:10:45 +02:00
|
|
|
ui->listView->setEditTriggers(QAbstractItemView::NoEditTriggers);
|
2014-06-28 17:09:23 +02:00
|
|
|
ui->listView->setItemDelegate(new CenteredEditingDelegate(this));
|
2019-07-25 01:02:30 +02:00
|
|
|
ui->listView->setContextMenuPolicy(Qt::CustomContextMenu);
|
|
|
|
connect(ui->listView, &QListView::customContextMenuRequested, this, &ScreenshotsPage::ShowContextMenu);
|
2014-06-28 17:09:23 +02:00
|
|
|
connect(ui->listView, SIGNAL(activated(QModelIndex)), SLOT(onItemActivated(QModelIndex)));
|
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
bool ScreenshotsPage::eventFilter(QObject* obj, QEvent* evt)
|
2014-06-28 17:09:23 +02:00
|
|
|
{
|
|
|
|
if (obj != ui->listView)
|
|
|
|
return QWidget::eventFilter(obj, evt);
|
2023-08-02 18:35:35 +02:00
|
|
|
if (evt->type() != QEvent::KeyPress) {
|
2014-06-28 17:09:23 +02:00
|
|
|
return QWidget::eventFilter(obj, evt);
|
|
|
|
}
|
2023-08-02 18:35:35 +02:00
|
|
|
QKeyEvent* keyEvent = static_cast<QKeyEvent*>(evt);
|
2021-12-03 03:02:58 +01:00
|
|
|
|
|
|
|
if (keyEvent->matches(QKeySequence::Copy)) {
|
2021-12-03 16:08:11 +01:00
|
|
|
on_actionCopy_File_s_triggered();
|
2021-12-03 03:02:58 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
switch (keyEvent->key()) {
|
|
|
|
case Qt::Key_Delete:
|
|
|
|
on_actionDelete_triggered();
|
|
|
|
return true;
|
|
|
|
case Qt::Key_F2:
|
|
|
|
on_actionRename_triggered();
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
break;
|
2014-06-28 17:09:23 +02:00
|
|
|
}
|
|
|
|
return QWidget::eventFilter(obj, evt);
|
|
|
|
}
|
|
|
|
|
2022-02-22 18:23:53 +00:00
|
|
|
void ScreenshotsPage::retranslate()
|
|
|
|
{
|
|
|
|
ui->retranslateUi(this);
|
|
|
|
}
|
|
|
|
|
2014-06-28 17:09:23 +02:00
|
|
|
ScreenshotsPage::~ScreenshotsPage()
|
|
|
|
{
|
|
|
|
delete ui;
|
|
|
|
}
|
|
|
|
|
2019-07-25 01:02:30 +02:00
|
|
|
void ScreenshotsPage::ShowContextMenu(const QPoint& pos)
|
|
|
|
{
|
|
|
|
auto menu = ui->toolBar->createContextMenu(this, tr("Context menu"));
|
2021-12-03 16:29:28 +01:00
|
|
|
|
|
|
|
if (ui->listView->selectionModel()->selectedRows().size() > 1) {
|
2023-08-02 18:35:35 +02:00
|
|
|
menu->removeAction(ui->actionCopy_Image);
|
2021-12-03 16:29:28 +01:00
|
|
|
}
|
|
|
|
|
2019-07-25 01:02:30 +02:00
|
|
|
menu->exec(ui->listView->mapToGlobal(pos));
|
|
|
|
delete menu;
|
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
QMenu* ScreenshotsPage::createPopupMenu()
|
2019-07-17 02:01:29 +02:00
|
|
|
{
|
|
|
|
QMenu* filteredMenu = QMainWindow::createPopupMenu();
|
2023-08-02 18:35:35 +02:00
|
|
|
filteredMenu->removeAction(ui->toolBar->toggleViewAction());
|
2019-07-17 02:01:29 +02:00
|
|
|
return filteredMenu;
|
|
|
|
}
|
|
|
|
|
2014-06-28 17:09:23 +02:00
|
|
|
void ScreenshotsPage::onItemActivated(QModelIndex index)
|
|
|
|
{
|
|
|
|
if (!index.isValid())
|
|
|
|
return;
|
|
|
|
auto info = m_model->fileInfo(index);
|
|
|
|
QString fileName = info.absoluteFilePath();
|
2016-01-05 07:32:52 +01:00
|
|
|
DesktopServices::openFile(info.absoluteFilePath());
|
2014-06-28 17:09:23 +02:00
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
void ScreenshotsPage::onCurrentSelectionChanged(const QItemSelection& selected)
|
2022-05-09 00:54:47 -04:00
|
|
|
{
|
|
|
|
bool allReadable = !selected.isEmpty();
|
|
|
|
bool allWritable = !selected.isEmpty();
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
for (auto index : selected.indexes()) {
|
2022-05-09 00:54:47 -04:00
|
|
|
if (!index.isValid())
|
|
|
|
break;
|
|
|
|
auto info = m_model->fileInfo(index);
|
|
|
|
if (!info.isReadable())
|
|
|
|
allReadable = false;
|
|
|
|
if (!info.isWritable())
|
|
|
|
allWritable = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ui->actionUpload->setEnabled(allReadable);
|
|
|
|
ui->actionCopy_Image->setEnabled(allReadable);
|
|
|
|
ui->actionCopy_File_s->setEnabled(allReadable);
|
|
|
|
ui->actionDelete->setEnabled(allWritable);
|
|
|
|
ui->actionRename->setEnabled(allWritable);
|
|
|
|
}
|
|
|
|
|
2019-07-17 02:01:29 +02:00
|
|
|
void ScreenshotsPage::on_actionView_Folder_triggered()
|
2014-06-28 17:09:23 +02:00
|
|
|
{
|
2016-01-05 07:32:52 +01:00
|
|
|
DesktopServices::openDirectory(m_folder, true);
|
2014-06-28 17:09:23 +02:00
|
|
|
}
|
|
|
|
|
2019-07-17 02:01:29 +02:00
|
|
|
void ScreenshotsPage::on_actionUpload_triggered()
|
2014-06-28 17:09:23 +02:00
|
|
|
{
|
2017-08-19 23:14:06 +02:00
|
|
|
auto selection = ui->listView->selectionModel()->selectedRows();
|
2014-06-28 17:09:23 +02:00
|
|
|
if (selection.isEmpty())
|
|
|
|
return;
|
|
|
|
|
2022-12-14 15:02:04 +00:00
|
|
|
QString text;
|
2023-06-27 19:31:36 +03:00
|
|
|
QUrl baseUrl(BuildConfig.IMGUR_BASE_URL);
|
2022-12-14 15:02:04 +00:00
|
|
|
if (selection.size() > 1)
|
2023-06-27 19:31:36 +03:00
|
|
|
text = tr("You are about to upload %1 screenshots to %2.\n"
|
|
|
|
"You should double-check for personal information.\n\n"
|
2022-12-14 15:02:04 +00:00
|
|
|
"Are you sure?")
|
2023-06-27 19:31:36 +03:00
|
|
|
.arg(QString::number(selection.size()), baseUrl.host());
|
2022-12-14 15:02:04 +00:00
|
|
|
else
|
2023-06-27 19:31:36 +03:00
|
|
|
text = tr("You are about to upload the selected screenshot to %1.\n"
|
|
|
|
"You should double-check for personal information.\n\n"
|
|
|
|
"Are you sure?")
|
|
|
|
.arg(baseUrl.host());
|
2022-12-14 15:02:04 +00:00
|
|
|
|
|
|
|
auto response = CustomMessageBox::selectable(this, "Confirm Upload", text, QMessageBox::Warning, QMessageBox::Yes | QMessageBox::No,
|
|
|
|
QMessageBox::No)
|
|
|
|
->exec();
|
|
|
|
|
|
|
|
if (response != QMessageBox::Yes)
|
|
|
|
return;
|
|
|
|
|
2021-11-21 23:21:12 +01:00
|
|
|
QList<ScreenShot::Ptr> uploaded;
|
2021-12-31 05:27:59 +01:00
|
|
|
auto job = NetJob::Ptr(new NetJob("Screenshot Upload", APPLICATION->network()));
|
2023-08-02 18:35:35 +02:00
|
|
|
if (selection.size() < 2) {
|
2019-06-18 14:41:07 +02:00
|
|
|
auto item = selection.at(0);
|
|
|
|
auto info = m_model->fileInfo(item);
|
|
|
|
auto screenshot = std::make_shared<ScreenShot>(info);
|
|
|
|
job->addNetAction(ImgurUpload::make(screenshot));
|
|
|
|
|
|
|
|
m_uploadActive = true;
|
|
|
|
ProgressDialog dialog(this);
|
2021-12-29 17:08:15 +01:00
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
if (dialog.execWithTask(job.get()) != QDialog::Accepted) {
|
|
|
|
CustomMessageBox::selectable(this, tr("Failed to upload screenshots!"), tr("Unknown error"), QMessageBox::Warning)->exec();
|
|
|
|
} else {
|
2019-06-18 14:41:07 +02:00
|
|
|
auto link = screenshot->m_url;
|
2023-08-02 18:35:35 +02:00
|
|
|
QClipboard* clipboard = QApplication::clipboard();
|
2019-06-18 14:41:07 +02:00
|
|
|
clipboard->setText(link);
|
|
|
|
CustomMessageBox::selectable(
|
2023-08-02 18:35:35 +02:00
|
|
|
this, tr("Upload finished"),
|
|
|
|
tr("The <a href=\"%1\">link to the uploaded screenshot</a> has been placed in your clipboard.").arg(link),
|
|
|
|
QMessageBox::Information)
|
|
|
|
->exec();
|
2019-06-18 14:41:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
m_uploadActive = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
for (auto item : selection) {
|
2014-06-28 17:09:23 +02:00
|
|
|
auto info = m_model->fileInfo(item);
|
|
|
|
auto screenshot = std::make_shared<ScreenShot>(info);
|
|
|
|
uploaded.push_back(screenshot);
|
|
|
|
job->addNetAction(ImgurUpload::make(screenshot));
|
|
|
|
}
|
|
|
|
SequentialTask task;
|
2021-12-31 05:27:59 +01:00
|
|
|
auto albumTask = NetJob::Ptr(new NetJob("Imgur Album Creation", APPLICATION->network()));
|
2014-06-28 17:09:23 +02:00
|
|
|
auto imgurAlbum = ImgurAlbumCreation::make(uploaded);
|
|
|
|
albumTask->addNetAction(imgurAlbum);
|
2021-11-20 17:08:34 +01:00
|
|
|
task.addTask(job);
|
|
|
|
task.addTask(albumTask);
|
2015-12-03 23:00:51 +01:00
|
|
|
m_uploadActive = true;
|
2014-06-28 17:09:23 +02:00
|
|
|
ProgressDialog prog(this);
|
2023-08-02 18:35:35 +02:00
|
|
|
if (prog.execWithTask(&task) != QDialog::Accepted) {
|
|
|
|
CustomMessageBox::selectable(this, tr("Failed to upload screenshots!"), tr("Unknown error"), QMessageBox::Warning)->exec();
|
|
|
|
} else {
|
2014-07-07 00:02:04 +02:00
|
|
|
auto link = QString("https://imgur.com/a/%1").arg(imgurAlbum->id());
|
2023-08-02 18:35:35 +02:00
|
|
|
QClipboard* clipboard = QApplication::clipboard();
|
2014-07-07 00:02:04 +02:00
|
|
|
clipboard->setText(link);
|
2023-08-02 18:35:35 +02:00
|
|
|
CustomMessageBox::selectable(this, tr("Upload finished"),
|
|
|
|
tr("The <a href=\"%1\">link to the uploaded album</a> has been placed in your clipboard.").arg(link),
|
|
|
|
QMessageBox::Information)
|
|
|
|
->exec();
|
2014-06-28 17:09:23 +02:00
|
|
|
}
|
2015-12-03 23:00:51 +01:00
|
|
|
m_uploadActive = false;
|
2014-06-28 17:09:23 +02:00
|
|
|
}
|
|
|
|
|
2021-12-03 16:08:11 +01:00
|
|
|
void ScreenshotsPage::on_actionCopy_Image_triggered()
|
2021-12-03 03:02:58 +01:00
|
|
|
{
|
|
|
|
auto selection = ui->listView->selectionModel()->selectedRows();
|
2023-08-02 18:35:35 +02:00
|
|
|
if (selection.size() < 1) {
|
2021-12-03 03:02:58 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// You can only copy one image to the clipboard. In the case of multiple selected files, only the first one gets copied.
|
|
|
|
auto item = selection[0];
|
|
|
|
auto info = m_model->fileInfo(item);
|
|
|
|
QImage image(info.absoluteFilePath());
|
|
|
|
Q_ASSERT(!image.isNull());
|
|
|
|
QApplication::clipboard()->setImage(image, QClipboard::Clipboard);
|
|
|
|
}
|
|
|
|
|
2021-12-03 16:08:11 +01:00
|
|
|
void ScreenshotsPage::on_actionCopy_File_s_triggered()
|
|
|
|
{
|
|
|
|
auto selection = ui->listView->selectionModel()->selectedRows();
|
2023-08-02 18:35:35 +02:00
|
|
|
if (selection.size() < 1) {
|
2021-12-03 16:08:11 +01:00
|
|
|
// Don't do anything so we don't empty the users clipboard
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString buf = "";
|
2023-08-02 18:35:35 +02:00
|
|
|
for (auto item : selection) {
|
2021-12-03 16:08:11 +01:00
|
|
|
auto info = m_model->fileInfo(item);
|
|
|
|
buf += "file:///" + info.absoluteFilePath() + "\r\n";
|
|
|
|
}
|
|
|
|
QMimeData* mimeData = new QMimeData();
|
|
|
|
mimeData->setData("text/uri-list", buf.toLocal8Bit());
|
|
|
|
QApplication::clipboard()->setMimeData(mimeData);
|
|
|
|
}
|
|
|
|
|
2019-07-17 02:01:29 +02:00
|
|
|
void ScreenshotsPage::on_actionDelete_triggered()
|
2014-06-28 17:09:23 +02:00
|
|
|
{
|
2022-12-14 15:02:04 +00:00
|
|
|
auto selected = ui->listView->selectionModel()->selectedIndexes();
|
|
|
|
|
|
|
|
int count = ui->listView->selectionModel()->selectedRows().size();
|
|
|
|
QString text;
|
|
|
|
if (count > 1)
|
2022-12-26 14:33:50 +00:00
|
|
|
text = tr("You are about to delete %1 screenshots.\n"
|
2022-12-14 15:02:04 +00:00
|
|
|
"This may be permanent and they will be gone from the folder.\n\n"
|
|
|
|
"Are you sure?")
|
|
|
|
.arg(count);
|
|
|
|
else
|
2022-12-26 14:33:50 +00:00
|
|
|
text = tr("You are about to delete the selected screenshot.\n"
|
2022-12-14 15:02:04 +00:00
|
|
|
"This may be permanent and it will be gone from the folder.\n\n"
|
|
|
|
"Are you sure?")
|
|
|
|
.arg(count);
|
2014-06-28 17:09:23 +02:00
|
|
|
|
2022-12-14 15:02:04 +00:00
|
|
|
auto response =
|
2022-12-17 09:26:06 +00:00
|
|
|
CustomMessageBox::selectable(this, tr("Confirm Deletion"), text, QMessageBox::Warning, QMessageBox::Yes | QMessageBox::No)->exec();
|
2022-12-14 15:02:04 +00:00
|
|
|
|
|
|
|
if (response != QMessageBox::Yes)
|
2014-06-28 17:09:23 +02:00
|
|
|
return;
|
|
|
|
|
2023-08-02 18:35:35 +02:00
|
|
|
for (auto item : selected) {
|
2022-12-14 15:02:04 +00:00
|
|
|
if (FS::trash(m_model->filePath(item)))
|
|
|
|
continue;
|
|
|
|
|
2014-06-28 17:09:23 +02:00
|
|
|
m_model->remove(item);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-17 02:01:29 +02:00
|
|
|
void ScreenshotsPage::on_actionRename_triggered()
|
2014-06-28 17:09:23 +02:00
|
|
|
{
|
|
|
|
auto selection = ui->listView->selectionModel()->selectedIndexes();
|
|
|
|
if (selection.isEmpty())
|
|
|
|
return;
|
|
|
|
ui->listView->edit(selection[0]);
|
|
|
|
// TODO: mass renaming
|
|
|
|
}
|
|
|
|
|
2018-03-19 02:36:12 +01:00
|
|
|
void ScreenshotsPage::openedImpl()
|
2014-06-28 17:09:23 +02:00
|
|
|
{
|
2023-08-02 18:35:35 +02:00
|
|
|
if (!m_valid) {
|
2016-11-22 00:56:48 +01:00
|
|
|
m_valid = FS::ensureFolderPathExists(m_folder);
|
|
|
|
}
|
2023-08-02 18:35:35 +02:00
|
|
|
if (m_valid) {
|
2014-06-28 17:09:23 +02:00
|
|
|
QString path = QDir(m_folder).absolutePath();
|
2016-11-22 00:56:48 +01:00
|
|
|
auto idx = m_model->setRootPath(path);
|
2023-08-02 18:35:35 +02:00
|
|
|
if (idx.isValid()) {
|
2016-11-22 00:56:48 +01:00
|
|
|
ui->listView->setModel(m_filterModel.get());
|
2023-08-02 18:35:35 +02:00
|
|
|
connect(ui->listView->selectionModel(), &QItemSelectionModel::selectionChanged, this,
|
|
|
|
&ScreenshotsPage::onCurrentSelectionChanged);
|
|
|
|
onCurrentSelectionChanged(ui->listView->selectionModel()->selection()); // set initial button enable states
|
2016-11-22 00:56:48 +01:00
|
|
|
ui->listView->setRootIndex(m_filterModel->mapFromSource(idx));
|
2023-08-02 18:35:35 +02:00
|
|
|
} else {
|
2016-11-22 00:56:48 +01:00
|
|
|
ui->listView->setModel(nullptr);
|
|
|
|
}
|
2014-06-28 17:09:23 +02:00
|
|
|
}
|
2022-11-19 17:12:31 -03:00
|
|
|
|
|
|
|
auto const setting_name = QString("WideBarVisibility_%1").arg(id());
|
|
|
|
if (!APPLICATION->settings()->contains(setting_name))
|
|
|
|
m_wide_bar_setting = APPLICATION->settings()->registerSetting(setting_name);
|
|
|
|
else
|
|
|
|
m_wide_bar_setting = APPLICATION->settings()->getSetting(setting_name);
|
|
|
|
|
|
|
|
ui->toolBar->setVisibilityState(m_wide_bar_setting->get().toByteArray());
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScreenshotsPage::closedImpl()
|
|
|
|
{
|
|
|
|
m_wide_bar_setting->set(ui->toolBar->getVisibilityState());
|
2014-06-28 17:09:23 +02:00
|
|
|
}
|
2014-07-05 13:26:37 +02:00
|
|
|
|
|
|
|
#include "ScreenshotsPage.moc"
|