diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000000000000000000000000000000000000..69e668d4f250ecf7ae2ecb5df2d22cd48f184776 --- /dev/null +++ b/.gitignore @@ -0,0 +1,2 @@ +*.pro.user +pkg/build/ diff --git a/Makefile b/Makefile deleted file mode 100644 index 456639edac6999bf5b16259ad1bd8d610880e024..0000000000000000000000000000000000000000 --- a/Makefile +++ /dev/null @@ -1,38 +0,0 @@ -SHELL = /bin/sh - -TARGET = xdgurl -srcdir = ./src - -DESTDIR = -prefix = /usr/local -exec_prefix = $(prefix) -bindir = $(exec_prefix)/bin -datadir = $(prefix)/share - -INSTALL = install -INSTALL_PROGRAM = $(INSTALL) -D -m 755 -INSTALL_DATA = $(INSTALL) -D -m 644 -RM = rm -f - -.PHONY: all rebuild build clean install uninstall - -all: rebuild ; - -rebuild: clean build ; - -build: $(TARGET) ; - -clean: - $(RM) ./$(TARGET) - -install: build - $(INSTALL_PROGRAM) ./$(TARGET) $(DESTDIR)$(bindir)/$(TARGET) - $(INSTALL_DATA) $(srcdir)/$(TARGET).desktop $(DESTDIR)$(datadir)/applications/$(TARGET).desktop - -uninstall: - $(RM) $(DESTDIR)$(bindir)/$(TARGET) - $(RM) $(DESTDIR)$(datadir)/applications/$(TARGET).desktop - -$(TARGET): - # Just copy for now - install -m 755 $(srcdir)/$(TARGET).py ./$(TARGET) diff --git a/deployment.pri b/deployment.pri new file mode 100644 index 0000000000000000000000000000000000000000..30f39b844554120225fee3e8c25b9c9c2d8ef43d --- /dev/null +++ b/deployment.pri @@ -0,0 +1,21 @@ +unix:!android { + isEmpty(PREFIX) { + PREFIX = /usr/local + } + + SRCDIR = src + BINDIR = $${PREFIX}/bin + DATADIR = $${PREFIX}/share + + target.path = $${BINDIR} + + desktop.files = $${SRCDIR}/desktop/$${TARGET}.desktop + desktop.path = $${DATADIR}/applications + + icon.files = $${SRCDIR}/desktop/$${TARGET}.svg + icon.path = $${DATADIR}/icons/hicolor/scalable/apps + + INSTALLS += target desktop icon +} + +export(INSTALLS) diff --git a/pkg/arch/PKGBUILD b/pkg/arch/PKGBUILD index 64cd7ff4467d045026c3930c9b661dc6a3d8b806..f84a7ef01d22ce33a114d52fdc807e07b51c42a6 100644 --- a/pkg/arch/PKGBUILD +++ b/pkg/arch/PKGBUILD @@ -1,13 +1,13 @@ # Maintainer: Akira Ohgaki <akiraohgaki@gmail.com> pkgname=xdgurl -pkgver=1.0.1 +pkgver=2.0.0 pkgrel=1 pkgdesc="An install helper program for desktop stuff." arch=('i686' 'x86_64') url="https://github.com/xdgurl/xdgurl" license=('GPL3') -depends=('tk') +depends=('qt5-base>=5.3.0' 'qt5-svg>=5.3.0' 'qt5-declarative>=5.3.0' 'qt5-quickcontrols>=5.3.0') #source=("https://github.com/xdgurl/xdgurl/archive/release-$pkgver.tar.gz") source=("$pkgname.tar.gz") md5sums=() #autofill using updpkgsums @@ -15,11 +15,12 @@ md5sums=() #autofill using updpkgsums build() { #cd "$pkgname-release-$pkgver" cd $pkgname + qmake PREFIX="/usr" make } package() { #cd "$pkgname-release-$pkgver" cd $pkgname - make DESTDIR="$pkgdir" prefix="/usr" install + make INSTALL_ROOT="$pkgdir" install } diff --git a/pkg/build.sh b/pkg/build.sh index 6e6782b838c6da56fc7f0042c3df495ac4f2d32c..2e96ccf75f4e901111d8610388e331b341507d19 100644 --- a/pkg/build.sh +++ b/pkg/build.sh @@ -1,31 +1,56 @@ #!/bin/sh -cd `dirname $0` +TMPDIR=/tmp +PKGDIR=`cd $(dirname $0) && pwd` +BUILDDIR=$PKGDIR/build +PROJDIR=`dirname $PKGDIR` +PROJNAME=`basename $PROJDIR` +PARENTDIR=`dirname $PROJDIR` build_ubuntu() { - mkdir ./build - cp -r ../src ./build/ - cp ../Makefile ./build/ - cp -r ./ubuntu/debian ./build/ - cd ./build + #sudo apt install build-essential qt5-default libqt5svg5-dev qtdeclarative5-dev devscripts debhelper fakeroot + + cd $PARENTDIR + tar -czvf $TMPDIR/$PROJNAME.tar.gz --exclude ".git" $PROJNAME + + mkdir $BUILDDIR + + mv $TMPDIR/$PROJNAME.tar.gz $BUILDDIR + tar -xzvf $BUILDDIR/$PROJNAME.tar.gz -C $BUILDDIR + cp -r $PKGDIR/ubuntu/debian $BUILDDIR/$PROJNAME + + cd $BUILDDIR/$PROJNAME debuild -uc -us -b } build_fedora() { - tar -czvf /tmp/xdgurl.tar.gz ../../xdgurl - mkdir -p ./build/SOURCES - mkdir -p ./build/SPECS - mv /tmp/xdgurl.tar.gz ./build/SOURCES/ - cp ./fedora/xdgurl.spec ./build/SPECS/ - rpmbuild --define '_topdir '`pwd`'/build' -bb ./build/SPECS/xdgurl.spec + #sudo dnf install make automake gcc gcc-c++ libtool qt5-qtbase-devel qt5-qtsvg-devel qt5-qtdeclarative-devel rpm-build + + cd $PARENTDIR + tar -czvf $TMPDIR/$PROJNAME.tar.gz --exclude ".git" $PROJNAME + + mkdir $BUILDDIR + mkdir $BUILDDIR/SOURCES + mkdir $BUILDDIR/SPECS + + mv $TMPDIR/$PROJNAME.tar.gz $BUILDDIR/SOURCES + cp $PKGDIR/fedora/xdgurl.spec $BUILDDIR/SPECS + + rpmbuild --define "_topdir $BUILDDIR" -bb $BUILDDIR/SPECS/xdgurl.spec } build_arch() { - tar -czvf /tmp/xdgurl.tar.gz ../../xdgurl - mkdir ./build - mv /tmp/xdgurl.tar.gz ./build/ - cp ./arch/PKGBUILD ./build/ - cd ./build + #sudo pacman -S base-devel qt5-base qt5-svg qt5-declarative qt5-quickcontrols + + cd $PARENTDIR + tar -czvf $TMPDIR/$PROJNAME.tar.gz --exclude ".git" $PROJNAME + + mkdir $BUILDDIR + + mv $TMPDIR/$PROJNAME.tar.gz $BUILDDIR + cp $PKGDIR/arch/PKGBUILD $BUILDDIR + + cd $BUILDDIR updpkgsums makepkg -s } diff --git a/pkg/fedora/xdgurl.spec b/pkg/fedora/xdgurl.spec index e24f53c9532b343f13d4ef83a9c3a7c0a18300de..d161a343a5c0a7734285ef708afb7787ac3f3717 100644 --- a/pkg/fedora/xdgurl.spec +++ b/pkg/fedora/xdgurl.spec @@ -1,6 +1,6 @@ Summary: An install helper program for desktop stuff Name: xdgurl -Version: 1.0.1 +Version: 2.0.0 Release: 1%{?dist} License: GPLv3+ Group: Applications/Internet @@ -9,7 +9,8 @@ URL: https://github.com/xdgurl/xdgurl #Source0: https://github.com/xdgurl/xdgurl/archive/release-%{version}.tar.gz Source0: %{name}.tar.gz -Requires: tkinter, python3-tkinter +Requires: qt5-qtbase >= 5.3.0, qt5-qtbase-gui >= 5.3.0, qt5-qtsvg >= 5.3.0, qt5-qtdeclarative >= 5.3.0, qt5-qtquickcontrols >= 5.3.0 +BuildRequires: make, automake, gcc, gcc-c++, libtool, qt5-qtbase-devel, qt5-qtsvg-devel, qt5-qtdeclarative-devel, rpm-build %description An install helper program for desktop stuff. @@ -20,20 +21,27 @@ An install helper program for desktop stuff. %build %define debug_package %{nil} +qmake-qt5 PREFIX="/usr" make %install -make DESTDIR="%{buildroot}" prefix="/usr" install +make INSTALL_ROOT="%{buildroot}" install %files %defattr(-,root,root) %{_bindir}/%{name} %{_datadir}/applications/%{name}.desktop +%{_datadir}/icons/hicolor/scalable/apps/%{name}.svg %clean rm -rf %{buildroot} %changelog +* Fri Oct 28 2016 Akira Ohgaki <akiraohgaki@gmail.com> - 2.0.0-1 +- Re-implemented xdgurl as Qt program +- Download progress bar +- Add install-type "bin" + * Fri Jul 15 2016 Akira Ohgaki <akiraohgaki@gmail.com> - 1.0.1-1 - Clean successfull message - Return exit code diff --git a/pkg/ubuntu/debian/changelog b/pkg/ubuntu/debian/changelog index 1e481316d798260a86f4393875fb5611a55eb8b8..a8ef66f2fbc93c46ee3b0ccd922fe7a02a28a6dd 100644 --- a/pkg/ubuntu/debian/changelog +++ b/pkg/ubuntu/debian/changelog @@ -1,3 +1,11 @@ +xdgurl (2.0.0-0ubuntu1) xenial; urgency=low + + * Re-implemented xdgurl as Qt program + * Download progress bar + * Add install-type "bin" + + -- Akira Ohgaki <akiraohgaki@gmail.com> Fri, 28 Oct 2016 08:53:57 +0000 + xdgurl (1.0.1-0ubuntu1) xenial; urgency=low * Clean successfull message diff --git a/pkg/ubuntu/debian/control b/pkg/ubuntu/debian/control index 68a1d227a8200f93b03a4cdabd1a5422b5dc9bf1..6a8eb36d11c3f42d3950924b14ed9cf278cf20a8 100644 --- a/pkg/ubuntu/debian/control +++ b/pkg/ubuntu/debian/control @@ -2,11 +2,11 @@ Source: xdgurl Section: web Priority: optional Maintainer: Akira Ohgaki <akiraohgaki@gmail.com> -Build-Depends: build-essential (>= 11), devscripts (>= 2.14), debhelper (>= 9), fakeroot (>= 1.20) +Build-Depends: build-essential (>= 11), qt5-default (>= 5.3.0), libqt5svg5-dev (>= 5.3.0), qtdeclarative5-dev (>= 5.3.0), devscripts (>= 2.14), debhelper (>= 9), fakeroot (>= 1.20) Standards-Version: 3.9.4 Package: xdgurl Architecture: any -Depends: ${shlibs:Depends}, ${misc:Depends}, python-tk (>= 2.7.0), python3-tk (>= 3.2.0) -Description: xdgurl +Depends: ${shlibs:Depends}, ${misc:Depends}, libqt5svg5 (>= 5.3.0), qtdeclarative5-qtquick2-plugin (>= 5.3.0), qtdeclarative5-window-plugin (>= 5.3.0), qtdeclarative5-controls-plugin (>= 5.3.0), qtdeclarative5-dialogs-plugin (>= 5.3.0) +Description: An install helper program for desktop stuff An install helper program for desktop stuff. diff --git a/pkg/ubuntu/debian/rules b/pkg/ubuntu/debian/rules index 999ca813027742773a158ff28651d583379299ac..3fc604ca255505614239a3ba2f7f82d0071c3592 100755 --- a/pkg/ubuntu/debian/rules +++ b/pkg/ubuntu/debian/rules @@ -3,8 +3,11 @@ %: dh $@ +override_dh_auto_configure: + qmake PREFIX="/usr" + override_dh_auto_install: - make DESTDIR="$(CURDIR)/debian/tmp" prefix="/usr" install + make INSTALL_ROOT="$(CURDIR)/debian/tmp" install override_dh_shlibdeps: # ignore diff --git a/pkg/ubuntu/debian/xdgurl.install b/pkg/ubuntu/debian/xdgurl.install index dcca62ee52466ef8bd412092cf039b29539b6a9a..44123c67f5636dc6d774bd04461f377155ea4872 100644 --- a/pkg/ubuntu/debian/xdgurl.install +++ b/pkg/ubuntu/debian/xdgurl.install @@ -1,2 +1,3 @@ usr/bin/xdgurl usr/share/applications/xdgurl.desktop +usr/share/icons/hicolor/scalable/apps/xdgurl.svg diff --git a/rpath.pri b/rpath.pri new file mode 100644 index 0000000000000000000000000000000000000000..b3cd65e83d8e1df1c1d7bc94254dc152a2a91226 --- /dev/null +++ b/rpath.pri @@ -0,0 +1,5 @@ +unix:!android { + QMAKE_LFLAGS += -Wl,-rpath=\\\$\$ORIGIN + QMAKE_LFLAGS += -Wl,-rpath=/usr/lib/$${TARGET} + QMAKE_RPATH = +} diff --git a/src/app/configs/application.json b/src/app/configs/application.json new file mode 100644 index 0000000000000000000000000000000000000000..415e29a2189bdad9330f60f85a291b9b1b9775bf --- /dev/null +++ b/src/app/configs/application.json @@ -0,0 +1,13 @@ +{ + "id": "xdgurl", + "name": "xdgurl", + "version": "2.0.0", + "organization": "xdgurl", + "domain": "com.xdgurl.xdgurl", + "icon": ":/desktop/xdgurl.svg", + "description": "An install helper program for desktop stuff.", + "license": "GPL-3+", + "author": "Akira Ohgaki", + "contact": "akiraohgaki@gmail.com", + "homepage": "https://github.com/xdgurl/xdgurl" +} diff --git a/src/app/configs/configs.qrc b/src/app/configs/configs.qrc new file mode 100644 index 0000000000000000000000000000000000000000..a27f28af5718bcfb48d18dd500717897c9b6e84d --- /dev/null +++ b/src/app/configs/configs.qrc @@ -0,0 +1,7 @@ +<RCC> + <qresource prefix="/configs"> + <file>application.json</file> + <file>destinations.json</file> + <file>destinations_alias.json</file> + </qresource> +</RCC> diff --git a/src/app/configs/destinations.json b/src/app/configs/destinations.json new file mode 100644 index 0000000000000000000000000000000000000000..1ea1bbf332a39ffa2f62bc8633089035ee5755ea --- /dev/null +++ b/src/app/configs/destinations.json @@ -0,0 +1,38 @@ +{ + "bin": "$HOME/.bin", + "downloads": "$HOME/Downloads", + "documents": "$HOME/Documents", + "pictures": "$HOME/Pictures", + "music": "$HOME/Music", + "videos": "$HOME/Videos", + "wallpapers": "$XDG_DATA_HOME/wallpapers", + "fonts": "$HOME/.fonts", + "cursors": "$HOME/.icons", + "icons": "$XDG_DATA_HOME/icons", + "emoticons": "$XDG_DATA_HOME/emoticons", + "themes": "$HOME/.themes", + "emerald_themes": "$HOME/.emerald/themes", + "enlightenment_themes": "$HOME/.e/e/themes", + "enlightenment_backgrounds": "$HOME/.e/e/backgrounds", + "fluxbox_styles": "$HOME/.fluxbox/styles", + "pekwm_themes": "$HOME/.pekwm/themes", + "icewm_themes": "$HOME/.icewm/themes", + "plasma_plasmoids": "$XDG_DATA_HOME/plasma/plasmoids", + "plasma_look_and_feel": "$XDG_DATA_HOME/plasma/look-and-feel", + "plasma_desktopthemes": "$XDG_DATA_HOME/plasma/desktoptheme", + "kwin_effects": "$XDG_DATA_HOME/kwin/effects", + "kwin_scripts": "$XDG_DATA_HOME/kwin/scripts", + "kwin_tabbox": "$XDG_DATA_HOME/kwin/tabbox", + "aurorae_themes": "$XDG_DATA_HOME/aurorae/themes", + "dekorator_themes": "$XDG_DATA_HOME/deKorator/themes", + "qtcurve": "$XDG_DATA_HOME/QtCurve", + "color_schemes": "$XDG_DATA_HOME/color-schemes", + "gnome_shell_extensions": "$XDG_DATA_HOME/gnome-shell/extensions", + "cinnamon_applets": "$XDG_DATA_HOME/cinnamon/applets", + "cinnamon_desklets": "$XDG_DATA_HOME/cinnamon/desklets", + "cinnamon_extensions": "$XDG_DATA_HOME/cinnamon/extensions", + "nautilus_scripts": "$XDG_DATA_HOME/nautilus/scripts", + "amarok_scripts": "$KDEHOME/share/apps/amarok/scripts", + "yakuake_skins": "$KDEHOME/share/apps/yakuake/skins", + "cairo_clock_themes": "$HOME/.cairo-clock/themes" +} diff --git a/src/app/configs/destinations_alias.json b/src/app/configs/destinations_alias.json new file mode 100644 index 0000000000000000000000000000000000000000..438fb4dc92cf31716b7741064141b8a10a83f20d --- /dev/null +++ b/src/app/configs/destinations_alias.json @@ -0,0 +1,17 @@ +{ + "gnome_shell_themes": "themes", + "cinnamon_themes": "themes", + "gtk2_themes": "themes", + "gtk3_themes": "themes", + "metacity_themes": "themes", + "xfwm4_themes": "themes", + "openbox_themes": "themes", + "kvantum_themes": "themes", + "compiz_themes": "emerald_themes", + "beryl_themes": "emerald_themes", + "plasma4_plasmoids": "plasma_plasmoids", + "plasma5_plasmoids": "plasma_plasmoids", + "plasma5_look_and_feel": "plasma_look_and_feel", + "plasma5_desktopthemes": "plasma_desktopthemes", + "plasma_color_schemes": "color_schemes" +} diff --git a/src/app/handlers/xdgurl.cpp b/src/app/handlers/xdgurl.cpp new file mode 100644 index 0000000000000000000000000000000000000000..487a83ba353471e8f9b5b12a856748d3e4f5396e --- /dev/null +++ b/src/app/handlers/xdgurl.cpp @@ -0,0 +1,286 @@ +#include "xdgurl.h" + +#include <QUrl> +#include <QUrlQuery> +#include <QTemporaryFile> +#include <QNetworkReply> +#include <QDesktopServices> + +#include "../../libs/utils/config.h" +#include "../../libs/utils/network.h" +#include "../../libs/utils/file.h" +#include "../../libs/utils/package.h" + +namespace handlers { + +XdgUrl::XdgUrl(const QString &xdgUrl, utils::Config *config, utils::Network *network, QObject *parent) : + QObject(parent), xdgUrl_(xdgUrl), config_(config), network_(network) +{ + parse(); + loadDestinations(); + + connect(network_, &utils::Network::finished, this, &handlers::XdgUrl::downloaded); + connect(network_, &utils::Network::downloadProgress, this, &handlers::XdgUrl::downloadProgress); +} + +void XdgUrl::process() +{ + /** + * xdgs scheme is a reserved name, so the process of xdgs + * is the same process of the xdg scheme currently. + */ + + if (!isValid()) { + QJsonObject result; + result["status"] = QString("error_validation"); + result["message"] = QString("Invalid XDG-URL " + xdgUrl_); + emit error(result); + return; + } + + network_->get(QUrl(metadata_["url"].toString())); + emit started(); +} + +bool XdgUrl::isValid() +{ + QString scheme = metadata_["scheme"].toString(); + QString command = metadata_["command"].toString(); + QString url = metadata_["url"].toString(); + QString type = metadata_["type"].toString(); + QString filename = metadata_["filename"].toString(); + + if ((scheme == "xdg" || scheme == "xdgs") + && (command == "download" || command == "install") + && QUrl(url).isValid() + && destinations_.contains(type) + && !filename.isEmpty()) { + return true; + } + + return false; +} + +void XdgUrl::openDestination() +{ + if (!destination_.isEmpty()) { + QDesktopServices::openUrl(QUrl("file://" + destination_)); + } +} + +QString XdgUrl::xdgUrl() const +{ + return xdgUrl_; +} + +QJsonObject XdgUrl::metadata() const +{ + return metadata_; +} + +void XdgUrl::downloaded(QNetworkReply *reply) +{ + if (reply->error() != QNetworkReply::NoError) { + QJsonObject result; + result["status"] = QString("error_network"); + result["message"] = reply->errorString(); + emit error(result); + return; + } + else if (reply->hasRawHeader("Location")) { + QString redirectUrl = QString(reply->rawHeader("Location")); + if (redirectUrl.startsWith("/")) { + redirectUrl = reply->url().authority() + redirectUrl; + } + network_->get(QUrl(redirectUrl)); + return; + } + else if (reply->hasRawHeader("Refresh")) { + QString refreshUrl = QString(reply->rawHeader("Refresh")).split("url=").last(); + if (refreshUrl.startsWith("/")) { + refreshUrl = reply->url().authority() + refreshUrl; + } + network_->get(QUrl(refreshUrl)); + return; + } + + if (metadata_["command"].toString() == "download") { + saveDownloadedFile(reply); + } + else if (metadata_["command"].toString() == "install") { + installDownloadedFile(reply); + } +} + +void XdgUrl::parse() +{ + QUrl url(xdgUrl_); + QUrlQuery query(url); + + metadata_["scheme"] = QString("xdg"); + metadata_["command"] = QString("download"); + metadata_["url"] = QString(""); + metadata_["type"] = QString("downloads"); + metadata_["filename"] = QString(""); + + if (!url.scheme().isEmpty()) { + metadata_["scheme"] = url.scheme(); + } + + if (!url.host().isEmpty()) { + metadata_["command"] = url.host(); + } + + if (query.hasQueryItem("url") && !query.queryItemValue("url").isEmpty()) { + metadata_["url"] = query.queryItemValue("url", QUrl::FullyDecoded); + } + + if (query.hasQueryItem("type") && !query.queryItemValue("type").isEmpty()) { + metadata_["type"] = query.queryItemValue("type", QUrl::FullyDecoded); + } + + if (query.hasQueryItem("filename") && !query.queryItemValue("filename").isEmpty()) { + metadata_["filename"] = QUrl(query.queryItemValue("filename", QUrl::FullyDecoded)).fileName(); + } + + if (!metadata_["url"].toString().isEmpty() && metadata_["filename"].toString().isEmpty()) { + metadata_["filename"] = QUrl(metadata_["url"].toString()).fileName(); + } +} + +void XdgUrl::loadDestinations() +{ + QJsonObject configDestinations = config_->get("destinations"); + QJsonObject configDestinationsAlias = config_->get("destinations_alias"); + + foreach (const QString key, configDestinations.keys()) { + destinations_[key] = convertPathString(configDestinations[key].toString()); + } + + foreach (const QString key, configDestinationsAlias.keys()) { + QString value = configDestinationsAlias[key].toString(); + if (destinations_.contains(value)) { + destinations_[key] = destinations_.value(value); + } + } +} + +QString XdgUrl::convertPathString(const QString &path) +{ + QString newPath = path; + + if (newPath.contains("$HOME")) { + newPath.replace("$HOME", utils::File::homePath()); + } + else if (newPath.contains("$XDG_DATA_HOME")) { + newPath.replace("$XDG_DATA_HOME", utils::File::genericDataPath()); + } + else if (newPath.contains("$KDEHOME")) { + newPath.replace("$KDEHOME", utils::File::kdehomePath()); + } + + return newPath; +} + +void XdgUrl::saveDownloadedFile(QNetworkReply *reply) +{ + QJsonObject result; + + QTemporaryFile temporaryFile; + + if (!temporaryFile.open() || temporaryFile.write(reply->readAll()) == -1) { + result["status"] = QString("error_save"); + result["message"] = temporaryFile.errorString(); + emit error(result); + return; + } + + QString type = metadata_["type"].toString(); + QString destination = destinations_[type].toString(); + QString path = destination + "/" + metadata_["filename"].toString(); + + utils::File::makeDir(destination); + utils::File::remove(path); // Remove previous downloaded file + + if (!temporaryFile.copy(path)) { + result["status"] = QString("error_save"); + result["message"] = temporaryFile.errorString(); + emit error(result); + return; + } + + destination_ = destination; + + result["status"] = QString("success_download"); + result["message"] = QString("The file has been stored into " + destination); + emit finished(result); +} + +void XdgUrl::installDownloadedFile(QNetworkReply *reply) +{ + QJsonObject result; + + QTemporaryFile temporaryFile; + + if (!temporaryFile.open() || temporaryFile.write(reply->readAll()) == -1) { + result["status"] = QString("error_save"); + result["message"] = temporaryFile.errorString(); + emit error(result); + return; + } + + QString type = metadata_["type"].toString(); + QString destination = destinations_[type].toString(); + QString path = destination + "/" + metadata_["filename"].toString(); + + utils::File::makeDir(destination); + utils::File::remove(path); // Remove previous downloaded file + + if (type == "bin" + && utils::Package::installProgram(temporaryFile.fileName(), path)) { + result["message"] = QString("The program has been installed into " + destination); + } + else if ((type == "plasma_plasmoids" || type == "plasma4_plasmoids" || type == "plasma5_plasmoids") + && utils::Package::installPlasmapkg(temporaryFile.fileName(), "plasmoid")) { + result["message"] = QString("The plasmoid has been installed"); + } + else if ((type == "plasma_look_and_feel" || type == "plasma5_look_and_feel") + && utils::Package::installPlasmapkg(temporaryFile.fileName(), "lookandfeel")) { + result["message"] = QString("The plasma look and feel has been installed"); + } + else if ((type == "plasma_desktopthemes" || type == "plasma5_desktopthemes") + && utils::Package::installPlasmapkg(temporaryFile.fileName(), "theme")) { + result["message"] = QString("The plasma desktop theme has been installed"); + } + else if (type == "kwin_effects" + && utils::Package::installPlasmapkg(temporaryFile.fileName(), "kwineffect")) { + result["message"] = QString("The KWin effect has been installed"); + } + else if (type == "kwin_scripts" + && utils::Package::installPlasmapkg(temporaryFile.fileName(), "kwinscript")) { + result["message"] = QString("The KWin script has been installed"); + } + else if (type == "kwin_tabbox" + && utils::Package::installPlasmapkg(temporaryFile.fileName(), "windowswitcher")) { + result["message"] = QString("The KWin window switcher has been installed"); + } + else if (utils::Package::uncompressArchive(temporaryFile.fileName(), destination)) { + result["message"] = QString("The archive file has been uncompressed into " + destination); + } + else if (temporaryFile.copy(path)) { + result["message"] = QString("The file has been stored into " + destination); + } + else { + result["status"] = QString("error_install"); + result["message"] = temporaryFile.errorString(); + emit error(result); + return; + } + + destination_ = destination; + + result["status"] = QString("success_install"); + emit finished(result); +} + +} // namespace handlers diff --git a/src/app/handlers/xdgurl.h b/src/app/handlers/xdgurl.h new file mode 100644 index 0000000000000000000000000000000000000000..3cd2cd030cc882508a1372891b5eb4bf29205bf8 --- /dev/null +++ b/src/app/handlers/xdgurl.h @@ -0,0 +1,54 @@ +#pragma once + +#include <QObject> +#include <QJsonObject> + +class QNetworkReply; + +namespace utils { +class Config; +class Network; +} + +namespace handlers { + +class XdgUrl : public QObject +{ + Q_OBJECT + +public: + explicit XdgUrl(const QString &xdgUrl, utils::Config *config, utils::Network *network, QObject *parent = 0); + +signals: + void started(); + void finished(const QJsonObject &result); + void error(const QJsonObject &result); + void downloadProgress(const qint64 &received, const qint64 &total); + +public slots: + void process(); + bool isValid(); + void openDestination(); + QString xdgUrl() const; + QJsonObject metadata() const; + +private slots: + void downloaded(QNetworkReply *reply); + +private: + void parse(); + void loadDestinations(); + QString convertPathString(const QString &path); + void saveDownloadedFile(QNetworkReply *reply); + void installDownloadedFile(QNetworkReply *reply); + + QString xdgUrl_; + utils::Config *config_; + utils::Network *network_; + + QJsonObject metadata_; + QJsonObject destinations_; + QString destination_; +}; + +} // namespace handlers diff --git a/src/app/main.cpp b/src/app/main.cpp new file mode 100644 index 0000000000000000000000000000000000000000..30d59795b5f327874556b7090344257caf1a76a4 --- /dev/null +++ b/src/app/main.cpp @@ -0,0 +1,59 @@ +#include <QtGlobal> +#include <QString> +#include <QStringList> +#include <QUrl> +#include <QJsonObject> +#include <QCommandLineParser> +#include <QCoreApplication> +#include <QGuiApplication> +#include <QIcon> +#include <QQmlApplicationEngine> +#include <QQmlContext> + +#include "../libs/utils/config.h" +#include "../libs/utils/network.h" + +#include "handlers/xdgurl.h" + +int main(int argc, char *argv[]) +{ + // Init +#if QT_VERSION >= QT_VERSION_CHECK(5, 6, 0) + QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling); +#endif + QGuiApplication app(argc, argv); + utils::Config *config = new utils::Config(":/configs"); + utils::Network *network = new utils::Network(true); + + QJsonObject configApplication = config->get("application"); + + app.setApplicationName(configApplication["name"].toString()); + app.setApplicationVersion(configApplication["version"].toString()); + app.setOrganizationName(configApplication["organization"].toString()); + app.setOrganizationDomain(configApplication["domain"].toString()); + app.setWindowIcon(QIcon::fromTheme(configApplication["id"].toString(), QIcon(configApplication["icon"].toString()))); + + // Setup CLI + QCommandLineParser clParser; + clParser.setApplicationDescription(configApplication["description"].toString()); + clParser.addHelpOption(); + clParser.addVersionOption(); + clParser.addPositionalArgument("xdgurl", "XDG-URL"); + clParser.process(app); + + QStringList args = clParser.positionalArguments(); + + if (args.size() != 1) { + clParser.showHelp(1); + } + + QString xdgUrl = args.at(0); + + // Setup QML + QQmlApplicationEngine qmlAppEngine; + QQmlContext *qmlContext = qmlAppEngine.rootContext(); + qmlContext->setContextProperty("xdgUrlHandler", new handlers::XdgUrl(xdgUrl, config, network)); + qmlAppEngine.load(QUrl("qrc:/qml/main.qml")); + + return app.exec(); +} diff --git a/src/app/qml/main.qml b/src/app/qml/main.qml new file mode 100644 index 0000000000000000000000000000000000000000..3b1d8ea857e5c30f85ff559aa36be43270913539 --- /dev/null +++ b/src/app/qml/main.qml @@ -0,0 +1,155 @@ +import QtQuick 2.3 +import QtQuick.Window 2.0 +import QtQuick.Controls 1.2 +import QtQuick.Dialogs 1.2 + +import 'scripts/Utility.js' as Utility + +Window { + id: root + title: Qt.application.name + width: 400 + height: 200 + minimumWidth: 400 + minimumHeight: 200 + maximumWidth: 800 + maximumHeight: 400 + + MessageDialog { + id: confirmDialog + title: root.title + icon: StandardIcon.Question + text: '' + informativeText: '' + detailedText: '' + standardButtons: StandardButton.Ok | StandardButton.Cancel + onAccepted: xdgUrlHandler.process() + onRejected: Qt.quit() + } + + MessageDialog { + id: infoDialog + title: root.title + icon: StandardIcon.Information + text: '' + informativeText: '' + detailedText: '' + standardButtons: StandardButton.Open | StandardButton.Close + onAccepted: { + xdgUrlHandler.openDestination(); + Qt.quit(); + } + onRejected: Qt.quit() + } + + MessageDialog { + id: errorDialog + title: root.title + icon: StandardIcon.Warning + text: '' + informativeText: '' + detailedText: '' + standardButtons: StandardButton.Close + onRejected: Qt.quit() + } + + Dialog { + id: progressDialog + title: root.title + contentItem: Item { + implicitWidth: 400 + implicitHeight: 150 + Column { + anchors.fill: parent + anchors.margins: 12 + spacing: 8 + Label { + id: primaryLabel + text: '' + font.bold: true + } + Label { + id: informativeLabel + text: '' + } + ProgressBar { + id: progressBar + maximumValue: 1 + minimumValue: 0 + value: 0 + anchors.left: parent.left + anchors.right: parent.right + } + Label { + id: progressLabel + text: '' + anchors.right: parent.right + } + Button { + id: cancelButton + text: 'Cancel' + anchors.right: parent.right + onClicked: Qt.quit() + } + } + } + property alias primaryLabel: primaryLabel + property alias informativeLabel: informativeLabel + property alias progressBar: progressBar + property alias progressLabel: progressLabel + } + + Component.onCompleted: { + var metadata = xdgUrlHandler.metadata(); + var primaryMessages = { + 'success_download': 'Download successfull', + 'success_install': 'Installation successfull', + 'error_validation': 'Validation error', + 'error_network': 'Network error', + 'error_save': 'Saving file failed', + 'error_install': 'Installation failed' + }; + + xdgUrlHandler.started.connect(function() { + progressDialog.open(); + }); + + xdgUrlHandler.finished.connect(function(result) { + progressDialog.close(); + infoDialog.text = primaryMessages[result.status]; + infoDialog.informativeText = metadata.filename; + infoDialog.detailedText = result.message; + infoDialog.open(); + }); + + xdgUrlHandler.error.connect(function(result) { + progressDialog.close(); + errorDialog.text = primaryMessages[result.status]; + errorDialog.informativeText = metadata.filename; + errorDialog.detailedText = result.message; + errorDialog.open(); + }); + + xdgUrlHandler.downloadProgress.connect(function(received, total) { + progressDialog.primaryLabel.text = 'Downloading... '; + progressDialog.informativeLabel.text = metadata.filename; + progressDialog.progressBar.value = received / total; + progressDialog.progressLabel.text = Utility.convertByteToHumanReadable(received) + + ' / ' + Utility.convertByteToHumanReadable(total) + }); + + if (xdgUrlHandler.isValid()) { + confirmDialog.text = 'Do you want to ' + metadata.command + '?'; + confirmDialog.informativeText = metadata.filename; + confirmDialog.detailedText = 'URL: ' + metadata.url + '\n\n' + + 'File: ' + metadata.filename + '\n\n' + + 'Type: ' + metadata.type; + confirmDialog.open(); + } + else { + errorDialog.text = 'Validation error'; + errorDialog.detailedText = 'Invalid XDG-URL ' + xdgUrlHandler.xdgUrl(); + errorDialog.open(); + } + } +} diff --git a/src/app/qml/qml.qrc b/src/app/qml/qml.qrc new file mode 100644 index 0000000000000000000000000000000000000000..640e9f07108d13e701a9470cb50262b1ce40e70b --- /dev/null +++ b/src/app/qml/qml.qrc @@ -0,0 +1,6 @@ +<RCC> + <qresource prefix="/qml"> + <file>main.qml</file> + <file>scripts/Utility.js</file> + </qresource> +</RCC> diff --git a/src/app/qml/scripts/Utility.js b/src/app/qml/scripts/Utility.js new file mode 100644 index 0000000000000000000000000000000000000000..d60a332b4da6f562ed69926b8c6c459e1daef7d7 --- /dev/null +++ b/src/app/qml/scripts/Utility.js @@ -0,0 +1,43 @@ +.pragma library + +function convertByteToHumanReadable(bytes) { + bytes = parseFloat(bytes); + var kb = 1024; + var mb = 1024 * kb; + var gb = 1024 * mb; + var tb = 1024 * gb; + var pb = 1024 * tb; + var eb = 1024 * pb; + var zb = 1024 * eb; + var yb = 1024 * zb; + + var text = ''; + if (bytes < kb) { + text = bytes.toFixed(0) + ' B'; + } + else if (bytes < mb) { + text = (bytes / kb).toFixed(2) + ' KB'; + } + else if (bytes < gb) { + text = (bytes / mb).toFixed(2) + ' MB'; + } + else if (bytes < tb) { + text = (bytes / gb).toFixed(2) + ' GB'; + } + else if (bytes < pb) { + text = (bytes / tb).toFixed(2) + ' TB'; + } + else if (bytes < eb) { + text = (bytes / pb).toFixed(2) + ' PB'; + } + else if (bytes < zb) { + text = (bytes / eb).toFixed(2) + ' EB'; + } + else if (bytes < yb) { + text = (bytes / zb).toFixed(2) + ' ZB'; + } + else if (bytes >= yb) { + text = (bytes / yb).toFixed(2) + ' YB'; + } + return text; +} diff --git a/src/desktop/desktop.qrc b/src/desktop/desktop.qrc new file mode 100644 index 0000000000000000000000000000000000000000..2e0475004292b32a2a8c875f6fcbb03092c91fc4 --- /dev/null +++ b/src/desktop/desktop.qrc @@ -0,0 +1,5 @@ +<RCC> + <qresource prefix="/desktop"> + <file>xdgurl.svg</file> + </qresource> +</RCC> diff --git a/src/xdgurl.desktop b/src/desktop/xdgurl.desktop similarity index 93% rename from src/xdgurl.desktop rename to src/desktop/xdgurl.desktop index 688ca021c06b73247aa4f47294fc0e66585d6e8f..aba57f117912ed2f3ba256709fd5dac7464a5735 100644 --- a/src/xdgurl.desktop +++ b/src/desktop/xdgurl.desktop @@ -1,6 +1,7 @@ [Desktop Entry] Name=xdgurl Exec=xdgurl %u +Icon=xdgurl Type=Application Terminal=false NoDisplay=true diff --git a/src/desktop/xdgurl.svg b/src/desktop/xdgurl.svg new file mode 100644 index 0000000000000000000000000000000000000000..a55a8e481d1abbe8fa585539fe9a41cdc3b53679 --- /dev/null +++ b/src/desktop/xdgurl.svg @@ -0,0 +1,75 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + version="1.1" + id="svg2" + viewBox="0 0 1000 1000" + height="1000" + width="1000"> + <defs + id="defs4" /> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <g + transform="translate(0,-52.362161)" + id="layer1"> + <circle + r="475" + cy="552.36218" + cx="500" + id="path4138" + style="fill:#19a2ff;fill-opacity:1;stroke:none;stroke-width:1;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" /> + <circle + r="450" + cy="552.36218" + cx="500" + id="path4138-0" + style="fill:#53b2ff;fill-opacity:1;stroke:none;stroke-width:1;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" /> + <g + transform="translate(0,20)" + id="g4205"> + <path + id="path4145" + style="font-style:normal;font-variant:normal;font-weight:500;font-stretch:normal;font-size:medium;line-height:125%;font-family:Futura;-inkscape-font-specification:'Futura Medium';letter-spacing:0px;word-spacing:0px;fill:#ffffff;fill-opacity:1;stroke:none;stroke-width:1px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1" + d="m 192.87313,533.47911 -74.36008,-101.11803 56.93857,0 46.10326,64.24502 46.95309,-64.24502 58.63824,0 -76.48467,101.11803 88.59474,119.21524 -56.93858,0 -60.76282,-82.34221 -62.25002,82.34221 -58.42577,0 91.99404,-119.21524 z" /> + <path + id="path4147" + style="font-style:normal;font-variant:normal;font-weight:500;font-stretch:normal;font-size:medium;line-height:125%;font-family:Futura;-inkscape-font-specification:'Futura Medium';letter-spacing:0px;word-spacing:0px;fill:#ffffff;fill-opacity:1;stroke:none;stroke-width:1px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1" + d="m 472.62967,312.47491 48.01537,0 0,340.84056 -48.01537,0 0,-21.67066 c -18.83789,18.41297 -40.15445,27.61946 -63.94968,27.61946 -28.32765,0 -51.83961,-10.33959 -70.53586,-31.01878 -18.55461,-21.1041 -27.83192,-47.44882 -27.83192,-79.03415 0,-30.87715 9.27731,-56.65531 27.83192,-77.3345 18.41298,-20.82082 41.57084,-31.23123 69.47357,-31.23123 24.22015,0 45.8908,9.91468 65.01197,29.74403 z M 359.17742,549.21134 c 0,19.82935 5.31143,35.97612 15.9343,48.44029 10.90615,12.60581 24.64506,18.90871 41.21674,18.90871 17.70478,0 32.01024,-6.09045 42.91639,-18.27134 10.90615,-12.60581 16.35922,-28.61093 16.35922,-48.01537 0,-19.40445 -5.45307,-35.40957 -16.35922,-48.01538 -10.90615,-12.32253 -25.06997,-18.48379 -42.49148,-18.48379 -16.43004,0 -30.16895,6.23208 -41.21673,18.69625 -10.90615,12.60581 -16.35922,28.18602 -16.35922,46.74063 z" /> + <path + id="path4149" + style="font-style:normal;font-variant:normal;font-weight:500;font-stretch:normal;font-size:medium;line-height:125%;font-family:Futura;-inkscape-font-specification:'Futura Medium';letter-spacing:0px;word-spacing:0px;fill:#ffffff;fill-opacity:1;stroke:none;stroke-width:1px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1" + d="m 722.24998,648.95295 c 0,9.77304 -0.3541,18.34215 -1.06229,25.70734 -0.56655,7.50683 -1.41638,14.09301 -2.54949,19.75854 -3.39932,15.58021 -10.05632,28.96502 -19.97099,40.15445 -18.69625,21.52901 -44.4036,32.29352 -77.12204,32.29352 -27.61947,0 -50.35241,-7.43601 -68.19883,-22.30802 -18.41297,-15.29693 -29.03584,-36.47186 -31.86861,-63.52477 l 48.01537,0 c 1.8413,10.19796 4.88652,18.05889 9.13567,23.58278 9.91468,12.88908 24.36178,19.33362 43.34131,19.33362 34.98465,0 52.47698,-21.4582 52.47698,-64.37459 l 0,-28.89421 c -18.97953,19.40444 -40.86264,29.10666 -65.64934,29.10666 -28.18601,0 -51.27305,-10.19795 -69.26111,-30.59386 -18.1297,-20.67919 -27.19455,-46.52817 -27.19455,-77.54695 0,-30.16895 8.42748,-55.80548 25.28243,-76.90959 18.1297,-22.37884 42.06657,-33.56826 71.81061,-33.56826 26.06144,0 47.73209,9.70222 65.01196,29.10666 l 0,-27.37031 47.80292,0 z m -45.8908,-98.15532 c 0,-20.11264 -5.38225,-36.18859 -16.14676,-48.22784 C 649.30627,490.24726 635.3549,484.086 618.35831,484.086 c -18.1297,0 -32.43517,6.72782 -42.9164,20.18345 -9.48976,12.03925 -14.23464,27.61946 -14.23464,46.74063 0,18.83789 4.74488,34.27647 14.23464,46.31572 10.33959,13.17236 24.64506,19.75854 42.9164,19.75854 18.27133,0 32.71844,-6.657 43.34131,-19.971 9.77304,-12.03925 14.65956,-27.47782 14.65956,-46.31571 z" /> + <path + style="fill:#ffffff;fill-opacity:1;stroke:none;stroke-width:1;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" + d="m 739.50519,490.21255 36,0 0,36 -36,0 z" + id="rect4156" /> + <path + style="fill:#ffffff;fill-opacity:1;stroke:none;stroke-width:1;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" + d="m 739.53345,589.21057 36,0 0,36 -36,0 z" + id="rect4156-9" /> + <path + style="fill:#ffffff;fill-opacity:1;stroke:none;stroke-width:1;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" + d="m 852.99238,412.46896 22,0 -94.33913,290.17369 -22,0 z" + id="rect4156-5" /> + <path + style="fill:#ffffff;fill-opacity:1;stroke:none;stroke-width:1;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" + d="m 902.99238,412.46896 22,0 -94.33913,290.17369 -22,0 z" + id="rect4156-5-3" /> + </g> + </g> +</svg> diff --git a/src/libs/utils/android.cpp b/src/libs/utils/android.cpp new file mode 100644 index 0000000000000000000000000000000000000000..9cca280565c8b419b83065f6fb63e671a61970db --- /dev/null +++ b/src/libs/utils/android.cpp @@ -0,0 +1,38 @@ +/** + * A library for Qt app + * + * LICENSE: The GNU Lesser General Public License, version 3.0 + * + * @author Akira Ohgaki <akiraohgaki@gmail.com> + * @copyright Akira Ohgaki + * @license https://opensource.org/licenses/LGPL-3.0 The GNU Lesser General Public License, version 3.0 + * @link https://github.com/akiraohgaki/qt-libs + */ + +#include "android.h" + +#include <QAndroidJniObject> + +namespace utils { + +Android::Android(QObject *parent) : QObject(parent) +{} + +bool Android::openApk(const QString &uri) +{ + QAndroidJniObject activity = QAndroidJniObject::callStaticObjectMethod("org/qtproject/qt5/android/QtNative", "activity", "()Landroid/app/Activity;"); + if (activity.isValid()) { + QAndroidJniObject fileUri = QAndroidJniObject::fromString(uri); + QAndroidJniObject parsedUri = QAndroidJniObject::callStaticObjectMethod("android/net/Uri", "parse", "(Ljava/lang/String;)Landroid/net/Uri;", fileUri.object()); + QAndroidJniObject mimeType = QAndroidJniObject::fromString("application/vnd.android.package-archive"); + QAndroidJniObject activityKind = QAndroidJniObject::fromString("android.intent.action.VIEW"); + QAndroidJniObject intent("android/content/Intent", "(Ljava/lang/String;)V", activityKind.object()); + intent = intent.callObjectMethod("setDataAndType", "(Landroid/net/Uri;Ljava/lang/String;)Landroid/content/Intent;", parsedUri.object(), mimeType.object()); + intent = intent.callObjectMethod("setFlags", "(I)Landroid/content/Intent;", 0x10000000); // 0x10000000 = FLAG_ACTIVITY_NEW_TASK + activity.callObjectMethod("startActivity", "(Landroid/content/Intent;)V", intent.object()); + return true; + } + return false; +} + +} // namespace utils diff --git a/src/libs/utils/android.h b/src/libs/utils/android.h new file mode 100644 index 0000000000000000000000000000000000000000..bbbccd36a693dc5ca086cfa1db8d4b87eabaa0e0 --- /dev/null +++ b/src/libs/utils/android.h @@ -0,0 +1,28 @@ +/** + * A library for Qt app + * + * LICENSE: The GNU Lesser General Public License, version 3.0 + * + * @author Akira Ohgaki <akiraohgaki@gmail.com> + * @copyright Akira Ohgaki + * @license https://opensource.org/licenses/LGPL-3.0 The GNU Lesser General Public License, version 3.0 + * @link https://github.com/akiraohgaki/qt-libs + */ + +#pragma once + +#include <QObject> + +namespace utils { + +class Android : public QObject +{ + Q_OBJECT + +public: + explicit Android(QObject *parent = 0); + + static bool openApk(const QString &uri); +}; + +} // namespace utils diff --git a/src/libs/utils/config.cpp b/src/libs/utils/config.cpp new file mode 100644 index 0000000000000000000000000000000000000000..13b1018af9a1ee4f0feab0315794875aa8a3bb43 --- /dev/null +++ b/src/libs/utils/config.cpp @@ -0,0 +1,50 @@ +/** + * A library for Qt app + * + * LICENSE: The GNU Lesser General Public License, version 3.0 + * + * @author Akira Ohgaki <akiraohgaki@gmail.com> + * @copyright Akira Ohgaki + * @license https://opensource.org/licenses/LGPL-3.0 The GNU Lesser General Public License, version 3.0 + * @link https://github.com/akiraohgaki/qt-libs + */ + +#include "config.h" + +#include "file.h" +#include "json.h" + +namespace utils { + +Config::Config(const QString &configsDir, QObject *parent) : + QObject(parent), configsDir_(configsDir) +{} + +QJsonObject Config::get(const QString &name) +{ + QString configFile = configsDir_ + "/" + name + ".json"; + + if (!cacheData_.contains(name)) { + QString json = utils::File::readText(configFile); + if (json.isEmpty()) { + json = "{}"; // Blank JSON data as default + } + cacheData_[name] = utils::Json::convertStrToObj(json); + } + return cacheData_[name].toObject(); +} + +bool Config::set(const QString &name, const QJsonObject &jsonObj) +{ + QString configFile = configsDir_ + "/" + name + ".json"; + QString json = utils::Json::convertObjToStr(jsonObj); + + utils::File::makeDir(configsDir_); + if (utils::File::writeText(configFile, json)) { + cacheData_[name] = jsonObj; + return true; + } + return false; +} + +} // namespace utils diff --git a/src/libs/utils/config.h b/src/libs/utils/config.h new file mode 100644 index 0000000000000000000000000000000000000000..9949407873709013bd414ff9b695c31e21c05f6d --- /dev/null +++ b/src/libs/utils/config.h @@ -0,0 +1,34 @@ +/** + * A library for Qt app + * + * LICENSE: The GNU Lesser General Public License, version 3.0 + * + * @author Akira Ohgaki <akiraohgaki@gmail.com> + * @copyright Akira Ohgaki + * @license https://opensource.org/licenses/LGPL-3.0 The GNU Lesser General Public License, version 3.0 + * @link https://github.com/akiraohgaki/qt-libs + */ + +#pragma once + +#include <QObject> +#include <QJsonObject> + +namespace utils { + +class Config : public QObject +{ + Q_OBJECT + +public: + explicit Config(const QString &configsDir, QObject *parent = 0); + + QJsonObject get(const QString &name); + bool set(const QString &name, const QJsonObject &jsonObj); + +private: + QString configsDir_; + QJsonObject cacheData_; +}; + +} // namespace utils diff --git a/src/libs/utils/file.cpp b/src/libs/utils/file.cpp new file mode 100644 index 0000000000000000000000000000000000000000..459292abf14210ace0df70dde873e9e71612f6ca --- /dev/null +++ b/src/libs/utils/file.cpp @@ -0,0 +1,201 @@ +/** + * A library for Qt app + * + * LICENSE: The GNU Lesser General Public License, version 3.0 + * + * @author Akira Ohgaki <akiraohgaki@gmail.com> + * @copyright Akira Ohgaki + * @license https://opensource.org/licenses/LGPL-3.0 The GNU Lesser General Public License, version 3.0 + * @link https://github.com/akiraohgaki/qt-libs + */ + +#include "file.h" + +#include <QIODevice> +#include <QStandardPaths> +#include <QDir> +#include <QFile> +#include <QFileInfo> +#include <QTextStream> + +namespace utils { + +File::File(QObject *parent) : QObject(parent) +{} + +QString File::rootPath() +{ + return QDir::rootPath(); +} + +QString File::tempPath() +{ + return QDir::tempPath(); +} + +QString File::homePath() +{ + return QDir::homePath(); +} + +QString File::genericDataPath() +{ + return QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation); +} + +QString File::genericConfigPath() +{ + return QStandardPaths::writableLocation(QStandardPaths::GenericConfigLocation); +} + +QString File::genericCachePath() +{ + return QStandardPaths::writableLocation(QStandardPaths::GenericCacheLocation); +} + +QString File::kdehomePath() +{ + // KDE System Administration/Environment Variables + // https://userbase.kde.org/KDE_System_Administration/Environment_Variables + + // KDE 4 maybe uses $KDEHOME + QString path = QString::fromLocal8Bit(qgetenv("KDEHOME").constData()); + if (path.isEmpty()) { + path = homePath() + "/.kde"; + } + return path; +} + +QFileInfoList File::readDir(const QString &path) +{ + QDir dir(path); + dir.setFilter(QDir::AllEntries | QDir::Hidden | QDir::System | QDir::NoDotAndDotDot); + //dir.setSorting(QDir::DirsFirst | QDir::Name); + return dir.entryInfoList(); +} + +bool File::makeDir(const QString &path) +{ + // This function will create all parent directories + QDir dir(path); + if (!dir.exists() && dir.mkpath(path)) { + return true; + } + return false; +} + +QString File::readText(const QString &path) +{ + QString data; + QFile file(path); + if (file.exists() && file.open(QIODevice::ReadOnly | QIODevice::Text)) { + QTextStream in(&file); + in.setCodec("UTF-8"); + data = in.readAll(); + file.close(); + } + return data; +} + +bool File::writeText(const QString &path, const QString &data) +{ + QFile file(path); + if (file.open(QIODevice::WriteOnly | QIODevice::Text)) { + QTextStream out(&file); + out.setCodec("UTF-8"); + out << data; + file.close(); + return true; + } + return false; +} + +QByteArray File::readBinary(const QString &path) +{ + QByteArray data; + QFile file(path); + if (file.exists() && file.open(QIODevice::ReadOnly)) { + data = file.readAll(); + file.close(); + } + return data; +} + +bool File::writeBinary(const QString &path, const QByteArray &data) +{ + QFile file(path); + if (file.open(QIODevice::WriteOnly)) { + file.write(data); + file.close(); + return true; + } + return false; +} + +bool File::copy(const QString &path, const QString &targetPath) +{ + // This function will copy files recursively + QFileInfo fileInfo(path); + if (fileInfo.isFile()) { + QFile file(path); + if (file.copy(targetPath)) { + return true; + } + } + else if (fileInfo.isDir()) { + QDir targetDir(targetPath); + QString targetDirName = targetDir.dirName(); + targetDir.cdUp(); + if (targetDir.mkdir(targetDirName)) { + QDir dir(path); + dir.setFilter(QDir::AllEntries | QDir::Hidden | QDir::System | QDir::NoDotAndDotDot); + QStringList entries = dir.entryList(); + foreach (const QString &entry, entries) { + if (!copy(path + "/" + entry, targetPath + "/" + entry)) { + return false; + } + } + return true; + } + } + return false; +} + +bool File::move(const QString &path, const QString &targetPath) +{ + QFileInfo fileInfo(path); + if (fileInfo.isFile()) { + QFile file(path); + if (file.rename(targetPath)) { + return true; + } + } + else if (fileInfo.isDir()) { + QDir dir(path); + if (dir.rename(path, targetPath)) { + return true; + } + } + return false; +} + +bool File::remove(const QString &path) +{ + // This function will remove files recursively + QFileInfo fileInfo(path); + if (fileInfo.isFile()) { + QFile file(path); + if (file.remove()) { + return true; + } + } + else if (fileInfo.isDir()) { + QDir dir(path); + if (dir.removeRecursively()) { + return true; + } + } + return false; +} + +} // namespace utils diff --git a/src/libs/utils/file.h b/src/libs/utils/file.h new file mode 100644 index 0000000000000000000000000000000000000000..0ec6548a0c14fb4efe3d616e31fa80c57666ef90 --- /dev/null +++ b/src/libs/utils/file.h @@ -0,0 +1,46 @@ +/** + * A library for Qt app + * + * LICENSE: The GNU Lesser General Public License, version 3.0 + * + * @author Akira Ohgaki <akiraohgaki@gmail.com> + * @copyright Akira Ohgaki + * @license https://opensource.org/licenses/LGPL-3.0 The GNU Lesser General Public License, version 3.0 + * @link https://github.com/akiraohgaki/qt-libs + */ + +#pragma once + +#include <QObject> + +class QFileInfo; +typedef QList<QFileInfo> QFileInfoList; + +namespace utils { + +class File : public QObject +{ + Q_OBJECT + +public: + explicit File(QObject *parent = 0); + + static QString rootPath(); + static QString tempPath(); + static QString homePath(); + static QString genericDataPath(); + static QString genericConfigPath(); + static QString genericCachePath(); + static QString kdehomePath(); + static QFileInfoList readDir(const QString &path); + static bool makeDir(const QString &path); + static QString readText(const QString &path); + static bool writeText(const QString &path, const QString &data); + static QByteArray readBinary(const QString &path); + static bool writeBinary(const QString &path, const QByteArray &data); + static bool copy(const QString &path, const QString &targetPath); + static bool move(const QString &path, const QString &targetPath); + static bool remove(const QString &path); +}; + +} // namespace utils diff --git a/src/libs/utils/json.cpp b/src/libs/utils/json.cpp new file mode 100644 index 0000000000000000000000000000000000000000..1573c847890f2d0c6365b9372542b9a4db355ec1 --- /dev/null +++ b/src/libs/utils/json.cpp @@ -0,0 +1,50 @@ +/** + * A library for Qt app + * + * LICENSE: The GNU Lesser General Public License, version 3.0 + * + * @author Akira Ohgaki <akiraohgaki@gmail.com> + * @copyright Akira Ohgaki + * @license https://opensource.org/licenses/LGPL-3.0 The GNU Lesser General Public License, version 3.0 + * @link https://github.com/akiraohgaki/qt-libs + */ + +#include "json.h" + +#include <QJsonDocument> +#include <QJsonObject> +#include <QJsonParseError> + +namespace utils { + +Json::Json(QObject *parent) : QObject(parent) +{} + +QString Json::convertObjToStr(const QJsonObject &jsonObj) +{ + QJsonDocument jsonDoc(jsonObj); + return QString::fromUtf8(jsonDoc.toJson()); +} + +QJsonObject Json::convertStrToObj(const QString &json) +{ + QJsonObject jsonObj; + QJsonParseError jsonError; + QJsonDocument jsonDoc = QJsonDocument::fromJson(json.toUtf8(), &jsonError); + if (jsonError.error == QJsonParseError::NoError && jsonDoc.isObject()) { + jsonObj = jsonDoc.object(); + } + return jsonObj; +} + +bool Json::isValid(const QString &json) +{ + QJsonParseError jsonError; + QJsonDocument::fromJson(json.toUtf8(), &jsonError); + if (jsonError.error == QJsonParseError::NoError) { + return true; + } + return false; +} + +} // namespace utils diff --git a/src/libs/utils/json.h b/src/libs/utils/json.h new file mode 100644 index 0000000000000000000000000000000000000000..f53527ad12b13650f4a5634d005f52e4bcaf5cfe --- /dev/null +++ b/src/libs/utils/json.h @@ -0,0 +1,30 @@ +/** + * A library for Qt app + * + * LICENSE: The GNU Lesser General Public License, version 3.0 + * + * @author Akira Ohgaki <akiraohgaki@gmail.com> + * @copyright Akira Ohgaki + * @license https://opensource.org/licenses/LGPL-3.0 The GNU Lesser General Public License, version 3.0 + * @link https://github.com/akiraohgaki/qt-libs + */ + +#pragma once + +#include <QObject> + +namespace utils { + +class Json : public QObject +{ + Q_OBJECT + +public: + explicit Json(QObject *parent = 0); + + static QString convertObjToStr(const QJsonObject &jsonObj); + static QJsonObject convertStrToObj(const QString &json); + static bool isValid(const QString &json); +}; + +} // namespace utils diff --git a/src/libs/utils/network.cpp b/src/libs/utils/network.cpp new file mode 100644 index 0000000000000000000000000000000000000000..2256a941a0a761bd76ddc39fb2c57ea7138c8a47 --- /dev/null +++ b/src/libs/utils/network.cpp @@ -0,0 +1,59 @@ +/** + * A library for Qt app + * + * LICENSE: The GNU Lesser General Public License, version 3.0 + * + * @author Akira Ohgaki <akiraohgaki@gmail.com> + * @copyright Akira Ohgaki + * @license https://opensource.org/licenses/LGPL-3.0 The GNU Lesser General Public License, version 3.0 + * @link https://github.com/akiraohgaki/qt-libs + */ + +#include "network.h" + +#include <QEventLoop> +#include <QNetworkAccessManager> +#include <QNetworkRequest> +#include <QNetworkReply> + +namespace utils { + +Network::Network(const bool &async, QObject *parent) : + QObject(parent), async_(async) +{ + manager_ = new QNetworkAccessManager(this); + connect(manager_, &QNetworkAccessManager::finished, this, &utils::Network::finished); + if (!async_) { + eventLoop_ = new QEventLoop(); + connect(manager_, &QNetworkAccessManager::finished, eventLoop_, &QEventLoop::quit); + } +} + +Network::~Network() +{ + manager_->deleteLater(); + if (!async_) { + delete eventLoop_; + } +} + +QNetworkReply *Network::head(const QUrl &uri) +{ + QNetworkReply *reply = manager_->head(QNetworkRequest(uri)); + if (!async_) { + eventLoop_->exec(); + } + return reply; +} + +QNetworkReply *Network::get(const QUrl &uri) +{ + QNetworkReply *reply = manager_->get(QNetworkRequest(uri)); + connect(reply, &QNetworkReply::downloadProgress, this, &utils::Network::downloadProgress); + if (!async_) { + eventLoop_->exec(); + } + return reply; +} + +} // namespace utils diff --git a/src/libs/utils/network.h b/src/libs/utils/network.h new file mode 100644 index 0000000000000000000000000000000000000000..fcb835028def3b28e0e99d72f792a813a561a5bd --- /dev/null +++ b/src/libs/utils/network.h @@ -0,0 +1,43 @@ +/** + * A library for Qt app + * + * LICENSE: The GNU Lesser General Public License, version 3.0 + * + * @author Akira Ohgaki <akiraohgaki@gmail.com> + * @copyright Akira Ohgaki + * @license https://opensource.org/licenses/LGPL-3.0 The GNU Lesser General Public License, version 3.0 + * @link https://github.com/akiraohgaki/qt-libs + */ + +#pragma once + +#include <QObject> + +class QEventLoop; +class QNetworkAccessManager; +class QNetworkReply; + +namespace utils { + +class Network : public QObject +{ + Q_OBJECT + +public: + explicit Network(const bool &async = true, QObject *parent = 0); + ~Network(); + + QNetworkReply *head(const QUrl &uri); + QNetworkReply *get(const QUrl &uri); + +signals: + void finished(QNetworkReply *reply); + void downloadProgress(const qint64 &received, const qint64 &total); + +private: + bool async_; + QNetworkAccessManager *manager_; + QEventLoop *eventLoop_; +}; + +} // namespace utils diff --git a/src/libs/utils/package.cpp b/src/libs/utils/package.cpp new file mode 100644 index 0000000000000000000000000000000000000000..6196c063404fc176325665e6a0fba447b58abc83 --- /dev/null +++ b/src/libs/utils/package.cpp @@ -0,0 +1,120 @@ +/** + * A library for Qt app + * + * LICENSE: The GNU Lesser General Public License, version 3.0 + * + * @author Akira Ohgaki <akiraohgaki@gmail.com> + * @copyright Akira Ohgaki + * @license https://opensource.org/licenses/LGPL-3.0 The GNU Lesser General Public License, version 3.0 + * @link https://github.com/akiraohgaki/qt-libs + */ + +#include "package.h" + +#include <QJsonObject> +#include <QMimeDatabase> +#include <QProcess> + +namespace utils { + +Package::Package(QObject *parent) : QObject(parent) +{} + +bool Package::installProgram(const QString &path, const QString &targetPath) +{ + QString program = "install"; + QStringList arguments; + arguments << "-m" << "755" << "-p" << path << targetPath; + return execute(program, arguments); +} + +bool Package::installFile(const QString &path, const QString &targetPath) +{ + QString program = "install"; + QStringList arguments; + arguments << "-m" << "644" << "-p" << path << targetPath; + return execute(program, arguments); +} + +bool Package::installPlasmapkg(const QString &path, const QString &type) +{ + QString program = "plasmapkg2"; + QStringList arguments; + arguments << "-t" << type << "-i" << path; + return execute(program, arguments); +} + +bool Package::uninstallPlasmapkg(const QString &path, const QString &type) +{ + QString program = "plasmapkg2"; + QStringList arguments; + arguments << "-t" << type << "-r" << path; + return execute(program, arguments); +} + +bool Package::uncompressArchive(const QString &path, const QString &targetDir) +{ + QJsonObject archiveTypes; + archiveTypes["application/x-tar"] = QString("tar"); + archiveTypes["application/x-gzip"] = QString("tar"); + archiveTypes["application/gzip"] = QString("tar"); + archiveTypes["application/x-bzip"] = QString("tar"); + archiveTypes["application/x-bzip2"] = QString("tar"); + archiveTypes["application/x-xz"] = QString("tar"); + archiveTypes["application/x-lzma"] = QString("tar"); + archiveTypes["application/x-lzip"] = QString("tar"); + archiveTypes["application/x-compressed-tar"] = QString("tar"); + archiveTypes["application/x-bzip-compressed-tar"] = QString("tar"); + archiveTypes["application/x-bzip2-compressed-tar"] = QString("tar"); + archiveTypes["application/x-xz-compressed-tar"] = QString("tar"); + archiveTypes["application/x-lzma-compressed-tar"] = QString("tar"); + archiveTypes["application/x-lzip-compressed-tar"] = QString("tar"); + archiveTypes["application/zip"] = QString("zip"); + archiveTypes["application/x-7z-compressed"] = QString("7z"); + archiveTypes["application/x-rar"] = QString("rar"); + archiveTypes["application/x-rar-compressed"] = QString("rar"); + + QMimeDatabase mimeDb; + QString mimeType = mimeDb.mimeTypeForFile(path).name(); + + if (archiveTypes.contains(mimeType)) { + QString archiveType = archiveTypes[mimeType].toString(); + + QString program; + QStringList arguments; + + if (archiveType == "tar") { + program = "tar"; + arguments << "-xf" << path << "-C" << targetDir; + } + else if (archiveType == "zip") { + program = "unzip"; + arguments << "-o" << path << "-d" << targetDir; + } + else if (archiveType == "7z") { + program = "7z"; + arguments << "x" << path << "-o" + targetDir; // No space between -o and directory + } + else if (archiveType == "rar") { + program = "unrar"; + arguments << "e" << path << targetDir; + } + + return execute(program, arguments); + } + + return false; +} + +bool Package::execute(const QString &program, const QStringList &arguments) +{ + QProcess process; + process.start(program, arguments); + if (process.waitForFinished()) { + process.waitForReadyRead(); + return true; + } + return false; +} + +} // namespace utils diff --git a/src/libs/utils/package.h b/src/libs/utils/package.h new file mode 100644 index 0000000000000000000000000000000000000000..9f086dded976aa2384584a8c9e05d204f9681daf --- /dev/null +++ b/src/libs/utils/package.h @@ -0,0 +1,35 @@ +/** + * A library for Qt app + * + * LICENSE: The GNU Lesser General Public License, version 3.0 + * + * @author Akira Ohgaki <akiraohgaki@gmail.com> + * @copyright Akira Ohgaki + * @license https://opensource.org/licenses/LGPL-3.0 The GNU Lesser General Public License, version 3.0 + * @link https://github.com/akiraohgaki/qt-libs + */ + +#pragma once + +#include <QObject> + +namespace utils { + +class Package : public QObject +{ + Q_OBJECT + +public: + explicit Package(QObject *parent = 0); + + static bool installProgram(const QString &path, const QString &targetPath); + static bool installFile(const QString &path, const QString &targetPath); + static bool installPlasmapkg(const QString &path, const QString &type = "plasmoid"); + static bool uninstallPlasmapkg(const QString &path, const QString &type = "plasmoid"); + static bool uncompressArchive(const QString &path, const QString &targetDir); + +private: + static bool execute(const QString &program, const QStringList &arguments); +}; + +} // namespace utils diff --git a/src/xdgurl.py b/src/xdgurl.py deleted file mode 100644 index 11ae5085dca88a767b79fce38c2f09fea4fd872c..0000000000000000000000000000000000000000 --- a/src/xdgurl.py +++ /dev/null @@ -1,377 +0,0 @@ -#!/usr/bin/env python - -"""xdgurl -An install helper program for desktop stuff. - -Copyright: 2016, Akira Ohgaki -License: GPL-3+ - -https://github.com/xdgurl/xdgurl -""" - -import sys -import os -import shutil -import json -import tempfile -import mimetypes -import subprocess -import argparse - -if sys.version_info.major >= 3: - import urllib.request - import urllib.error - import urllib.parse - import tkinter - import tkinter.messagebox -else: - import urllib - import urlparse - import Tkinter - import tkMessageBox - -class XdgUrl: - - def __init__(self, xdg_url=''): - self.xdg_url = xdg_url - self.meta = self.parse() - - self.temp_dir = tempfile.gettempdir() - self.home_dir = os.path.expanduser('~') - #self.config_dir = os.path.join(self.home_dir, '.config', 'xdgurl') - self.data_dir = os.path.join(self.home_dir, '.local', 'share') - self.kde_data_dir = os.path.join(self.home_dir, '.kde', 'share') - - #self.config = {} - - self.destinations = { - 'downloads': os.path.join(self.home_dir, 'Downloads'), - 'documents': os.path.join(self.home_dir, 'Documents'), - 'pictures': os.path.join(self.home_dir, 'Pictures'), - 'music': os.path.join(self.home_dir, 'Music'), - 'videos': os.path.join(self.home_dir, 'Videos'), - 'wallpapers': os.path.join(self.data_dir, 'wallpapers'), - 'fonts': os.path.join(self.home_dir, '.fonts'), - 'cursors': os.path.join(self.home_dir, '.icons'), - 'icons': os.path.join(self.data_dir, 'icons'), - 'emoticons': os.path.join(self.data_dir, 'emoticons'), - 'themes': os.path.join(self.home_dir, '.themes'), - 'emerald_themes': os.path.join(self.home_dir, '.emerald', 'themes'), - 'enlightenment_themes': os.path.join(self.home_dir, '.e', 'e', 'themes'), - 'enlightenment_backgrounds': os.path.join(self.home_dir, '.e', 'e', 'backgrounds'), - 'fluxbox_styles': os.path.join(self.home_dir, '.fluxbox', 'styles'), - 'pekwm_themes': os.path.join(self.home_dir, '.pekwm', 'themes'), - 'icewm_themes': os.path.join(self.home_dir, '.icewm', 'themes'), - 'plasma_plasmoids': os.path.join(self.data_dir, 'plasma', 'plasmoids'), - 'plasma_look_and_feel': os.path.join(self.data_dir, 'plasma', 'look-and-feel'), - 'plasma_desktopthemes': os.path.join(self.data_dir, 'plasma', 'desktoptheme'), - 'kwin_effects': os.path.join(self.data_dir, 'kwin', 'effects'), - 'kwin_scripts': os.path.join(self.data_dir, 'kwin', 'scripts'), - 'kwin_tabbox': os.path.join(self.data_dir, 'kwin', 'tabbox'), - 'aurorae_themes': os.path.join(self.data_dir, 'aurorae', 'themes'), - 'dekorator_themes': os.path.join(self.data_dir, 'deKorator', 'themes'), - 'qtcurve': os.path.join(self.data_dir, 'QtCurve'), - 'color_schemes': os.path.join(self.data_dir, 'color-schemes'), - 'gnome_shell_extensions': os.path.join(self.data_dir, 'gnome-shell', 'extensions'), - 'cinnamon_applets': os.path.join(self.data_dir, 'cinnamon', 'applets'), - 'cinnamon_desklets': os.path.join(self.data_dir, 'cinnamon', 'desklets'), - 'cinnamon_extensions': os.path.join(self.data_dir, 'cinnamon', 'extensions'), - 'nautilus_scripts': os.path.join(self.data_dir, 'nautilus', 'scripts'), - 'amarok_scripts': os.path.join(self.kde_data_dir, 'apps', 'amarok', 'scripts'), - 'yakuake_skins': os.path.join(self.kde_data_dir, 'apps', 'yakuake', 'skins'), - 'cairo_clock_themes': os.path.join(self.home_dir, '.cairo-clock', 'themes') - } - self.destinations_alias = { - 'gnome_shell_themes': self.destinations['themes'], - 'cinnamon_themes': self.destinations['themes'], - 'gtk2_themes': self.destinations['themes'], - 'gtk3_themes': self.destinations['themes'], - 'metacity_themes': self.destinations['themes'], - 'xfwm4_themes': self.destinations['themes'], - 'openbox_themes': self.destinations['themes'], - 'kvantum_themes': self.destinations['themes'], - 'compiz_themes': self.destinations['emerald_themes'], - 'beryl_themes': self.destinations['emerald_themes'], - 'plasma4_plasmoids': self.destinations['plasma_plasmoids'], - 'plasma5_plasmoids': self.destinations['plasma_plasmoids'], - 'plasma5_look_and_feel': self.destinations['plasma_look_and_feel'], - 'plasma5_desktopthemes': self.destinations['plasma_desktopthemes'], - 'plasma_color_schemes': self.destinations['color_schemes'] - } - self.destinations.update(self.destinations_alias) - - self.archive_types = { - 'tar': [ - 'application/x-tar', - 'application/x-gzip', - 'application/gzip', - 'application/x-bzip', - 'application/x-bzip2', - 'application/x-xz', - 'application/x-lzma', - 'application/x-lzip', - 'application/x-compressed-tar', - 'application/x-bzip-compressed-tar', - 'application/x-bzip2-compressed-tar', - 'application/x-xz-compressed-tar', - 'application/x-lzma-compressed-tar', - 'application/x-lzip-compressed-tar' - ], - 'zip': ['application/zip'], - '7z': ['application/x-7z-compressed'], - 'rar': [ - 'application/x-rar', - 'application/x-rar-compressed' - ] - } - - #self.config.update(self.read_config('config')) - #self.destinations.update(self.read_config('destinations')) - - def read_config(self, name): - path = os.path.join(self.config_dir, name + '.json') - data = {} - if os.path.isfile(path): - f = open(path, 'r') - data = json.load(f) - f.close() - return data - - def write_config(self, name, data): - if not os.path.isdir(self.config_dir): - os.makedirs(self.config_dir) - path = os.path.join(self.config_dir, name + '.json') - f = open(path, 'w') - json.dump(data, f) - f.close() - - def parse(self): - meta = { - 'command': 'download', - 'url': '', - 'type': 'downloads', - 'filename': '' - } - - if sys.version_info.major >= 3: - parse_result = urllib.parse.urlparse(self.xdg_url) - query = urllib.parse.parse_qs(parse_result.query) - else: - parse_result = urlparse.urlparse(self.xdg_url) - query = urlparse.parse_qs(parse_result.query) - - if parse_result.netloc: - meta['command'] = parse_result.netloc - - if 'url' in query and query['url'][0]: - if sys.version_info.major >= 3: - meta['url'] = urllib.parse.unquote(query['url'][0]) - else: - meta['url'] = urllib.unquote(query['url'][0]) - - if 'type' in query and query['type'][0]: - if sys.version_info.major >= 3: - meta['type'] = urllib.parse.unquote(query['type'][0]) - else: - meta['type'] = urllib.unquote(query['type'][0]) - - if 'filename' in query and query['filename'][0]: - if sys.version_info.major >= 3: - meta['filename'] = urllib.parse.unquote(query['filename'][0]).split('?')[0] - else: - meta['filename'] = urllib.unquote(query['filename'][0]).split('?')[0] - - if meta['url'] and not meta['filename']: - meta['filename'] = os.path.basename(meta['url']).split('?')[0] - - return meta - - """ - def detect_desktop_environment(self): - desktop_environment = 'unknown' - if os.environ.get('KDE_FULL_SESSION') == 'true': - desktop_environment = 'kde' - elif os.environ.get('GNOME_DESKTOP_SESSION_ID'): - desktop_environment = 'gnome' - return desktop_environment - """ - - def install_plasmapkg(self, path, type='plasmoid'): - status = subprocess.call(['plasmapkg2', '-t', type, '-i', path]) - if status == 0: - return True - return False - - def uncompress_archive(self, path, target_dir): - (mimetype, encoding) = mimetypes.guess_type(path) - status = None - - if mimetype in self.archive_types['tar']: - status = subprocess.call(['tar', '-xf', path, '-C', target_dir]) - elif mimetype in self.archive_types['zip']: - status = subprocess.call(['unzip', '-o', path, '-d', target_dir]) - elif mimetype in self.archive_types['7z']: - # No space between -o and directory - status = subprocess.call(['7z', 'x', path, '-o' + target_dir]) - elif mimetype in self.archive_types['rar']: - status = subprocess.call(['unrar', 'e', path, target_dir]) - - if status == 0: - return True - return False - - def download(self): - url = self.meta['url'] - type = self.meta['type'] - filename = self.meta['filename'] - destination = self.destinations[type] - - temp_path = os.path.join(self.temp_dir, filename) - path = os.path.join(destination, filename) - - print('Retrieving a file from ' + url) - if sys.version_info.major >= 3: - urllib.request.urlretrieve(url, temp_path) - else: - urllib.urlretrieve(url, temp_path) - - print('Creating a directory ' + destination) - if not os.path.isdir(destination): - os.makedirs(destination) - - print('Saving a file to ' + path) - shutil.move(temp_path, path) - - print('Done') - - def install(self): - url = self.meta['url'] - type = self.meta['type'] - filename = self.meta['filename'] - destination = self.destinations[type] - - temp_path = os.path.join(self.temp_dir, filename) - path = os.path.join(destination, filename) - - print('Retrieving a file from ' + url) - if sys.version_info.major >= 3: - urllib.request.urlretrieve(url, temp_path) - else: - urllib.urlretrieve(url, temp_path) - - print('Creating a directory ' + destination) - if not os.path.isdir(destination): - os.makedirs(destination) - - print('Installing') - if (type in ['plasma_plasmoids', 'plasma4_plasmoids', 'plasma5_plasmoids'] - and self.install_plasmapkg(temp_path, 'plasmoid')): - print('The plasmoid has been installed') - os.remove(temp_path) - elif (type in ['plasma_look_and_feel', 'plasma5_look_and_feel'] - and self.install_plasmapkg(temp_path, 'lookandfeel')): - print('The plasma look and feel has been installed') - os.remove(temp_path) - elif (type in ['plasma_desktopthemes', 'plasma5_desktopthemes'] - and self.install_plasmapkg(temp_path, 'theme')): - print('The plasma desktop theme has been installed') - os.remove(temp_path) - elif (type == 'kwin_effects' - and self.install_plasmapkg(temp_path, 'kwineffect')): - print('The KWin effect has been installed') - os.remove(temp_path) - elif (type == 'kwin_scripts' - and self.install_plasmapkg(temp_path, 'kwinscript')): - print('The KWin script has been installed') - os.remove(temp_path) - elif (type == 'kwin_tabbox' - and self.install_plasmapkg(temp_path, 'windowswitcher')): - print('The KWin window switcher has been installed') - os.remove(temp_path) - elif self.uncompress_archive(temp_path, destination): - print('The archive file has been uncompressed into ' + destination) - os.remove(temp_path) - else: - print('Saving a file to ' + path) - shutil.move(temp_path, path) - - print('Done') - - def execute(self): - if (self.meta['command'] in ['download', 'install'] - and self.meta['url'] - and self.meta['type'] in self.destinations - and self.meta['filename']): - if self.meta['command'] == 'download': - self.download() - elif self.meta['command'] == 'install': - self.install() - else: - raise Exception('Incorrect XDG-URL ' + self.xdg_url) - -def main(): - program = 'xdgurl' - version = '1.0.1' - - parser = argparse.ArgumentParser( - prog=program, - description='An install helper program for desktop stuff', - epilog='Check more information on https://github.com/xdgurl/xdgurl' - ) - parser.add_argument( - '-v', '--version', - action='version', - version='%(prog)s ' + version - ) - parser.add_argument('xdg_url', help='XDG-URL') - args = parser.parse_args() - - if args.xdg_url: - core = XdgUrl(args.xdg_url) - if sys.version_info.major >= 3: - window = tkinter.Tk() - else: - window = Tkinter.Tk() - window.withdraw() - - info_text = 'Download: ' + core.meta['filename'] + '\nFrom: ' + core.meta['url'] - if core.meta['command'] == 'install': - info_text = 'Install: ' + core.meta['filename'] + '\nFrom: ' + core.meta['url'] - - message = 'Do you want to continue?' - - print(info_text) - if sys.version_info.major >= 3: - confirm = tkinter.messagebox.askyesno(program, info_text + '\n\n' + message) - else: - confirm = tkMessageBox.askyesno(program, info_text + '\n\n' + message) - - if confirm: - try: - core.execute(); - except Exception as e: - message = 'Download failed' - if core.meta['command'] == 'install': - message = 'Installation failed' - - print(message) - if sys.version_info.major >= 3: - tkinter.messagebox.showerror(program, info_text + '\n\n' + message + '\n' + str(e)) - else: - tkMessageBox.showerror(program, info_text + '\n\n' + message + '\n' + str(e)) - - return str(e) # stderr and exit code 1 - else: - message = 'Download successfull' - if core.meta['command'] == 'install': - message = 'Installation successfull' - - print(message) - if sys.version_info.major >= 3: - tkinter.messagebox.showinfo(program, message) - else: - tkMessageBox.showinfo(program, message) - return 0 - -if __name__ == '__main__': - sys.exit(main()) diff --git a/xdgurl.pro b/xdgurl.pro new file mode 100644 index 0000000000000000000000000000000000000000..ed63a8fa24b3b45116d02a0aed7297f0eda78165 --- /dev/null +++ b/xdgurl.pro @@ -0,0 +1,54 @@ +TARGET = xdgurl + +TEMPLATE = app + +QT += \ + core \ + gui \ + widgets \ + qml \ + quick \ + svg \ + network + +CONFIG += c++11 + +SOURCES += \ + src/app/main.cpp \ + src/app/handlers/xdgurl.cpp \ + src/libs/utils/config.cpp \ + src/libs/utils/network.cpp \ + src/libs/utils/file.cpp \ + src/libs/utils/json.cpp \ + src/libs/utils/package.cpp + +HEADERS += \ + src/app/handlers/xdgurl.h \ + src/libs/utils/config.h \ + src/libs/utils/network.h \ + src/libs/utils/file.h \ + src/libs/utils/json.h \ + src/libs/utils/package.h + +RESOURCES += \ + src/app/configs/configs.qrc \ + src/app/qml/qml.qrc \ + src/desktop/desktop.qrc + +DISTFILES += \ + README.md \ + pkg/build.sh \ + pkg/ubuntu/debian/changelog \ + pkg/ubuntu/debian/compat \ + pkg/ubuntu/debian/control \ + pkg/ubuntu/debian/copyright \ + pkg/ubuntu/debian/rules \ + pkg/ubuntu/debian/xdgurl.install \ + pkg/fedora/xdgurl.spec \ + pkg/arch/PKGBUILD + +# Additional RPATH +#include(rpath.pri) + +# Deployment rules +include(deployment.pri)