Mercurial > minori
changeset 250:c130f47f6f48
*: many many changes
e.g. the search page is actually implemented now!
author | Paper <paper@paper.us.eu.org> |
---|---|
date | Sun, 04 Feb 2024 21:17:17 -0500 (11 months ago) |
parents | 6b2441c776dd |
children | 4d635d3e168a |
files | Makefile.am configure.ac include/core/anime.h include/core/config.h include/core/session.h include/gui/pages/search.h include/gui/pages/statistics.h include/gui/window.h include/services/anilist.h include/services/services.h m4/autotroll.m4 m4/m4_ax_have_qt.m4 src/core/config.cc src/core/filesystem.cc src/core/strings.cc src/gui/dialog/information.cc src/gui/pages/anime_list.cc src/gui/pages/search.cc src/gui/pages/statistics.cc src/gui/pages/torrents.cc src/gui/widgets/anime_info.cc src/gui/widgets/sidebar.cc src/gui/window.cc src/main.cc src/services/anilist.cc src/services/services.cc src/sys/glib/dark_theme.cc src/sys/osx/filesystem.cc |
diffstat | 28 files changed, 1502 insertions(+), 584 deletions(-) [+] |
line wrap: on
line diff
--- a/Makefile.am Wed Jan 24 20:18:59 2024 -0500 +++ b/Makefile.am Sun Feb 04 21:17:17 2024 -0500 @@ -5,23 +5,23 @@ rc/locale/es.ts .ts.qm: - @MKDIR_P@ `dirname $@`; \ - @QT_LRELEASE@ $< -qm $@ + $(MKDIR_P) `dirname $@`; \ + $(LRELEASE) $< -qm $@ minori_locale_qm = $(minori_locale_ts:.ts=.qm) # this has to be in the root build folder -translations.qrc: $(minori_locale_qm) +rc/locale/translations.qrc: $(minori_locale_qm) printf "<!DOCTYPE rcc><RCC version=\"1.0\">\n\t<qresource prefix=\"locale/\">\n" > $@; \ for q in $(minori_locale_qm); do \ - printf "\t\t<file alias=\"%s\">%s</file>\n" "`basename $$q`" "$$q" >> $@; \ - done; \ + printf "\t\t<file>%s</file>\n" "`basename $$q`" >> $@; \ + done; printf "\t</qresource>\n</RCC>\n" >> $@; minori_qtrc = \ $(top_srcdir)/rc/icons/icons.qrc \ $(top_srcdir)/rc/player_data.qrc \ - translations.qrc + rc/locale/translations.qrc if BUILD_WIN @@ -30,7 +30,7 @@ endif rc/final_qrc.cc: $(minori_qtrc) - @QT_RCC@ -o $@ $(minori_qtrc) + $(RCC) -o $@ $(minori_qtrc) minori_qtheaders = \ include/core/http.h \ @@ -84,8 +84,8 @@ if BUILD_GLIB files_glib = src/sys/glib/dark_theme.cc -cflags_glib = @GIO_CFLAGS@ -libs_glib = @GIO_LIBS@ +cflags_glib = $(GLIB_CFLAGS) +libs_glib = $(GLIB_LIBS) endif if BUILD_WIN @@ -101,8 +101,9 @@ .rc.$(OBJEXT): $(WINDRES) $(WRCFLAGS) -i $< -o $@ files_windres=rc/win32/version.rc rc/win32/resource.rc -endif -endif +endif # BUILD_WINDRES + +endif # BUILD_WIN if BUILD_OSX files_osx = src/sys/osx/dark_theme.cc src/sys/osx/filesystem.cc src/sys/osx/permissions.cc @@ -156,9 +157,11 @@ $(files_osx) \ $(files_glib) \ $(files_win) \ + $(files_windres) + +nodist_minori_SOURCES = \ $(minori_moc_sources) \ - rc/final_qrc.cc \ - $(files_windres) + rc/final_qrc.cc minori_includes = \ -I$(top_srcdir)/include \ @@ -167,16 +170,16 @@ -I$(top_srcdir)/dep/anitomy \ -I$(top_srcdir)/dep -minori_CPPFLAGS = @LIBCURL_CPPFLAGS@ $(minori_includes) -minori_CXXFLAGS = @QT_CXXFLAGS@ $(cflags_osx) $(cflags_glib) $(cflags_win) -std=c++17 -minori_LDFLAGS = $(ldflags_osx) $(ldflags_win) +minori_CPPFLAGS = $(QT_CPPFLAGS) $(LIBCURL_CPPFLAGS) $(minori_includes) +minori_CXXFLAGS = $(cflags_osx) $(cflags_glib) $(cflags_win) +minori_LDFLAGS = $(QT_LDFLAGS) $(ldflags_osx) $(ldflags_win) minori_DEPENDENCIES = dep/pugixml/libpugixml.la dep/animia/libanimia.la dep/anitomy/libanitomy.la -minori_LDADD = $(libs_glib) $(libs_osx) $(libs_win) @LIBCURL@ @QT_LIBS@ dep/pugixml/libpugixml.la dep/animia/libanimia.la dep/anitomy/libanitomy.la +minori_LDADD = $(libs_glib) $(LIBCURL) $(QT_LIBS) $(libs_osx) $(libs_win) $(minori_DEPENDENCIES) .h_moc.cc: - @MKDIR_P@ -- `dirname $@` - @QT_MOC@ -o $@ $(minori_includes) $< + $(MKDIR_P) -- `dirname $@` + $(MOC) -o $@ $(minori_includes) $< SUFFIXES = .h _moc.cc .ts .qm SUBDIRS = $(subdirs)
--- a/configure.ac Wed Jan 24 20:18:59 2024 -0500 +++ b/configure.ac Sun Feb 04 21:17:17 2024 -0500 @@ -9,61 +9,59 @@ AM_INIT_AUTOMAKE([-Wall -Wportability foreign subdir-objects]) -# Do we have a C++17 compiler +dnl Do we have a C++17 compiler : ${CXXFLAGS=""} AC_PROG_CXX AX_CXX_COMPILE_STDCXX([17], [noext], [mandatory]) -# Init libtool +dnl Init libtool AM_PROG_AR LT_INIT -# Qt? -AX_HAVE_QT +dnl Qt? +AT_WITH_QT([widgets gui core], [], [], [have_qt=no], [have_qt=yes]) + +AS_IF([test "x$have_qt" = "xno"], [AC_MSG_ERROR([*** Qt not found.])]) -if test "x$have_qt" = "xno"; then - AC_MSG_ERROR([*** Qt not found.]) -fi +dnl need this for moc +AC_PROG_MKDIR_P +AC_SUBST([MKDIR_P]) -# need this for moc -AC_PROG_MKDIR_P - -# libcurl? +dnl libcurl? LIBCURL_CHECK_CONFIG([yes], [7.7.2], [have_libcurl=yes], [have_libcurl=no]) -if test "x$have_libcurl" = "xno"; then - AC_MSG_ERROR([*** libcurl not found.]) -fi +AS_IF([test "x$have_libcurl" = "xno"], [AC_MSG_ERROR([*** libcurl not found.])]) build_windows=no build_osx=no +build_linux=no build_glib=no -case "${host_os}" in - cygwin*|mingw*) - # Windows - build_windows=yes - AC_CHECK_TOOL([WINDRES], [windres]) - AC_SUBST(WINDRES) - AC_PROG_SED # We need sed for version numbers in windres - AC_SUBST(SED) - AC_DEFINE(WIN32) - ;; - darwin*) - # Mac OS X - build_osx=yes - AC_DEFINE(MACOSX) - ;; - *) - if test "x$host_os" = "xlinux"; then - AC_DEFINE(LINUX) - fi - # Everything else - AC_SUBST([GIO_CFLAGS]) - AC_SUBST([GIO_LIBS]) - PKG_CHECK_MODULES([GIO], [gio-2.0], [build_glib=yes], []) - ;; -esac +AS_CASE(["$host_os"], + [cygwin*|mingw*], [build_windows=yes], + [darwin*], [build_osx=yes], + [linux*], [build_linux=yes], + []) + +if test "x$build_windows" = "xyes"; then + AC_DEFINE([WIN32]) + + dnl Check for windres + AC_CHECK_TOOL([WINDRES], [windres]) + AC_SUBST([WINDRES]) +elif test "x$build_osx" = "xyes"; then + AC_DEFINE([MACOSX]) +else + AS_IF([test "x$build_linux" = "xyes"], [AC_DEFINE([linux])]) + + PKG_CHECK_MODULES([GLIB], [gio-2.0 glib-2.0], [build_glib=yes], [build_glib=no]) + if test "x$build_glib" = "xyes"; then + AC_DEFINE([GLIB]) + + AC_SUBST([GLIB_CFLAGS]) + AC_SUBST([GLIB_LIBS]) + fi +fi AM_CONDITIONAL([BUILD_WIN], [test "x$build_windows" = "xyes"]) AM_CONDITIONAL([BUILD_OSX], [test "x$build_osx" = "xyes"])
--- a/include/core/anime.h Wed Jan 24 20:18:59 2024 -0500 +++ b/include/core/anime.h Sun Feb 04 21:17:17 2024 -0500 @@ -147,7 +147,7 @@ std::vector<std::string> GetProducers() const; SeriesFormat GetFormat() const; SeriesSeason GetSeason() const; - int GetAudienceScore() const; /* should be double once MAL and Kitsu are implemented */ + int GetAudienceScore() const; std::string GetSynopsis() const; int GetDuration() const; std::string GetPosterUrl() const;
--- a/include/core/config.h Wed Jan 24 20:18:59 2024 -0500 +++ b/include/core/config.h Sun Feb 04 21:17:17 2024 -0500 @@ -20,7 +20,7 @@ class Config { public: int Load(); - int Save() const; + int Save(); Anime::Services service; Theme::Theme theme;
--- a/include/core/session.h Wed Jan 24 20:18:59 2024 -0500 +++ b/include/core/session.h Sun Feb 04 21:17:17 2024 -0500 @@ -7,6 +7,8 @@ #include "semver/semver.hpp" +class MainWindow; + struct Session { public: Session() { timer.start(); }
--- a/include/gui/pages/search.h Wed Jan 24 20:18:59 2024 -0500 +++ b/include/gui/pages/search.h Sun Feb 04 21:17:17 2024 -0500 @@ -1,12 +1,66 @@ #ifndef __gui__pages__search_h #define __gui__pages__search_h -#include <QWidget> + +#include "core/anime.h" + +#include <QFrame> +#include <QAbstractListModel> +#include <QSortFilterProxyModel> +#include <QItemSelection> + +class QTreeView; + +class SearchPageListSortFilter final : public QSortFilterProxyModel { + Q_OBJECT + + public: + SearchPageListSortFilter(QObject* parent = nullptr); + + protected: + bool lessThan(const QModelIndex& l, const QModelIndex& r) const override; +}; + +class SearchPageListModel final : public QAbstractListModel { + Q_OBJECT -class SearchPage final : public QWidget { + public: + enum columns { + SR_TITLE, + SR_TYPE, + SR_EPISODES, + SR_SCORE, + SR_SEASON, + + NB_COLUMNS + }; + + SearchPageListModel(QObject* parent); + ~SearchPageListModel() override = default; + int rowCount(const QModelIndex& parent = QModelIndex()) const override; + int columnCount(const QModelIndex& parent = QModelIndex()) const override; + QVariant data(const QModelIndex& index, int role) const override; + QVariant headerData(const int section, const Qt::Orientation orientation, const int role) const override; + Qt::ItemFlags flags(const QModelIndex& index) const override; + + void ParseSearch(const std::vector<int>& ids); + Anime::Anime* GetAnimeFromIndex(const QModelIndex& index) const; + + private: + std::vector<int> ids; +}; + +class SearchPage final : public QFrame { Q_OBJECT public: SearchPage(QWidget* parent = nullptr); -}; + void Search(const std::string& search); + void DisplayListMenu(); + void ItemDoubleClicked(); + private: + SearchPageListModel* model = nullptr; + SearchPageListSortFilter* sort_model = nullptr; + QTreeView* treeview = nullptr; +}; #endif // __gui__pages__search_h
--- a/include/gui/pages/statistics.h Wed Jan 24 20:18:59 2024 -0500 +++ b/include/gui/pages/statistics.h Sun Feb 04 21:17:17 2024 -0500 @@ -22,9 +22,6 @@ void showEvent(QShowEvent*) override; private: - std::string MinutesToDateString(int minutes); - std::string SecondsToDateString(int seconds); - std::shared_ptr<TextWidgets::LabelledSection> _anime_list; std::shared_ptr<Graph<int>> _score_distribution_graph; std::shared_ptr<TextWidgets::LabelledSection> _application;
--- a/include/gui/window.h Wed Jan 24 20:18:59 2024 -0500 +++ b/include/gui/window.h Sun Feb 04 21:17:17 2024 -0500 @@ -13,6 +13,8 @@ #include <QThread> #include "gui/widgets/sidebar.h" +class QMenu; + Q_DECLARE_METATYPE(std::vector<std::string>); class PlayingThread : public QThread { @@ -32,11 +34,24 @@ Q_OBJECT public: + enum class Pages { + NOW_PLAYING, + + ANIME_LIST, + HISTORY, + STATISTICS, + + SEARCH, + SEASONS, + TORRENTS + }; + MainWindow(QWidget* parent = nullptr); void SetActivePage(QWidget* page); void CreateBars(); void AddMainWidgets(); void RetranslateUI(); + void UpdateFolderMenu(); void AsyncSynchronize(QAction* action, QStackedWidget* stack); void changeEvent(QEvent* event) override; void showEvent(QShowEvent* event) override; @@ -47,7 +62,9 @@ std::unique_ptr<QStackedWidget> stack = nullptr; std::unique_ptr<SideBar> sidebar = nullptr; - std::unique_ptr<PlayingThread> thread = nullptr; + std::unique_ptr<PlayingThread> thread = nullptr; + + QMenu* folder_menu = nullptr; }; #endif // __window_h
--- a/include/services/anilist.h Wed Jan 24 20:18:59 2024 -0500 +++ b/include/services/anilist.h Sun Feb 04 21:17:17 2024 -0500 @@ -1,9 +1,8 @@ #ifndef __services__anilist_h #define __services__anilist_h -#include "core/anime.h" -#include "core/json.h" -#include <curl/curl.h> +#include <vector> +#include <string> namespace Services { namespace AniList { @@ -13,6 +12,9 @@ /* Read queries */ int GetAnimeList(); +/* Search query */ +std::vector<int> Search(const std::string& search); + /* Write queries (mutations) */ int UpdateAnimeEntry(int id);
--- a/include/services/services.h Wed Jan 24 20:18:59 2024 -0500 +++ b/include/services/services.h Sun Feb 04 21:17:17 2024 -0500 @@ -1,9 +1,13 @@ #ifndef __services__services_h #define __services__services_h +#include <vector> +#include <string> + namespace Services { void Synchronize(); +std::vector<int> Search(const std::string& search); void UpdateAnimeEntry(int id); bool Authorize();
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/m4/autotroll.m4 Sun Feb 04 21:17:17 2024 -0500 @@ -0,0 +1,720 @@ +# Build Qt apps with the autotools (Autoconf/Automake). +# M4 macros. +# +# This file is part of AutoTroll. +# +# Copyright (C) 2006-2018 Benoit Sigoure <benoit.sigoure@lrde.epita.fr> +# Copyright (C) 2012-2023 Werner Lemberg <wl@gnu.org> +# +# AutoTroll 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; if not, write to the Free Software +# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA +# 02110-1301, USA. +# +# In addition, as a special exception, the copyright holders of +# AutoTroll give you unlimited permission to copy, distribute and +# modify the configure scripts that are the output of Autoconf when +# processing the macros of AutoTroll. You need not follow the terms +# of the GNU General Public License when using or distributing such +# scripts, even though portions of the text of AutoTroll appear in +# them. The GNU General Public License (GPL) does govern all other +# use of the material that constitutes AutoTroll. +# +# This special exception to the GPL applies to versions of AutoTroll +# released by the copyright holders of AutoTroll. Note that people +# who make modified versions of AutoTroll are not obligated to grant +# this special exception for their modified versions; it is their +# choice whether to do so. The GNU General Public License gives +# permission to release a modified version without this exception; +# this exception also makes it possible to release a modified version +# which carries forward this exception. + + # ------------- # + # DOCUMENTATION # + # ------------- # + +# Disclaimer: Tested with Qt 4.2, 4.8, Qt 5.x, and Qt 6 only. +# Feedback welcome. Simply invoke AT_WITH_QT in your configure.ac. +# AT_WITH_QT can take arguments which are documented in depth below. +# The default arguments are equivalent to the default .pro file +# generated by qmake. +# +# Invoking AT_WITH_QT will do the following: +# +# - Add option `--with-qt[=ARG]' to your configure script. Possible +# values for ARG are `yes' (which is the default) and `no' to +# enable and disable Qt support, respectively, or a path to the +# directory which contains the Qt binaries in case you have a +# non-stardard location. +# +# - Add option `--without-qt', which is equivalent to `--with-qt=no'. +# +# - On MacOS, add `-spec macx-g++' or `-spec macx-lang' (if `$CXX +# --version' output contains `clang'). This can be overridden with +# the QMAKESPEC environment variable, for example +# +# QMAKESPEC='macx-clang' ./configure ... +# +# (The QMAKESPEC variable is honoured for non-MacOS builds also.) +# +# - If Qt support is enabled, define C preprocessor macro HAVE_QT. +# +# - Find the programs `qmake', `moc', `uic', and `rcc' and save them +# in the make variables $(QMAKE), $(MOC), $(UIC), and $(RCC). +# +# - Save the path to Qt binaries in $(QT_PATH). +# +# - Find the flags necessary to compile and link Qt, that is: +# +# * $(QT_DEFINES): -D's defined by qmake. +# * $(QT_CFLAGS): CFLAGS as defined by qmake (C?!) +# * $(QT_CXXFLAGS): CXXFLAGS as defined by qmake. +# * $(QT_INCPATH): -I's defined by qmake. +# * $(QT_CPPFLAGS): Same as $(QT_DEFINES) + $(QT_INCPATH). +# * $(QT_LFLAGS): LFLAGS defined by qmake. +# * $(QT_LDFLAGS): Same thing as $(QT_LFLAGS). +# * $(QT_LIBS): LIBS defined by qmake. +# +# - Provide @QT_STATIC_PLUGINS@, which holds some additional C++ +# declarations necessary for linking with static Qt plugins (for +# dynamic Qt builds it contains a dummy typedef declaration +# instead). Use this substitution in a `foo.cpp.in' C++ template +# file or something similar, which must be registered in +# configure.ac's call to AC_CONFIG_FILES so that a proper `foo.cpp' +# file gets created. Then compile and link `foo.cpp' with your +# program in the usual automake way. +# +# NOTE: It is not possible to automatically detect whether a Qt +# release earlier than version 5 is built as a static library! For +# this reason, @QT_STATIC_PLUGINS@ always contains the dummy +# typedef declaration if not using Qt5. +# +# You *MUST* invoke $(MOC) and/or $(UIC) by yourself where necessary. +# AutoTroll provides you with Makerules to ease this; here is a sample +# Makefile.am to use with AutoTroll which builds the code given in +# chapter 7 of the Qt Tutorial +# (http://doc.trolltech.com/4.2/tutorial-t7.html). +# +# ------------------------------------------------------------------------- +# include $(top_srcdir)/build-aux/autotroll.mk +# +# ACLOCAL_AMFLAGS = -I build-aux +# +# bin_PROGRAMS = lcdrange +# lcdrange_SOURCES = $(BUILT_SOURCES) lcdrange.cpp lcdrange.h main.cpp +# lcdrange_CXXFLAGS = $(QT_CXXFLAGS) $(AM_CXXFLAGS) +# lcdrange_CPPFLAGS = $(QT_CPPFLAGS) $(AM_CPPFLAGS) +# lcdrange_LDFLAGS = $(QT_LDFLAGS) $(LDFLAGS) +# lcdrange_LDADD = $(QT_LIBS) $(LDADD) +# +# BUILT_SOURCES = lcdrange.moc.cpp +# ------------------------------------------------------------------------- +# +# Note that your MOC, UIC, and RCC files *MUST* be listed explicitly +# in BUILT_SOURCES. If you name them properly (e.g. `.moc.cc', +# `.qrc.cc', `.ui.cc' -- of course you can use `.cpp' or `.cxx' or +# `.C' rather than `.cc') AutoTroll will build them automagically for +# you, using implicit rules defined in `autotroll.mk'. + +m4_define([_AUTOTROLL_SERIAL], + [m4_translit([ +# serial 17 +], [# +], [])]) + + +m4_ifdef([AX_INSTEAD_IF], + [], + [AC_DEFUN([AX_INSTEAD_IF], + [m4_ifval([$1], + [AC_MSG_WARN([$2]) + [$1]], + [AC_MSG_ERROR([$2])])])]) + + +# AX_PATH_TOOLS(VARIABLE, PROGS-TO-CHECK-FOR, [VALUE-IF-NOT-FOUND], [PATH]) +# ------------------------------------------------------------------------- +AC_DEFUN([AX_PATH_TOOLS], + [for ax_tool in $2; do + AC_PATH_TOOL([$1], [$ax_tool], [], [$4]) + test -n "$$1" && break + done + m4_ifval([$3], [test -n "$$1" || $1="$3"]) + ]) + + +m4_pattern_forbid([^AT_]) +m4_pattern_forbid([^_AT_]) + + +# AT_WITH_QT([QT_modules], [QT_config], [QT_misc], [RUN-IF-FAILED], [RUN-IF-OK]) +# ------------------------------------------------------------------------------ +# Enable Qt support and add an option --with-qt to the configure +# script. +# +# The QT_modules argument is optional and defines extra modules to +# enable or disable (it's equivalent to the QT variable in .pro +# files). Modules can be specified as follows: +# +# AT_WITH_QT => No argument -> No QT value. +# Qmake sets it to "core gui" by +# default. +# AT_WITH_QT([xml]) => QT += xml +# AT_WITH_QT([+xml]) => QT += xml +# AT_WITH_QT([-gui]) => QT -= gui +# AT_WITH_QT([xml -gui +sql svg]) => QT += xml sql svg +# QT -= gui +# +# The QT_config argument is also optional and follows the same +# convention as QT_modules. Instead of changing the QT variable, it +# changes the CONFIG variable, which is used to tweak configuration +# and compiler options. +# +# The last argument, QT_misc (also optional) will be copied as-is the +# .pro file used to guess how to compile Qt apps. You may use it to +# further tweak the build process of Qt apps if tweaking the QT or +# CONFIG variables isn't enough for you (for example, to control which +# static plugins get used). +# +# RUN-IF-FAILED is arbitrary code to execute if Qt cannot be found or +# if any problem happens. If this argument is omitted, then +# AC_MSG_ERROR will be called. RUN-IF-OK is arbitrary code to execute +# if Qt was successfully found. + +AC_DEFUN([AT_WITH_QT], + [AC_REQUIRE([AC_CANONICAL_HOST]) + AC_REQUIRE([AC_CANONICAL_BUILD]) + AC_REQUIRE([AC_PROG_CXX]) + + echo "$as_me: this is autotroll.m4[]_AUTOTROLL_SERIAL" \ + >& AS_MESSAGE_LOG_FD + + # This is a hack to get decent flow control with `break'. + for _qt_ignored in once; do + + AC_ARG_WITH([qt], + AS_HELP_STRING([--with-qt@<:@=ARG@:>@], + [Qt support. ARG can be `yes' (the default), `no', + or a path to Qt binaries; if `yes' or empty, + use PATH and some default directories to find Qt binaries])) + + if test x"$with_qt" = x"no"; then + break + else + AC_DEFINE([HAVE_QT],[1], + [Define if the Qt framework is available.]) + fi + + if test x"$with_qt" = x"yes"; then + QT_PATH= + else + QT_PATH=$with_qt + fi + + # Find Qt. + AC_ARG_VAR([QT_PATH], + [path to Qt binaries]) + QT_TOOL_PATH=$QT_PATH:$PATH + + # Find qmake. + AC_ARG_VAR([QMAKE], + [Qt Makefile generator command]) + AX_PATH_TOOLS([QMAKE], + [qmake qmake-qt5 qmake-qt4 qmake-qt3], + [missing], + [$QT_TOOL_PATH]) + if test x"$QMAKE" = xmissing; then + if test x"$with_qt" = "x"; then + with_qt="no" + else + AX_INSTEAD_IF([$4], + [Cannot find qmake. Try --with-qt=PATH.]) + fi + break + fi + + # Find moc (Meta Object Compiler). + AC_ARG_VAR([MOC], + [Qt Meta Object Compiler command]) + AX_PATH_TOOLS([MOC], + [moc moc-qt5 moc-qt4 moc-qt3], + [missing], + [$QT_TOOL_PATH]) + if test x"$MOC" = xmissing; then + AX_INSTEAD_IF([$4], + [Cannot find moc (Meta Object Compiler). Try --with-qt=PATH.]) + break + fi + + # Find uic (User Interface Compiler). + AC_ARG_VAR([UIC], + [Qt User Interface Compiler command]) + AX_PATH_TOOLS([UIC], + [uic uic-qt5 uic-qt4 uic-qt3 uic3], + [missing], + [$QT_TOOL_PATH]) + if test x"$UIC" = xmissing; then + AX_INSTEAD_IF([$4], + [Cannot find uic (User Interface Compiler). Try --with-qt=PATH.]) + break + fi + + # Find rcc (Qt Resource Compiler). + AC_ARG_VAR([RCC], + [Qt Resource Compiler command]) + AX_PATH_TOOLS([RCC], + [rcc rcc-qt5], + [missing], + [$QT_TOOL_PATH]) + if test x"$RCC" = xmissing; then + AC_MSG_WARN( + [Cannot find rcc (Qt Resource Compiler). Try --with-qt=PATH.]) + fi + + AC_ARG_VAR([LUPDATE], + [Qt Linguist updater command]) + AX_PATH_TOOLS([LUPDATE], + [lupdate lupdate-qt5 lrelease-qt6], + [missing], + [$QT_TOOL_PATH]) + if test "x$LUPDATE" = "xmissing"; then + AC_MSG_WARN( + [Cannot find lupdate (Qt Linguist updater). Try --with-qt=PATH.]) + fi + + AC_ARG_VAR([LRELEASE], + [Qt Linguist compiler command]) + AX_PATH_TOOLS([LRELEASE], + [lrelease lrelease-qt5 lrelease-qt6], + [missing], + [$QT_TOOL_PATH]) + if test "x$LRELEASE" = "xmissing"; then + AC_MSG_WARN( + [Cannot find lrelease (Qt Linguist compiler). Try --with-qt=PATH.]) + fi + + AC_MSG_CHECKING([whether host operating system is Darwin]) + at_darwin=no + at_qmake_args= + case $host_os in + dnl ( + darwin*) + at_darwin=yes + ;; + esac + AC_MSG_RESULT([$at_darwin]) + + AC_MSG_CHECKING([whether QMAKESPEC environment variable is set]) + if test x"$QMAKESPEC" = x; then + if test x"$at_darwin" = xyes; then + if $CXX --version | grep -q -i clang; then + at_qmake_args='-spec macx-clang' + else + at_qmake_args='-spec macx-g++' + fi + AC_MSG_RESULT([no, using $at_qmake_args]) + else + AC_MSG_RESULT([no]) + fi + else + AC_MSG_RESULT([yes, using $QMAKESPEC]) + fi + + # If we don't know the path to Qt, guess it from the path to + # qmake. + if test x"$QT_PATH" = x; then + QT_PATH=`dirname "$QMAKE"` + fi + if test x"$QT_PATH" = x; then + AX_INSTEAD_IF([$4], + [Cannot find your Qt installation. Try --with-qt=PATH.]) + break + fi + AC_SUBST([QT_PATH]) + + # Get ready to build a test-app with Qt. + if mkdir conftest.dir \ + && cd conftest.dir; then + : + else + AX_INSTEAD_IF([$4], + [Cannot mkdir conftest.dir or cd to that directory.]) + break + fi + + cat >conftest.h <<_ASEOF + +#include <QObject> + +class Foo: public QObject +{ + Q_OBJECT; +public: + Foo(); + ~Foo() {} +public Q_SLOTS: + void setValue(int value); +Q_SIGNALS: + void valueChanged(int newValue); +private: + int value_; +}; + +_ASEOF + + cat >conftest.cpp <<_ASEOF + +#include "conftest.h" + +Foo::Foo() + : value_ (42) +{ + connect(this, SIGNAL(valueChanged(int)), + this, SLOT(setValue(int))); +} + +void Foo::setValue(int value) +{ + value_ = value; +} + +int main() +{ + Foo f; +} + +_ASEOF + + if $QMAKE -project; then + : + else + AX_INSTEAD_IF([$4], + [Calling $QMAKE -project failed.]) + break + fi + + # Find the .pro file generated by qmake. + pro_file=conftest.dir.pro + test -f $pro_file || pro_file=`echo *.pro` + if test -f "$pro_file"; then + : + else + AX_INSTEAD_IF([$4], + [Can't find the .pro file generated by Qmake.]) + break + fi + + dnl This is for Qt5; for Qt4 it does nothing special. + _AT_TWEAK_PRO_FILE([QT], [+widgets]) + + dnl Undocumented qmake: always use absolute paths. + dnl Defaults to 4. + _AT_TWEAK_PRO_FILE([QMAKE_PROJECT_DEPTH], [0]) + + dnl Tweak the value of QT in the .pro file if we have a first + dnl argument. + m4_ifval([$1], + [_AT_TWEAK_PRO_FILE([QT], [$1])]) + + dnl Tweak the value of CONFIG in the .pro file if we have a + dnl second argument. + m4_ifval([$2], + [_AT_TWEAK_PRO_FILE([CONFIG], [$2])]) + + m4_ifval([$3], + [ # Add the extra-settings the user wants to set in the .pro + # file. + echo "$3" >>"$pro_file" + ]) + + echo "$as_me:$LINENO: Invoking $QMAKE on $pro_file" \ + >& AS_MESSAGE_LOG_FD + sed 's/^/| /' "$pro_file" >& AS_MESSAGE_LOG_FD + + if $QMAKE $at_qmake_args; then + : + else + AX_INSTEAD_IF([$4], + [Calling $QMAKE $at_qmake_args failed.]) + break + fi + + # Try to compile a simple Qt app. + AC_CACHE_CHECK([whether we can build a simple Qt application], + [at_cv_qt_build], + [at_cv_qt_build=ko + : ${MAKE=make} + + if $MAKE >& AS_MESSAGE_LOG_FD 2>&1; then + at_cv_qt_build='ok, looks like Qt 4, Qt 5, or Qt 6' + else + echo "$as_me:$LINENO: Build failed, trying to #include <qobject.h> instead" \ + >& AS_MESSAGE_LOG_FD + sed 's/<QObject>/<qobject.h>/' conftest.h > tmp.h \ + && mv tmp.h conftest.h + if $MAKE >& AS_MESSAGE_LOG_FD 2>&1; then + at_cv_qt_build='ok, looks like Qt 3' + else + # Sometimes (such as on Debian) build will fail because Qt + # hasn't been installed in debug mode and qmake tries (by + # default) to build apps in debug mode => Try again in + # release mode. + echo "$as_me:$LINENO: Build failed, trying to enforce release mode" \ + >& AS_MESSAGE_LOG_FD + + _AT_TWEAK_PRO_FILE([CONFIG], [+release]) + + sed 's/<qobject.h>/<QObject>/' conftest.h > tmp.h \ + && mv tmp.h conftest.h + if $MAKE >& AS_MESSAGE_LOG_FD 2>&1; then + at_cv_qt_build='ok, looks like Qt 4 or Qt 5, release mode forced' + else + echo "$as_me:$LINENO: Build failed, trying to #include <qobject.h> instead" \ + >& AS_MESSAGE_LOG_FD + sed 's/<QObject>/<qobject.h>/' conftest.h > tmp.h \ + && mv tmp.h conftest.h + if $MAKE >& AS_MESSAGE_LOG_FD 2>&1; then + at_cv_qt_build='ok, looks like Qt 3, release mode forced' + else + at_cv_qt_build=ko + echo "$as_me:$LINENO: failed program was:" \ + >& AS_MESSAGE_LOG_FD + sed 's/^/| /' conftest.h >& AS_MESSAGE_LOG_FD + echo "$as_me:$LINENO: failed program was:" \ + >& AS_MESSAGE_LOG_FD + sed 's/^/| /' conftest.cpp >& AS_MESSAGE_LOG_FD + fi # if make with Qt3-style #include and release mode forced. + fi # if make with Qt4/5-style #include and release mode forced. + fi # if make with Qt3-style #include. + fi # if make with Qt4/5-style #include. + ])dnl end: AC_CACHE_CHECK(at_cv_qt_build) + + if test x"$at_cv_qt_build" = xko; then + AX_INSTEAD_IF([$4], + [Cannot build a test Qt program]) + cd .. + break + fi + + QT_VERSION_MAJOR=`echo "$at_cv_qt_build" | sed 's/[[^0-9]]*//g'` + AC_SUBST([QT_VERSION_MAJOR]) + + # This sed filter is applied after an expression of the form + # /^FOO.*=/!d; it starts by removing the beginning of the line + # (using the empty regular expression //, which repeats the last + # regular expression match), removing references to SUBLIBS, + # removing unnecessary whitespace at the beginning, then prefixing + # our exported variables with QT_. Note that `LDFLAGS' is + # intentionally omitted. + qt_sed_filter='s///; + s/$(SUBLIBS)//g; + s/^ *//; + s/\$(DEFINES)/$(QT_DEFINES)/g; + s/\$(CFLAGS)/$(QT_CFLAGS)/g; + s/\$(CXXFLAGS)/$(QT_CXXFLAGS)/g; + s/\$(INCPATH)/$(QT_INCPATH)/g; + s/\$(CPPFLAGS)/$(QT_CPPFLAGS)/g; + s/\$(LFLAGS)/$(QT_LFLAGS)/g; + s/\$(LIBS)/$(QT_LIBS)/g' + + # Find the Makefile (qmake happens to generate a fake Makefile + # which invokes a Makefile.Debug or Makefile.Release). If we + # have both, we'll pick the Makefile.Release. The reason is that + # this release uses -Os and debug -g. We can override -Os by + # passing another -O but we usually don't override -g. + if test -f Makefile.Release; then + at_mfile='Makefile.Release' + else + at_mfile='Makefile' + fi + if test -f $at_mfile; then + : + else + AX_INSTEAD_IF([$4], + [Cannot find the Makefile generated by qmake.]) + cd .. + break + fi + + # Find the DEFINES of Qt (should have been named CPPFLAGS). + AC_CACHE_CHECK([for the DEFINES to use with Qt], + [at_cv_env_QT_DEFINES], + [at_cv_env_QT_DEFINES=`sed "/^DEFINES@<:@^A-Z=@:>@*=/!d; + $qt_sed_filter" $at_mfile`]) + AC_SUBST([QT_DEFINES], + [$at_cv_env_QT_DEFINES]) + + # Find the CFLAGS of Qt. (We can use Qt in C?!) + AC_CACHE_CHECK([for the CFLAGS to use with Qt], + [at_cv_env_QT_CFLAGS], + [at_cv_env_QT_CFLAGS=`sed "/^CFLAGS@<:@^A-Z=@:>@*=/!d; + $qt_sed_filter" $at_mfile`]) + AC_SUBST([QT_CFLAGS], + [$at_cv_env_QT_CFLAGS]) + + # Find the CXXFLAGS of Qt. + AC_CACHE_CHECK([for the CXXFLAGS to use with Qt], + [at_cv_env_QT_CXXFLAGS], + [at_cv_env_QT_CXXFLAGS=`sed "/^CXXFLAGS@<:@^A-Z=@:>@*=/!d; + $qt_sed_filter" $at_mfile`]) + AC_SUBST([QT_CXXFLAGS], + [$at_cv_env_QT_CXXFLAGS]) + + # Find the INCPATH of Qt. + AC_CACHE_CHECK([for the INCPATH to use with Qt], + [at_cv_env_QT_INCPATH], + [at_cv_env_QT_INCPATH=`sed "/^INCPATH@<:@^A-Z=@:>@*=/!d; + $qt_sed_filter" $at_mfile`]) + AC_SUBST([QT_INCPATH], + [$at_cv_env_QT_INCPATH]) + + AC_SUBST([QT_CPPFLAGS], + ["$at_cv_env_QT_DEFINES $at_cv_env_QT_INCPATH"]) + + # Find the LFLAGS of Qt (should have been named LDFLAGS). + AC_CACHE_CHECK([for the LDFLAGS to use with Qt], + [at_cv_env_QT_LDFLAGS], + [at_cv_env_QT_LDFLAGS=`sed "/^LFLAGS@<:@^A-Z=@:>@*=/!d; + $qt_sed_filter" $at_mfile`]) + AC_SUBST([QT_LFLAGS], + [$at_cv_env_QT_LDFLAGS]) + AC_SUBST([QT_LDFLAGS], + [$at_cv_env_QT_LDFLAGS]) + + # Find the LIBS of Qt. + AC_CACHE_CHECK([for the LIBS to use with Qt], + [at_cv_env_QT_LIBS], + [at_cv_env_QT_LIBS=`sed "/^LIBS@<:@^A-Z@:>@*=/!d; + $qt_sed_filter" $at_mfile` + if test x$at_darwin = xyes; then + # Fix QT_LIBS: as of today Libtool (GNU Libtool 1.5.23a) + # doesn't handle -F properly. The "bug" has been fixed on 22 + # October 2006 by Peter O'Gorman but we provide backward + # compatibility here. + at_cv_env_QT_LIBS=`echo "$at_cv_env_QT_LIBS" \ + | sed 's/^-F/-Wl,-F/; + s/ -F/ -Wl,-F/g'` + fi]) + AC_SUBST([QT_LIBS], + [$at_cv_env_QT_LIBS]) + + # We can't use AC_CACHE_CHECK for data that contains newlines. + AC_MSG_CHECKING([for necessary static plugin code]) + # find static plugin data generated by qmake + if test -f conftest.dir_plugin_import.cpp; then + QT_STATIC_PLUGINS=`cat conftest.dir_plugin_import.cpp` + else + QT_STATIC_PLUGINS="\ +// We have Qt earlier than version 5 or a dynamic build. +// Provide dummy typedef to avoid empty source code. +typedef int _qt_not_a_static_build;" + fi + AC_SUBST([QT_STATIC_PLUGINS]) + AM_SUBST_NOTMAKE([QT_STATIC_PLUGINS]) + AC_MSG_RESULT([$QT_STATIC_PLUGINS]) + + cd .. && rm -rf conftest.dir + + # Run the user code + $5 + + done # end hack (useless FOR to be able to use break) + ]) + + +# AT_REQUIRE_QT_VERSION(QT_version, [RUN-IF-FAILED], [RUN-IF-OK]) +# --------------------------------------------------------------- +# Check (using qmake) that Qt's version "matches" QT_version. Must be +# run *AFTER* AT_WITH_QT. Requires autoconf 2.60. +# +# This macro is ignored if Qt support has been disabled (using +# `--with-qt=no' or `--without-qt'). +# +# RUN-IF-FAILED is arbitrary code to execute if Qt cannot be found or +# if any problem happens. If this argument is omitted, then +# AC_MSG_ERROR will be called. RUN-IF-OK is arbitrary code to execute +# if Qt was successfully found. +# +# This macro provides the Qt version in $(QT_VERSION). + +AC_DEFUN([AT_REQUIRE_QT_VERSION], + [AC_PREREQ([2.60]) + + # This is a hack to get decent flow control with `break'. + for _qt_ignored in once; do + + if test x"$with_qt" = x"no"; then + break + fi + + if test x"$QMAKE" = x; then + AX_INSTEAD_IF([$2], + [\$QMAKE is empty. Did you invoke AT@&t@_WITH_QT before AT@&t@_REQUIRE_QT_VERSION?]) + break + fi + + AC_CACHE_CHECK([for Qt's version], + [at_cv_QT_VERSION], + [echo "$as_me:$LINENO: Running $QMAKE --version:" \ + >& AS_MESSAGE_LOG_FD + $QMAKE --version >& AS_MESSAGE_LOG_FD 2>&1 + qmake_version_sed=['/^.*\([0-9][0-9]*\.[0-9][0-9]*\.[0-9][0-9]*\).*$/!d;s//\1/'] + at_cv_QT_VERSION=`$QMAKE --version 2>&1 \ + | sed "$qmake_version_sed"`]) + if test x"$at_cv_QT_VERSION" = x; then + AX_INSTEAD_IF([$2], + [Cannot detect Qt's version.]) + break + fi + AC_SUBST([QT_VERSION], + [$at_cv_QT_VERSION]) + AS_VERSION_COMPARE([$QT_VERSION], [$1], + [AX_INSTEAD_IF([$2], + [This package requires Qt $1 or above.]) + break]) + + # Run the user code + $3 + + done # end hack (useless FOR to be able to use break) + ]) + + +# _AT_TWEAK_PRO_FILE(QT_VAR, VALUE) +# --------------------------------- +# @internal. Tweak the variable QT_VAR in the .pro file. VALUE is an +# IFS-separated list of values, and each value is rewritten as +# follows: +# +# +value => QT_VAR += value +# -value => QT_VAR -= value +# value => QT_VAR += value + +AC_DEFUN([_AT_TWEAK_PRO_FILE], + [ # Tweak the value of $1 in the .pro file for $2. + qt_conf='' + for at_mod in $2; do + at_mod=`echo "$at_mod" | sed 's/^-//; tough + s/^+//; beef + :ough + s/^/$1 -= /;n + :eef + s/^/$1 += /'` + qt_conf="\ +$qt_conf +$at_mod" + done + echo "$qt_conf" | sed 1d >>"$pro_file" + ]) + +# eof
--- a/m4/m4_ax_have_qt.m4 Wed Jan 24 20:18:59 2024 -0500 +++ /dev/null Thu Jan 01 00:00:00 1970 +0000 @@ -1,268 +0,0 @@ -# =========================================================================== -# https://www.gnu.org/software/autoconf-archive/ax_have_qt.html -# =========================================================================== -# -# SYNOPSIS -# -# AX_HAVE_QT -# -# DESCRIPTION -# -# Searches $PATH and queries qmake for Qt include files, libraries and Qt -# binary utilities. The macro only supports Qt5 or later. -# -# The following shell variable is set to either "yes" or "no": -# -# have_qt -# -# Additionally, the following variables are exported: -# -# QT_CXXFLAGS -# QT_LIBS -# QT_MOC -# QT_UIC -# QT_RCC -# QT_LRELEASE -# QT_LUPDATE -# QT_DIR -# QMAKE -# -# which respectively contain an "-I" flag pointing to the Qt include -# directory, link flags necessary to link with Qt and X, the full path to -# the meta object compiler and the user interface compiler both, and -# finally the variable QTDIR as Qt likes to see it defined. -# -# Example lines for Makefile.in: -# -# CXXFLAGS = @QT_CXXFLAGS@ -# MOC = @QT_MOC@ -# -# After the variables have been set, a trial compile and link is performed -# to check the correct functioning of the meta object compiler. This test -# may fail when the different detected elements stem from different -# releases of the Qt framework. In that case, an error message is emitted -# and configure stops. -# -# No common variables such as $LIBS or $CFLAGS are polluted. -# -# LICENSE -# -# Copyright (c) 2008 Bastiaan Veelo <Bastiaan@Veelo.net> -# Copyright (c) 2014 Alex Henrie <alexhenrie24@gmail.com> -# Copyright (c) 2024 Paper <mrpapersonic@gmail.com> -# -# Copying and distribution of this file, with or without modification, are -# permitted in any medium without royalty provided the copyright notice -# and this notice are preserved. This file is offered as-is, without any -# warranty. - -#serial 25 - -AU_ALIAS([BNV_HAVE_QT], [AX_HAVE_QT]) -AC_DEFUN([AX_HAVE_QT], -[ - AC_REQUIRE([AC_PROG_CXX]) - AC_REQUIRE([AC_PATH_X]) - AC_REQUIRE([AC_PATH_XTRA]) - # openSUSE leap 15.3 installs qmake-qt5, not qmake, for example. - # Store the full name (like qmake-qt5) into QMAKE - # and the specifier (like -qt5 or empty) into am_have_qt_qmexe_suff. - AC_ARG_VAR([QMAKE],"Qt make tool") - AC_CHECK_TOOLS([QMAKE],[qmake qmake-qt6 qmake-qt5],[false]) - - AC_MSG_CHECKING(for Qt) - am_have_qt_qmexe_suff=`echo $QMAKE | sed 's,^.*qmake,,'` - # If we have Qt5 or later in the path, we're golden - ver=`$QMAKE --version | grep -o "Qt version ."` - - if test "$ver" ">" "Qt version 4"; then - have_qt=yes - # This pro file dumps qmake's variables, but it only works on Qt 5 or later - am_have_qt_dir=`mktemp -d` - am_have_qt_pro="$am_have_qt_dir/test.pro" - am_have_qt_stash="$am_have_qt_dir/.qmake.stash" - am_have_qt_makefile="$am_have_qt_dir/Makefile" - # http://qt-project.org/doc/qt-5/qmake-variable-reference.html#qt - cat > $am_have_qt_pro << EOF -win32 { - CONFIG -= debug_and_release - CONFIG += release -} - -CONFIG += sdk_no_version_check - -# use absolute paths, useful on windows -# where qmake really loves giving relative paths -QMAKE_PROJECT_DEPTH = 0 - -# commented out all the modules we don't use -#qtHaveModule(axcontainer): QT += axcontainer -#qtHaveModule(axserver): QT += axserver -#qtHaveModule(concurrent): QT += concurrent -qtHaveModule(core): QT += core -#qtHaveModule(dbus): QT += dbus -#qtHaveModule(declarative): QT += declarative -#qtHaveModule(designer): QT += designer -qtHaveModule(gui): QT += gui -#qtHaveModule(help): QT += help -#qtHaveModule(multimedia): QT += multimedia -#qtHaveModule(multimediawidgets): QT += multimediawidgets -#qtHaveModule(network): QT += network -#qtHaveModule(opengl): QT += opengl -#qtHaveModule(printsupport): QT += printsupport -#qtHaveModule(qml): QT += qml -#qtHaveModule(qmltest): QT += qmltest -#qtHaveModule(x11extras): QT += x11extras -#qtHaveModule(script): QT += script -#qtHaveModule(scripttools): QT += scripttools -#qtHaveModule(sensors): QT += sensors -#qtHaveModule(serialport): QT += serialport -#qtHaveModule(sql): QT += sql -#qtHaveModule(svg): QT += svg -#qtHaveModule(testlib): QT += testlib -#qtHaveModule(uitools): QT += uitools -#qtHaveModule(webkit): QT += webkit -#qtHaveModule(webkitwidgets): QT += webkitwidgets -#qtHaveModule(xml): QT += xml -#qtHaveModule(xmlpatterns): QT += xmlpatterns -qtHaveModule(widgets): QT += widgets -percent.target = % -percent.commands = @echo -n "\$(\$(@))\ " -QMAKE_EXTRA_TARGETS += percent -EOF - am_have_qt_makefile_cxxflags=`cat << EOF -include $am_have_qt_makefile - -VAR: - @echo \\$(CXXFLAGS) \\$(INCPATH) -EOF` - am_have_qt_makefile_libs=`cat << EOF -include $am_have_qt_makefile - -VAR: - @echo \\$(LIBS) -EOF` - $QMAKE "$am_have_qt_pro" -o "$am_have_qt_makefile" - QT_CXXFLAGS=`cd $am_have_qt_dir; echo "\$am_have_qt_makefile_cxxflags" | make -s -f - VAR` - QT_LIBS=`cd $am_have_qt_dir; echo "\$am_have_qt_makefile_libs" | make -s -f - VAR` - rm $am_have_qt_pro $am_have_qt_stash $am_have_qt_makefile - rmdir $am_have_qt_dir - - # Look for specific tools in $PATH - QT_MOC=`which moc$am_have_qt_qmexe_suff` - QT_UIC=`which uic$am_have_qt_qmexe_suff` - QT_RCC=`which rcc$am_have_qt_qmexe_suff` - QT_LRELEASE=`which lrelease$am_have_qt_qmexe_suff` - QT_LUPDATE=`which lupdate$am_have_qt_qmexe_suff` - - # Get Qt version from qmake - QT_DIR=`$QMAKE --version | grep -o -E /.+` - - # All variables are defined, report the result - AC_MSG_RESULT([$have_qt: - QT_CXXFLAGS=$QT_CXXFLAGS - QT_DIR=$QT_DIR - QT_LIBS=$QT_LIBS - QT_UIC=$QT_UIC - QT_MOC=$QT_MOC - QT_RCC=$QT_RCC - QT_LRELEASE=$QT_LRELEASE - QT_LUPDATE=$QT_LUPDATE]) - else - # Qt was not found - have_qt=no - QT_CXXFLAGS= - QT_DIR= - QT_LIBS= - QT_UIC= - QT_MOC= - QT_RCC= - QT_LRELEASE= - QT_LUPDATE= - AC_MSG_RESULT($have_qt) - fi - AC_SUBST(QT_CXXFLAGS) - AC_SUBST(QT_DIR) - AC_SUBST(QT_LIBS) - AC_SUBST(QT_UIC) - AC_SUBST(QT_MOC) - AC_SUBST(QT_RCC) - AC_SUBST(QT_LRELEASE) - AC_SUBST(QT_LUPDATE) - AC_SUBST(QMAKE) - - #### Being paranoid: - if test x"$have_qt" = xyes; then - AC_MSG_CHECKING(correct functioning of Qt installation) - AC_CACHE_VAL(ax_cv_qt_test_result, - [ - cat > ax_qt_test.h << EOF -#include <qobject.h> -class Test : public QObject -{ -Q_OBJECT -public: - Test() {} - ~Test() {} -public slots: - void receive() {} -signals: - void send(); -}; -EOF - - cat > ax_qt_main.$ac_ext << EOF -#include "ax_qt_test.h" -#include <qapplication.h> -int main( int argc, char **argv ) -{ - QApplication app( argc, argv ); - Test t; - QObject::connect( &t, SIGNAL(send()), &t, SLOT(receive()) ); -} -EOF - - ax_cv_qt_test_result="failure" - ax_try_1="$QT_MOC ax_qt_test.h -o moc_ax_qt_test.$ac_ext >/dev/null 2>/dev/null" - AC_TRY_EVAL(ax_try_1) - if test x"$ac_status" != x0; then - echo "$ax_err_1" >&AS_MESSAGE_LOG_FD - echo "configure: could not run $QT_MOC on:" >&AS_MESSAGE_LOG_FD - cat ax_qt_test.h >&AS_MESSAGE_LOG_FD - else - ax_try_2="$CXX $QT_CXXFLAGS -c $CXXFLAGS -o moc_ax_qt_test.o moc_ax_qt_test.$ac_ext >/dev/null 2>/dev/null" - AC_TRY_EVAL(ax_try_2) - if test x"$ac_status" != x0; then - echo "$ax_err_2" >&AS_MESSAGE_LOG_FD - echo "configure: could not compile:" >&AS_MESSAGE_LOG_FD - cat moc_ax_qt_test.$ac_ext >&AS_MESSAGE_LOG_FD - else - ax_try_3="$CXX $QT_CXXFLAGS -c $CXXFLAGS -o ax_qt_main.o ax_qt_main.$ac_ext >/dev/null 2>/dev/null" - AC_TRY_EVAL(ax_try_3) - if test x"$ac_status" != x0; then - echo "$ax_err_3" >&AS_MESSAGE_LOG_FD - echo "configure: could not compile:" >&AS_MESSAGE_LOG_FD - cat ax_qt_main.$ac_ext >&AS_MESSAGE_LOG_FD - else - ax_try_4="$CXX -o ax_qt_main ax_qt_main.o moc_ax_qt_test.o $QT_LIBS $LIBS >/dev/null 2>/dev/null" - AC_TRY_EVAL(ax_try_4) - if test x"$ac_status" != x0; then - echo "$ax_err_4" >&AS_MESSAGE_LOG_FD - else - ax_cv_qt_test_result="success" - fi - fi - fi - fi - ])dnl AC_CACHE_VAL ax_cv_qt_test_result - AC_MSG_RESULT([$ax_cv_qt_test_result]) - if test x"$ax_cv_qt_test_result" = "xfailure"; then - AC_MSG_ERROR([Failed to find matching components of a complete - Qt installation. Try using more options, - see ./configure --help.]) - fi - - rm -f ax_qt_test.h moc_ax_qt_test.$ac_ext moc_ax_qt_test.o \ - ax_qt_main.$ac_ext ax_qt_main.o ax_qt_main - fi -])
--- a/src/core/config.cc Wed Jan 24 20:18:59 2024 -0500 +++ b/src/core/config.cc Sun Feb 04 21:17:17 2024 -0500 @@ -23,6 +23,8 @@ #include <QFile> #include <QTextStream> +#include <iostream> + /* I'll use an INI-based config file instead of using an * XML file like Taiga. * @@ -89,14 +91,16 @@ } } + locale.RefreshAvailableLocales(); locale.SetActiveLocale(QLocale(Strings::ToQString(INI::GetIniValue<std::string>(ini, "General", "Locale", "en_US")))); theme.SetTheme(Translate::ToTheme(INI::GetIniValue<std::string>(ini, "Appearance", "Theme", "Default"))); { std::vector<std::string> v = Strings::Split(INI::GetIniValue<std::string>(ini, "Library", "Folders", ""), ";"); - library.paths = std::set(std::make_move_iterator(v.begin()), - std::make_move_iterator(v.end())); + for (const auto& s : v) + if (!library.paths.count(s)) + library.paths.insert(s); } library.real_time_monitor = INI::GetIniValue<bool>(ini, "Library", "Real-time monitor", true); @@ -104,7 +108,7 @@ return 0; } -int Config::Save() const { +int Config::Save() { std::filesystem::path cfg_path = Filesystem::GetConfigPath(); Filesystem::CreateDirectories(cfg_path);
--- a/src/core/filesystem.cc Wed Jan 24 20:18:59 2024 -0500 +++ b/src/core/filesystem.cc Sun Feb 04 21:17:17 2024 -0500 @@ -1,23 +1,10 @@ -#ifdef WIN32 -# include <shlobj.h> -#elif defined(MACOSX) -# include "sys/osx/filesystem.h" -#elif defined(__linux__) -# include <pwd.h> -# include <sys/types.h> -#endif - -#ifndef WIN32 -# include <errno.h> -# include <unistd.h> -# include <sys/stat.h> -#endif - #include "core/filesystem.h" #include "core/config.h" #include "core/strings.h" + +#include <QStandardPaths> + #include <filesystem> -#include <limits.h> namespace Filesystem { @@ -33,39 +20,18 @@ } std::filesystem::path GetDotPath() { + /* + * Windows: ~/AppData/Roaming/Minori + * macOS: ~/Library/Application Support/Minori + * ...: ~/.config/minori + * + * FIXME: are windows and mac properly cased? + */ #ifdef WIN32 - std::filesystem::path path; - wchar_t* buf; - - if (SHGetKnownFolderPath(FOLDERID_RoamingAppData, KF_FLAG_CREATE, NULL, &buf) == S_OK) - path = buf; - else - return std::filesystem::path(); - - CoTaskMemFree(buf); - - return path / CONFIG_DIR; -#elif defined(MACOSX) - std::string appsupport; - if (!osx::GetApplicationSupportDirectory(appsupport)) - return ""; - - return std::filesystem::path(appsupport) / CONFIG_DIR; -#else // just assume POSIX - std::filesystem::path path; - const char* home = getenv("HOME"); - -# ifdef __linux__ - if (!home) - home = getpwuid(getuid())->pw_dir; -# endif // __linux__ - - /* only do this if the home directory was really found */ - if (home) - return std::filesystem::path(home) / ".config" / CONFIG_DIR; - else - return std::filesystem::path(); -#endif // !WIN32 && !MACOSX + return Strings::ToUtf8String(QStandardPaths::writableLocation(QStandardPaths::AppDataLocation)); +#else + return Strings::ToUtf8String(QStandardPaths::writableLocation(QStandardPaths::AppConfigLocation)); +#endif } std::filesystem::path GetConfigPath() {
--- a/src/core/strings.cc Wed Jan 24 20:18:59 2024 -0500 +++ b/src/core/strings.cc Sun Feb 04 21:17:17 2024 -0500 @@ -23,7 +23,7 @@ /* ew */ std::string Implode(const std::vector<std::string>& vector, const std::string& delimiter) { if (vector.size() < 1) - return "-"; + return ""; std::string out; @@ -38,7 +38,7 @@ std::string Implode(const std::set<std::string>& set, const std::string& delimiter) { if (set.size() < 1) - return "-"; + return ""; std::string out; @@ -52,6 +52,9 @@ } std::vector<std::string> Split(const std::string &text, const std::string& delimiter) { + if (text.length() < 1) + return {}; + std::vector<std::string> tokens; std::size_t start = 0, end = 0; @@ -91,7 +94,8 @@ } /* removes dumb HTML tags because anilist is aids and - gives us HTML for synopses :/ */ + * gives us HTML for synopses :/ +*/ std::string RemoveHtmlTags(std::string string) { while (string.find("<") != std::string::npos) { auto startpos = string.find("<");
--- a/src/gui/dialog/information.cc Wed Jan 24 20:18:59 2024 -0500 +++ b/src/gui/dialog/information.cc Sun Feb 04 21:17:17 2024 -0500 @@ -30,6 +30,9 @@ /* TODO: Taiga disables rendering of the tab widget entirely when the anime is not part of a list, which sucks. Think of a better way to implement this later. */ void InformationDialog::SaveData(Anime::Anime& anime) { + if (!anime.IsInUserList()) + return; + anime.SetUserProgress(_progress); anime.SetUserScore(_score); anime.SetUserIsRewatching(_rewatching); @@ -98,7 +101,7 @@ tabbed_widget->addTab(main_information_widget, tr("Main information")); } - { + if (anime.IsInUserList()) { /* My list and settings */ QWidget* settings_widget = new QWidget(tabbed_widget); settings_widget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Maximum);
--- a/src/gui/pages/anime_list.cc Wed Jan 24 20:18:59 2024 -0500 +++ b/src/gui/pages/anime_list.cc Sun Feb 04 21:17:17 2024 -0500 @@ -275,7 +275,6 @@ void AnimeListPage::DisplayListMenu() { QMenu* menu = new QMenu(this); menu->setAttribute(Qt::WA_DeleteOnClose); - menu->setTitle(tr("Column visibility")); menu->setToolTipsVisible(true); AnimeListPageModel* source_model = @@ -424,6 +423,7 @@ void AnimeListPage::showEvent(QShowEvent*) { SetupLayout(); + Refresh(); } /* --------- QTabWidget replication end ---------- */ @@ -497,6 +497,4 @@ SetColumnDefaults(); setFocusPolicy(Qt::TabFocus); setFocusProxy(tab_bar); - - Refresh(); }
--- a/src/gui/pages/search.cc Wed Jan 24 20:18:59 2024 -0500 +++ b/src/gui/pages/search.cc Sun Feb 04 21:17:17 2024 -0500 @@ -1,4 +1,346 @@ #include "gui/pages/search.h" +#include "core/anime.h" +#include "core/anime_db.h" +#include "core/strings.h" +#include "core/http.h" +#include "core/session.h" +#include "core/filesystem.h" +#include "gui/widgets/text.h" +#include "gui/dialog/information.h" +#include "track/media.h" +#include "gui/translate/anime.h" +#include "services/services.h" + +#include <QHeaderView> +#include <QVBoxLayout> +#include <QToolBar> +#include <QTreeView> +#include <QDate> +#include <QMenu> + +#include <iostream> +#include <sstream> +#include <fstream> +#include <algorithm> + +#include "pugixml.hpp" +#include "anitomy/anitomy.h" + +SearchPageListSortFilter::SearchPageListSortFilter(QObject* parent) : QSortFilterProxyModel(parent) { +} + +bool SearchPageListSortFilter::lessThan(const QModelIndex& l, const QModelIndex& r) const { + QVariant left = sourceModel()->data(l, sortRole()); + QVariant right = sourceModel()->data(r, sortRole()); + + switch (left.userType()) { + case QMetaType::Int: + case QMetaType::UInt: + case QMetaType::LongLong: + case QMetaType::ULongLong: + return left.toInt() < right.toInt(); + case QMetaType::QDate: + return left.toDate() < right.toDate(); + case QMetaType::QString: + default: // meh + return QString::compare(left.toString(), right.toString(), Qt::CaseInsensitive) < 0; + } +} + +/* -------------------------------------------- */ + +SearchPageListModel::SearchPageListModel(QObject* parent) : QAbstractListModel(parent) { +} + +void SearchPageListModel::ParseSearch(const std::vector<int>& ids) { + /* hack!!! */ + if (!rowCount(index(0))) { + beginInsertRows(QModelIndex(), 0, 0); + endInsertRows(); + } + + beginResetModel(); + + this->ids = ids; + + endResetModel(); +} + +int SearchPageListModel::rowCount(const QModelIndex& parent) const { + return ids.size(); + (void)(parent); +} + +int SearchPageListModel::columnCount(const QModelIndex& parent) const { + return NB_COLUMNS; + (void)(parent); +} + +QVariant SearchPageListModel::headerData(const int section, const Qt::Orientation orientation, const int role) const { + switch (role) { + case Qt::DisplayRole: { + switch (section) { + case SR_TITLE: return tr("Anime title"); + case SR_EPISODES: return tr("Episode"); + case SR_TYPE: return tr("Type"); + case SR_SCORE: return tr("Score"); + case SR_SEASON: return tr("Season"); + default: return {}; + } + break; + } + case Qt::TextAlignmentRole: { + switch (section) { + case SR_TITLE: return QVariant(Qt::AlignLeft | Qt::AlignVCenter); + case SR_TYPE: return QVariant(Qt::AlignHCenter | Qt::AlignVCenter); + case SR_EPISODES: + case SR_SCORE: + case SR_SEASON: return QVariant(Qt::AlignRight | Qt::AlignVCenter); + default: return {}; + } + break; + } + } + return QAbstractListModel::headerData(section, orientation, role); +} + +QVariant SearchPageListModel::data(const QModelIndex& index, int role) const { + if (!index.isValid()) + return QVariant(); + + const Anime::Anime& anime = Anime::db.items[ids[index.row()]]; + + switch (role) { + case Qt::DisplayRole: + switch (index.column()) { + case SR_TITLE: return Strings::ToQString(anime.GetUserPreferredTitle()); + case SR_TYPE: return Strings::ToQString(Translate::ToLocalString(anime.GetFormat())); + case SR_EPISODES: return anime.GetEpisodes(); + case SR_SCORE: return QString::number(anime.GetAudienceScore()) + "%"; + case SR_SEASON: return Strings::ToQString(Translate::ToLocalString(anime.GetSeason())) + " " + QString::number(anime.GetAirDate().GetYear().value_or(2000)); + default: return {}; + } + break; + case Qt::UserRole: + switch (index.column()) { + case SR_SCORE: return anime.GetAudienceScore(); + case SR_EPISODES: return anime.GetEpisodes(); + case SR_SEASON: return anime.GetAirDate().GetAsQDate(); + /* We have to use this to work around some stupid + * "conversion ambiguous" error on Linux + */ + default: return data(index, Qt::DisplayRole); + } + break; + case Qt::SizeHintRole: { + switch (index.column()) { + default: { + /* max horizontal size of 100, height size = size of current font */ + const QString d = data(index, Qt::DisplayRole).toString(); + const QFontMetrics metric = QFontMetrics(QFont()); + + return QSize(std::max(metric.horizontalAdvance(d), 100), metric.height()); + } + } + break; + } + case Qt::TextAlignmentRole: + return headerData(index.column(), Qt::Horizontal, Qt::TextAlignmentRole); + } + return QVariant(); +} + +Qt::ItemFlags SearchPageListModel::flags(const QModelIndex& index) const { + if (!index.isValid()) + return Qt::NoItemFlags; + + return Qt::ItemIsEnabled | Qt::ItemIsSelectable; +} + +Anime::Anime* SearchPageListModel::GetAnimeFromIndex(const QModelIndex& index) const { + return &Anime::db.items[ids[index.row()]]; +} + +void SearchPage::DisplayListMenu() { + QMenu* menu = new QMenu(this); + menu->setAttribute(Qt::WA_DeleteOnClose); + menu->setToolTipsVisible(true); + + const QItemSelection selection = sort_model->mapSelectionToSource(treeview->selectionModel()->selection()); + + bool add_to_list_enable = true; -SearchPage::SearchPage(QWidget* parent) : QWidget(parent) { + std::set<Anime::Anime*> animes; + for (const auto& index : selection.indexes()) { + if (!index.isValid()) + continue; + + Anime::Anime* anime = model->GetAnimeFromIndex(index); + if (anime) { + animes.insert(anime); + if (anime->IsInUserList()) + add_to_list_enable = false; + } + } + + menu->addAction(tr("Information"), [this, animes] { + for (auto& anime : animes) { + InformationDialog* dialog = new InformationDialog(*anime, [this, anime] { + //UpdateAnime(anime->GetId()); + }, InformationDialog::PAGE_MAIN_INFO, this); + + dialog->show(); + dialog->raise(); + dialog->activateWindow(); + } + }); + menu->addSeparator(); + { + QMenu* submenu = menu->addMenu(tr("Add to list...")); + submenu->addAction(tr("Currently watching"), [animes]{ + for (auto& anime : animes) { + if (!anime->IsInUserList()) + anime->AddToUserList(); + anime->SetUserStatus(Anime::ListStatus::CURRENT); + Services::UpdateAnimeEntry(anime->GetId()); + } + }); + submenu->addAction(tr("Completed"), [animes]{ + for (auto& anime : animes) { + if (!anime->IsInUserList()) + anime->AddToUserList(); + anime->SetUserStatus(Anime::ListStatus::COMPLETED); + Services::UpdateAnimeEntry(anime->GetId()); + } + }); + submenu->addAction(tr("On hold"), [animes]{ + for (auto& anime : animes) { + if (!anime->IsInUserList()) + anime->AddToUserList(); + anime->SetUserStatus(Anime::ListStatus::PAUSED); + Services::UpdateAnimeEntry(anime->GetId()); + } + }); + submenu->addAction(tr("Dropped"), [animes]{ + for (auto& anime : animes) { + if (!anime->IsInUserList()) + anime->AddToUserList(); + anime->SetUserStatus(Anime::ListStatus::DROPPED); + Services::UpdateAnimeEntry(anime->GetId()); + } + }); + submenu->addAction(tr("Plan to watch"), [animes]{ + for (auto& anime : animes) { + if (!anime->IsInUserList()) + anime->AddToUserList(); + anime->SetUserStatus(Anime::ListStatus::PLANNING); + Services::UpdateAnimeEntry(anime->GetId()); + } + }); + submenu->setEnabled(add_to_list_enable); + } + menu->popup(QCursor::pos()); } + +void SearchPage::ItemDoubleClicked() { + /* throw out any other garbage */ + const QItemSelection selection = sort_model->mapSelectionToSource(treeview->selectionModel()->selection()); + if (!selection.indexes().first().isValid()) + return; + + const QModelIndex index = model->index(selection.indexes().first().row()); + Anime::Anime* anime = model->GetAnimeFromIndex(index); + + InformationDialog* dialog = new InformationDialog(*anime, [this, anime] { + //UpdateAnime(anime->GetId()); + }, InformationDialog::PAGE_MAIN_INFO, this); + + dialog->show(); + dialog->raise(); + dialog->activateWindow(); +} + +SearchPage::SearchPage(QWidget* parent) : QFrame(parent) { + setFrameShape(QFrame::Box); + setFrameShadow(QFrame::Sunken); + + QVBoxLayout* layout = new QVBoxLayout(this); + layout->setContentsMargins(0, 0, 0, 0); + layout->setSpacing(0); + + { + /* Toolbar */ + QToolBar* toolbar = new QToolBar(this); + toolbar->setMovable(false); + + { + QLineEdit* line_edit = new QLineEdit("", toolbar); + connect(line_edit, &QLineEdit::returnPressed, this, [this, line_edit]{ + /* static thread here. */ + static QThread* thread = nullptr; + + if (thread) + return; + + thread = QThread::create([this, line_edit]{ + model->ParseSearch(Services::Search(Strings::ToUtf8String(line_edit->text()))); + }); + + connect(thread, &QThread::finished, this, []{ + thread->deleteLater(); + thread = nullptr; + }); + + thread->start(); + }); + toolbar->addWidget(line_edit); + } + + layout->addWidget(toolbar); + } + + { + QFrame* line = new QFrame(this); + line->setFrameShape(QFrame::HLine); + line->setFrameShadow(QFrame::Sunken); + line->setLineWidth(1); + layout->addWidget(line); + } + + { + treeview = new QTreeView(this); + treeview->setUniformRowHeights(true); + treeview->setAllColumnsShowFocus(false); + treeview->setAlternatingRowColors(true); + treeview->setSortingEnabled(true); + treeview->setSelectionMode(QAbstractItemView::ExtendedSelection); + treeview->setItemsExpandable(false); + treeview->setRootIsDecorated(false); + treeview->setContextMenuPolicy(Qt::CustomContextMenu); + treeview->setFrameShape(QFrame::NoFrame); + + { + sort_model = new SearchPageListSortFilter(treeview); + model = new SearchPageListModel(treeview); + sort_model->setSourceModel(model); + sort_model->setSortRole(Qt::UserRole); + sort_model->setSortCaseSensitivity(Qt::CaseInsensitive); + treeview->setModel(sort_model); + } + + // set column sizes + treeview->setColumnWidth(SearchPageListModel::SR_TITLE, 400); + treeview->setColumnWidth(SearchPageListModel::SR_TYPE, 60); + treeview->setColumnWidth(SearchPageListModel::SR_EPISODES, 60); + treeview->setColumnWidth(SearchPageListModel::SR_SCORE, 60); + treeview->setColumnWidth(SearchPageListModel::SR_SEASON, 100); + + treeview->header()->setStretchLastSection(false); + + /* Double click stuff */ + connect(treeview, &QAbstractItemView::doubleClicked, this, &SearchPage::ItemDoubleClicked); + connect(treeview, &QWidget::customContextMenuRequested, this, &SearchPage::DisplayListMenu); + + layout->addWidget(treeview); + } +}
--- a/src/gui/pages/statistics.cc Wed Jan 24 20:18:59 2024 -0500 +++ b/src/gui/pages/statistics.cc Sun Feb 04 21:17:17 2024 -0500 @@ -14,6 +14,12 @@ #include <QWidget> #include <sstream> +#include <cmath> + +enum class TimeUnits { + SECONDS, + MINUTES +}; StatisticsPage::StatisticsPage(QWidget* parent) : QFrame(parent) { QVBoxLayout* layout = new QVBoxLayout(this); @@ -74,47 +80,49 @@ UpdateStatistics(); } -/* me abusing macros :) */ -static void add_time_segment(std::ostringstream& str, int x, const std::string_view& s, const std::string_view& p) { - if (x > 0) - str << x << ((x == 1) ? s : p); -} +/* [in] enum TimeUnits unit: + * which unit to stop on + * [in] int amount: + * amount of units to parse + * [in, defaults to 1.0] double unit_in_seconds: + * equivalent of one of 'amount' in seconds, e.g. minutes would be 60.0 +*/ +static std::string TimeToDateString(TimeUnits unit, int amount, double unit_in_seconds = 1.0) { + /* avoid calculating this twice */ + const double years_conv = (31556952.0 / unit_in_seconds); + const double months_conv = (2629746.0 / unit_in_seconds); + const double days_conv = (86400.0 / unit_in_seconds); + const double hours_conv = (3600.0 / unit_in_seconds); + const double minutes_conv = (60.0 / unit_in_seconds); + const double seconds_conv = (1.0 / unit_in_seconds); -std::string StatisticsPage::MinutesToDateString(const int minutes) { - /* ew */ - int years = (minutes * (1 / 525949.2F)); - int months = (minutes * (1 / 43829.1F)) - (years * 12); - int days = (minutes * (1 / 1440.0F)) - (years * 365.2425F) - (months * 30.436875F); - int hours = (minutes * (1 / 60.0F)) - (years * 8765.82F) - (months * 730.485F) - (days * 24); - int rest_minutes = (minutes) - (years * 525949.2F) - (months * 43829.1F) - (days * 1440) - (hours * 60); - std::ostringstream return_stream; - add_time_segment(return_stream, years, " year ", " years "); - add_time_segment(return_stream, months, " month ", " months "); - add_time_segment(return_stream, days, " day ", " days "); - add_time_segment(return_stream, hours, " hour ", " hours "); - if (rest_minutes > 0 || return_stream.str().size() == 0) - return_stream << rest_minutes << ((rest_minutes == 1) ? " minute" : " minutes"); - return return_stream.str(); -} + const int years = amount / years_conv; + const int months = std::fmod(amount, years_conv) / months_conv; + const int days = std::fmod(amount, months_conv) / days_conv; + const int hours = std::fmod(amount, days_conv) / hours_conv; + const int minutes = std::fmod(amount, hours_conv) / minutes_conv; + const int seconds = std::fmod(amount, minutes_conv) / seconds_conv; + + const auto add_time_segment = [](std::ostringstream& str, int amount, const std::string_view& singular, const std::string_view& plural, bool always = false) { + if (amount > 0 || always) + str << amount << ((amount == 1) ? singular : plural); + }; -std::string StatisticsPage::SecondsToDateString(const int sec) { - /* this is all fairly unnecessary, but works:tm: */ - int years = sec * (1 / 31556952.0F); - int months = sec * (1 / 2629746.0F) - (years * 12); - int days = sec * (1 / 86400.0F) - (years * 365.2425F) - (months * 30.436875F); - int hours = sec * (1 / 3600.0F) - (years * 8765.82F) - (months * 730.485F) - (days * 24); - int minutes = (sec) * (1 / 60.0F) - (years * 525949.2F) - (months * 43829.1F) - (days * 1440.0F) - (hours * 60.0F); - int seconds = - sec - (years * 31556952.0F) - (months * 2629746.0F) - (days * 86400.0F) - (hours * 3600.0F) - (minutes * 60.0F); - std::ostringstream return_stream; - add_time_segment(return_stream, years, " year ", " years "); - add_time_segment(return_stream, months, " month ", " months "); - add_time_segment(return_stream, days, " day ", " days "); - add_time_segment(return_stream, hours, " hour ", " hours "); - add_time_segment(return_stream, minutes, " minute ", " minutes "); - if (seconds > 0 || return_stream.str().size() == 0) - return_stream << seconds << ((seconds == 1) ? " second" : " seconds"); - return return_stream.str(); + std::ostringstream string; + add_time_segment(string, years, " year ", " years "); + add_time_segment(string, months, " month ", " months "); + add_time_segment(string, days, " day ", " days "); + add_time_segment(string, hours, " hour ", " hours "); + + if (unit == TimeUnits::MINUTES) { + add_time_segment(string, minutes, " minute", " minutes", true); + return string.str(); + } else { + add_time_segment(string, minutes, " minute ", " minutes "); + } + + add_time_segment(string, seconds, " second", " seconds", true); + return string.str(); } inline int GetTotalWithScore(const int score) { @@ -131,8 +139,8 @@ QTextStream ts(&string); ts << Anime::db.GetTotalAnimeAmount() << '\n'; ts << Anime::db.GetTotalEpisodeAmount() << '\n'; - ts << MinutesToDateString(Anime::db.GetTotalWatchedAmount()).c_str() << '\n'; - ts << MinutesToDateString(Anime::db.GetTotalPlannedAmount()).c_str() << '\n'; + ts << Strings::ToQString(TimeToDateString(TimeUnits::MINUTES, Anime::db.GetTotalWatchedAmount(), 60.0)) << '\n'; + ts << Strings::ToQString(TimeToDateString(TimeUnits::MINUTES, Anime::db.GetTotalPlannedAmount(), 60.0)) << '\n'; ts << Anime::db.GetAverageScore() << '\n'; ts << Anime::db.GetScoreDeviation(); _anime_list->GetParagraph()->SetText(string); @@ -142,7 +150,7 @@ _score_distribution_graph->AddItem(i, GetTotalWithScore(i)); string = ""; - ts << Strings::ToQString(SecondsToDateString(session.uptime() / 1000)) << '\n'; + ts << Strings::ToQString(TimeToDateString(TimeUnits::SECONDS, session.uptime() / 1000)) << '\n'; ts << session.GetRequests(); /* Application */ // UiUtils::SetPlainTextEditData(application_data, QString::number(session.uptime() / 1000));
--- a/src/gui/pages/torrents.cc Wed Jan 24 20:18:59 2024 -0500 +++ b/src/gui/pages/torrents.cc Sun Feb 04 21:17:17 2024 -0500 @@ -6,10 +6,10 @@ #include "gui/widgets/text.h" #include "track/media.h" +#include <QHeaderView> #include <QVBoxLayout> #include <QToolBar> #include <QTreeView> -#include <QMainWindow> #include <QByteArray> #include <QDataStream> #include <QThread> @@ -24,10 +24,11 @@ #include "anitomy/anitomy.h" /* This file is very, very similar to the anime list page. - - It differs from Taiga in that it uses tabs instead of - those "groups", but those are custom painted and a pain in the ass to - maintain over multiple platforms. */ + * + * It differs from Taiga in that it uses tabs instead of + * those "groups", but those are custom painted and a pain in the ass to + * maintain over multiple platforms. +*/ TorrentsPageListSortFilter::TorrentsPageListSortFilter(QObject* parent) : QSortFilterProxyModel(parent) { } @@ -398,6 +399,8 @@ treeview->setColumnWidth(TorrentsPageListModel::TL_FILENAME, 200); treeview->setColumnWidth(TorrentsPageListModel::TL_RELEASEDATE, 190); + treeview->header()->setStretchLastSection(false); + layout->addWidget(treeview); } }
--- a/src/gui/widgets/anime_info.cc Wed Jan 24 20:18:59 2024 -0500 +++ b/src/gui/widgets/anime_info.cc Sun Feb 04 21:17:17 2024 -0500 @@ -35,12 +35,13 @@ * QString because QTextStream sucks and assumes * Latin1 (on Windows?) */ + const auto genres = anime.GetGenres(); details_data_s << Strings::ToQString(Translate::ToLocalString(anime.GetFormat())) << "\n" << anime.GetEpisodes() << "\n" - << Strings::ToQString(Translate::ToLocalString(anime.GetUserStatus())) << "\n" + << Strings::ToQString(Translate::ToLocalString(anime.GetAiringStatus())) << "\n" << Strings::ToQString(Translate::ToLocalString(anime.GetSeason())) << " " << anime.GetAirDate().GetYear().value_or(2000) << "\n" - << Strings::ToQString(Strings::Implode(anime.GetGenres(), ", ")) << "\n" + << Strings::ToQString((genres.size() > 1) ? Strings::Implode(genres, ", ") : "-") << "\n" << anime.GetAudienceScore() << "%"; _details->GetParagraph()->SetText(details_data);
--- a/src/gui/widgets/sidebar.cc Wed Jan 24 20:18:59 2024 -0500 +++ b/src/gui/widgets/sidebar.cc Sun Feb 04 21:17:17 2024 -0500 @@ -4,6 +4,8 @@ #include <QListWidgetItem> #include <QMouseEvent> +#include <iostream> + SideBar::SideBar(QWidget* parent) : QListWidget(parent) { setFrameShape(QFrame::NoFrame); setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); @@ -30,6 +32,7 @@ void SideBar::SetBackgroundColor(QColor color) { viewport()->setAutoFillBackground(color != Qt::transparent); + QPalette pal(palette()); pal.setColor(QPalette::Window, color); setPalette(pal); @@ -66,26 +69,30 @@ } int SideBar::AddSeparatorsToIndex(int index) { - int i, j; - for (i = 0, j = 0; i < index;) { - i++; - if (IndexIsSeparator(indexFromItem(item(i)))) - j++; + int i = 0, separators = 0, items = 0; + + for (; items <= index; ) { + if (IndexIsSeparator(indexFromItem(item(i++)))) { + separators++; + } else { + items++; + } } - return i + j; + + return index + separators; } int SideBar::RemoveSeparatorsFromIndex(int index) { - int i, j; - for (i = 0, j = 0; i < index; i++) { + int i = 0, items = 0; + for (; i < index; i++) { if (!IndexIsSeparator(indexFromItem(item(i)))) - j++; + items++; } - return j; + return items; } bool SideBar::IndexIsSeparator(QModelIndex index) const { - return !(index.isValid() && index.flags() & Qt::ItemIsEnabled); + return !index.isValid() || !(index.flags() & Qt::ItemIsEnabled); } QItemSelectionModel::SelectionFlags SideBar::selectionCommand(const QModelIndex& index, const QEvent*) const {
--- a/src/gui/window.cc Wed Jan 24 20:18:59 2024 -0500 +++ b/src/gui/window.cc Sun Feb 04 21:17:17 2024 -0500 @@ -48,18 +48,6 @@ # include "sys/win32/dark_theme.h" #endif -enum class Pages { - NOW_PLAYING, - - ANIME_LIST, - HISTORY, - STATISTICS, - - SEARCH, - SEASONS, - TORRENTS -}; - void PlayingThread::run() { std::vector<std::string> files; Track::Media::GetCurrentlyPlaying(files); @@ -124,6 +112,7 @@ void MainWindow::AddMainWidgets() { int page = static_cast<int>(Pages::ANIME_LIST); + if (sidebar.get()) { main_widget->layout()->removeWidget(sidebar.get()); sidebar.reset(); @@ -149,10 +138,13 @@ sidebar->AddItem(tr("Torrents"), SideBar::CreateIcon(":/icons/16x16/feed.png")); stack.reset(new QStackedWidget(main_widget.get())); + stack->addWidget(new NowPlayingPage(main_widget.get())); + /* ---- */ stack->addWidget(new AnimeListPage(main_widget.get())); stack->addWidget(new HistoryPage(main_widget.get())); stack->addWidget(new StatisticsPage(main_widget.get())); + /* ---- */ stack->addWidget(new SearchPage(main_widget.get())); stack->addWidget(new SeasonsPage(main_widget.get())); stack->addWidget(new TorrentsPage(main_widget.get())); @@ -166,7 +158,6 @@ void MainWindow::CreateBars() { QMenuBar* menubar = new QMenuBar(this); - QMenu* folder_menu; /* this is used twice, so we declare it here */ QAction* sync_action; { @@ -176,36 +167,7 @@ { folder_menu = menu->addMenu(tr("&Library folders")); - /* add in all of our existing folders... */ - std::size_t i = 0; - for (const auto& path : session.config.library.paths) { - const QString folder = Strings::ToQString(path); - QAction* action = folder_menu->addAction(folder, [folder]{ - QDesktopServices::openUrl(QUrl::fromLocalFile(folder)); - }); - if (i < 9) - action->setShortcut(QKeySequence(Qt::ALT | static_cast<Qt::Modifier>(Qt::Key_1 + i))); - else if (i == 9) - action->setShortcut(QKeySequence(Qt::ALT | Qt::Key_0)); - /* don't bother with a shortcut in case of more... */ - i++; - } - - folder_menu->addSeparator(); - - { - folder_menu->addAction(tr("&Add new folder..."), [this]{ - const QString dir = QFileDialog::getExistingDirectory(this, tr("Open Directory"), - QDir::homePath(), - QFileDialog::ShowDirsOnly - | QFileDialog::DontResolveSymlinks); - if (dir.isEmpty()) - return; - session.config.library.paths.insert(Strings::ToUtf8String(dir)); - /* we have to recreate the menu bar to add the new folder */ - CreateBars(); - }); - } + UpdateFolderMenu(); } { @@ -229,7 +191,7 @@ menu->addSeparator(); { - QAction* action = menu->addAction(tr("E&xit"), qApp, &QApplication::quit); + QAction* action = menu->addAction(tr("E&xit"), this, &MainWindow::close); action->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_Q)); } } @@ -314,7 +276,7 @@ QAction* action = menu->addAction(tr("&Settings"), [this] { SettingsDialog dialog(this); dialog.exec(); - CreateBars(); + UpdateFolderMenu(); }); action->setMenuRole(QAction::PreferencesRole); } @@ -326,60 +288,78 @@ { /* Pages... */ - std::map<QAction*, int> page_to_index_map = {}; - - QActionGroup* pages_group = new QActionGroup(this); + QActionGroup* pages_group = new QActionGroup(menu); pages_group->setExclusive(true); { QAction* action = pages_group->addAction(menu->addAction(tr("&Now Playing"))); action->setCheckable(true); - page_to_index_map[action] = 0; + connect(action, &QAction::toggled, this, [this] { + sidebar->SetCurrentItem(0); + }); } { QAction* action = pages_group->addAction(menu->addAction(tr("&Anime List"))); action->setCheckable(true); action->setChecked(true); - page_to_index_map[action] = 1; + connect(action, &QAction::toggled, this, [this] { + sidebar->SetCurrentItem(1); + }); } { QAction* action = pages_group->addAction(menu->addAction(tr("&History"))); action->setCheckable(true); - page_to_index_map[action] = 2; + connect(action, &QAction::toggled, this, [this] { + sidebar->SetCurrentItem(2); + }); } { QAction* action = pages_group->addAction(menu->addAction(tr("&Statistics"))); action->setCheckable(true); - page_to_index_map[action] = 3; + connect(action, &QAction::toggled, this, [this] { + sidebar->SetCurrentItem(3); + }); } { QAction* action = pages_group->addAction(menu->addAction(tr("S&earch"))); action->setCheckable(true); - page_to_index_map[action] = 4; + connect(action, &QAction::toggled, this, [this] { + sidebar->SetCurrentItem(4); + }); } { QAction* action = pages_group->addAction(menu->addAction(tr("Se&asons"))); action->setCheckable(true); - page_to_index_map[action] = 5; + connect(action, &QAction::toggled, this, [this] { + sidebar->SetCurrentItem(5); + }); } { QAction* action = pages_group->addAction(menu->addAction(tr("&Torrents"))); action->setCheckable(true); - page_to_index_map[action] = 6; + connect(action, &QAction::toggled, this, [this] { + sidebar->SetCurrentItem(6); + }); } /* pain in my ass */ - connect(sidebar.get(), &SideBar::CurrentItemChanged, this, - [pages_group](int index) { pages_group->actions()[index]->setChecked(true); }); + connect(sidebar.get(), &SideBar::CurrentItemChanged, this, [pages_group](int index) { + QAction* checked = pages_group->checkedAction(); - connect(pages_group, &QActionGroup::triggered, this, - [this, page_to_index_map](QAction* action) { sidebar->SetCurrentItem(page_to_index_map.at(action)); }); + const QList<QAction*>& actions = pages_group->actions(); + if (index > actions.size()) + return; + + if (checked) + checked->setChecked(false); + actions[index]->setChecked(true); + }); } menu->addSeparator(); @@ -479,12 +459,55 @@ toolbar->addAction(QIcon(":/icons/24x24/gear.png"), tr("S&ettings"), [this] { SettingsDialog dialog(this); dialog.exec(); - CreateBars(); + /* library folders might have changed! */ + UpdateFolderMenu(); }); addToolBar(toolbar); } } +void MainWindow::UpdateFolderMenu() { + if (!folder_menu) + return; + + folder_menu->clear(); + + /* add in all of our existing folders... */ + std::size_t i = 0; + for (const auto& path : session.config.library.paths) { + const QString folder = Strings::ToQString(path); + QAction* action = folder_menu->addAction(folder, [folder]{ + QDesktopServices::openUrl(QUrl::fromLocalFile(folder)); + }); + + if (i < 9) { + /* Qt::Key_1 is equivalent to 1 in ASCII, so we can use the same + * stupid `'0' + i` trick here + */ + action->setShortcut(QKeySequence(Qt::ALT | static_cast<Qt::Modifier>(Qt::Key_1 + i))); + } else if (i == 9) { + action->setShortcut(QKeySequence(Qt::ALT | Qt::Key_0)); + } + /* don't bother with a shortcut in case of more... */ + i++; + } + + folder_menu->addSeparator(); + + { + folder_menu->addAction(tr("&Add new folder..."), [this]{ + const QString dir = QFileDialog::getExistingDirectory(this, tr("Open Directory"), + QDir::homePath(), + QFileDialog::ShowDirsOnly + | QFileDialog::DontResolveSymlinks); + if (dir.isEmpty()) + return; + session.config.library.paths.insert(Strings::ToUtf8String(dir)); + UpdateFolderMenu(); + }); + } +} + void MainWindow::SetActivePage(QWidget* page) { this->setCentralWidget(page); }
--- a/src/main.cc Wed Jan 24 20:18:59 2024 -0500 +++ b/src/main.cc Sun Feb 04 21:17:17 2024 -0500 @@ -1,7 +1,9 @@ #include "core/session.h" #include "core/anime_db.h" #include "core/strings.h" +#include "services/anilist.h" #include "gui/window.h" + #include <QApplication> #include <QStyleFactory> #include <QTranslator> @@ -13,10 +15,11 @@ int main(int argc, char** argv) { QApplication app(argc, argv); + app.setApplicationName("minori"); + app.setApplicationDisplayName("Minori"); app.setAttribute(Qt::AA_DontShowIconsInMenus, true); session.config.Load(); - session.config.locale.RefreshAvailableLocales(); Anime::db.LoadDatabaseFromDisk(); MainWindow window;
--- a/src/services/anilist.cc Wed Jan 24 20:18:59 2024 -0500 +++ b/src/services/anilist.cc Sun Feb 04 21:17:17 2024 -0500 @@ -244,6 +244,59 @@ return 1; } +/* return is a vector of anime ids */ +std::vector<int> Search(const std::string& search) { + constexpr std::string_view query = + "query ($search: String) {\n" + " Page (page: 1, perPage: 50) {\n" + " media (search: $search, type: ANIME) {\n" + " coverImage {\n" + " large\n" + " }\n" + " id\n" + " title {\n" + " romaji\n" + " english\n" + " native\n" + " }\n" + " format\n" + " status\n" + " averageScore\n" + " season\n" + " startDate {\n" + " year\n" + " month\n" + " day\n" + " }\n" + " genres\n" + " episodes\n" + " duration\n" + " synonyms\n" + " description(asHtml: false)\n" + " }\n" + " }\n" + "}\n"; + + // clang-format off + nlohmann::json json = { + {"query", query}, + {"variables", { + {"search", search} + }} + }; + // clang-format on + + auto res = SendJSONRequest(json); + + std::vector<int> ret; + ret.reserve(res["data"]["Page"]["media"].size()); + + for (const auto& media : res["data"]["Page"]["media"].items()) + ret.push_back(ParseMediaJson(media.value())); + + return ret; +} + int UpdateAnimeEntry(int id) { /** * possible values: @@ -263,18 +316,17 @@ * float[] advancedScores, * Date startedAt, * Date completedAt - **/ + **/ Anime::Anime& anime = Anime::db.items[id]; if (!anime.IsInUserList()) return 0; - constexpr std::string_view query = "mutation ($media_id: Int, $progress: Int, $status: MediaListStatus, $score: Int, " - "$notes: String, $start: FuzzyDateInput, $comp: FuzzyDateInput, $repeat: Int) {\n" - " SaveMediaListEntry (mediaId: $media_id, progress: $progress, status: $status, " - "scoreRaw: $score, notes: $notes, startedAt: $start, completedAt: $comp, repeat: $repeat) {\n" - " id\n" - " }\n" - "}\n"; + constexpr std::string_view query = + "mutation ($media_id: Int, $progress: Int, $status: MediaListStatus, $score: Int, $notes: String, $start: FuzzyDateInput, $comp: FuzzyDateInput, $repeat: Int) {\n" + " SaveMediaListEntry (mediaId: $media_id, progress: $progress, status: $status, scoreRaw: $score, notes: $notes, startedAt: $start, completedAt: $comp, repeat: $repeat) {\n" + " id\n" + " }\n" + "}\n"; // clang-format off nlohmann::json json = { {"query", query},
--- a/src/services/services.cc Wed Jan 24 20:18:59 2024 -0500 +++ b/src/services/services.cc Sun Feb 04 21:17:17 2024 -0500 @@ -2,7 +2,6 @@ #include "core/session.h" #include "gui/dialog/settings.h" #include "services/anilist.h" -#include <QMessageBox> namespace Services { @@ -13,6 +12,13 @@ } } +std::vector<int> Search(const std::string& search) { + switch (session.config.service) { + case Anime::Services::ANILIST: return AniList::Search(search); + default: return {}; + } +} + void UpdateAnimeEntry(int id) { switch (session.config.service) { case Anime::Services::ANILIST: AniList::UpdateAnimeEntry(id); break;
--- a/src/sys/glib/dark_theme.cc Wed Jan 24 20:18:59 2024 -0500 +++ b/src/sys/glib/dark_theme.cc Sun Feb 04 21:17:17 2024 -0500 @@ -1,11 +1,12 @@ #include <gio/gio.h> #include <cstring> - -#include <iostream> +#include <string_view> namespace glib { bool IsInDarkTheme() { + bool success = false; + GSettings* settings = ::g_settings_new("org.gnome.desktop.interface"); if (!settings) return false; @@ -19,12 +20,31 @@ if (!str) /* how */ return false; - bool success = !std::strcmp(str, "prefer-dark"); + success |= !std::strcmp(str, "prefer-dark"); + + ::g_variant_unref(val); + + if (success) { + ::g_object_unref(settings); + return success; + } + + GVariant* gtk_theme = ::g_settings_get_value(settings, "gtk-theme"); + if (!gtk_theme) + return false; - /* unref these */ - ::g_variant_unref(val); + const gchar* gtk_theme_str; + ::g_variant_get(gtk_theme, "&s", gtk_theme_str); + if (!gtk_theme_str) + return false; + + static constexpr std::string_view suffix = "-dark"; + + size_t gtk_theme_len = strlen(gtk_theme_str); + + success |= !std::strncmp(gtk_theme_str + gtk_theme_len - suffix.length(), suffix.data(), suffix.length()); + ::g_object_unref(settings); - return success; }
--- a/src/sys/osx/filesystem.cc Wed Jan 24 20:18:59 2024 -0500 +++ /dev/null Thu Jan 01 00:00:00 1970 +0000 @@ -1,51 +0,0 @@ -#include "sys/osx/filesystem.h" - -#include <CoreFoundation/CoreFoundation.h> -#include <objc/runtime.h> - -#include <string> - -/* These constants are defined in Foundation but not - * exposed to CoreFoundation users. -*/ -static constexpr unsigned long NSApplicationSupportDirectory = 14; -static constexpr unsigned long NSUserDomainMask = 1; - -extern "C" { - CFArrayRef NSSearchPathForDirectoriesInDomains(unsigned long directory, unsigned long domainMask, BOOL expandTilde); -} - -namespace osx { - -bool GetApplicationSupportDirectory(std::string& result) { - // NSArray* strings = NSSearchPathForDirectoriesInDomains(NSApplicationSupportDirectory, NSUserDomainMask, ON); - const CFArrayRef strings = NSSearchPathForDirectoriesInDomains(NSApplicationSupportDirectory, NSUserDomainMask, true); - if (!strings) - return false; - - // NSIndex index = [strings count]; - const CFIndex count = CFArrayGetCount(strings); - if (count < 1) { - CFRelease(strings); - return false; - } - - // NSString* string = [strings objectAtIndex: 0]; - const CFStringRef string = reinterpret_cast<CFStringRef>(CFArrayGetValueAtIndex(strings, 0)); - if (!string) { - CFRelease(strings); - return false; - } - - // result = [string UTF8String]; - result.resize(CFStringGetMaximumSizeForEncoding(CFStringGetLength(string), kCFStringEncodingUTF8) + 1); - if (!CFStringGetCString(string, &result.front(), result.length(), kCFStringEncodingUTF8)) { - CFRelease(strings); - return false; - } - result.resize(result.find_first_of('\0')); - - return true; -} - -} // namespace osx