# HG changeset patch # User Paper # Date 1715545867 14400 # Node ID 9a88e1725fd2d7ac59e2be74dd5fbbf804562d44 # Parent 9347e2eaf6e5ea97271bfb42da0cb8bd041fd484 *: refactor lots of stuff I forgot to put this into different commits, oops! anyway, it doesn't really matter *that* much since this is an unfinished hobby project anyway. once it starts getting stable commit history will be more important, but for now it's not that big of a deal diff -r 9347e2eaf6e5 -r 9a88e1725fd2 Makefile.am --- a/Makefile.am Wed May 08 17:32:28 2024 -0400 +++ b/Makefile.am Sun May 12 16:31:07 2024 -0400 @@ -20,7 +20,8 @@ minori_qtrc = \ $(top_srcdir)/rc/icons/icons.qrc \ - $(top_srcdir)/rc/animone.qrc + $(top_srcdir)/rc/animone.qrc \ + rc/locale/translations.qrc # various things we want to distribute @@ -182,6 +183,8 @@ minori_utf8proc_sources = \ dep/utf8proc/utf8proc.c +minori_moc_sources = $(minori_qtheaders:.h=_moc.cc) + minori_SOURCES = \ src/core/anime_db.cc \ src/core/anime.cc \ @@ -239,11 +242,8 @@ $(minori_locale_qm) \ $(minori_moc_sources) \ $(minori_utf8proc_sources) \ - rc/locale/translations.qrc \ rc/final_qrc.cc -minori_moc_sources = $(minori_qtheaders:.h=_moc.cc) - minori_includes = \ -I$(top_srcdir)/include \ -I$(top_srcdir)/dep/animone/include \ diff -r 9347e2eaf6e5 -r 9a88e1725fd2 configure.ac --- a/configure.ac Wed May 08 17:32:28 2024 -0400 +++ b/configure.ac Sun May 12 16:31:07 2024 -0400 @@ -23,49 +23,65 @@ dnl Qt? + + PKG_CHECK_MODULES([QT], [Qt5Core >= 5.7.0 Qt5Widgets Qt5Gui], [ QT_PATH="$(eval $PKG_CONFIG --variable=exec_prefix Qt5Core)" QT_HOST_PATH="$(eval $PKG_CONFIG --variable=host_bins Qt5Core)" - QT_VERSION="$(eval $PKG_CONFIG --modversion Qt5Gui)" + AC_PATH_PROGS(QT_MOC, [moc-qt5 moc], moc, ["${QT_HOST_PATH}" "${QT_PATH}/bin"]) AC_PATH_PROGS(QT_RCC, [rcc-qt5 rcc], rcc, ["${QT_HOST_PATH}" "${QT_PATH}/bin"]) AC_PATH_PROGS(QT_UIC, [uic-qt5 uic], uic, ["${QT_HOST_PATH}" "${QT_PATH}/bin"]) AC_PATH_PROGS(QT_LRELEASE, [lrelease-qt5 lrelease], lrelease, ["${QT_HOST_PATH}" "${QT_PATH}/bin"]) AC_PATH_PROGS(QT_LUPDATE, [lupdate-qt5 lupdate], lupdate, ["${QT_HOST_PATH}" "${QT_PATH}/bin"]) -], [ - AC_MSG_ERROR([${QT_PKG_ERRORS}.]) -]) -dnl On some platforms (see: Debian), Qt is built with -dnl `-reduce-relocations`, which requires applications -dnl to be built with position-independent code. -dnl -dnl Unfortunately there's no way to check for this -dnl without using qmake (bleugh), so we use this check -dnl to see if qglobal.h can be included without PIC. + dnl On some platforms (see: Debian), Qt is built with + dnl `-reduce-relocations`, which requires applications + dnl to be built with position-independent code. + dnl + dnl Unfortunately there's no way to check for this + dnl without using qmake (bleugh), so we use this check + dnl to see if qglobal.h can be included without PIC. -AC_MSG_CHECKING([for Qt requiring -fPIC]) + AC_MSG_CHECKING([for Qt requiring -fPIC]) -saved_CXXFLAGS="$CXXFLAGS" -CXXFLAGS="$CXXFLAGS $QT_CFLAGS" -AC_COMPILE_IFELSE( - [AC_LANG_PROGRAM([#include ], [])], - [ - AC_MSG_RESULT([no]) - CXXFLAGS="$saved_CXXFLAGS" - ], - [ - AC_MSG_RESULT([yes]) - CXXFLAGS="$saved_CXXFLAGS -fPIC -DPIC" - ] -) + saved_CXXFLAGS="$CXXFLAGS" + CXXFLAGS="$CXXFLAGS $QT_CFLAGS" + AC_COMPILE_IFELSE( + [AC_LANG_PROGRAM([#include ], [])], + [ + AC_MSG_RESULT([no]) + CXXFLAGS="$saved_CXXFLAGS" + ], + [ + AC_MSG_RESULT([yes]) + CXXFLAGS="$saved_CXXFLAGS -fPIC -DPIC" + ] + ) +], [ + PKG_CHECK_MODULES([QT], [Qt6Core Qt6Widgets Qt6Gui], [ + QT_BIN_DIRECTORY="$(eval $PKG_CONFIG --variable=bindir Qt6Core)" + QT_LIBEXEC_DIRECTORY="$(eval $PKG_CONFIG --variable=libexecdir Qt6Core)" + + AC_PATH_PROGS(QT_MOC, [moc], moc, ["${QT_LIBEXEC_DIRECTORY}"]) + AC_PATH_PROGS(QT_RCC, [rcc], rcc, ["${QT_LIBEXEC_DIRECTORY}"]) + AC_PATH_PROGS(QT_UIC, [uic], uic, ["${QT_LIBEXEC_DIRECTORY}"]) + AC_PATH_PROGS(QT_LRELEASE, [lrelease], lrelease, ["${QT_BIN_DIRECTORY}"]) + AC_PATH_PROGS(QT_LUPDATE, [lupdate], lupdate, ["${QT_BIN_DIRECTORY}"]) + + dnl in my testing Qt 6 seems to require PIC unconditionally... + CXXFLAGS="$CXXFLAGS -fPIC" + ], [ + AC_MSG_ERROR([${QT_PKG_ERRORS}.]) + ]) +]) dnl need this for moc AC_PROG_MKDIR_P AC_SUBST([MKDIR_P]) dnl libcurl? -LIBCURL_CHECK_CONFIG([yes], [7.7.2], [have_libcurl=yes], [have_libcurl=no]) +LIBCURL_CHECK_CONFIG([yes], [7.87.0], [have_libcurl=yes], [have_libcurl=no]) AS_IF([test "x$have_libcurl" = "xno"], [AC_MSG_ERROR([*** libcurl not found.])]) diff -r 9347e2eaf6e5 -r 9a88e1725fd2 include/core/http.h --- a/include/core/http.h Wed May 08 17:32:28 2024 -0400 +++ b/include/core/http.h Sun May 12 16:31:07 2024 -0400 @@ -3,55 +3,54 @@ #include #include + #include #include +#include namespace HTTP { -QByteArray Get(const std::string& url, const std::vector& headers = {}); -QByteArray Post(const std::string& url, const std::string& data, const std::vector& headers = {}); +enum class Type { + Get, + Post +}; -class GetThread : public QThread { +QByteArray Request(const std::string& url, const std::vector& headers = {}, const std::string& data = "", Type type = Type::Get); + +class RequestThread final : public QThread { Q_OBJECT public: - GetThread(const std::string& u, const std::vector& h = {}, QObject* parent = nullptr) - : QThread(parent) { - url = u; - headers = h; - } + RequestThread(Type type = Type::Get, QObject* parent = nullptr); + RequestThread(const std::string& url, const std::vector& headers = {}, + const std::string& data = "", Type type = Type::Get, QObject* parent = nullptr); + ~RequestThread(); + + void SetUrl(const std::string& url); + void SetHeaders(const std::vector& headers); + void SetData(const std::string& data); + void SetType(Type type); + + void Stop(); signals: void ReceivedData(const QByteArray& ba); protected: - void run() override { emit ReceivedData(Get(url, headers)); } - - std::string url; - std::vector headers; -}; - -class PostThread : public QThread { - Q_OBJECT + void run() override; + static size_t WriteCallback(void* contents, size_t size, size_t nmemb, void* userdata); -public: - PostThread(const std::string& u, const std::string& d, const std::vector& h = {}, - QObject* parent = nullptr) - : QThread(parent) { - url = u; - data = d; - headers = h; - } + std::string url_; + std::string data_; + std::vector headers_; + Type type_; -signals: - void ReceivedData(const QByteArray& ba); + /* these are passed to the write callback */ + QByteArray array_; + bool cancelled_ = false; -protected: - void run() override { emit ReceivedData(Post(url, data, headers)); } - - std::string url; - std::string data; - std::vector headers; + /* don't fuck this up */ + std::mutex callback_data_mutex_; }; } // namespace HTTP diff -r 9347e2eaf6e5 -r 9a88e1725fd2 include/core/session.h --- a/include/core/session.h Wed May 08 17:32:28 2024 -0400 +++ b/include/core/session.h Sun May 12 16:31:07 2024 -0400 @@ -7,21 +7,22 @@ #include "semver/semver.hpp" -class MainWindow; +#include struct Session { public: Session() { timer.start(); } /* we literally *cannot* be lying to the user by doing this */ void IncrementRequests() { requests++; }; - int GetRequests() { return requests; }; + unsigned int GetRequests() { return requests; }; int uptime() { return timer.elapsed(); } Config config; static constexpr semver::version version{PACKAGE_VERSION}; private: - unsigned int requests = 0; + /* IncrementRequests() gets called by different threads */ + std::atomic requests = 0; QElapsedTimer timer; }; diff -r 9347e2eaf6e5 -r 9a88e1725fd2 include/gui/pages/anime_list.h --- a/include/gui/pages/anime_list.h Wed May 08 17:32:28 2024 -0400 +++ b/include/gui/pages/anime_list.h Sun May 12 16:31:07 2024 -0400 @@ -79,7 +79,7 @@ Q_OBJECT public: - AnimeListPage(QWidget* parent); + AnimeListPage(QWidget* parent = nullptr); void Refresh(); protected: diff -r 9347e2eaf6e5 -r 9a88e1725fd2 include/gui/widgets/anime_info.h --- a/include/gui/widgets/anime_info.h Wed May 08 17:32:28 2024 -0400 +++ b/include/gui/widgets/anime_info.h Sun May 12 16:31:07 2024 -0400 @@ -2,12 +2,7 @@ #define MINORI_GUI_WIDGETS_ANIME_INFO_H_ #include - -namespace TextWidgets { -class OneLineSection; -class LabelledSection; -class SelectableSection; -} // namespace TextWidgets +#include "gui/widgets/text.h" namespace Anime { class Anime; @@ -22,9 +17,9 @@ void SetAnime(const Anime::Anime& anime); private: - std::shared_ptr _title = nullptr; - std::shared_ptr _details = nullptr; - std::shared_ptr _synopsis = nullptr; + TextWidgets::OneLineSection _title; + TextWidgets::LabelledSection _details; + TextWidgets::SelectableSection _synopsis; }; #endif // MINORI_GUI_WIDGETS_ANIME_INFO_H_ diff -r 9347e2eaf6e5 -r 9a88e1725fd2 include/gui/widgets/poster.h --- a/include/gui/widgets/poster.h Wed May 08 17:32:28 2024 -0400 +++ b/include/gui/widgets/poster.h Sun May 12 16:31:07 2024 -0400 @@ -4,6 +4,7 @@ #include #include "gui/widgets/clickable_label.h" +#include "core/http.h" class QWidget; namespace Anime { @@ -25,6 +26,8 @@ void RenderToLabel(); private: + HTTP::RequestThread get_thread_; + QImage img_; QString service_url_; ClickableLabel label_; diff -r 9347e2eaf6e5 -r 9a88e1725fd2 include/gui/widgets/text.h --- a/include/gui/widgets/text.h Wed May 08 17:32:28 2024 -0400 +++ b/include/gui/widgets/text.h Sun May 12 16:31:07 2024 -0400 @@ -21,16 +21,24 @@ void SetText(const QString& title); private: - QLabel* static_text_title; - QFrame* static_text_line; + QLabel static_text_title; + QFrame static_text_line; }; -class Paragraph : public QLabel { +class Paragraph : public QWidget { Q_OBJECT public: Paragraph(const QString& text, QWidget* parent = nullptr); void SetText(const QString& text); + QPlainTextEdit* GetLabel(); + +protected: + QSize minimumSizeHint() const; + QSize sizeHint() const; + +private: + QPlainTextEdit text_edit; }; class LabelledParagraph final : public QWidget { @@ -38,21 +46,27 @@ public: LabelledParagraph(const QString& label, const QString& data, QWidget* parent = nullptr); - Paragraph* GetLabels(); - Paragraph* GetParagraph(); + QLabel* GetLabels(); + QLabel* GetData(); + + /* synonymous with GetData(), kept for compatibility. don't use in new code!!! */ + QLabel* GetParagraph(); private: - Paragraph* labels; - Paragraph* paragraph; + QLabel labels_; + QLabel data_; }; -class Line : public Paragraph { +class Line : public QWidget { Q_OBJECT public: Line(QWidget* parent = nullptr); Line(const QString& text, QWidget* parent = nullptr); void SetText(const QString& text); + +protected: + QLineEdit line_edit_; }; class Title final : public Line { @@ -81,8 +95,9 @@ public: LabelledSection(const QString& title, const QString& label, const QString& data, QWidget* parent = nullptr); Header* GetHeader(); - Paragraph* GetLabels(); - Paragraph* GetParagraph(); + QLabel* GetLabels(); + QLabel* GetData(); + QLabel* GetParagraph(); private: Header* header; diff -r 9347e2eaf6e5 -r 9a88e1725fd2 include/gui/window.h --- a/include/gui/window.h Wed May 08 17:32:28 2024 -0400 +++ b/include/gui/window.h Sun May 12 16:31:07 2024 -0400 @@ -1,22 +1,31 @@ #ifndef MINORI_WINDOW_H_ #define MINORI_WINDOW_H_ + #include "core/config.h" -#include -#include +#include "gui/widgets/sidebar.h" -/* *could* be forward-declared, but this causes - any file that #includes this to have to #include - these as well due to unique_ptr */ -#include "gui/widgets/sidebar.h" +/* pages; these should really be in a namespace */ +#include "gui/pages/anime_list.h" +#include "gui/pages/history.h" +#include "gui/pages/now_playing.h" +#include "gui/pages/search.h" +#include "gui/pages/seasons.h" +#include "gui/pages/statistics.h" +#include "gui/pages/torrents.h" + +#include #include #include #include #include #include +#include + class QMenu; class AnimeListPage; +/* ... :) */ Q_DECLARE_METATYPE(std::vector); class MainWindowPlayingThread final : public QThread { @@ -80,6 +89,14 @@ QStackedWidget stack_; SideBar sidebar_; + AnimeListPage anime_list_page_; + HistoryPage history_page_; + NowPlayingPage now_playing_page_; + SearchPage search_page_; + SeasonsPage seasons_page_; + StatisticsPage statistics_page_; + TorrentsPage torrents_page_; + MainWindowPlayingThread playing_thread_; QTimer playing_thread_timer_; diff -r 9347e2eaf6e5 -r 9a88e1725fd2 src/core/http.cc --- a/src/core/http.cc Wed May 08 17:32:28 2024 -0400 +++ b/src/core/http.cc Sun May 12 16:31:07 2024 -0400 @@ -13,16 +13,18 @@ return size * nmemb; } -QByteArray Get(const std::string& url, const std::vector& headers) { +QByteArray Request(const std::string& url, const std::vector& headers, const std::string& data, Type type) { struct curl_slist* list = NULL; QByteArray userdata; CURL* curl = curl_easy_init(); if (curl) { - for (const std::string& h : headers) { + for (const std::string& h : headers) list = curl_slist_append(list, h.c_str()); - } + curl_easy_setopt(curl, CURLOPT_URL, url.c_str()); + if (type == Type::Post) + curl_easy_setopt(curl, CURLOPT_POSTFIELDS, data.c_str()); curl_easy_setopt(curl, CURLOPT_HTTPHEADER, list); curl_easy_setopt(curl, CURLOPT_WRITEDATA, &userdata); curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, &WriteCallback); @@ -38,30 +40,99 @@ return userdata; } -QByteArray Post(const std::string& url, const std::string& data, const std::vector& headers) { +/* this function is static */ +size_t RequestThread::WriteCallback(void* contents, size_t size, size_t nmemb, void* userdata) { + RequestThread* thread = reinterpret_cast(userdata); + + const std::lock_guard lock(thread->callback_data_mutex_); + + /* stop writing, then! */ + if (thread->cancelled_) + return CURL_WRITEFUNC_ERROR; + + /* else, continue on as normal */ + thread->array_.append(reinterpret_cast(contents), size * nmemb); + return size * nmemb; +} + +RequestThread::RequestThread(Type type, QObject* parent) : QThread(parent) { + SetType(type); +} + +RequestThread::RequestThread(const std::string& url, const std::vector& headers, + const std::string& data, Type type, QObject* parent) + : QThread(parent) { + SetUrl(url); + SetData(data); + SetHeaders(headers); + SetType(type); +} + +RequestThread::~RequestThread() { + /* block until the function can safely exit. + * + * this sucks. find out a better way to do this, which will probably + * be to put all of the threads in a pool */ + Stop(); + wait(); +} + +void RequestThread::SetUrl(const std::string& url) { + url_ = url; +} + +void RequestThread::SetHeaders(const std::vector& headers) { + headers_ = headers; +} + +void RequestThread::SetData(const std::string& data) { + data_ = data; +} + +void RequestThread::SetType(Type type) { + type_ = type; +} + +void RequestThread::run() { struct curl_slist* list = NULL; - QByteArray userdata; CURL* curl = curl_easy_init(); if (curl) { - for (const std::string& h : headers) { + curl_easy_setopt(curl, CURLOPT_URL, url_.c_str()); + + if (type_ == Type::Post) + curl_easy_setopt(curl, CURLOPT_POSTFIELDS, data_.c_str()); + + for (const std::string& h : headers_) list = curl_slist_append(list, h.c_str()); - } - curl_easy_setopt(curl, CURLOPT_URL, url.c_str()); - curl_easy_setopt(curl, CURLOPT_POSTFIELDS, data.c_str()); curl_easy_setopt(curl, CURLOPT_HTTPHEADER, list); - curl_easy_setopt(curl, CURLOPT_WRITEDATA, &userdata); - curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, &WriteCallback); + + curl_easy_setopt(curl, CURLOPT_WRITEDATA, this); + curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, &RequestThread::WriteCallback); + /* Use system certs... useful on Windows. */ curl_easy_setopt(curl, CURLOPT_SSL_OPTIONS, CURLSSLOPT_NATIVE_CA); - curl_easy_setopt(curl, CURLOPT_NOSIGNAL, 1); // threading + + /* does something with threading, don't remember what though */ + curl_easy_setopt(curl, CURLOPT_NOSIGNAL, 1); + CURLcode res = curl_easy_perform(curl); session.IncrementRequests(); curl_easy_cleanup(curl); - if (res != CURLE_OK) + + callback_data_mutex_.lock(); + if (res != CURLE_OK && !(res == CURLE_WRITE_ERROR && cancelled_)) std::cerr << "curl_easy_perform(curl) failed!: " << curl_easy_strerror(res) << std::endl; + callback_data_mutex_.unlock(); } - return userdata; + + emit ReceivedData(array_); + array_.clear(); +} + +void RequestThread::Stop() { + const std::lock_guard lock(callback_data_mutex_); + cancelled_ = true; } } // namespace HTTP diff -r 9347e2eaf6e5 -r 9a88e1725fd2 src/gui/dialog/information.cc --- a/src/gui/dialog/information.cc Wed May 08 17:32:28 2024 -0400 +++ b/src/gui/dialog/information.cc Sun May 12 16:31:07 2024 -0400 @@ -171,7 +171,7 @@ QComboBox* combo_box = new QComboBox(section); for (unsigned int i = 0; i < Anime::ListStatuses.size(); i++) - combo_box->addItem(Strings::ToQString(Translate::ToString(Anime::ListStatuses[i])), + combo_box->addItem(Strings::ToQString(Translate::ToLocalString(Anime::ListStatuses[i])), static_cast(Anime::ListStatuses[i])); connect(combo_box, QOverload::of(&QComboBox::currentIndexChanged), this, diff -r 9347e2eaf6e5 -r 9a88e1725fd2 src/gui/pages/anime_list.cc --- a/src/gui/pages/anime_list.cc Wed May 08 17:32:28 2024 -0400 +++ b/src/gui/pages/anime_list.cc Sun May 12 16:31:07 2024 -0400 @@ -320,6 +320,7 @@ dialog->show(); dialog->raise(); dialog->activateWindow(); + connect(dialog, &InformationDialog::finished, dialog, &InformationDialog::deleteLater); } }); menu->addSeparator(); @@ -331,6 +332,7 @@ dialog->show(); dialog->raise(); dialog->activateWindow(); + connect(dialog, &InformationDialog::finished, dialog, &InformationDialog::deleteLater); } }); menu->addAction(tr("Delete from list..."), [this, animes] { @@ -361,6 +363,7 @@ dialog->show(); dialog->raise(); dialog->activateWindow(); + connect(dialog, &InformationDialog::finished, dialog, &InformationDialog::deleteLater); } void AnimeListPage::RefreshList() { diff -r 9347e2eaf6e5 -r 9a88e1725fd2 src/gui/pages/seasons.cc --- a/src/gui/pages/seasons.cc Wed May 08 17:32:28 2024 -0400 +++ b/src/gui/pages/seasons.cc Sun May 12 16:31:07 2024 -0400 @@ -48,7 +48,7 @@ toolbar->setMovable(false); { - /* hard-coded this value */ + /* todo: clean this up... this sucks... */ static constexpr Date::Year last_year = 1960; auto create_year_menu = [this](QWidget* parent, QMenu* parent_menu, Date::Year year){ @@ -72,12 +72,12 @@ }; /* we'll be extinct by the time this code breaks, so I guess it's fine :) */ - const Date::Year year = static_cast(QDate::currentDate().year()); - const Date::Year year_before_collapse = GetClosestDecade(year) - 10; + const Date::Year current_year = static_cast(QDate::currentDate().year()); + const Date::Year year_before_collapse = GetClosestDecade(current_year) - 10; season_button = new QToolButton(toolbar); QMenu* full_season_menu = new QMenu(season_button); - for (Date::Year c = year; c >= year_before_collapse; c--) + for (Date::Year c = current_year; c >= year_before_collapse; c--) create_year_menu(season_button, full_season_menu, c); full_season_menu->addSeparator(); @@ -103,15 +103,15 @@ { /* links */ QMenu* menu = new QMenu(button); - menu->addAction("Airing status"); - menu->addAction("List status"); - menu->addAction("Type"); + menu->addAction(tr("Airing status")); + menu->addAction(tr("List status")); + menu->addAction(tr("Type")); button->setMenu(menu); } button->setToolButtonStyle(Qt::ToolButtonTextBesideIcon); button->setIcon(QIcon(":/icons/16x16/category.png")); - button->setText("Group by:"); + button->setText(tr("Group by:")); button->setPopupMode(QToolButton::InstantPopup); toolbar->addWidget(button); } @@ -122,17 +122,17 @@ { /* links */ QMenu* menu = new QMenu(button); - menu->addAction("Airing date"); - menu->addAction("Episodes"); - menu->addAction("Popularity"); - menu->addAction("Score"); - menu->addAction("Title"); + menu->addAction(tr("Airing date")); + menu->addAction(tr("Episodes")); + menu->addAction(tr("Popularity")); + menu->addAction(tr("Score")); + menu->addAction(tr("Title")); button->setMenu(menu); } button->setToolButtonStyle(Qt::ToolButtonTextBesideIcon); button->setIcon(QIcon(":/icons/16x16/sort-quantity-descending.png")); - button->setText("Sort by:"); + button->setText(tr("Sort by:")); button->setPopupMode(QToolButton::InstantPopup); toolbar->addWidget(button); } @@ -143,14 +143,14 @@ { /* links */ QMenu* menu = new QMenu(button); - menu->addAction("Images"); - menu->addAction("Details"); + menu->addAction(tr("Images")); + menu->addAction(tr("Details")); button->setMenu(menu); } button->setToolButtonStyle(Qt::ToolButtonTextBesideIcon); button->setIcon(QIcon(":/icons/16x16/ui-scroll-pane-detail.png")); - button->setText("View:"); + button->setText(tr("View:")); button->setPopupMode(QToolButton::InstantPopup); toolbar->addWidget(button); } diff -r 9347e2eaf6e5 -r 9a88e1725fd2 src/gui/pages/statistics.cc --- a/src/gui/pages/statistics.cc Wed May 08 17:32:28 2024 -0400 +++ b/src/gui/pages/statistics.cc Sun May 12 16:31:07 2024 -0400 @@ -141,7 +141,7 @@ 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); + _anime_list->GetData()->setText(string); _score_distribution_graph->Clear(); for (int i = 10; i <= 100; i += 10) @@ -152,5 +152,5 @@ ts << session.GetRequests(); /* Application */ // UiUtils::SetPlainTextEditData(application_data, QString::number(session.uptime() / 1000)); - _application->GetParagraph()->SetText(string); + _application->GetData()->setText(string); } diff -r 9347e2eaf6e5 -r 9a88e1725fd2 src/gui/pages/torrents.cc --- a/src/gui/pages/torrents.cc Wed May 08 17:32:28 2024 -0400 +++ b/src/gui/pages/torrents.cc Sun May 12 16:31:07 2024 -0400 @@ -65,9 +65,10 @@ const std::filesystem::path torrents_dir = Filesystem::GetTorrentsPath(); std::filesystem::create_directories(torrents_dir); - HTTP::GetThread* thread = new HTTP::GetThread(link, {}, this); + /* this sucks */ + HTTP::RequestThread* thread = new HTTP::RequestThread(link, {}, "", HTTP::Type::Get, this); - connect(thread, &HTTP::GetThread::ReceivedData, this, [this, torrents_dir, filename](const QByteArray& data) { + connect(thread, &HTTP::RequestThread::ReceivedData, this, [this, torrents_dir, filename](const QByteArray& data) { std::ofstream file(torrents_dir / filename, std::ofstream::out | std::ofstream::trunc); if (!file) return; // wat @@ -75,14 +76,14 @@ file.write(data.data(), data.size()); file.close(); }); - connect(thread, &HTTP::GetThread::finished, thread, &HTTP::GetThread::deleteLater); + connect(thread, &HTTP::RequestThread::finished, thread, &HTTP::RequestThread::deleteLater); thread->start(); } } QByteArray TorrentsPageListModel::DownloadTorrentList() { - return HTTP::Get(session.config.torrents.feed_link); + return HTTP::Request(session.config.torrents.feed_link); } void TorrentsPageListModel::ParseFeedDescription(const std::string& description, Torrent& torrent) { @@ -416,9 +417,9 @@ if (!model) return; - HTTP::GetThread* thread = new HTTP::GetThread(session.config.torrents.feed_link); + HTTP::RequestThread* thread = new HTTP::RequestThread(session.config.torrents.feed_link); - connect(thread, &HTTP::GetThread::ReceivedData, this, [&](const QByteArray& ba) { + connect(thread, &HTTP::RequestThread::ReceivedData, this, [&](const QByteArray& ba) { /* This is to make sure we aren't in a different thread * messing around with GUI stuff */ @@ -426,7 +427,7 @@ model->ParseTorrentList(ba); treeview->setUpdatesEnabled(true); }); - connect(thread, &QThread::finished, thread, &QThread::deleteLater); + connect(thread, &HTTP::RequestThread::finished, thread, &HTTP::RequestThread::deleteLater); thread->start(); } diff -r 9347e2eaf6e5 -r 9a88e1725fd2 src/gui/theme.cc --- a/src/gui/theme.cc Wed May 08 17:32:28 2024 -0400 +++ b/src/gui/theme.cc Sun May 12 16:31:07 2024 -0400 @@ -7,6 +7,7 @@ #include #include #include +#include #ifdef MACOSX # include "sys/osx/dark_theme.h" #elif defined(WIN32) diff -r 9347e2eaf6e5 -r 9a88e1725fd2 src/gui/widgets/anime_button.cc --- a/src/gui/widgets/anime_button.cc Wed May 08 17:32:28 2024 -0400 +++ b/src/gui/widgets/anime_button.cc Sun May 12 16:31:07 2024 -0400 @@ -91,10 +91,10 @@ { const QLocale& locale = session.config.locale.GetLocale(); - _info.GetParagraph()->SetText(locale.toString(anime.GetAirDate().GetAsQDate(), "dd MMM yyyy") + "\n" + - QString::number(anime.GetEpisodes()) + "\n" + - Strings::ToQString(Strings::Implode(anime.GetGenres(), ", ")) + "\n" + "...\n" + - QString::number(anime.GetAudienceScore()) + "%\n" + "..."); + _info.GetData()->setText(locale.toString(anime.GetAirDate().GetAsQDate(), "dd MMM yyyy") + "\n" + + QString::number(anime.GetEpisodes()) + "\n" + + Strings::ToQString(Strings::Implode(anime.GetGenres(), ", ")) + "\n" + "...\n" + + QString::number(anime.GetAudienceScore()) + "%\n" + "..."); } _synopsis.SetText(Strings::ToQString(anime.GetSynopsis())); diff -r 9347e2eaf6e5 -r 9a88e1725fd2 src/gui/widgets/anime_info.cc --- a/src/gui/widgets/anime_info.cc Wed May 08 17:32:28 2024 -0400 +++ b/src/gui/widgets/anime_info.cc Sun May 12 16:31:07 2024 -0400 @@ -6,21 +6,16 @@ #include #include -AnimeInfoWidget::AnimeInfoWidget(QWidget* parent) : QWidget(parent) { +AnimeInfoWidget::AnimeInfoWidget(QWidget* parent) + : QWidget(parent) + , _title(tr("Alternative titles"), "") + , _details(tr("Details"), tr("Type:\nEpisodes:\nStatus:\nSeason:\nGenres:\nScore:"), "") + , _synopsis(tr("Synopsis"), "") { QVBoxLayout* layout = new QVBoxLayout(this); - _title.reset(new TextWidgets::OneLineSection(tr("Alternative titles"), "", this)); - layout->addWidget(_title.get()); - - _details.reset(new TextWidgets::LabelledSection( - tr("Details"), tr("Type:\nEpisodes:\nStatus:\nSeason:\nGenres:\nScore:"), "", this)); - layout->addWidget(_details.get()); - - _synopsis.reset(new TextWidgets::SelectableSection(tr("Synopsis"), "", this)); - _synopsis->GetParagraph()->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Minimum); - layout->addWidget(_synopsis.get()); - - layout->addStretch(); + layout->addWidget(&_title); + layout->addWidget(&_details); + layout->addWidget(&_synopsis); } AnimeInfoWidget::AnimeInfoWidget(const Anime::Anime& anime, QWidget* parent) : AnimeInfoWidget(parent) { @@ -29,15 +24,15 @@ void AnimeInfoWidget::SetAnime(const Anime::Anime& anime) { /* alt titles */ - _title->GetLine()->SetText(Strings::ToQString(Strings::Implode(anime.GetTitleSynonyms(), ", "))); + _title.GetLine()->SetText(Strings::ToQString(Strings::Implode(anime.GetTitleSynonyms(), ", "))); /* details */ QString details_data; QTextStream details_data_s(&details_data); + /* we have to convert ALL of these strings to * QString because QTextStream sucks and assumes - * Latin1 (on Windows?) - */ + * Latin1 (on Windows?) */ const auto genres = anime.GetGenres(); details_data_s << Strings::ToQString(Translate::ToLocalString(anime.GetFormat())) << "\n" << anime.GetEpisodes() << "\n" @@ -46,9 +41,9 @@ << anime.GetAirDate().GetYear().value_or(2000) << "\n" << Strings::ToQString((genres.size() > 1) ? Strings::Implode(genres, ", ") : "-") << "\n" << anime.GetAudienceScore() << "%"; - _details->GetParagraph()->SetText(details_data); + _details.GetData()->setText(details_data); - _synopsis->GetParagraph()->SetText(Strings::ToQString(anime.GetSynopsis())); + _synopsis.GetParagraph()->SetText(Strings::ToQString(anime.GetSynopsis())); updateGeometry(); } diff -r 9347e2eaf6e5 -r 9a88e1725fd2 src/gui/widgets/poster.cc --- a/src/gui/widgets/poster.cc Wed May 08 17:32:28 2024 -0400 +++ b/src/gui/widgets/poster.cc Sun May 12 16:31:07 2024 -0400 @@ -27,6 +27,8 @@ label_.setAlignment(Qt::AlignCenter); layout->addWidget(&label_); + + connect(&get_thread_, &HTTP::RequestThread::ReceivedData, this, &Poster::ImageDownloadFinished); } Poster::Poster(const Anime::Anime& anime, QWidget* parent) : Poster(parent) { @@ -35,14 +37,12 @@ void Poster::SetAnime(const Anime::Anime& anime) { /* todo: only download on showEvent() */ - { - HTTP::GetThread* thread = new HTTP::GetThread(anime.GetPosterUrl(), {}, this); + if (get_thread_.isRunning()) + get_thread_.Stop(); + get_thread_.wait(); - connect(thread, &HTTP::GetThread::ReceivedData, this, &Poster::ImageDownloadFinished); - connect(thread, &HTTP::GetThread::finished, thread, &HTTP::GetThread::deleteLater); - - thread->start(); - } + get_thread_.SetUrl(anime.GetPosterUrl()); + get_thread_.start(); std::optional url = anime.GetServiceUrl(session.config.service); if (url) diff -r 9347e2eaf6e5 -r 9a88e1725fd2 src/gui/widgets/sidebar.cc --- a/src/gui/widgets/sidebar.cc Wed May 08 17:32:28 2024 -0400 +++ b/src/gui/widgets/sidebar.cc Sun May 12 16:31:07 2024 -0400 @@ -30,10 +30,6 @@ void SideBar::SetBackgroundColor(QColor color) { viewport()->setAutoFillBackground(color != Qt::transparent); - - QPalette pal(palette()); - pal.setColor(QPalette::Window, color); - setPalette(pal); } QListWidgetItem* SideBar::AddItem(QString name, QIcon icon) { diff -r 9347e2eaf6e5 -r 9a88e1725fd2 src/gui/widgets/text.cc --- a/src/gui/widgets/text.cc Wed May 08 17:32:28 2024 -0400 +++ b/src/gui/widgets/text.cc Sun May 12 16:31:07 2024 -0400 @@ -6,82 +6,130 @@ #include #include #include +#include +#include + +/* WARNING: GARBAGE CODE FOLLOWS + * + * This file is filled with spaghetti to make this + * stupid text render how I want it to. + * + * many cases of hacking with setSizePolicy() are seen + * all around this file. Edit it only if really + * necessary, please. +*/ namespace TextWidgets { -Header::Header(const QString& title, QWidget* parent) : QWidget(parent) { +Header::Header(const QString& title, QWidget* parent) + : QWidget(parent) + , static_text_title(title) { QVBoxLayout* layout = new QVBoxLayout(this); setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Maximum); - static_text_title = new QLabel(title, this); - static_text_title->setTextFormat(Qt::PlainText); + static_text_title.setTextFormat(Qt::PlainText); { - QFont font = static_text_title->font(); + QFont font = static_text_title.font(); font.setWeight(QFont::Bold); - static_text_title->setFont(font); + static_text_title.setFont(font); } - static_text_line = new QFrame(this); - static_text_line->setFrameShape(QFrame::HLine); - static_text_line->setFrameShadow(QFrame::Sunken); - static_text_line->setFixedHeight(2); + static_text_line.setFrameShape(QFrame::HLine); + static_text_line.setFrameShadow(QFrame::Sunken); + static_text_line.setFixedHeight(2); - layout->addWidget(static_text_title); - layout->addWidget(static_text_line); + layout->addWidget(&static_text_title); + layout->addWidget(&static_text_line); layout->setSpacing(0); layout->setContentsMargins(0, 0, 0, 0); } void Header::SetText(const QString& text) { - static_text_title->setText(text); + static_text_title.setText(text); updateGeometry(); } -/* for now, this is a QLabel with a couple of default settings. - * - * eventually I'll have to implement this as a QScrollArea, just in case - * some random text decides to overflow or something. - */ -Paragraph::Paragraph(const QString& text, QWidget* parent) : QLabel(text, parent) { - setTextInteractionFlags(Qt::TextBrowserInteraction); - setFrameShape(QFrame::NoFrame); - setCursor(Qt::IBeamCursor); /* emulate Taiga */ - setWordWrap(true); +Paragraph::Paragraph(const QString& text, QWidget* parent) : QWidget(parent) { + /* meh */ + setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Expanding); + + QVBoxLayout* layout = new QVBoxLayout(this); + layout->setSpacing(0); + layout->setContentsMargins(0, 0, 0, 0); + + text_edit.setTextInteractionFlags(Qt::TextBrowserInteraction); + text_edit.setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); + text_edit.setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff); + text_edit.setStyleSheet("background: transparent;"); + + text_edit.document()->setDocumentMargin(0); - setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Minimum); + SetText(text); + + layout->addWidget(&text_edit); +} + +void Paragraph::SetText(const QString& text) { + text_edit.document()->setPlainText(text); + + /* return the view to the start */ + QTextCursor cursor = text_edit.textCursor(); + cursor.setPosition(0); + text_edit.setTextCursor(cursor); +} + +QSize Paragraph::minimumSizeHint() const { + return QSize(0, 0); } -/* kept here for legacy reasons, see explanation above */ -void Paragraph::SetText(const QString& text) { - setText(text); +/* highly based upon... some stackoverflow answer for PyQt */ +QSize Paragraph::sizeHint() const { + QTextDocument* doc = text_edit.document(); + doc->adjustSize(); + + long h = 0; + for (QTextBlock line = doc->begin(); line != doc->end(); line = line.next()) + h += doc->documentLayout()->blockBoundingRect(line).height(); + + return QSize(doc->size().width(), h); +} + +QPlainTextEdit* Paragraph::GetLabel() { + return &text_edit; } -/* Equivalent to Paragraph(), but disables word wrap. */ -Line::Line(QWidget* parent) : Paragraph("", parent) { - setWordWrap(false); +Line::Line(QWidget* parent) : QWidget(parent) { + QVBoxLayout* layout = new QVBoxLayout(this); + layout->setSpacing(0); + layout->setContentsMargins(0, 0, 0, 0); + + line_edit_.setReadOnly(true); + line_edit_.setFrame(false); + line_edit_.setStyleSheet("background: transparent;"); + + layout->addWidget(&line_edit_); } -Line::Line(const QString& text, QWidget* parent) : Paragraph(text, parent) { - setWordWrap(false); +Line::Line(const QString& text, QWidget* parent) : Line(parent) { + SetText(text); } -/* legacy function, don't use in new code */ void Line::SetText(const QString& text) { - setText(text); + line_edit_.setText(text); + line_edit_.setCursorPosition(0); } Title::Title(const QString& title, QWidget* parent) : Line(title, parent) { - QFont fnt(font()); + QFont fnt(line_edit_.font()); fnt.setPixelSize(16); - setFont(fnt); + line_edit_.setFont(fnt); - QPalette pal(palette()); - pal.setColor(QPalette::Text, pal.color(QPalette::Highlight)); - setPalette(pal); + line_edit_.setForegroundRole(QPalette::Highlight); } -Section::Section(const QString& title, const QString& data, QWidget* parent) : QWidget(parent) { +Section::Section(const QString& title, const QString& data, QWidget* parent) + : QWidget(parent) { QVBoxLayout* layout = new QVBoxLayout(this); header = new Header(title, this); @@ -90,9 +138,9 @@ QHBoxLayout* content_layout = new QHBoxLayout(content); paragraph = new Paragraph(data, this); - paragraph->setTextInteractionFlags(Qt::NoTextInteraction); - paragraph->setAttribute(Qt::WidgetAttribute::WA_TransparentForMouseEvents); - paragraph->setWordWrap(QTextOption::NoWrap); + paragraph->GetLabel()->setTextInteractionFlags(Qt::NoTextInteraction); + paragraph->GetLabel()->setAttribute(Qt::WidgetAttribute::WA_TransparentForMouseEvents); + paragraph->GetLabel()->setWordWrapMode(QTextOption::NoWrap); content_layout->addWidget(paragraph); content_layout->setSpacing(0); @@ -113,33 +161,32 @@ return paragraph; } -LabelledParagraph::LabelledParagraph(const QString& label, const QString& data, QWidget* parent) : QWidget(parent) { +/* despite being named a "labelled paragraph" this uses QLabels for simplicity */ +LabelledParagraph::LabelledParagraph(const QString& label, const QString& data, QWidget* parent) + : QWidget(parent) + , labels_(label) + , data_(data) { QHBoxLayout* ly = new QHBoxLayout(this); - labels = new Paragraph(label, this); - labels->setTextInteractionFlags(Qt::NoTextInteraction); - labels->setAttribute(Qt::WidgetAttribute::WA_TransparentForMouseEvents); - labels->setWordWrap(QTextOption::NoWrap); - labels->setSizePolicy(QSizePolicy::Maximum, QSizePolicy::Expanding); + labels_.setSizePolicy(QSizePolicy::Maximum, QSizePolicy::Maximum); + data_.setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Maximum); - paragraph = new Paragraph(data, this); - paragraph->setTextInteractionFlags(Qt::NoTextInteraction); - paragraph->setAttribute(Qt::WidgetAttribute::WA_TransparentForMouseEvents); - paragraph->setWordWrap(QTextOption::NoWrap); - paragraph->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Expanding); - - ly->addWidget(labels, 0, Qt::AlignTop); - ly->addWidget(paragraph, 0, Qt::AlignTop); + ly->addWidget(&labels_, 0, Qt::AlignTop); + ly->addWidget(&data_, 0, Qt::AlignTop); ly->setSpacing(20); ly->setContentsMargins(0, 0, 0, 0); } -Paragraph* LabelledParagraph::GetLabels() { - return labels; +QLabel* LabelledParagraph::GetLabels() { + return &labels_; } -Paragraph* LabelledParagraph::GetParagraph() { - return paragraph; +QLabel* LabelledParagraph::GetData() { + return &data_; +} + +QLabel* LabelledParagraph::GetParagraph() { + return GetData(); } LabelledSection::LabelledSection(const QString& title, const QString& label, const QString& data, QWidget* parent) @@ -151,7 +198,7 @@ // this is not accessible from the object because there's really // no reason to make it accessible... content = new LabelledParagraph(label, data, this); - content->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Minimum); + content->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Maximum); content->setContentsMargins(12, 0, 0, 0); layout->addWidget(header); @@ -164,11 +211,15 @@ return header; } -Paragraph* LabelledSection::GetLabels() { +QLabel* LabelledSection::GetLabels() { return content->GetLabels(); } -Paragraph* LabelledSection::GetParagraph() { +QLabel* LabelledSection::GetData() { + return content->GetData(); +} + +QLabel* LabelledSection::GetParagraph() { return content->GetParagraph(); } diff -r 9347e2eaf6e5 -r 9a88e1725fd2 src/gui/window.cc --- a/src/gui/window.cc Wed May 08 17:32:28 2024 -0400 +++ b/src/gui/window.cc Sun May 12 16:31:07 2024 -0400 @@ -74,21 +74,32 @@ action_->setEnabled(true); } -MainWindow::MainWindow(QWidget* parent) : QMainWindow(parent), async_synchronize_thread_(nullptr, nullptr) { +MainWindow::MainWindow(QWidget* parent) + : QMainWindow(parent) + , async_synchronize_thread_(nullptr, nullptr) { setWindowIcon(QIcon(":/icons/favicon.png")); sidebar_.setFixedWidth(128); sidebar_.setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Expanding); - connect(&sidebar_, &SideBar::CurrentItemChanged, &stack_, &QStackedWidget::setCurrentIndex); - new QHBoxLayout(&main_widget_); + CreateBars(); + + stack_.addWidget(&now_playing_page_); + /* ---- */ + stack_.addWidget(&anime_list_page_); + stack_.addWidget(&history_page_); + stack_.addWidget(&statistics_page_); + /* ---- */ + stack_.addWidget(&search_page_); + stack_.addWidget(&seasons_page_); + stack_.addWidget(&torrents_page_); + AddMainWidgets(); + sidebar_.SetCurrentItem(static_cast(Pages::ANIME_LIST)); setCentralWidget(&main_widget_); - CreateBars(); - NowPlayingPage* page = reinterpret_cast(stack_.widget(static_cast(Pages::NOW_PLAYING))); connect(&playing_thread_, &MainWindowPlayingThread::Done, this, [page](const std::vector& files) { @@ -123,8 +134,9 @@ playing_thread_timer_.start(5000); } +/* Does the main part of what Qt's generic "RetranslateUI" function would do */ void MainWindow::AddMainWidgets() { - int page = static_cast(Pages::ANIME_LIST); + int page = sidebar_.GetCurrentItem(); sidebar_.clear(); @@ -138,20 +150,6 @@ sidebar_.AddItem(tr("Seasons"), SideBar::CreateIcon(":/icons/16x16/calendar.png")); sidebar_.AddItem(tr("Torrents"), SideBar::CreateIcon(":/icons/16x16/feed.png")); - while (stack_.count()) - stack_.removeWidget(stack_.widget(0)); - - /* can we allocate these on the stack? */ - stack_.addWidget(new NowPlayingPage(&main_widget_)); - /* ---- */ - stack_.addWidget(new AnimeListPage(&main_widget_)); - stack_.addWidget(new HistoryPage(&main_widget_)); - stack_.addWidget(new StatisticsPage(&main_widget_)); - /* ---- */ - stack_.addWidget(new SearchPage(&main_widget_)); - stack_.addWidget(new SeasonsPage(&main_widget_)); - stack_.addWidget(new TorrentsPage(&main_widget_)); - sidebar_.SetCurrentItem(page); main_widget_.layout()->addWidget(&sidebar_); @@ -334,6 +332,8 @@ } /* pain in the ass */ + disconnect(&sidebar_, &SideBar::CurrentItemChanged, nullptr, nullptr); + connect(&sidebar_, &SideBar::CurrentItemChanged, &stack_, &QStackedWidget::setCurrentIndex); connect(&sidebar_, &SideBar::CurrentItemChanged, this, [pages_group](int index) { QAction* checked = pages_group->checkedAction(); @@ -508,8 +508,7 @@ } void MainWindow::RetranslateUI() { - /* This kinda sucks but nobody's really going to be changing - the application language all the time :p */ + /* This sucks a LOT */ setUpdatesEnabled(false); AddMainWidgets(); CreateBars(); @@ -532,14 +531,18 @@ QMainWindow::showEvent(event); #ifdef WIN32 /* Technically this *should* be - session.config.theme.IsInDarkTheme() && win32::IsInDarkTheme() - but I prefer the title bar being black even when light mode - is enabled :/ */ + * session.config.theme.IsInDarkTheme() && win32::IsInDarkTheme() + * but I prefer the title bar being black even when light mode + * is enabled :/ */ win32::SetTitleBarsToBlack(session.config.theme.IsInDarkTheme()); #endif } void MainWindow::closeEvent(QCloseEvent* event) { + playing_thread_timer_.stop(); + playing_thread_.wait(); + async_synchronize_thread_.wait(); + session.config.Save(); Anime::db.SaveDatabaseToDisk(); event->accept(); diff -r 9347e2eaf6e5 -r 9a88e1725fd2 src/services/anilist.cc --- a/src/services/anilist.cc Wed May 08 17:32:28 2024 -0400 +++ b/src/services/anilist.cc Sun May 12 16:31:07 2024 -0400 @@ -18,6 +18,7 @@ #include #include +#include #include @@ -26,7 +27,7 @@ namespace Services { namespace AniList { -static constexpr int CLIENT_ID = 13706; +static constexpr std::string_view CLIENT_ID = "13706"; class Account { public: @@ -42,13 +43,13 @@ static Account account; -std::string SendRequest(std::string data) { +static std::string SendRequest(const std::string& data) { std::vector headers = {"Authorization: Bearer " + account.AuthToken(), "Accept: application/json", "Content-Type: application/json"}; - return Strings::ToUtf8String(HTTP::Post("https://graphql.anilist.co", data, headers)); + return Strings::ToUtf8String(HTTP::Request("https://graphql.anilist.co", headers, data, HTTP::Type::Post)); } -nlohmann::json SendJSONRequest(nlohmann::json data) { +static nlohmann::json SendJSONRequest(const nlohmann::json& data) { std::string request = SendRequest(data.dump()); if (request.empty()) { std::cerr << "[AniList] JSON Request returned an empty result!" << std::endl; @@ -72,7 +73,7 @@ return ret; } -void ParseListStatus(std::string status, Anime::Anime& anime) { +static void ParseListStatus(std::string status, Anime::Anime& anime) { static const std::unordered_map map = { {"CURRENT", Anime::ListStatus::Current }, {"PLANNING", Anime::ListStatus::Planning }, @@ -95,7 +96,7 @@ anime.SetUserStatus(map.at(status)); } -std::string ListStatusToString(const Anime::Anime& anime) { +static std::string ListStatusToString(const Anime::Anime& anime) { if (anime.GetUserIsRewatching() && anime.GetUserStatus() == Anime::ListStatus::Current) return "REWATCHING"; @@ -109,7 +110,7 @@ return "CURRENT"; } -void ParseTitle(const nlohmann::json& json, Anime::Anime& anime) { +static void ParseTitle(const nlohmann::json& json, Anime::Anime& anime) { nlohmann::json::json_pointer g = "/native"_json_pointer; if (json.contains(g) && json[g].is_string()) anime.SetTitle(Anime::TitleLanguage::Native, json[g]); @@ -123,7 +124,7 @@ anime.SetTitle(Anime::TitleLanguage::Romaji, json[g]); } -int ParseMediaJson(const nlohmann::json& json) { +static int ParseMediaJson(const nlohmann::json& json) { int id = JSON::GetNumber(json, "/id"_json_pointer); if (!id) return 0; @@ -159,7 +160,7 @@ return id; } -int ParseListItem(const nlohmann::json& json) { +static int ParseListItem(const nlohmann::json& json) { int id = ParseMediaJson(json["media"]); Anime::Anime& anime = Anime::db.items[id]; @@ -179,7 +180,7 @@ return id; } -int ParseList(const nlohmann::json& json) { +static int ParseList(const nlohmann::json& json) { for (const auto& entry : json["entries"].items()) { ParseListItem(entry.value()); } @@ -336,6 +337,10 @@ if (!anime.IsInUserList()) return 0; + std::optional service_id = anime.GetServiceId(Anime::Service::AniList); + if (!service_id) + 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" @@ -348,7 +353,7 @@ nlohmann::json json = { {"query", query}, {"variables", { - {"media_id", anime.GetId()}, + {"media_id", Strings::ToInt(service_id.value())}, {"progress", anime.GetUserProgress()}, {"status", ListStatusToString(anime)}, {"score", anime.GetUserScore()}, @@ -365,7 +370,7 @@ return JSON::GetNumber(ret, "/data/SaveMediaListEntry/id"_json_pointer, 0); } -int ParseUser(const nlohmann::json& json) { +static int ParseUser(const nlohmann::json& json) { account.SetUserId(JSON::GetNumber(json, "/id"_json_pointer, 0)); return account.UserId(); } @@ -373,7 +378,7 @@ bool AuthorizeUser() { /* Prompt for PIN */ QDesktopServices::openUrl(QUrl(Strings::ToQString("https://anilist.co/api/v2/oauth/authorize?client_id=" + - Strings::ToUtf8String(CLIENT_ID) + "&response_type=token"))); + std::string(CLIENT_ID) + "&response_type=token"))); bool ok; QString token = QInputDialog::getText(