mirror of
https://github.com/bggRGjQaUbCoE/PiliPlus.git
synced 2026-04-19 19:01:24 +08:00
allow user to configure window decorations(CSD/SSD) (#1839)
* allow user to configure window decorations(CSD/SSD) * rename piliplus.desktop to com.example.piliplus.desktop * update --------- Co-authored-by: dom <githubaccount56556@proton.me>
This commit is contained in:
16
.github/workflows/linux_x64.yml
vendored
16
.github/workflows/linux_x64.yml
vendored
@@ -80,7 +80,7 @@ jobs:
|
|||||||
printf "复制文件...\n"
|
printf "复制文件...\n"
|
||||||
cp -r ../build/linux/x64/release/bundle/* opt/PiliPlus
|
cp -r ../build/linux/x64/release/bundle/* opt/PiliPlus
|
||||||
cp -r ../assets/linux/DEBIAN .
|
cp -r ../assets/linux/DEBIAN .
|
||||||
cp ../assets/linux/piliplus.desktop usr/share/applications
|
cp ../assets/linux/com.example.piliplus.desktop usr/share/applications
|
||||||
cp ../assets/images/logo/logo.png usr/share/icons/hicolor/512x512/apps/piliplus.png
|
cp ../assets/images/logo/logo.png usr/share/icons/hicolor/512x512/apps/piliplus.png
|
||||||
|
|
||||||
printf "修改控制文件...\n"
|
printf "修改控制文件...\n"
|
||||||
@@ -122,7 +122,7 @@ jobs:
|
|||||||
SRC_DIR="$PWD/piliplus-${{ env.version }}"
|
SRC_DIR="$PWD/piliplus-${{ env.version }}"
|
||||||
mkdir -p "$SRC_DIR/bundle" "$SRC_DIR/assets"
|
mkdir -p "$SRC_DIR/bundle" "$SRC_DIR/assets"
|
||||||
cp -r build/linux/x64/release/bundle/* "$SRC_DIR/bundle/"
|
cp -r build/linux/x64/release/bundle/* "$SRC_DIR/bundle/"
|
||||||
cp assets/linux/piliplus.desktop "$SRC_DIR/assets/piliplus.desktop"
|
cp assets/linux/com.example.piliplus.desktop "$SRC_DIR/assets/com.example.piliplus.desktop"
|
||||||
cp assets/images/logo/logo.png "$SRC_DIR/assets/piliplus.png"
|
cp assets/images/logo/logo.png "$SRC_DIR/assets/piliplus.png"
|
||||||
tar -zcvf "$RPM_BUILD_ROOT/SOURCES/piliplus-${{ env.version }}.tar.gz" -C "$PWD" "piliplus-${{ env.version }}"
|
tar -zcvf "$RPM_BUILD_ROOT/SOURCES/piliplus-${{ env.version }}.tar.gz" -C "$PWD" "piliplus-${{ env.version }}"
|
||||||
|
|
||||||
@@ -155,7 +155,7 @@ jobs:
|
|||||||
|
|
||||||
# 桌面集成
|
# 桌面集成
|
||||||
mkdir -p %{buildroot}/usr/share/applications
|
mkdir -p %{buildroot}/usr/share/applications
|
||||||
install -m 644 assets/piliplus.desktop %{buildroot}/usr/share/applications/piliplus.desktop
|
install -m 644 assets/com.example.piliplus.desktop %{buildroot}/usr/share/applications/com.example.piliplus.desktop
|
||||||
|
|
||||||
mkdir -p %{buildroot}/usr/share/icons/hicolor/512x512/apps
|
mkdir -p %{buildroot}/usr/share/icons/hicolor/512x512/apps
|
||||||
install -m 644 assets/piliplus.png %{buildroot}/usr/share/icons/hicolor/512x512/apps/piliplus.png
|
install -m 644 assets/piliplus.png %{buildroot}/usr/share/icons/hicolor/512x512/apps/piliplus.png
|
||||||
@@ -171,7 +171,7 @@ jobs:
|
|||||||
%files
|
%files
|
||||||
/opt/PiliPlus
|
/opt/PiliPlus
|
||||||
/usr/bin/piliplus
|
/usr/bin/piliplus
|
||||||
/usr/share/applications/piliplus.desktop
|
/usr/share/applications/com.example.piliplus.desktop
|
||||||
/usr/share/icons/hicolor/512x512/apps/piliplus.png
|
/usr/share/icons/hicolor/512x512/apps/piliplus.png
|
||||||
|
|
||||||
%changelog
|
%changelog
|
||||||
@@ -207,8 +207,8 @@ jobs:
|
|||||||
cp -r build/linux/x64/release/bundle/* "$APPDIR/usr/bin/"
|
cp -r build/linux/x64/release/bundle/* "$APPDIR/usr/bin/"
|
||||||
|
|
||||||
printf "复制桌面文件和图标...\n"
|
printf "复制桌面文件和图标...\n"
|
||||||
cp assets/linux/piliplus.desktop "$APPDIR/piliplus.desktop"
|
cp assets/linux/com.example.piliplus.desktop "$APPDIR/com.example.piliplus.desktop"
|
||||||
cp assets/linux/piliplus.desktop "$APPDIR/usr/share/applications/piliplus.desktop"
|
cp assets/linux/com.example.piliplus.desktop "$APPDIR/usr/share/applications/com.example.piliplus.desktop"
|
||||||
cp assets/images/logo/logo.png "$APPDIR/piliplus.png"
|
cp assets/images/logo/logo.png "$APPDIR/piliplus.png"
|
||||||
cp assets/images/logo/logo.png "$APPDIR/usr/share/icons/hicolor/512x512/apps/piliplus.png"
|
cp assets/images/logo/logo.png "$APPDIR/usr/share/icons/hicolor/512x512/apps/piliplus.png"
|
||||||
|
|
||||||
@@ -224,8 +224,8 @@ jobs:
|
|||||||
chmod +x "$APPDIR/AppRun"
|
chmod +x "$APPDIR/AppRun"
|
||||||
|
|
||||||
printf "修改桌面文件中的 Exec 路径...\n"
|
printf "修改桌面文件中的 Exec 路径...\n"
|
||||||
sed -i 's|Exec=piliplus|Exec=piliplus|g' "$APPDIR/piliplus.desktop"
|
sed -i 's|Exec=piliplus|Exec=piliplus|g' "$APPDIR/com.example.piliplus.desktop"
|
||||||
sed -i 's|Icon=piliplus|Icon=piliplus|g' "$APPDIR/piliplus.desktop"
|
sed -i 's|Icon=piliplus|Icon=piliplus|g' "$APPDIR/com.example.piliplus.desktop"
|
||||||
|
|
||||||
printf "打包 AppImage...\n"
|
printf "打包 AppImage...\n"
|
||||||
ARCH=x86_64 ./appimagetool-x86_64.AppImage "$APPDIR" "PiliPlus_linux_${{ env.version }}_amd64.AppImage"
|
ARCH=x86_64 ./appimagetool-x86_64.AppImage "$APPDIR" "PiliPlus_linux_${{ env.version }}_amd64.AppImage"
|
||||||
|
|||||||
@@ -56,6 +56,16 @@ List<SettingsModel> get styleSettings => [
|
|||||||
needReboot: true,
|
needReboot: true,
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
|
if (Platform.isLinux)
|
||||||
|
const SwitchModel(
|
||||||
|
title: '使用SSD(Server-Side Decoration)',
|
||||||
|
subtitle: '(Linux)强制使用Server-Side Decoration',
|
||||||
|
leading: Icon(Icons.web_asset),
|
||||||
|
setKey: SettingBoxKey.useSSD,
|
||||||
|
defaultVal: false,
|
||||||
|
needReboot: true,
|
||||||
|
onChanged: GStorage.syncToDisk,
|
||||||
|
),
|
||||||
SwitchModel(
|
SwitchModel(
|
||||||
title: '横屏适配',
|
title: '横屏适配',
|
||||||
subtitle: '启用横屏布局与逻辑,平板、折叠屏等可开启;建议全屏方向设为【不改变当前方向】',
|
subtitle: '启用横屏布局与逻辑,平板、折叠屏等可开启;建议全屏方向设为【不改变当前方向】',
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import 'dart:convert';
|
import 'dart:convert';
|
||||||
|
import 'dart:io';
|
||||||
|
|
||||||
import 'package:PiliPlus/models/model_owner.dart';
|
import 'package:PiliPlus/models/model_owner.dart';
|
||||||
import 'package:PiliPlus/models/user/danmaku_rule_adapter.dart';
|
import 'package:PiliPlus/models/user/danmaku_rule_adapter.dart';
|
||||||
@@ -61,6 +62,11 @@ abstract final class GStorage {
|
|||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static Future<File> syncToDisk([_]) {
|
||||||
|
final jsonPath = path.join(appSupportDirPath, 'settings.json');
|
||||||
|
return File(jsonPath).writeAsString(exportAllSettings());
|
||||||
|
}
|
||||||
|
|
||||||
static String exportAllSettings() {
|
static String exportAllSettings() {
|
||||||
return Utils.jsonEncoder.convert({
|
return Utils.jsonEncoder.convert({
|
||||||
setting.name: setting.toMap(),
|
setting.name: setting.toMap(),
|
||||||
|
|||||||
@@ -157,7 +157,8 @@ abstract final class SettingBoxKey {
|
|||||||
showWindowTitleBar = 'showWindowTitleBar',
|
showWindowTitleBar = 'showWindowTitleBar',
|
||||||
desktopVolume = 'desktopVolume',
|
desktopVolume = 'desktopVolume',
|
||||||
showTrayIcon = 'showTrayIcon',
|
showTrayIcon = 'showTrayIcon',
|
||||||
uiScale = 'uiScale';
|
uiScale = 'uiScale',
|
||||||
|
useSSD = 'useSSD';
|
||||||
|
|
||||||
static const String subtitlePreferenceV2 = 'subtitlePreferenceV2',
|
static const String subtitlePreferenceV2 = 'subtitlePreferenceV2',
|
||||||
enableDragSubtitle = 'enableDragSubtitle',
|
enableDragSubtitle = 'enableDragSubtitle',
|
||||||
|
|||||||
@@ -914,6 +914,9 @@ abstract final class Pref {
|
|||||||
static bool get showWindowTitleBar =>
|
static bool get showWindowTitleBar =>
|
||||||
_setting.get(SettingBoxKey.showWindowTitleBar, defaultValue: true);
|
_setting.get(SettingBoxKey.showWindowTitleBar, defaultValue: true);
|
||||||
|
|
||||||
|
static bool get useSSD =>
|
||||||
|
_setting.get(SettingBoxKey.useSSD, defaultValue: false);
|
||||||
|
|
||||||
static double get desktopVolume =>
|
static double get desktopVolume =>
|
||||||
_setting.get(SettingBoxKey.desktopVolume, defaultValue: 1.0);
|
_setting.get(SettingBoxKey.desktopVolume, defaultValue: 1.0);
|
||||||
|
|
||||||
|
|||||||
@@ -5,18 +5,21 @@
|
|||||||
#include <gdk/gdkx.h>
|
#include <gdk/gdkx.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include <fstream>
|
||||||
|
#include <iterator>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
#include "flutter/generated_plugin_registrant.h"
|
#include "flutter/generated_plugin_registrant.h"
|
||||||
|
|
||||||
struct _MyApplication {
|
struct _MyApplication {
|
||||||
GtkApplication parent_instance;
|
GtkApplication parent_instance;
|
||||||
char** dart_entrypoint_arguments;
|
char **dart_entrypoint_arguments;
|
||||||
};
|
};
|
||||||
|
|
||||||
G_DEFINE_TYPE(MyApplication, my_application, GTK_TYPE_APPLICATION)
|
G_DEFINE_TYPE(MyApplication, my_application, GTK_TYPE_APPLICATION)
|
||||||
|
|
||||||
// Called when first Flutter frame received.
|
// Called when first Flutter frame received.
|
||||||
static void first_frame_cb(MyApplication* self, FlView *view)
|
static void first_frame_cb(MyApplication *self, FlView *view) {
|
||||||
{
|
|
||||||
gtk_widget_show(gtk_widget_get_toplevel(GTK_WIDGET(view)));
|
gtk_widget_show(gtk_widget_get_toplevel(GTK_WIDGET(view)));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -33,9 +36,9 @@ static gboolean window_delete_event_cb(GtkWidget *widget, GdkEvent *event,
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Implements GApplication::activate.
|
// Implements GApplication::activate.
|
||||||
static void my_application_activate(GApplication* application) {
|
static void my_application_activate(GApplication *application) {
|
||||||
MyApplication* self = MY_APPLICATION(application);
|
MyApplication *self = MY_APPLICATION(application);
|
||||||
GtkWindow* window =
|
GtkWindow *window =
|
||||||
GTK_WINDOW(gtk_application_window_new(GTK_APPLICATION(application)));
|
GTK_WINDOW(gtk_application_window_new(GTK_APPLICATION(application)));
|
||||||
|
|
||||||
// Use a header bar when running in GNOME as this is the common style used
|
// Use a header bar when running in GNOME as this is the common style used
|
||||||
@@ -45,18 +48,40 @@ static void my_application_activate(GApplication* application) {
|
|||||||
// in case the window manager does more exotic layout, e.g. tiling.
|
// in case the window manager does more exotic layout, e.g. tiling.
|
||||||
// If running on Wayland assume the header bar will work (may need changing
|
// If running on Wayland assume the header bar will work (may need changing
|
||||||
// if future cases occur).
|
// if future cases occur).
|
||||||
gboolean use_header_bar = TRUE;
|
|
||||||
|
const gboolean use_header_bar = [window]() -> gboolean {
|
||||||
|
auto UseSSD = []() -> bool {
|
||||||
|
const gchar *config_root = g_get_user_data_dir();
|
||||||
|
gchar *full_path_c = g_build_filename(config_root, "com.example.piliplus",
|
||||||
|
"settings.json", NULL);
|
||||||
|
std::string config_path(full_path_c);
|
||||||
|
g_free(full_path_c);
|
||||||
|
std::ifstream f(config_path);
|
||||||
|
if (!f.is_open()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string content{std::istreambuf_iterator<char>(f), {}};
|
||||||
|
return (content.find("\"useSSD\": true,") != std::string::npos);
|
||||||
|
// user choose to use SSD
|
||||||
|
};
|
||||||
|
if (UseSSD())
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
#ifdef GDK_WINDOWING_X11
|
#ifdef GDK_WINDOWING_X11
|
||||||
GdkScreen* screen = gtk_window_get_screen(window);
|
GdkScreen *screen = gtk_window_get_screen(window);
|
||||||
if (GDK_IS_X11_SCREEN(screen)) {
|
if (GDK_IS_X11_SCREEN(screen)) {
|
||||||
const gchar* wm_name = gdk_x11_screen_get_window_manager_name(screen);
|
const gchar *wm_name = gdk_x11_screen_get_window_manager_name(screen);
|
||||||
if (g_strcmp0(wm_name, "GNOME Shell") != 0) {
|
if (g_strcmp0(wm_name, "GNOME Shell") != 0) {
|
||||||
use_header_bar = FALSE;
|
return FALSE;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
return TRUE;
|
||||||
|
}();
|
||||||
|
|
||||||
if (use_header_bar) {
|
if (use_header_bar) {
|
||||||
GtkHeaderBar* header_bar = GTK_HEADER_BAR(gtk_header_bar_new());
|
GtkHeaderBar *header_bar = GTK_HEADER_BAR(gtk_header_bar_new());
|
||||||
gtk_widget_show(GTK_WIDGET(header_bar));
|
gtk_widget_show(GTK_WIDGET(header_bar));
|
||||||
gtk_header_bar_set_title(header_bar, "piliplus");
|
gtk_header_bar_set_title(header_bar, "piliplus");
|
||||||
gtk_header_bar_set_show_close_button(header_bar, TRUE);
|
gtk_header_bar_set_show_close_button(header_bar, TRUE);
|
||||||
@@ -68,11 +93,13 @@ static void my_application_activate(GApplication* application) {
|
|||||||
gtk_window_set_default_size(window, 1280, 720);
|
gtk_window_set_default_size(window, 1280, 720);
|
||||||
|
|
||||||
g_autoptr(FlDartProject) project = fl_dart_project_new();
|
g_autoptr(FlDartProject) project = fl_dart_project_new();
|
||||||
fl_dart_project_set_dart_entrypoint_arguments(project, self->dart_entrypoint_arguments);
|
fl_dart_project_set_dart_entrypoint_arguments(
|
||||||
|
project, self->dart_entrypoint_arguments);
|
||||||
|
|
||||||
FlView* view = fl_view_new(project);
|
FlView *view = fl_view_new(project);
|
||||||
GdkRGBA background_color;
|
GdkRGBA background_color;
|
||||||
// Background defaults to black, override it here if necessary, e.g. #00000000 for transparent.
|
// Background defaults to black, override it here if necessary, e.g. #00000000
|
||||||
|
// for transparent.
|
||||||
gdk_rgba_parse(&background_color, "#000000");
|
gdk_rgba_parse(&background_color, "#000000");
|
||||||
fl_view_set_background_color(view, &background_color);
|
fl_view_set_background_color(view, &background_color);
|
||||||
gtk_widget_show(GTK_WIDGET(view));
|
gtk_widget_show(GTK_WIDGET(view));
|
||||||
@@ -80,11 +107,13 @@ static void my_application_activate(GApplication* application) {
|
|||||||
|
|
||||||
// Show the window when Flutter renders.
|
// Show the window when Flutter renders.
|
||||||
// Requires the view to be realized so we can start rendering.
|
// Requires the view to be realized so we can start rendering.
|
||||||
g_signal_connect_swapped(view, "first-frame", G_CALLBACK(first_frame_cb), self);
|
g_signal_connect_swapped(view, "first-frame", G_CALLBACK(first_frame_cb),
|
||||||
|
self);
|
||||||
gtk_widget_realize(GTK_WIDGET(view));
|
gtk_widget_realize(GTK_WIDGET(view));
|
||||||
|
|
||||||
// Connect the delete-event signal to handle window close.
|
// Connect the delete-event signal to handle window close.
|
||||||
g_signal_connect(window, "delete-event", G_CALLBACK(window_delete_event_cb), NULL);
|
g_signal_connect(window, "delete-event", G_CALLBACK(window_delete_event_cb),
|
||||||
|
NULL);
|
||||||
|
|
||||||
fl_register_plugins(FL_PLUGIN_REGISTRY(view));
|
fl_register_plugins(FL_PLUGIN_REGISTRY(view));
|
||||||
|
|
||||||
@@ -92,16 +121,18 @@ static void my_application_activate(GApplication* application) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Implements GApplication::local_command_line.
|
// Implements GApplication::local_command_line.
|
||||||
static gboolean my_application_local_command_line(GApplication* application, gchar*** arguments, int* exit_status) {
|
static gboolean my_application_local_command_line(GApplication *application,
|
||||||
MyApplication* self = MY_APPLICATION(application);
|
gchar ***arguments,
|
||||||
|
int *exit_status) {
|
||||||
|
MyApplication *self = MY_APPLICATION(application);
|
||||||
// Strip out the first argument as it is the binary name.
|
// Strip out the first argument as it is the binary name.
|
||||||
self->dart_entrypoint_arguments = g_strdupv(*arguments + 1);
|
self->dart_entrypoint_arguments = g_strdupv(*arguments + 1);
|
||||||
|
|
||||||
g_autoptr(GError) error = nullptr;
|
g_autoptr(GError) error = nullptr;
|
||||||
if (!g_application_register(application, nullptr, &error)) {
|
if (!g_application_register(application, nullptr, &error)) {
|
||||||
g_warning("Failed to register: %s", error->message);
|
g_warning("Failed to register: %s", error->message);
|
||||||
*exit_status = 1;
|
*exit_status = 1;
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
g_application_activate(application);
|
g_application_activate(application);
|
||||||
@@ -111,8 +142,8 @@ static gboolean my_application_local_command_line(GApplication* application, gch
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Implements GApplication::startup.
|
// Implements GApplication::startup.
|
||||||
static void my_application_startup(GApplication* application) {
|
static void my_application_startup(GApplication *application) {
|
||||||
//MyApplication* self = MY_APPLICATION(object);
|
// MyApplication* self = MY_APPLICATION(object);
|
||||||
|
|
||||||
// Perform any actions required at application startup.
|
// Perform any actions required at application startup.
|
||||||
|
|
||||||
@@ -120,8 +151,8 @@ static void my_application_startup(GApplication* application) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Implements GApplication::shutdown.
|
// Implements GApplication::shutdown.
|
||||||
static void my_application_shutdown(GApplication* application) {
|
static void my_application_shutdown(GApplication *application) {
|
||||||
//MyApplication* self = MY_APPLICATION(object);
|
// MyApplication* self = MY_APPLICATION(object);
|
||||||
|
|
||||||
// Perform any actions required at application shutdown.
|
// Perform any actions required at application shutdown.
|
||||||
|
|
||||||
@@ -129,30 +160,30 @@ static void my_application_shutdown(GApplication* application) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Implements GObject::dispose.
|
// Implements GObject::dispose.
|
||||||
static void my_application_dispose(GObject* object) {
|
static void my_application_dispose(GObject *object) {
|
||||||
MyApplication* self = MY_APPLICATION(object);
|
MyApplication *self = MY_APPLICATION(object);
|
||||||
g_clear_pointer(&self->dart_entrypoint_arguments, g_strfreev);
|
g_clear_pointer(&self->dart_entrypoint_arguments, g_strfreev);
|
||||||
G_OBJECT_CLASS(my_application_parent_class)->dispose(object);
|
G_OBJECT_CLASS(my_application_parent_class)->dispose(object);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void my_application_class_init(MyApplicationClass* klass) {
|
static void my_application_class_init(MyApplicationClass *klass) {
|
||||||
G_APPLICATION_CLASS(klass)->activate = my_application_activate;
|
G_APPLICATION_CLASS(klass)->activate = my_application_activate;
|
||||||
G_APPLICATION_CLASS(klass)->local_command_line = my_application_local_command_line;
|
G_APPLICATION_CLASS(klass)->local_command_line =
|
||||||
|
my_application_local_command_line;
|
||||||
G_APPLICATION_CLASS(klass)->startup = my_application_startup;
|
G_APPLICATION_CLASS(klass)->startup = my_application_startup;
|
||||||
G_APPLICATION_CLASS(klass)->shutdown = my_application_shutdown;
|
G_APPLICATION_CLASS(klass)->shutdown = my_application_shutdown;
|
||||||
G_OBJECT_CLASS(klass)->dispose = my_application_dispose;
|
G_OBJECT_CLASS(klass)->dispose = my_application_dispose;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void my_application_init(MyApplication* self) {}
|
static void my_application_init(MyApplication *self) {}
|
||||||
|
|
||||||
MyApplication* my_application_new() {
|
MyApplication *my_application_new() {
|
||||||
// Set the program name to the application ID, which helps various systems
|
// Set the program name to the application ID, which helps various systems
|
||||||
// like GTK and desktop environments map this running application to its
|
// like GTK and desktop environments map this running application to its
|
||||||
// corresponding .desktop file. This ensures better integration by allowing
|
// corresponding .desktop file. This ensures better integration by allowing
|
||||||
// the application to be recognized beyond its binary name.
|
// the application to be recognized beyond its binary name.
|
||||||
g_set_prgname(APPLICATION_ID);
|
g_set_prgname(APPLICATION_ID);
|
||||||
|
|
||||||
return MY_APPLICATION(g_object_new(my_application_get_type(),
|
return MY_APPLICATION(g_object_new(
|
||||||
"application-id", APPLICATION_ID,
|
my_application_get_type(), "application-id", APPLICATION_ID, nullptr));
|
||||||
nullptr));
|
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user