mirror of
https://github.com/ZoiteChat/zoitechat.git
synced 2026-06-11 17:30:18 +00:00
Compare commits
57 Commits
20b552ad9d
...
notificati
| Author | SHA1 | Date | |
|---|---|---|---|
| 99f02cd8e1 | |||
|
|
d93c9aa780 | ||
| ea56504aee | |||
| 46b91edfdf | |||
|
|
d2dfde519d | ||
| 1eac56f22c | |||
|
|
fb491a6bb2 | ||
| 3da7c89b66 | |||
|
|
e842cf3a57 | ||
| 85b0e8f1a6 | |||
|
|
b8e03ff6c1 | ||
| 54b1703d67 | |||
| 15d647a0ec | |||
|
|
353558ddb3 | ||
|
|
9f58d30050 | ||
|
|
4f0632cdf1 | ||
|
|
ff8ba71948 | ||
|
|
a367591327 | ||
|
|
c91925fbc2 | ||
| 479f1649ef | |||
|
|
06f69184b6 | ||
|
|
3471d9a57c | ||
| 28d4035477 | |||
|
|
3e1d151efd | ||
| 556cfc3036 | |||
| c49b757be6 | |||
| cec7e2caf3 | |||
| 9a0c07a461 | |||
| c7064c18b9 | |||
|
|
fb897310c8 | ||
| 5944849326 | |||
| 1255f1e6c7 | |||
| d7bc09d859 | |||
| f84a448351 | |||
| 7e34690e0c | |||
|
|
6f6d378600 | ||
|
|
216b463b8f | ||
| 4ad84cb5e5 | |||
| 19e0946717 | |||
| 0de1ad06cd | |||
|
|
72427006dd | ||
|
|
0e5f702651 | ||
| 4f1b0fc838 | |||
|
|
23d0963c2d | ||
|
|
dcb35fb80f | ||
| b1768854c3 | |||
| 4ed4eaf8e8 | |||
|
|
d167b53b17 | ||
| 18eff80a30 | |||
| a44ec5f624 | |||
|
|
5da518f50e | ||
|
|
ebb11a8ac5 | ||
|
|
dd9c5db1aa | ||
| 984ac9763e | |||
|
|
ad67af2f8f | ||
| cc8460d366 | |||
| 1dc9b9c956 |
36
.github/workflows/appimage-build.yml
vendored
36
.github/workflows/appimage-build.yml
vendored
@@ -34,7 +34,7 @@ jobs:
|
|||||||
libwayland-client0 libwayland-cursor0 libwayland-egl1 \
|
libwayland-client0 libwayland-cursor0 libwayland-egl1 \
|
||||||
libxkbcommon0 \
|
libxkbcommon0 \
|
||||||
libgtk-3-bin libglib2.0-bin shared-mime-info gsettings-desktop-schemas \
|
libgtk-3-bin libglib2.0-bin shared-mime-info gsettings-desktop-schemas \
|
||||||
libluajit-5.1-dev libpci-dev libperl-dev libssl-dev libayatana-appindicator3-dev \
|
liblua5.4-dev libpci-dev libperl-dev libssl-dev libayatana-appindicator3-dev \
|
||||||
perl python3 python3-minimal python3-dev python3-cffi mono-devel desktop-file-utils \
|
perl python3 python3-minimal python3-dev python3-cffi mono-devel desktop-file-utils \
|
||||||
fonts-noto-color-emoji breeze-gtk-theme \
|
fonts-noto-color-emoji breeze-gtk-theme \
|
||||||
patchelf file curl
|
patchelf file curl
|
||||||
@@ -86,11 +86,21 @@ jobs:
|
|||||||
cp -a /usr/lib/x86_64-linux-gnu/python3/dist-packages AppDir/usr/lib/x86_64-linux-gnu/python3/
|
cp -a /usr/lib/x86_64-linux-gnu/python3/dist-packages AppDir/usr/lib/x86_64-linux-gnu/python3/
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
if [ -d "/usr/lib/x86_64-linux-gnu/perl-base" ]; then
|
||||||
|
install -d AppDir/usr/lib/x86_64-linux-gnu
|
||||||
|
cp -a /usr/lib/x86_64-linux-gnu/perl-base AppDir/usr/lib/x86_64-linux-gnu/
|
||||||
|
fi
|
||||||
|
|
||||||
if [ -d "/usr/lib/x86_64-linux-gnu/perl" ]; then
|
if [ -d "/usr/lib/x86_64-linux-gnu/perl" ]; then
|
||||||
install -d AppDir/usr/lib/x86_64-linux-gnu
|
install -d AppDir/usr/lib/x86_64-linux-gnu
|
||||||
cp -a /usr/lib/x86_64-linux-gnu/perl AppDir/usr/lib/x86_64-linux-gnu/
|
cp -a /usr/lib/x86_64-linux-gnu/perl AppDir/usr/lib/x86_64-linux-gnu/
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
if [ -d "/usr/lib/x86_64-linux-gnu/perl5" ]; then
|
||||||
|
install -d AppDir/usr/lib/x86_64-linux-gnu
|
||||||
|
cp -a /usr/lib/x86_64-linux-gnu/perl5 AppDir/usr/lib/x86_64-linux-gnu/
|
||||||
|
fi
|
||||||
|
|
||||||
if [ -d "/usr/share/perl" ]; then
|
if [ -d "/usr/share/perl" ]; then
|
||||||
install -d AppDir/usr/share
|
install -d AppDir/usr/share
|
||||||
cp -a /usr/share/perl AppDir/usr/share/
|
cp -a /usr/share/perl AppDir/usr/share/
|
||||||
@@ -100,6 +110,10 @@ jobs:
|
|||||||
install -d AppDir/usr/share
|
install -d AppDir/usr/share
|
||||||
cp -a /usr/share/perl5 AppDir/usr/share/
|
cp -a /usr/share/perl5 AppDir/usr/share/
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
perl -MFile::Spec -e 'print "Build host File::Spec: $INC{\"File/Spec.pm\"}\n"'
|
||||||
|
find AppDir/usr -path '*/File/Spec.pm' -print -quit | grep -q .
|
||||||
|
|
||||||
if compgen -G '/usr/lib/x86_64-linux-gnu/libpython3*.so*' > /dev/null; then
|
if compgen -G '/usr/lib/x86_64-linux-gnu/libpython3*.so*' > /dev/null; then
|
||||||
install -d AppDir/usr/lib/x86_64-linux-gnu
|
install -d AppDir/usr/lib/x86_64-linux-gnu
|
||||||
cp -a /usr/lib/x86_64-linux-gnu/libpython3*.so* AppDir/usr/lib/x86_64-linux-gnu/
|
cp -a /usr/lib/x86_64-linux-gnu/libpython3*.so* AppDir/usr/lib/x86_64-linux-gnu/
|
||||||
@@ -162,7 +176,7 @@ jobs:
|
|||||||
|
|
||||||
APPDIR="${APPDIR:-$(dirname "$(readlink -f "$0")")}"
|
APPDIR="${APPDIR:-$(dirname "$(readlink -f "$0")")}"
|
||||||
|
|
||||||
export PATH="${PATH:-/usr/bin:/bin}:$APPDIR/usr/bin"
|
export PATH="$APPDIR/usr/bin:${PATH:-/usr/bin:/bin}"
|
||||||
export LD_LIBRARY_PATH="$APPDIR/usr/lib:$APPDIR/usr/lib/x86_64-linux-gnu:${LD_LIBRARY_PATH:-}"
|
export LD_LIBRARY_PATH="$APPDIR/usr/lib:$APPDIR/usr/lib/x86_64-linux-gnu:${LD_LIBRARY_PATH:-}"
|
||||||
export XDG_DATA_DIRS="$APPDIR/usr/share:${XDG_DATA_DIRS:-/usr/local/share:/usr/share}"
|
export XDG_DATA_DIRS="$APPDIR/usr/share:${XDG_DATA_DIRS:-/usr/local/share:/usr/share}"
|
||||||
export GTK_EXE_PREFIX="$APPDIR/usr"
|
export GTK_EXE_PREFIX="$APPDIR/usr"
|
||||||
@@ -211,6 +225,23 @@ jobs:
|
|||||||
|
|
||||||
unset GTK_MODULES
|
unset GTK_MODULES
|
||||||
|
|
||||||
|
perl5lib_entries=""
|
||||||
|
for dir in \
|
||||||
|
"$APPDIR/usr/lib/x86_64-linux-gnu/perl-base" \
|
||||||
|
"$APPDIR/usr/lib/x86_64-linux-gnu/perl"/* \
|
||||||
|
"$APPDIR/usr/lib/x86_64-linux-gnu/perl5"/* \
|
||||||
|
"$APPDIR/usr/share/perl"/* \
|
||||||
|
"$APPDIR/usr/share/perl5"
|
||||||
|
do
|
||||||
|
if [ -d "$dir" ]; then
|
||||||
|
perl5lib_entries="${perl5lib_entries:+$perl5lib_entries:}$dir"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
if [ -n "$perl5lib_entries" ]; then
|
||||||
|
export PERL5LIB="$perl5lib_entries${PERL5LIB:+:$PERL5LIB}"
|
||||||
|
fi
|
||||||
|
|
||||||
export PYTHONHOME="$APPDIR/usr"
|
export PYTHONHOME="$APPDIR/usr"
|
||||||
python_stdlib_dir="$(find "$APPDIR/usr/lib" -maxdepth 1 -type d -name 'python3.*' | head -n 1 || true)"
|
python_stdlib_dir="$(find "$APPDIR/usr/lib" -maxdepth 1 -type d -name 'python3.*' | head -n 1 || true)"
|
||||||
pythonpath_entries=""
|
pythonpath_entries=""
|
||||||
@@ -256,7 +287,6 @@ jobs:
|
|||||||
EOF
|
EOF
|
||||||
chmod +x AppRun
|
chmod +x AppRun
|
||||||
|
|
||||||
|
|
||||||
VERSION="$(git describe --tags --always)"
|
VERSION="$(git describe --tags --always)"
|
||||||
|
|
||||||
./linuxdeploy-x86_64.AppImage \
|
./linuxdeploy-x86_64.AppImage \
|
||||||
|
|||||||
1
.github/workflows/manjaro-package-build.yml
vendored
1
.github/workflows/manjaro-package-build.yml
vendored
@@ -34,7 +34,6 @@ jobs:
|
|||||||
openssl \
|
openssl \
|
||||||
libcanberra \
|
libcanberra \
|
||||||
libayatana-appindicator \
|
libayatana-appindicator \
|
||||||
luajit \
|
|
||||||
iso-codes \
|
iso-codes \
|
||||||
lua \
|
lua \
|
||||||
perl \
|
perl \
|
||||||
|
|||||||
4
.github/workflows/windows-build.yml
vendored
4
.github/workflows/windows-build.yml
vendored
@@ -65,8 +65,8 @@ jobs:
|
|||||||
Download-WithRetry -Url https://github.com/jrsoftware/issrc/releases/download/is-6_7_1/innosetup-6.7.1.exe -OutFile deps\innosetup-unicode.exe
|
Download-WithRetry -Url https://github.com/jrsoftware/issrc/releases/download/is-6_7_1/innosetup-6.7.1.exe -OutFile deps\innosetup-unicode.exe
|
||||||
& deps\innosetup-unicode.exe /VERYSILENT | Out-Null
|
& deps\innosetup-unicode.exe /VERYSILENT | Out-Null
|
||||||
|
|
||||||
Download-WithRetry -Url https://github.com/ZoiteChat/gvsbuild/releases/download/zoitechat-2.18.0-pre1/GTK3_Gvsbuild_zoitechat-2.18.0-pre1_${{ matrix.platform }}.7z -OutFile deps\gtk-${{ matrix.arch }}.7z
|
Download-WithRetry -Url https://github.com/ZoiteChat/gvsbuild/releases/download/zoitechat-2.18.1/GTK3_Gvsbuild_zoitechat-2.18.1_x64.zip -OutFile deps\gtk-${{ matrix.arch }}.zip
|
||||||
& 7z.exe x deps\gtk-${{ matrix.arch }}.7z -oC:\gtk-build\gtk\x64\release
|
Expand-Archive -LiteralPath deps\gtk-${{ matrix.arch }}.zip -DestinationPath C:\gtk-build\gtk\x64\release -Force
|
||||||
|
|
||||||
Download-WithRetry -Url https://repo.msys2.org/mingw/x86_64/mingw-w64-x86_64-hicolor-icon-theme-0.18-1-any.pkg.tar.zst -OutFile deps\hicolor-icon-theme.pkg.tar.zst
|
Download-WithRetry -Url https://repo.msys2.org/mingw/x86_64/mingw-w64-x86_64-hicolor-icon-theme-0.18-1-any.pkg.tar.zst -OutFile deps\hicolor-icon-theme.pkg.tar.zst
|
||||||
python -c "import tarfile,zstandard,pathlib;archive=pathlib.Path(r'deps\\hicolor-icon-theme.pkg.tar.zst');target=pathlib.Path(r'C:\\gtk-build\\gtk\\x64\\release');dctx=zstandard.ZstdDecompressor();f=archive.open('rb');reader=dctx.stream_reader(f);tf=tarfile.open(fileobj=reader,mode='r|');[tf.extract(m,path=target) for m in tf if m.name.startswith('mingw64/share/icons/hicolor/')];tf.close();reader.close();f.close()"
|
python -c "import tarfile,zstandard,pathlib;archive=pathlib.Path(r'deps\\hicolor-icon-theme.pkg.tar.zst');target=pathlib.Path(r'C:\\gtk-build\\gtk\\x64\\release');dctx=zstandard.ZstdDecompressor();f=archive.open('rb');reader=dctx.stream_reader(f);tf=tarfile.open(fileobj=reader,mode='r|');[tf.extract(m,path=target) for m in tf if m.name.startswith('mingw64/share/icons/hicolor/')];tf.close();reader.close();f.close()"
|
||||||
|
|||||||
@@ -1,7 +1,28 @@
|
|||||||
ZoiteChat ChangeLog
|
ZoiteChat ChangeLog
|
||||||
=================
|
=================
|
||||||
|
|
||||||
|
2.18.1 (2026-05-21)
|
||||||
|
-------------------
|
||||||
|
|
||||||
|
- Migrated D-Bus handling to GDBus.
|
||||||
|
- Enabled mouse wheel channel switching by default and consumed handled tab wheel events.
|
||||||
|
- Switched main panes to native GTK scrollbars.
|
||||||
|
- Tightened userlist button and meter layout.
|
||||||
|
- Made the native Windows file chooser modal.
|
||||||
|
- Set the Windows AppUserModelID before GTK startup.
|
||||||
|
- Updated Windows CI to use the new GTK3 bundle zip.
|
||||||
|
- Updated Windows installer architecture checks to use x64-compatible detection.
|
||||||
|
- Bumped the Flatpak Perl runtime to 5.42.2.
|
||||||
|
- Fixed palette color reads to preserve base GTK state.
|
||||||
|
- Fixed auto-replace whole-word matching.
|
||||||
|
- Fixed checksum file stream cleanup.
|
||||||
|
- Fixed byte handling in the Python console.
|
||||||
|
- Guarded GTK drag-and-drop handlers against null windows.
|
||||||
|
- Fixed size_t and integer handling issues.
|
||||||
|
- Removed the Winamp plugin.
|
||||||
|
|
||||||
2.18.0 (2026-04-20)
|
2.18.0 (2026-04-20)
|
||||||
|
-------------------
|
||||||
|
|
||||||
- Added optional close buttons on tabs.
|
- Added optional close buttons on tabs.
|
||||||
- Added Ctrl+W to close tabs and Ctrl+Shift+T to reopen recently closed tabs.
|
- Added Ctrl+W to close tabs and Ctrl+Shift+T to reopen recently closed tabs.
|
||||||
@@ -19,6 +40,7 @@ ZoiteChat ChangeLog
|
|||||||
- Improved AppStream metainfo validation.
|
- Improved AppStream metainfo validation.
|
||||||
|
|
||||||
2.18.0~pre6 (2026-03-30)
|
2.18.0~pre6 (2026-03-30)
|
||||||
|
------------------------
|
||||||
|
|
||||||
- Applied app theme CSS to the menubar consistently across the app.
|
- Applied app theme CSS to the menubar consistently across the app.
|
||||||
- Restored horizontal separator lines in menus.
|
- Restored horizontal separator lines in menus.
|
||||||
|
|||||||
@@ -29,6 +29,28 @@
|
|||||||
<id>zoitechat.desktop</id>
|
<id>zoitechat.desktop</id>
|
||||||
</provides>
|
</provides>
|
||||||
<releases>
|
<releases>
|
||||||
|
<release date="2026-05-21" version="2.18.1">
|
||||||
|
<description>
|
||||||
|
<ul>
|
||||||
|
<li>Migrated D-Bus handling to GDBus.</li>
|
||||||
|
<li>Enabled mouse wheel channel switching by default and consumed handled tab wheel events.</li>
|
||||||
|
<li>Switched main panes to native GTK scrollbars.</li>
|
||||||
|
<li>Tightened userlist button and meter layout.</li>
|
||||||
|
<li>Made the native Windows file chooser modal.</li>
|
||||||
|
<li>Set the Windows AppUserModelID before GTK startup.</li>
|
||||||
|
<li>Updated Windows CI to use the new GTK3 bundle zip.</li>
|
||||||
|
<li>Updated Windows installer architecture checks to use x64-compatible detection.</li>
|
||||||
|
<li>Bumped the Flatpak Perl runtime to 5.42.2.</li>
|
||||||
|
<li>Fixed palette color reads to preserve base GTK state.</li>
|
||||||
|
<li>Fixed auto-replace whole-word matching.</li>
|
||||||
|
<li>Fixed checksum file stream cleanup.</li>
|
||||||
|
<li>Fixed byte handling in the Python console.</li>
|
||||||
|
<li>Guarded GTK drag-and-drop handlers against null windows.</li>
|
||||||
|
<li>Fixed size_t and integer handling issues.</li>
|
||||||
|
<li>Removed the Winamp plugin.</li>
|
||||||
|
</ul>
|
||||||
|
</description>
|
||||||
|
</release>
|
||||||
<release date="2026-04-20" version="2.18.0">
|
<release date="2026-04-20" version="2.18.0">
|
||||||
<description>
|
<description>
|
||||||
<p>Tabs and navigation:</p>
|
<p>Tabs and navigation:</p>
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
project('zoitechat', 'c',
|
project('zoitechat', 'c',
|
||||||
version: '2.18.0',
|
version: '2.18.1',
|
||||||
meson_version: '>= 0.55.0',
|
meson_version: '>= 0.55.0',
|
||||||
default_options: [
|
default_options: [
|
||||||
'c_std=c17',
|
'c_std=c17',
|
||||||
|
|||||||
@@ -104,12 +104,14 @@ thread_sha256_file (GTask *task, GFile *file, gpointer task_data, GCancellable *
|
|||||||
g_checksum_update (checksum, buffer, ret);
|
g_checksum_update (checksum, buffer, ret);
|
||||||
|
|
||||||
if (error) {
|
if (error) {
|
||||||
g_checksum_free (checksum);
|
|
||||||
g_task_return_error (task, error);
|
g_task_return_error (task, error);
|
||||||
return;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
g_task_return_pointer (task, g_strdup (g_checksum_get_string (checksum)), g_free);
|
g_task_return_pointer (task, g_strdup (g_checksum_get_string (checksum)), g_free);
|
||||||
|
cleanup:
|
||||||
|
g_input_stream_close(G_INPUT_STREAM(istream), NULL, NULL);
|
||||||
|
g_object_unref(istream);
|
||||||
g_checksum_free (checksum);
|
g_checksum_free (checksum);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -91,27 +91,13 @@ static const signed char fish_unbase64[256] = {
|
|||||||
#include <openssl/provider.h>
|
#include <openssl/provider.h>
|
||||||
static OSSL_PROVIDER *legacy_provider;
|
static OSSL_PROVIDER *legacy_provider;
|
||||||
static OSSL_PROVIDER *default_provider;
|
static OSSL_PROVIDER *default_provider;
|
||||||
static OSSL_LIB_CTX *ossl_ctx;
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
int fish_init(void)
|
int fish_init(void)
|
||||||
{
|
{
|
||||||
#if OPENSSL_VERSION_NUMBER >= 0x30000000L
|
#if OPENSSL_VERSION_NUMBER >= 0x30000000L
|
||||||
ossl_ctx = OSSL_LIB_CTX_new();
|
legacy_provider = OSSL_PROVIDER_load(NULL, "legacy");
|
||||||
if (!ossl_ctx)
|
default_provider = OSSL_PROVIDER_load(NULL, "default");
|
||||||
return 0;
|
|
||||||
|
|
||||||
legacy_provider = OSSL_PROVIDER_load(ossl_ctx, "legacy");
|
|
||||||
if (!legacy_provider) {
|
|
||||||
fish_deinit();
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
default_provider = OSSL_PROVIDER_load(ossl_ctx, "default");
|
|
||||||
if (!default_provider) {
|
|
||||||
fish_deinit();
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
@@ -129,10 +115,6 @@ void fish_deinit(void)
|
|||||||
default_provider = NULL;
|
default_provider = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ossl_ctx) {
|
|
||||||
OSSL_LIB_CTX_free(ossl_ctx);
|
|
||||||
ossl_ctx = NULL;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -278,7 +260,9 @@ char *fish_cipher(const char *plaintext, size_t plaintext_len, const char *key,
|
|||||||
}
|
}
|
||||||
|
|
||||||
#if OPENSSL_VERSION_NUMBER >= 0x30000000L
|
#if OPENSSL_VERSION_NUMBER >= 0x30000000L
|
||||||
cipher = EVP_CIPHER_fetch(ossl_ctx, "BF-CBC", NULL);
|
cipher = EVP_CIPHER_fetch(NULL, "BF-CBC", NULL);
|
||||||
|
if (!cipher)
|
||||||
|
cipher = (EVP_CIPHER *) EVP_bf_cbc();
|
||||||
#else
|
#else
|
||||||
cipher = (EVP_CIPHER *) EVP_bf_cbc();
|
cipher = (EVP_CIPHER *) EVP_bf_cbc();
|
||||||
#endif
|
#endif
|
||||||
@@ -286,7 +270,9 @@ char *fish_cipher(const char *plaintext, size_t plaintext_len, const char *key,
|
|||||||
} else if (mode == EVP_CIPH_ECB_MODE) {
|
} else if (mode == EVP_CIPH_ECB_MODE) {
|
||||||
|
|
||||||
#if OPENSSL_VERSION_NUMBER >= 0x30000000L
|
#if OPENSSL_VERSION_NUMBER >= 0x30000000L
|
||||||
cipher = EVP_CIPHER_fetch(ossl_ctx, "BF-ECB", NULL);
|
cipher = EVP_CIPHER_fetch(NULL, "BF-ECB", NULL);
|
||||||
|
if (!cipher)
|
||||||
|
cipher = (EVP_CIPHER *) EVP_bf_ecb();
|
||||||
#else
|
#else
|
||||||
cipher = (EVP_CIPHER *) EVP_bf_ecb();
|
cipher = (EVP_CIPHER *) EVP_bf_ecb();
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -421,7 +421,7 @@ static int handle_keyx_notice(char *word[], char *word_eol[], void *userdata) {
|
|||||||
zoitechat_commandf(ph, "quote NOTICE %s :DH1080_FINISH %s%s", sender, pub_key, (mode == FISH_CBC_MODE) ? " CBC" : "");
|
zoitechat_commandf(ph, "quote NOTICE %s :DH1080_FINISH %s%s", sender, pub_key, (mode == FISH_CBC_MODE) ? " CBC" : "");
|
||||||
g_free(pub_key);
|
g_free(pub_key);
|
||||||
} else {
|
} else {
|
||||||
zoitechat_print(ph, "Failed to generate keys");
|
zoitechat_printf(ph, "Failed to generate keys");
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
} else if (!strcmp (dh_message, "DH1080_FINISH")) {
|
} else if (!strcmp (dh_message, "DH1080_FINISH")) {
|
||||||
@@ -446,7 +446,7 @@ static int handle_keyx_notice(char *word[], char *word_eol[], void *userdata) {
|
|||||||
zoitechat_printf(ph, "Stored new key for %s (%s)", sender, fish_modes[mode]);
|
zoitechat_printf(ph, "Stored new key for %s (%s)", sender, fish_modes[mode]);
|
||||||
g_free(secret_key);
|
g_free(secret_key);
|
||||||
} else {
|
} else {
|
||||||
zoitechat_print(ph, "Failed to create secret key!");
|
zoitechat_printf(ph, "Failed to create secret key!");
|
||||||
}
|
}
|
||||||
|
|
||||||
cleanup:
|
cleanup:
|
||||||
@@ -548,7 +548,7 @@ static int handle_keyx(char *word[], char *word_eol[], void *userdata) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if ((query_ctx && ctx_type != 3) || (!query_ctx && !irc_is_query(target))) {
|
if ((query_ctx && ctx_type != 3) || (!query_ctx && !irc_is_query(target))) {
|
||||||
zoitechat_print(ph, "You can only exchange keys with individuals");
|
zoitechat_printf(ph, "You can only exchange keys with individuals");
|
||||||
return ZOITECHAT_EAT_ALL;
|
return ZOITECHAT_EAT_ALL;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -560,7 +560,7 @@ static int handle_keyx(char *word[], char *word_eol[], void *userdata) {
|
|||||||
|
|
||||||
g_free(pub_key);
|
g_free(pub_key);
|
||||||
} else {
|
} else {
|
||||||
zoitechat_print(ph, "Failed to generate keys");
|
zoitechat_printf(ph, "Failed to generate keys");
|
||||||
}
|
}
|
||||||
|
|
||||||
return ZOITECHAT_EAT_ALL;
|
return ZOITECHAT_EAT_ALL;
|
||||||
@@ -577,7 +577,7 @@ static int handle_crypt_topic(char *word[], char *word_eol[], void *userdata) {
|
|||||||
GSList *encrypted_list;
|
GSList *encrypted_list;
|
||||||
|
|
||||||
if (!*topic) {
|
if (!*topic) {
|
||||||
zoitechat_print(ph, usage_topic);
|
zoitechat_printf(ph, "%s", usage_topic);
|
||||||
return ZOITECHAT_EAT_ALL;
|
return ZOITECHAT_EAT_ALL;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -624,7 +624,7 @@ static int handle_crypt_notice(char *word[], char *word_eol[], void *userdata) {
|
|||||||
GSList *encrypted_list, *encrypted_item;
|
GSList *encrypted_list, *encrypted_item;
|
||||||
|
|
||||||
if (!*target || !*notice) {
|
if (!*target || !*notice) {
|
||||||
zoitechat_print(ph, usage_notice);
|
zoitechat_printf(ph, "%s", usage_notice);
|
||||||
return ZOITECHAT_EAT_ALL;
|
return ZOITECHAT_EAT_ALL;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -676,7 +676,7 @@ static int handle_crypt_msg(char *word[], char *word_eol[], void *userdata) {
|
|||||||
GSList *encrypted_list, *encrypted_item;
|
GSList *encrypted_list, *encrypted_item;
|
||||||
|
|
||||||
if (!*target || !*message) {
|
if (!*target || !*message) {
|
||||||
zoitechat_print(ph, usage_msg);
|
zoitechat_printf(ph, "%s", usage_msg);
|
||||||
return ZOITECHAT_EAT_ALL;
|
return ZOITECHAT_EAT_ALL;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -805,11 +805,15 @@ int zoitechat_plugin_init(zoitechat_plugin *plugin_handle,
|
|||||||
zoitechat_hook_server_attrs(ph, "TOPIC", ZOITECHAT_PRI_NORM, handle_incoming, NULL);
|
zoitechat_hook_server_attrs(ph, "TOPIC", ZOITECHAT_PRI_NORM, handle_incoming, NULL);
|
||||||
zoitechat_hook_server_attrs(ph, "332", ZOITECHAT_PRI_NORM, handle_incoming, NULL);
|
zoitechat_hook_server_attrs(ph, "332", ZOITECHAT_PRI_NORM, handle_incoming, NULL);
|
||||||
|
|
||||||
if (!fish_init())
|
if (!fish_init()) {
|
||||||
|
zoitechat_printf(ph, "FiSHLiM failed to initialize crypto backend");
|
||||||
return 0;
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
if (!dh1080_init())
|
if (!dh1080_init()) {
|
||||||
|
zoitechat_printf(ph, "FiSHLiM failed to initialize DH1080");
|
||||||
return 0;
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
pending_exchanges = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, g_free);
|
pending_exchanges = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, g_free);
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,19 @@
|
|||||||
if cc.get_id() == 'msvc'
|
if cc.get_id() == 'msvc'
|
||||||
lua_dep = cc.find_library('lua51')
|
lua_dep = cc.find_library('lua51')
|
||||||
else
|
else
|
||||||
lua_dep = dependency(get_option('with-lua'))
|
lua_opt = get_option('with-lua')
|
||||||
|
lua_dep = dependency(lua_opt, required: false)
|
||||||
|
if not lua_dep.found() and lua_opt == 'lua-5.4'
|
||||||
|
foreach lua_name : ['lua5.4', 'lua-5.3', 'lua5.3', 'lua']
|
||||||
|
lua_dep = dependency(lua_name, required: false)
|
||||||
|
if lua_dep.found()
|
||||||
|
break
|
||||||
|
endif
|
||||||
|
endforeach
|
||||||
|
endif
|
||||||
|
if not lua_dep.found()
|
||||||
|
error('Dependency "' + lua_opt + '" not found')
|
||||||
|
endif
|
||||||
endif
|
endif
|
||||||
|
|
||||||
shared_module('lua', 'lua.c',
|
shared_module('lua', 'lua.c',
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ else:
|
|||||||
if not hasattr(sys, 'argv'):
|
if not hasattr(sys, 'argv'):
|
||||||
sys.argv = ['<zoitechat>']
|
sys.argv = ['<zoitechat>']
|
||||||
|
|
||||||
VERSION = b'2.18.0'
|
VERSION = b'2.18.1'
|
||||||
PLUGIN_NAME = ffi.new('char[]', b'Python')
|
PLUGIN_NAME = ffi.new('char[]', b'Python')
|
||||||
PLUGIN_DESC = ffi.new('char[]', b'Python %d.%d scripting interface' % (sys.version_info[0], sys.version_info[1]))
|
PLUGIN_DESC = ffi.new('char[]', b'Python %d.%d scripting interface' % (sys.version_info[0], sys.version_info[1]))
|
||||||
PLUGIN_VERSION = ffi.new('char[]', VERSION)
|
PLUGIN_VERSION = ffi.new('char[]', VERSION)
|
||||||
@@ -320,9 +320,9 @@ def _on_say_command(word, word_eol, userdata):
|
|||||||
return 0
|
return 0
|
||||||
|
|
||||||
try:
|
try:
|
||||||
python = _cstr(word_eol[1])
|
python = __decode(_cstr(word_eol[1]))
|
||||||
except Exception:
|
except Exception:
|
||||||
python = b''
|
python = ''
|
||||||
|
|
||||||
if not python:
|
if not python:
|
||||||
return 1
|
return 1
|
||||||
|
|||||||
17
readme.md
17
readme.md
@@ -31,12 +31,19 @@
|
|||||||
<br />
|
<br />
|
||||||
|
|
||||||
|
|
||||||
ZoiteChat is an HexChat based IRC client for Windows and UNIX-like operating systems.
|
ZoiteChat is a GTK3 IRC client based on HexChat, available for Windows and UNIX-like operating systems.
|
||||||
See [IRCHelp.org](http://irchelp.org) for information about IRC in general.
|
|
||||||
For more information on ZoiteChat please read our [documentation](https://docs.zoitechat.org/):
|
|
||||||
- [Downloads](https://zoitechat.org/download)
|
|
||||||
|
|
||||||
- [Troubleshooting](troubleshooting.md)
|
Features include HexChat-compatible Python, Perl and Lua scripting support, a plugin API,
|
||||||
|
multiple server/channel windows, spell checking, multiple authentication methods including SASL,
|
||||||
|
and customizable notifications.
|
||||||
|
|
||||||
|
See [IRCHelp.org](http://irchelp.org) for information about IRC in general.
|
||||||
|
|
||||||
|
For more information on ZoiteChat:
|
||||||
|
|
||||||
|
- [Main Documentation](https://docs.zoitechat.org/) and [FAQ](https://docs.zoitechat.org/en/latest/faq.html)
|
||||||
|
|
||||||
|
- [Downloads](https://zoitechat.org/download.php)
|
||||||
|
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|||||||
@@ -447,6 +447,7 @@ const struct prefs vars[] =
|
|||||||
{"gui_tab_layout", P_OFFINT (hex_gui_tab_layout), TYPE_INT},
|
{"gui_tab_layout", P_OFFINT (hex_gui_tab_layout), TYPE_INT},
|
||||||
{"gui_tab_closebuttons", P_OFFINT (hex_gui_tab_closebuttons), TYPE_BOOL},
|
{"gui_tab_closebuttons", P_OFFINT (hex_gui_tab_closebuttons), TYPE_BOOL},
|
||||||
{"gui_tab_middleclose", P_OFFINT (hex_gui_tab_middleclose), TYPE_BOOL},
|
{"gui_tab_middleclose", P_OFFINT (hex_gui_tab_middleclose), TYPE_BOOL},
|
||||||
|
{"gui_mouse_scroll_speed", P_OFFINT (hex_gui_mouse_scroll_speed), TYPE_INT},
|
||||||
{"gui_tab_newtofront", P_OFFINT (hex_gui_tab_newtofront), TYPE_INT},
|
{"gui_tab_newtofront", P_OFFINT (hex_gui_tab_newtofront), TYPE_INT},
|
||||||
{"gui_tab_pos", P_OFFINT (hex_gui_tab_pos), TYPE_INT},
|
{"gui_tab_pos", P_OFFINT (hex_gui_tab_pos), TYPE_INT},
|
||||||
{"gui_tab_scrollchans", P_OFFINT (hex_gui_tab_scrollchans), TYPE_BOOL},
|
{"gui_tab_scrollchans", P_OFFINT (hex_gui_tab_scrollchans), TYPE_BOOL},
|
||||||
@@ -548,6 +549,10 @@ const struct prefs vars[] =
|
|||||||
#endif
|
#endif
|
||||||
{"net_bind_host", P_OFFSET (hex_net_bind_host), TYPE_STR},
|
{"net_bind_host", P_OFFSET (hex_net_bind_host), TYPE_STR},
|
||||||
{"net_ping_timeout", P_OFFINT (hex_net_ping_timeout), TYPE_INT, zoitechat_reinit_timers},
|
{"net_ping_timeout", P_OFFINT (hex_net_ping_timeout), TYPE_INT, zoitechat_reinit_timers},
|
||||||
|
{"net_lag_check", P_OFFINT (hex_net_lag_check), TYPE_INT, zoitechat_reinit_timers},
|
||||||
|
{"net_keepalive_idle", P_OFFINT (hex_net_keepalive_idle), TYPE_INT},
|
||||||
|
{"net_keepalive_interval", P_OFFINT (hex_net_keepalive_interval), TYPE_INT},
|
||||||
|
{"net_keepalive_count", P_OFFINT (hex_net_keepalive_count), TYPE_INT},
|
||||||
{"net_proxy_auth", P_OFFINT (hex_net_proxy_auth), TYPE_BOOL},
|
{"net_proxy_auth", P_OFFINT (hex_net_proxy_auth), TYPE_BOOL},
|
||||||
{"net_proxy_host", P_OFFSET (hex_net_proxy_host), TYPE_STR},
|
{"net_proxy_host", P_OFFSET (hex_net_proxy_host), TYPE_STR},
|
||||||
{"net_proxy_pass", P_OFFSET (hex_net_proxy_pass), TYPE_STR},
|
{"net_proxy_pass", P_OFFSET (hex_net_proxy_pass), TYPE_STR},
|
||||||
@@ -782,6 +787,8 @@ load_default_config(void)
|
|||||||
prefs.hex_gui_tab_middleclose = 1;
|
prefs.hex_gui_tab_middleclose = 1;
|
||||||
prefs.hex_gui_tab_server = 1;
|
prefs.hex_gui_tab_server = 1;
|
||||||
prefs.hex_gui_tab_sort = 1;
|
prefs.hex_gui_tab_sort = 1;
|
||||||
|
prefs.hex_gui_tab_scrollchans = 1;
|
||||||
|
prefs.hex_gui_mouse_scroll_speed = 10;
|
||||||
prefs.hex_gui_topicbar = 1;
|
prefs.hex_gui_topicbar = 1;
|
||||||
prefs.hex_gui_transparency = 255;
|
prefs.hex_gui_transparency = 255;
|
||||||
prefs.hex_gui_tray = 1;
|
prefs.hex_gui_tray = 1;
|
||||||
@@ -857,6 +864,10 @@ load_default_config(void)
|
|||||||
prefs.hex_irc_ban_type = 1;
|
prefs.hex_irc_ban_type = 1;
|
||||||
prefs.hex_irc_join_delay = 5;
|
prefs.hex_irc_join_delay = 5;
|
||||||
prefs.hex_net_ping_timeout = 60;
|
prefs.hex_net_ping_timeout = 60;
|
||||||
|
prefs.hex_net_lag_check = 60;
|
||||||
|
prefs.hex_net_keepalive_idle = 60;
|
||||||
|
prefs.hex_net_keepalive_interval = 20;
|
||||||
|
prefs.hex_net_keepalive_count = 3;
|
||||||
prefs.hex_net_reconnect_delay = 10;
|
prefs.hex_net_reconnect_delay = 10;
|
||||||
prefs.hex_notify_timeout = 15;
|
prefs.hex_notify_timeout = 15;
|
||||||
prefs.hex_text_max_indent = 256;
|
prefs.hex_text_max_indent = 256;
|
||||||
@@ -1167,7 +1178,10 @@ set_showval (session *sess, const struct prefs *var, char *tbuf)
|
|||||||
switch (var->type)
|
switch (var->type)
|
||||||
{
|
{
|
||||||
case TYPE_STR:
|
case TYPE_STR:
|
||||||
sprintf (tbuf + len, "\0033:\017 %s\n", (char *) &prefs + var->offset);
|
{
|
||||||
|
const char *value = (char *) &prefs + var->offset;
|
||||||
|
sprintf (tbuf + len, "\0033:\017 %s\n", value ? value : "");
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case TYPE_INT:
|
case TYPE_INT:
|
||||||
sprintf (tbuf + len, "\0033:\017 %d\n", *((int *) &prefs + var->offset));
|
sprintf (tbuf + len, "\0033:\017 %d\n", *((int *) &prefs + var->offset));
|
||||||
|
|||||||
@@ -34,6 +34,9 @@
|
|||||||
|
|
||||||
#ifndef WIN32
|
#ifndef WIN32
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
#ifdef HAVE_NETINET_TCP_H
|
||||||
|
#include <netinet/tcp.h>
|
||||||
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define WANTSOCKET
|
#define WANTSOCKET
|
||||||
@@ -43,6 +46,9 @@
|
|||||||
|
|
||||||
#define NETWORK_PRIVATE
|
#define NETWORK_PRIVATE
|
||||||
#include "network.h"
|
#include "network.h"
|
||||||
|
#include "zoitechat.h"
|
||||||
|
|
||||||
|
extern struct zoitechatprefs prefs;
|
||||||
|
|
||||||
#define RAND_INT(n) ((int)(rand() / (RAND_MAX + 1.0) * (n)))
|
#define RAND_INT(n) ((int)(rand() / (RAND_MAX + 1.0) * (n)))
|
||||||
|
|
||||||
@@ -58,6 +64,27 @@ net_set_socket_options (int sok)
|
|||||||
setsockopt (sok, SOL_SOCKET, SO_REUSEADDR, (char *) &sw, sizeof (sw));
|
setsockopt (sok, SOL_SOCKET, SO_REUSEADDR, (char *) &sw, sizeof (sw));
|
||||||
sw = 1;
|
sw = 1;
|
||||||
setsockopt (sok, SOL_SOCKET, SO_KEEPALIVE, (char *) &sw, sizeof (sw));
|
setsockopt (sok, SOL_SOCKET, SO_KEEPALIVE, (char *) &sw, sizeof (sw));
|
||||||
|
#ifdef TCP_KEEPIDLE
|
||||||
|
{
|
||||||
|
int keepidle = prefs.hex_net_keepalive_idle;
|
||||||
|
if (keepidle > 0)
|
||||||
|
setsockopt (sok, IPPROTO_TCP, TCP_KEEPIDLE, (char *) &keepidle, sizeof (keepidle));
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef TCP_KEEPINTVL
|
||||||
|
{
|
||||||
|
int keepintvl = prefs.hex_net_keepalive_interval;
|
||||||
|
if (keepintvl > 0)
|
||||||
|
setsockopt (sok, IPPROTO_TCP, TCP_KEEPINTVL, (char *) &keepintvl, sizeof (keepintvl));
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef TCP_KEEPCNT
|
||||||
|
{
|
||||||
|
int keepcnt = prefs.hex_net_keepalive_count;
|
||||||
|
if (keepcnt > 0)
|
||||||
|
setsockopt (sok, IPPROTO_TCP, TCP_KEEPCNT, (char *) &keepcnt, sizeof (keepcnt));
|
||||||
|
}
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
char *
|
char *
|
||||||
|
|||||||
@@ -4166,10 +4166,10 @@ const struct commands xc_cmds[] = {
|
|||||||
static int
|
static int
|
||||||
command_compare (const void *a, const void *b)
|
command_compare (const void *a, const void *b)
|
||||||
{
|
{
|
||||||
return g_ascii_strcasecmp (a, ((struct commands *)b)->name);
|
return g_ascii_strcasecmp (a, ((const struct commands *)b)->name);
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct commands *
|
static const struct commands *
|
||||||
find_internal_command (char *name)
|
find_internal_command (char *name)
|
||||||
{
|
{
|
||||||
/* the "-1" is to skip the NULL terminator */
|
/* the "-1" is to skip the NULL terminator */
|
||||||
@@ -4205,7 +4205,7 @@ usercommand_show_help (session *sess, char *name)
|
|||||||
static void
|
static void
|
||||||
help (session *sess, char *tbuf, char *helpcmd, int quiet)
|
help (session *sess, char *tbuf, char *helpcmd, int quiet)
|
||||||
{
|
{
|
||||||
struct commands *cmd;
|
const struct commands *cmd;
|
||||||
|
|
||||||
if (plugin_show_help (sess, helpcmd))
|
if (plugin_show_help (sess, helpcmd))
|
||||||
return;
|
return;
|
||||||
@@ -4397,7 +4397,7 @@ void
|
|||||||
check_special_chars (char *cmd, int do_ascii) /* check for %X */
|
check_special_chars (char *cmd, int do_ascii) /* check for %X */
|
||||||
{
|
{
|
||||||
int occur = 0;
|
int occur = 0;
|
||||||
int len = strlen (cmd);
|
size_t len = strlen (cmd);
|
||||||
char *buf, *utf;
|
char *buf, *utf;
|
||||||
char tbuf[4];
|
char tbuf[4];
|
||||||
int i = 0, j = 0;
|
int i = 0, j = 0;
|
||||||
@@ -4763,7 +4763,7 @@ handle_command (session *sess, char *cmd, int check_spch)
|
|||||||
char *word[PDIWORDS+1];
|
char *word[PDIWORDS+1];
|
||||||
char *word_eol[PDIWORDS+1];
|
char *word_eol[PDIWORDS+1];
|
||||||
static int command_level = 0;
|
static int command_level = 0;
|
||||||
struct commands *int_cmd;
|
const struct commands *int_cmd;
|
||||||
char *pdibuf;
|
char *pdibuf;
|
||||||
char *tbuf;
|
char *tbuf;
|
||||||
int len;
|
int len;
|
||||||
|
|||||||
@@ -1017,7 +1017,7 @@ process_named_msg (session *sess, char *type, char *word[], char *word_eol[],
|
|||||||
char *account;
|
char *account;
|
||||||
char ip[128], nick[NICKLEN];
|
char ip[128], nick[NICKLEN];
|
||||||
char *text, *ex;
|
char *text, *ex;
|
||||||
int len = strlen (type);
|
size_t len = strlen (type);
|
||||||
|
|
||||||
/* fill in the "ip" and "nick" buffers */
|
/* fill in the "ip" and "nick" buffers */
|
||||||
ex = strchr (word[1], '!');
|
ex = strchr (word[1], '!');
|
||||||
|
|||||||
@@ -22,6 +22,7 @@
|
|||||||
|
|
||||||
#include <ctype.h>
|
#include <ctype.h>
|
||||||
#include <inttypes.h>
|
#include <inttypes.h>
|
||||||
|
#include <limits.h>
|
||||||
#include <stdarg.h>
|
#include <stdarg.h>
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
@@ -74,6 +75,18 @@ typedef struct
|
|||||||
static bool string_builder_init (StringBuilder *builder);
|
static bool string_builder_init (StringBuilder *builder);
|
||||||
static void string_builder_free (StringBuilder *builder);
|
static void string_builder_free (StringBuilder *builder);
|
||||||
static bool string_builder_append (StringBuilder *builder, const char *text);
|
static bool string_builder_append (StringBuilder *builder, const char *text);
|
||||||
|
static int size_to_int (size_t value);
|
||||||
|
|
||||||
|
static int
|
||||||
|
size_to_int (size_t value)
|
||||||
|
{
|
||||||
|
if (value > (size_t) INT_MAX)
|
||||||
|
{
|
||||||
|
return INT_MAX;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (int) value;
|
||||||
|
}
|
||||||
|
|
||||||
char *
|
char *
|
||||||
sysinfo_get_cpu (void)
|
sysinfo_get_cpu (void)
|
||||||
@@ -511,6 +524,7 @@ static char *read_hdd_info (IWbemClassObject *object)
|
|||||||
static char *bstr_to_utf8 (BSTR bstr)
|
static char *bstr_to_utf8 (BSTR bstr)
|
||||||
{
|
{
|
||||||
int utf8_len;
|
int utf8_len;
|
||||||
|
int wide_len;
|
||||||
char *utf8;
|
char *utf8;
|
||||||
|
|
||||||
if (bstr == NULL)
|
if (bstr == NULL)
|
||||||
@@ -518,7 +532,8 @@ static char *bstr_to_utf8 (BSTR bstr)
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
utf8_len = WideCharToMultiByte (CP_UTF8, 0, bstr, SysStringLen (bstr), NULL, 0, NULL, NULL);
|
wide_len = size_to_int ((size_t) SysStringLen (bstr));
|
||||||
|
utf8_len = WideCharToMultiByte (CP_UTF8, 0, bstr, wide_len, NULL, 0, NULL, NULL);
|
||||||
if (utf8_len <= 0)
|
if (utf8_len <= 0)
|
||||||
{
|
{
|
||||||
return NULL;
|
return NULL;
|
||||||
@@ -530,7 +545,7 @@ static char *bstr_to_utf8 (BSTR bstr)
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (WideCharToMultiByte (CP_UTF8, 0, bstr, SysStringLen (bstr), utf8, utf8_len, NULL, NULL) <= 0)
|
if (WideCharToMultiByte (CP_UTF8, 0, bstr, wide_len, utf8, utf8_len, NULL, NULL) <= 0)
|
||||||
{
|
{
|
||||||
free (utf8);
|
free (utf8);
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|||||||
@@ -321,7 +321,7 @@ url_check_line (char *buf)
|
|||||||
for (i = 0; i < ARRAY_SIZE (commands); i++)
|
for (i = 0; i < ARRAY_SIZE (commands); i++)
|
||||||
{
|
{
|
||||||
char *cmd = commands[i];
|
char *cmd = commands[i];
|
||||||
int len = strlen (cmd);
|
size_t len = strlen (cmd);
|
||||||
|
|
||||||
if (strncmp (cmd, po, len) == 0)
|
if (strncmp (cmd, po, len) == 0)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -98,7 +98,7 @@ path_part (char *file, char *path, int pathlen)
|
|||||||
char * /* like strstr(), but nocase */
|
char * /* like strstr(), but nocase */
|
||||||
nocasestrstr (const char *s, const char *wanted)
|
nocasestrstr (const char *s, const char *wanted)
|
||||||
{
|
{
|
||||||
register const int len = strlen (wanted);
|
register const size_t len = strlen (wanted);
|
||||||
|
|
||||||
if (len == 0)
|
if (len == 0)
|
||||||
return (char *)s;
|
return (char *)s;
|
||||||
|
|||||||
@@ -2,10 +2,10 @@
|
|||||||
* Copyright (C) 1998-2010 Peter Zelezny.
|
* Copyright (C) 1998-2010 Peter Zelezny.
|
||||||
* Copyright (C) 2009-2013 Berke Viktor.
|
* Copyright (C) 2009-2013 Berke Viktor.
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or
|
||||||
* it under the terms of the GNU General Public License as published by
|
* modify it under the terms of the GNU General Public License
|
||||||
* the Free Software Foundation; either version 2 of the License, or
|
* as published by the Free Software Foundation; either version 2
|
||||||
* (at your option) any later version.
|
* of the License, or (at your option) any later version.
|
||||||
*
|
*
|
||||||
* This program is distributed in the hope that it will be useful,
|
* This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
@@ -13,8 +13,8 @@
|
|||||||
* GNU General Public License for more details.
|
* GNU General Public License for more details.
|
||||||
*
|
*
|
||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with this program; if not, write to the Free Software
|
* along with this program; if not, see
|
||||||
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
* <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/* You can distribute this header with your plugins for easy compilation */
|
/* You can distribute this header with your plugins for easy compilation */
|
||||||
|
|||||||
@@ -379,6 +379,7 @@ lag_check (void)
|
|||||||
char tbuf[128];
|
char tbuf[128];
|
||||||
time_t now = time (0);
|
time_t now = time (0);
|
||||||
time_t lag;
|
time_t lag;
|
||||||
|
time_t ping_age;
|
||||||
|
|
||||||
tim = make_ping_time ();
|
tim = make_ping_time ();
|
||||||
|
|
||||||
@@ -388,7 +389,7 @@ lag_check (void)
|
|||||||
if (serv->connected && serv->end_of_motd)
|
if (serv->connected && serv->end_of_motd)
|
||||||
{
|
{
|
||||||
lag = now - serv->ping_recv;
|
lag = now - serv->ping_recv;
|
||||||
if (prefs.hex_net_ping_timeout != 0 && lag > prefs.hex_net_ping_timeout && lag > 0)
|
if (serv->lag_sent && prefs.hex_net_ping_timeout != 0 && lag > prefs.hex_net_ping_timeout && lag > 0)
|
||||||
{
|
{
|
||||||
sprintf (tbuf, "%" G_GINT64_FORMAT, (gint64) lag);
|
sprintf (tbuf, "%" G_GINT64_FORMAT, (gint64) lag);
|
||||||
EMIT_SIGNAL (XP_TE_PINGTIMEOUT, serv->server_session, tbuf, NULL,
|
EMIT_SIGNAL (XP_TE_PINGTIMEOUT, serv->server_session, tbuf, NULL,
|
||||||
@@ -397,12 +398,12 @@ lag_check (void)
|
|||||||
serv->auto_reconnect (serv, FALSE, -1);
|
serv->auto_reconnect (serv, FALSE, -1);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
{
|
||||||
|
ping_age = now - serv->ping_recv;
|
||||||
|
if (!serv->lag_sent && prefs.hex_net_lag_check > 0 && ping_age >= prefs.hex_net_lag_check)
|
||||||
{
|
{
|
||||||
g_snprintf (tbuf, sizeof (tbuf), "LAG%lu", tim);
|
g_snprintf (tbuf, sizeof (tbuf), "LAG%lu", tim);
|
||||||
serv->p_ping (serv, "", tbuf);
|
serv->p_ping (serv, "", tbuf);
|
||||||
|
|
||||||
if (!serv->lag_sent)
|
|
||||||
{
|
|
||||||
serv->lag_sent = tim;
|
serv->lag_sent = tim;
|
||||||
fe_set_lag (serv, -1);
|
fe_set_lag (serv, -1);
|
||||||
}
|
}
|
||||||
@@ -525,7 +526,7 @@ zoitechat_reinit_timers (void)
|
|||||||
if ((prefs.hex_net_ping_timeout != 0 || prefs.hex_gui_lagometer)
|
if ((prefs.hex_net_ping_timeout != 0 || prefs.hex_gui_lagometer)
|
||||||
&& lag_check_tag == 0)
|
&& lag_check_tag == 0)
|
||||||
{
|
{
|
||||||
lag_check_tag = fe_timeout_add_seconds (30, zoitechat_lag_check, NULL);
|
lag_check_tag = fe_timeout_add_seconds (1, zoitechat_lag_check, NULL);
|
||||||
}
|
}
|
||||||
else if ((!prefs.hex_net_ping_timeout && !prefs.hex_gui_lagometer)
|
else if ((!prefs.hex_net_ping_timeout && !prefs.hex_gui_lagometer)
|
||||||
&& lag_check_tag != 0)
|
&& lag_check_tag != 0)
|
||||||
|
|||||||
@@ -267,6 +267,7 @@ struct zoitechatprefs
|
|||||||
int hex_gui_tab_layout;
|
int hex_gui_tab_layout;
|
||||||
int hex_gui_tab_closebuttons;
|
int hex_gui_tab_closebuttons;
|
||||||
int hex_gui_tab_middleclose;
|
int hex_gui_tab_middleclose;
|
||||||
|
int hex_gui_mouse_scroll_speed;
|
||||||
int hex_gui_tab_newtofront;
|
int hex_gui_tab_newtofront;
|
||||||
int hex_gui_tab_pos;
|
int hex_gui_tab_pos;
|
||||||
int hex_gui_tab_small;
|
int hex_gui_tab_small;
|
||||||
@@ -289,6 +290,10 @@ struct zoitechatprefs
|
|||||||
int hex_irc_join_delay;
|
int hex_irc_join_delay;
|
||||||
int hex_irc_notice_pos;
|
int hex_irc_notice_pos;
|
||||||
int hex_net_ping_timeout;
|
int hex_net_ping_timeout;
|
||||||
|
int hex_net_lag_check;
|
||||||
|
int hex_net_keepalive_idle;
|
||||||
|
int hex_net_keepalive_interval;
|
||||||
|
int hex_net_keepalive_count;
|
||||||
int hex_net_proxy_port;
|
int hex_net_proxy_port;
|
||||||
int hex_net_proxy_type; /* 0=disabled, 1=wingate 2=socks4, 3=socks5, 4=http */
|
int hex_net_proxy_type; /* 0=disabled, 1=wingate 2=socks4, 3=socks5, 4=http */
|
||||||
int hex_net_proxy_use; /* 0=all 1=IRC_ONLY 2=DCC_ONLY */
|
int hex_net_proxy_use; /* 0=all 1=IRC_ONLY 2=DCC_ONLY */
|
||||||
|
|||||||
@@ -452,7 +452,7 @@ void
|
|||||||
fe_add_chan_list (server *serv, char *chan, char *users, char *topic)
|
fe_add_chan_list (server *serv, char *chan, char *users, char *topic)
|
||||||
{
|
{
|
||||||
chanlistrow *next_row;
|
chanlistrow *next_row;
|
||||||
int len = strlen (chan) + 1;
|
size_t len = strlen (chan) + 1;
|
||||||
|
|
||||||
/* we allocate the struct and channel string in one go */
|
/* we allocate the struct and channel string in one go */
|
||||||
next_row = g_malloc (sizeof (chanlistrow) + len);
|
next_row = g_malloc (sizeof (chanlistrow) + len);
|
||||||
|
|||||||
@@ -304,20 +304,32 @@ tab_scroll_right_down_clicked (GtkWidget *widget, chanview *cv)
|
|||||||
static gboolean
|
static gboolean
|
||||||
tab_scroll_cb (GtkWidget *widget, GdkEventScroll *event, gpointer cv)
|
tab_scroll_cb (GtkWidget *widget, GdkEventScroll *event, gpointer cv)
|
||||||
{
|
{
|
||||||
|
int direction = cv_scroll_direction (event);
|
||||||
|
int i;
|
||||||
|
|
||||||
if (prefs.hex_gui_tab_scrollchans)
|
if (prefs.hex_gui_tab_scrollchans)
|
||||||
{
|
{
|
||||||
if (event->direction == GDK_SCROLL_DOWN)
|
if (direction != 0)
|
||||||
mg_switch_page (1, 1);
|
{
|
||||||
else if (event->direction == GDK_SCROLL_UP)
|
for (i = 0; i < cv_scroll_step_count (); i++)
|
||||||
mg_switch_page (1, -1);
|
mg_switch_page (1, direction);
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
/* mouse wheel scrolling */
|
if (direction < 0)
|
||||||
if (event->direction == GDK_SCROLL_UP)
|
{
|
||||||
|
for (i = 0; i < cv_scroll_step_count (); i++)
|
||||||
tab_scroll_left_up_clicked (widget, cv);
|
tab_scroll_left_up_clicked (widget, cv);
|
||||||
else if (event->direction == GDK_SCROLL_DOWN)
|
return TRUE;
|
||||||
|
}
|
||||||
|
else if (direction > 0)
|
||||||
|
{
|
||||||
|
for (i = 0; i < cv_scroll_step_count (); i++)
|
||||||
tab_scroll_right_down_clicked (widget, cv);
|
tab_scroll_right_down_clicked (widget, cv);
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
@@ -345,17 +357,17 @@ cv_tabs_init (chanview *cv)
|
|||||||
|
|
||||||
viewport = gtk_scrolled_window_new (0, 0);
|
viewport = gtk_scrolled_window_new (0, 0);
|
||||||
gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (viewport), GTK_SHADOW_NONE);
|
gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (viewport), GTK_SHADOW_NONE);
|
||||||
gtk_scrolled_window_set_overlay_scrolling (GTK_SCROLLED_WINDOW (viewport), FALSE);
|
|
||||||
if (cv->vertical)
|
if (cv->vertical)
|
||||||
gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (viewport),
|
gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (viewport),
|
||||||
GTK_POLICY_NEVER, GTK_POLICY_ALWAYS);
|
GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC);
|
||||||
else
|
else
|
||||||
gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (viewport),
|
gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (viewport),
|
||||||
GTK_POLICY_ALWAYS, GTK_POLICY_NEVER);
|
GTK_POLICY_AUTOMATIC, GTK_POLICY_NEVER);
|
||||||
gtk_scrolled_window_set_min_content_width (GTK_SCROLLED_WINDOW (viewport), 1);
|
gtk_scrolled_window_set_min_content_width (GTK_SCROLLED_WINDOW (viewport), 1);
|
||||||
gtk_scrolled_window_set_min_content_height (GTK_SCROLLED_WINDOW (viewport), 1);
|
gtk_scrolled_window_set_min_content_height (GTK_SCROLLED_WINDOW (viewport), 1);
|
||||||
gtk_widget_set_hexpand (viewport, TRUE);
|
gtk_widget_set_hexpand (viewport, TRUE);
|
||||||
gtk_widget_set_vexpand (viewport, TRUE);
|
gtk_widget_set_vexpand (viewport, TRUE);
|
||||||
|
cv_add_scroll_events (viewport);
|
||||||
g_signal_connect (G_OBJECT (viewport), "scroll-event",
|
g_signal_connect (G_OBJECT (viewport), "scroll-event",
|
||||||
G_CALLBACK (tab_scroll_cb), cv);
|
G_CALLBACK (tab_scroll_cb), cv);
|
||||||
gtk_box_pack_start (GTK_BOX (outer), viewport, 1, 1, 0);
|
gtk_box_pack_start (GTK_BOX (outer), viewport, 1, 1, 0);
|
||||||
@@ -677,9 +689,11 @@ cv_tabs_add (chanview *cv, chan *ch, char *name, GtkTreeIter *parent)
|
|||||||
gtk_widget_set_name (but, "zoitechat-tab");
|
gtk_widget_set_name (but, "zoitechat-tab");
|
||||||
gtk_widget_set_size_request (but, -1, 14);
|
gtk_widget_set_size_request (but, -1, 14);
|
||||||
gtk_widget_add_events (but, GDK_POINTER_MOTION_MASK | GDK_LEAVE_NOTIFY_MASK);
|
gtk_widget_add_events (but, GDK_POINTER_MOTION_MASK | GDK_LEAVE_NOTIFY_MASK);
|
||||||
|
cv_add_scroll_events (but);
|
||||||
hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 2);
|
hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 2);
|
||||||
label = gtk_label_new (name);
|
label = gtk_label_new (name);
|
||||||
close_button = gtk_button_new ();
|
close_button = gtk_button_new ();
|
||||||
|
cv_add_scroll_events (close_button);
|
||||||
gtk_style_context_add_class (gtk_widget_get_style_context (close_button), "flat");
|
gtk_style_context_add_class (gtk_widget_get_style_context (close_button), "flat");
|
||||||
close_icon = gtk_image_new_from_icon_name ("window-close-symbolic", GTK_ICON_SIZE_MENU);
|
close_icon = gtk_image_new_from_icon_name ("window-close-symbolic", GTK_ICON_SIZE_MENU);
|
||||||
gtk_image_set_pixel_size (GTK_IMAGE (close_icon), 8);
|
gtk_image_set_pixel_size (GTK_IMAGE (close_icon), 8);
|
||||||
@@ -695,6 +709,10 @@ cv_tabs_add (chanview *cv, chan *ch, char *name, GtkTreeIter *parent)
|
|||||||
/* used to trap right-clicks */
|
/* used to trap right-clicks */
|
||||||
g_signal_connect (G_OBJECT (but), "button-press-event",
|
g_signal_connect (G_OBJECT (but), "button-press-event",
|
||||||
G_CALLBACK (tab_click_cb), ch);
|
G_CALLBACK (tab_click_cb), ch);
|
||||||
|
g_signal_connect (G_OBJECT (but), "scroll-event",
|
||||||
|
G_CALLBACK (tab_scroll_cb), cv);
|
||||||
|
g_signal_connect (G_OBJECT (close_button), "scroll-event",
|
||||||
|
G_CALLBACK (tab_scroll_cb), cv);
|
||||||
g_signal_connect (G_OBJECT (but), "motion-notify-event",
|
g_signal_connect (G_OBJECT (but), "motion-notify-event",
|
||||||
G_CALLBACK (tab_close_motion_cb), ch);
|
G_CALLBACK (tab_close_motion_cb), ch);
|
||||||
g_signal_connect (G_OBJECT (but), "leave-notify-event",
|
g_signal_connect (G_OBJECT (but), "leave-notify-event",
|
||||||
|
|||||||
@@ -111,12 +111,14 @@ cv_tree_scroll_event_cb (GtkWidget *widget, GdkEventScroll *event, gpointer user
|
|||||||
{
|
{
|
||||||
if (prefs.hex_gui_tab_scrollchans)
|
if (prefs.hex_gui_tab_scrollchans)
|
||||||
{
|
{
|
||||||
if (event->direction == GDK_SCROLL_DOWN)
|
int direction = cv_scroll_direction (event);
|
||||||
mg_switch_page (1, 1);
|
int i;
|
||||||
else if (event->direction == GDK_SCROLL_UP)
|
|
||||||
mg_switch_page (1, -1);
|
|
||||||
|
|
||||||
return TRUE;
|
if (direction != 0)
|
||||||
|
for (i = 0; i < cv_scroll_step_count (); i++)
|
||||||
|
mg_switch_page (1, direction);
|
||||||
|
|
||||||
|
return direction != 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
@@ -141,6 +143,7 @@ cv_tree_init (chanview *cv)
|
|||||||
win = gtk_scrolled_window_new (0, 0);
|
win = gtk_scrolled_window_new (0, 0);
|
||||||
gtk_widget_set_hexpand (win, TRUE);
|
gtk_widget_set_hexpand (win, TRUE);
|
||||||
gtk_widget_set_vexpand (win, TRUE);
|
gtk_widget_set_vexpand (win, TRUE);
|
||||||
|
cv_add_scroll_events (win);
|
||||||
|
|
||||||
/*gtk_container_set_border_width (GTK_CONTAINER (win), 1);*/
|
/*gtk_container_set_border_width (GTK_CONTAINER (win), 1);*/
|
||||||
gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (win),
|
gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (win),
|
||||||
@@ -154,6 +157,7 @@ cv_tree_init (chanview *cv)
|
|||||||
view = gtk_tree_view_new_with_model (GTK_TREE_MODEL (cv->store));
|
view = gtk_tree_view_new_with_model (GTK_TREE_MODEL (cv->store));
|
||||||
gtk_widget_set_hexpand (view, TRUE);
|
gtk_widget_set_hexpand (view, TRUE);
|
||||||
gtk_widget_set_vexpand (view, TRUE);
|
gtk_widget_set_vexpand (view, TRUE);
|
||||||
|
cv_add_scroll_events (view);
|
||||||
gtk_widget_set_name (view, "zoitechat-tree");
|
gtk_widget_set_name (view, "zoitechat-tree");
|
||||||
{
|
{
|
||||||
ThemeWidgetStyleValues style_values;
|
ThemeWidgetStyleValues style_values;
|
||||||
|
|||||||
@@ -95,6 +95,46 @@ static chan *cv_find_chan_by_number (chanview *cv, int num);
|
|||||||
static int cv_find_number_of_chan (chanview *cv, chan *find_ch);
|
static int cv_find_number_of_chan (chanview *cv, chan *find_ch);
|
||||||
static void cv_find_neighbors_for_removal (chanview *cv, chan *find_ch, chan **left_ch, chan **first_ch);
|
static void cv_find_neighbors_for_removal (chanview *cv, chan *find_ch, chan **left_ch, chan **first_ch);
|
||||||
|
|
||||||
|
static void
|
||||||
|
cv_add_scroll_events (GtkWidget *widget)
|
||||||
|
{
|
||||||
|
gtk_widget_add_events (widget, GDK_SCROLL_MASK | GDK_SMOOTH_SCROLL_MASK);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
cv_scroll_direction (GdkEventScroll *event)
|
||||||
|
{
|
||||||
|
gdouble dx;
|
||||||
|
gdouble dy;
|
||||||
|
|
||||||
|
if (event->direction == GDK_SCROLL_SMOOTH && gdk_event_get_scroll_deltas ((GdkEvent *) event, &dx, &dy))
|
||||||
|
{
|
||||||
|
if (dy > 0)
|
||||||
|
return 1;
|
||||||
|
if (dy < 0)
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (event->direction)
|
||||||
|
{
|
||||||
|
case GDK_SCROLL_DOWN:
|
||||||
|
return 1;
|
||||||
|
case GDK_SCROLL_UP:
|
||||||
|
return -1;
|
||||||
|
default:
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
cv_scroll_step_count (void)
|
||||||
|
{
|
||||||
|
int speed = prefs.hex_gui_mouse_scroll_speed;
|
||||||
|
if (speed < 1)
|
||||||
|
speed = 1;
|
||||||
|
return (speed + 9) / 10;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* ======= TABS ======= */
|
/* ======= TABS ======= */
|
||||||
|
|
||||||
|
|||||||
@@ -124,6 +124,25 @@ create_msg_dialog (gchar *title, gchar *message)
|
|||||||
|
|
||||||
static char *win32_argv0_dir;
|
static char *win32_argv0_dir;
|
||||||
|
|
||||||
|
static void
|
||||||
|
win32_set_appusermodelid (void)
|
||||||
|
{
|
||||||
|
HMODULE shell32;
|
||||||
|
HRESULT (WINAPI *set_appid) (PCWSTR);
|
||||||
|
|
||||||
|
shell32 = GetModuleHandleW (L"shell32.dll");
|
||||||
|
if (!shell32)
|
||||||
|
shell32 = LoadLibraryW (L"shell32.dll");
|
||||||
|
if (!shell32)
|
||||||
|
return;
|
||||||
|
|
||||||
|
set_appid = (HRESULT (WINAPI *) (PCWSTR)) GetProcAddress (shell32, "SetCurrentProcessExplicitAppUserModelID");
|
||||||
|
if (!set_appid)
|
||||||
|
return;
|
||||||
|
|
||||||
|
set_appid (L"ZoiteChat.Desktop.Notify");
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
win32_set_gsettings_schema_dir (void)
|
win32_set_gsettings_schema_dir (void)
|
||||||
{
|
{
|
||||||
@@ -422,6 +441,7 @@ fe_args (int argc, char *argv[])
|
|||||||
|
|
||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
win32_set_gsettings_schema_dir ();
|
win32_set_gsettings_schema_dir ();
|
||||||
|
win32_set_appusermodelid ();
|
||||||
win32_configure_pixbuf_loaders ();
|
win32_configure_pixbuf_loaders ();
|
||||||
|
|
||||||
/* this is mainly for irc:// URL handling. When windows calls us from */
|
/* this is mainly for irc:// URL handling. When windows calls us from */
|
||||||
|
|||||||
@@ -1973,8 +1973,37 @@ replace_handle (GtkWidget *t)
|
|||||||
{
|
{
|
||||||
ptrdiff_t found_offset = found - text;
|
ptrdiff_t found_offset = found - text;
|
||||||
ptrdiff_t found_end_offset = found_offset + (ptrdiff_t) pop_len;
|
ptrdiff_t found_end_offset = found_offset + (ptrdiff_t) pop_len;
|
||||||
|
gboolean start_ok;
|
||||||
|
gboolean end_ok;
|
||||||
int rank;
|
int rank;
|
||||||
ptrdiff_t distance;
|
ptrdiff_t distance;
|
||||||
|
const char *before = found;
|
||||||
|
const char *after = found + pop_len;
|
||||||
|
|
||||||
|
if (before > text)
|
||||||
|
{
|
||||||
|
before = g_utf8_find_prev_char (text, before);
|
||||||
|
start_ok = !before || (!g_unichar_isalnum (g_utf8_get_char (before)) && g_utf8_get_char (before) != '_');
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
start_ok = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*after != '\0')
|
||||||
|
{
|
||||||
|
end_ok = !g_unichar_isalnum (g_utf8_get_char (after)) && g_utf8_get_char (after) != '_';
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
end_ok = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!start_ok || !end_ok)
|
||||||
|
{
|
||||||
|
found++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if (cursor_byte_offset >= found_offset && cursor_byte_offset <= found_end_offset)
|
if (cursor_byte_offset >= found_offset && cursor_byte_offset <= found_end_offset)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -272,13 +272,6 @@ gtkutil_apply_palette (GtkWidget *widget, const GdkRGBA *bg, const GdkRGBA *fg,
|
|||||||
theme_manager_apply_palette_widget (widget, bg, fg, font_desc);
|
theme_manager_apply_palette_widget (widget, bg, fg, font_desc);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
gtkutil_file_req_destroy (GtkWidget * wid, struct file_req *freq)
|
|
||||||
{
|
|
||||||
freq->callback (freq->userdata, NULL);
|
|
||||||
g_free (freq);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gtkutil_check_file (char *filename, struct file_req *freq)
|
gtkutil_check_file (char *filename, struct file_req *freq)
|
||||||
{
|
{
|
||||||
@@ -390,26 +383,6 @@ gtkutil_file_req_done_chooser (GtkFileChooser *fs, struct file_req *freq)
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
gtkutil_file_req_done (GtkWidget * wid, struct file_req *freq)
|
|
||||||
{
|
|
||||||
gtkutil_file_req_done_chooser (GTK_FILE_CHOOSER (freq->dialog), freq);
|
|
||||||
gtk_widget_destroy (freq->dialog);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
gtkutil_file_req_response (GtkWidget *dialog, gint res, struct file_req *freq)
|
|
||||||
{
|
|
||||||
if (res == GTK_RESPONSE_ACCEPT)
|
|
||||||
{
|
|
||||||
gtkutil_file_req_done (dialog, freq);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
gtk_widget_destroy (dialog);
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef WIN32
|
|
||||||
static gboolean
|
static gboolean
|
||||||
gtkutil_native_dialog_unref_idle (gpointer native)
|
gtkutil_native_dialog_unref_idle (gpointer native)
|
||||||
{
|
{
|
||||||
@@ -423,27 +396,16 @@ gtkutil_native_file_req_response (GtkNativeDialog *dialog, gint res, struct file
|
|||||||
if (res == GTK_RESPONSE_ACCEPT)
|
if (res == GTK_RESPONSE_ACCEPT)
|
||||||
gtkutil_file_req_done_chooser (GTK_FILE_CHOOSER (dialog), freq);
|
gtkutil_file_req_done_chooser (GTK_FILE_CHOOSER (dialog), freq);
|
||||||
|
|
||||||
/* Match gtk dialog flow by always sending NULL to indicate completion. */
|
|
||||||
freq->callback (freq->userdata, NULL);
|
freq->callback (freq->userdata, NULL);
|
||||||
g_free (freq);
|
g_free (freq);
|
||||||
|
|
||||||
/*
|
|
||||||
* Defer unref until idle to avoid disposing the native chooser while
|
|
||||||
* still in the button-release signal stack on Windows.
|
|
||||||
*/
|
|
||||||
g_idle_add (gtkutil_native_dialog_unref_idle, dialog);
|
g_idle_add (gtkutil_native_dialog_unref_idle, dialog);
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
void
|
void
|
||||||
gtkutil_file_req (GtkWindow *parent, const char *title, void *callback, void *userdata, char *filter, char *extensions,
|
gtkutil_file_req (GtkWindow *parent, const char *title, void *callback, void *userdata, char *filter, char *extensions,
|
||||||
int flags)
|
int flags)
|
||||||
{
|
{
|
||||||
struct file_req *freq;
|
struct file_req *freq;
|
||||||
GtkWidget *dialog;
|
|
||||||
GtkFileFilter *filefilter;
|
|
||||||
char *token;
|
|
||||||
char *tokenbuffer;
|
|
||||||
const char *xdir;
|
const char *xdir;
|
||||||
GtkWindow *effective_parent = parent;
|
GtkWindow *effective_parent = parent;
|
||||||
|
|
||||||
@@ -453,7 +415,6 @@ gtkutil_file_req (GtkWindow *parent, const char *title, void *callback, void *us
|
|||||||
|
|
||||||
xdir = get_xdir ();
|
xdir = get_xdir ();
|
||||||
|
|
||||||
#ifdef WIN32
|
|
||||||
{
|
{
|
||||||
GtkFileChooserNative *native = gtk_file_chooser_native_new (
|
GtkFileChooserNative *native = gtk_file_chooser_native_new (
|
||||||
title,
|
title,
|
||||||
@@ -522,110 +483,14 @@ gtkutil_file_req (GtkWindow *parent, const char *title, void *callback, void *us
|
|||||||
|
|
||||||
g_signal_connect (native, "response",
|
g_signal_connect (native, "response",
|
||||||
G_CALLBACK (gtkutil_native_file_req_response), freq);
|
G_CALLBACK (gtkutil_native_file_req_response), freq);
|
||||||
|
|
||||||
|
if (flags & FRF_MODAL)
|
||||||
|
gtk_native_dialog_set_modal (GTK_NATIVE_DIALOG (native), TRUE);
|
||||||
|
|
||||||
gtk_native_dialog_show (GTK_NATIVE_DIALOG (native));
|
gtk_native_dialog_show (GTK_NATIVE_DIALOG (native));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
if (flags & FRF_WRITE)
|
|
||||||
{
|
|
||||||
dialog = gtk_file_chooser_dialog_new (title, NULL,
|
|
||||||
GTK_FILE_CHOOSER_ACTION_SAVE,
|
|
||||||
_("_Cancel"), GTK_RESPONSE_CANCEL,
|
|
||||||
_("_Save"), GTK_RESPONSE_ACCEPT,
|
|
||||||
NULL);
|
|
||||||
|
|
||||||
if (!(flags & FRF_NOASKOVERWRITE))
|
|
||||||
gtk_file_chooser_set_do_overwrite_confirmation (GTK_FILE_CHOOSER (dialog), TRUE);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
dialog = gtk_file_chooser_dialog_new (title, NULL,
|
|
||||||
GTK_FILE_CHOOSER_ACTION_OPEN,
|
|
||||||
_("_Cancel"), GTK_RESPONSE_CANCEL,
|
|
||||||
_("_Open"), GTK_RESPONSE_ACCEPT,
|
|
||||||
NULL);
|
|
||||||
|
|
||||||
theme_manager_attach_window (dialog);
|
|
||||||
|
|
||||||
if (filter && filter[0] && (flags & FRF_FILTERISINITIAL))
|
|
||||||
{
|
|
||||||
if (flags & FRF_WRITE)
|
|
||||||
{
|
|
||||||
char temp[1024];
|
|
||||||
path_part (filter, temp, sizeof (temp));
|
|
||||||
if (temp[0] && g_file_test (temp, G_FILE_TEST_IS_DIR))
|
|
||||||
gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog), temp);
|
|
||||||
else if (xdir && xdir[0] && g_file_test (xdir, G_FILE_TEST_IS_DIR))
|
|
||||||
gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog), xdir);
|
|
||||||
gtk_file_chooser_set_current_name (GTK_FILE_CHOOSER (dialog), file_part (filter));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (g_file_test (filter, G_FILE_TEST_IS_DIR))
|
|
||||||
gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog), filter);
|
|
||||||
else if (xdir && xdir[0] && g_file_test (xdir, G_FILE_TEST_IS_DIR))
|
|
||||||
gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog), xdir);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (!(flags & FRF_RECENTLYUSED))
|
|
||||||
{
|
|
||||||
if (xdir && xdir[0] && g_file_test (xdir, G_FILE_TEST_IS_DIR))
|
|
||||||
gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog), xdir);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (flags & FRF_MULTIPLE)
|
|
||||||
gtk_file_chooser_set_select_multiple (GTK_FILE_CHOOSER (dialog), TRUE);
|
|
||||||
if (flags & FRF_CHOOSEFOLDER)
|
|
||||||
gtk_file_chooser_set_action (GTK_FILE_CHOOSER (dialog), GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER);
|
|
||||||
|
|
||||||
if ((flags & FRF_EXTENSIONS || flags & FRF_MIMETYPES) && extensions != NULL)
|
|
||||||
{
|
|
||||||
filefilter = gtk_file_filter_new ();
|
|
||||||
tokenbuffer = g_strdup (extensions);
|
|
||||||
token = strtok (tokenbuffer, ";");
|
|
||||||
|
|
||||||
while (token != NULL)
|
|
||||||
{
|
|
||||||
if (flags & FRF_EXTENSIONS)
|
|
||||||
gtk_file_filter_add_pattern (filefilter, token);
|
|
||||||
else
|
|
||||||
gtk_file_filter_add_mime_type (filefilter, token);
|
|
||||||
token = strtok (NULL, ";");
|
|
||||||
}
|
|
||||||
|
|
||||||
g_free (tokenbuffer);
|
|
||||||
gtk_file_chooser_set_filter (GTK_FILE_CHOOSER (dialog), filefilter);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (xdir && xdir[0] && g_file_test (xdir, G_FILE_TEST_IS_DIR))
|
|
||||||
{
|
|
||||||
GError *shortcut_error = NULL;
|
|
||||||
|
|
||||||
gtk_file_chooser_add_shortcut_folder (GTK_FILE_CHOOSER (dialog), xdir, &shortcut_error);
|
|
||||||
if (shortcut_error)
|
|
||||||
g_error_free (shortcut_error);
|
|
||||||
}
|
|
||||||
freq = g_new (struct file_req, 1);
|
|
||||||
freq->dialog = dialog;
|
|
||||||
freq->flags = flags;
|
|
||||||
freq->callback = callback;
|
|
||||||
freq->userdata = userdata;
|
|
||||||
|
|
||||||
g_signal_connect (G_OBJECT (dialog), "response",
|
|
||||||
G_CALLBACK (gtkutil_file_req_response), freq);
|
|
||||||
g_signal_connect (G_OBJECT (dialog), "destroy",
|
|
||||||
G_CALLBACK (gtkutil_file_req_destroy), (gpointer) freq);
|
|
||||||
|
|
||||||
if (effective_parent)
|
|
||||||
gtk_window_set_transient_for (GTK_WINDOW (dialog), effective_parent);
|
|
||||||
|
|
||||||
if (flags & FRF_MODAL)
|
|
||||||
{
|
|
||||||
g_assert (effective_parent);
|
|
||||||
gtk_window_set_modal (GTK_WINDOW (dialog), TRUE);
|
|
||||||
}
|
|
||||||
|
|
||||||
gtk_widget_show (dialog);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
@@ -972,7 +837,7 @@ gtkutil_copy_to_clipboard (GtkWidget *widget, GdkAtom selection,
|
|||||||
win = gtk_widget_get_toplevel (GTK_WIDGET (widget));
|
win = gtk_widget_get_toplevel (GTK_WIDGET (widget));
|
||||||
if (gtk_widget_is_toplevel (win))
|
if (gtk_widget_is_toplevel (win))
|
||||||
{
|
{
|
||||||
int len = strlen (str);
|
gint len = (gint) strlen (str);
|
||||||
|
|
||||||
if (selection)
|
if (selection)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -902,7 +902,12 @@ mg_windowstate_cb (GtkWindow *wid, GdkEventWindowState *event, gpointer userdata
|
|||||||
if ((event->changed_mask & GDK_WINDOW_STATE_ICONIFIED) &&
|
if ((event->changed_mask & GDK_WINDOW_STATE_ICONIFIED) &&
|
||||||
(event->new_window_state & GDK_WINDOW_STATE_ICONIFIED) &&
|
(event->new_window_state & GDK_WINDOW_STATE_ICONIFIED) &&
|
||||||
prefs.hex_gui_tray_minimize && prefs.hex_gui_tray &&
|
prefs.hex_gui_tray_minimize && prefs.hex_gui_tray &&
|
||||||
gtkutil_tray_icon_supported (wid))
|
gtkutil_tray_icon_supported (wid)
|
||||||
|
#ifndef WIN32
|
||||||
|
)
|
||||||
|
#else
|
||||||
|
&& !gtk_window_is_active (wid))
|
||||||
|
#endif
|
||||||
{
|
{
|
||||||
tray_toggle_visibility (TRUE);
|
tray_toggle_visibility (TRUE);
|
||||||
}
|
}
|
||||||
@@ -947,6 +952,7 @@ mg_windowstate_cb (GtkWindow *wid, GdkEventWindowState *event, gpointer userdata
|
|||||||
if (sess && sess->gui && GTK_IS_WIDGET (sess->gui->window))
|
if (sess && sess->gui && GTK_IS_WIDGET (sess->gui->window))
|
||||||
gtk_widget_queue_draw (sess->gui->window);
|
gtk_widget_queue_draw (sess->gui->window);
|
||||||
|
|
||||||
|
if (current_sess && current_sess->gui)
|
||||||
menu_set_fullscreen (current_sess->gui, prefs.hex_gui_win_fullscreen);
|
menu_set_fullscreen (current_sess->gui, prefs.hex_gui_win_fullscreen);
|
||||||
|
|
||||||
#ifdef G_OS_WIN32
|
#ifdef G_OS_WIN32
|
||||||
@@ -2412,9 +2418,9 @@ mg_userlist_button (GtkWidget * box, char *label, char *cmd,
|
|||||||
g_signal_connect (G_OBJECT (wid), "clicked",
|
g_signal_connect (G_OBJECT (wid), "clicked",
|
||||||
G_CALLBACK (userlist_button_cb), cmd);
|
G_CALLBACK (userlist_button_cb), cmd);
|
||||||
gtk_widget_set_hexpand (wid, TRUE);
|
gtk_widget_set_hexpand (wid, TRUE);
|
||||||
gtk_widget_set_vexpand (wid, TRUE);
|
gtk_widget_set_vexpand (wid, FALSE);
|
||||||
gtk_widget_set_halign (wid, GTK_ALIGN_FILL);
|
gtk_widget_set_halign (wid, GTK_ALIGN_FILL);
|
||||||
gtk_widget_set_valign (wid, GTK_ALIGN_FILL);
|
gtk_widget_set_valign (wid, GTK_ALIGN_CENTER);
|
||||||
gtk_grid_attach (GTK_GRID (box), wid, a, c, b - a, d - c);
|
gtk_grid_attach (GTK_GRID (box), wid, a, c, b - a, d - c);
|
||||||
show_and_unfocus (wid);
|
show_and_unfocus (wid);
|
||||||
}
|
}
|
||||||
@@ -2699,7 +2705,6 @@ mg_changui_destroy (session *sess)
|
|||||||
/* it fixes: Gdk-CRITICAL **: gdk_colormap_get_screen: */
|
/* it fixes: Gdk-CRITICAL **: gdk_colormap_get_screen: */
|
||||||
/* assertion `GDK_IS_COLORMAP (cmap)' failed */
|
/* assertion `GDK_IS_COLORMAP (cmap)' failed */
|
||||||
ret = sess->gui->window;
|
ret = sess->gui->window;
|
||||||
g_free (sess->gui);
|
|
||||||
sess->gui = NULL;
|
sess->gui = NULL;
|
||||||
}
|
}
|
||||||
return ret;
|
return ret;
|
||||||
@@ -2721,13 +2726,17 @@ mg_link_irctab (session *sess, int focus)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
session_gui *old_gui;
|
||||||
|
|
||||||
mg_unpopulate (sess);
|
mg_unpopulate (sess);
|
||||||
|
old_gui = sess->gui;
|
||||||
win = mg_changui_destroy (sess);
|
win = mg_changui_destroy (sess);
|
||||||
mg_changui_new (sess, sess->res, 1, focus);
|
mg_changui_new (sess, sess->res, 1, focus);
|
||||||
/* the buffer is now attached to a different widget */
|
/* the buffer is now attached to a different widget */
|
||||||
((xtext_buffer *)sess->res->buffer)->xtext = (GtkXText *)sess->gui->xtext;
|
((xtext_buffer *)sess->res->buffer)->xtext = (GtkXText *)sess->gui->xtext;
|
||||||
if (win)
|
if (win)
|
||||||
gtk_widget_destroy (win);
|
gtk_widget_destroy (win);
|
||||||
|
g_free (old_gui);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@@ -3413,8 +3422,13 @@ mg_create_textarea (session *sess, GtkWidget *box)
|
|||||||
inbox = mg_box_new (GTK_ORIENTATION_HORIZONTAL, FALSE, 2);
|
inbox = mg_box_new (GTK_ORIENTATION_HORIZONTAL, FALSE, 2);
|
||||||
gtk_box_pack_start (GTK_BOX (vbox), inbox, TRUE, TRUE, 0);
|
gtk_box_pack_start (GTK_BOX (vbox), inbox, TRUE, TRUE, 0);
|
||||||
|
|
||||||
frame = gtk_frame_new (NULL);
|
frame = gtk_scrolled_window_new (NULL, NULL);
|
||||||
gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
|
gtk_widget_set_hexpand (frame, TRUE);
|
||||||
|
gtk_widget_set_vexpand (frame, TRUE);
|
||||||
|
gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (frame),
|
||||||
|
GTK_SHADOW_IN);
|
||||||
|
gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (frame),
|
||||||
|
GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC);
|
||||||
gtk_box_pack_start (GTK_BOX (inbox), frame, TRUE, TRUE, 0);
|
gtk_box_pack_start (GTK_BOX (inbox), frame, TRUE, TRUE, 0);
|
||||||
|
|
||||||
theme_get_xtext_colors_for_widget (frame, xtext_palette, XTEXT_COLS);
|
theme_get_xtext_colors_for_widget (frame, xtext_palette, XTEXT_COLS);
|
||||||
@@ -3431,9 +3445,7 @@ mg_create_textarea (session *sess, GtkWidget *box)
|
|||||||
g_signal_connect (G_OBJECT (xtext), "word_click",
|
g_signal_connect (G_OBJECT (xtext), "word_click",
|
||||||
G_CALLBACK (mg_word_clicked), NULL);
|
G_CALLBACK (mg_word_clicked), NULL);
|
||||||
|
|
||||||
gui->vscrollbar = gtk_scrollbar_new (GTK_ORIENTATION_VERTICAL,
|
gui->vscrollbar = gtk_scrolled_window_get_vscrollbar (GTK_SCROLLED_WINDOW (frame));
|
||||||
GTK_XTEXT (xtext)->adj);
|
|
||||||
gtk_box_pack_start (GTK_BOX (inbox), gui->vscrollbar, FALSE, TRUE, 0);
|
|
||||||
|
|
||||||
gtk_drag_dest_set (gui->vscrollbar, 5, dnd_dest_targets, 2,
|
gtk_drag_dest_set (gui->vscrollbar, 5, dnd_dest_targets, 2,
|
||||||
GDK_ACTION_MOVE | GDK_ACTION_COPY | GDK_ACTION_LINK);
|
GDK_ACTION_MOVE | GDK_ACTION_COPY | GDK_ACTION_LINK);
|
||||||
@@ -3476,7 +3488,7 @@ mg_create_meters (session_gui *gui, GtkWidget *parent_box)
|
|||||||
GtkWidget *infbox, *wid, *box;
|
GtkWidget *infbox, *wid, *box;
|
||||||
|
|
||||||
gui->meter_box = infbox = box = mg_box_new (GTK_ORIENTATION_VERTICAL, FALSE, 1);
|
gui->meter_box = infbox = box = mg_box_new (GTK_ORIENTATION_VERTICAL, FALSE, 1);
|
||||||
gtk_box_pack_start (GTK_BOX (parent_box), box, 0, 0, 0);
|
gtk_box_pack_end (GTK_BOX (parent_box), box, 0, 0, 0);
|
||||||
|
|
||||||
if ((prefs.hex_gui_lagometer & 2) || (prefs.hex_gui_throttlemeter & 2))
|
if ((prefs.hex_gui_lagometer & 2) || (prefs.hex_gui_throttlemeter & 2))
|
||||||
{
|
{
|
||||||
@@ -3661,8 +3673,14 @@ mg_create_userlist (session_gui *gui, GtkWidget *box)
|
|||||||
gtk_box_pack_start (GTK_BOX (box), vbox, TRUE, TRUE, 0);
|
gtk_box_pack_start (GTK_BOX (box), vbox, TRUE, TRUE, 0);
|
||||||
|
|
||||||
gui->namelistinfo = gtk_label_new (NULL);
|
gui->namelistinfo = gtk_label_new (NULL);
|
||||||
gtk_label_set_xalign (GTK_LABEL (gui->namelistinfo), 0.0f);
|
gtk_label_set_xalign (GTK_LABEL (gui->namelistinfo), 0.5f);
|
||||||
gtk_widget_set_halign (gui->namelistinfo, GTK_ALIGN_START);
|
gtk_label_set_justify (GTK_LABEL (gui->namelistinfo), GTK_JUSTIFY_CENTER);
|
||||||
|
gtk_label_set_ellipsize (GTK_LABEL (gui->namelistinfo), PANGO_ELLIPSIZE_END);
|
||||||
|
gtk_label_set_width_chars (GTK_LABEL (gui->namelistinfo), 1);
|
||||||
|
gtk_widget_set_margin_start (gui->namelistinfo, 0);
|
||||||
|
gtk_widget_set_margin_end (gui->namelistinfo, 0);
|
||||||
|
gtk_widget_set_hexpand (gui->namelistinfo, TRUE);
|
||||||
|
gtk_widget_set_halign (gui->namelistinfo, GTK_ALIGN_FILL);
|
||||||
if (prefs.hex_gui_ulist_count)
|
if (prefs.hex_gui_ulist_count)
|
||||||
gtk_box_pack_start (GTK_BOX (vbox), gui->namelistinfo, 0, 0, 0);
|
gtk_box_pack_start (GTK_BOX (vbox), gui->namelistinfo, 0, 0, 0);
|
||||||
|
|
||||||
@@ -4718,7 +4736,14 @@ mg_win32_filter (GdkXEvent *xevent, GdkEvent *event, gpointer data)
|
|||||||
{
|
{
|
||||||
if (strcmp (command, "__WIN32_TASKBAR_TOGGLE__") == 0)
|
if (strcmp (command, "__WIN32_TASKBAR_TOGGLE__") == 0)
|
||||||
{
|
{
|
||||||
if (gtk_widget_get_visible (current_sess->gui->window))
|
GdkWindowState state = 0;
|
||||||
|
GdkWindow *gdk_window = gtk_widget_get_window (current_sess->gui->window);
|
||||||
|
|
||||||
|
if (gdk_window)
|
||||||
|
state = gdk_window_get_state (gdk_window);
|
||||||
|
|
||||||
|
if (gtk_widget_get_visible (current_sess->gui->window)
|
||||||
|
&& (state & GDK_WINDOW_STATE_ICONIFIED) == 0)
|
||||||
fe_ctrl_gui (current_sess, FE_GUI_ICONIFY, 0);
|
fe_ctrl_gui (current_sess, FE_GUI_ICONIFY, 0);
|
||||||
else
|
else
|
||||||
fe_ctrl_gui (current_sess, FE_GUI_SHOW, 0);
|
fe_ctrl_gui (current_sess, FE_GUI_SHOW, 0);
|
||||||
@@ -5232,9 +5257,14 @@ static void
|
|||||||
mg_handle_drop (GtkWidget *widget, int y, int *pos, int *other_pos)
|
mg_handle_drop (GtkWidget *widget, int y, int *pos, int *other_pos)
|
||||||
{
|
{
|
||||||
int height;
|
int height;
|
||||||
|
GdkWindow *window;
|
||||||
session_gui *gui = current_sess->gui;
|
session_gui *gui = current_sess->gui;
|
||||||
|
|
||||||
height = gdk_window_get_height (gtk_widget_get_window (widget));
|
window = gtk_widget_get_window (widget);
|
||||||
|
if (!window)
|
||||||
|
return;
|
||||||
|
|
||||||
|
height = gdk_window_get_height (window);
|
||||||
|
|
||||||
if (y < height / 2)
|
if (y < height / 2)
|
||||||
{
|
{
|
||||||
@@ -5312,6 +5342,9 @@ mg_drag_begin_cb (GtkWidget *widget, GdkDragContext *context, gpointer userdata)
|
|||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
window = gtk_widget_get_window (widget);
|
window = gtk_widget_get_window (widget);
|
||||||
|
if (!window)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
width = gdk_window_get_width (window);
|
width = gdk_window_get_width (window);
|
||||||
height = gdk_window_get_height (window);
|
height = gdk_window_get_height (window);
|
||||||
|
|
||||||
@@ -5389,11 +5422,16 @@ mg_drag_motion_cb (GtkWidget *widget, GdkDragContext *context, int x, int y, gui
|
|||||||
width = allocation.width;
|
width = allocation.width;
|
||||||
height = allocation.height;
|
height = allocation.height;
|
||||||
window = gtk_widget_get_window (widget);
|
window = gtk_widget_get_window (widget);
|
||||||
|
if (!window)
|
||||||
|
return FALSE;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
ox = oy = 0;
|
ox = oy = 0;
|
||||||
window = gtk_widget_get_window (widget);
|
window = gtk_widget_get_window (widget);
|
||||||
|
if (!window)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
width = gdk_window_get_width (window);
|
width = gdk_window_get_width (window);
|
||||||
height = gdk_window_get_height (window);
|
height = gdk_window_get_height (window);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2309,7 +2309,7 @@ menu_reorder (GtkMenu *menu, GtkWidget *item, int pos)
|
|||||||
if (pos < 0) /* position offset from end/bottom */
|
if (pos < 0) /* position offset from end/bottom */
|
||||||
{
|
{
|
||||||
GList *children = gtk_container_get_children (GTK_CONTAINER (menu));
|
GList *children = gtk_container_get_children (GTK_CONTAINER (menu));
|
||||||
int length = g_list_length (children);
|
gint length = (gint) g_list_length (children);
|
||||||
|
|
||||||
g_list_free (children);
|
g_list_free (children);
|
||||||
gtk_menu_reorder_child (menu, item, (length + pos) - 1);
|
gtk_menu_reorder_child (menu, item, (length + pos) - 1);
|
||||||
@@ -2381,7 +2381,7 @@ menu_add_sub (GtkWidget *menu, menu_entry *me)
|
|||||||
if (pos < 0) /* position offset from end/bottom */
|
if (pos < 0) /* position offset from end/bottom */
|
||||||
{
|
{
|
||||||
GList *children = gtk_container_get_children (GTK_CONTAINER (menu));
|
GList *children = gtk_container_get_children (GTK_CONTAINER (menu));
|
||||||
int length = g_list_length (children);
|
gint length = (gint) g_list_length (children);
|
||||||
|
|
||||||
g_list_free (children);
|
g_list_free (children);
|
||||||
pos = length + pos;
|
pos = length + pos;
|
||||||
|
|||||||
@@ -75,7 +75,7 @@ notification_backend_show (const char *title, const char *text)
|
|||||||
"Notify",
|
"Notify",
|
||||||
g_variant_builder_end (¶ms),
|
g_variant_builder_end (¶ms),
|
||||||
G_DBUS_CALL_FLAGS_NONE,
|
G_DBUS_CALL_FLAGS_NONE,
|
||||||
1000,
|
-1,
|
||||||
NULL,
|
NULL,
|
||||||
(GAsyncReadyCallback)on_notify_ready,
|
(GAsyncReadyCallback)on_notify_ready,
|
||||||
NULL);
|
NULL);
|
||||||
@@ -108,7 +108,7 @@ notification_backend_init (const char **error)
|
|||||||
"GetCapabilities",
|
"GetCapabilities",
|
||||||
NULL,
|
NULL,
|
||||||
G_DBUS_CALL_FLAGS_NONE,
|
G_DBUS_CALL_FLAGS_NONE,
|
||||||
30,
|
1000,
|
||||||
NULL,
|
NULL,
|
||||||
&err);
|
&err);
|
||||||
|
|
||||||
|
|||||||
@@ -920,7 +920,10 @@ tray_menu_notify_cb (GObject *tray, GParamSpec *pspec, gpointer user_data)
|
|||||||
{
|
{
|
||||||
if (!tray_backend_is_embedded ())
|
if (!tray_backend_is_embedded ())
|
||||||
{
|
{
|
||||||
tray_restore_timer = g_timeout_add(500, (GSourceFunc)tray_menu_try_restore, NULL);
|
if (!tray_restore_timer)
|
||||||
|
{
|
||||||
|
tray_restore_timer = g_timeout_add (500, (GSourceFunc) tray_menu_try_restore, NULL);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -936,9 +939,10 @@ tray_menu_notify_cb (GObject *tray, GParamSpec *pspec, gpointer user_data)
|
|||||||
static gboolean
|
static gboolean
|
||||||
tray_menu_try_restore (void)
|
tray_menu_try_restore (void)
|
||||||
{
|
{
|
||||||
tray_cleanup();
|
tray_restore_timer = 0;
|
||||||
tray_init();
|
tray_cleanup ();
|
||||||
return TRUE;
|
tray_init ();
|
||||||
|
return G_SOURCE_REMOVE;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
|||||||
@@ -193,6 +193,7 @@ static const setting appearance_advanced_settings[] =
|
|||||||
{ST_HEADER, N_("Advanced"),0,0,0},
|
{ST_HEADER, N_("Advanced"),0,0,0},
|
||||||
{ST_EFILE, N_ ("Background image:"), P_OFFSETNL (hex_text_background), 0, 0, sizeof prefs.hex_text_background},
|
{ST_EFILE, N_ ("Background image:"), P_OFFSETNL (hex_text_background), 0, 0, sizeof prefs.hex_text_background},
|
||||||
{ST_HSCALE, N_("Window opacity:"), P_OFFINTNL(hex_gui_transparency),0,0,0},
|
{ST_HSCALE, N_("Window opacity:"), P_OFFINTNL(hex_gui_transparency),0,0,0},
|
||||||
|
{ST_HSCALE, N_("Mouse wheel scroll speed (Slower ← → Faster):"), P_OFFINTNL(hex_gui_mouse_scroll_speed), 0, 0, 100},
|
||||||
|
|
||||||
{ST_END, 0, 0, 0, 0, 0}
|
{ST_END, 0, 0, 0, 0, 0}
|
||||||
};
|
};
|
||||||
@@ -582,6 +583,7 @@ static const setting advanced_settings[] =
|
|||||||
{ST_TOGGLE, N_("Display lists in compact mode"), P_OFFINTNL(hex_gui_compact), N_("Use less spacing between user list/channel tree rows."), 0, 0},
|
{ST_TOGGLE, N_("Display lists in compact mode"), P_OFFINTNL(hex_gui_compact), N_("Use less spacing between user list/channel tree rows."), 0, 0},
|
||||||
{ST_TOGGLE, N_("Use server time if supported"), P_OFFINTNL(hex_irc_cap_server_time), N_("Display timestamps obtained from server if it supports the time-server extension."), 0, 0},
|
{ST_TOGGLE, N_("Use server time if supported"), P_OFFINTNL(hex_irc_cap_server_time), N_("Display timestamps obtained from server if it supports the time-server extension."), 0, 0},
|
||||||
{ST_TOGGLE, N_("Automatically reconnect to servers on disconnect"), P_OFFINTNL(hex_net_auto_reconnect), 0, 0, 1},
|
{ST_TOGGLE, N_("Automatically reconnect to servers on disconnect"), P_OFFINTNL(hex_net_auto_reconnect), 0, 0, 1},
|
||||||
|
{ST_NUMBER, N_("Lag check interval:"), P_OFFINTNL(hex_net_lag_check), 0, (const char **)N_("seconds."), 9999},
|
||||||
{ST_NUMBER, N_("Auto reconnect delay:"), P_OFFINTNL(hex_net_reconnect_delay), 0, 0, 9999},
|
{ST_NUMBER, N_("Auto reconnect delay:"), P_OFFINTNL(hex_net_reconnect_delay), 0, 0, 9999},
|
||||||
{ST_NUMBER, N_("Auto join delay:"), P_OFFINTNL(hex_irc_join_delay), 0, 0, 9999},
|
{ST_NUMBER, N_("Auto join delay:"), P_OFFINTNL(hex_irc_join_delay), 0, 0, 9999},
|
||||||
{ST_MENU, N_("Ban Type:"), P_OFFINTNL(hex_irc_ban_type), N_("Attempt to use this banmask when banning or quieting. (requires irc_who_join)"), bantypemenu, 0},
|
{ST_MENU, N_("Ban Type:"), P_OFFINTNL(hex_irc_ban_type), N_("Attempt to use this banmask when banning or quieting. (requires irc_who_join)"), bantypemenu, 0},
|
||||||
@@ -655,6 +657,11 @@ static const setting network_settings[] =
|
|||||||
{ST_MENU, N_("Type:"), P_OFFINTNL(hex_net_proxy_type), 0, proxytypes, 0},
|
{ST_MENU, N_("Type:"), P_OFFINTNL(hex_net_proxy_type), 0, proxytypes, 0},
|
||||||
{ST_MENU, N_("Use proxy for:"), P_OFFINTNL(hex_net_proxy_use), 0, proxyuse, 0},
|
{ST_MENU, N_("Use proxy for:"), P_OFFINTNL(hex_net_proxy_use), 0, proxyuse, 0},
|
||||||
|
|
||||||
|
{ST_HEADER, N_("Connection Health"), 0, 0, 0, 0},
|
||||||
|
{ST_NUMBER, N_("TCP keepalive idle:"), P_OFFINTNL(hex_net_keepalive_idle), 0, (const char **)N_("seconds."), 7200},
|
||||||
|
{ST_NUMBER, N_("TCP keepalive interval:"), P_OFFINTNL(hex_net_keepalive_interval), 0, (const char **)N_("seconds."), 600},
|
||||||
|
{ST_NUMBER, N_("TCP keepalive probes:"), P_OFFINTNL(hex_net_keepalive_count), 0, 0, 20},
|
||||||
|
|
||||||
{ST_HEADER, N_("Proxy Authentication"), 0, 0, 0, 0},
|
{ST_HEADER, N_("Proxy Authentication"), 0, 0, 0, 0},
|
||||||
{ST_TOGGLE, N_("Use authentication (HTTP or SOCKS5 only)"), P_OFFINTNL(hex_net_proxy_auth), 0, 0, 0},
|
{ST_TOGGLE, N_("Use authentication (HTTP or SOCKS5 only)"), P_OFFINTNL(hex_net_proxy_auth), 0, 0, 0},
|
||||||
{ST_ENTRY, N_("Username:"), P_OFFSETNL(hex_net_proxy_user), 0, 0, sizeof prefs.hex_net_proxy_user},
|
{ST_ENTRY, N_("Username:"), P_OFFSETNL(hex_net_proxy_user), 0, 0, sizeof prefs.hex_net_proxy_user},
|
||||||
@@ -1364,7 +1371,7 @@ setup_entry_cb (GtkEntry *entry, setting *set)
|
|||||||
int size;
|
int size;
|
||||||
int pos;
|
int pos;
|
||||||
unsigned char *p = (unsigned char*)gtk_entry_get_text (entry);
|
unsigned char *p = (unsigned char*)gtk_entry_get_text (entry);
|
||||||
int len = strlen (p);
|
size_t len = strlen ((const char *) p);
|
||||||
|
|
||||||
/* need to truncate? */
|
/* need to truncate? */
|
||||||
if (len >= set->extra)
|
if (len >= set->extra)
|
||||||
@@ -2180,7 +2187,7 @@ unslash (char *dir)
|
|||||||
{
|
{
|
||||||
if (dir[0])
|
if (dir[0])
|
||||||
{
|
{
|
||||||
int len = strlen (dir) - 1;
|
size_t len = strlen (dir) - 1;
|
||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
if (dir[len] == '/' || dir[len] == '\\')
|
if (dir[len] == '/' || dir[len] == '\\')
|
||||||
#else
|
#else
|
||||||
|
|||||||
@@ -93,23 +93,14 @@ userlist_apply_saved_column_width (GtkTreeViewColumn *column, int width)
|
|||||||
static void
|
static void
|
||||||
userlist_update_min_width (session *sess)
|
userlist_update_min_width (session *sess)
|
||||||
{
|
{
|
||||||
GtkRequisition minimum;
|
|
||||||
GtkRequisition natural;
|
|
||||||
GtkWidget *scrolled_window;
|
GtkWidget *scrolled_window;
|
||||||
int width;
|
|
||||||
|
|
||||||
if (!sess || !sess->gui || !sess->gui->user_box || !sess->gui->namelistinfo || !sess->gui->user_tree)
|
if (!sess || !sess->gui || !sess->gui->user_tree)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
gtk_widget_get_preferred_size (sess->gui->namelistinfo, &minimum, &natural);
|
|
||||||
width = MAX (minimum.width, natural.width);
|
|
||||||
if (width < 1)
|
|
||||||
width = 1;
|
|
||||||
|
|
||||||
scrolled_window = gtk_widget_get_parent (sess->gui->user_tree);
|
scrolled_window = gtk_widget_get_parent (sess->gui->user_tree);
|
||||||
if (GTK_IS_SCROLLED_WINDOW (scrolled_window))
|
if (GTK_IS_SCROLLED_WINDOW (scrolled_window))
|
||||||
gtk_scrolled_window_set_min_content_width (GTK_SCROLLED_WINDOW (scrolled_window), width);
|
gtk_scrolled_window_set_min_content_width (GTK_SCROLLED_WINDOW (scrolled_window), 1);
|
||||||
gtk_widget_set_size_request (sess->gui->user_box, width, -1);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
GdkPixbuf *
|
GdkPixbuf *
|
||||||
@@ -909,17 +900,19 @@ userlist_create (GtkWidget *box)
|
|||||||
};
|
};
|
||||||
|
|
||||||
sw = gtk_scrolled_window_new (NULL, NULL);
|
sw = gtk_scrolled_window_new (NULL, NULL);
|
||||||
|
gtk_widget_set_hexpand (sw, TRUE);
|
||||||
|
gtk_widget_set_vexpand (sw, TRUE);
|
||||||
gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (sw),
|
gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (sw),
|
||||||
GTK_SHADOW_ETCHED_IN);
|
GTK_SHADOW_IN);
|
||||||
gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (sw),
|
gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (sw),
|
||||||
prefs.hex_gui_ulist_show_hosts ?
|
GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
|
||||||
GTK_POLICY_AUTOMATIC :
|
gtk_scrolled_window_set_min_content_width (GTK_SCROLLED_WINDOW (sw), 1);
|
||||||
GTK_POLICY_NEVER,
|
|
||||||
GTK_POLICY_AUTOMATIC);
|
|
||||||
gtk_box_pack_start (GTK_BOX (box), sw, TRUE, TRUE, 0);
|
gtk_box_pack_start (GTK_BOX (box), sw, TRUE, TRUE, 0);
|
||||||
gtk_widget_show (sw);
|
gtk_widget_show (sw);
|
||||||
|
|
||||||
treeview = gtk_tree_view_new ();
|
treeview = gtk_tree_view_new ();
|
||||||
|
gtk_widget_set_hexpand (treeview, TRUE);
|
||||||
|
gtk_widget_set_vexpand (treeview, TRUE);
|
||||||
gtk_widget_set_name (treeview, "zoitechat-userlist");
|
gtk_widget_set_name (treeview, "zoitechat-userlist");
|
||||||
gtk_widget_set_can_focus (treeview, TRUE);
|
gtk_widget_set_can_focus (treeview, TRUE);
|
||||||
gtk_tree_view_set_headers_visible (GTK_TREE_VIEW (treeview), FALSE);
|
gtk_tree_view_set_headers_visible (GTK_TREE_VIEW (treeview), FALSE);
|
||||||
|
|||||||
@@ -110,6 +110,15 @@ enum
|
|||||||
TARGET_COMPOUND_TEXT
|
TARGET_COMPOUND_TEXT
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
PROP_0,
|
||||||
|
PROP_HADJUSTMENT,
|
||||||
|
PROP_VADJUSTMENT,
|
||||||
|
PROP_HSCROLL_POLICY,
|
||||||
|
PROP_VSCROLL_POLICY
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
/* Selection targets for PRIMARY selection / copy-paste.
|
/* Selection targets for PRIMARY selection / copy-paste.
|
||||||
*
|
*
|
||||||
@@ -158,7 +167,8 @@ gtk_xtext_install_selection_targets (GtkWidget *widget)
|
|||||||
|
|
||||||
static guint xtext_signals[LAST_SIGNAL];
|
static guint xtext_signals[LAST_SIGNAL];
|
||||||
|
|
||||||
G_DEFINE_TYPE (GtkXText, gtk_xtext, GTK_TYPE_WIDGET)
|
G_DEFINE_TYPE_WITH_CODE (GtkXText, gtk_xtext, GTK_TYPE_WIDGET,
|
||||||
|
G_IMPLEMENT_INTERFACE (GTK_TYPE_SCROLLABLE, NULL))
|
||||||
|
|
||||||
char *nocasestrstr (const char *text, const char *tofind); /* util.c */
|
char *nocasestrstr (const char *text, const char *tofind); /* util.c */
|
||||||
int xtext_get_stamp_str (time_t, char **);
|
int xtext_get_stamp_str (time_t, char **);
|
||||||
@@ -171,6 +181,10 @@ static void gtk_xtext_adjustment_changed (GtkAdjustment * adj,
|
|||||||
GtkXText * xtext);
|
GtkXText * xtext);
|
||||||
static void gtk_xtext_scroll_adjustments (GtkXText *xtext, GtkAdjustment *hadj,
|
static void gtk_xtext_scroll_adjustments (GtkXText *xtext, GtkAdjustment *hadj,
|
||||||
GtkAdjustment *vadj);
|
GtkAdjustment *vadj);
|
||||||
|
static void gtk_xtext_set_property (GObject *object, guint prop_id,
|
||||||
|
const GValue *value, GParamSpec *pspec);
|
||||||
|
static void gtk_xtext_get_property (GObject *object, guint prop_id,
|
||||||
|
GValue *value, GParamSpec *pspec);
|
||||||
static int gtk_xtext_render_ents (GtkXText * xtext, textentry *, textentry *);
|
static int gtk_xtext_render_ents (GtkXText * xtext, textentry *, textentry *);
|
||||||
static void gtk_xtext_recalc_widths (xtext_buffer *buf, int);
|
static void gtk_xtext_recalc_widths (xtext_buffer *buf, int);
|
||||||
static void gtk_xtext_fix_indent (xtext_buffer *buf);
|
static void gtk_xtext_fix_indent (xtext_buffer *buf);
|
||||||
@@ -794,6 +808,9 @@ gtk_xtext_init (GtkXText * xtext)
|
|||||||
xtext->recycle = FALSE;
|
xtext->recycle = FALSE;
|
||||||
xtext->dont_render = FALSE;
|
xtext->dont_render = FALSE;
|
||||||
xtext->dont_render2 = FALSE;
|
xtext->dont_render2 = FALSE;
|
||||||
|
xtext->hadj = NULL;
|
||||||
|
xtext->hscroll_policy = GTK_SCROLL_MINIMUM;
|
||||||
|
xtext->vscroll_policy = GTK_SCROLL_MINIMUM;
|
||||||
gtk_xtext_scroll_adjustments (xtext, NULL, NULL);
|
gtk_xtext_scroll_adjustments (xtext, NULL, NULL);
|
||||||
|
|
||||||
gtk_xtext_install_selection_targets (GTK_WIDGET (xtext));
|
gtk_xtext_install_selection_targets (GTK_WIDGET (xtext));
|
||||||
@@ -886,6 +903,66 @@ gtk_xtext_adjustment_changed (GtkAdjustment * adj, GtkXText * xtext)
|
|||||||
xtext->buffer->old_value = xtext_adj_get_value (adj);
|
xtext->buffer->old_value = xtext_adj_get_value (adj);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gtk_xtext_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec)
|
||||||
|
{
|
||||||
|
GtkXText *xtext = GTK_XTEXT (object);
|
||||||
|
GtkAdjustment *adj;
|
||||||
|
|
||||||
|
switch (prop_id)
|
||||||
|
{
|
||||||
|
case PROP_HADJUSTMENT:
|
||||||
|
adj = g_value_get_object (value);
|
||||||
|
if (xtext->hadj == adj)
|
||||||
|
break;
|
||||||
|
if (xtext->hadj)
|
||||||
|
g_object_unref (xtext->hadj);
|
||||||
|
xtext->hadj = adj ? g_object_ref (adj) : NULL;
|
||||||
|
g_object_notify (object, "hadjustment");
|
||||||
|
break;
|
||||||
|
case PROP_VADJUSTMENT:
|
||||||
|
gtk_xtext_scroll_adjustments (xtext, NULL, g_value_get_object (value));
|
||||||
|
g_object_notify (object, "vadjustment");
|
||||||
|
break;
|
||||||
|
case PROP_HSCROLL_POLICY:
|
||||||
|
xtext->hscroll_policy = g_value_get_enum (value);
|
||||||
|
g_object_notify (object, "hscroll-policy");
|
||||||
|
break;
|
||||||
|
case PROP_VSCROLL_POLICY:
|
||||||
|
xtext->vscroll_policy = g_value_get_enum (value);
|
||||||
|
g_object_notify (object, "vscroll-policy");
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gtk_xtext_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec)
|
||||||
|
{
|
||||||
|
GtkXText *xtext = GTK_XTEXT (object);
|
||||||
|
|
||||||
|
switch (prop_id)
|
||||||
|
{
|
||||||
|
case PROP_HADJUSTMENT:
|
||||||
|
g_value_set_object (value, xtext->hadj);
|
||||||
|
break;
|
||||||
|
case PROP_VADJUSTMENT:
|
||||||
|
g_value_set_object (value, xtext->adj);
|
||||||
|
break;
|
||||||
|
case PROP_HSCROLL_POLICY:
|
||||||
|
g_value_set_enum (value, xtext->hscroll_policy);
|
||||||
|
break;
|
||||||
|
case PROP_VSCROLL_POLICY:
|
||||||
|
g_value_set_enum (value, xtext->vscroll_policy);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
GtkWidget *
|
GtkWidget *
|
||||||
gtk_xtext_new (const XTextColor *palette, int separator)
|
gtk_xtext_new (const XTextColor *palette, int separator)
|
||||||
{
|
{
|
||||||
@@ -952,6 +1029,11 @@ gtk_xtext_cleanup (GtkXText *xtext)
|
|||||||
xtext->adj = NULL;
|
xtext->adj = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (xtext->hadj)
|
||||||
|
{
|
||||||
|
g_object_unref (G_OBJECT (xtext->hadj));
|
||||||
|
xtext->hadj = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
if (xtext->hand_cursor)
|
if (xtext->hand_cursor)
|
||||||
{
|
{
|
||||||
@@ -2218,6 +2300,12 @@ gtk_xtext_leave_notify (GtkWidget * widget, GdkEventCrossing * event)
|
|||||||
xtext->hilight_ent = NULL;
|
xtext->hilight_ent = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (xtext->tooltip_stamp_set)
|
||||||
|
{
|
||||||
|
gtk_widget_set_tooltip_text (widget, NULL);
|
||||||
|
xtext->tooltip_stamp_set = FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -2384,7 +2472,7 @@ gtk_xtext_motion_notify (GtkWidget * widget, GdkEventMotion * event)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (xtext->urlcheck_function == NULL)
|
if (xtext->urlcheck_function == NULL)
|
||||||
return FALSE;
|
goto tooltip_check;
|
||||||
|
|
||||||
word_type = gtk_xtext_get_word_adjust (xtext, x, y, &word_ent, &offset, &len);
|
word_type = gtk_xtext_get_word_adjust (xtext, x, y, &word_ent, &offset, &len);
|
||||||
if (word_type > 0)
|
if (word_type > 0)
|
||||||
@@ -2422,6 +2510,46 @@ gtk_xtext_motion_notify (GtkWidget * widget, GdkEventMotion * event)
|
|||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
tooltip_check:
|
||||||
|
if (xtext->buffer->time_stamp && xtext->buffer->indent > 0 && x >= 0 && x < xtext->stamp_width)
|
||||||
|
{
|
||||||
|
textentry *ent = gtk_xtext_find_char (xtext, x, y, NULL, NULL);
|
||||||
|
if (ent && (!xtext->tooltip_stamp_set || xtext->tooltip_stamp != ent->stamp))
|
||||||
|
{
|
||||||
|
char tooltip[96];
|
||||||
|
strftime_utf8 (tooltip, sizeof (tooltip), "%Y-%m-%d", ent->stamp);
|
||||||
|
gtk_widget_set_tooltip_text (widget, tooltip);
|
||||||
|
xtext->tooltip_stamp = ent->stamp;
|
||||||
|
xtext->tooltip_stamp_set = TRUE;
|
||||||
|
}
|
||||||
|
if (ent)
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
else if (!xtext->buffer->time_stamp && x >= xtext->buffer->indent)
|
||||||
|
{
|
||||||
|
textentry *ent = gtk_xtext_find_char (xtext, x, y, NULL, NULL);
|
||||||
|
if (ent && ent->stamp && (!xtext->tooltip_stamp_set || xtext->tooltip_stamp != ent->stamp))
|
||||||
|
{
|
||||||
|
char tooltip[128];
|
||||||
|
char date[64];
|
||||||
|
char *stamp_text;
|
||||||
|
strftime_utf8 (date, sizeof (date), "%Y-%m-%d", ent->stamp);
|
||||||
|
xtext_get_stamp_str (ent->stamp, &stamp_text);
|
||||||
|
g_snprintf (tooltip, sizeof (tooltip), "%s %s", date, stamp_text);
|
||||||
|
gtk_widget_set_tooltip_text (widget, tooltip);
|
||||||
|
g_free (stamp_text);
|
||||||
|
xtext->tooltip_stamp = ent->stamp;
|
||||||
|
xtext->tooltip_stamp_set = TRUE;
|
||||||
|
}
|
||||||
|
if (ent)
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
else if (xtext->tooltip_stamp_set)
|
||||||
|
{
|
||||||
|
gtk_widget_set_tooltip_text (widget, NULL);
|
||||||
|
xtext->tooltip_stamp_set = FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
gtk_xtext_leave_notify (widget, NULL);
|
gtk_xtext_leave_notify (widget, NULL);
|
||||||
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
@@ -2856,19 +2984,41 @@ gtk_xtext_scroll (GtkWidget *widget, GdkEventScroll *event)
|
|||||||
{
|
{
|
||||||
GtkXText *xtext = GTK_XTEXT (widget);
|
GtkXText *xtext = GTK_XTEXT (widget);
|
||||||
gfloat new_value;
|
gfloat new_value;
|
||||||
|
gfloat step;
|
||||||
|
gdouble dx;
|
||||||
|
gdouble dy;
|
||||||
|
int direction = 0;
|
||||||
|
int speed = prefs.hex_gui_mouse_scroll_speed;
|
||||||
|
|
||||||
if (event->direction == GDK_SCROLL_UP) /* mouse wheel pageUp */
|
if (speed < 1)
|
||||||
|
speed = 1;
|
||||||
|
step = (xtext_adj_get_page_increment (xtext->adj) * speed) / 100.0f;
|
||||||
|
|
||||||
|
if (event->direction == GDK_SCROLL_SMOOTH &&
|
||||||
|
gdk_event_get_scroll_deltas ((GdkEvent *)event, &dx, &dy))
|
||||||
|
{
|
||||||
|
if (dy > 0)
|
||||||
|
direction = 1;
|
||||||
|
else if (dy < 0)
|
||||||
|
direction = -1;
|
||||||
|
}
|
||||||
|
else if (event->direction == GDK_SCROLL_UP)
|
||||||
|
direction = -1;
|
||||||
|
else if (event->direction == GDK_SCROLL_DOWN)
|
||||||
|
direction = 1;
|
||||||
|
|
||||||
|
if (direction < 0)
|
||||||
{
|
{
|
||||||
new_value = xtext_adj_get_value (xtext->adj) -
|
new_value = xtext_adj_get_value (xtext->adj) -
|
||||||
(xtext_adj_get_page_increment (xtext->adj) / 10);
|
step;
|
||||||
if (new_value < xtext_adj_get_lower (xtext->adj))
|
if (new_value < xtext_adj_get_lower (xtext->adj))
|
||||||
new_value = xtext_adj_get_lower (xtext->adj);
|
new_value = xtext_adj_get_lower (xtext->adj);
|
||||||
xtext_adj_set_value (xtext->adj, new_value);
|
xtext_adj_set_value (xtext->adj, new_value);
|
||||||
}
|
}
|
||||||
else if (event->direction == GDK_SCROLL_DOWN) /* mouse wheel pageDn */
|
else if (direction > 0)
|
||||||
{
|
{
|
||||||
new_value = xtext_adj_get_value (xtext->adj) +
|
new_value = xtext_adj_get_value (xtext->adj) +
|
||||||
(xtext_adj_get_page_increment (xtext->adj) / 10);
|
step;
|
||||||
if (new_value > (xtext_adj_get_upper (xtext->adj) -
|
if (new_value > (xtext_adj_get_upper (xtext->adj) -
|
||||||
xtext_adj_get_page_size (xtext->adj)))
|
xtext_adj_get_page_size (xtext->adj)))
|
||||||
new_value = xtext_adj_get_upper (xtext->adj) -
|
new_value = xtext_adj_get_upper (xtext->adj) -
|
||||||
@@ -2876,7 +3026,7 @@ gtk_xtext_scroll (GtkWidget *widget, GdkEventScroll *event)
|
|||||||
xtext_adj_set_value (xtext->adj, new_value);
|
xtext_adj_set_value (xtext->adj, new_value);
|
||||||
}
|
}
|
||||||
|
|
||||||
return FALSE;
|
return direction != 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@@ -2924,6 +3074,13 @@ gtk_xtext_class_init (GtkXTextClass * class)
|
|||||||
widget_class = (GtkWidgetClass *) class;
|
widget_class = (GtkWidgetClass *) class;
|
||||||
xtext_class = (GtkXTextClass *) class;
|
xtext_class = (GtkXTextClass *) class;
|
||||||
|
|
||||||
|
object_class->set_property = gtk_xtext_set_property;
|
||||||
|
object_class->get_property = gtk_xtext_get_property;
|
||||||
|
g_object_class_override_property (object_class, PROP_HADJUSTMENT, "hadjustment");
|
||||||
|
g_object_class_override_property (object_class, PROP_VADJUSTMENT, "vadjustment");
|
||||||
|
g_object_class_override_property (object_class, PROP_HSCROLL_POLICY, "hscroll-policy");
|
||||||
|
g_object_class_override_property (object_class, PROP_VSCROLL_POLICY, "vscroll-policy");
|
||||||
|
|
||||||
xtext_signals[WORD_CLICK] =
|
xtext_signals[WORD_CLICK] =
|
||||||
g_signal_new ("word_click",
|
g_signal_new ("word_click",
|
||||||
G_TYPE_FROM_CLASS (object_class),
|
G_TYPE_FROM_CLASS (object_class),
|
||||||
@@ -4830,13 +4987,12 @@ gtk_xtext_check_marker_visibility (GtkXText * xtext)
|
|||||||
static void
|
static void
|
||||||
gtk_xtext_unstrip_color (gint start, gint end, GSList *slp, GList **gl, gint maxo)
|
gtk_xtext_unstrip_color (gint start, gint end, GSList *slp, GList **gl, gint maxo)
|
||||||
{
|
{
|
||||||
gint off1, off2, curlen;
|
gint off1, off2;
|
||||||
GSList *cursl;
|
GSList *cursl;
|
||||||
offsets_t marks;
|
offsets_t marks;
|
||||||
offlen_t *meta;
|
offlen_t *meta;
|
||||||
|
|
||||||
off1 = 0;
|
off1 = 0;
|
||||||
curlen = 0;
|
|
||||||
cursl = slp;
|
cursl = slp;
|
||||||
while (cursl)
|
while (cursl)
|
||||||
{
|
{
|
||||||
@@ -4846,7 +5002,6 @@ gtk_xtext_unstrip_color (gint start, gint end, GSList *slp, GList **gl, gint max
|
|||||||
off1 = meta->off + start;
|
off1 = meta->off + start;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
curlen += meta->len;
|
|
||||||
start -= meta->len;
|
start -= meta->len;
|
||||||
end -= meta->len;
|
end -= meta->len;
|
||||||
cursl = g_slist_next (cursl);
|
cursl = g_slist_next (cursl);
|
||||||
@@ -4861,7 +5016,6 @@ gtk_xtext_unstrip_color (gint start, gint end, GSList *slp, GList **gl, gint max
|
|||||||
off2 = meta->off + end;
|
off2 = meta->off + end;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
curlen += meta->len;
|
|
||||||
end -= meta->len;
|
end -= meta->len;
|
||||||
cursl = g_slist_next (cursl);
|
cursl = g_slist_next (cursl);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -134,6 +134,9 @@ struct _GtkXText
|
|||||||
xtext_buffer *selection_buffer;
|
xtext_buffer *selection_buffer;
|
||||||
|
|
||||||
GtkAdjustment *adj;
|
GtkAdjustment *adj;
|
||||||
|
GtkAdjustment *hadj;
|
||||||
|
GtkScrollablePolicy hscroll_policy;
|
||||||
|
GtkScrollablePolicy vscroll_policy;
|
||||||
cairo_surface_t *background_surface; /* 0 = use palette[19] */
|
cairo_surface_t *background_surface; /* 0 = use palette[19] */
|
||||||
cairo_surface_t *background_clip_surface;
|
cairo_surface_t *background_clip_surface;
|
||||||
GdkWindow *draw_window; /* points to ->window */
|
GdkWindow *draw_window; /* points to ->window */
|
||||||
@@ -187,6 +190,8 @@ struct _GtkXText
|
|||||||
textentry *hilight_ent;
|
textentry *hilight_ent;
|
||||||
int hilight_start;
|
int hilight_start;
|
||||||
int hilight_end;
|
int hilight_end;
|
||||||
|
time_t tooltip_stamp;
|
||||||
|
unsigned int tooltip_stamp_set:1;
|
||||||
|
|
||||||
guint16 fontwidth[128]; /* each char's width, only the ASCII ones */
|
guint16 fontwidth[128]; /* each char's width, only the ASCII ones */
|
||||||
|
|
||||||
|
|||||||
@@ -313,7 +313,8 @@ fe_print_text (struct session *sess, char *text, time_t stamp,
|
|||||||
gboolean no_activity)
|
gboolean no_activity)
|
||||||
{
|
{
|
||||||
int dotime = FALSE;
|
int dotime = FALSE;
|
||||||
int comma, k, i = 0, j = 0, len = strlen (text);
|
int comma, k, i = 0, j = 0;
|
||||||
|
size_t len = strlen (text);
|
||||||
|
|
||||||
unsigned char *newtext = g_malloc (len + 1024);
|
unsigned char *newtext = g_malloc (len + 1024);
|
||||||
|
|
||||||
|
|||||||
@@ -26,12 +26,19 @@
|
|||||||
#include <windows.h>
|
#include <windows.h>
|
||||||
|
|
||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
|
#include <climits>
|
||||||
|
|
||||||
#include "typedef.h" // for ssize_t
|
#include "typedef.h" // for ssize_t
|
||||||
#include <enchant-provider.h>
|
#include <enchant-provider.h>
|
||||||
|
|
||||||
ENCHANT_PLUGIN_DECLARE ("win8")
|
ENCHANT_PLUGIN_DECLARE ("win8")
|
||||||
|
|
||||||
|
static int
|
||||||
|
size_to_int (size_t value)
|
||||||
|
{
|
||||||
|
return value > static_cast<size_t>(INT_MAX) ? INT_MAX : static_cast<int>(value);
|
||||||
|
}
|
||||||
|
|
||||||
static char *
|
static char *
|
||||||
utf16_to_utf8 (const wchar_t * const str, bool from_bcp47)
|
utf16_to_utf8 (const wchar_t * const str, bool from_bcp47)
|
||||||
{
|
{
|
||||||
@@ -136,7 +143,7 @@ static void
|
|||||||
win8_dict_add_to_personal (EnchantDict *dict, const char *const word, size_t len)
|
win8_dict_add_to_personal (EnchantDict *dict, const char *const word, size_t len)
|
||||||
{
|
{
|
||||||
auto checker = static_cast<ISpellChecker*>(dict->user_data);
|
auto checker = static_cast<ISpellChecker*>(dict->user_data);
|
||||||
wchar_t *wword = utf8_to_utf16 (word, static_cast<int>(len), false);
|
wchar_t *wword = utf8_to_utf16 (word, size_to_int (len), false);
|
||||||
|
|
||||||
checker->Add (wword);
|
checker->Add (wword);
|
||||||
std::free (wword);
|
std::free (wword);
|
||||||
@@ -146,7 +153,7 @@ static void
|
|||||||
win8_dict_add_to_session (EnchantDict *dict, const char *const word, size_t len)
|
win8_dict_add_to_session (EnchantDict *dict, const char *const word, size_t len)
|
||||||
{
|
{
|
||||||
auto checker = static_cast<ISpellChecker*>(dict->user_data);
|
auto checker = static_cast<ISpellChecker*>(dict->user_data);
|
||||||
wchar_t *wword = utf8_to_utf16 (word, static_cast<int>(len), false);
|
wchar_t *wword = utf8_to_utf16 (word, size_to_int (len), false);
|
||||||
|
|
||||||
checker->Ignore (wword);
|
checker->Ignore (wword);
|
||||||
std::free (wword);
|
std::free (wword);
|
||||||
@@ -156,7 +163,7 @@ static int
|
|||||||
win8_dict_check (EnchantDict *dict, const char *const word, size_t len)
|
win8_dict_check (EnchantDict *dict, const char *const word, size_t len)
|
||||||
{
|
{
|
||||||
auto checker = static_cast<ISpellChecker*>(dict->user_data);
|
auto checker = static_cast<ISpellChecker*>(dict->user_data);
|
||||||
wchar_t *wword = utf8_to_utf16 (word, static_cast<int>(len), false);
|
wchar_t *wword = utf8_to_utf16 (word, size_to_int (len), false);
|
||||||
IEnumSpellingError *errors;
|
IEnumSpellingError *errors;
|
||||||
ISpellingError *error = nullptr;
|
ISpellingError *error = nullptr;
|
||||||
HRESULT hr;
|
HRESULT hr;
|
||||||
@@ -184,7 +191,7 @@ static char **
|
|||||||
win8_dict_suggest (EnchantDict *dict, const char *const word, size_t len, size_t *out_n_suggs)
|
win8_dict_suggest (EnchantDict *dict, const char *const word, size_t len, size_t *out_n_suggs)
|
||||||
{
|
{
|
||||||
auto checker = static_cast<ISpellChecker*>(dict->user_data);
|
auto checker = static_cast<ISpellChecker*>(dict->user_data);
|
||||||
wchar_t *wword = utf8_to_utf16 (word, static_cast<int>(len), false);
|
wchar_t *wword = utf8_to_utf16 (word, size_to_int (len), false);
|
||||||
IEnumString *suggestions;
|
IEnumString *suggestions;
|
||||||
HRESULT hr;
|
HRESULT hr;
|
||||||
|
|
||||||
|
|||||||
@@ -30,8 +30,8 @@ PrivilegesRequired=none
|
|||||||
ShowComponentSizes=no
|
ShowComponentSizes=no
|
||||||
CreateUninstallRegKey=not IsTaskSelected('portable')
|
CreateUninstallRegKey=not IsTaskSelected('portable')
|
||||||
Uninstallable=not IsTaskSelected('portable')
|
Uninstallable=not IsTaskSelected('portable')
|
||||||
ArchitecturesAllowed=x64
|
ArchitecturesAllowed=x64compatible
|
||||||
ArchitecturesInstallIn64BitMode=x64
|
ArchitecturesInstallIn64BitMode=x64compatible
|
||||||
MinVersion=6.1
|
MinVersion=6.1
|
||||||
WizardImageFile={#PROJECTDIR}wizardimage.bmp
|
WizardImageFile={#PROJECTDIR}wizardimage.bmp
|
||||||
WizardSmallImageFile={#PROJECTDIR}wizardsmallimage.bmp
|
WizardSmallImageFile={#PROJECTDIR}wizardsmallimage.bmp
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
2.18.0
|
2.18.1
|
||||||
|
|||||||
Reference in New Issue
Block a user