Compare commits
17 Commits
v1.0.0-rc.
...
appimage
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
092fbf72b6 | ||
|
|
4d3e77bc9a | ||
|
|
6ef16b9625 | ||
|
|
ae57550b46 | ||
|
|
b62f1bab1f | ||
|
|
5f9211f98f | ||
|
|
cfbb4152c3 | ||
|
|
fa83e2d9a8 | ||
|
|
58ecbbc6f4 | ||
|
|
77300ed178 | ||
|
|
99df18ef89 | ||
|
|
6681c0591a | ||
|
|
8597416bb5 | ||
|
|
ce94b44997 | ||
|
|
2741e1afaa | ||
|
|
b2f5f8e7d0 | ||
|
|
c4c32b1615 |
@@ -1,4 +1,4 @@
|
||||
cmake_minimum_required(VERSION 3.0)
|
||||
cmake_minimum_required(VERSION 3.11)
|
||||
project(discord-screenaudio)
|
||||
|
||||
set(CMAKE_CXX_STANDARD 17)
|
||||
@@ -13,7 +13,7 @@ find_package(Qt5 CONFIG REQUIRED COMPONENTS
|
||||
WebEngineWidgets
|
||||
)
|
||||
|
||||
set(discord-screenaudio_SRC
|
||||
set(${CMAKE_PROJECT_NAME}_SRC
|
||||
src/main.cpp
|
||||
src/mainwindow.cpp
|
||||
src/virtmic.cpp
|
||||
@@ -43,9 +43,12 @@ endif()
|
||||
|
||||
add_subdirectory(submodules/rohrkabel)
|
||||
|
||||
add_executable(discord-screenaudio ${discord-screenaudio_SRC})
|
||||
add_executable(${CMAKE_PROJECT_NAME} ${${CMAKE_PROJECT_NAME}_SRC})
|
||||
|
||||
target_link_libraries(discord-screenaudio Qt5::Widgets Qt5::WebEngineWidgets rohrkabel)
|
||||
target_link_libraries(${CMAKE_PROJECT_NAME} Qt5::Widgets Qt5::WebEngineWidgets rohrkabel)
|
||||
|
||||
install(TARGETS discord-screenaudio DESTINATION bin)
|
||||
install(PROGRAMS assets/discord-screenaudio.desktop DESTINATION ${CMAKE_INSTALL_PREFIX}/share/applications)
|
||||
install(TARGETS ${CMAKE_PROJECT_NAME} DESTINATION bin)
|
||||
install(PROGRAMS assets/${CMAKE_PROJECT_NAME}.desktop DESTINATION share/applications)
|
||||
install(FILES assets/discord.png DESTINATION share/pixmaps)
|
||||
|
||||
add_custom_target(appimage ${CMAKE_SOURCE_DIR}/scripts/build-appimage.sh WORKING_DIRECTORY ${CMAKE_SOURCE_DIR})
|
||||
|
||||
18
README.md
18
README.md
@@ -9,11 +9,18 @@ of [@edisionnano](https://github.com/edisionnano) and the
|
||||
|
||||

|
||||
|
||||
The purpose of this project is **not** to provide an alternative to the original
|
||||
Discord client. Rather, it should be used in addition to the original client in
|
||||
case you want to stream something, maybe used with a second account. For
|
||||
anything else, this client has way too many things that work less well than in
|
||||
the original client.
|
||||
|
||||
## Known Issues
|
||||
|
||||
- Only works with **PipeWire**
|
||||
- Only works on **X11**
|
||||
- Can only share primary screen (no other screens or specific applications) (see [#1](https://github.com/maltejur/discord-screenaudio/issues/1))
|
||||
- Can only share primary screen (no other screens or specific applications) (see
|
||||
[#1](https://github.com/maltejur/discord-screenaudio/issues/1))
|
||||
|
||||
## Installation
|
||||
|
||||
@@ -36,7 +43,14 @@ On Debian:
|
||||
|
||||
### Building
|
||||
|
||||
To build the program, run this in the source directory:
|
||||
First, clone the repository:
|
||||
|
||||
```bash
|
||||
git clone https://github.com/maltejur/discord-screenaudio.git
|
||||
cd discord-screenaudio
|
||||
```
|
||||
|
||||
Then, to build the program, run this in the source directory:
|
||||
|
||||
```bash
|
||||
cmake -B build
|
||||
|
||||
@@ -4,3 +4,4 @@ Name=discord-screenaudio
|
||||
Exec=discord-screenaudio
|
||||
Icon=discord
|
||||
Terminal=false
|
||||
Categories=Audio;
|
||||
|
||||
BIN
assets/discord.png
Normal file
BIN
assets/discord.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 6.8 KiB |
@@ -117,7 +117,7 @@ setInterval(() => {
|
||||
) {
|
||||
for (const el of document.getElementsByClassName("info-3pQQBb")) {
|
||||
const aboutEl = document.createElement("div");
|
||||
aboutEl.innerText = "discord-screenaudio v1.0.0-rc.6";
|
||||
aboutEl.innerText = "discord-screenaudio v1.0.0-rc.7 (AppImage)";
|
||||
aboutEl.style.fontSize = "12px";
|
||||
aboutEl.style.color = "var(--text-muted)";
|
||||
aboutEl.classList.add("dirscordScreenaudioAboutText");
|
||||
@@ -129,3 +129,7 @@ setInterval(() => {
|
||||
document.getElementById("manage-streams-change-windows")?.remove();
|
||||
document.querySelector(`[aria-label="Stream Settings"]`)?.remove();
|
||||
}, 500);
|
||||
|
||||
// Fix for broken discord notifications after restart
|
||||
// (https://github.com/maltejur/discord-screenaudio/issues/17)
|
||||
Notification.requestPermission();
|
||||
|
||||
4
scripts/Dockerfile
Normal file
4
scripts/Dockerfile
Normal file
@@ -0,0 +1,4 @@
|
||||
FROM debian:11
|
||||
ADD ./prepare-container.sh /tmp/prepare-container.sh
|
||||
RUN /tmp/prepare-container.sh
|
||||
WORKDIR /work
|
||||
32
scripts/build-appimage.sh
Executable file
32
scripts/build-appimage.sh
Executable file
@@ -0,0 +1,32 @@
|
||||
#!/usr/bin/bash
|
||||
|
||||
set -e
|
||||
|
||||
# Check if inside of docker container
|
||||
if [ ! -f /.dockerenv ]; then
|
||||
# Check if docker is available
|
||||
if ! command -v docker >/dev/null 2>&1; then
|
||||
echo "Error: Docker is not available."
|
||||
exit 1
|
||||
fi
|
||||
docker run --rm -u $(id -u) -v "$PWD":/work discord-screenaudio-buildenv bash /work/scripts/build-appimage.sh
|
||||
exit 0
|
||||
fi
|
||||
|
||||
tmpdir="$(mktemp -d)"
|
||||
builddir="$tmpdir/build"
|
||||
appdir="$tmpdir/AppDir"
|
||||
export CMAKE_GENERATOR="Ninja"
|
||||
cmake -B "$builddir" -S "$PWD"
|
||||
cmake --build "$builddir" --config Release
|
||||
DESTDIR="$appdir" cmake --install "$builddir" --prefix "/usr"
|
||||
# Include libnss related files
|
||||
mkdir -p "$appdir/AppDir/usr/lib/"
|
||||
cp -rv "/usr/lib/x86_64-linux-gnu/nss" "$appdir/usr/lib/"
|
||||
|
||||
VERSION="$(cat version)" linuxdeploy \
|
||||
--appdir "$appdir" \
|
||||
--icon-file "assets/discord.png" \
|
||||
--plugin qt \
|
||||
--exclude-library "libpipewire-0.3.so.0" \
|
||||
--output appimage
|
||||
37
scripts/prepare-container.sh
Executable file
37
scripts/prepare-container.sh
Executable file
@@ -0,0 +1,37 @@
|
||||
#!/bin/sh
|
||||
|
||||
set -e
|
||||
cd "$(dirname "$0")"
|
||||
|
||||
if [ ! -f /.dockerenv ]; then
|
||||
# Check if docker is available
|
||||
if ! command -v docker >/dev/null 2>&1; then
|
||||
echo "Error: Docker is not available."
|
||||
exit 1
|
||||
fi
|
||||
docker build -t discord-screenaudio-buildenv .
|
||||
exit 0
|
||||
fi
|
||||
|
||||
echo "-> Installing dependencies with apt..."
|
||||
export DEBIAN_FRONTEND=noninteractive
|
||||
apt-get update
|
||||
apt-get install -y curl build-essential qtbase5-dev qtwebengine5-dev qt5-qmake cmake ninja-build pkg-config git libpipewire-0.3-dev file
|
||||
|
||||
echo "-> Installing linuxdeploy..."
|
||||
tmpdir="$(mktemp -d)"
|
||||
install_appimage() {
|
||||
curl -Lo "$tmpdir/$1.AppImage" "$2"
|
||||
chmod +x "$tmpdir/$1.AppImage"
|
||||
sed '0,/AI\x02/{s|AI\x02|\x00\x00\x00|}' -i -i "$tmpdir/$1.AppImage"
|
||||
(cd "$tmpdir" && ./$1.AppImage --appimage-extract)
|
||||
mv -v "$tmpdir/squashfs-root" "/opt/$1"
|
||||
ln -sv "/opt/$1/AppRun" "/usr/local/bin/$1"
|
||||
}
|
||||
install_appimage "linuxdeploy" "https://github.com/linuxdeploy/linuxdeploy/releases/download/continuous/linuxdeploy-x86_64.AppImage"
|
||||
install_appimage "linuxdeploy-plugin-qt" "https://github.com/linuxdeploy/linuxdeploy-plugin-qt/releases/download/continuous/linuxdeploy-plugin-qt-x86_64.AppImage"
|
||||
|
||||
echo "-> Cleaning up..."
|
||||
rm -rf "$tmpdir"
|
||||
rm /tmp/prepare-container.sh
|
||||
apt-get clean
|
||||
@@ -26,6 +26,7 @@ DiscordPage::DiscordPage(QWidget *parent) : QWebEnginePage(parent) {
|
||||
settings()->setAttribute(QWebEngineSettings::FullScreenSupportEnabled, true);
|
||||
settings()->setAttribute(QWebEngineSettings::PlaybackRequiresUserGesture,
|
||||
false);
|
||||
settings()->setAttribute(QWebEngineSettings::JavascriptCanOpenWindows, false);
|
||||
|
||||
setUrl(QUrl("https://discord.com/app"));
|
||||
|
||||
@@ -36,7 +37,7 @@ DiscordPage::DiscordPage(QWidget *parent) : QWebEnginePage(parent) {
|
||||
}
|
||||
|
||||
void DiscordPage::injectScript(QString source) {
|
||||
qDebug() << "[main ] Injecting " << source;
|
||||
qDebug() << "[main ] Injecting" << source;
|
||||
|
||||
QFile userscript(source);
|
||||
|
||||
@@ -77,14 +78,25 @@ void DiscordPage::featurePermissionRequested(const QUrl &securityOrigin,
|
||||
bool DiscordPage::acceptNavigationRequest(const QUrl &url,
|
||||
QWebEnginePage::NavigationType type,
|
||||
bool isMainFrame) {
|
||||
qDebug() << url;
|
||||
if (type == QWebEnginePage::NavigationTypeLinkClicked) {
|
||||
QDesktopServices::openUrl(url);
|
||||
return false;
|
||||
}
|
||||
// if (type == QWebEnginePage::NavigationTypeLinkClicked) {
|
||||
// QDesktopServices::openUrl(url);
|
||||
// return false;
|
||||
// }
|
||||
return true;
|
||||
};
|
||||
|
||||
bool ExternalPage::acceptNavigationRequest(const QUrl &url,
|
||||
QWebEnginePage::NavigationType type,
|
||||
bool isMainFrame) {
|
||||
QDesktopServices::openUrl(url);
|
||||
deleteLater();
|
||||
return false;
|
||||
}
|
||||
|
||||
QWebEnginePage *DiscordPage::createWindow(QWebEnginePage::WebWindowType type) {
|
||||
return new ExternalPage;
|
||||
}
|
||||
|
||||
void DiscordPage::stopVirtmic() {
|
||||
if (m_virtmicProcess.state() == QProcess::Running) {
|
||||
qDebug() << "[virtmic] Stopping Virtmic";
|
||||
|
||||
@@ -4,6 +4,7 @@
|
||||
#include "virtmic.h"
|
||||
|
||||
#include <QProcess>
|
||||
#include <QWebEngineFullScreenRequest>
|
||||
#include <QWebEnginePage>
|
||||
|
||||
class DiscordPage : public QWebEnginePage {
|
||||
@@ -18,6 +19,7 @@ private:
|
||||
bool acceptNavigationRequest(const QUrl &url,
|
||||
QWebEnginePage::NavigationType type,
|
||||
bool isMainFrame) override;
|
||||
QWebEnginePage *createWindow(QWebEnginePage::WebWindowType type) override;
|
||||
void
|
||||
javaScriptConsoleMessage(QWebEnginePage::JavaScriptConsoleMessageLevel level,
|
||||
const QString &message, int lineNumber,
|
||||
@@ -31,3 +33,15 @@ private Q_SLOTS:
|
||||
QWebEnginePage::Feature feature);
|
||||
void startStream(QString target, uint width, uint height, uint frameRate);
|
||||
};
|
||||
|
||||
// Will immediately get destroyed again but is needed for navigation to
|
||||
// target="_blank" links, since QWebEnginePage::newWindowRequested is
|
||||
// only available sinec Qt 6.3.
|
||||
class ExternalPage : public QWebEnginePage {
|
||||
Q_OBJECT
|
||||
|
||||
private:
|
||||
bool acceptNavigationRequest(const QUrl &url,
|
||||
QWebEnginePage::NavigationType type,
|
||||
bool isMainFrame) override;
|
||||
};
|
||||
|
||||
@@ -7,7 +7,7 @@
|
||||
int main(int argc, char *argv[]) {
|
||||
QApplication app(argc, argv);
|
||||
QApplication::setApplicationName("discord-screenaudio");
|
||||
QApplication::setApplicationVersion("1.0.0-rc.6");
|
||||
QApplication::setApplicationVersion("1.0.0-rc.7");
|
||||
|
||||
QCommandLineParser parser;
|
||||
parser.setApplicationDescription(
|
||||
|
||||
@@ -23,8 +23,25 @@ MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent) {
|
||||
}
|
||||
|
||||
void MainWindow::setupWebView() {
|
||||
m_webView = new QWebEngineView(this);
|
||||
auto page = new DiscordPage(this);
|
||||
connect(page, &QWebEnginePage::fullScreenRequested, this,
|
||||
&MainWindow::fullScreenRequested);
|
||||
|
||||
m_webView = new QWebEngineView(this);
|
||||
m_webView->setPage(page);
|
||||
|
||||
setCentralWidget(m_webView);
|
||||
}
|
||||
|
||||
void MainWindow::fullScreenRequested(
|
||||
QWebEngineFullScreenRequest fullScreenRequest) {
|
||||
fullScreenRequest.accept();
|
||||
if (fullScreenRequest.toggleOn()) {
|
||||
m_wasMaximized = isMaximized();
|
||||
showFullScreen();
|
||||
} else {
|
||||
m_wasMaximized ? showMaximized() : showNormal();
|
||||
}
|
||||
}
|
||||
|
||||
void MainWindow::closeEvent(QCloseEvent *event) { QApplication::quit(); }
|
||||
|
||||
@@ -21,4 +21,9 @@ private:
|
||||
QWebEngineView *m_webView;
|
||||
QWebEngineProfile *prepareProfile();
|
||||
DiscordPage *m_discordPage;
|
||||
void closeEvent(QCloseEvent *event) override;
|
||||
bool m_wasMaximized;
|
||||
|
||||
private Q_SLOTS:
|
||||
void fullScreenRequested(QWebEngineFullScreenRequest fullScreenRequest);
|
||||
};
|
||||
|
||||
Reference in New Issue
Block a user