diff --git a/MainWindow/Window.cpp b/MainWindow/Window.cpp
index a6ffb98d..7e5330d3 100644
--- a/MainWindow/Window.cpp
+++ b/MainWindow/Window.cpp
@@ -1,1827 +1,1830 @@
/* Copyright (C) 2003-2020 The KPhotoAlbum Development Team
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; either
version 2 of the License, or (at your option) any later version.
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; see the file COPYING. If not, write to
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Boston, MA 02110-1301, USA.
*/
#include "Window.h"
#include
#include
#ifdef HAVE_STDLIB_H
#include
#endif
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
+#include
#include
#include
#include
#include
#include // for #if KIO_VERSION...
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#ifdef KF5Purpose_FOUND
#include
#endif
#include "AutoStackImages.h"
#include "BreadcrumbViewer.h"
#include "CopyPopup.h"
#include "DeleteDialog.h"
#include "DirtyIndicator.h"
#include "DuplicateMerger/DuplicateMerger.h"
#include "ExternalPopup.h"
#include "FeatureDialog.h"
#include "ImageCounter.h"
#include "InvalidDateFinder.h"
#include "Logging.h"
#include "Options.h"
#include "SearchBar.h"
#include "SplashScreen.h"
#include "StatisticsDialog.h"
#include "StatusBar.h"
#include "TokenEditor.h"
#include "UpdateVideoThumbnail.h"
#include "WelcomeDialog.h"
#ifdef HAVE_MARBLE
#include
"),
i18n("Feature has not been configured"));
}
}
void MainWindow::Window::setupStatusBar()
{
m_statusBar = new MainWindow::StatusBar;
setStatusBar(m_statusBar);
setLocked(Settings::SettingsData::instance()->locked(), true, false);
connect(m_statusBar, &StatusBar::thumbnailSettingsRequested, [this]() { this->slotOptions(); m_settingsDialog->activatePage(Settings::SettingsPage::ThumbnailsPage); });
}
void MainWindow::Window::slotRecreateExifDB()
{
Exif::Database::instance()->recreate();
}
void MainWindow::Window::useNextVideoThumbnail()
{
UpdateVideoThumbnail::useNext(selected());
}
void MainWindow::Window::usePreviousVideoThumbnail()
{
UpdateVideoThumbnail::usePrevious(selected());
}
void MainWindow::Window::mergeDuplicates()
{
DuplicateMerger *merger = new DuplicateMerger;
merger->show();
}
void MainWindow::Window::slotThumbnailSizeChanged()
{
+ QPixmapCache::clear();
+
QString thumbnailSizeMsg = i18nc("@info:status",
//xgettext:no-c-format
"Thumbnail width: %1px (storage size: %2px)",
Settings::SettingsData::instance()->actualThumbnailSize(),
Settings::SettingsData::instance()->thumbnailSize());
m_statusBar->showMessage(thumbnailSizeMsg, 4000);
}
void MainWindow::Window::createSearchBar()
{
// Set up the search tool bar
SearchBar *searchBar = new SearchBar(this);
searchBar->setLineEditEnabled(false);
searchBar->setObjectName(QString::fromUtf8("searchBar"));
connect(searchBar, &SearchBar::textChanged, m_browser, &Browser::BrowserWidget::slotLimitToMatch);
connect(searchBar, &SearchBar::returnPressed, m_browser, &Browser::BrowserWidget::slotInvokeSeleted);
connect(searchBar, &SearchBar::keyPressed, m_browser, &Browser::BrowserWidget::scrollKeyPressed);
connect(m_browser, &Browser::BrowserWidget::viewChanged, searchBar, &SearchBar::reset);
connect(m_browser, &Browser::BrowserWidget::isSearchable, searchBar, &SearchBar::setLineEditEnabled);
m_filterWidget = m_thumbnailView->createFilterWidget(this);
addToolBar(m_filterWidget);
m_filterWidget->setObjectName(QString::fromUtf8("filterBar"));
connect(m_browser, &Browser::BrowserWidget::viewChanged, ThumbnailView::ThumbnailFacade::instance(), &ThumbnailView::ThumbnailFacade::clearFilter);
connect(m_browser, &Browser::BrowserWidget::isFilterable, m_filterWidget, &ThumbnailView::FilterWidget::setEnabled);
}
void MainWindow::Window::executeStartupActions()
{
Q_ASSERT(m_thumbnailCache);
new ImageManager::ThumbnailBuilder(m_statusBar, this, m_thumbnailCache);
if (!Settings::SettingsData::instance()->incrementalThumbnails())
ImageManager::ThumbnailBuilder::instance()->buildMissing();
connect(m_thumbnailCache, &ImageManager::ThumbnailCache::cacheInvalidated,
this, &Window::slotBuildThumbnailsIfWanted);
if (!FeatureDialog::hasVideoThumbnailer()) {
BackgroundTaskManager::JobManager::instance()->addJob(
new BackgroundJobs::SearchForVideosWithoutLengthInfo);
BackgroundTaskManager::JobManager::instance()->addJob(
new BackgroundJobs::SearchForVideosWithoutVideoThumbnailsJob);
}
}
void MainWindow::Window::checkIfVideoThumbnailerIsInstalled()
{
if (Options::the()->demoMode())
return;
if (!FeatureDialog::hasVideoThumbnailer()) {
KMessageBox::information(this,
i18n("Unable to find ffmpeg on the system.
"
"Without it, KPhotoAlbum will not be able to display video thumbnails and video lengths. "
"Please install the ffmpeg package
"),
i18n("Video thumbnails are not available"), QString::fromLatin1("VideoThumbnailerNotInstalled"));
}
}
bool MainWindow::Window::anyVideosSelected() const
{
const auto selectedFiles = selected();
for (const DB::FileName &fileName : selectedFiles) {
if (Utilities::isVideo(fileName))
return true;
}
return false;
}
void MainWindow::Window::setHistogramVisibilty(bool visible) const
{
if (visible) {
m_dateBar->show();
m_dateBarLine->show();
} else {
m_dateBar->hide();
m_dateBarLine->hide();
}
}
void MainWindow::Window::slotImageRotated(const DB::FileName &fileName)
{
// An image has been rotated by the annotation dialog or the viewer.
// We have to reload the respective thumbnail to get it in the right angle
thumbnailCache()->removeThumbnail(fileName);
}
bool MainWindow::Window::dbIsDirty() const
{
return m_statusBar->mp_dirtyIndicator->isSaveDirty();
}
#ifdef HAVE_MARBLE
void MainWindow::Window::showPositionBrowser()
{
auto positionBrowser = positionBrowserWidget();
m_stack->setCurrentWidget(positionBrowser);
updateStates(false);
}
Map::MapView *MainWindow::Window::positionBrowserWidget()
{
if (!m_positionBrowser) {
m_positionBrowser = new Map::MapView(m_stack, Map::UsageType::InlineMapView);
m_stack->addWidget(m_positionBrowser);
}
return m_positionBrowser;
}
#endif
UserFeedback MainWindow::Window::askWarningContinueCancel(const QString &msg, const QString &title, const QString &dialogId)
{
auto answer = KMessageBox::warningContinueCancel(this, msg, title, KStandardGuiItem::cont(), KStandardGuiItem::cancel(), dialogId);
return (answer == KMessageBox::Continue) ? UserFeedback::Confirm : UserFeedback::Deny;
}
UserFeedback MainWindow::Window::askQuestionYesNo(const QString &msg, const QString &title, const QString &dialogId)
{
auto answer = KMessageBox::questionYesNo(this, msg, title, KStandardGuiItem::yes(), KStandardGuiItem::no(), dialogId);
return (answer == KMessageBox::Yes) ? UserFeedback::Confirm : UserFeedback::Deny;
}
void MainWindow::Window::showInformation(const QString &msg, const QString &title, const QString &dialogId)
{
KMessageBox::information(this, msg, title, dialogId);
}
void MainWindow::Window::showSorry(const QString &msg, const QString &title, const QString &)
{
KMessageBox::sorry(this, msg, title);
}
void MainWindow::Window::showError(const QString &msg, const QString &title, const QString &)
{
KMessageBox::error(this, msg, title);
}
bool MainWindow::Window::isDialogDisabled(const QString &dialogId)
{
// Note(jzarl): there are different methods for different kinds of dialogs.
// However, all these methods share exactly the same code in KMessageBox.
// If that ever changes, we can still update our implementation - until then I won't just copy a stupid API...
return !KMessageBox::shouldBeShownContinue(dialogId);
}
// vi:expandtab:tabstop=4 shiftwidth=4:
diff --git a/Settings/SettingsData.cpp b/Settings/SettingsData.cpp
index cd9bfe01..7e568b2d 100644
--- a/Settings/SettingsData.cpp
+++ b/Settings/SettingsData.cpp
@@ -1,598 +1,595 @@
/* Copyright (C) 2003-2020 The KPhotoAlbum Development Team
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; either version 2 of
the License or (at your option) version 3 or any later version
accepted by the membership of KDE e. V. (or its successor approved
by the membership of KDE e. V.), which shall act as a proxy
defined in Section 14 of version 3 of the license.
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 .
*/
#include "SettingsData.h"
#include
#include
#include
#include
-#include
#include
#include
#include
#define STR(x) QString::fromLatin1(x)
#define value(GROUP, OPTION, DEFAULT) \
KSharedConfig::openConfig()->group(GROUP).readEntry(OPTION, DEFAULT)
#define setValue(GROUP, OPTION, VALUE) \
{ \
KConfigGroup group = KSharedConfig::openConfig()->group(GROUP); \
group.writeEntry(OPTION, VALUE); \
group.sync(); \
}
#define getValueFunc_(TYPE, FUNC, GROUP, OPTION, DEFAULT) \
TYPE SettingsData::FUNC() const \
{ \
return (TYPE)value(GROUP, OPTION, DEFAULT); \
}
#define setValueFunc_(FUNC, TYPE, GROUP, OPTION, VALUE) \
void SettingsData::FUNC(const TYPE v) \
{ \
setValue(GROUP, OPTION, VALUE); \
}
#define getValueFunc(TYPE, FUNC, GROUP, DEFAULT) getValueFunc_(TYPE, FUNC, #GROUP, #FUNC, DEFAULT)
#define setValueFunc(FUNC, TYPE, GROUP, OPTION) setValueFunc_(FUNC, TYPE, #GROUP, #OPTION, v)
// TODO(mfwitten): document parameters.
#define property_(GET_TYPE, GET_FUNC, GET_VALUE, SET_FUNC, SET_TYPE, SET_VALUE, GROUP, OPTION, GET_DEFAULT_1, GET_DEFAULT_2, GET_DEFAULT_2_TYPE) \
GET_TYPE SettingsData::GET_FUNC() const \
{ \
KConfigGroup g = KSharedConfig::openConfig()->group(GROUP); \
\
if (!g.hasKey(OPTION)) \
return GET_DEFAULT_1; \
\
GET_DEFAULT_2_TYPE v = g.readEntry(OPTION, (GET_DEFAULT_2_TYPE)GET_DEFAULT_2); \
return (GET_TYPE)GET_VALUE; \
} \
setValueFunc_(SET_FUNC, SET_TYPE, GROUP, OPTION, SET_VALUE)
#define property(GET_TYPE, GET_FUNC, SET_FUNC, SET_TYPE, SET_VALUE, GROUP, OPTION, GET_DEFAULT) \
getValueFunc_(GET_TYPE, GET_FUNC, GROUP, OPTION, GET_DEFAULT) \
setValueFunc_(SET_FUNC, SET_TYPE, GROUP, OPTION, SET_VALUE)
#define property_copy(GET_FUNC, SET_FUNC, TYPE, GROUP, GET_DEFAULT) \
property(TYPE, GET_FUNC, SET_FUNC, TYPE, v, #GROUP, #GET_FUNC, GET_DEFAULT)
#define property_ref_(GET_FUNC, SET_FUNC, TYPE, GROUP, GET_DEFAULT) \
property(TYPE, GET_FUNC, SET_FUNC, TYPE &, v, GROUP, #GET_FUNC, GET_DEFAULT)
#define property_ref(GET_FUNC, SET_FUNC, TYPE, GROUP, GET_DEFAULT) \
property(TYPE, GET_FUNC, SET_FUNC, TYPE &, v, #GROUP, #GET_FUNC, GET_DEFAULT)
#define property_enum(GET_FUNC, SET_FUNC, TYPE, GROUP, GET_DEFAULT) \
property(TYPE, GET_FUNC, SET_FUNC, TYPE, (int)v, #GROUP, #GET_FUNC, (int)GET_DEFAULT)
#define property_sset(GET_FUNC, SET_FUNC, GROUP, GET_DEFAULT) \
property_(StringSet, GET_FUNC, v.toSet(), SET_FUNC, StringSet &, v.toList(), #GROUP, #GET_FUNC, GET_DEFAULT, QStringList(), QStringList)
/**
* smoothScale() is called from the image loading thread, therefore we need
* to cache it this way, rather than going to KConfig.
*/
static bool _smoothScale = true;
using namespace Settings;
const WindowType Settings::MainWindow = "MainWindow";
const WindowType Settings::AnnotationDialog = "AnnotationDialog";
SettingsData *SettingsData::s_instance = nullptr;
SettingsData *SettingsData::instance()
{
if (!s_instance)
qFatal("instance called before loading a setup!");
return s_instance;
}
bool SettingsData::ready()
{
return s_instance;
}
void SettingsData::setup(const QString &imageDirectory, DB::UIDelegate &delegate)
{
if (!s_instance)
s_instance = new SettingsData(imageDirectory, delegate);
}
SettingsData::SettingsData(const QString &imageDirectory, DB::UIDelegate &delegate)
: m_UI(delegate)
{
m_hasAskedAboutTimeStamps = false;
QString s = STR("/");
m_imageDirectory = imageDirectory.endsWith(s) ? imageDirectory : imageDirectory + s;
_smoothScale = value("Viewer", "smoothScale", true);
// Split the list of Exif comments that should be stripped automatically to a list
QStringList commentsToStrip = value("General", "commentsToStrip", QString::fromLatin1("Exif_JPEG_PICTURE-,-OLYMPUS DIGITAL CAMERA-,-JENOPTIK DIGITAL CAMERA-,-")).split(QString::fromLatin1("-,-"), QString::SkipEmptyParts);
for (QString &comment : commentsToStrip)
comment.replace(QString::fromLatin1(",,"), QString::fromLatin1(","));
m_EXIFCommentsToStrip = commentsToStrip;
}
/////////////////
//// General ////
/////////////////
// clang-format off
property_copy(useEXIFRotate, setUseEXIFRotate, bool, General, true)
property_copy(useEXIFComments, setUseEXIFComments, bool, General, true)
property_copy(stripEXIFComments, setStripEXIFComments, bool, General, true)
property_copy(commentsToStrip, setCommentsToStrip, QString, General, "" /* see constructor */)
property_copy(searchForImagesOnStart, setSearchForImagesOnStart, bool, General, true)
property_copy(ignoreFileExtension, setIgnoreFileExtension, bool, General, false)
property_copy(skipSymlinks, setSkipSymlinks, bool, General, false)
property_copy(skipRawIfOtherMatches, setSkipRawIfOtherMatches, bool, General, false)
property_copy(useRawThumbnail, setUseRawThumbnail, bool, General, true)
property_copy(useRawThumbnailSize, setUseRawThumbnailSize, QSize, General, QSize(1024, 768))
property_copy(useCompressedIndexXML, setUseCompressedIndexXML, bool, General, true)
property_copy(compressBackup, setCompressBackup, bool, General, true)
property_copy(showSplashScreen, setShowSplashScreen, bool, General, true)
property_copy(showHistogram, setShowHistogram, bool, General, true)
property_copy(autoSave, setAutoSave, int, General, 5)
property_copy(backupCount, setBackupCount, int, General, 5)
property_enum(tTimeStamps, setTTimeStamps, TimeStampTrust, General, Always)
property_copy(excludeDirectories, setExcludeDirectories, QString, General, QString::fromLatin1("xml,ThumbNails,.thumbs"))
#ifdef KPA_ENABLE_REMOTECONTROL
property_copy(recentAndroidAddress, setRecentAndroidAddress, QString, General, QString())
property_copy(listenForAndroidDevicesOnStartup, setListenForAndroidDevicesOnStartup, bool, General, false)
#endif
getValueFunc(QString, colorScheme, General, QString())
void SettingsData::setColorScheme(const QString &path) {
if (path != colorScheme())
{
setValue("General", "colorScheme", path);
emit colorSchemeChanged();
}
}
getValueFunc(QSize, histogramSize, General, QSize(15, 30))
getValueFunc(ViewSortType, viewSortType, General, (int)SortLastUse)
getValueFunc(AnnotationDialog::MatchType, matchType, General, (int)AnnotationDialog::MatchFromWordStart)
getValueFunc(bool, histogramUseLinearScale, General, false)
// clang-format on
void SettingsData::setHistogramUseLinearScale(const bool useLinearScale)
{
if (useLinearScale == histogramUseLinearScale())
return;
setValue("General", "histogramUseLinearScale", useLinearScale);
emit histogramScaleChanged();
}
void SettingsData::setHistogramSize(const QSize &size)
{
if (size == histogramSize())
return;
setValue("General", "histogramSize", size);
emit histogramSizeChanged(size);
}
void SettingsData::setViewSortType(const ViewSortType tp)
{
if (tp == viewSortType())
return;
setValue("General", "viewSortType", (int)tp);
emit viewSortTypeChanged(tp);
}
void SettingsData::setMatchType(const AnnotationDialog::MatchType mt)
{
if (mt == matchType())
return;
setValue("General", "matchType", (int)mt);
emit matchTypeChanged(mt);
}
bool SettingsData::trustTimeStamps()
{
if (tTimeStamps() == Always)
return true;
else if (tTimeStamps() == Never)
return false;
else {
if (!m_hasAskedAboutTimeStamps) {
const QString txt = i18n("When reading time information of images, their Exif info is used. "
"Exif info may, however, not be supported by your KPhotoAlbum installation, "
"or no valid information may be in the file. "
"As a backup, KPhotoAlbum may use the timestamp of the image - this may, "
"however, not be valid in case the image is scanned in. "
"So the question is, should KPhotoAlbum trust the time stamp on your images?");
const QString logMsg = QString::fromUtf8("Trust timestamps for this session?");
auto answer = uiDelegate().questionYesNo(logMsg, txt, i18n("Trust Time Stamps?"));
if (answer == DB::UserFeedback::Confirm)
m_trustTimeStamps = true;
else
m_trustTimeStamps = false;
m_hasAskedAboutTimeStamps = true;
}
return m_trustTimeStamps;
}
}
////////////////////////////////
//// File Version Detection ////
////////////////////////////////
// clang-format off
property_copy(detectModifiedFiles, setDetectModifiedFiles, bool, FileVersionDetection, true)
property_copy(modifiedFileComponent, setModifiedFileComponent, QString, FileVersionDetection, "^(.*)-edited.([^.]+)$")
property_copy(originalFileComponent, setOriginalFileComponent, QString, FileVersionDetection, "\\1.\\2")
property_copy(moveOriginalContents, setMoveOriginalContents, bool, FileVersionDetection, false)
property_copy(autoStackNewFiles, setAutoStackNewFiles, bool, FileVersionDetection, true)
property_copy(copyFileComponent, setCopyFileComponent, QString, FileVersionDetection, "(.[^.]+)$")
property_copy(copyFileReplacementComponent, setCopyFileReplacementComponent, QString, FileVersionDetection, "-edited\\1")
property_copy(loadOptimizationPreset, setLoadOptimizationPreset, int, FileVersionDetection, 0)
property_copy(overlapLoadMD5, setOverlapLoadMD5, bool, FileVersionDetection, false)
property_copy(preloadThreadCount, setPreloadThreadCount, int, FileVersionDetection, 1)
property_copy(thumbnailPreloadThreadCount, setThumbnailPreloadThreadCount, int, FileVersionDetection, 1)
property_copy(thumbnailBuilderThreadCount, setThumbnailBuilderThreadCount, int, FileVersionDetection, 0)
// clang-format on
////////////////////
//// Thumbnails ////
////////////////////
// clang-format off
property_copy(displayLabels, setDisplayLabels, bool, Thumbnails, true)
property_copy(displayCategories, setDisplayCategories, bool, Thumbnails, false)
property_copy(autoShowThumbnailView, setAutoShowThumbnailView, int, Thumbnails, 20)
property_copy(showNewestThumbnailFirst, setShowNewestFirst, bool, Thumbnails, false)
property_copy(thumbnailDisplayGrid, setThumbnailDisplayGrid, bool, Thumbnails, false)
property_copy(previewSize, setPreviewSize, int, Thumbnails, 256)
property_copy(thumbnailSpace, setThumbnailSpace, int, Thumbnails, 4)
// not available via GUI, but should be consistent (and maybe confgurable for powerusers):
property_copy(minimumThumbnailSize, setMinimumThumbnailSize, int, Thumbnails, 32)
property_copy(maximumThumbnailSize, setMaximumThumbnailSize, int, Thumbnails, 4096)
property_enum(thumbnailAspectRatio, setThumbnailAspectRatio, ThumbnailAspectRatio, Thumbnails, Aspect_3_2)
property_copy(incrementalThumbnails, setIncrementalThumbnails, bool, Thumbnails, true)
// database specific so that changing it doesn't invalidate the thumbnail cache for other databases:
getValueFunc_(int, thumbnailSize, groupForDatabase("Thumbnails"), "thumbSize", 256)
// clang-format on
void SettingsData::setThumbnailSize(int value)
{
// enforce limits:
value = qBound(minimumThumbnailSize(), value, maximumThumbnailSize());
if (value != thumbnailSize())
emit thumbnailSizeChanged(value);
setValue(groupForDatabase("Thumbnails"), "thumbSize", value);
}
int SettingsData::actualThumbnailSize() const
{
// this is database specific since it's a derived value of thumbnailSize
int retval = value(groupForDatabase("Thumbnails"), "actualThumbSize", 0);
// if no value has been set, use thumbnailSize
if (retval == 0)
retval = thumbnailSize();
return retval;
}
void SettingsData::setActualThumbnailSize(int value)
{
- QPixmapCache::clear();
-
// enforce limits:
value = qBound(minimumThumbnailSize(), value, thumbnailSize());
if (value != actualThumbnailSize()) {
setValue(groupForDatabase("Thumbnails"), "actualThumbSize", value);
emit actualThumbnailSizeChanged(value);
}
}
////////////////
//// Viewer ////
////////////////
// clang-format off
property_ref(viewerSize, setViewerSize, QSize, Viewer, QSize(1024, 768))
property_ref(slideShowSize, setSlideShowSize, QSize, Viewer, QSize(1024, 768))
property_copy(launchViewerFullScreen, setLaunchViewerFullScreen, bool, Viewer, false)
property_copy(launchSlideShowFullScreen, setLaunchSlideShowFullScreen, bool, Viewer, true)
property_copy(showInfoBox, setShowInfoBox, bool, Viewer, true)
property_copy(showLabel, setShowLabel, bool, Viewer, true)
property_copy(showDescription, setShowDescription, bool, Viewer, true)
property_copy(showDate, setShowDate, bool, Viewer, true)
property_copy(showImageSize, setShowImageSize, bool, Viewer, true)
property_copy(showRating, setShowRating, bool, Viewer, true)
property_copy(showTime, setShowTime, bool, Viewer, true)
property_copy(showFilename, setShowFilename, bool, Viewer, false)
property_copy(showEXIF, setShowEXIF, bool, Viewer, true)
property_copy(slideShowInterval, setSlideShowInterval, int, Viewer, 5)
property_copy(viewerCacheSize, setViewerCacheSize, int, Viewer, 195)
property_copy(infoBoxWidth, setInfoBoxWidth, int, Viewer, 400)
property_copy(infoBoxHeight, setInfoBoxHeight, int, Viewer, 300)
property_enum(infoBoxPosition, setInfoBoxPosition, Position, Viewer, Bottom)
property_enum(viewerStandardSize, setViewerStandardSize, StandardViewSize, Viewer, FullSize)
// clang-format on
bool SettingsData::smoothScale() const
{
return _smoothScale;
}
void SettingsData::setSmoothScale(bool b)
{
_smoothScale = b;
setValue("Viewer", "smoothScale", b);
}
////////////////////
//// Categories ////
////////////////////
// clang-format off
property_ref(untaggedCategory, setUntaggedCategory, QString, General, i18n("Events"))
property_ref(untaggedTag, setUntaggedTag, QString, General, i18n("untagged"))
property_copy(untaggedImagesTagVisible, setUntaggedImagesTagVisible, bool, General, false)
// clang-format on
//////////////
//// Exif ////
//////////////
// clang-format off
property_sset(exifForViewer, setExifForViewer, Exif, StringSet())
property_sset(exifForDialog, setExifForDialog, Exif, Exif::Info::instance()->standardKeys())
property_ref(iptcCharset, setIptcCharset, QString, Exif, QString())
// clang-format on
/////////////////////
//// Exif Import ////
/////////////////////
// clang-format off
property_copy(updateExifData, setUpdateExifData, bool, ExifImport, true)
property_copy(updateImageDate, setUpdateImageDate, bool, ExifImport, false)
property_copy(useModDateIfNoExif, setUseModDateIfNoExif, bool, ExifImport, true)
property_copy(updateOrientation, setUpdateOrientation, bool, ExifImport, false)
property_copy(updateDescription, setUpdateDescription, bool, ExifImport, false)
// clang-format on
///////////////////////
//// Miscellaneous ////
///////////////////////
// clang-format off
property_ref_(HTMLBaseDir, setHTMLBaseDir, QString, groupForDatabase("HTML Settings"), QString::fromLocal8Bit(qgetenv("HOME")) + STR("/public_html"))
property_ref_(HTMLBaseURL, setHTMLBaseURL, QString, groupForDatabase("HTML Settings"), STR("file://") + HTMLBaseDir())
property_ref_(HTMLDestURL, setHTMLDestURL, QString, groupForDatabase("HTML Settings"), STR("file://") + HTMLBaseDir())
property_ref_(HTMLCopyright, setHTMLCopyright, QString, groupForDatabase("HTML Settings"), STR(""))
property_ref_(HTMLDate, setHTMLDate, int, groupForDatabase("HTML Settings"), true)
property_ref_(HTMLTheme, setHTMLTheme, int, groupForDatabase("HTML Settings"), -1)
property_ref_(HTMLKimFile, setHTMLKimFile, int, groupForDatabase("HTML Settings"), true)
property_ref_(HTMLInlineMovies, setHTMLInlineMovies, int, groupForDatabase("HTML Settings"), true)
property_ref_(HTML5Video, setHTML5Video, int, groupForDatabase("HTML Settings"), true)
property_ref_(HTML5VideoGenerate, setHTML5VideoGenerate, int, groupForDatabase("HTML Settings"), true)
property_ref_(HTMLThumbSize, setHTMLThumbSize, int, groupForDatabase("HTML Settings"), 128)
property_ref_(HTMLNumOfCols, setHTMLNumOfCols, int, groupForDatabase("HTML Settings"), 5)
property_ref_(HTMLSizes, setHTMLSizes, QString, groupForDatabase("HTML Settings"), STR(""))
property_ref_(HTMLIncludeSelections, setHTMLIncludeSelections, QString, groupForDatabase("HTML Settings"), STR(""))
// clang-format on
QDate SettingsData::fromDate() const
{
QString date = value("Miscellaneous", "fromDate", STR(""));
return date.isEmpty() ? QDate(QDate::currentDate().year(), 1, 1) : QDate::fromString(date, Qt::ISODate);
}
void SettingsData::setFromDate(const QDate &date)
{
if (date.isValid())
setValue("Miscellaneous", "fromDate", date.toString(Qt::ISODate));
}
QDate SettingsData::toDate() const
{
QString date = value("Miscellaneous", "toDate", STR(""));
return date.isEmpty() ? QDate(QDate::currentDate().year() + 1, 1, 1) : QDate::fromString(date, Qt::ISODate);
}
void SettingsData::setToDate(const QDate &date)
{
if (date.isValid())
setValue("Miscellaneous", "toDate", date.toString(Qt::ISODate));
}
QString SettingsData::imageDirectory() const
{
return m_imageDirectory;
}
QString SettingsData::groupForDatabase(const char *setting) const
{
return STR("%1 - %2").arg(STR(setting)).arg(imageDirectory());
}
DB::ImageSearchInfo SettingsData::currentLock() const
{
return DB::ImageSearchInfo::loadLock();
}
void SettingsData::setCurrentLock(const DB::ImageSearchInfo &info, bool exclude)
{
info.saveLock();
setValue(groupForDatabase("Privacy Settings"), "exclude", exclude);
}
bool SettingsData::lockExcludes() const
{
return value(groupForDatabase("Privacy Settings"), "exclude", false);
}
getValueFunc_(bool, locked, groupForDatabase("Privacy Settings"), "locked", false)
void SettingsData::setLocked(bool lock, bool force)
{
if (lock == locked() && !force)
return;
setValue(groupForDatabase("Privacy Settings"), "locked", lock);
emit locked(lock, lockExcludes());
}
void SettingsData::setWindowGeometry(WindowType win, const QRect &geometry)
{
setValue("Window Geometry", win, geometry);
}
QRect SettingsData::windowGeometry(WindowType win) const
{
return value("Window Geometry", win, QRect(0, 0, 800, 600));
}
double Settings::SettingsData::getThumbnailAspectRatio() const
{
double ratio = 1.0;
switch (Settings::SettingsData::instance()->thumbnailAspectRatio()) {
case Settings::Aspect_16_9:
ratio = 9.0 / 16;
break;
case Settings::Aspect_4_3:
ratio = 3.0 / 4;
break;
case Settings::Aspect_3_2:
ratio = 2.0 / 3;
break;
case Settings::Aspect_9_16:
ratio = 16 / 9.0;
break;
case Settings::Aspect_3_4:
ratio = 4 / 3.0;
break;
case Settings::Aspect_2_3:
ratio = 3 / 2.0;
break;
case Settings::Aspect_1_1:
ratio = 1.0;
break;
}
return ratio;
}
QStringList Settings::SettingsData::EXIFCommentsToStrip()
{
return m_EXIFCommentsToStrip;
}
void Settings::SettingsData::setEXIFCommentsToStrip(QStringList EXIFCommentsToStrip)
{
m_EXIFCommentsToStrip = EXIFCommentsToStrip;
}
bool Settings::SettingsData::getOverlapLoadMD5() const
{
switch (Settings::SettingsData::instance()->loadOptimizationPreset()) {
case Settings::LoadOptimizationSlowNVME:
case Settings::LoadOptimizationFastNVME:
return true;
break;
case Settings::LoadOptimizationManual:
return Settings::SettingsData::instance()->overlapLoadMD5();
break;
case Settings::LoadOptimizationHardDisk:
case Settings::LoadOptimizationNetwork:
case Settings::LoadOptimizationSataSSD:
default:
return false;
break;
}
}
int Settings::SettingsData::getPreloadThreadCount() const
{
switch (Settings::SettingsData::instance()->loadOptimizationPreset()) {
case Settings::LoadOptimizationManual:
return Settings::SettingsData::instance()->preloadThreadCount();
break;
case Settings::LoadOptimizationSlowNVME:
case Settings::LoadOptimizationFastNVME:
case Settings::LoadOptimizationSataSSD:
return qMax(1, qMin(16, QThread::idealThreadCount()));
break;
case Settings::LoadOptimizationHardDisk:
case Settings::LoadOptimizationNetwork:
default:
return 1;
break;
}
}
int Settings::SettingsData::getThumbnailPreloadThreadCount() const
{
switch (Settings::SettingsData::instance()->loadOptimizationPreset()) {
case Settings::LoadOptimizationManual:
return Settings::SettingsData::instance()->thumbnailPreloadThreadCount();
break;
case Settings::LoadOptimizationSlowNVME:
case Settings::LoadOptimizationFastNVME:
case Settings::LoadOptimizationSataSSD:
return qMax(1, qMin(16, QThread::idealThreadCount() / 2));
break;
case Settings::LoadOptimizationHardDisk:
case Settings::LoadOptimizationNetwork:
default:
return 1;
break;
}
}
int Settings::SettingsData::getThumbnailBuilderThreadCount() const
{
switch (Settings::SettingsData::instance()->loadOptimizationPreset()) {
case Settings::LoadOptimizationManual:
return Settings::SettingsData::instance()->thumbnailBuilderThreadCount();
break;
case Settings::LoadOptimizationSlowNVME:
case Settings::LoadOptimizationFastNVME:
case Settings::LoadOptimizationSataSSD:
case Settings::LoadOptimizationHardDisk:
case Settings::LoadOptimizationNetwork:
default:
return qMax(1, qMin(16, QThread::idealThreadCount() - 1));
break;
}
}
DB::UIDelegate &SettingsData::uiDelegate() const
{
return m_UI;
}
// vi:expandtab:tabstop=4 shiftwidth=4: