mirror of
				https://github.com/Tha14/toxic.git
				synced 2025-10-26 09:06:45 +01:00 
			
		
		
		
	Compare commits
	
		
			114 Commits
		
	
	
		
			v0.8.3
			...
			TokTok-mas
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | ae94bc593b | ||
|  | 81eb58532e | ||
|  | 8464ea9a7a | ||
|  | b77bff35a1 | ||
|  | eb964b64c2 | ||
|  | 2ff9d29491 | ||
|  | 2640919318 | ||
|  | 2fcbc4fa1c | ||
|  | 4330bf5867 | ||
|  | 3f1b7cdd26 | ||
|  | 1e985c1456 | ||
|  | 61740bda85 | ||
|  | 0d8e6d713e | ||
|  | 39e4ff8bd6 | ||
|  | 0434ac186a | ||
|  | 8d9d51640c | ||
|  | c4c0c0d1f4 | ||
|  | 3f2826bd66 | ||
|  | 7b7ea0e386 | ||
|  | d35a38735b | ||
|  | f0c4906fdc | ||
|  | 56ba61e061 | ||
|  | 898d89e95a | ||
|  | 1fd1e27bdf | ||
|  | 8e84ac58d4 | ||
|  | 9d65997871 | ||
|  | da2889f3ab | ||
|  | 312b38d253 | ||
|  | 0554bf0240 | ||
|  | 53a7530e8a | ||
|  | 41be04a142 | ||
|  | 31f36318a2 | ||
|  | f882fdf608 | ||
|  | 7e1e410307 | ||
|  | c135c812c2 | ||
|  | 6c239193ab | ||
|  | de7db08352 | ||
|  | ba5ded9bc2 | ||
|  | 4581dee4fc | ||
|  | d75d6e8b60 | ||
|  | 142ce642f0 | ||
|  | 7dead5ec96 | ||
|  | ddcf224db2 | ||
|  | daf794c4a2 | ||
|  | dac0124f0f | ||
|  | 15b7a30925 | ||
|  | 77ab71f26f | ||
|  | 68e1ba312d | ||
|  | 752fc6d619 | ||
|  | 16bcb27ca7 | ||
|  | 71d7d355a6 | ||
|  | 4188b392cc | ||
|  | 811fbfbb1e | ||
|  | 32eb7d3040 | ||
|  | 42763905d7 | ||
|  | f64300d1d6 | ||
|  | 1a723f0e8e | ||
|  | a86884c40e | ||
|  | 3f02e119f4 | ||
|  | 1bbd50aac7 | ||
|  | e7a0c32a68 | ||
|  | 7560bc9547 | ||
|  | 2b43340c90 | ||
|  | ff1620c923 | ||
|  | 1303053a27 | ||
|  | 91f194c821 | ||
|  | 478762f76c | ||
|  | 4d96d6a753 | ||
|  | 3cdcfbf4e5 | ||
|  | 4c302da503 | ||
|  | 26b5fe8f9d | ||
|  | 22d60232fb | ||
|  | e428879beb | ||
|  | 3015138a5a | ||
|  | 9c06ad608b | ||
|  | 015dbd9a96 | ||
|  | a7466c3142 | ||
|  | f012007cc4 | ||
|  | dcf3baf60f | ||
|  | 4bda799a4b | ||
|  | bdeae33d48 | ||
|  | 47591d5298 | ||
|  | b5ace27a3e | ||
|  | b334622d36 | ||
|  | 4bfb344caa | ||
|  | 16d96d6faf | ||
|  | 0ab2bad226 | ||
|  | 68db926f9f | ||
|  | b270c1e8b7 | ||
|  | e7142e49fd | ||
|  | 610906d07f | ||
|  | 6f72a191ba | ||
|  | dd5fa236ae | ||
|  | 51e1ab94b3 | ||
|  | ddc8c53abf | ||
|  | 46513017e3 | ||
|  | 98cb7f58c0 | ||
|  | 206bf407fd | ||
|  | 0a8ac4de3b | ||
|  | 87d54acad0 | ||
|  | 45ff6d8bac | ||
|  | 437dd8baeb | ||
|  | b080236ee5 | ||
|  | 116bff8cef | ||
|  | ddeca171a0 | ||
|  | 127f9462e0 | ||
|  | 4b5a9abbd4 | ||
|  | bb2257973e | ||
|  | 12b9cd2386 | ||
|  | 2cbe8fa880 | ||
|  | 2e39bee05a | ||
|  | 05eda76643 | ||
|  | f7b73af9a7 | ||
|  | 73aaa44d12 | 
							
								
								
									
										1
									
								
								.github/CODEOWNERS
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.github/CODEOWNERS
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| /.github/	@TokTok/admins | ||||
							
								
								
									
										2
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| --- | ||||
| github: [JFreegman] | ||||
							
								
								
									
										17
									
								
								.github/settings.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								.github/settings.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| --- | ||||
| _extends: .github | ||||
|  | ||||
| repository: | ||||
|   name: toxic | ||||
|   description: An ncurses-based Tox client | ||||
|   topics: tox, console, chat | ||||
|  | ||||
| branches: | ||||
|   - name: "master" | ||||
|     protection: | ||||
|       required_status_checks: | ||||
|         contexts: | ||||
|           - Codacy/PR Quality Review | ||||
|           - CodeFactor | ||||
|           - Travis CI - Pull Request | ||||
|           - code-review/reviewable | ||||
							
								
								
									
										4
									
								
								.restyled.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								.restyled.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| --- | ||||
| restylers: | ||||
|   - astyle: | ||||
|       arguments: ["--options=astylerc"] | ||||
							
								
								
									
										43
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						
									
										43
									
								
								.travis.yml
									
									
									
									
									
								
							| @@ -1,11 +1,16 @@ | ||||
| --- | ||||
| language: python | ||||
| python: nightly | ||||
| dist: xenial | ||||
| os: linux | ||||
|  | ||||
| addons: | ||||
| jobs: | ||||
|   include: | ||||
|     - env: JOB=linux | ||||
|  | ||||
|       addons: | ||||
|         apt: | ||||
|           packages: | ||||
|       - astyle | ||||
|             - libalut-dev | ||||
|             - libconfig-dev | ||||
|             - libnotify-dev | ||||
| @@ -14,11 +19,11 @@ addons: | ||||
|             - libqrencode-dev | ||||
|             - libvpx-dev | ||||
|  | ||||
| cache: | ||||
|       cache: | ||||
|         directories: | ||||
|           - $HOME/cache | ||||
|  | ||||
| install: | ||||
|       install: | ||||
|         # Where to find libraries. | ||||
|         - export LD_LIBRARY_PATH=$HOME/cache/usr/lib | ||||
|         - export PKG_CONFIG_PATH=$HOME/cache/usr/lib/pkgconfig | ||||
| @@ -28,10 +33,28 @@ install: | ||||
|         # c-toxcore | ||||
|         - git clone --depth=1 https://github.com/TokTok/c-toxcore ../c-toxcore | ||||
|         - test -f $HOME/cache/usr/lib/libtoxcore.so || (cd ../c-toxcore && cmake -B_build -H. -DCMAKE_INSTALL_PREFIX:PATH=$HOME/cache/usr && make -C_build install -j$(nproc)) | ||||
|   # astyle | ||||
|   - wget -O ../astyle.tar.gz https://deb.debian.org/debian/pool/main/a/astyle/astyle_2.06.orig.tar.gz | ||||
|   - test -f $HOME/cache/astyle/build/gcc/bin/astyle || (tar -xf ../astyle.tar.gz -C "$HOME/cache" && make -C "$HOME/cache/astyle/build/gcc" -j2) | ||||
|  | ||||
| script: | ||||
|   - $HOME/cache/astyle/build/gcc/bin/astyle --options=astylerc $(find . -name "*.[ch]") | ||||
|   - make ENABLE_PYTHON=1 -j2 | ||||
|       script: | ||||
|         - make ENABLE_PYTHON=1 -j$(nproc) | ||||
|  | ||||
|     - env: JOB=macos | ||||
|       os: macos | ||||
|       language: c | ||||
|  | ||||
|       cache: | ||||
|         directories: | ||||
|           - $HOME/cache | ||||
|  | ||||
|       install: | ||||
|         - brew install | ||||
|           freealut | ||||
|           libconfig | ||||
|           libqrencode | ||||
|           libsodium | ||||
|           openal-soft | ||||
|         - export LDFLAGS="-L/usr/local/Cellar/openal-soft/1.21.0/lib" | ||||
|         - git clone --depth=1 https://github.com/TokTok/c-toxcore ../c-toxcore | ||||
|         - test -f /usr/local/lib/libtoxcore.dylib || (cd ../c-toxcore && cmake -B_build -H. && make -C_build install -j$(nproc)) | ||||
|  | ||||
|       script: | ||||
|         - make ENABLE_PYTHON=1 DISABLE_DESKTOP_NOTIFY=1 DISABLE_X11=1 DISABLE_AV=1 DISABLE_SOUND_NOTIFY=1 -j$(nproc) | ||||
|   | ||||
							
								
								
									
										25
									
								
								BUILD.bazel
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								BUILD.bazel
									
									
									
									
									
								
							| @@ -1,21 +1,30 @@ | ||||
| load("@rules_cc//cc:defs.bzl", "cc_binary") | ||||
| load("//tools/project:build_defs.bzl", "project") | ||||
|  | ||||
| project() | ||||
|  | ||||
| cc_binary( | ||||
|     name = "toxic", | ||||
|     srcs = glob([ | ||||
|     srcs = glob( | ||||
|         [ | ||||
|             "src/*.c", | ||||
|             "src/*.h", | ||||
|     ]), | ||||
|         ], | ||||
|         exclude = ["src/video*"], | ||||
|     ) + select({ | ||||
|         "//tools/config:linux": glob(["src/video*"]), | ||||
|         "//tools/config:osx": [], | ||||
|     }), | ||||
|     copts = [ | ||||
|         "-std=gnu99", | ||||
|         "-DAUDIO", | ||||
|         "-DPACKAGE_DATADIR='\"data\"'", | ||||
|         "-DPYTHON", | ||||
|         "-DQRCODE", | ||||
|         "-DVIDEO", | ||||
|         "-Wno-error=unused-result", | ||||
|     ], | ||||
|     ] + select({ | ||||
|         "//tools/config:linux": ["-DVIDEO"], | ||||
|         "//tools/config:osx": [], | ||||
|     }), | ||||
|     deps = [ | ||||
|         "//c-toxcore", | ||||
|         "@curl", | ||||
| @@ -25,6 +34,8 @@ cc_binary( | ||||
|         "@ncurses", | ||||
|         "@openal", | ||||
|         "@python3//:python", | ||||
|         "@x11", | ||||
|     ], | ||||
|     ] + select({ | ||||
|         "//tools/config:linux": ["@x11"], | ||||
|         "//tools/config:osx": [], | ||||
|     }), | ||||
| ) | ||||
|   | ||||
| @@ -55,8 +55,12 @@ Run `make doc` in the build directory after editing the asciidoc files to regene | ||||
|   * `DISABLE_X11=1` → Disable X11 support (needed for focus tracking) | ||||
|   * `DISABLE_AV=1` → Disable audio call support | ||||
|   * `DISABLE_SOUND_NOTIFY=1` → Disable sound notifications support | ||||
|   * `DISABLE_QRCODE` → Disable QR exporting support | ||||
|   * `DISABLE_QRPNG` → Disable support for exporting QR as PNG | ||||
|   * `DISABLE_DESKTOP_NOTIFY=1` → Disable desktop notifications support | ||||
|   * `ENABLE_PYTHON=1` → Build toxic with Python scripting support | ||||
|   * `ENABLE_RELEASE=1` → Build toxic without debug symbols and with full compiler optimizations | ||||
|   * `ENABLE_ASAN=1` → Build toxic with LLVM Address Sanitizer enabled | ||||
|  | ||||
| * `DESTDIR=""` Specifies the base install directory for binaries and data files (e.g.: DESTDIR="/tmp/build/pkg") | ||||
|  | ||||
|   | ||||
							
								
								
									
										23
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										23
									
								
								Makefile
									
									
									
									
									
								
							| @@ -5,8 +5,7 @@ CFG_DIR = $(BASE_DIR)/cfg | ||||
|  | ||||
| LIBS = toxcore ncursesw libconfig libcurl | ||||
|  | ||||
| CFLAGS ?= -g | ||||
| CFLAGS += -std=gnu99 -pthread -Wall -fstack-protector-all | ||||
| CFLAGS ?= -std=c99 -pthread -Wall -Wpedantic -Wunused -fstack-protector-all -Wvla -Wno-missing-braces | ||||
| CFLAGS += '-DTOXICVER="$(VERSION)"' -DHAVE_WIDECHAR -D_XOPEN_SOURCE_EXTENDED -D_FILE_OFFSET_BITS=64 | ||||
| CFLAGS += '-DPACKAGE_DATADIR="$(abspath $(DATADIR))"' | ||||
| CFLAGS += ${USER_CFLAGS} | ||||
| @@ -14,10 +13,26 @@ LDFLAGS ?= | ||||
| LDFLAGS += ${USER_LDFLAGS} | ||||
|  | ||||
| OBJ = autocomplete.o avatars.o bootstrap.o chat.o chat_commands.o configdir.o curl_util.o execute.o | ||||
| OBJ += file_transfers.o friendlist.o global_commands.o group_commands.o groupchat.o help.o input.o | ||||
| OBJ += file_transfers.o friendlist.o global_commands.o conference_commands.o conference.o help.o input.o | ||||
| OBJ += line_info.o log.o message_queue.o misc_tools.o name_lookup.o notify.o prompt.o qr_code.o settings.o | ||||
| OBJ += term_mplex.o toxic.o toxic_strings.o windows.o | ||||
|  | ||||
| # Check if debug build is enabled | ||||
| RELEASE := $(shell if [ -z "$(ENABLE_RELEASE)" ] || [ "$(ENABLE_RELEASE)" = "0" ] ; then echo disabled ; else echo enabled ; fi) | ||||
| ifneq ($(RELEASE), enabled) | ||||
| 	CFLAGS += -O0 -g -DDEBUG | ||||
| 	LDFLAGS += -O0 | ||||
| else | ||||
| 	CFLAGS += -O2 -flto | ||||
| 	LDFLAGS += -O2 -flto | ||||
| endif | ||||
|  | ||||
| # Check if LLVM Address Sanitizer is enabled | ||||
| ENABLE_ASAN := $(shell if [ -z "$(ENABLE_ASAN)" ] || [ "$(ENABLE_ASAN)" = "0" ] ; then echo disabled ; else echo enabled ; fi) | ||||
| ifneq ($(ENABLE_ASAN), disabled) | ||||
| 	CFLAGS += -fsanitize=address -fno-omit-frame-pointer | ||||
| endif | ||||
|  | ||||
| # Check on wich system we are running | ||||
| UNAME_S = $(shell uname -s) | ||||
| ifeq ($(UNAME_S), Linux) | ||||
| @@ -70,7 +85,7 @@ $(BUILD_DIR)/%.o: $(SRC_DIR)/%.c | ||||
| 	fi | ||||
| 	@echo "  CC    $(@:$(BUILD_DIR)/%=%)" | ||||
| 	@$(CC) $(CFLAGS) -o $(BUILD_DIR)/$*.o -c $(SRC_DIR)/$*.c | ||||
| 	@$(CC) -MM $(CFLAGS) $(SRC_DIR)/$*.c > $(BUILD_DIR)/$*.d | ||||
| 	@$(CC) -MM $(CFLAGS) $(SRC_DIR)/$*.c >$(BUILD_DIR)/$*.d | ||||
|  | ||||
| clean: | ||||
| 	rm -f $(BUILD_DIR)/*.d $(BUILD_DIR)/*.o $(BUILD_DIR)/toxic | ||||
|   | ||||
| @@ -3,9 +3,9 @@ | ||||
|        src="https://scan.coverity.com/projects/4975/badge.svg"/> | ||||
| </a> | ||||
|  | ||||
| Toxic is a [Tox](https://tox.chat)-based instant messenging and video chat client. | ||||
| Toxic is a [Tox](https://tox.chat)-based instant messaging and video chat client. | ||||
|  | ||||
| [](https://i.imgur.com/san99Z2.png) | ||||
| [](https://i.imgur.com/TwYA8L0.png) | ||||
|  | ||||
| ## Installation | ||||
| [See the install instructions](/INSTALL.md) | ||||
|   | ||||
| @@ -55,9 +55,9 @@ author = 'Jakob Kreuze' | ||||
| # built documents. | ||||
| # | ||||
| # The short X.Y version. | ||||
| version = '0.8.3' | ||||
| version = '0.10.0' | ||||
| # The full version, including alpha/beta/rc tags. | ||||
| release = '0.8.3' | ||||
| release = '0.10.0' | ||||
|  | ||||
| # The language for content autogenerated by Sphinx. Refer to documentation | ||||
| # for a list of supported languages. | ||||
|   | ||||
| @@ -8,13 +8,14 @@ else | ||||
| endif | ||||
|  | ||||
| # Check if we can build audio support | ||||
| CHECK_AUDIO_LIBS = $(shell $(PKG_CONFIG) --exists $(AUDIO_LIBS) || echo -n "error") | ||||
| CHECK_AUDIO_LIBS := $(shell $(PKG_CONFIG) --exists $(AUDIO_LIBS) || echo -n "error") | ||||
| ifneq ($(CHECK_AUDIO_LIBS), error) | ||||
|     LIBS += $(AUDIO_LIBS) | ||||
|     LDFLAGS += -lm | ||||
|     CFLAGS += $(AUDIO_CFLAGS) | ||||
|     OBJ += $(AUDIO_OBJ) | ||||
| else ifneq ($(MAKECMDGOALS), clean) | ||||
|     MISSING_AUDIO_LIBS = $(shell for lib in $(AUDIO_LIBS) ; do if ! $(PKG_CONFIG) --exists $$lib ; then echo $$lib ; fi ; done) | ||||
|     MISSING_AUDIO_LIBS := $(shell for lib in $(AUDIO_LIBS) ; do if ! $(PKG_CONFIG) --exists $$lib ; then echo $$lib ; fi ; done) | ||||
|     $(warning WARNING -- Toxic will be compiled without audio support) | ||||
|     $(warning WARNING -- You need these libraries for audio support) | ||||
|     $(warning WARNING -- $(MISSING_AUDIO_LIBS)) | ||||
|   | ||||
| @@ -1,19 +1,19 @@ | ||||
| CHECKS_DIR = $(CFG_DIR)/checks | ||||
|  | ||||
| # Check if we want build X11 support | ||||
| X11 = $(shell if [ -z "$(DISABLE_X11)" ] || [ "$(DISABLE_X11)" = "0" ] ; then echo enabled ; else echo disabled ; fi) | ||||
| X11 := $(shell if [ -z "$(DISABLE_X11)" ] || [ "$(DISABLE_X11)" = "0" ] ; then echo enabled ; else echo disabled ; fi) | ||||
| ifneq ($(X11), disabled) | ||||
|     -include $(CHECKS_DIR)/x11.mk | ||||
| endif | ||||
|  | ||||
| # Check if we want build audio support | ||||
| AUDIO = $(shell if [ -z "$(DISABLE_AV)" ] || [ "$(DISABLE_AV)" = "0" ] ; then echo enabled ; else echo disabled ; fi) | ||||
| AUDIO := $(shell if [ -z "$(DISABLE_AV)" ] || [ "$(DISABLE_AV)" = "0" ] ; then echo enabled ; else echo disabled ; fi) | ||||
| ifneq ($(AUDIO), disabled) | ||||
|     -include $(CHECKS_DIR)/audio.mk | ||||
| endif | ||||
|  | ||||
| # Check if we want build video support | ||||
| VIDEO = $(shell if [ -z "$(DISABLE_VI)" ] || [ "$(DISABLE_VI)" = "0" ] ; then echo enabled ; else echo disabled ; fi) | ||||
| VIDEO := $(shell if [ -z "$(DISABLE_VI)" ] || [ "$(DISABLE_VI)" = "0" ] ; then echo enabled ; else echo disabled ; fi) | ||||
| ifneq ($(X11), disabled) | ||||
| ifneq ($(AUDIO), disabled) | ||||
| ifneq ($(VIDEO), disabled) | ||||
| @@ -23,42 +23,42 @@ endif | ||||
| endif | ||||
|  | ||||
| # Check if we want build sound notifications support | ||||
| SND_NOTIFY = $(shell if [ -z "$(DISABLE_SOUND_NOTIFY)" ] || [ "$(DISABLE_SOUND_NOTIFY)" = "0" ] ; then echo enabled ; else echo disabled ; fi) | ||||
| SND_NOTIFY := $(shell if [ -z "$(DISABLE_SOUND_NOTIFY)" ] || [ "$(DISABLE_SOUND_NOTIFY)" = "0" ] ; then echo enabled ; else echo disabled ; fi) | ||||
| ifneq ($(SND_NOTIFY), disabled) | ||||
|     -include $(CHECKS_DIR)/sound_notifications.mk | ||||
| endif | ||||
|  | ||||
| # Check if we want build desktop notifications support | ||||
| DESK_NOTIFY = $(shell if [ -z "$(DISABLE_DESKTOP_NOTIFY)" ] || [ "$(DISABLE_DESKTOP_NOTIFY)" = "0" ] ; then echo enabled ; else echo disabled ; fi) | ||||
| DESK_NOTIFY := $(shell if [ -z "$(DISABLE_DESKTOP_NOTIFY)" ] || [ "$(DISABLE_DESKTOP_NOTIFY)" = "0" ] ; then echo enabled ; else echo disabled ; fi) | ||||
| ifneq ($(DESK_NOTIFY), disabled) | ||||
|     -include $(CHECKS_DIR)/desktop_notifications.mk | ||||
| endif | ||||
|  | ||||
| # Check if we want build QR export support | ||||
| QR_CODE = $(shell if [ -z "$(DISABLE_QRCODE)" ] || [ "$(DISABLE_QRCODE)" = "0" ] ; then echo enabled ; else echo disabled ; fi) | ||||
| QR_CODE := $(shell if [ -z "$(DISABLE_QRCODE)" ] || [ "$(DISABLE_QRCODE)" = "0" ] ; then echo enabled ; else echo disabled ; fi) | ||||
| ifneq ($(QR_CODE), disabled) | ||||
|     -include $(CHECKS_DIR)/qr.mk | ||||
| endif | ||||
|  | ||||
| # Check if we want build QR exported as PNG support | ||||
| QR_PNG = $(shell if [ -z "$(DISABLE_QRPNG)" ] || [ "$(DISABLE_QRPNG)" = "0" ] ; then echo enabled ; else echo disabled ; fi) | ||||
| QR_PNG := $(shell if [ -z "$(DISABLE_QRPNG)" ] || [ "$(DISABLE_QRPNG)" = "0" ] ; then echo enabled ; else echo disabled ; fi) | ||||
| ifneq ($(QR_PNG), disabled) | ||||
|     -include $(CHECKS_DIR)/qr_png.mk | ||||
| endif | ||||
|  | ||||
| # Check if we want build Python scripting support | ||||
| PYTHON = $(shell if [ -z "$(ENABLE_PYTHON)" ] || [ "$(ENABLE_PYTHON)" = "0" ] ; then echo disabled ; else echo enabled ; fi) | ||||
| PYTHON := $(shell if [ -z "$(ENABLE_PYTHON)" ] || [ "$(ENABLE_PYTHON)" = "0" ] ; then echo disabled ; else echo enabled ; fi) | ||||
| ifneq ($(PYTHON), disabled) | ||||
|     -include $(CHECKS_DIR)/python.mk | ||||
| endif | ||||
|  | ||||
| # Check if we can build Toxic | ||||
| CHECK_LIBS = $(shell $(PKG_CONFIG) --exists $(LIBS) || echo -n "error") | ||||
| CHECK_LIBS := $(shell $(PKG_CONFIG) --exists $(LIBS) || echo -n "error") | ||||
| ifneq ($(CHECK_LIBS), error) | ||||
|     CFLAGS += $(shell $(PKG_CONFIG) --cflags $(LIBS)) | ||||
|     LDFLAGS += $(shell $(PKG_CONFIG) --libs $(LIBS)) | ||||
| else ifneq ($(MAKECMDGOALS), clean) | ||||
|     MISSING_LIBS = $(shell for lib in $(LIBS) ; do if ! $(PKG_CONFIG) --exists $$lib ; then echo $$lib ; fi ; done) | ||||
|     MISSING_LIBS := $(shell for lib in $(LIBS) ; do if ! $(PKG_CONFIG) --exists $$lib ; then echo $$lib ; fi ; done) | ||||
|     $(warning ERROR -- Cannot compile Toxic) | ||||
|     $(warning ERROR -- You need these libraries) | ||||
|     $(warning ERROR -- $(MISSING_LIBS)) | ||||
|   | ||||
| @@ -3,12 +3,12 @@ DESK_NOTIFY_LIBS = libnotify | ||||
| DESK_NOTIFY_CFLAGS = -DBOX_NOTIFY | ||||
|  | ||||
| # Check if we can build desktop notifications support | ||||
| CHECK_DESK_NOTIFY_LIBS = $(shell $(PKG_CONFIG) --exists $(DESK_NOTIFY_LIBS) || echo -n "error") | ||||
| CHECK_DESK_NOTIFY_LIBS := $(shell $(PKG_CONFIG) --exists $(DESK_NOTIFY_LIBS) || echo -n "error") | ||||
| ifneq ($(CHECK_DESK_NOTIFY_LIBS), error) | ||||
|     LIBS += $(DESK_NOTIFY_LIBS) | ||||
|     CFLAGS += $(DESK_NOTIFY_CFLAGS) | ||||
| else ifneq ($(MAKECMDGOALS), clean) | ||||
|     MISSING_DESK_NOTIFY_LIBS = $(shell for lib in $(DESK_NOTIFY_LIBS) ; do if ! $(PKG_CONFIG) --exists $$lib ; then echo $$lib ; fi ; done) | ||||
|     MISSING_DESK_NOTIFY_LIBS := $(shell for lib in $(DESK_NOTIFY_LIBS) ; do if ! $(PKG_CONFIG) --exists $$lib ; then echo $$lib ; fi ; done) | ||||
|     $(warning WARNING -- Toxic will be compiled without desktop notifications support) | ||||
|     $(warning WARNING -- You need these libraries for desktop notifications support) | ||||
|     $(warning WARNING -- $(MISSING_DESK_NOTIFY_LIBS)) | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| # Variables for video call support | ||||
| VIDEO_LIBS = vpx x11 | ||||
| VIDEO_LIBS = openal vpx x11 | ||||
| VIDEO_CFLAGS = -DVIDEO | ||||
| ifneq (, $(findstring video_device.o, $(OBJ))) | ||||
|     VIDEO_OBJ = video_call.o | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| # Variables for X11 support | ||||
| X11_LIBS = x11 | ||||
| X11_CFLAGS = -DX11 | ||||
| X11_OBJ = xtra.o | ||||
| X11_OBJ = x11focus.o | ||||
|  | ||||
| # Check if we can build X11 support | ||||
| CHECK_X11_LIBS = $(shell $(PKG_CONFIG) --exists $(X11_LIBS) || echo -n "error") | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| # Version | ||||
| TOXIC_VERSION = 0.8.3 | ||||
| TOXIC_VERSION = 0.10.0 | ||||
| REV = $(shell git rev-list HEAD --count 2>/dev/null || echo -n "error") | ||||
| ifneq (, $(findstring error, $(REV))) | ||||
|     VERSION = $(TOXIC_VERSION) | ||||
|   | ||||
| @@ -17,6 +17,8 @@ help: | ||||
| 	@echo "  DISABLE_QRCODE:         Set to \"1\" to force building without QR export support" | ||||
| 	@echo "  DISABLE_QRPNG:          Set to \"1\" to force building without QR exported as PNG support" | ||||
| 	@echo "  ENABLE_PYTHON:          Set to \"1\" to enable building with Python scripting support" | ||||
| 	@echo "  ENABLE_RELEASE:         Set to \"1\" to build without debug symbols and with full compiler optimizations" | ||||
| 	@echo "  ENABLE_ASAN:            Set to \"1\" to build with LLVM address sanitizer enabled. | ||||
| 	@echo "  USER_CFLAGS:            Add custom flags to default CFLAGS" | ||||
| 	@echo "  USER_LDFLAGS:           Add custom flags to default LDFLAGS" | ||||
| 	@echo "  PREFIX:                 Specify a prefix directory for binaries, data files,... (default is \"$(abspath $(PREFIX))\")" | ||||
|   | ||||
							
								
								
									
										11
									
								
								doc/toxic.1
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								doc/toxic.1
									
									
									
									
									
								
							| @@ -1,13 +1,13 @@ | ||||
| '\" t | ||||
| .\"     Title: toxic | ||||
| .\"    Author: [see the "AUTHORS" section] | ||||
| .\" Generator: DocBook XSL Stylesheets v1.78.1 <http://docbook.sf.net/> | ||||
| .\"      Date: 2016-09-20 | ||||
| .\" Generator: DocBook XSL Stylesheets v1.79.1 <http://docbook.sf.net/> | ||||
| .\"      Date: 2020-05-04 | ||||
| .\"    Manual: Toxic Manual | ||||
| .\"    Source: toxic __VERSION__ | ||||
| .\"  Language: English | ||||
| .\" | ||||
| .TH "TOXIC" "1" "2016\-09\-20" "toxic __VERSION__" "Toxic Manual" | ||||
| .TH "TOXIC" "1" "2020\-05\-04" "toxic __VERSION__" "Toxic Manual" | ||||
| .\" ----------------------------------------------------------------- | ||||
| .\" * Define some portability stuff | ||||
| .\" ----------------------------------------------------------------- | ||||
| @@ -78,6 +78,11 @@ instead of | ||||
| Show help message | ||||
| .RE | ||||
| .PP | ||||
| \-l, \-\-logging | ||||
| .RS 4 | ||||
| Enable toxcore logging to stderr | ||||
| .RE | ||||
| .PP | ||||
| \-n, \-\-nodes nodes\-file | ||||
| .RS 4 | ||||
| Use specified | ||||
|   | ||||
| @@ -40,6 +40,9 @@ OPTIONS | ||||
| -h, --help:: | ||||
|     Show help message | ||||
|  | ||||
| -l, --logging:: | ||||
|     Enable toxcore logging to stderr | ||||
|  | ||||
| -n, --nodes nodes-file:: | ||||
|     Use specified 'nodes-file' for DHT bootstrap nodes instead of '~/.config/tox/DHTnodes.json' | ||||
|  | ||||
|   | ||||
| @@ -2,12 +2,12 @@ | ||||
| .\"     Title: toxic.conf | ||||
| .\"    Author: [see the "AUTHORS" section] | ||||
| .\" Generator: DocBook XSL Stylesheets v1.79.1 <http://docbook.sf.net/> | ||||
| .\"      Date: 2016-09-20 | ||||
| .\"      Date: 2020-11-18 | ||||
| .\"    Manual: Toxic Manual | ||||
| .\"    Source: toxic __VERSION__ | ||||
| .\"  Language: English | ||||
| .\" | ||||
| .TH "TOXIC\&.CONF" "5" "2016\-09\-20" "toxic __VERSION__" "Toxic Manual" | ||||
| .TH "TOXIC\&.CONF" "5" "2020\-11\-18" "toxic __VERSION__" "Toxic Manual" | ||||
| .\" ----------------------------------------------------------------- | ||||
| .\" * Define some portability stuff | ||||
| .\" ----------------------------------------------------------------- | ||||
| @@ -93,6 +93,26 @@ Enable or disable acoustic alerts on events\&. true or false | ||||
| Select between native terminal colors and toxic color theme\&. true or false | ||||
| .RE | ||||
| .PP | ||||
| \fBcolor_bar_bg\fR | ||||
| .RS 4 | ||||
| set background color of chat status bars\&. (black, white, red, green, blue, cyan, yellow, magenta) | ||||
| .RE | ||||
| .PP | ||||
| \fBcolor_bar_fg\fR | ||||
| .RS 4 | ||||
| set foreground (text) color of chat status bars\&. (black, white, red, green, blue, cyan, yellow, magenta) | ||||
| .RE | ||||
| .PP | ||||
| \fBcolor_bar_accent\fR | ||||
| .RS 4 | ||||
| set foreground accent color of chat status bars\&. (black, white, red, green, blue, cyan, yellow, magenta) | ||||
| .RE | ||||
| .PP | ||||
| \fBcolor_bar_notify\fR | ||||
| .RS 4 | ||||
| set foreground notify (and typing) color in chat status bar\&. (black, white, red, green, blue, cyan, yellow, magenta) | ||||
| .RE | ||||
| .PP | ||||
| \fBautolog\fR | ||||
| .RS 4 | ||||
| Enable or disable autologging\&. true or false | ||||
| @@ -120,7 +140,12 @@ Enable friend connection change notifications\&. true or false | ||||
| .PP | ||||
| \fBnodelist_update_freq\fR | ||||
| .RS 4 | ||||
| How often in days to update the DHT nodes list\&. (0 to disable updates) | ||||
| How often in days to update the DHT nodes list\&. (integer; 0 to disable) | ||||
| .RE | ||||
| .PP | ||||
| \fBautosave_freq\fR | ||||
| .RS 4 | ||||
| How often in seconds to auto\-save the Tox data file\&. (integer; 0 to disable) | ||||
| .RE | ||||
| .PP | ||||
| \fBhistory_size\fR | ||||
| @@ -128,6 +153,11 @@ How often in days to update the DHT nodes list\&. (0 to disable updates) | ||||
| Maximum lines for chat window history\&. Integer value\&. (for example: 700) | ||||
| .RE | ||||
| .PP | ||||
| \fBnotification_timeout\fR | ||||
| .RS 4 | ||||
| Time in milliseconds to display a notification\&. Integer value\&. (for example: 3000) | ||||
| .RE | ||||
| .PP | ||||
| \fBline_join\fR | ||||
| .RS 4 | ||||
| Indicator for when someone connects or joins a group\&. Three characters max for line_ settings\&. | ||||
| @@ -207,9 +237,24 @@ Audio output device\&. Integer value\&. Number corresponds to | ||||
| /lsdev out | ||||
| .RE | ||||
| .PP | ||||
| \fBVAD_treshold\fR | ||||
| \fBVAD_threshold\fR | ||||
| .RS 4 | ||||
| Voice Activity Detection treshold\&. Float value\&. Recommended values are around 40\&.0 | ||||
| Voice Activity Detection threshold\&. Float value\&. Recommended values are 1\&.0\-40\&.0 | ||||
| .RE | ||||
| .PP | ||||
| \fBconference_audio_channels\fR | ||||
| .RS 4 | ||||
| Number of channels for conference audio broadcast\&. Integer value\&. 1 (mono) or 2 (stereo) | ||||
| .RE | ||||
| .PP | ||||
| \fBchat_audio_channels\fR | ||||
| .RS 4 | ||||
| Number of channels for 1\-on\-1 audio broadcast\&. Integer value\&. 1 (mono) or 2 (stereo) | ||||
| .RE | ||||
| .PP | ||||
| \fBpush_to_talk\fR | ||||
| .RS 4 | ||||
| Enable/Disable Push\-To\-Talk for conference audio chats (active key is F2)\&. true or false | ||||
| .RE | ||||
| .RE | ||||
| .PP | ||||
| @@ -341,16 +386,6 @@ Key combination to scroll half page down\&. | ||||
| Key combination to scroll to page bottom\&. | ||||
| .RE | ||||
| .PP | ||||
| \fBpeer_list_up\fR | ||||
| .RS 4 | ||||
| Key combination to scroll contacts list up\&. | ||||
| .RE | ||||
| .PP | ||||
| \fBpeer_list_down\fR | ||||
| .RS 4 | ||||
| Key combination to scroll contacts list down\&. | ||||
| .RE | ||||
| .PP | ||||
| \fBtoggle_peerlist\fR | ||||
| .RS 4 | ||||
| Toggle the peer list on and off\&. | ||||
|   | ||||
| @@ -60,6 +60,18 @@ OPTIONS | ||||
|     *native_colors*;; | ||||
|         Select between native terminal colors and toxic color theme. true or false | ||||
|  | ||||
|     *color_bar_bg*;; | ||||
|         set background color of chat status bars. (black, white, red, green, blue, cyan, yellow, magenta) | ||||
|  | ||||
|     *color_bar_fg*;; | ||||
|         set foreground (text) color of chat status bars. (black, white, red, green, blue, cyan, yellow, magenta) | ||||
|  | ||||
|     *color_bar_accent*;; | ||||
|         set foreground accent color of chat status bars. (black, white, red, green, blue, cyan, yellow, magenta) | ||||
|  | ||||
|     *color_bar_notify*;; | ||||
|         set foreground notify (and typing) color in chat status bar. (black, white, red, green, blue, cyan, yellow, magenta) | ||||
|  | ||||
|     *autolog*;; | ||||
|         Enable or disable autologging. true or false | ||||
|  | ||||
| @@ -76,11 +88,17 @@ OPTIONS | ||||
|         Enable friend connection change notifications. true or false | ||||
|  | ||||
|     *nodelist_update_freq*;; | ||||
|         How often in days to update the DHT nodes list. (0 to disable updates) | ||||
|         How often in days to update the DHT nodes list. (integer; 0 to disable) | ||||
|  | ||||
|     *autosave_freq*;; | ||||
|         How often in seconds to auto-save the Tox data file. (integer; 0 to disable) | ||||
|  | ||||
|     *history_size*;; | ||||
|         Maximum lines for chat window history. Integer value. (for example: 700) | ||||
|  | ||||
|     *notification_timeout*;; | ||||
|         Time in milliseconds to display a notification. Integer value. (for example: 3000) | ||||
|  | ||||
|     *line_join*;; | ||||
|         Indicator for when someone connects or joins a group. | ||||
|         Three characters max for line_ settings. | ||||
| @@ -129,9 +147,18 @@ OPTIONS | ||||
|     *output_device*;; | ||||
|         Audio output device. Integer value. Number corresponds to `/lsdev out` | ||||
|  | ||||
|     *VAD_treshold*;; | ||||
|         Voice Activity Detection treshold.  Float value. Recommended values are | ||||
|         around 40.0 | ||||
|     *VAD_threshold*;; | ||||
|         Voice Activity Detection threshold.  Float value. Recommended values are | ||||
|         1.0-40.0 | ||||
|  | ||||
|     *conference_audio_channels*;; | ||||
|         Number of channels for conference audio broadcast. Integer value. 1 (mono) or 2 (stereo) | ||||
|  | ||||
|     *chat_audio_channels*;; | ||||
|         Number of channels for 1-on-1 audio broadcast. Integer value. 1 (mono) or 2 (stereo) | ||||
|  | ||||
|     *push_to_talk*;; | ||||
|         Enable/Disable Push-To-Talk for conference audio chats (active key is F2). true or false | ||||
|  | ||||
| *tox*:: | ||||
|     Configuration related to paths. | ||||
| @@ -215,12 +242,6 @@ OPTIONS | ||||
|     *page_bottom*;; | ||||
|         Key combination to scroll to page bottom. | ||||
|  | ||||
|     *peer_list_up*;; | ||||
|         Key combination to scroll contacts list up. | ||||
|  | ||||
|     *peer_list_down*;; | ||||
|         Key combination to scroll contacts list down. | ||||
|  | ||||
|     *toggle_peerlist*;; | ||||
|         Toggle the peer list on and off. | ||||
|  | ||||
|   | ||||
| @@ -9,20 +9,32 @@ ui = { | ||||
|   alerts=true; | ||||
|  | ||||
|   // Output a bell when receiving a message (see manpage) | ||||
|   bell_on_message=true | ||||
|   bell_on_message=true; | ||||
|  | ||||
|   // Output a bell when receiving a filetransfer (see manpage) | ||||
|   bell_on_filetrans=true | ||||
|   bell_on_filetrans=true; | ||||
|  | ||||
|   // Don't output a bell when a filetransfer was accepted (see manpage) | ||||
|   bell_on_filetrans_accept=false | ||||
|   bell_on_filetrans_accept=false; | ||||
|  | ||||
|   // Output a bell when receiving a group/call invite (see manpage) | ||||
|   bell_on_invite=true | ||||
|   bell_on_invite=true; | ||||
|  | ||||
|   // true to use native terminal colours, false to use toxic default colour theme | ||||
|   native_colors=false; | ||||
|  | ||||
|   // set background color of chat status bars (black, white, red, green, blue, cyan, yellow, magenta) | ||||
|   color_bar_bg="blue"; | ||||
|  | ||||
|   // set foreground (text) color of chat status bars (black, white, red, green, blue, cyan, yellow, magenta) | ||||
|   color_bar_fg="white"; | ||||
|  | ||||
|   // set foreground accent color of chat status bars (black, white, red, green, blue, cyan, yellow, magenta) | ||||
|   color_bar_accent="cyan"; | ||||
|  | ||||
|   // set foreground notify (and typing) color in chat status bar (black, white, red, green, blue, cyan, yellow, magenta) | ||||
|   color_bar_notify="yellow"; | ||||
|  | ||||
|   // true to enable autologging, false to disable | ||||
|   autolog=false; | ||||
|  | ||||
| @@ -30,7 +42,7 @@ ui = { | ||||
|   time_format=24; | ||||
|  | ||||
|   // Timestamp format string according to date/strftime format. Overrides time_format setting | ||||
|   timestamp_format="%H:%M:%S"; | ||||
|   timestamp_format="%H:%M"; | ||||
|  | ||||
|   // true to show you when others are typing a message in 1-on-1 chats | ||||
|   show_typing_other=true; | ||||
| @@ -47,20 +59,26 @@ ui = { | ||||
|   // How often in days to update the DHT nodes list. (0 to disable updates) | ||||
|   nodeslist_update_freq=7; | ||||
|  | ||||
|   // How often in seconds to auto-save the Tox data file. (0 to disable periodic auto-saves) | ||||
|   autosave_freq=600; | ||||
|  | ||||
|   // maximum lines for chat window history | ||||
|   history_size=700; | ||||
|  | ||||
|   // Indicator for display when someone connects or joins a group. | ||||
|   // time in milliseconds to display a notification | ||||
|   notification_timeout=6000; | ||||
|  | ||||
|   // Indicator for display when someone connects or joins a group | ||||
|   line_join="-->"; | ||||
|  | ||||
|   // Indicator for display when someone disconnects or leaves a group. | ||||
|   // Indicator for display when someone disconnects or leaves a group | ||||
|   line_quit="<--"; | ||||
|  | ||||
|   // Indicator for alert messages. | ||||
|   line_alert="-!-"; | ||||
|  | ||||
|   // Indicator for normal messages. | ||||
|   line_normal="---"; | ||||
|   line_normal="-"; | ||||
|  | ||||
|   // true to change status based on screen/tmux attach/detach, false to disable | ||||
|   mplex_away=true; | ||||
| @@ -76,8 +94,17 @@ audio = { | ||||
|   // preferred audio output device; numbers correspond to /lsdev out | ||||
|   output_device=0; | ||||
|  | ||||
|   // default VAD treshold; float (recommended values are around 40) | ||||
|   VAD_treshold=40.0; | ||||
|   // default VAD threshold; float (recommended values are 1.0-40.0) | ||||
|   VAD_threshold=5.0; | ||||
|  | ||||
|   // Number of channels to use for conference audio broadcasts; 1 for mono, 2 for stereo. | ||||
|   conference_audio_channels=1; | ||||
|  | ||||
|   // Number of channels to use for 1-on-1 audio broadcasts; 1 for mono, 2 for stereo. | ||||
|   chat_audio_channels=2; | ||||
|  | ||||
|   // toggle conference push-to-talk | ||||
|   push_to_talk=false; | ||||
| }; | ||||
|  | ||||
| tox = { | ||||
| @@ -116,8 +143,6 @@ keys = { | ||||
|   half_page_up="Ctrl+F"; | ||||
|   half_page_down="Ctrl+V"; | ||||
|   page_bottom="Ctrl+H"; | ||||
|   peer_list_up="Ctrl+["; | ||||
|   peer_list_down="Ctrl+]"; | ||||
|   toggle_peerlist="Ctrl+b"; | ||||
|   toggle_peerlist="Ctrl+B"; | ||||
|   toggle_paste_mode="Ctrl+T"; | ||||
| }; | ||||
|   | ||||
							
								
								
									
										30
									
								
								src/api.c
									
									
									
									
									
								
							
							
						
						
									
										30
									
								
								src/api.c
									
									
									
									
									
								
							| @@ -99,18 +99,16 @@ void api_send(const char *msg) | ||||
|     } | ||||
|  | ||||
|     char *name = api_get_nick(); | ||||
|     char  timefrmt[TIME_STR_SIZE]; | ||||
|  | ||||
|     if (name == NULL) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     self_window = get_active_window(); | ||||
|     get_time_str(timefrmt, sizeof(timefrmt)); | ||||
|  | ||||
|     strncpy((char *) self_window->chatwin->line, msg, sizeof(self_window->chatwin->line)); | ||||
|     add_line_to_hist(self_window->chatwin); | ||||
|     int id = line_info_add(self_window, timefrmt, name, NULL, OUT_MSG, 0, 0, "%s", msg); | ||||
|     int id = line_info_add(self_window, true, name, NULL, OUT_MSG, 0, 0, "%s", msg); | ||||
|     cqueue_add(self_window->chatwin->cqueue, msg, strlen(msg), OUT_MSG, id); | ||||
|     free(name); | ||||
| } | ||||
| @@ -156,7 +154,7 @@ void cmd_run(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX | ||||
|             error_str = "Only one argument allowed."; | ||||
|         } | ||||
|  | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, error_str); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, error_str); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -165,7 +163,7 @@ void cmd_run(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX | ||||
|     if (fp == NULL) { | ||||
|         error_str = "Path does not exist."; | ||||
|  | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, error_str); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, error_str); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -175,36 +173,36 @@ void cmd_run(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX | ||||
|  | ||||
| void invoke_autoruns(WINDOW *window, ToxWindow *self) | ||||
| { | ||||
|     struct dirent *dir; | ||||
|     char    abspath_buf[PATH_MAX + 1], err_buf[PATH_MAX + 1]; | ||||
|     size_t  path_len; | ||||
|     DIR    *d; | ||||
|     FILE   *fp; | ||||
|     char abspath_buf[PATH_MAX + 256]; | ||||
|     char err_buf[PATH_MAX + 128]; | ||||
|  | ||||
|     if (user_settings->autorun_path[0] == '\0') { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     d = opendir(user_settings->autorun_path); | ||||
|     DIR *d = opendir(user_settings->autorun_path); | ||||
|  | ||||
|     if (d == NULL) { | ||||
|         snprintf(err_buf, PATH_MAX + 1, "Autorun path does not exist: %s", user_settings->autorun_path); | ||||
|         snprintf(err_buf, sizeof(err_buf), "Autorun path does not exist: %s", user_settings->autorun_path); | ||||
|         api_display(err_buf); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     struct dirent *dir = NULL; | ||||
|  | ||||
|     cur_window  = window; | ||||
|  | ||||
|     self_window = self; | ||||
|  | ||||
|     while ((dir = readdir(d)) != NULL) { | ||||
|         path_len = strlen(dir->d_name); | ||||
|         size_t path_len = strlen(dir->d_name); | ||||
|  | ||||
|         if (!strcmp(dir->d_name + path_len - 3, ".py")) { | ||||
|             snprintf(abspath_buf, PATH_MAX + 1, "%s%s", user_settings->autorun_path, dir->d_name); | ||||
|             fp = fopen(abspath_buf, "r"); | ||||
|             snprintf(abspath_buf, sizeof(abspath_buf), "%s%s", user_settings->autorun_path, dir->d_name); | ||||
|             FILE *fp = fopen(abspath_buf, "r"); | ||||
|  | ||||
|             if (fp == NULL) { | ||||
|                 snprintf(err_buf, PATH_MAX + 1, "Invalid path: %s", abspath_buf); | ||||
|                 snprintf(err_buf, sizeof(err_buf), "Invalid path: %s", abspath_buf); | ||||
|                 api_display(err_buf); | ||||
|                 continue; | ||||
|             } | ||||
|   | ||||
							
								
								
									
										890
									
								
								src/audio_call.c
									
									
									
									
									
								
							
							
						
						
									
										890
									
								
								src/audio_call.c
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -27,7 +27,7 @@ | ||||
|  | ||||
| #include "audio_device.h" | ||||
|  | ||||
| typedef enum _AudioError { | ||||
| typedef enum AudioError { | ||||
|     ae_None = 0, | ||||
|     ae_StartingCaptureDevice = 1 << 0, | ||||
|     ae_StartingOutputDevice = 1 << 1, | ||||
| @@ -35,7 +35,7 @@ typedef enum _AudioError { | ||||
| } AudioError; | ||||
|  | ||||
| #ifdef VIDEO | ||||
| typedef enum _VideoError { | ||||
| typedef enum VideoError { | ||||
|     ve_None = 0, | ||||
|     ve_StartingCaptureDevice = 1 << 0, | ||||
|     ve_StartingOutputDevice = 1 << 1, | ||||
| @@ -44,14 +44,27 @@ typedef enum _VideoError { | ||||
|  | ||||
| #endif /* VIDEO */ | ||||
|  | ||||
| /* Status transitions: | ||||
|  * None -> Pending (call invitation made or received); | ||||
|  * Pending -> None (invitation rejected or failed); | ||||
|  * Pending -> Active (call starts); | ||||
|  * Active -> None (call ends). | ||||
|  */ | ||||
| typedef enum CallStatus { | ||||
|     cs_None = 0, | ||||
|     cs_Pending, | ||||
|     cs_Active | ||||
| } CallStatus; | ||||
|  | ||||
| typedef struct Call { | ||||
|     pthread_t ttid; /* Transmission thread id */ | ||||
|     bool ttas, has_output; /* Transmission thread active status (0 - stopped, 1- running) */ | ||||
|     uint32_t in_idx, out_idx; /* Audio Index */ | ||||
| #ifdef VIDEO | ||||
|     uint32_t vin_idx, vout_idx; /* Video Index */ | ||||
| #endif /* VIDEO */ | ||||
|     pthread_mutex_t mutex; | ||||
|     CallStatus status; | ||||
|     uint32_t state; /* ToxAV call state, valid when `status == cs_Active` */ | ||||
|     uint32_t in_idx, out_idx; /* Audio device index, or -1 if not open */ | ||||
|     uint32_t audio_bit_rate; /* Bit rate for sending audio */ | ||||
|  | ||||
|     uint32_t vin_idx, vout_idx; /* Video device index, or -1 if not open */ | ||||
|     uint32_t video_width, video_height; | ||||
|     uint32_t video_bit_rate; /* Bit rate for sending video; 0 for no video */ | ||||
| } Call; | ||||
|  | ||||
| struct CallControl { | ||||
| @@ -66,19 +79,17 @@ struct CallControl { | ||||
|     Call *calls; | ||||
|     uint32_t max_calls; | ||||
|  | ||||
|     uint32_t call_state; | ||||
|     bool pending_call; | ||||
|     bool audio_enabled; | ||||
|     bool video_enabled; | ||||
|  | ||||
|     uint32_t audio_bit_rate; | ||||
|     int32_t audio_frame_duration; | ||||
|     uint32_t audio_sample_rate; | ||||
|     uint8_t audio_channels; | ||||
|     uint32_t default_audio_bit_rate; | ||||
|  | ||||
|     uint32_t video_bit_rate; | ||||
|     int32_t video_frame_duration; | ||||
|  | ||||
|     uint32_t default_video_width, default_video_height; | ||||
|     uint32_t default_video_bit_rate; | ||||
| }; | ||||
|  | ||||
| extern struct CallControl CallControl; | ||||
| @@ -86,9 +97,12 @@ extern struct CallControl CallControl; | ||||
| /* You will have to pass pointer to first member of 'windows' declared in windows.c */ | ||||
| ToxAV *init_audio(ToxWindow *self, Tox *tox); | ||||
| void terminate_audio(void); | ||||
| int start_transmission(ToxWindow *self, Call *call); | ||||
| int stop_transmission(Call *call, uint32_t friend_number); | ||||
|  | ||||
| bool init_call(Call *call); | ||||
|  | ||||
| void place_call(ToxWindow *self); | ||||
| void stop_current_call(ToxWindow *self); | ||||
|  | ||||
| void init_friend_AV(uint32_t index); | ||||
| void del_friend_AV(uint32_t index); | ||||
|  | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -30,8 +30,9 @@ | ||||
| #define AUDIO_DEVICE_H | ||||
|  | ||||
| #define OPENAL_BUFS 5 | ||||
| #define MAX_OPENAL_DEVICES 32 | ||||
| #define MAX_DEVICES 32 | ||||
| #include <inttypes.h> | ||||
|  | ||||
| #include "windows.h" | ||||
|  | ||||
| typedef enum DeviceType { | ||||
| @@ -55,42 +56,42 @@ typedef enum DeviceError { | ||||
| typedef void (*DataHandleCallback)(const int16_t *, uint32_t size, void *data); | ||||
|  | ||||
|  | ||||
| #ifdef AUDIO | ||||
| DeviceError init_devices(ToxAV *av); | ||||
| #else | ||||
| DeviceError init_devices(void); | ||||
| #endif /* AUDIO */ | ||||
|  | ||||
| void get_devices_names(void); | ||||
| void get_al_device_names(void); | ||||
| DeviceError terminate_devices(void); | ||||
|  | ||||
| /* Callback handles ready data from INPUT device */ | ||||
| DeviceError register_device_callback(int32_t friend_number, uint32_t device_idx, DataHandleCallback callback, | ||||
|                                      void *data, bool enable_VAD); | ||||
| void *get_device_callback_data(uint32_t device_idx); | ||||
|  | ||||
| /* toggle device mute */ | ||||
| DeviceError device_mute(DeviceType type, uint32_t device_idx); | ||||
|  | ||||
| #ifdef AUDIO | ||||
| DeviceError device_set_VAD_treshold(uint32_t device_idx, float value); | ||||
| #endif | ||||
| bool device_is_muted(DeviceType type, uint32_t device_idx); | ||||
|  | ||||
| DeviceError device_set_VAD_threshold(uint32_t device_idx, float value); | ||||
|  | ||||
| float device_get_VAD_threshold(uint32_t device_idx); | ||||
|  | ||||
| DeviceError set_source_position(uint32_t device_idx, float x, float y, float z); | ||||
|  | ||||
| DeviceError set_al_device(DeviceType type, int32_t selection); | ||||
|  | ||||
| DeviceError set_primary_device(DeviceType type, int32_t selection); | ||||
| DeviceError open_primary_device(DeviceType type, uint32_t *device_idx, uint32_t sample_rate, uint32_t frame_duration, | ||||
|                                 uint8_t channels); | ||||
| /* Start device */ | ||||
| DeviceError open_device(DeviceType type, int32_t selection, uint32_t *device_idx, uint32_t sample_rate, | ||||
|                         uint32_t frame_duration, uint8_t channels); | ||||
| DeviceError open_input_device(uint32_t *device_idx, | ||||
|                               DataHandleCallback cb, void *cb_data, bool enable_VAD, | ||||
|                               uint32_t sample_rate, uint32_t frame_duration, uint8_t channels); | ||||
| DeviceError open_output_device(uint32_t *device_idx, | ||||
|                                uint32_t sample_rate, uint32_t frame_duration, uint8_t channels); | ||||
|  | ||||
| /* Stop device */ | ||||
| DeviceError close_device(DeviceType type, uint32_t device_idx); | ||||
|  | ||||
| /* Write data to device */ | ||||
| /* Write data to output device */ | ||||
| DeviceError write_out(uint32_t device_idx, const int16_t *data, uint32_t length, uint8_t channels, | ||||
|                       uint32_t sample_rate); | ||||
|  | ||||
| void print_devices(ToxWindow *self, DeviceType type); | ||||
| void get_primary_device_name(DeviceType type, char *buf, int size); | ||||
| /* return current input volume as float in range 0.0-100.0 */ | ||||
| float get_input_volume(void); | ||||
|  | ||||
| void print_al_devices(ToxWindow *self, DeviceType type); | ||||
|  | ||||
| DeviceError selection_valid(DeviceType type, int32_t selection); | ||||
| #endif /* AUDIO_DEVICE_H */ | ||||
|   | ||||
| @@ -20,9 +20,9 @@ | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <limits.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <limits.h> | ||||
|  | ||||
| #ifdef __APPLE__ | ||||
| #include <sys/types.h> | ||||
| @@ -31,27 +31,24 @@ | ||||
| #include <dirent.h> | ||||
| #endif /* __APPLE__ */ | ||||
|  | ||||
| #include "windows.h" | ||||
| #include "toxic.h" | ||||
| #include "misc_tools.h" | ||||
| #include "line_info.h" | ||||
| #include "execute.h" | ||||
| #include "configdir.h" | ||||
| #include "execute.h" | ||||
| #include "line_info.h" | ||||
| #include "misc_tools.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| static void print_matches(ToxWindow *self, Tox *m, const void *list, size_t n_items, size_t size) | ||||
| static void print_ac_matches(ToxWindow *self, Tox *m, char **list, size_t n_matches) | ||||
| { | ||||
|     if (m) { | ||||
|         execute(self->chatwin->history, self, m, "/clear", GLOBAL_COMMAND_MODE); | ||||
|     } | ||||
|  | ||||
|     const char *L = (char *) list; | ||||
|     int i; | ||||
|  | ||||
|     for (i = 0; i < n_items; ++i) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "%s", &L[i * size]); | ||||
|     for (size_t i = 0; i < n_matches; ++i) { | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "%s", list[i]); | ||||
|     } | ||||
|  | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "");   /* formatting */ | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, ""); | ||||
| } | ||||
|  | ||||
| /* puts match in match buffer. if more than one match, add first n chars that are identical. | ||||
| @@ -59,19 +56,19 @@ static void print_matches(ToxWindow *self, Tox *m, const void *list, size_t n_it | ||||
|  * | ||||
|  * Returns the length of the match. | ||||
|  */ | ||||
| static size_t get_str_match(ToxWindow *self, char *match, size_t match_sz, char (*matches)[MAX_STR_SIZE], int n) | ||||
| static size_t get_str_match(ToxWindow *self, char *match, size_t match_sz, const char **matches, size_t n_items, | ||||
|                             size_t max_size) | ||||
| { | ||||
|     if (n == 1) { | ||||
|     UNUSED_VAR(self); | ||||
|  | ||||
|     if (n_items == 1) { | ||||
|         return snprintf(match, match_sz, "%s", matches[0]); | ||||
|     } | ||||
|  | ||||
|     int i; | ||||
|  | ||||
|     for (i = 0; i < MAX_STR_SIZE; ++i) { | ||||
|     for (size_t i = 0; i < max_size; ++i) { | ||||
|         char ch1 = matches[0][i]; | ||||
|         int j; | ||||
|  | ||||
|         for (j = 0; j < n; ++j) { | ||||
|         for (size_t j = 0; j < n_items; ++j) { | ||||
|             char ch2 = matches[j][i]; | ||||
|  | ||||
|             if (ch1 != ch2 || !ch1) { | ||||
| @@ -90,8 +87,7 @@ static size_t get_str_match(ToxWindow *self, char *match, size_t match_sz, char | ||||
|  * then fills line with the complete word. e.g. "Hello jo" would complete the line | ||||
|  * with "Hello john". If multiple matches, prints out all the matches and semi-completes line. | ||||
|  * | ||||
|  * list is a pointer to the list of strings being compared, n_items is the number of items | ||||
|  * in the list, and size is the size of each item in the list. | ||||
|  * `list` is a pointer to `n_items` strings. Each string in the list must be <= MAX_STR_SIZE. | ||||
|  * | ||||
|  * dir_search should be true if the line being completed is a file path. | ||||
|  * | ||||
| @@ -100,7 +96,7 @@ static size_t get_str_match(ToxWindow *self, char *match, size_t match_sz, char | ||||
|  * | ||||
|  * Note: This function should not be called directly. Use complete_line() and complete_path() instead. | ||||
|  */ | ||||
| static int complete_line_helper(ToxWindow *self, const void *list, size_t n_items, size_t size, bool dir_search) | ||||
| static int complete_line_helper(ToxWindow *self, const char **list, const size_t n_items, bool dir_search) | ||||
| { | ||||
|     ChatContext *ctx = self->chatwin; | ||||
|  | ||||
| @@ -108,13 +104,12 @@ static int complete_line_helper(ToxWindow *self, const void *list, size_t n_item | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (ctx->len >= MAX_STR_SIZE || size > MAX_STR_SIZE) { | ||||
|     if (ctx->len >= MAX_STR_SIZE) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     const char *L = (const char *) list; | ||||
|     const char *endchrs = " "; | ||||
|     char ubuf[MAX_STR_SIZE] = {0}; | ||||
|     char ubuf[MAX_STR_SIZE]; | ||||
|  | ||||
|     /* work with multibyte string copy of buf for simplicity */ | ||||
|     if (wcs_to_mbs_buf(ubuf, ctx->line, sizeof(ubuf)) == -1) { | ||||
| @@ -123,10 +118,10 @@ static int complete_line_helper(ToxWindow *self, const void *list, size_t n_item | ||||
|  | ||||
|     /* isolate substring from space behind pos to pos */ | ||||
|     char tmp[MAX_STR_SIZE]; | ||||
|     snprintf(tmp, sizeof(tmp), "%s", ubuf); | ||||
|     tmp[ctx->pos] = '\0'; | ||||
|     memcpy(tmp, ubuf, ctx->pos); | ||||
|     tmp[ctx->pos] = 0; | ||||
|  | ||||
|     const char *s = strrchr(tmp, ' '); | ||||
|     const char *s = dir_search ? strchr(tmp, ' ') : strrchr(tmp, ' '); | ||||
|     char *sub = calloc(1, strlen(ubuf) + 1); | ||||
|  | ||||
|     if (sub == NULL) { | ||||
| @@ -152,38 +147,43 @@ static int complete_line_helper(ToxWindow *self, const void *list, size_t n_item | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (!sub[0]) { | ||||
|     if (!sub[0] && !(dir_search && n_items == 1)) { | ||||
|         free(sub); | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     int s_len = strlen(sub); | ||||
|     size_t n_matches = 0; | ||||
|     char matches[n_items][MAX_STR_SIZE]; | ||||
|     int i = 0; | ||||
|  | ||||
|     char **matches = (char **) malloc_ptr_array(n_items, MAX_STR_SIZE); | ||||
|  | ||||
|     if (matches == NULL) { | ||||
|         free(sub); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     /* put all list matches in matches array */ | ||||
|     for (i = 0; i < n_items; ++i) { | ||||
|         char str[MAX_CMDNAME_SIZE + 1]; | ||||
|         snprintf(str, sizeof(str), "%s", &L[i * size]); | ||||
|  | ||||
|         if (strncasecmp(str, sub, s_len) == 0) { | ||||
|             strcpy(matches[n_matches++], str); | ||||
|     for (size_t i = 0; i < n_items; ++i) { | ||||
|         if (strncasecmp(list[i], sub, s_len) == 0) { | ||||
|             snprintf(matches[n_matches++], MAX_STR_SIZE, "%s", list[i]); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     free(sub); | ||||
|  | ||||
|     if (!n_matches) { | ||||
|         free_ptr_array((void **) matches); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (!dir_search && n_matches > 1) { | ||||
|         print_matches(self, NULL, matches, n_matches, MAX_STR_SIZE); | ||||
|         print_ac_matches(self, NULL, matches, n_matches); | ||||
|     } | ||||
|  | ||||
|     char match[MAX_STR_SIZE]; | ||||
|     size_t match_len = get_str_match(self, match, sizeof(match), matches, n_matches); | ||||
|     size_t match_len = get_str_match(self, match, sizeof(match), (const char **) matches, n_matches, MAX_STR_SIZE); | ||||
|  | ||||
|     free_ptr_array((void **) matches); | ||||
|  | ||||
|     if (match_len == 0) { | ||||
|         return 0; | ||||
| @@ -219,11 +219,21 @@ static int complete_line_helper(ToxWindow *self, const void *list, size_t n_item | ||||
|  | ||||
|     /* If path points to a file with no extension don't append a forward slash */ | ||||
|     if (dir_search && *endchrs == '/') { | ||||
|         const char *path_start = strchr(ubuf+1, '/'); | ||||
|         const char *path_start = strchr(ubuf + 1, '/'); | ||||
|  | ||||
|         if (!path_start) {  // should never happen | ||||
|         if (!path_start) { | ||||
|             path_start = strchr(ubuf + 1, ' '); | ||||
|  | ||||
|             if (!path_start) { | ||||
|                 return -1; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (strlen(path_start) < 2) { | ||||
|             return -1; | ||||
|         } | ||||
|  | ||||
|         ++path_start; | ||||
|  | ||||
|         if (file_type(path_start) == FILE_TYPE_REGULAR) { | ||||
|             endchrs = ""; | ||||
| @@ -249,14 +259,14 @@ static int complete_line_helper(ToxWindow *self, const void *list, size_t n_item | ||||
|     return diff; | ||||
| } | ||||
|  | ||||
| int complete_line(ToxWindow *self, const void *list, size_t n_items, size_t size) | ||||
| int complete_line(ToxWindow *self, const char **list, size_t n_items) | ||||
| { | ||||
|     return complete_line_helper(self, list, n_items, size, false); | ||||
|     return complete_line_helper(self, list, n_items, false); | ||||
| } | ||||
|  | ||||
| static int complete_path(ToxWindow *self, const void *list, size_t n_items, size_t size) | ||||
| static int complete_path(ToxWindow *self, const char **list, const size_t n_items) | ||||
| { | ||||
|     return complete_line_helper(self, list, n_items, size, true); | ||||
|     return complete_line_helper(self, list, n_items, true); | ||||
| } | ||||
|  | ||||
| /* Transforms a tab complete starting with the shorthand "~" into the full home directory. */ | ||||
| @@ -267,9 +277,9 @@ static void complete_home_dir(ToxWindow *self, char *path, int pathsize, const c | ||||
|     char homedir[MAX_STR_SIZE] = {0}; | ||||
|     get_home_dir(homedir, sizeof(homedir)); | ||||
|  | ||||
|     char newline[MAX_STR_SIZE]; | ||||
|     char newline[MAX_STR_SIZE + 1]; | ||||
|     snprintf(newline, sizeof(newline), "%s %s%s", cmd, homedir, path + 1); | ||||
|     snprintf(path, pathsize, "%s", &newline[cmdlen-1]); | ||||
|     snprintf(path, pathsize, "%s", &newline[cmdlen - 1]); | ||||
|  | ||||
|     wchar_t wline[MAX_STR_SIZE]; | ||||
|  | ||||
| @@ -288,21 +298,33 @@ static void complete_home_dir(ToxWindow *self, char *path, int pathsize, const c | ||||
|     ctx->len = ctx->pos; | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Return true if the first `p_len` chars in `s` are equal to `p` and `s` is a valid directory name. | ||||
|  */ | ||||
| static bool is_partial_match(const char *s, const char *p, size_t p_len) | ||||
| { | ||||
|     if (s == NULL || p == NULL) { | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     return strncmp(s, p, p_len) == 0 && strcmp(".", s) != 0 && strcmp("..", s) != 0; | ||||
| } | ||||
|  | ||||
| /* Attempts to match /command "<incomplete-dir>" line to matching directories. | ||||
|  * If there is only one match the line is auto-completed. | ||||
|  * | ||||
|  * Returns the diff between old len and new len of ctx->line on success. | ||||
|  * Returns -1 if no matches or more than one match. | ||||
|  */ | ||||
| #define MAX_DIRS 512 | ||||
| #define MAX_DIRS 75 | ||||
| int dir_match(ToxWindow *self, Tox *m, const wchar_t *line, const wchar_t *cmd) | ||||
| { | ||||
|     char b_path[MAX_STR_SIZE]; | ||||
|     char b_name[MAX_STR_SIZE]; | ||||
|     char b_path[MAX_STR_SIZE + 1]; | ||||
|     char b_name[MAX_STR_SIZE + 1]; | ||||
|     char b_cmd[MAX_STR_SIZE]; | ||||
|     const wchar_t *tmpline = &line[wcslen(cmd) + 1];   /* start after "/command " */ | ||||
|  | ||||
|     if (wcs_to_mbs_buf(b_path, tmpline, sizeof(b_path)) == -1) { | ||||
|     if (wcs_to_mbs_buf(b_path, tmpline, sizeof(b_path) - 1) == -1) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
| @@ -311,7 +333,7 @@ int dir_match(ToxWindow *self, Tox *m, const wchar_t *line, const wchar_t *cmd) | ||||
|     } | ||||
|  | ||||
|     if (b_path[0] == '~') { | ||||
|         complete_home_dir(self, b_path, sizeof(b_path), b_cmd, strlen(b_cmd) + 2); | ||||
|         complete_home_dir(self, b_path, sizeof(b_path) - 1, b_cmd, strlen(b_cmd) + 2); | ||||
|     } | ||||
|  | ||||
|     int si = char_rfind(b_path, '/', strlen(b_path)); | ||||
| @@ -320,28 +342,33 @@ int dir_match(ToxWindow *self, Tox *m, const wchar_t *line, const wchar_t *cmd) | ||||
|         b_path[0] = '.'; | ||||
|         b_path[1] = '\0'; | ||||
|     } else if (!si && b_path[0] != '/') {    /* look for matches in pwd */ | ||||
|         char tmp[MAX_STR_SIZE]; | ||||
|         snprintf(tmp, sizeof(tmp), ".%s", b_path); | ||||
|         snprintf(b_path, sizeof(b_path), "%s", tmp); | ||||
|         memmove(b_path + 1, b_path, sizeof(b_path) - 1); | ||||
|         b_path[0] = '.'; | ||||
|     } | ||||
|  | ||||
|     snprintf(b_name, sizeof(b_name), "%s", &b_path[si + 1]); | ||||
|     b_path[si + 1] = '\0'; | ||||
|     int b_name_len = strlen(b_name); | ||||
|     size_t b_name_len = strlen(b_name); | ||||
|     DIR *dp = opendir(b_path); | ||||
|  | ||||
|     if (dp == NULL) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     char dirnames[MAX_DIRS][NAME_MAX + 1]; | ||||
|     char **dirnames = (char **) malloc_ptr_array(MAX_DIRS, NAME_MAX + 1); | ||||
|  | ||||
|     if (dirnames == NULL) { | ||||
|         closedir(dp); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     struct dirent *entry; | ||||
|  | ||||
|     int dircount = 0; | ||||
|  | ||||
|     while ((entry = readdir(dp)) && dircount < MAX_DIRS) { | ||||
|         if (strncmp(entry->d_name, b_name, b_name_len) == 0 | ||||
|                 && strcmp(".", entry->d_name) && strcmp("..", entry->d_name)) { | ||||
|             snprintf(dirnames[dircount], sizeof(dirnames[dircount]), "%s", entry->d_name); | ||||
|         if (is_partial_match(entry->d_name, b_name, b_name_len)) { | ||||
|             snprintf(dirnames[dircount], NAME_MAX + 1, "%s", entry->d_name); | ||||
|             ++dircount; | ||||
|         } | ||||
|     } | ||||
| @@ -349,13 +376,18 @@ int dir_match(ToxWindow *self, Tox *m, const wchar_t *line, const wchar_t *cmd) | ||||
|     closedir(dp); | ||||
|  | ||||
|     if (dircount == 0) { | ||||
|         free_ptr_array((void **) dirnames); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (dircount > 1) { | ||||
|         qsort(dirnames, dircount, NAME_MAX + 1, qsort_strcasecmp_hlpr); | ||||
|         print_matches(self, m, dirnames, dircount, NAME_MAX + 1); | ||||
|         qsort(dirnames, dircount, sizeof(char *), qsort_ptr_char_array_helper); | ||||
|         print_ac_matches(self, m, dirnames, dircount); | ||||
|     } | ||||
|  | ||||
|     return complete_path(self, dirnames, dircount, NAME_MAX + 1); | ||||
|     int ret = complete_path(self, (const char **) dirnames, dircount); | ||||
|  | ||||
|     free_ptr_array((void **) dirnames); | ||||
|  | ||||
|     return ret; | ||||
| } | ||||
|   | ||||
| @@ -23,18 +23,23 @@ | ||||
| #ifndef AUTOCOMPLETE_H | ||||
| #define AUTOCOMPLETE_H | ||||
|  | ||||
| #include "windows.h" | ||||
|  | ||||
| /* | ||||
|  * Looks for all instances in list that begin with the last entered word in line according to pos, | ||||
|  * then fills line with the complete word. e.g. "Hello jo" would complete the line | ||||
|  * with "Hello john". If multiple matches, prints out all the matches and semi-completes line. | ||||
|  * | ||||
|  * list is a pointer to the list of strings being compared, n_items is the number of items | ||||
|  * in the list, and size is the size of each item in the list. | ||||
| * `list` is a pointer to `n_items` strings. | ||||
|  * | ||||
|  * dir_search should be true if the line being completed is a file path. | ||||
|  * | ||||
|  * Returns the difference between the old len and new len of line on success. | ||||
|  * Returns -1 on error. | ||||
|  * | ||||
|  * Note: This function should not be called directly. Use complete_line() and complete_path() instead. | ||||
|  */ | ||||
| int complete_line(ToxWindow *self, const void *list, size_t n_items, size_t size); | ||||
| int complete_line(ToxWindow *self, const char **list, size_t n_items); | ||||
|  | ||||
| /* Attempts to match /command "<incomplete-dir>" line to matching directories. | ||||
|  * If there is only one match the line is auto-completed. | ||||
|   | ||||
| @@ -20,14 +20,14 @@ | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <stdlib.h> | ||||
| #include <stdio.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
|  | ||||
| #include "misc_tools.h" | ||||
| #include "avatars.h" | ||||
| #include "file_transfers.h" | ||||
| #include "friendlist.h" | ||||
| #include "avatars.h" | ||||
| #include "misc_tools.h" | ||||
|  | ||||
| extern FriendsList Friends; | ||||
|  | ||||
| @@ -39,21 +39,54 @@ static struct Avatar { | ||||
|     off_t size; | ||||
| } Avatar; | ||||
|  | ||||
| /* Compares the first size bytes of fp to signature. | ||||
|  * | ||||
|  * Returns 0 if they are the same | ||||
|  * Returns 1 if they differ | ||||
|  * Returns -1 on error. | ||||
|  * | ||||
|  * On success this function will seek back to the beginning of fp. | ||||
|  */ | ||||
| static int check_file_signature(const unsigned char *signature, size_t size, FILE *fp) | ||||
| { | ||||
|     char *buf = malloc(size); | ||||
|  | ||||
|     if (buf == NULL) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (fread(buf, size, 1, fp) != 1) { | ||||
|         free(buf); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     int ret = memcmp(signature, buf, size); | ||||
|  | ||||
|     free(buf); | ||||
|  | ||||
|     if (fseek(fp, 0L, SEEK_SET) == -1) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     return ret == 0 ? 0 : 1; | ||||
| } | ||||
|  | ||||
| static void avatar_clear(void) | ||||
| { | ||||
|     memset(&Avatar, 0, sizeof(struct Avatar)); | ||||
|     Avatar = (struct Avatar) { | ||||
|         0 | ||||
|     }; | ||||
| } | ||||
|  | ||||
| /* Sends avatar to friendnum. | ||||
| /* Sends avatar to friendnumber. | ||||
|  * | ||||
|  * Returns 0 on success. | ||||
|  * Returns -1 on failure. | ||||
|  */ | ||||
| int avatar_send(Tox *m, uint32_t friendnum) | ||||
| int avatar_send(Tox *m, uint32_t friendnumber) | ||||
| { | ||||
|     Tox_Err_File_Send err; | ||||
|     uint32_t filenum = tox_file_send(m, friendnum, TOX_FILE_KIND_AVATAR, (size_t) Avatar.size, | ||||
|     uint32_t filenumber = tox_file_send(m, friendnumber, TOX_FILE_KIND_AVATAR, (size_t) Avatar.size, | ||||
|                                         NULL, (uint8_t *) Avatar.name, Avatar.name_len, &err); | ||||
|  | ||||
|     if (Avatar.size == 0) { | ||||
| @@ -61,11 +94,11 @@ int avatar_send(Tox *m, uint32_t friendnum) | ||||
|     } | ||||
|  | ||||
|     if (err != TOX_ERR_FILE_SEND_OK) { | ||||
|         fprintf(stderr, "tox_file_send failed for friendnumber %d (error %d)\n", friendnum, err); | ||||
|         fprintf(stderr, "tox_file_send failed for friendnumber %u (error %d)\n", friendnumber, err); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     struct FileTransfer *ft = new_file_transfer(NULL, friendnum, filenum, FILE_TRANSFER_SEND, TOX_FILE_KIND_AVATAR); | ||||
|     struct FileTransfer *ft = new_file_transfer(NULL, friendnumber, filenumber, FILE_TRANSFER_SEND, TOX_FILE_KIND_AVATAR); | ||||
|  | ||||
|     if (!ft) { | ||||
|         return -1; | ||||
| @@ -86,9 +119,7 @@ int avatar_send(Tox *m, uint32_t friendnum) | ||||
| /* Sends avatar to all friends */ | ||||
| static void avatar_send_all(Tox *m) | ||||
| { | ||||
|     size_t i; | ||||
|  | ||||
|     for (i = 0; i < Friends.max_idx; ++i) { | ||||
|     for (size_t i = 0; i < Friends.max_idx; ++i) { | ||||
|         if (Friends.list[i].connection_status != TOX_CONNECTION_NONE) { | ||||
|             avatar_send(m, Friends.list[i].num); | ||||
|         } | ||||
| @@ -112,7 +143,7 @@ int avatar_set(Tox *m, const char *path, size_t path_len) | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     char PNG_signature[8] = {0x89, 0x50, 0x4E, 0x47, 0x0D, 0x0A, 0x1A, 0x0A}; | ||||
|     unsigned char PNG_signature[8] = {0x89, 0x50, 0x4E, 0x47, 0x0D, 0x0A, 0x1A, 0x0A}; | ||||
|  | ||||
|     if (check_file_signature(PNG_signature, sizeof(PNG_signature), fp) != 0) { | ||||
|         fclose(fp); | ||||
| @@ -149,6 +180,13 @@ void avatar_unset(Tox *m) | ||||
|     avatar_send_all(m); | ||||
| } | ||||
|  | ||||
| void on_avatar_friend_connection_status(Tox *m, uint32_t friendnumber, Tox_Connection connection_status) | ||||
| { | ||||
|     if (connection_status == TOX_CONNECTION_NONE) { | ||||
|         kill_avatar_file_transfers_friend(m, friendnumber); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void on_avatar_file_control(Tox *m, struct FileTransfer *ft, Tox_File_Control control) | ||||
| { | ||||
|     switch (control) { | ||||
| @@ -196,21 +234,29 @@ void on_avatar_chunk_request(Tox *m, struct FileTransfer *ft, uint64_t position, | ||||
|         ft->position = position; | ||||
|     } | ||||
|  | ||||
|     uint8_t send_data[length]; | ||||
|     size_t send_length = fread(send_data, 1, sizeof(send_data), ft->file); | ||||
|     uint8_t *send_data = malloc(length); | ||||
|  | ||||
|     if (send_length != length) { | ||||
|     if (send_data == NULL) { | ||||
|         close_file_transfer(NULL, m, ft, TOX_FILE_CONTROL_CANCEL, NULL, silent); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     size_t send_length = fread(send_data, 1, length, ft->file); | ||||
|  | ||||
|     if (send_length != length) { | ||||
|         close_file_transfer(NULL, m, ft, TOX_FILE_CONTROL_CANCEL, NULL, silent); | ||||
|         free(send_data); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     Tox_Err_File_Send_Chunk err; | ||||
|     tox_file_send_chunk(m, ft->friendnum, ft->filenum, position, send_data, send_length, &err); | ||||
|     tox_file_send_chunk(m, ft->friendnumber, ft->filenumber, position, send_data, send_length, &err); | ||||
|  | ||||
|     free(send_data); | ||||
|  | ||||
|     if (err != TOX_ERR_FILE_SEND_CHUNK_OK) { | ||||
|         fprintf(stderr, "tox_file_send_chunk failed in avatar callback (error %d)\n", err); | ||||
|     } | ||||
|  | ||||
|     ft->position += send_length; | ||||
|     ft->last_keep_alive = get_unix_time(); | ||||
| } | ||||
|   | ||||
| @@ -23,6 +23,8 @@ | ||||
| #ifndef AVATARS_H | ||||
| #define AVATARS_H | ||||
|  | ||||
| #include "file_transfers.h" | ||||
|  | ||||
| #define MAX_AVATAR_FILE_SIZE 65536 | ||||
|  | ||||
| /* Sends avatar to friendnum. | ||||
| @@ -48,5 +50,6 @@ void avatar_unset(Tox *m); | ||||
|  | ||||
| void on_avatar_chunk_request(Tox *m, struct FileTransfer *ft, uint64_t position, size_t length); | ||||
| void on_avatar_file_control(Tox *m, struct FileTransfer *ft, Tox_File_Control control); | ||||
| void on_avatar_friend_connection_status(Tox *m, uint32_t friendnumber, Tox_Connection connection_status); | ||||
|  | ||||
| #endif /* AVATARS_H */ | ||||
|   | ||||
| @@ -20,21 +20,24 @@ | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <arpa/inet.h> | ||||
| #include <limits.h> | ||||
| #include <netinet/in.h> | ||||
| #include <stdbool.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <stdbool.h> | ||||
| #include <limits.h> | ||||
| #include <sys/socket.h> | ||||
|  | ||||
| #include <curl/curl.h> | ||||
| #include <tox/tox.h> | ||||
|  | ||||
| #include "line_info.h" | ||||
| #include "windows.h" | ||||
| #include "misc_tools.h" | ||||
| #include "configdir.h" | ||||
| #include "curl_util.h" | ||||
| #include "settings.h" | ||||
| #include "line_info.h" | ||||
| #include "misc_tools.h" | ||||
| #include "prompt.h" | ||||
| #include "settings.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| extern struct arg_opts arg_opts; | ||||
| extern struct user_settings *user_settings; | ||||
| @@ -105,6 +108,37 @@ static struct DHT_Nodes { | ||||
|     time_t last_updated; | ||||
| } Nodes; | ||||
|  | ||||
| /* Return true if address appears to be a valid ipv4 address. */ | ||||
| static bool is_ip4_address(const char *address) | ||||
| { | ||||
|     struct sockaddr_in s_addr; | ||||
|     return inet_pton(AF_INET, address, &(s_addr.sin_addr)) != 0; | ||||
| } | ||||
|  | ||||
| /* Return true if address roughly appears to be a valid ipv6 address. | ||||
|  * | ||||
|  * TODO: Improve this function (inet_pton behaves strangely with ipv6). | ||||
|  * for now the only guarantee is that it won't return true if the | ||||
|  * address is a domain or ipv4 address, and should only be used if you're | ||||
|  * reasonably sure that the address is one of the three (ipv4, ipv6 or a domain). | ||||
|  */ | ||||
| static bool is_ip6_address(const char *address) | ||||
| { | ||||
|     size_t num_colons = 0; | ||||
|     char ch = 0; | ||||
|  | ||||
|     for (size_t i = 0; (ch = address[i]); ++i) { | ||||
|         if (ch == '.') { | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         if (ch == ':') { | ||||
|             ++num_colons; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return num_colons > 1 && num_colons < 8; | ||||
| } | ||||
|  | ||||
| /* Determine if a node is offline by comparing the age of the nodeslist | ||||
|  * to the last time the node was successfully pinged. | ||||
| @@ -241,6 +275,7 @@ on_exit: | ||||
|  * Return -2 if http lookup failed. | ||||
|  * Return -3 if http reponse was empty. | ||||
|  * Return -4 if data could not be written to disk. | ||||
|  * Return -5 if memory allocation fails. | ||||
|  */ | ||||
| static int update_DHT_nodeslist(const char *nodes_path) | ||||
| { | ||||
| @@ -254,26 +289,34 @@ static int update_DHT_nodeslist(const char *nodes_path) | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     struct Recv_Curl_Data recv_data; | ||||
|     struct Recv_Curl_Data *recv_data = calloc(1, sizeof(struct Recv_Curl_Data)); | ||||
|  | ||||
|     memset(&recv_data, 0, sizeof(struct Recv_Curl_Data)); | ||||
|     if (recv_data == NULL) { | ||||
|         fclose(fp); | ||||
|         return -5; | ||||
|     } | ||||
|  | ||||
|     if (curl_fetch_nodes_JSON(&recv_data) == -1) { | ||||
|     if (curl_fetch_nodes_JSON(recv_data) == -1) { | ||||
|         free(recv_data); | ||||
|         fclose(fp); | ||||
|         return -2; | ||||
|     } | ||||
|  | ||||
|     if (recv_data.length == 0) { | ||||
|     if (recv_data->length == 0) { | ||||
|         free(recv_data); | ||||
|         fclose(fp); | ||||
|         return -3; | ||||
|     } | ||||
|  | ||||
|     if (fwrite(recv_data.data, recv_data.length, 1, fp) != 1) { | ||||
|     if (fwrite(recv_data->data, recv_data->length, 1, fp) != 1) { | ||||
|         free(recv_data); | ||||
|         fclose(fp); | ||||
|         return -4; | ||||
|     } | ||||
|  | ||||
|     free(recv_data); | ||||
|     fclose(fp); | ||||
|  | ||||
|     return 1; | ||||
| } | ||||
|  | ||||
| @@ -430,6 +473,8 @@ static int extract_node(const char *line, struct Node *node) | ||||
| /* Loads the DHT nodeslist to memory from json encoded nodes file. */ | ||||
| void *load_nodeslist_thread(void *data) | ||||
| { | ||||
|     UNUSED_VAR(data); | ||||
|  | ||||
|     char nodes_path[PATH_MAX]; | ||||
|     get_nodeslist_path(nodes_path, sizeof(nodes_path)); | ||||
|  | ||||
|   | ||||
							
								
								
									
										672
									
								
								src/chat.c
									
									
									
									
									
								
							
							
						
						
									
										672
									
								
								src/chat.c
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -23,8 +23,8 @@ | ||||
| #ifndef CHAT_H | ||||
| #define CHAT_H | ||||
|  | ||||
| #include "windows.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| /* set CTRL to -1 if we don't want to send a control signal. | ||||
|    set msg to NULL if we don't want to display a message */ | ||||
|   | ||||
| @@ -23,23 +23,25 @@ | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
|  | ||||
| #include "chat.h" | ||||
| #include "conference.h" | ||||
| #include "execute.h" | ||||
| #include "file_transfers.h" | ||||
| #include "friendlist.h" | ||||
| #include "line_info.h" | ||||
| #include "misc_tools.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "misc_tools.h" | ||||
| #include "friendlist.h" | ||||
| #include "execute.h" | ||||
| #include "line_info.h" | ||||
| #include "groupchat.h" | ||||
| #include "chat.h" | ||||
| #include "file_transfers.h" | ||||
|  | ||||
| extern ToxWindow *prompt; | ||||
| extern FriendsList Friends; | ||||
|  | ||||
| void cmd_cancelfile(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|  | ||||
|     if (argc < 2) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Requires type in|out and the file ID."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Requires type in|out and the file ID."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -48,7 +50,7 @@ void cmd_cancelfile(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*ar | ||||
|     long int idx = strtol(argv[2], NULL, 10); | ||||
|  | ||||
|     if ((idx == 0 && strcmp(argv[2], "0")) || idx >= MAX_FILES || idx < 0) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invalid file ID."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid file ID."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -60,17 +62,17 @@ void cmd_cancelfile(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*ar | ||||
|     } else if (strcasecmp(inoutstr, "out") == 0) { | ||||
|         ft = get_file_transfer_struct_index(self->num, idx, FILE_TRANSFER_SEND); | ||||
|     } else { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Type must be 'in' or 'out'."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Type must be 'in' or 'out'."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (!ft) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invalid file ID."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid file ID."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (ft->state == FILE_TRANSFER_INACTIVE) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invalid file ID."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid file ID."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -78,91 +80,123 @@ void cmd_cancelfile(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*ar | ||||
|     close_file_transfer(self, m, ft, TOX_FILE_CONTROL_CANCEL, msg, silent); | ||||
| } | ||||
|  | ||||
| void cmd_groupinvite(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| void cmd_conference_invite(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|  | ||||
|     if (argc < 1) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Group number required."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Conference number required."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     long int groupnum = strtol(argv[1], NULL, 10); | ||||
|     long int conferencenum = strtol(argv[1], NULL, 10); | ||||
|  | ||||
|     if ((groupnum == 0 && strcmp(argv[1], "0")) || groupnum < 0 || groupnum == LONG_MAX) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invalid group number."); | ||||
|     if ((conferencenum == 0 && strcmp(argv[1], "0")) || conferencenum < 0 || conferencenum == LONG_MAX) { | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid conference number."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     Tox_Err_Conference_Invite err; | ||||
|  | ||||
|     if (!tox_conference_invite(m, self->num, groupnum, &err)) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to invite contact to group (error %d)", err); | ||||
|     if (!tox_conference_invite(m, self->num, conferencenum, &err)) { | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to invite contact to conference (error %d)", err); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invited contact to Group %d.", groupnum); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invited contact to Conference %ld.", conferencenum); | ||||
| } | ||||
|  | ||||
| void cmd_join_group(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| void cmd_conference_join(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|     UNUSED_VAR(argc); | ||||
|     UNUSED_VAR(argv); | ||||
|  | ||||
|     if (get_num_active_windows() >= MAX_WINDOWS_NUM) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, RED, " * Warning: Too many windows are open."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, RED, " * Warning: Too many windows are open."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     const char *groupkey = Friends.list[self->num].group_invite.key; | ||||
|     uint16_t length = Friends.list[self->num].group_invite.length; | ||||
|     uint8_t type = Friends.list[self->num].group_invite.type; | ||||
|     const char *conferencekey = Friends.list[self->num].conference_invite.key; | ||||
|     uint16_t length = Friends.list[self->num].conference_invite.length; | ||||
|     uint8_t type = Friends.list[self->num].conference_invite.type; | ||||
|  | ||||
|     if (!Friends.list[self->num].group_invite.pending) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "No pending group chat invite."); | ||||
|     if (!Friends.list[self->num].conference_invite.pending) { | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "No pending conference invite."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (type != TOX_CONFERENCE_TYPE_TEXT) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Toxic does not support audio groups."); | ||||
|         return; | ||||
|     } | ||||
|     uint32_t conferencenum; | ||||
|  | ||||
|     if (type == TOX_CONFERENCE_TYPE_TEXT) { | ||||
|         Tox_Err_Conference_Join err; | ||||
|  | ||||
|     uint32_t groupnum = tox_conference_join(m, self->num, (const uint8_t *) groupkey, length, &err); | ||||
|         conferencenum = tox_conference_join(m, self->num, (const uint8_t *) conferencekey, length, &err); | ||||
|  | ||||
|         if (err != TOX_ERR_CONFERENCE_JOIN_OK) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Group chat instance failed to initialize (error %d)", err); | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Conference instance failed to initialize (error %d)", err); | ||||
|             return; | ||||
|         } | ||||
|     } else if (type == TOX_CONFERENCE_TYPE_AV) { | ||||
| #ifdef AUDIO | ||||
|         conferencenum = toxav_join_av_groupchat(m, self->num, (const uint8_t *) conferencekey, length, | ||||
|                                                 audio_conference_callback, NULL); | ||||
|  | ||||
|         if (conferencenum == (uint32_t) -1) { | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Audio conference instance failed to initialize"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|     if (init_groupchat_win(prompt, m, groupnum, type) == -1) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Group chat window failed to initialize."); | ||||
|         tox_conference_delete(m, groupnum, NULL); | ||||
| #else | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Audio support disabled by compile-time option."); | ||||
|         return; | ||||
| #endif | ||||
|     } else { | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Unknown conference type %d", type); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (init_conference_win(m, conferencenum, type, NULL, 0) == -1) { | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Conference window failed to initialize."); | ||||
|         tox_conference_delete(m, conferencenum, NULL); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| #ifdef AUDIO | ||||
|  | ||||
|     if (type == TOX_CONFERENCE_TYPE_AV) { | ||||
|         if (!init_conference_audio_input(m, conferencenum)) { | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Audio capture failed; use \"/audio on\" to try again."); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| #endif | ||||
| } | ||||
|  | ||||
| void cmd_savefile(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|  | ||||
|     if (argc < 1) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "File ID required."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "File ID required."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     long int idx = strtol(argv[1], NULL, 10); | ||||
|  | ||||
|     if ((idx == 0 && strcmp(argv[1], "0")) || idx < 0 || idx >= MAX_FILES) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "No pending file transfers with that ID."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "No pending file transfers with that ID."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     struct FileTransfer *ft = get_file_transfer_struct_index(self->num, idx, FILE_TRANSFER_RECV); | ||||
|  | ||||
|     if (!ft) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "No pending file transfers with that ID."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "No pending file transfers with that ID."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (ft->state != FILE_TRANSFER_PENDING) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "No pending file transfers with that ID."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "No pending file transfers with that ID."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -173,18 +207,18 @@ void cmd_savefile(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv | ||||
|     } | ||||
|  | ||||
|     Tox_Err_File_Control err; | ||||
|     tox_file_control(m, self->num, ft->filenum, TOX_FILE_CONTROL_RESUME, &err); | ||||
|     tox_file_control(m, self->num, ft->filenumber, TOX_FILE_CONTROL_RESUME, &err); | ||||
|  | ||||
|     if (err != TOX_ERR_FILE_CONTROL_OK) { | ||||
|         goto on_recv_error; | ||||
|     } | ||||
|  | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Saving file [%d] as: '%s'", idx, ft->file_path); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Saving file [%ld] as: '%s'", idx, ft->file_path); | ||||
|  | ||||
|     /* prep progress bar line */ | ||||
|     char progline[MAX_STR_SIZE]; | ||||
|     init_progress_bar(progline); | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "%s", progline); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "%s", progline); | ||||
|  | ||||
|     ft->line_id = self->chatwin->hst->line_end->id + 2; | ||||
|     ft->state = FILE_TRANSFER_STARTED; | ||||
| @@ -195,33 +229,35 @@ on_recv_error: | ||||
|  | ||||
|     switch (err) { | ||||
|         case TOX_ERR_FILE_CONTROL_FRIEND_NOT_FOUND: | ||||
|             line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "File transfer failed: Friend not found."); | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "File transfer failed: Friend not found."); | ||||
|             return; | ||||
|  | ||||
|         case TOX_ERR_FILE_CONTROL_FRIEND_NOT_CONNECTED: | ||||
|             line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "File transfer failed: Friend is not online."); | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "File transfer failed: Friend is not online."); | ||||
|             return; | ||||
|  | ||||
|         case TOX_ERR_FILE_CONTROL_NOT_FOUND: | ||||
|             line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "File transfer failed: Invalid filenumber."); | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "File transfer failed: Invalid filenumber."); | ||||
|             return; | ||||
|  | ||||
|         case TOX_ERR_FILE_CONTROL_SENDQ: | ||||
|             line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "File transfer failed: Connection error."); | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "File transfer failed: Connection error."); | ||||
|             return; | ||||
|  | ||||
|         default: | ||||
|             line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "File transfer failed (error %d)\n", err); | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "File transfer failed (error %d)\n", err); | ||||
|             return; | ||||
|     } | ||||
| } | ||||
|  | ||||
| void cmd_sendfile(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|  | ||||
|     const char *errmsg = NULL; | ||||
|  | ||||
|     if (argc < 1) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "File path required."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "File path required."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -230,21 +266,21 @@ void cmd_sendfile(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv | ||||
|     int path_len = strlen(path); | ||||
|  | ||||
|     if (path_len >= MAX_STR_SIZE) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "File path exceeds character limit."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "File path exceeds character limit."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     FILE *file_to_send = fopen(path, "r"); | ||||
|  | ||||
|     if (file_to_send == NULL) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "File not found."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "File not found."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     off_t filesize = file_size(path); | ||||
|  | ||||
|     if (filesize == 0) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invalid file."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid file."); | ||||
|         fclose(file_to_send); | ||||
|         return; | ||||
|     } | ||||
| @@ -274,7 +310,7 @@ void cmd_sendfile(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv | ||||
|  | ||||
|     char sizestr[32]; | ||||
|     bytes_convert_str(sizestr, sizeof(sizestr), filesize); | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Sending file [%d]: '%s' (%s)", filenum, file_name, sizestr); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Sending file [%d]: '%s' (%s)", filenum, file_name, sizestr); | ||||
|  | ||||
|     return; | ||||
|  | ||||
| @@ -302,7 +338,7 @@ on_send_error: | ||||
|             break; | ||||
|     } | ||||
|  | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "%s", errmsg); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "%s", errmsg); | ||||
|     tox_file_control(m, self->num, filenum, TOX_FILE_CONTROL_CANCEL, NULL); | ||||
|     fclose(file_to_send); | ||||
| } | ||||
|   | ||||
| @@ -23,12 +23,12 @@ | ||||
| #ifndef CHAT_COMMANDS_H | ||||
| #define CHAT_COMMANDS_H | ||||
|  | ||||
| #include "windows.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| void cmd_cancelfile(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_groupinvite(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_join_group(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_conference_invite(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_conference_join(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_savefile(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_sendfile(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
|  | ||||
| @@ -45,8 +45,9 @@ void cmd_bitrate(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SI | ||||
| #endif /* AUDIO */ | ||||
|  | ||||
| #ifdef VIDEO | ||||
| void cmd_vcall(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_video(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_ccur_video_device(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_res(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| #endif /* VIDEO */ | ||||
|  | ||||
| #endif /* CHAT_COMMANDS_H */ | ||||
|   | ||||
							
								
								
									
										1418
									
								
								src/conference.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1418
									
								
								src/conference.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										118
									
								
								src/conference.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										118
									
								
								src/conference.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,118 @@ | ||||
| /*  conference.h | ||||
|  * | ||||
|  * | ||||
|  *  Copyright (C) 2014 Toxic All Rights Reserved. | ||||
|  * | ||||
|  *  This file is part of Toxic. | ||||
|  * | ||||
|  *  Toxic is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|  *  the Free Software Foundation, either version 3 of the License, or | ||||
|  *  (at your option) any later version. | ||||
|  * | ||||
|  *  Toxic is distributed in the hope that it will be useful, | ||||
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  *  GNU General Public License for more details. | ||||
|  * | ||||
|  *  You should have received a copy of the GNU General Public License | ||||
|  *  along with Toxic.  If not, see <http://www.gnu.org/licenses/>. | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #ifndef CONFERENCE_H | ||||
| #define CONFERENCE_H | ||||
|  | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| #define CONFERENCE_MAX_TITLE_LENGTH TOX_MAX_NAME_LENGTH | ||||
| #define SIDEBAR_WIDTH 16 | ||||
|  | ||||
| typedef struct ConferencePeer { | ||||
|     bool       active; | ||||
|  | ||||
|     uint8_t    pubkey[TOX_PUBLIC_KEY_SIZE]; | ||||
|     uint32_t   peernum;    /* index in chat->peer_list */ | ||||
|  | ||||
|     char       name[TOX_MAX_NAME_LENGTH]; | ||||
|     size_t     name_length; | ||||
|  | ||||
|     bool       sending_audio; | ||||
|     uint32_t   audio_out_idx; | ||||
|     time_t     last_audio_time; | ||||
| } ConferencePeer; | ||||
|  | ||||
| typedef struct AudioInputCallbackData { | ||||
|     Tox *tox; | ||||
|     uint32_t conferencenum; | ||||
| } AudioInputCallbackData; | ||||
|  | ||||
| #define PUBKEY_STRING_SIZE (2 * TOX_PUBLIC_KEY_SIZE + 1) | ||||
| typedef struct NameListEntry { | ||||
|     char name[TOX_MAX_NAME_LENGTH]; | ||||
|     char pubkey_str[PUBKEY_STRING_SIZE]; | ||||
|     uint32_t peernum; | ||||
| } NameListEntry; | ||||
|  | ||||
|  | ||||
| typedef struct { | ||||
|     int chatwin; | ||||
|     bool active; | ||||
|     uint8_t type; | ||||
|     int side_pos;    /* current position of the sidebar - used for scrolling up and down */ | ||||
|     time_t start_time; | ||||
|  | ||||
|     char title[CONFERENCE_MAX_TITLE_LENGTH + 1]; | ||||
|     size_t title_length; | ||||
|  | ||||
|     ConferencePeer *peer_list; | ||||
|     uint32_t max_idx; | ||||
|  | ||||
|     NameListEntry *name_list; | ||||
|     uint32_t num_peers; | ||||
|  | ||||
|     bool push_to_talk_enabled; | ||||
|     time_t ptt_last_pushed; | ||||
|  | ||||
|     bool audio_enabled; | ||||
|     time_t last_sent_audio; | ||||
|     uint32_t audio_in_idx; | ||||
|     AudioInputCallbackData audio_input_callback_data; | ||||
| } ConferenceChat; | ||||
|  | ||||
| /* Frees all Toxic associated data structures for a conference (does not call tox_conference_delete() ) */ | ||||
| void free_conference(ToxWindow *self, uint32_t conferencenum); | ||||
|  | ||||
| int init_conference_win(Tox *m, uint32_t conferencenum, uint8_t type, const char *title, size_t length); | ||||
|  | ||||
| /* destroys and re-creates conference window with or without the peerlist */ | ||||
| void redraw_conference_win(ToxWindow *self); | ||||
|  | ||||
| void conference_set_title(ToxWindow *self, uint32_t conferencesnum, const char *title, size_t length); | ||||
| void conference_rename_log_path(Tox *m, uint32_t conferencenum, const char *new_title); | ||||
| int conference_enable_logging(ToxWindow *self, Tox *m, uint32_t conferencenum, struct chatlog *log); | ||||
|  | ||||
| /* Puts `(NameListEntry *)`s in `entries` for each matched peer, up to a maximum | ||||
|  * of `maxpeers`. | ||||
|  * Maches each peer whose name or pubkey begins with `prefix`. | ||||
|  * If `prefix` is exactly the pubkey of a peer, matches only that peer. | ||||
|  * return number of entries placed in `entries`. | ||||
|  */ | ||||
| uint32_t get_name_list_entries_by_prefix(uint32_t conferencenum, const char *prefix, NameListEntry **entries, | ||||
|         uint32_t maxpeers); | ||||
|  | ||||
| bool init_conference_audio_input(Tox *tox, uint32_t conferencenum); | ||||
| bool enable_conference_audio(Tox *tox, uint32_t conferencenum); | ||||
| bool disable_conference_audio(Tox *tox, uint32_t conferencenum); | ||||
| bool toggle_conference_push_to_talk(uint32_t conferencenum, bool enabled); | ||||
| void audio_conference_callback(void *tox, uint32_t conferencenum, uint32_t peernum, | ||||
|                                const int16_t *pcm, unsigned int samples, uint8_t channels, uint32_t | ||||
|                                sample_rate, void *userdata); | ||||
|  | ||||
| bool conference_mute_self(uint32_t conferencenum); | ||||
| bool conference_mute_peer(const Tox *m, uint32_t conferencenum, uint32_t peernum); | ||||
| bool conference_set_VAD_threshold(uint32_t conferencenum, float threshold); | ||||
| float conference_get_VAD_threshold(uint32_t conferencenum); | ||||
|  | ||||
| #endif /* CONFERENCE_H */ | ||||
							
								
								
									
										210
									
								
								src/conference_commands.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										210
									
								
								src/conference_commands.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,210 @@ | ||||
| /*  conference_commands.c | ||||
|  * | ||||
|  * | ||||
|  *  Copyright (C) 2014 Toxic All Rights Reserved. | ||||
|  * | ||||
|  *  This file is part of Toxic. | ||||
|  * | ||||
|  *  Toxic is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|  *  the Free Software Foundation, either version 3 of the License, or | ||||
|  *  (at your option) any later version. | ||||
|  * | ||||
|  *  Toxic is distributed in the hope that it will be useful, | ||||
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  *  GNU General Public License for more details. | ||||
|  * | ||||
|  *  You should have received a copy of the GNU General Public License | ||||
|  *  along with Toxic.  If not, see <http://www.gnu.org/licenses/>. | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
|  | ||||
| #include "conference.h" | ||||
| #include "line_info.h" | ||||
| #include "log.h" | ||||
| #include "misc_tools.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| static void print_err(ToxWindow *self, const char *error_str) | ||||
| { | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "%s", error_str); | ||||
| } | ||||
|  | ||||
| void cmd_conference_set_title(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|  | ||||
|     Tox_Err_Conference_Title err; | ||||
|     char title[CONFERENCE_MAX_TITLE_LENGTH + 1]; | ||||
|  | ||||
|     if (argc < 1) { | ||||
|         size_t tlen = tox_conference_get_title_size(m, self->num, &err); | ||||
|  | ||||
|         if (err != TOX_ERR_CONFERENCE_TITLE_OK || tlen >= sizeof(title)) { | ||||
|             print_err(self, "Title is not set"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (!tox_conference_get_title(m, self->num, (uint8_t *) title, &err)) { | ||||
|             print_err(self, "Title is not set"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         title[tlen] = '\0'; | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Title is set to: %s", title); | ||||
|  | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     size_t len = strlen(argv[1]); | ||||
|  | ||||
|     if (len >= sizeof(title)) { | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to set title: max length exceeded."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     snprintf(title, sizeof(title), "%s", argv[1]); | ||||
|  | ||||
|     if (!tox_conference_set_title(m, self->num, (uint8_t *) title, len, &err)) { | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to set title (error %d)", err); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     conference_rename_log_path(m, self->num, title);  // must be called first | ||||
|  | ||||
|     conference_set_title(self, self->num, title, len); | ||||
|  | ||||
|     char selfnick[TOX_MAX_NAME_LENGTH]; | ||||
|     tox_self_get_name(m, (uint8_t *) selfnick); | ||||
|  | ||||
|     size_t sn_len = tox_self_get_name_size(m); | ||||
|     selfnick[sn_len] = '\0'; | ||||
|  | ||||
|     line_info_add(self, true, selfnick, NULL, NAME_CHANGE, 0, 0, " set the conference title to: %s", title); | ||||
|  | ||||
|     char tmp_event[MAX_STR_SIZE + 20]; | ||||
|     snprintf(tmp_event, sizeof(tmp_event), "set title to %s", title); | ||||
|     write_to_log(tmp_event, selfnick, self->chatwin->log, true); | ||||
| } | ||||
|  | ||||
| #ifdef AUDIO | ||||
| void cmd_enable_audio(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|  | ||||
|     bool enable; | ||||
|  | ||||
|     if (argc == 1 && !strcasecmp(argv[1], "on")) { | ||||
|         enable = true; | ||||
|     } else if (argc == 1 && !strcasecmp(argv[1], "off")) { | ||||
|         enable = false; | ||||
|     } else { | ||||
|         print_err(self, "Please specify: on | off"); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (enable ? enable_conference_audio(m, self->num) : disable_conference_audio(m, self->num)) { | ||||
|         print_err(self, enable ? "Enabled conference audio. Use the '/ptt' command to toggle Push-To-Talk." | ||||
|                   : "Disabled conference audio"); | ||||
|     } else { | ||||
|         print_err(self, enable ? "Failed to enable audio" : "Failed to disable audio"); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void cmd_conference_mute(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|  | ||||
|     if (argc < 1) { | ||||
|         if (conference_mute_self(self->num)) { | ||||
|             print_err(self, "Toggled self audio mute status"); | ||||
|         } else { | ||||
|             print_err(self, "No audio input to mute"); | ||||
|         } | ||||
|     } else { | ||||
|         NameListEntry *entries[16]; | ||||
|         uint32_t n = get_name_list_entries_by_prefix(self->num, argv[1], entries, 16); | ||||
|  | ||||
|         if (n == 0) { | ||||
|             print_err(self, "No such peer"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (n > 1) { | ||||
|             print_err(self, "Multiple matching peers (use /mute [public key] to disambiguate):"); | ||||
|  | ||||
|             for (uint32_t i = 0; i < n; ++i) { | ||||
|                 line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "%s: %s", entries[i]->pubkey_str, entries[i]->name); | ||||
|             } | ||||
|  | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (conference_mute_peer(m, self->num, entries[0]->peernum)) { | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Toggled audio mute status of %s", entries[0]->name); | ||||
|         } else { | ||||
|             print_err(self, "Peer is not on the call"); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| void cmd_conference_sense(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|     UNUSED_VAR(m); | ||||
|  | ||||
|     if (argc == 0) { | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Current VAD threshold: %.1f", | ||||
|                       (double) conference_get_VAD_threshold(self->num)); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (argc > 1) { | ||||
|         print_err(self, "Only one argument allowed."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     char *end; | ||||
|     float value = strtof(argv[1], &end); | ||||
|  | ||||
|     if (*end) { | ||||
|         print_err(self, "Invalid input"); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (conference_set_VAD_threshold(self->num, value)) { | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Set VAD threshold to %.1f", (double) value); | ||||
|     } else { | ||||
|         print_err(self, "Failed to set conference audio input sensitivity."); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void cmd_conference_push_to_talk(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|     UNUSED_VAR(m); | ||||
|  | ||||
|     bool enable; | ||||
|  | ||||
|     if (argc == 1 && !strcasecmp(argv[1], "on")) { | ||||
|         enable = true; | ||||
|     } else if (argc == 1 && !strcasecmp(argv[1], "off")) { | ||||
|         enable = false; | ||||
|     } else { | ||||
|         print_err(self, "Please specify: on | off"); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (!toggle_conference_push_to_talk(self->num, enable)) { | ||||
|         print_err(self, "Failed to toggle push to talk."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     print_err(self, enable ? "Push-To-Talk is enabled. Push F2 to activate" : "Push-To-Talk is disabled"); | ||||
| } | ||||
| #endif /* AUDIO */ | ||||
| @@ -1,4 +1,4 @@ | ||||
| /*  xtra.h
 | ||||
| /*  conference_commands.h
 | ||||
|  * | ||||
|  * | ||||
|  *  Copyright (C) 2014 Toxic All Rights Reserved. | ||||
| @@ -20,21 +20,16 @@ | ||||
|  * | ||||
|  */ | ||||
| 
 | ||||
| #ifndef XTRA_H | ||||
| #define XTRA_H | ||||
| #ifndef CONFERENCE_COMMANDS_H | ||||
| #define CONFERENCE_COMMANDS_H | ||||
| 
 | ||||
| /* NOTE: If no xlib present don't compile */ | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| 
 | ||||
| typedef enum { | ||||
|     DT_plain, | ||||
|     DT_file_list | ||||
| } | ||||
| DropType; | ||||
| void cmd_conference_set_title(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_enable_audio(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_conference_mute(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_conference_sense(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_conference_push_to_talk(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| 
 | ||||
| typedef void (*drop_callback)(const char *, DropType); | ||||
| 
 | ||||
| int               init_xtra(drop_callback d); | ||||
| void              terminate_xtra(void); | ||||
| int               is_focused(void); /* returns bool */ | ||||
| 
 | ||||
| #endif /* XTRA_H */ | ||||
| #endif /* CONFERENCE_COMMANDS_H */ | ||||
| @@ -20,18 +20,18 @@ | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <string.h> | ||||
| #include <stdlib.h> | ||||
| #include <stdio.h> | ||||
| #include <sys/types.h> | ||||
| #include <sys/stat.h> | ||||
| #include <errno.h> | ||||
| #include <unistd.h> | ||||
| #include <pwd.h> | ||||
| #include <stdio.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <sys/stat.h> | ||||
| #include <sys/types.h> | ||||
| #include <unistd.h> | ||||
|  | ||||
| #include "toxic.h" | ||||
| #include "configdir.h" | ||||
| #include "misc_tools.h" | ||||
| #include "toxic.h" | ||||
|  | ||||
| /* get the user's home directory. */ | ||||
| void get_home_dir(char *home, int size) | ||||
|   | ||||
| @@ -23,6 +23,8 @@ | ||||
| #ifndef CURL_UTIL_H | ||||
| #define CURL_UTIL_H | ||||
|  | ||||
| #include <stdint.h> | ||||
|  | ||||
| /* List based on Mozilla's recommended configurations for modern browsers */ | ||||
| #define TLS_CIPHER_SUITE_LIST "ECDHE-RSA-AES128-GCM-SHA256:ECDHE-ECDSA-AES128-GCM-SHA256:ECDHE-RSA-AES256-GCM-SHA384:ECDHE-ECDSA-AES256-GCM-SHA384:DHE-RSA-AES128-GCM-SHA256:DHE-DSS-AES128-GCM-SHA256:kEDH+AESGCM:ECDHE-RSA-AES128-SHA256:ECDHE-ECDSA-AES128-SHA256:ECDHE-RSA-AES128-SHA:ECDHE-ECDSA-AES128-SHA:ECDHE-RSA-AES256-SHA384:ECDHE-ECDSA-AES256-SHA384:ECDHE-RSA-AES256-SHA:ECDHE-ECDSA-AES256-SHA:DHE-RSA-AES128-SHA256:DHE-RSA-AES128-SHA:DHE-DSS-AES128-SHA256:DHE-RSA-AES256-SHA256:DHE-DSS-AES256-SHA:DHE-RSA-AES256-SHA:!aNULL:!eNULL:!EXPORT:!DES:!RC4:!3DES:!MD5:!PSK" | ||||
|  | ||||
|   | ||||
| @@ -20,20 +20,20 @@ | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <assert.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <assert.h> | ||||
|  | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "execute.h" | ||||
| #include "api.h" | ||||
| #include "chat_commands.h" | ||||
| #include "execute.h" | ||||
| #include "global_commands.h" | ||||
| #include "group_commands.h" | ||||
| #include "conference_commands.h" | ||||
| #include "line_info.h" | ||||
| #include "misc_tools.h" | ||||
| #include "notify.h" | ||||
| #include "api.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| struct cmd_func { | ||||
|     const char *name; | ||||
| @@ -48,7 +48,7 @@ static struct cmd_func global_commands[] = { | ||||
|     { "/connect",   cmd_connect       }, | ||||
|     { "/decline",   cmd_decline       }, | ||||
|     { "/exit",      cmd_quit          }, | ||||
|     { "/group",     cmd_groupchat     }, | ||||
|     { "/conference", cmd_conference    }, | ||||
|     { "/help",      cmd_prompt_help   }, | ||||
|     { "/log",       cmd_log           }, | ||||
|     { "/myid",      cmd_myid          }, | ||||
| @@ -78,8 +78,8 @@ static struct cmd_func global_commands[] = { | ||||
|  | ||||
| static struct cmd_func chat_commands[] = { | ||||
|     { "/cancel",    cmd_cancelfile        }, | ||||
|     { "/invite",    cmd_groupinvite }, | ||||
|     { "/join",      cmd_join_group  }, | ||||
|     { "/invite",    cmd_conference_invite }, | ||||
|     { "/join",      cmd_conference_join   }, | ||||
|     { "/savefile",  cmd_savefile          }, | ||||
|     { "/sendfile",  cmd_sendfile          }, | ||||
| #ifdef AUDIO | ||||
| @@ -92,26 +92,30 @@ static struct cmd_func chat_commands[] = { | ||||
|     { "/bitrate",   cmd_bitrate           }, | ||||
| #endif /* AUDIO */ | ||||
| #ifdef VIDEO | ||||
|     { "/vcall",     cmd_vcall             }, | ||||
|     { "/video",     cmd_video             }, | ||||
|     { "/res",       cmd_res               }, | ||||
| #endif /* VIDEO */ | ||||
|     { NULL,         NULL                  }, | ||||
| }; | ||||
|  | ||||
| static struct cmd_func group_commands[] = { | ||||
|     { "/title",     cmd_set_title   }, | ||||
| static struct cmd_func conference_commands[] = { | ||||
|     { "/title",     cmd_conference_set_title }, | ||||
|  | ||||
| #ifdef AUDIO | ||||
|     { "/mute",      cmd_mute        }, | ||||
|     { "/sense",     cmd_sense       }, | ||||
|     { "/audio",     cmd_enable_audio }, | ||||
|     { "/mute",      cmd_conference_mute   }, | ||||
|     { "/ptt",       cmd_conference_push_to_talk }, | ||||
|     { "/sense",     cmd_conference_sense  }, | ||||
| #endif /* AUDIO */ | ||||
|     { NULL,         NULL             }, | ||||
| }; | ||||
|  | ||||
|  | ||||
| #ifdef PYTHON | ||||
| #define SPECIAL_COMMANDS 6 | ||||
| #define SPECIAL_COMMANDS 7 | ||||
| #else | ||||
| #define SPECIAL_COMMANDS 5 | ||||
| #define SPECIAL_COMMANDS 6 | ||||
| #endif /* PYTHON */ | ||||
|  | ||||
| /* Special commands are commands that only take one argument even if it contains spaces */ | ||||
| @@ -122,8 +126,9 @@ static const char special_commands[SPECIAL_COMMANDS][MAX_CMDNAME_SIZE] = { | ||||
| #ifdef PYTHON | ||||
|     "/run", | ||||
| #endif /* PYTHON */ | ||||
|     "/title", | ||||
|     "/sendfile", | ||||
|     "/title", | ||||
|     "/mute", | ||||
| }; | ||||
|  | ||||
| /* Returns true if input command is in the special_commands array. */ | ||||
| @@ -145,7 +150,7 @@ static bool is_special_command(const char *input) | ||||
|  * | ||||
|  * Returns the number of arguments. | ||||
|  */ | ||||
| static int parse_special_command(WINDOW *w, ToxWindow *self, const char *input, char (*args)[MAX_STR_SIZE]) | ||||
| static int parse_special_command(const char *input, char (*args)[MAX_STR_SIZE]) | ||||
| { | ||||
|     int len = strlen(input); | ||||
|     int s = char_find(0, input, ' '); | ||||
| @@ -167,10 +172,10 @@ static int parse_special_command(WINDOW *w, ToxWindow *self, const char *input, | ||||
|  * | ||||
|  * Returns the number of arguments. | ||||
|  */ | ||||
| static int parse_command(WINDOW *w, ToxWindow *self, const char *input, char (*args)[MAX_STR_SIZE]) | ||||
| static int parse_command(const char *input, char (*args)[MAX_STR_SIZE]) | ||||
| { | ||||
|     if (is_special_command(input)) { | ||||
|         return parse_special_command(w, self, input, args); | ||||
|         return parse_special_command(input, args); | ||||
|     } | ||||
|  | ||||
|     char *cmd = strdup(input); | ||||
| @@ -227,7 +232,7 @@ void execute(WINDOW *w, ToxWindow *self, Tox *m, const char *input, int mode) | ||||
|     } | ||||
|  | ||||
|     char args[MAX_NUM_ARGS][MAX_STR_SIZE]; | ||||
|     int num_args = parse_command(w, self, input, args); | ||||
|     int num_args = parse_command(input, args); | ||||
|  | ||||
|     if (num_args <= 0) { | ||||
|         return; | ||||
| @@ -246,8 +251,8 @@ void execute(WINDOW *w, ToxWindow *self, Tox *m, const char *input, int mode) | ||||
|  | ||||
|             break; | ||||
|  | ||||
|         case GROUPCHAT_COMMAND_MODE: | ||||
|             if (do_command(w, self, m, num_args, group_commands, args) == 0) { | ||||
|         case CONFERENCE_COMMAND_MODE: | ||||
|             if (do_command(w, self, m, num_args, conference_commands, args) == 0) { | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
| @@ -266,5 +271,5 @@ void execute(WINDOW *w, ToxWindow *self, Tox *m, const char *input, int mode) | ||||
|  | ||||
| #endif | ||||
|  | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invalid command."); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid command."); | ||||
| } | ||||
|   | ||||
| @@ -31,7 +31,7 @@ | ||||
| enum { | ||||
|     GLOBAL_COMMAND_MODE, | ||||
|     CHAT_COMMAND_MODE, | ||||
|     GROUPCHAT_COMMAND_MODE, | ||||
|     CONFERENCE_COMMAND_MODE, | ||||
| }; | ||||
|  | ||||
| void execute(WINDOW *w, ToxWindow *self, Tox *m, const char *input, int mode); | ||||
|   | ||||
| @@ -20,23 +20,24 @@ | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <string.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <time.h> | ||||
| #include <unistd.h> | ||||
|  | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "friendlist.h" | ||||
| #include "file_transfers.h" | ||||
| #include "friendlist.h" | ||||
| #include "line_info.h" | ||||
| #include "misc_tools.h" | ||||
| #include "notify.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| extern FriendsList Friends; | ||||
|  | ||||
| /* number of "#"'s in file transfer progress bar. Keep well below MAX_STR_SIZE */ | ||||
| #define NUM_PROG_MARKS 50 | ||||
| #define STR_BUF_SIZE 30 | ||||
|  | ||||
| /* creates initial progress line that will be updated during file transfer. | ||||
|    Assumes progline has room for at least MAX_STR_SIZE bytes */ | ||||
| @@ -59,13 +60,15 @@ void print_progress_bar(ToxWindow *self, double bps, double pct_done, uint32_t l | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     char pct_str[24]; | ||||
|     char pct_str[STR_BUF_SIZE]; | ||||
|     snprintf(pct_str, sizeof(pct_str), "%.1f%%", pct_done); | ||||
|  | ||||
|     char bps_str[24]; | ||||
|     char bps_str[STR_BUF_SIZE]; | ||||
|     bytes_convert_str(bps_str, sizeof(bps_str), bps); | ||||
|  | ||||
|     char prog_line[NUM_PROG_MARKS + 1] = {0}; | ||||
|     char prog_line[NUM_PROG_MARKS + 1]; | ||||
|     prog_line[0] = 0; | ||||
|  | ||||
|     int n = pct_done / (100 / NUM_PROG_MARKS); | ||||
|     int i, j; | ||||
|  | ||||
| @@ -81,13 +84,21 @@ void print_progress_bar(ToxWindow *self, double bps, double pct_done, uint32_t l | ||||
|         strcat(prog_line, "-"); | ||||
|     } | ||||
|  | ||||
|     char full_line[strlen(pct_str) + NUM_PROG_MARKS + strlen(bps_str) + 7]; | ||||
|     snprintf(full_line, sizeof(full_line), "%s [%s] %s/s", pct_str, prog_line, bps_str); | ||||
|     size_t line_buf_size = strlen(pct_str) + NUM_PROG_MARKS + strlen(bps_str) + 7; | ||||
|     char *full_line = malloc(line_buf_size); | ||||
|  | ||||
|     if (full_line == NULL) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     snprintf(full_line, line_buf_size, "%s [%s] %s/s", pct_str, prog_line, bps_str); | ||||
|  | ||||
|     line_info_set(self, line_id, full_line); | ||||
|  | ||||
|     free(full_line); | ||||
| } | ||||
|  | ||||
| static void refresh_progress_helper(ToxWindow *self, Tox *m, struct FileTransfer *ft) | ||||
| static void refresh_progress_helper(ToxWindow *self, struct FileTransfer *ft) | ||||
| { | ||||
|     if (ft->state == FILE_TRANSFER_INACTIVE) { | ||||
|         return; | ||||
| @@ -107,33 +118,36 @@ static void refresh_progress_helper(ToxWindow *self, Tox *m, struct FileTransfer | ||||
| } | ||||
|  | ||||
| /* refreshes active file transfer status bars. */ | ||||
| void refresh_file_transfer_progress(ToxWindow *self, Tox *m, uint32_t friendnum) | ||||
| void refresh_file_transfer_progress(ToxWindow *self, uint32_t friendnumber) | ||||
| { | ||||
|     size_t i; | ||||
|  | ||||
|     for (i = 0; i < MAX_FILES; ++i) { | ||||
|         refresh_progress_helper(self, m, &Friends.list[friendnum].file_receiver[i]); | ||||
|         refresh_progress_helper(self, m, &Friends.list[friendnum].file_sender[i]); | ||||
|     for (size_t i = 0; i < MAX_FILES; ++i) { | ||||
|         refresh_progress_helper(self, &Friends.list[friendnumber].file_receiver[i]); | ||||
|         refresh_progress_helper(self, &Friends.list[friendnumber].file_sender[i]); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /* Returns a pointer to friendnum's FileTransfer struct associated with filenum. | ||||
|  * Returns NULL if filenum is invalid. | ||||
|  */ | ||||
| struct FileTransfer *get_file_transfer_struct(uint32_t friendnum, uint32_t filenum) | ||||
| static void clear_file_transfer(struct FileTransfer *ft) | ||||
| { | ||||
|     size_t i; | ||||
|     *ft = (struct FileTransfer) { | ||||
|         0 | ||||
|     }; | ||||
| } | ||||
|  | ||||
|     for (i = 0; i < MAX_FILES; ++i) { | ||||
|         struct FileTransfer *ft_send = &Friends.list[friendnum].file_sender[i]; | ||||
| /* Returns a pointer to friendnumber's FileTransfer struct associated with filenumber. | ||||
|  * Returns NULL if filenumber is invalid. | ||||
|  */ | ||||
| struct FileTransfer *get_file_transfer_struct(uint32_t friendnumber, uint32_t filenumber) | ||||
| { | ||||
|     for (size_t i = 0; i < MAX_FILES; ++i) { | ||||
|         struct FileTransfer *ft_send = &Friends.list[friendnumber].file_sender[i]; | ||||
|  | ||||
|         if (ft_send->state != FILE_TRANSFER_INACTIVE && ft_send->filenum == filenum) { | ||||
|         if (ft_send->state != FILE_TRANSFER_INACTIVE && ft_send->filenumber == filenumber) { | ||||
|             return ft_send; | ||||
|         } | ||||
|  | ||||
|         struct FileTransfer *ft_recv = &Friends.list[friendnum].file_receiver[i]; | ||||
|         struct FileTransfer *ft_recv = &Friends.list[friendnumber].file_receiver[i]; | ||||
|  | ||||
|         if (ft_recv->state != FILE_TRANSFER_INACTIVE && ft_recv->filenum == filenum) { | ||||
|         if (ft_recv->state != FILE_TRANSFER_INACTIVE && ft_recv->filenumber == filenumber) { | ||||
|             return ft_recv; | ||||
|         } | ||||
|     } | ||||
| @@ -144,19 +158,17 @@ struct FileTransfer *get_file_transfer_struct(uint32_t friendnum, uint32_t filen | ||||
| /* Returns a pointer to the FileTransfer struct associated with index with the direction specified. | ||||
|  * Returns NULL on failure. | ||||
|  */ | ||||
| struct FileTransfer *get_file_transfer_struct_index(uint32_t friendnum, uint32_t index, | ||||
| struct FileTransfer *get_file_transfer_struct_index(uint32_t friendnumber, uint32_t index, | ||||
|         FILE_TRANSFER_DIRECTION direction) | ||||
| { | ||||
|     if (direction != FILE_TRANSFER_RECV && direction != FILE_TRANSFER_SEND) { | ||||
|         return NULL; | ||||
|     } | ||||
|  | ||||
|     size_t i; | ||||
|  | ||||
|     for (i = 0; i < MAX_FILES; ++i) { | ||||
|     for (size_t i = 0; i < MAX_FILES; ++i) { | ||||
|         struct FileTransfer *ft = direction == FILE_TRANSFER_SEND ? | ||||
|                                       &Friends.list[friendnum].file_sender[i] : | ||||
|                                       &Friends.list[friendnum].file_receiver[i]; | ||||
|                                       &Friends.list[friendnumber].file_sender[i] : | ||||
|                                       &Friends.list[friendnumber].file_receiver[i]; | ||||
|  | ||||
|         if (ft->state != FILE_TRANSFER_INACTIVE && ft->index == index) { | ||||
|             return ft; | ||||
| @@ -169,23 +181,19 @@ struct FileTransfer *get_file_transfer_struct_index(uint32_t friendnum, uint32_t | ||||
| /* Returns a pointer to an unused file sender. | ||||
|  * Returns NULL if all file senders are in use. | ||||
|  */ | ||||
| static struct FileTransfer *new_file_sender(ToxWindow *window, uint32_t friendnum, uint32_t filenum, uint8_t type) | ||||
| static struct FileTransfer *new_file_sender(ToxWindow *window, uint32_t friendnumber, uint32_t filenumber, uint8_t type) | ||||
| { | ||||
|     size_t i; | ||||
|  | ||||
|     for (i = 0; i < MAX_FILES; ++i) { | ||||
|         struct FileTransfer *ft = &Friends.list[friendnum].file_sender[i]; | ||||
|     for (size_t i = 0; i < MAX_FILES; ++i) { | ||||
|         struct FileTransfer *ft = &Friends.list[friendnumber].file_sender[i]; | ||||
|  | ||||
|         if (ft->state == FILE_TRANSFER_INACTIVE) { | ||||
|             memset(ft, 0, sizeof(struct FileTransfer)); | ||||
|             clear_file_transfer(ft); | ||||
|             ft->window = window; | ||||
|             ft->index = i; | ||||
|             ft->friendnum = friendnum; | ||||
|             ft->filenum = filenum; | ||||
|             ft->friendnumber = friendnumber; | ||||
|             ft->filenumber = filenumber; | ||||
|             ft->file_type = type; | ||||
|             ft->last_keep_alive = get_unix_time(); | ||||
|             ft->state = FILE_TRANSFER_PENDING; | ||||
|             ft->direction = FILE_TRANSFER_SEND; | ||||
|             return ft; | ||||
|         } | ||||
|     } | ||||
| @@ -196,23 +204,20 @@ static struct FileTransfer *new_file_sender(ToxWindow *window, uint32_t friendnu | ||||
| /* Returns a pointer to an unused file receiver. | ||||
|  * Returns NULL if all file receivers are in use. | ||||
|  */ | ||||
| static struct FileTransfer *new_file_receiver(ToxWindow *window, uint32_t friendnum, uint32_t filenum, uint8_t type) | ||||
| static struct FileTransfer *new_file_receiver(ToxWindow *window, uint32_t friendnumber, uint32_t filenumber, | ||||
|         uint8_t type) | ||||
| { | ||||
|     size_t i; | ||||
|  | ||||
|     for (i = 0; i < MAX_FILES; ++i) { | ||||
|         struct FileTransfer *ft = &Friends.list[friendnum].file_receiver[i]; | ||||
|     for (size_t i = 0; i < MAX_FILES; ++i) { | ||||
|         struct FileTransfer *ft = &Friends.list[friendnumber].file_receiver[i]; | ||||
|  | ||||
|         if (ft->state == FILE_TRANSFER_INACTIVE) { | ||||
|             memset(ft, 0, sizeof(struct FileTransfer)); | ||||
|             clear_file_transfer(ft); | ||||
|             ft->window = window; | ||||
|             ft->index = i; | ||||
|             ft->friendnum = friendnum; | ||||
|             ft->filenum = filenum; | ||||
|             ft->friendnumber = friendnumber; | ||||
|             ft->filenumber = filenumber; | ||||
|             ft->file_type = type; | ||||
|             ft->last_keep_alive = get_unix_time(); | ||||
|             ft->state = FILE_TRANSFER_PENDING; | ||||
|             ft->direction = FILE_TRANSFER_RECV; | ||||
|             return ft; | ||||
|         } | ||||
|     } | ||||
| @@ -223,15 +228,15 @@ static struct FileTransfer *new_file_receiver(ToxWindow *window, uint32_t friend | ||||
| /* Initializes an unused file transfer and returns its pointer. | ||||
|  * Returns NULL on failure. | ||||
|  */ | ||||
| struct FileTransfer *new_file_transfer(ToxWindow *window, uint32_t friendnum, uint32_t filenum, | ||||
| struct FileTransfer *new_file_transfer(ToxWindow *window, uint32_t friendnumber, uint32_t filenumber, | ||||
|                                        FILE_TRANSFER_DIRECTION direction, uint8_t type) | ||||
| { | ||||
|     if (direction == FILE_TRANSFER_RECV) { | ||||
|         return new_file_receiver(window, friendnum, filenum, type); | ||||
|         return new_file_receiver(window, friendnumber, filenumber, type); | ||||
|     } | ||||
|  | ||||
|     if (direction == FILE_TRANSFER_SEND) { | ||||
|         return new_file_sender(window, friendnum, filenum, type); | ||||
|         return new_file_sender(window, friendnumber, filenumber, type); | ||||
|     } | ||||
|  | ||||
|     return NULL; | ||||
| @@ -259,7 +264,7 @@ void close_file_transfer(ToxWindow *self, Tox *m, struct FileTransfer *ft, int C | ||||
|     } | ||||
|  | ||||
|     if (CTRL >= 0) { | ||||
|         tox_file_control(m, ft->friendnum, ft->filenum, (Tox_File_Control) CTRL, NULL); | ||||
|         tox_file_control(m, ft->friendnumber, ft->filenumber, (Tox_File_Control) CTRL, NULL); | ||||
|     } | ||||
|  | ||||
|     if (message && self) { | ||||
| @@ -269,28 +274,36 @@ void close_file_transfer(ToxWindow *self, Tox *m, struct FileTransfer *ft, int C | ||||
|             box_notify(self, sound_type, NT_NOFOCUS | NT_WNDALERT_2, &self->active_box, self->name, "%s", message); | ||||
|         } | ||||
|  | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "%s", message); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "%s", message); | ||||
|     } | ||||
|  | ||||
|     memset(ft, 0, sizeof(struct FileTransfer)); | ||||
|     clear_file_transfer(ft); | ||||
| } | ||||
|  | ||||
| /* Kills all active file transfers for friendnum */ | ||||
| void kill_all_file_transfers_friend(Tox *m, uint32_t friendnum) | ||||
| /* Kills active outgoing avatar file transfers for friendnumber */ | ||||
| void kill_avatar_file_transfers_friend(Tox *m, uint32_t friendnumber) | ||||
| { | ||||
|     size_t i; | ||||
|     for (size_t i = 0; i < MAX_FILES; ++i) { | ||||
|         struct FileTransfer *ft = &Friends.list[friendnumber].file_sender[i]; | ||||
|  | ||||
|     for (i = 0; i < MAX_FILES; ++i) { | ||||
|         close_file_transfer(NULL, m, &Friends.list[friendnum].file_sender[i], TOX_FILE_CONTROL_CANCEL, NULL, silent); | ||||
|         close_file_transfer(NULL, m, &Friends.list[friendnum].file_receiver[i], TOX_FILE_CONTROL_CANCEL, NULL, silent); | ||||
|         if (ft->file_type == TOX_FILE_KIND_AVATAR) { | ||||
|             close_file_transfer(NULL, m, ft, TOX_FILE_CONTROL_CANCEL, NULL, silent); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| /* Kills all active file transfers for friendnumber */ | ||||
| void kill_all_file_transfers_friend(Tox *m, uint32_t friendnumber) | ||||
| { | ||||
|     for (size_t i = 0; i < MAX_FILES; ++i) { | ||||
|         close_file_transfer(NULL, m, &Friends.list[friendnumber].file_sender[i], TOX_FILE_CONTROL_CANCEL, NULL, silent); | ||||
|         close_file_transfer(NULL, m, &Friends.list[friendnumber].file_receiver[i], TOX_FILE_CONTROL_CANCEL, NULL, silent); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void kill_all_file_transfers(Tox *m) | ||||
| { | ||||
|     size_t i; | ||||
|  | ||||
|     for (i = 0; i < Friends.max_idx; ++i) { | ||||
|     for (size_t i = 0; i < Friends.max_idx; ++i) { | ||||
|         kill_all_file_transfers_friend(m, Friends.list[i].num); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -25,9 +25,9 @@ | ||||
|  | ||||
| #include <limits.h> | ||||
|  | ||||
| #include "notify.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "notify.h" | ||||
|  | ||||
| #define KiB 1024 | ||||
| #define MiB 1048576       /* 1024^2 */ | ||||
| @@ -51,18 +51,16 @@ struct FileTransfer { | ||||
|     ToxWindow *window; | ||||
|     FILE *file; | ||||
|     FILE_TRANSFER_STATE state; | ||||
|     FILE_TRANSFER_DIRECTION direction; | ||||
|     uint8_t file_type; | ||||
|     char file_name[TOX_MAX_FILENAME_LENGTH + 1]; | ||||
|     char file_path[PATH_MAX + 1];    /* Not used by senders */ | ||||
|     double   bps; | ||||
|     uint32_t filenum; | ||||
|     uint32_t friendnum; | ||||
|     uint32_t filenumber; | ||||
|     uint32_t friendnumber; | ||||
|     size_t   index; | ||||
|     uint64_t file_size; | ||||
|     uint64_t position; | ||||
|     time_t   last_line_progress;   /* The last time we updated the progress bar */ | ||||
|     time_t   last_keep_alive;  /* The last time we sent or received data */ | ||||
|     uint32_t line_id; | ||||
|     uint8_t  file_id[TOX_FILE_ID_LENGTH]; | ||||
| }; | ||||
| @@ -75,24 +73,24 @@ void init_progress_bar(char *progline); | ||||
| void print_progress_bar(ToxWindow *self, double pct_done, double bps, uint32_t line_id); | ||||
|  | ||||
| /* refreshes active file transfer status bars. */ | ||||
| void refresh_file_transfer_progress(ToxWindow *self, Tox *m, uint32_t friendnum); | ||||
| void refresh_file_transfer_progress(ToxWindow *self, uint32_t friendnumber); | ||||
|  | ||||
| /* Returns a pointer to friendnum's FileTransfer struct associated with filenum. | ||||
|  * Returns NULL if filenum is invalid. | ||||
| /* Returns a pointer to friendnumber's FileTransfer struct associated with filenumber. | ||||
|  * Returns NULL if filenumber is invalid. | ||||
|  */ | ||||
| struct FileTransfer *get_file_transfer_struct(uint32_t friendnum, uint32_t filenum); | ||||
| struct FileTransfer *get_file_transfer_struct(uint32_t friendnumber, uint32_t filenumber); | ||||
|  | ||||
|  | ||||
| /* Returns a pointer to the FileTransfer struct associated with index with the direction specified. | ||||
|  * Returns NULL on failure. | ||||
|  */ | ||||
| struct FileTransfer *get_file_transfer_struct_index(uint32_t friendnum, uint32_t index, | ||||
| struct FileTransfer *get_file_transfer_struct_index(uint32_t friendnumber, uint32_t index, | ||||
|         FILE_TRANSFER_DIRECTION direction); | ||||
|  | ||||
| /* Initializes an unused file transfer and returns its pointer. | ||||
|  * Returns NULL on failure. | ||||
|  */ | ||||
| struct FileTransfer *new_file_transfer(ToxWindow *window, uint32_t friendnum, uint32_t filenum, | ||||
| struct FileTransfer *new_file_transfer(ToxWindow *window, uint32_t friendnumber, uint32_t filenumber, | ||||
|                                        FILE_TRANSFER_DIRECTION direction, uint8_t type); | ||||
|  | ||||
| /* Closes file transfer ft. | ||||
| @@ -103,8 +101,11 @@ struct FileTransfer *new_file_transfer(ToxWindow *window, uint32_t friendnum, ui | ||||
| void close_file_transfer(ToxWindow *self, Tox *m, struct FileTransfer *ft, int CTRL, const char *message, | ||||
|                          Notification sound_type); | ||||
|  | ||||
| /* Kills all active file transfers for friendnum */ | ||||
| void kill_all_file_transfers_friend(Tox *m, uint32_t friendnum); | ||||
| /* Kills active outgoing avatar file transfers for friendnumber */ | ||||
| void kill_avatar_file_transfers_friend(Tox *m, uint32_t friendnumber); | ||||
|  | ||||
| /* Kills all active file transfers for friendnumber */ | ||||
| void kill_all_file_transfers_friend(Tox *m, uint32_t friendnumber); | ||||
|  | ||||
| void kill_all_file_transfers(Tox *m); | ||||
|  | ||||
|   | ||||
							
								
								
									
										273
									
								
								src/friendlist.c
									
									
									
									
									
								
							
							
						
						
									
										273
									
								
								src/friendlist.c
									
									
									
									
									
								
							| @@ -20,26 +20,26 @@ | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <string.h> | ||||
| #include <stdint.h> | ||||
| #include <stdlib.h> | ||||
| #include <time.h> | ||||
| #include <arpa/inet.h> | ||||
| #include <assert.h> | ||||
| #include <stdint.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <time.h> | ||||
|  | ||||
| #include <tox/tox.h> | ||||
|  | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "avatars.h" | ||||
| #include "chat.h" | ||||
| #include "friendlist.h" | ||||
| #include "misc_tools.h" | ||||
| #include "line_info.h" | ||||
| #include "settings.h" | ||||
| #include "notify.h" | ||||
| #include "help.h" | ||||
| #include "line_info.h" | ||||
| #include "log.h" | ||||
| #include "avatars.h" | ||||
| #include "misc_tools.h" | ||||
| #include "notify.h" | ||||
| #include "settings.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| #ifdef AUDIO | ||||
| #include "audio_call.h" | ||||
| @@ -116,8 +116,8 @@ static void realloc_blocklist(int n) | ||||
| void kill_friendlist(ToxWindow *self) | ||||
| { | ||||
|     for (size_t i = 0; i < Friends.max_idx; ++i) { | ||||
|         if (Friends.list[i].active && Friends.list[i].group_invite.key != NULL) { | ||||
|             free(Friends.list[i].group_invite.key); | ||||
|         if (Friends.list[i].active && Friends.list[i].conference_invite.key != NULL) { | ||||
|             free(Friends.list[i].conference_invite.key); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -127,6 +127,20 @@ void kill_friendlist(ToxWindow *self) | ||||
|     del_window(self); | ||||
| } | ||||
|  | ||||
| static void clear_blocklist_index(size_t idx) | ||||
| { | ||||
|     Blocked.list[idx] = (BlockedFriend) { | ||||
|         0 | ||||
|     }; | ||||
| } | ||||
|  | ||||
| static void clear_friendlist_index(size_t idx) | ||||
| { | ||||
|     Friends.list[idx] = (ToxicFriend) { | ||||
|         0 | ||||
|     }; | ||||
| } | ||||
|  | ||||
| /* Saves the blocklist to path. If there are no items in the blocklist the | ||||
|  * empty file will be removed. | ||||
|  * | ||||
| @@ -160,8 +174,7 @@ static int save_blocklist(char *path) | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             BlockedFriend tmp; | ||||
|             memset(&tmp, 0, sizeof(BlockedFriend)); | ||||
|             BlockedFriend tmp = {0}; | ||||
|             tmp.namelength = htons(Blocked.list[i].namelength); | ||||
|             memcpy(tmp.name, Blocked.list[i].name, Blocked.list[i].namelength + 1);  // Include null byte | ||||
|             memcpy(tmp.pub_key, Blocked.list[i].pub_key, TOX_PUBLIC_KEY_SIZE); | ||||
| @@ -187,13 +200,21 @@ static int save_blocklist(char *path) | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     char temp_path[strlen(path) + strlen(TEMP_BLOCKLIST_EXT) + 1]; | ||||
|     snprintf(temp_path, sizeof(temp_path), "%s%s", path, TEMP_BLOCKLIST_EXT); | ||||
|     size_t temp_buf_size = strlen(path) + strlen(TEMP_BLOCKLIST_EXT) + 1; | ||||
|     char *temp_path = malloc(temp_buf_size); | ||||
|  | ||||
|     if (temp_path == NULL) { | ||||
|         free(data); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     snprintf(temp_path, temp_buf_size, "%s%s", path, TEMP_BLOCKLIST_EXT); | ||||
|  | ||||
|     FILE *fp = fopen(temp_path, "wb"); | ||||
|  | ||||
|     if (fp == NULL) { | ||||
|         free(data); | ||||
|         free(temp_path); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
| @@ -201,6 +222,7 @@ static int save_blocklist(char *path) | ||||
|         fprintf(stderr, "Failed to write blocklist data.\n"); | ||||
|         fclose(fp); | ||||
|         free(data); | ||||
|         free(temp_path); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
| @@ -208,9 +230,12 @@ static int save_blocklist(char *path) | ||||
|     free(data); | ||||
|  | ||||
|     if (rename(temp_path, path) != 0) { | ||||
|         free(temp_path); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     free(temp_path); | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| @@ -235,15 +260,22 @@ int load_blocklist(char *path) | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     char data[len]; | ||||
|     char *data = malloc(len); | ||||
|  | ||||
|     if (data == NULL) { | ||||
|         fclose(fp); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (fread(data, len, 1, fp) != 1) { | ||||
|         fclose(fp); | ||||
|         free(data); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (len % sizeof(BlockedFriend) != 0) { | ||||
|         fclose(fp); | ||||
|         free(data); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
| @@ -251,12 +283,9 @@ int load_blocklist(char *path) | ||||
|     Blocked.max_idx = num; | ||||
|     realloc_blocklist(num); | ||||
|  | ||||
|     int i; | ||||
|  | ||||
|     for (i = 0; i < num; ++i) { | ||||
|         BlockedFriend tmp; | ||||
|         memset(&tmp, 0, sizeof(BlockedFriend)); | ||||
|         memset(&Blocked.list[i], 0, sizeof(BlockedFriend)); | ||||
|     for (int i = 0; i < num; ++i) { | ||||
|         BlockedFriend tmp = {0}; | ||||
|         clear_blocklist_index(i); | ||||
|  | ||||
|         memcpy(&tmp, data + i * sizeof(BlockedFriend), sizeof(BlockedFriend)); | ||||
|         Blocked.list[i].namelength = ntohs(tmp.namelength); | ||||
| @@ -279,6 +308,8 @@ int load_blocklist(char *path) | ||||
|     } | ||||
|  | ||||
|     fclose(fp); | ||||
|     free(data); | ||||
|  | ||||
|     sort_blocklist_index(); | ||||
|  | ||||
|     return 0; | ||||
| @@ -308,7 +339,9 @@ void sort_friendlist_index(void) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (Friends.num_friends > 0) { | ||||
|         qsort(Friends.index, Friends.num_friends, sizeof(uint32_t), index_name_cmp); | ||||
|     } | ||||
| } | ||||
|  | ||||
| static int index_name_cmp_block(const void *n1, const void *n2) | ||||
| @@ -344,6 +377,10 @@ static void update_friend_last_online(uint32_t num, time_t timestamp) | ||||
| static void friendlist_onMessage(ToxWindow *self, Tox *m, uint32_t num, Tox_Message_Type type, const char *str, | ||||
|                                  size_t length) | ||||
| { | ||||
|     UNUSED_VAR(self); | ||||
|     UNUSED_VAR(type); | ||||
|     UNUSED_VAR(length); | ||||
|  | ||||
|     if (num >= Friends.max_idx) { | ||||
|         return; | ||||
|     } | ||||
| @@ -360,16 +397,15 @@ static void friendlist_onMessage(ToxWindow *self, Tox *m, uint32_t num, Tox_Mess | ||||
|     char nick[TOX_MAX_NAME_LENGTH]; | ||||
|     get_nick_truncate(m, nick, num); | ||||
|  | ||||
|     char timefrmt[TIME_STR_SIZE]; | ||||
|     get_time_str(timefrmt, sizeof(timefrmt)); | ||||
|  | ||||
|     line_info_add(prompt, timefrmt, nick, NULL, IN_MSG, 0, 0, "%s", str); | ||||
|     line_info_add(prompt, NULL, NULL, NULL, SYS_MSG, 0, RED, "* Warning: Too many windows are open."); | ||||
|     line_info_add(prompt, true, nick, NULL, IN_MSG, 0, 0, "%s", str); | ||||
|     line_info_add(prompt, false, NULL, NULL, SYS_MSG, 0, RED, "* Warning: Too many windows are open."); | ||||
|     sound_notify(prompt, notif_error, NT_WNDALERT_1, NULL); | ||||
| } | ||||
|  | ||||
| static void friendlist_onConnectionChange(ToxWindow *self, Tox *m, uint32_t num, Tox_Connection connection_status) | ||||
| { | ||||
|     UNUSED_VAR(self); | ||||
|  | ||||
|     if (num >= Friends.max_idx) { | ||||
|         return; | ||||
|     } | ||||
| @@ -380,7 +416,7 @@ static void friendlist_onConnectionChange(ToxWindow *self, Tox *m, uint32_t num, | ||||
|         ++Friends.num_online; | ||||
|  | ||||
|         if (avatar_send(m, num) == -1) { | ||||
|             fprintf(stderr, "avatar_send failed for friend %d\n", num); | ||||
|             fprintf(stderr, "avatar_send failed for friend %u\n", num); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -392,6 +428,9 @@ static void friendlist_onConnectionChange(ToxWindow *self, Tox *m, uint32_t num, | ||||
|  | ||||
| static void friendlist_onNickChange(ToxWindow *self, Tox *m, uint32_t num, const char *nick, size_t length) | ||||
| { | ||||
|     UNUSED_VAR(self); | ||||
|     UNUSED_VAR(length); | ||||
|  | ||||
|     if (num >= Friends.max_idx) { | ||||
|         return; | ||||
|     } | ||||
| @@ -411,7 +450,9 @@ static void friendlist_onNickChange(ToxWindow *self, Tox *m, uint32_t num, const | ||||
|     tox_self_get_address(m, (uint8_t *) myid); | ||||
|  | ||||
|     if (strcmp(oldname, newnamecpy) != 0) { | ||||
|         rename_logfile(oldname, newnamecpy, myid, Friends.list[num].pub_key, Friends.list[num].chatwin); | ||||
|         if (rename_logfile(oldname, newnamecpy, myid, Friends.list[num].pub_key, Friends.list[num].chatwin) != 0) { | ||||
|             fprintf(stderr, "Failed to rename friend chat log from `%s` to `%s`\n", oldname, newnamecpy); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     sort_friendlist_index(); | ||||
| @@ -419,6 +460,9 @@ static void friendlist_onNickChange(ToxWindow *self, Tox *m, uint32_t num, const | ||||
|  | ||||
| static void friendlist_onStatusChange(ToxWindow *self, Tox *m, uint32_t num, Tox_User_Status status) | ||||
| { | ||||
|     UNUSED_VAR(self); | ||||
|     UNUSED_VAR(m); | ||||
|  | ||||
|     if (num >= Friends.max_idx) { | ||||
|         return; | ||||
|     } | ||||
| @@ -428,6 +472,8 @@ static void friendlist_onStatusChange(ToxWindow *self, Tox *m, uint32_t num, Tox | ||||
|  | ||||
| static void friendlist_onStatusMessageChange(ToxWindow *self, uint32_t num, const char *note, size_t length) | ||||
| { | ||||
|     UNUSED_VAR(self); | ||||
|  | ||||
|     if (length > TOX_MAX_STATUS_MESSAGE_LENGTH || num >= Friends.max_idx) { | ||||
|         return; | ||||
|     } | ||||
| @@ -438,8 +484,10 @@ static void friendlist_onStatusMessageChange(ToxWindow *self, uint32_t num, cons | ||||
|  | ||||
| void friendlist_onFriendAdded(ToxWindow *self, Tox *m, uint32_t num, bool sort) | ||||
| { | ||||
|     UNUSED_VAR(self); | ||||
|  | ||||
|     realloc_friends(Friends.max_idx + 1); | ||||
|     memset(&Friends.list[Friends.max_idx], 0, sizeof(ToxicFriend)); | ||||
|     clear_friendlist_index(Friends.max_idx); | ||||
|  | ||||
|     uint32_t i; | ||||
|  | ||||
| @@ -473,10 +521,11 @@ void friendlist_onFriendAdded(ToxWindow *self, Tox *m, uint32_t num, bool sort) | ||||
|  | ||||
|         update_friend_last_online(i, t); | ||||
|  | ||||
|         char tempname[TOX_MAX_NAME_LENGTH] = {0}; | ||||
|         get_nick_truncate(m, tempname, num); | ||||
|         snprintf(Friends.list[i].name, sizeof(Friends.list[i].name), "%s", tempname); | ||||
|         Friends.list[i].namelength = strlen(Friends.list[i].name); | ||||
|         char tempname[TOX_MAX_NAME_LENGTH + 1]; | ||||
|         int name_len = get_nick_truncate(m, tempname, num); | ||||
|         memcpy(Friends.list[i].name, tempname, name_len); | ||||
|         Friends.list[i].name[name_len] = 0; | ||||
|         Friends.list[i].namelength = name_len; | ||||
|  | ||||
|         if (i == Friends.max_idx) { | ||||
|             ++Friends.max_idx; | ||||
| @@ -495,10 +544,10 @@ void friendlist_onFriendAdded(ToxWindow *self, Tox *m, uint32_t num, bool sort) | ||||
| } | ||||
|  | ||||
| /* Puts blocked friend back in friendlist. fnum is new friend number, bnum is blocked number. */ | ||||
| static void friendlist_add_blocked(Tox *m, uint32_t fnum, uint32_t bnum) | ||||
| static void friendlist_add_blocked(uint32_t fnum, uint32_t bnum) | ||||
| { | ||||
|     realloc_friends(Friends.max_idx + 1); | ||||
|     memset(&Friends.list[Friends.max_idx], 0, sizeof(ToxicFriend)); | ||||
|     clear_friendlist_index(Friends.max_idx); | ||||
|  | ||||
|     int i; | ||||
|  | ||||
| @@ -536,6 +585,11 @@ static void friendlist_add_blocked(Tox *m, uint32_t fnum, uint32_t bnum) | ||||
| static void friendlist_onFileRecv(ToxWindow *self, Tox *m, uint32_t num, uint32_t filenum, | ||||
|                                   uint64_t file_size, const char *filename, size_t name_length) | ||||
| { | ||||
|     UNUSED_VAR(self); | ||||
|     UNUSED_VAR(file_size); | ||||
|     UNUSED_VAR(filename); | ||||
|     UNUSED_VAR(name_length); | ||||
|  | ||||
|     if (num >= Friends.max_idx) { | ||||
|         return; | ||||
|     } | ||||
| @@ -554,15 +608,21 @@ static void friendlist_onFileRecv(ToxWindow *self, Tox *m, uint32_t num, uint32_ | ||||
|     char nick[TOX_MAX_NAME_LENGTH]; | ||||
|     get_nick_truncate(m, nick, num); | ||||
|  | ||||
|     line_info_add(prompt, NULL, NULL, NULL, SYS_MSG, 0, RED, | ||||
|     line_info_add(prompt, false, NULL, NULL, SYS_MSG, 0, RED, | ||||
|                   "* File transfer from %s failed: too many windows are open.", nick); | ||||
|  | ||||
|     sound_notify(prompt, notif_error, NT_WNDALERT_1, NULL); | ||||
| } | ||||
|  | ||||
| static void friendlist_onGroupInvite(ToxWindow *self, Tox *m, int32_t num, uint8_t type, const char *group_pub_key, | ||||
| static void friendlist_onConferenceInvite(ToxWindow *self, Tox *m, int32_t num, uint8_t type, | ||||
|         const char *conference_pub_key, | ||||
|         uint16_t length) | ||||
| { | ||||
|     UNUSED_VAR(self); | ||||
|     UNUSED_VAR(type); | ||||
|     UNUSED_VAR(conference_pub_key); | ||||
|     UNUSED_VAR(length); | ||||
|  | ||||
|     if (num >= Friends.max_idx) { | ||||
|         return; | ||||
|     } | ||||
| @@ -579,14 +639,14 @@ static void friendlist_onGroupInvite(ToxWindow *self, Tox *m, int32_t num, uint8 | ||||
|     char nick[TOX_MAX_NAME_LENGTH]; | ||||
|     get_nick_truncate(m, nick, num); | ||||
|  | ||||
|     line_info_add(prompt, NULL, NULL, NULL, SYS_MSG, 0, RED, | ||||
|                   "* Group chat invite from %s failed: too many windows are open.", nick); | ||||
|     line_info_add(prompt, false, NULL, NULL, SYS_MSG, 0, RED, | ||||
|                   "* Conference chat invite from %s failed: too many windows are open.", nick); | ||||
|  | ||||
|     sound_notify(prompt, notif_error, NT_WNDALERT_1, NULL); | ||||
| } | ||||
|  | ||||
| /* move friendlist/blocklist cursor up and down */ | ||||
| static void select_friend(ToxWindow *self, wint_t key, int *selected, int num) | ||||
| static void select_friend(wint_t key, int *selected, int num) | ||||
| { | ||||
|     if (num <= 0) { | ||||
|         return; | ||||
| @@ -626,11 +686,11 @@ static void delete_friend(Tox *m, uint32_t f_num) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (Friends.list[f_num].group_invite.key != NULL) { | ||||
|         free(Friends.list[f_num].group_invite.key); | ||||
|     if (Friends.list[f_num].conference_invite.key != NULL) { | ||||
|         free(Friends.list[f_num].conference_invite.key); | ||||
|     } | ||||
|  | ||||
|     memset(&Friends.list[f_num], 0, sizeof(ToxicFriend)); | ||||
|     clear_friendlist_index(f_num); | ||||
|  | ||||
|     int i; | ||||
|  | ||||
| @@ -656,7 +716,7 @@ static void delete_friend(Tox *m, uint32_t f_num) | ||||
| } | ||||
|  | ||||
| /* activates delete friend popup */ | ||||
| static void del_friend_activate(ToxWindow *self, Tox *m, uint32_t f_num) | ||||
| static void del_friend_activate(uint32_t f_num) | ||||
| { | ||||
|     PendingDelete.popup = newwin(3, 22 + TOXIC_MAX_NAME_LENGTH, 8, 8); | ||||
|     PendingDelete.active = true; | ||||
| @@ -666,9 +726,9 @@ static void del_friend_activate(ToxWindow *self, Tox *m, uint32_t f_num) | ||||
| static void delete_blocked_friend(uint32_t bnum); | ||||
|  | ||||
| /* deactivates delete friend popup and deletes friend if instructed */ | ||||
| static void del_friend_deactivate(ToxWindow *self, Tox *m, wint_t key) | ||||
| static void del_friend_deactivate(Tox *m, wint_t key) | ||||
| { | ||||
|     if (key == 'y') { | ||||
|     if (key == L'y') { | ||||
|         if (blocklist_view == 0) { | ||||
|             delete_friend(m, PendingDelete.num); | ||||
|             sort_friendlist_index(); | ||||
| @@ -679,7 +739,11 @@ static void del_friend_deactivate(ToxWindow *self, Tox *m, wint_t key) | ||||
|     } | ||||
|  | ||||
|     delwin(PendingDelete.popup); | ||||
|     memset(&PendingDelete, 0, sizeof(PendingDelete)); | ||||
|  | ||||
|     PendingDelete = (struct PendingDel) { | ||||
|         0 | ||||
|     }; | ||||
|  | ||||
|     clear(); | ||||
|     refresh(); | ||||
| } | ||||
| @@ -717,7 +781,7 @@ static void draw_del_popup(void) | ||||
| /* deletes contact from blocked list */ | ||||
| static void delete_blocked_friend(uint32_t bnum) | ||||
| { | ||||
|     memset(&Blocked.list[bnum], 0, sizeof(BlockedFriend)); | ||||
|     clear_blocklist_index(bnum); | ||||
|  | ||||
|     int i; | ||||
|  | ||||
| @@ -740,12 +804,12 @@ static void delete_blocked_friend(uint32_t bnum) | ||||
| /* deletes contact from friendlist and puts in blocklist */ | ||||
| void block_friend(Tox *m, uint32_t fnum) | ||||
| { | ||||
|     if (Friends.num_friends <= 0) { | ||||
|     if (Friends.num_friends == 0) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     realloc_blocklist(Blocked.max_idx + 1); | ||||
|     memset(&Blocked.list[Blocked.max_idx], 0, sizeof(BlockedFriend)); | ||||
|     clear_blocklist_index(Blocked.max_idx); | ||||
|  | ||||
|     int i; | ||||
|  | ||||
| @@ -787,35 +851,38 @@ static void unblock_friend(Tox *m, uint32_t bnum) | ||||
|     uint32_t friendnum = tox_friend_add_norequest(m, (uint8_t *) Blocked.list[bnum].pub_key, &err); | ||||
|  | ||||
|     if (err != TOX_ERR_FRIEND_ADD_OK) { | ||||
|         line_info_add(prompt, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to unblock friend (error %d)", err); | ||||
|         line_info_add(prompt, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to unblock friend (error %d)", err); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     friendlist_add_blocked(m, friendnum, bnum); | ||||
|     friendlist_add_blocked(friendnum, bnum); | ||||
|     delete_blocked_friend(bnum); | ||||
|     sort_blocklist_index(); | ||||
|     sort_friendlist_index(); | ||||
| } | ||||
|  | ||||
| static void friendlist_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr) | ||||
| /* | ||||
|  * Return true if input is recognized by handler | ||||
|  */ | ||||
| static bool friendlist_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr) | ||||
| { | ||||
|  | ||||
|     if (self->help->active) { | ||||
|         help_onKey(self, key); | ||||
|         return; | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     if (key == 'h') { | ||||
|     if (key == L'h') { | ||||
|         help_init_menu(self); | ||||
|         return; | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     if (!blocklist_view && !Friends.num_friends && (key != KEY_RIGHT && key != KEY_LEFT)) { | ||||
|         return; | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     if (blocklist_view && !Blocked.num_blocked && (key != KEY_RIGHT && key != KEY_LEFT)) { | ||||
|         return; | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     int f = 0; | ||||
| @@ -828,19 +895,19 @@ static void friendlist_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr) | ||||
|  | ||||
|     /* lock screen and force decision on deletion popup */ | ||||
|     if (PendingDelete.active) { | ||||
|         if (key == 'y' || key == 'n') { | ||||
|             del_friend_deactivate(self, m, key); | ||||
|         if (key == L'y' || key == L'n') { | ||||
|             del_friend_deactivate(m, key); | ||||
|         } | ||||
|  | ||||
|         return; | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     if (key == ltr) { | ||||
|         return; | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     switch (key) { | ||||
|         case '\r': | ||||
|         case L'\r': | ||||
|             if (blocklist_view) { | ||||
|                 break; | ||||
|             } | ||||
| @@ -853,17 +920,17 @@ static void friendlist_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr) | ||||
|                 set_active_window_index(Friends.list[f].chatwin); | ||||
|             } else { | ||||
|                 const char *msg = "* Warning: Too many windows are open."; | ||||
|                 line_info_add(prompt, NULL, NULL, NULL, SYS_MSG, 0, RED, msg); | ||||
|                 line_info_add(prompt, false, NULL, NULL, SYS_MSG, 0, RED, msg); | ||||
|                 sound_notify(prompt, notif_error, NT_WNDALERT_1, NULL); | ||||
|             } | ||||
|  | ||||
|             break; | ||||
|  | ||||
|         case KEY_DC: | ||||
|             del_friend_activate(self, m, f); | ||||
|             del_friend_activate(f); | ||||
|             break; | ||||
|  | ||||
|         case 'b': | ||||
|         case L'b': | ||||
|             if (!blocklist_view) { | ||||
|                 block_friend(m, f); | ||||
|             } else { | ||||
| @@ -879,19 +946,23 @@ static void friendlist_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr) | ||||
|  | ||||
|         default: | ||||
|             if (blocklist_view == 0) { | ||||
|                 select_friend(self, key, &Friends.num_selected, Friends.num_friends); | ||||
|                 select_friend(key, &Friends.num_selected, Friends.num_friends); | ||||
|             } else { | ||||
|                 select_friend(self, key, &Blocked.num_selected, Blocked.num_blocked); | ||||
|                 select_friend(key, &Blocked.num_selected, Blocked.num_blocked); | ||||
|             } | ||||
|  | ||||
|             break; | ||||
|     } | ||||
|  | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| #define FLIST_OFST 6    /* Accounts for space at top and bottom */ | ||||
|  | ||||
| static void blocklist_onDraw(ToxWindow *self, Tox *m, int y2, int x2) | ||||
| { | ||||
|     UNUSED_VAR(m); | ||||
|  | ||||
|     wattron(self->window, A_BOLD); | ||||
|     wprintw(self->window, " Blocked: "); | ||||
|     wattroff(self->window, A_BOLD); | ||||
| @@ -948,7 +1019,7 @@ static void blocklist_onDraw(ToxWindow *self, Tox *m, int y2, int x2) | ||||
|         wmove(self->window, y2 - 1, 1); | ||||
|  | ||||
|         wattron(self->window, A_BOLD); | ||||
|         wprintw(self->window, "Key: "); | ||||
|         wprintw(self->window, "Public key: "); | ||||
|         wattroff(self->window, A_BOLD); | ||||
|  | ||||
|         int i; | ||||
| @@ -983,6 +1054,8 @@ static void friendlist_onDraw(ToxWindow *self, Tox *m) | ||||
|     wprintw(self->window, "key for help\n\n"); | ||||
|     wattroff(self->window, COLOR_PAIR(CYAN)); | ||||
|  | ||||
|     draw_window_bar(self); | ||||
|  | ||||
|     if (blocklist_view == 1) { | ||||
|         blocklist_onDraw(self, m, y2, x2); | ||||
|         return; | ||||
| @@ -995,7 +1068,7 @@ static void friendlist_onDraw(ToxWindow *self, Tox *m) | ||||
|     wprintw(self->window, " Online: "); | ||||
|     wattroff(self->window, A_BOLD); | ||||
|  | ||||
|     wprintw(self->window, "%d/%d \n\n", Friends.num_online, Friends.num_friends); | ||||
|     wprintw(self->window, "%zu/%zu \n\n", Friends.num_online, Friends.num_friends); | ||||
|  | ||||
|     if ((y2 - FLIST_OFST) <= 0) { | ||||
|         return; | ||||
| @@ -1024,9 +1097,9 @@ static void friendlist_onDraw(ToxWindow *self, Tox *m) | ||||
|         int num_selected = Friends.num_selected; | ||||
|         pthread_mutex_unlock(&Winthread.lock); | ||||
|  | ||||
|         if (is_active) { | ||||
|             bool f_selected = false; | ||||
|  | ||||
|         if (is_active) { | ||||
|             if (i == num_selected) { | ||||
|                 wattron(self->window, A_BOLD); | ||||
|                 wprintw(self->window, " > "); | ||||
| @@ -1173,7 +1246,7 @@ static void friendlist_onDraw(ToxWindow *self, Tox *m) | ||||
|         wmove(self->window, y2 - 1, 1); | ||||
|  | ||||
|         wattron(self->window, A_BOLD); | ||||
|         wprintw(self->window, "Key: "); | ||||
|         wprintw(self->window, "Public key: "); | ||||
|         wattroff(self->window, A_BOLD); | ||||
|  | ||||
|         int i; | ||||
| @@ -1191,6 +1264,21 @@ static void friendlist_onDraw(ToxWindow *self, Tox *m) | ||||
|     } | ||||
| } | ||||
|  | ||||
| void friendlist_onInit(ToxWindow *self, Tox *m) | ||||
| { | ||||
|     UNUSED_VAR(m); | ||||
|  | ||||
|     int x2; | ||||
|     int y2; | ||||
|     getmaxyx(self->window, y2, x2); | ||||
|  | ||||
|     if (y2 <= 0 || x2 <= 0) { | ||||
|         exit_toxic_err("failed in friendlist_onInit", FATALERR_CURSES); | ||||
|     } | ||||
|  | ||||
|     self->window_bar = subwin(self->window, WINDOW_BAR_HEIGHT, x2, y2 - 2, 0); | ||||
| } | ||||
|  | ||||
| void disable_chatwin(uint32_t f_num) | ||||
| { | ||||
|     Friends.list[f_num].chatwin = -1; | ||||
| @@ -1199,13 +1287,12 @@ void disable_chatwin(uint32_t f_num) | ||||
| #ifdef AUDIO | ||||
| static void friendlist_onAV(ToxWindow *self, ToxAV *av, uint32_t friend_number, int state) | ||||
| { | ||||
|     assert(0); | ||||
|     UNUSED_VAR(self); | ||||
|  | ||||
|     if (friend_number >= Friends.max_idx) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     assert(0); | ||||
|     Tox *m = toxav_get_tox(av); | ||||
|  | ||||
|     if (Friends.list[friend_number].chatwin == -1) { | ||||
| @@ -1217,10 +1304,10 @@ static void friendlist_onAV(ToxWindow *self, ToxAV *av, uint32_t friend_number, | ||||
|         } else { | ||||
|             char nick[TOX_MAX_NAME_LENGTH]; | ||||
|             get_nick_truncate(m, nick, Friends.list[friend_number].num); | ||||
|             line_info_add(prompt, NULL, NULL, NULL, SYS_MSG, 0, 0, "Audio action from: %s!", nick); | ||||
|             line_info_add(prompt, false, NULL, NULL, SYS_MSG, 0, 0, "Audio action from: %s!", nick); | ||||
|  | ||||
|             const char *errmsg = "* Warning: Too many windows are open."; | ||||
|             line_info_add(prompt, NULL, NULL, NULL, SYS_MSG, 0, RED, errmsg); | ||||
|             line_info_add(prompt, false, NULL, NULL, SYS_MSG, 0, RED, errmsg); | ||||
|  | ||||
|             sound_notify(prompt, notif_error, NT_WNDALERT_1, NULL); | ||||
|         } | ||||
| @@ -1240,6 +1327,26 @@ Tox_Connection get_friend_connection_status(uint32_t friendnumber) | ||||
|     return Friends.list[friendnumber].connection_status; | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Returns true if friend associated with `public_key` is in the block list. | ||||
|  * | ||||
|  * `public_key` must be at least TOX_PUBLIC_KEY_SIZE bytes. | ||||
|  */ | ||||
| bool friend_is_blocked(const char *public_key) | ||||
| { | ||||
|     for (size_t i = 0; i < Blocked.max_idx; ++i) { | ||||
|         if (!Blocked.list[i].active) { | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         if (memcmp(public_key, Blocked.list[i].pub_key, TOX_PUBLIC_KEY_SIZE) == 0) { | ||||
|             return true; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| ToxWindow *new_friendlist(void) | ||||
| { | ||||
|     ToxWindow *ret = calloc(1, sizeof(ToxWindow)); | ||||
| @@ -1248,8 +1355,9 @@ ToxWindow *new_friendlist(void) | ||||
|         exit_toxic_err("failed in new_friendlist", FATALERR_MEMORY); | ||||
|     } | ||||
|  | ||||
|     ret->is_friendlist = true; | ||||
|     ret->type = WINDOW_TYPE_FRIEND_LIST; | ||||
|  | ||||
|     ret->onInit = &friendlist_onInit; | ||||
|     ret->onKey = &friendlist_onKey; | ||||
|     ret->onDraw = &friendlist_onDraw; | ||||
|     ret->onFriendAdded = &friendlist_onFriendAdded; | ||||
| @@ -1259,7 +1367,7 @@ ToxWindow *new_friendlist(void) | ||||
|     ret->onStatusChange = &friendlist_onStatusChange; | ||||
|     ret->onStatusMessageChange = &friendlist_onStatusMessageChange; | ||||
|     ret->onFileRecv = &friendlist_onFileRecv; | ||||
|     ret->onGroupInvite = &friendlist_onGroupInvite; | ||||
|     ret->onConferenceInvite = &friendlist_onConferenceInvite; | ||||
|  | ||||
| #ifdef AUDIO | ||||
|     ret->onInvite = &friendlist_onAV; | ||||
| @@ -1273,7 +1381,6 @@ ToxWindow *new_friendlist(void) | ||||
|     ret->onEnd = &friendlist_onAV; | ||||
|  | ||||
|     ret->is_call = false; | ||||
|     ret->device_selection[0] = ret->device_selection[1] = -1; | ||||
| #endif /* AUDIO */ | ||||
|  | ||||
|     ret->num = -1; | ||||
| @@ -1286,6 +1393,6 @@ ToxWindow *new_friendlist(void) | ||||
|     } | ||||
|  | ||||
|     ret->help = help; | ||||
|     strcpy(ret->name, "contacts"); | ||||
|     strcpy(ret->name, "Contacts"); | ||||
|     return ret; | ||||
| } | ||||
|   | ||||
| @@ -25,9 +25,9 @@ | ||||
|  | ||||
| #include <time.h> | ||||
|  | ||||
| #include "file_transfers.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "file_transfers.h" | ||||
|  | ||||
| struct LastOnline { | ||||
|     uint64_t last_on; | ||||
| @@ -35,7 +35,7 @@ struct LastOnline { | ||||
|     char hour_min_str[TIME_STR_SIZE];    /* holds 12/24-hour time string e.g. "10:43 PM" */ | ||||
| }; | ||||
|  | ||||
| struct GroupChatInvite { | ||||
| struct ConferenceInvite { | ||||
|     char *key; | ||||
|     uint16_t length; | ||||
|     uint8_t type; | ||||
| @@ -57,7 +57,7 @@ typedef struct { | ||||
|     Tox_User_Status status; | ||||
|  | ||||
|     struct LastOnline last_online; | ||||
|     struct GroupChatInvite group_invite; | ||||
|     struct ConferenceInvite conference_invite; | ||||
|  | ||||
|     struct FileTransfer file_receiver[MAX_FILES]; | ||||
|     struct FileTransfer file_sender[MAX_FILES]; | ||||
| @@ -82,6 +82,7 @@ typedef struct { | ||||
| } FriendsList; | ||||
|  | ||||
| ToxWindow *new_friendlist(void); | ||||
| void friendlist_onInit(ToxWindow *self, Tox *m); | ||||
| void disable_chatwin(uint32_t f_num); | ||||
| int get_friendnum(uint8_t *name); | ||||
| int load_blocklist(char *data); | ||||
| @@ -93,4 +94,11 @@ Tox_Connection get_friend_connection_status(uint32_t friendnumber); | ||||
| /* sorts friendlist_index first by connection status then alphabetically */ | ||||
| void sort_friendlist_index(void); | ||||
|  | ||||
| /* | ||||
|  * Returns true if friend associated with `public_key` is in the block list. | ||||
|  * | ||||
|  * `public_key` must be at least TOX_PUBLIC_KEY_SIZE bytes. | ||||
|  */ | ||||
| bool friend_is_blocked(const char *public_key); | ||||
|  | ||||
| #endif /* end of include guard: FRIENDLIST_H */ | ||||
|   | ||||
| @@ -23,42 +23,45 @@ | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
|  | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "misc_tools.h" | ||||
| #include "friendlist.h" | ||||
| #include "log.h" | ||||
| #include "line_info.h" | ||||
| #include "groupchat.h" | ||||
| #include "prompt.h" | ||||
| #include "help.h" | ||||
| #include "term_mplex.h" | ||||
| #include "avatars.h" | ||||
| #include "conference.h" | ||||
| #include "friendlist.h" | ||||
| #include "help.h" | ||||
| #include "line_info.h" | ||||
| #include "log.h" | ||||
| #include "misc_tools.h" | ||||
| #include "name_lookup.h" | ||||
| #include "prompt.h" | ||||
| #include "qr_code.h" | ||||
| #include "term_mplex.h" | ||||
| #include "toxic.h" | ||||
| #include "toxic_strings.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| extern char *DATA_FILE; | ||||
| extern ToxWindow *prompt; | ||||
| extern FriendsList Friends; | ||||
| extern FriendRequests FrndRequests; | ||||
|  | ||||
| /* command functions */ | ||||
| void cmd_accept(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|  | ||||
|     if (argc < 1) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Request ID required."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Request ID required."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     long int req = strtol(argv[1], NULL, 10); | ||||
|  | ||||
|     if ((req == 0 && strcmp(argv[1], "0")) || req < 0 || req >= MAX_FRIEND_REQUESTS) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "No pending friend request with that ID."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "No pending friend request with that ID."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (!FrndRequests.request[req].active) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "No pending friend request with that ID."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "No pending friend request with that ID."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -66,14 +69,16 @@ void cmd_accept(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[ | ||||
|     uint32_t friendnum = tox_friend_add_norequest(m, FrndRequests.request[req].key, &err); | ||||
|  | ||||
|     if (err != TOX_ERR_FRIEND_ADD_OK) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to add friend (error %d\n)", err); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to add friend (error %d\n)", err); | ||||
|         return; | ||||
|     } else { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Friend request accepted."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Friend request accepted."); | ||||
|         on_friend_added(m, friendnum, true); | ||||
|     } | ||||
|  | ||||
|     memset(&FrndRequests.request[req], 0, sizeof(struct friend_request)); | ||||
|     FrndRequests.request[req] = (struct friend_request) { | ||||
|         0 | ||||
|     }; | ||||
|  | ||||
|     int i; | ||||
|  | ||||
| @@ -85,7 +90,6 @@ void cmd_accept(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[ | ||||
|  | ||||
|     FrndRequests.max_idx = i; | ||||
|     --FrndRequests.num_requests; | ||||
|  | ||||
| } | ||||
|  | ||||
| void cmd_add_helper(ToxWindow *self, Tox *m, const char *id_bin, const char *msg) | ||||
| @@ -133,17 +137,19 @@ void cmd_add_helper(ToxWindow *self, Tox *m, const char *id_bin, const char *msg | ||||
|  | ||||
|         /* fallthrough */ | ||||
|         default: | ||||
|             errmsg = "Faile to add friend: Unknown error."; | ||||
|             errmsg = "Failed to add friend: Unknown error."; | ||||
|             break; | ||||
|     } | ||||
|  | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, errmsg); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, errmsg); | ||||
| } | ||||
|  | ||||
| void cmd_add(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|  | ||||
|     if (argc < 1) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Tox ID or address required."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Tox ID or address required."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -152,7 +158,7 @@ void cmd_add(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX | ||||
|  | ||||
|     if (argc > 1) { | ||||
|         if (argv[2][0] != '\"') { | ||||
|             line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Message must be enclosed in quotes."); | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Message must be enclosed in quotes."); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
| @@ -186,13 +192,18 @@ void cmd_add(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX | ||||
|             xx[2] = '\0'; | ||||
|  | ||||
|             if (sscanf(xx, "%02x", &x) != 1) { | ||||
|                 line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invalid Tox ID."); | ||||
|                 line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid Tox ID."); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             id_bin[i] = x; | ||||
|         } | ||||
|  | ||||
|         if (friend_is_blocked(id_bin)) { | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Friend is in your block list."); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         cmd_add_helper(self, m, id_bin, msg); | ||||
|     } else {    /* assume id is a username@domain address and do http name server lookup */ | ||||
|         name_lookup(self, m, id_bin, id, msg); | ||||
| @@ -201,9 +212,11 @@ void cmd_add(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX | ||||
|  | ||||
| void cmd_avatar(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|  | ||||
|     if (argc != 1 || strlen(argv[1]) < 3) { | ||||
|         avatar_unset(m); | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Avatar has been unset."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Avatar has been unset."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -212,7 +225,7 @@ void cmd_avatar(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[ | ||||
|     int len = strlen(path); | ||||
|  | ||||
|     if (len <= 0) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invalid path."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid path."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -221,25 +234,31 @@ void cmd_avatar(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[ | ||||
|     get_file_name(filename, sizeof(filename), path); | ||||
|  | ||||
|     if (avatar_set(m, path, len) == -1) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, | ||||
|                       "Failed to set avatar. Avatars must be in PNG format and may not exceed %d bytes.", | ||||
|                       MAX_AVATAR_FILE_SIZE); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Avatar set to '%s'", filename); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Avatar set to '%s'", filename); | ||||
| } | ||||
|  | ||||
| void cmd_clear(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(m); | ||||
|     UNUSED_VAR(argc); | ||||
|     UNUSED_VAR(argv); | ||||
|  | ||||
|     line_info_clear(self->chatwin->hst); | ||||
|     force_refresh(window); | ||||
| } | ||||
|  | ||||
| void cmd_connect(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|  | ||||
|     if (argc != 3) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Require: <ip> <port> <key>"); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Require: <ip> <port> <key>"); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -250,14 +269,14 @@ void cmd_connect(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv) | ||||
|     long int port = strtol(port_str, NULL, 10); | ||||
|  | ||||
|     if (port <= 0 || port > MAX_PORT_RANGE) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invalid port."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid port."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     char key_binary[TOX_PUBLIC_KEY_SIZE * 2 + 1]; | ||||
|  | ||||
|     if (hex_string_to_bin(ascii_key, strlen(ascii_key), key_binary, TOX_PUBLIC_KEY_SIZE) == -1) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invalid key."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid key."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -267,15 +286,15 @@ void cmd_connect(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv) | ||||
|  | ||||
|     switch (err) { | ||||
|         case TOX_ERR_BOOTSTRAP_BAD_HOST: | ||||
|             line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Bootstrap failed: Invalid IP."); | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Bootstrap failed: Invalid IP."); | ||||
|             break; | ||||
|  | ||||
|         case TOX_ERR_BOOTSTRAP_BAD_PORT: | ||||
|             line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Bootstrap failed: Invalid port."); | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Bootstrap failed: Invalid port."); | ||||
|             break; | ||||
|  | ||||
|         case TOX_ERR_BOOTSTRAP_NULL: | ||||
|             line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Bootstrap failed."); | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Bootstrap failed."); | ||||
|             break; | ||||
|  | ||||
|         default: | ||||
| @@ -285,24 +304,29 @@ void cmd_connect(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv) | ||||
|  | ||||
| void cmd_decline(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|     UNUSED_VAR(m); | ||||
|  | ||||
|     if (argc < 1) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Request ID required."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Request ID required."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     long int req = strtol(argv[1], NULL, 10); | ||||
|  | ||||
|     if ((req == 0 && strcmp(argv[1], "0")) || req < 0 || req >= MAX_FRIEND_REQUESTS) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "No pending friend request with that ID."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "No pending friend request with that ID."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (!FrndRequests.request[req].active) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "No pending friend request with that ID."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "No pending friend request with that ID."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     memset(&FrndRequests.request[req], 0, sizeof(struct friend_request)); | ||||
|     FrndRequests.request[req] = (struct friend_request) { | ||||
|         0 | ||||
|     }; | ||||
|  | ||||
|     int i; | ||||
|  | ||||
| @@ -316,15 +340,17 @@ void cmd_decline(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv) | ||||
|     --FrndRequests.num_requests; | ||||
| } | ||||
|  | ||||
| void cmd_groupchat(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| void cmd_conference(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|  | ||||
|     if (get_num_active_windows() >= MAX_WINDOWS_NUM) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, RED, " * Warning: Too many windows are open."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, RED, " * Warning: Too many windows are open."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (argc < 1) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Please specify group type: text | audio"); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Please specify conference type: text | audio"); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -335,35 +361,59 @@ void cmd_groupchat(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*arg | ||||
|     } else if (!strcasecmp(argv[1], "text")) { | ||||
|         type = TOX_CONFERENCE_TYPE_TEXT; | ||||
|     } else { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Valid group types are: text | audio"); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Valid conference types are: text | audio"); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (type != TOX_CONFERENCE_TYPE_TEXT) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Toxic does not support audio groups."); | ||||
|         return; | ||||
|     } | ||||
|     uint32_t conferencenum = 0; | ||||
|  | ||||
|     if (type == TOX_CONFERENCE_TYPE_TEXT) { | ||||
|         Tox_Err_Conference_New err; | ||||
|  | ||||
|     uint32_t groupnum = tox_conference_new(m, &err); | ||||
|         conferencenum = tox_conference_new(m, &err); | ||||
|  | ||||
|         if (err != TOX_ERR_CONFERENCE_NEW_OK) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Group chat instance failed to initialize (error %d)", err); | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Conference instance failed to initialize (error %d)", err); | ||||
|             return; | ||||
|         } | ||||
|     } else if (type == TOX_CONFERENCE_TYPE_AV) { | ||||
| #ifdef AUDIO | ||||
|         conferencenum = toxav_add_av_groupchat(m, audio_conference_callback, NULL); | ||||
|  | ||||
|         if (conferencenum == (uint32_t) -1) { | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Audio conference instance failed to initialize"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|     if (init_groupchat_win(prompt, m, groupnum, type) == -1) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Group chat window failed to initialize."); | ||||
|         tox_conference_delete(m, groupnum, NULL); | ||||
| #else | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Audio support disabled by compile-time option."); | ||||
|         return; | ||||
| #endif | ||||
|     } | ||||
|  | ||||
|     if (init_conference_win(m, conferencenum, type, NULL, 0) == -1) { | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Conference window failed to initialize."); | ||||
|         tox_conference_delete(m, conferencenum, NULL); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Group chat [%d] created.", groupnum); | ||||
| #ifdef AUDIO | ||||
|  | ||||
|     if (type == TOX_CONFERENCE_TYPE_AV) { | ||||
|         if (!init_conference_audio_input(m, conferencenum)) { | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Audio capture failed; use \"/audio on\" to try again."); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| #endif | ||||
|  | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Conference [%d] created.", conferencenum); | ||||
| } | ||||
|  | ||||
| void cmd_log(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|  | ||||
|     const char *msg; | ||||
|     struct chatlog *log = self->chatwin->log; | ||||
|  | ||||
| @@ -374,69 +424,61 @@ void cmd_log(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX | ||||
|             msg = "Logging for this window is OFF; type \"/log on\" to enable."; | ||||
|         } | ||||
|  | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, msg); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, msg); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     const char *swch = argv[1]; | ||||
|  | ||||
|     if (!strcmp(swch, "1") || !strcmp(swch, "on")) { | ||||
|         char myid[TOX_ADDRESS_SIZE]; | ||||
|         tox_self_get_address(m, (uint8_t *) myid); | ||||
|  | ||||
|         int log_ret = -1; | ||||
|  | ||||
|         if (self->is_chat) { | ||||
|             Friends.list[self->num].logging_on = true; | ||||
|             log_ret = log_enable(self->name, myid, Friends.list[self->num].pub_key, log, LOG_CHAT); | ||||
|         } else if (self->is_prompt) { | ||||
|             log_ret = log_enable(self->name, myid, NULL, log, LOG_PROMPT); | ||||
|         } else if (self->is_groupchat) { | ||||
|             log_ret = log_enable(self->name, myid, NULL, log, LOG_GROUP); | ||||
|         } | ||||
|  | ||||
|         msg = log_ret == 0 ? "Logging enabled." : "Warning: Log failed to initialize."; | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, msg); | ||||
|         msg = log_enable(log) == 0 ? "Logging enabled." : "Warning: Failed to enable log."; | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, msg); | ||||
|         return; | ||||
|     } else if (!strcmp(swch, "0") || !strcmp(swch, "off")) { | ||||
|         if (self->is_chat) { | ||||
|         if (self->type == WINDOW_TYPE_CHAT) { | ||||
|             Friends.list[self->num].logging_on = false; | ||||
|         } | ||||
|  | ||||
|         log_disable(log); | ||||
|  | ||||
|         msg = "Logging disabled."; | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, msg); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, msg); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     msg = "Invalid option. Use \"/log on\" and \"/log off\" to toggle logging."; | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, msg); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, msg); | ||||
| } | ||||
|  | ||||
| void cmd_myid(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|     UNUSED_VAR(argc); | ||||
|     UNUSED_VAR(argv); | ||||
|  | ||||
|     char id_string[TOX_ADDRESS_SIZE * 2 + 1]; | ||||
|     char bin_id[TOX_ADDRESS_SIZE]; | ||||
|     tox_self_get_address(m, (uint8_t *) bin_id); | ||||
|  | ||||
|     if (bin_id_to_string(bin_id, sizeof(bin_id), id_string, sizeof(id_string)) == -1) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to print ID."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to print ID."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "%s", id_string); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "%s", id_string); | ||||
| } | ||||
|  | ||||
| #ifdef QRCODE | ||||
| void cmd_myqr(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|  | ||||
|     char id_string[TOX_ADDRESS_SIZE * 2 + 1]; | ||||
|     char bin_id[TOX_ADDRESS_SIZE]; | ||||
|     tox_self_get_address(m, (uint8_t *) bin_id); | ||||
|  | ||||
|     if (bin_id_to_string(bin_id, sizeof(bin_id), id_string, sizeof(id_string)) == -1) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to create QR code."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to create QR code."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -446,52 +488,88 @@ void cmd_myqr(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MA | ||||
|     nick[nick_len] = '\0'; | ||||
|  | ||||
|     size_t data_file_len = strlen(DATA_FILE); | ||||
|     char dir[data_file_len + 1]; | ||||
|     char *dir = malloc(data_file_len + 1); | ||||
|  | ||||
|     if (dir == NULL) { | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to create QR code: Out of memory."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     size_t dir_len = get_base_dir(DATA_FILE, data_file_len, dir); | ||||
|  | ||||
| #ifdef QRPNG | ||||
|  | ||||
|     if (argc == 0) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Required 'txt' or 'png'"); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Required 'txt' or 'png'"); | ||||
|         free(dir); | ||||
|         return; | ||||
|     } else if (!strcmp(argv[1], "txt")) { | ||||
|  | ||||
| #endif /* QRPNG */ | ||||
|         char qr_path[dir_len + nick_len + strlen(QRCODE_FILENAME_EXT) + 1]; | ||||
|         snprintf(qr_path, sizeof(qr_path), "%s%s%s", dir, nick, QRCODE_FILENAME_EXT); | ||||
|         size_t qr_path_buf_size = dir_len + nick_len + sizeof(QRCODE_FILENAME_EXT); | ||||
|         char *qr_path = malloc(qr_path_buf_size); | ||||
|  | ||||
|         if (ID_to_QRcode_txt(id_string, qr_path) == -1) { | ||||
|             line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to create QR code."); | ||||
|         if (qr_path == NULL) { | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to create QR code: Out of memory"); | ||||
|             free(dir); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "QR code has been printed to the file '%s'", qr_path); | ||||
|         snprintf(qr_path, qr_path_buf_size, "%s%s%s", dir, nick, QRCODE_FILENAME_EXT); | ||||
|  | ||||
|         if (ID_to_QRcode_txt(id_string, qr_path) == -1) { | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to create QR code."); | ||||
|             free(dir); | ||||
|             free(qr_path); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "QR code has been printed to the file '%s'", qr_path); | ||||
|  | ||||
|         free(qr_path); | ||||
|  | ||||
| #ifdef QRPNG | ||||
|     } else if (!strcmp(argv[1], "png")) { | ||||
|         char qr_path[dir_len + nick_len + strlen(QRCODE_FILENAME_EXT_PNG) + 1]; | ||||
|         snprintf(qr_path, sizeof(qr_path), "%s%s%s", dir, nick, QRCODE_FILENAME_EXT_PNG); | ||||
|         size_t qr_path_buf_size = dir_len + nick_len + sizeof(QRCODE_FILENAME_EXT_PNG); | ||||
|         char *qr_path = malloc(qr_path_buf_size); | ||||
|  | ||||
|         if (ID_to_QRcode_png(id_string, qr_path) == -1) { | ||||
|             line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to create QR code."); | ||||
|         if (qr_path == NULL) { | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to create QR code: Out of memory"); | ||||
|             free(dir); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "QR code has been printed to the file '%s'", qr_path); | ||||
|         snprintf(qr_path, qr_path_buf_size, "%s%s%s", dir, nick, QRCODE_FILENAME_EXT_PNG); | ||||
|  | ||||
|         if (ID_to_QRcode_png(id_string, qr_path) == -1) { | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to create QR code."); | ||||
|             free(dir); | ||||
|             free(qr_path); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "QR code has been printed to the file '%s'", qr_path); | ||||
|  | ||||
|         free(qr_path); | ||||
|  | ||||
|     } else { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Unknown option '%s' -- Required 'txt' or 'png'", argv[1]); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Unknown option '%s' -- Required 'txt' or 'png'", argv[1]); | ||||
|         free(dir); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| #endif /* QRPNG */ | ||||
|  | ||||
|     free(dir); | ||||
| } | ||||
| #endif /* QRCODE */ | ||||
|  | ||||
| void cmd_nick(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|  | ||||
|     if (argc < 1) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Input required."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Input required."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -500,7 +578,7 @@ void cmd_nick(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MA | ||||
|     size_t len = strlen(nick); | ||||
|  | ||||
|     if (!valid_nick(nick)) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invalid name."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid name."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -515,8 +593,10 @@ void cmd_nick(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MA | ||||
|  | ||||
| void cmd_note(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|  | ||||
|     if (argc < 1) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Input required."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Input required."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -531,7 +611,7 @@ void cmd_nospam(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[ | ||||
|         nospam = strtol(argv[1], NULL, 16); | ||||
|  | ||||
|         if ((nospam == 0 && strcmp(argv[1], "0")) || nospam < 0) { | ||||
|             line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invalid nospam value."); | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid nospam value."); | ||||
|             return; | ||||
|         } | ||||
|     } | ||||
| @@ -539,29 +619,44 @@ void cmd_nospam(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[ | ||||
|     uint32_t old_nospam = tox_self_get_nospam(m); | ||||
|     tox_self_set_nospam(m, (uint32_t) nospam); | ||||
|  | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Your new Tox ID is:"); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Your new Tox ID is:"); | ||||
|     cmd_myid(window, self, m, 0, NULL); | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, ""); | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, ""); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, | ||||
|                   "Any services that relied on your old ID will need to be updated manually."); | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "If you ever want your old Tox ID back, type '/nospam %X'", | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "If you ever want your old Tox ID back, type '/nospam %X'", | ||||
|                   old_nospam); | ||||
| } | ||||
|  | ||||
| void cmd_prompt_help(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|     UNUSED_VAR(m); | ||||
|     UNUSED_VAR(argc); | ||||
|     UNUSED_VAR(argv); | ||||
|  | ||||
|     help_init_menu(self); | ||||
| } | ||||
|  | ||||
| void cmd_quit(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|     UNUSED_VAR(argc); | ||||
|     UNUSED_VAR(argv); | ||||
|     UNUSED_VAR(self); | ||||
|  | ||||
|     exit_toxic_success(m); | ||||
| } | ||||
|  | ||||
| void cmd_requests(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|     UNUSED_VAR(m); | ||||
|     UNUSED_VAR(argc); | ||||
|     UNUSED_VAR(argv); | ||||
|  | ||||
|     if (FrndRequests.num_requests == 0) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "No pending friend requests."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "No pending friend requests."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -581,24 +676,26 @@ void cmd_requests(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv | ||||
|             strcat(id, d); | ||||
|         } | ||||
|  | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "%d : %s", i, id); | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "%s", FrndRequests.request[i].msg); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "%d : %s", i, id); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "%s", FrndRequests.request[i].msg); | ||||
|  | ||||
|         if (++count < FrndRequests.num_requests) { | ||||
|             line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, ""); | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, ""); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| void cmd_status(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|  | ||||
|     const char *errmsg; | ||||
|  | ||||
|     lock_status(); | ||||
|  | ||||
|     if (argc < 1) { | ||||
|         errmsg = "Require a status. Statuses are: online, busy and away."; | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, errmsg); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, errmsg); | ||||
|         goto finish; | ||||
|     } | ||||
|  | ||||
| @@ -613,13 +710,13 @@ void cmd_status(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[ | ||||
|         status = TOX_USER_STATUS_BUSY; | ||||
|     } else { | ||||
|         errmsg = "Invalid status. Valid statuses are: online, busy and away."; | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, errmsg); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, errmsg); | ||||
|         goto finish; | ||||
|     } | ||||
|  | ||||
|     tox_self_set_status(m, status); | ||||
|     prompt_update_status(prompt, status); | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Your status has been changed to %s.", status_str); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Your status has been changed to %s.", status_str); | ||||
|  | ||||
|  | ||||
| finish: | ||||
|   | ||||
| @@ -23,8 +23,8 @@ | ||||
| #ifndef GLOBAL_COMMANDS_H | ||||
| #define GLOBAL_COMMANDS_H | ||||
|  | ||||
| #include "windows.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| void cmd_accept(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_add(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| @@ -32,7 +32,7 @@ void cmd_avatar(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[ | ||||
| void cmd_clear(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_connect(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_decline(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_groupchat(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_conference(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_log(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| void cmd_myid(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| #ifdef QRCODE | ||||
|   | ||||
| @@ -1,79 +0,0 @@ | ||||
| /*  group_commands.c | ||||
|  * | ||||
|  * | ||||
|  *  Copyright (C) 2014 Toxic All Rights Reserved. | ||||
|  * | ||||
|  *  This file is part of Toxic. | ||||
|  * | ||||
|  *  Toxic is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|  *  the Free Software Foundation, either version 3 of the License, or | ||||
|  *  (at your option) any later version. | ||||
|  * | ||||
|  *  Toxic is distributed in the hope that it will be useful, | ||||
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  *  GNU General Public License for more details. | ||||
|  * | ||||
|  *  You should have received a copy of the GNU General Public License | ||||
|  *  along with Toxic.  If not, see <http://www.gnu.org/licenses/>. | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <string.h> | ||||
|  | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "line_info.h" | ||||
| #include "misc_tools.h" | ||||
| #include "log.h" | ||||
|  | ||||
| void cmd_set_title(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     Tox_Err_Conference_Title err; | ||||
|     char title[MAX_STR_SIZE]; | ||||
|  | ||||
|     if (argc < 1) { | ||||
|         size_t tlen = tox_conference_get_title_size(m, self->num, &err); | ||||
|  | ||||
|         if (err != TOX_ERR_CONFERENCE_TITLE_OK) { | ||||
|             line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Title is not set"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (!tox_conference_get_title(m, self->num, (uint8_t *) title, &err)) { | ||||
|             line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Title is not set"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         title[tlen] = '\0'; | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Title is set to: %s", title); | ||||
|  | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     snprintf(title, sizeof(title), "%s", argv[1]); | ||||
|     int len = strlen(title); | ||||
|  | ||||
|     if (!tox_conference_set_title(m, self->num, (uint8_t *) title, len, &err)) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to set title (error %d)", err); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     set_window_title(self, title, len); | ||||
|  | ||||
|     char timefrmt[TIME_STR_SIZE]; | ||||
|     char selfnick[TOX_MAX_NAME_LENGTH]; | ||||
|  | ||||
|     get_time_str(timefrmt, sizeof(timefrmt)); | ||||
|  | ||||
|     tox_self_get_name(m, (uint8_t *) selfnick); | ||||
|     size_t sn_len = tox_self_get_name_size(m); | ||||
|     selfnick[sn_len] = '\0'; | ||||
|  | ||||
|     line_info_add(self, timefrmt, selfnick, NULL, NAME_CHANGE, 0, 0, " set the group title to: %s", title); | ||||
|  | ||||
|     char tmp_event[MAX_STR_SIZE]; | ||||
|     snprintf(tmp_event, sizeof(tmp_event), "set title to %s", title); | ||||
|     write_to_log(tmp_event, selfnick, self->chatwin->log, true); | ||||
| } | ||||
							
								
								
									
										738
									
								
								src/groupchat.c
									
									
									
									
									
								
							
							
						
						
									
										738
									
								
								src/groupchat.c
									
									
									
									
									
								
							| @@ -1,738 +0,0 @@ | ||||
| /*  groupchat.c | ||||
|  * | ||||
|  * | ||||
|  *  Copyright (C) 2014 Toxic All Rights Reserved. | ||||
|  * | ||||
|  *  This file is part of Toxic. | ||||
|  * | ||||
|  *  Toxic is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|  *  the Free Software Foundation, either version 3 of the License, or | ||||
|  *  (at your option) any later version. | ||||
|  * | ||||
|  *  Toxic is distributed in the hope that it will be useful, | ||||
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  *  GNU General Public License for more details. | ||||
|  * | ||||
|  *  You should have received a copy of the GNU General Public License | ||||
|  *  along with Toxic.  If not, see <http://www.gnu.org/licenses/>. | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #ifndef _GNU_SOURCE | ||||
| #define _GNU_SOURCE    /* needed for strcasestr() and wcswidth() */ | ||||
| #endif | ||||
|  | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <assert.h> | ||||
| #include <time.h> | ||||
| #include <wchar.h> | ||||
| #include <unistd.h> | ||||
|  | ||||
| #ifdef AUDIO | ||||
| #ifdef __APPLE__ | ||||
| #include <OpenAL/al.h> | ||||
| #include <OpenAL/alc.h> | ||||
| #else | ||||
| #include <AL/al.h> | ||||
| #include <AL/alc.h> | ||||
| /* compatibility with older versions of OpenAL */ | ||||
| #ifndef ALC_ALL_DEVICES_SPECIFIER | ||||
| #include <AL/alext.h> | ||||
| #endif /* ALC_ALL_DEVICES_SPECIFIER */ | ||||
| #endif /* __APPLE__ */ | ||||
| #endif /* AUDIO */ | ||||
|  | ||||
| #include "windows.h" | ||||
| #include "toxic.h" | ||||
| #include "execute.h" | ||||
| #include "misc_tools.h" | ||||
| #include "groupchat.h" | ||||
| #include "prompt.h" | ||||
| #include "toxic_strings.h" | ||||
| #include "log.h" | ||||
| #include "line_info.h" | ||||
| #include "settings.h" | ||||
| #include "input.h" | ||||
| #include "help.h" | ||||
| #include "notify.h" | ||||
| #include "autocomplete.h" | ||||
| #include "audio_device.h" | ||||
|  | ||||
| extern char *DATA_FILE; | ||||
|  | ||||
| static GroupChat groupchats[MAX_GROUPCHAT_NUM]; | ||||
| static int max_groupchat_index = 0; | ||||
|  | ||||
| extern struct user_settings *user_settings; | ||||
| extern struct Winthread Winthread; | ||||
|  | ||||
| #ifdef PYTHON | ||||
| #define AC_NUM_GROUP_COMMANDS_PYTHON 1 | ||||
| #else | ||||
| #define AC_NUM_GROUP_COMMANDS_PYTHON 0 | ||||
| #endif /* PYTHON */ | ||||
| #ifdef QRCODE | ||||
| #define AC_NUM_GROUP_COMMANDS_QRCODE 1 | ||||
| #else | ||||
| #define AC_NUM_GROUP_COMMANDS_QRCODE 0 | ||||
| #endif /* QRCODE */ | ||||
| #define AC_NUM_GROUP_COMMANDS (19 + AC_NUM_GROUP_COMMANDS_PYTHON + AC_NUM_GROUP_COMMANDS_QRCODE) | ||||
|  | ||||
| /* Array of groupchat command names used for tab completion. */ | ||||
| static const char group_cmd_list[AC_NUM_GROUP_COMMANDS][MAX_CMDNAME_SIZE] = { | ||||
|     { "/accept"     }, | ||||
|     { "/add"        }, | ||||
|     { "/avatar"     }, | ||||
|     { "/clear"      }, | ||||
|     { "/close"      }, | ||||
|     { "/connect"    }, | ||||
|     { "/decline"    }, | ||||
|     { "/exit"       }, | ||||
|     { "/group"      }, | ||||
|     { "/help"       }, | ||||
|     { "/log"        }, | ||||
|     { "/myid"       }, | ||||
| #ifdef QRCODE | ||||
|     { "/myqr"       }, | ||||
| #endif /* QRCODE */ | ||||
|     { "/nick"       }, | ||||
|     { "/note"       }, | ||||
|     { "/nospam"     }, | ||||
|     { "/quit"       }, | ||||
|     { "/requests"   }, | ||||
|     { "/status"     }, | ||||
|     { "/title"      }, | ||||
|  | ||||
| #ifdef PYTHON | ||||
|  | ||||
|     { "/run"        }, | ||||
|  | ||||
| #endif /* PYTHON */ | ||||
| }; | ||||
|  | ||||
| static void kill_groupchat_window(ToxWindow *self) | ||||
| { | ||||
|     ChatContext *ctx = self->chatwin; | ||||
|  | ||||
|     log_disable(ctx->log); | ||||
|     line_info_cleanup(ctx->hst); | ||||
|     delwin(ctx->linewin); | ||||
|     delwin(ctx->history); | ||||
|     delwin(ctx->sidebar); | ||||
|     free(ctx->log); | ||||
|     free(ctx); | ||||
|     free(self->help); | ||||
|     del_window(self); | ||||
| } | ||||
|  | ||||
| int init_groupchat_win(ToxWindow *prompt, Tox *m, uint32_t groupnum, uint8_t type) | ||||
| { | ||||
|     if (groupnum > MAX_GROUPCHAT_NUM) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     ToxWindow *self = new_group_chat(m, groupnum); | ||||
|  | ||||
|     for (int i = 0; i <= max_groupchat_index; ++i) { | ||||
|         if (!groupchats[i].active) { | ||||
|             groupchats[i].chatwin = add_window(m, self); | ||||
|             groupchats[i].active = true; | ||||
|             groupchats[i].num_peers = 0; | ||||
|             groupchats[i].type = type; | ||||
|             groupchats[i].start_time = get_unix_time(); | ||||
|  | ||||
|             set_active_window_index(groupchats[i].chatwin); | ||||
|  | ||||
|             if (i == max_groupchat_index) { | ||||
|                 ++max_groupchat_index; | ||||
|             } | ||||
|  | ||||
|             return 0; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     kill_groupchat_window(self); | ||||
|  | ||||
|     return -1; | ||||
| } | ||||
|  | ||||
| void free_groupchat(ToxWindow *self, Tox *m, uint32_t groupnum) | ||||
| { | ||||
|     free(groupchats[groupnum].name_list); | ||||
|     free(groupchats[groupnum].peer_list); | ||||
|     memset(&groupchats[groupnum], 0, sizeof(GroupChat)); | ||||
|  | ||||
|     int i; | ||||
|  | ||||
|     for (i = max_groupchat_index; i > 0; --i) { | ||||
|         if (groupchats[i - 1].active) { | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     max_groupchat_index = i; | ||||
|     kill_groupchat_window(self); | ||||
| } | ||||
|  | ||||
| static void delete_groupchat(ToxWindow *self, Tox *m, uint32_t groupnum) | ||||
| { | ||||
|     tox_conference_delete(m, groupnum, NULL); | ||||
|     free_groupchat(self, m, groupnum); | ||||
| } | ||||
|  | ||||
| /* destroys and re-creates groupchat window with or without the peerlist */ | ||||
| void redraw_groupchat_win(ToxWindow *self) | ||||
| { | ||||
|     ChatContext *ctx = self->chatwin; | ||||
|  | ||||
|     endwin(); | ||||
|     refresh(); | ||||
|     clear(); | ||||
|  | ||||
|     int x2, y2; | ||||
|     getmaxyx(stdscr, y2, x2); | ||||
|     y2 -= 2; | ||||
|  | ||||
|     if (y2 <= 0 || x2 <= 0) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (ctx->sidebar) { | ||||
|         delwin(ctx->sidebar); | ||||
|         ctx->sidebar = NULL; | ||||
|     } | ||||
|  | ||||
|     delwin(ctx->linewin); | ||||
|     delwin(ctx->history); | ||||
|     delwin(self->window); | ||||
|  | ||||
|     self->window = newwin(y2, x2, 0, 0); | ||||
|     ctx->linewin = subwin(self->window, CHATBOX_HEIGHT, x2, y2 - CHATBOX_HEIGHT, 0); | ||||
|  | ||||
|     if (self->show_peerlist) { | ||||
|         ctx->history = subwin(self->window, y2 - CHATBOX_HEIGHT + 1, x2 - SIDEBAR_WIDTH - 1, 0, 0); | ||||
|         ctx->sidebar = subwin(self->window, y2 - CHATBOX_HEIGHT + 1, SIDEBAR_WIDTH, 0, x2 - SIDEBAR_WIDTH); | ||||
|     } else { | ||||
|         ctx->history = subwin(self->window, y2 - CHATBOX_HEIGHT + 1, x2, 0, 0); | ||||
|     } | ||||
|  | ||||
|     scrollok(ctx->history, 0); | ||||
|  | ||||
| } | ||||
|  | ||||
| static void groupchat_onGroupMessage(ToxWindow *self, Tox *m, uint32_t groupnum, uint32_t peernum, | ||||
|                                      Tox_Message_Type type, const char *msg, size_t len) | ||||
| { | ||||
|     if (self->num != groupnum) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     ChatContext *ctx = self->chatwin; | ||||
|  | ||||
|     char nick[TOX_MAX_NAME_LENGTH]; | ||||
|     get_group_nick_truncate(m, nick, peernum, groupnum); | ||||
|  | ||||
|     char selfnick[TOX_MAX_NAME_LENGTH]; | ||||
|     tox_self_get_name(m, (uint8_t *) selfnick); | ||||
|  | ||||
|     size_t sn_len = tox_self_get_name_size(m); | ||||
|     selfnick[sn_len] = '\0'; | ||||
|  | ||||
|     int nick_clr = strcmp(nick, selfnick) == 0 ? GREEN : CYAN; | ||||
|  | ||||
|     /* Only play sound if mentioned by someone else */ | ||||
|     if (strcasestr(msg, selfnick) && strcmp(selfnick, nick)) { | ||||
|         sound_notify(self, generic_message, NT_WNDALERT_0 | user_settings->bell_on_message, NULL); | ||||
|  | ||||
|         if (self->active_box != -1) { | ||||
|             box_silent_notify2(self, NT_NOFOCUS, self->active_box, "%s %s", nick, msg); | ||||
|         } else { | ||||
|             box_silent_notify(self, NT_NOFOCUS, &self->active_box, self->name, "%s %s", nick, msg); | ||||
|         } | ||||
|  | ||||
|         nick_clr = RED; | ||||
|     } else { | ||||
|         sound_notify(self, silent, NT_WNDALERT_1, NULL); | ||||
|     } | ||||
|  | ||||
|     char timefrmt[TIME_STR_SIZE]; | ||||
|     get_time_str(timefrmt, sizeof(timefrmt)); | ||||
|  | ||||
|     line_info_add(self, timefrmt, nick, NULL, type == TOX_MESSAGE_TYPE_NORMAL ? IN_MSG : IN_ACTION, 0, nick_clr, "%s", msg); | ||||
|     write_to_log(msg, nick, ctx->log, false); | ||||
| } | ||||
|  | ||||
| static void groupchat_onGroupTitleChange(ToxWindow *self, Tox *m, uint32_t groupnum, uint32_t peernum, | ||||
|         const char *title, | ||||
|         size_t length) | ||||
| { | ||||
|     ChatContext *ctx = self->chatwin; | ||||
|  | ||||
|     if (self->num != groupnum) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     set_window_title(self, title, length); | ||||
|  | ||||
|     char timefrmt[TIME_STR_SIZE]; | ||||
|     get_time_str(timefrmt, sizeof(timefrmt)); | ||||
|  | ||||
|     /* don't announce title when we join the room */ | ||||
|     if (!timed_out(groupchats[self->num].start_time, GROUP_EVENT_WAIT)) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     char nick[TOX_MAX_NAME_LENGTH]; | ||||
|     get_group_nick_truncate(m, nick, peernum, groupnum); | ||||
|     line_info_add(self, timefrmt, nick, NULL, NAME_CHANGE, 0, 0, " set the group title to: %s", title); | ||||
|  | ||||
|     char tmp_event[MAX_STR_SIZE]; | ||||
|     snprintf(tmp_event, sizeof(tmp_event), "set title to %s", title); | ||||
|     write_to_log(tmp_event, nick, ctx->log, true); | ||||
| } | ||||
|  | ||||
| static void group_update_name_list(uint32_t groupnum) | ||||
| { | ||||
|     GroupChat *chat = &groupchats[groupnum]; | ||||
|  | ||||
|     if (!chat) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (chat->name_list) { | ||||
|         free(chat->name_list); | ||||
|     } | ||||
|  | ||||
|     chat->name_list = malloc(sizeof(char *) * chat->num_peers * TOX_MAX_NAME_LENGTH); | ||||
|  | ||||
|     if (chat->name_list == NULL) { | ||||
|         exit_toxic_err("failed in group_update_name_list", FATALERR_MEMORY); | ||||
|     } | ||||
|  | ||||
|     uint32_t i, count = 0; | ||||
|  | ||||
|     for (i = 0; i < chat->max_idx; ++i) { | ||||
|         if (chat->peer_list[i].active) { | ||||
|             memcpy(&chat->name_list[count * TOX_MAX_NAME_LENGTH], chat->peer_list[i].name, chat->peer_list[i].name_length + 1); | ||||
|             ++count; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     qsort(chat->name_list, count, TOX_MAX_NAME_LENGTH, qsort_strcasecmp_hlpr); | ||||
| } | ||||
|  | ||||
| /* Reallocates groupnum's peer list. Increase is true if the list needs to grow. | ||||
|  * | ||||
|  * Returns 0 on success. | ||||
|  * Returns -1 on failure. | ||||
|  */ | ||||
| static int realloc_peer_list(GroupChat *chat, uint32_t num_peers) | ||||
| { | ||||
|     if (!chat) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (num_peers == 0) { | ||||
|         free(chat->peer_list); | ||||
|         chat->peer_list = NULL; | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     struct GroupPeer *tmp_list = realloc(chat->peer_list, num_peers * sizeof(struct GroupPeer)); | ||||
|  | ||||
|     if (!tmp_list) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     chat->peer_list = tmp_list; | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| static void update_peer_list(Tox *m, uint32_t groupnum, uint32_t num_peers) | ||||
| { | ||||
|     GroupChat *chat = &groupchats[groupnum]; | ||||
|  | ||||
|     if (!chat) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     realloc_peer_list(chat, num_peers); | ||||
|  | ||||
|     uint32_t i; | ||||
|  | ||||
|     for (i = 0; i < num_peers; ++i) { | ||||
|         GroupPeer *peer = &chat->peer_list[i]; | ||||
|  | ||||
|         Tox_Err_Conference_Peer_Query err; | ||||
|         size_t length = tox_conference_peer_get_name_size(m, groupnum, i, &err); | ||||
|  | ||||
|         if (err != TOX_ERR_CONFERENCE_PEER_QUERY_OK || length >= TOX_MAX_NAME_LENGTH) { | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         tox_conference_peer_get_name(m, groupnum, i, (uint8_t *) peer->name, &err); | ||||
|         peer->name[length] = 0; | ||||
|  | ||||
|         if (err != TOX_ERR_CONFERENCE_PEER_QUERY_OK) { | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         peer->active = true; | ||||
|         peer->name_length = length; | ||||
|         peer->peernumber = i; | ||||
|     } | ||||
|  | ||||
|     group_update_name_list(groupnum); | ||||
| } | ||||
|  | ||||
| static void groupchat_onGroupNameListChange(ToxWindow *self, Tox *m, uint32_t groupnum) | ||||
| { | ||||
|     if (self->num != groupnum) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (groupnum > max_groupchat_index) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     GroupChat *chat = &groupchats[groupnum]; | ||||
|     Tox_Err_Conference_Peer_Query err; | ||||
|  | ||||
|     uint32_t num_peers = tox_conference_peer_count(m, groupnum, &err); | ||||
|     uint32_t old_num = chat->num_peers; | ||||
|  | ||||
|     if (err == TOX_ERR_CONFERENCE_PEER_QUERY_OK) { | ||||
|         chat->num_peers = num_peers; | ||||
|     } else { | ||||
|         num_peers = old_num; | ||||
|     } | ||||
|  | ||||
|     chat->max_idx = num_peers; | ||||
|     update_peer_list(m, groupnum, num_peers); | ||||
| } | ||||
|  | ||||
| static void groupchat_onGroupPeerNameChange(ToxWindow *self, Tox *m, uint32_t groupnum, uint32_t peernum, | ||||
|         const char *name, size_t length) | ||||
| { | ||||
|     if (self->num != groupnum) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     GroupChat *chat = &groupchats[groupnum]; | ||||
|  | ||||
|     if (!chat) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     size_t i; | ||||
|  | ||||
|     for (i = 0; i < chat->max_idx; ++i) { | ||||
|         GroupPeer *peer = &chat->peer_list[i]; | ||||
|  | ||||
|         // Test against default tox name to prevent nick change spam on initial join (TODO: this is disgusting) | ||||
|         if (peer->active && peer->peernumber == peernum && peer->name_length > 0) { | ||||
|             ChatContext *ctx = self->chatwin; | ||||
|             char timefrmt[TIME_STR_SIZE]; | ||||
|             get_time_str(timefrmt, sizeof(timefrmt)); | ||||
|  | ||||
|             char tmp_event[TOXIC_MAX_NAME_LENGTH * 2 + 32]; | ||||
|             snprintf(tmp_event, sizeof(tmp_event), "is now known as %s", (const char *) name); | ||||
|  | ||||
|             write_to_log(tmp_event, peer->name, ctx->log, true); | ||||
|             line_info_add(self, timefrmt, peer->name, (const char *) name, NAME_CHANGE, 0, 0, " is now known as "); | ||||
|  | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     groupchat_onGroupNameListChange(self, m, groupnum); | ||||
| } | ||||
|  | ||||
| static void send_group_action(ToxWindow *self, ChatContext *ctx, Tox *m, char *action) | ||||
| { | ||||
|     if (action == NULL) { | ||||
|         wprintw(ctx->history, "Invalid syntax.\n"); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     Tox_Err_Conference_Send_Message err; | ||||
|  | ||||
|     if (!tox_conference_send_message(m, self->num, TOX_MESSAGE_TYPE_ACTION, (uint8_t *) action, strlen(action), &err)) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, RED, " * Failed to send action (error %d)", err); | ||||
|     } | ||||
| } | ||||
|  | ||||
| static void groupchat_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr) | ||||
| { | ||||
|     ChatContext *ctx = self->chatwin; | ||||
|  | ||||
|     int x, y, y2, x2; | ||||
|     getyx(self->window, y, x); | ||||
|     getmaxyx(self->window, y2, x2); | ||||
|  | ||||
|     if (x2 <= 0 || y2 <= 0) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (self->help->active) { | ||||
|         help_onKey(self, key); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (ctx->pastemode && key == '\r') { | ||||
|         key = '\n'; | ||||
|     } | ||||
|  | ||||
|     if (ltr || key == '\n') {    /* char is printable */ | ||||
|         input_new_char(self, key, x, y, x2, y2); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (line_info_onKey(self, key)) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (input_handle(self, key, x, y, x2, y2)) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (key == '\t') {  /* TAB key: auto-completes peer name or command */ | ||||
|         if (ctx->len > 0) { | ||||
|             int diff; | ||||
|  | ||||
|             /* TODO: make this not suck */ | ||||
|             if (ctx->line[0] != L'/' || wcscmp(ctx->line, L"/me") == 0) { | ||||
|                 diff = complete_line(self, groupchats[self->num].name_list, groupchats[self->num].num_peers, | ||||
|                                      TOX_MAX_NAME_LENGTH); | ||||
|             } else if (wcsncmp(ctx->line, L"/avatar ", wcslen(L"/avatar ")) == 0) { | ||||
|                 diff = dir_match(self, m, ctx->line, L"/avatar"); | ||||
|             } | ||||
|  | ||||
| #ifdef PYTHON | ||||
|             else if (wcsncmp(ctx->line, L"/run ", wcslen(L"/run ")) == 0) { | ||||
|                 diff = dir_match(self, m, ctx->line, L"/run"); | ||||
|             } | ||||
|  | ||||
| #endif | ||||
|  | ||||
|             else { | ||||
|                 diff = complete_line(self, group_cmd_list, AC_NUM_GROUP_COMMANDS, MAX_CMDNAME_SIZE); | ||||
|             } | ||||
|  | ||||
|             if (diff != -1) { | ||||
|                 if (x + diff > x2 - 1) { | ||||
|                     int wlen = MAX(0, wcswidth(ctx->line, sizeof(ctx->line) / sizeof(wchar_t))); | ||||
|                     ctx->start = wlen < x2 ? 0 : wlen - x2 + 1; | ||||
|                 } | ||||
|             } else { | ||||
|                 sound_notify(self, notif_error, 0, NULL); | ||||
|             } | ||||
|         } else { | ||||
|             sound_notify(self, notif_error, 0, NULL); | ||||
|         } | ||||
|     } else if (key == user_settings->key_peer_list_down) {    /* Scroll peerlist up and down one position */ | ||||
|         int L = y2 - CHATBOX_HEIGHT - SDBAR_OFST; | ||||
|  | ||||
|         if (groupchats[self->num].side_pos < groupchats[self->num].num_peers - L) { | ||||
|             ++groupchats[self->num].side_pos; | ||||
|         } | ||||
|     } else if (key == user_settings->key_peer_list_up) { | ||||
|         if (groupchats[self->num].side_pos > 0) { | ||||
|             --groupchats[self->num].side_pos; | ||||
|         } | ||||
|     } else if (key == '\r') { | ||||
|         rm_trailing_spaces_buf(ctx); | ||||
|  | ||||
|         if (!wstring_is_empty(ctx->line)) { | ||||
|             add_line_to_hist(ctx); | ||||
|  | ||||
|             wstrsubst(ctx->line, L'¶', L'\n'); | ||||
|  | ||||
|             char line[MAX_STR_SIZE]; | ||||
|  | ||||
|             if (wcs_to_mbs_buf(line, ctx->line, MAX_STR_SIZE) == -1) { | ||||
|                 memset(&line, 0, sizeof(line)); | ||||
|             } | ||||
|  | ||||
|             if (line[0] == '/') { | ||||
|                 if (strcmp(line, "/close") == 0) { | ||||
|                     delete_groupchat(self, m, self->num); | ||||
|                     return; | ||||
|                 } else if (strncmp(line, "/me ", strlen("/me ")) == 0) { | ||||
|                     send_group_action(self, ctx, m, line + strlen("/me ")); | ||||
|                 } else { | ||||
|                     execute(ctx->history, self, m, line, GROUPCHAT_COMMAND_MODE); | ||||
|                 } | ||||
|             } else { | ||||
|                 Tox_Err_Conference_Send_Message err; | ||||
|  | ||||
|                 if (!tox_conference_send_message(m, self->num, TOX_MESSAGE_TYPE_NORMAL, (uint8_t *) line, strlen(line), &err)) { | ||||
|                     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, RED, " * Failed to send message (error %d)", err); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         wclear(ctx->linewin); | ||||
|         wmove(self->window, y2 - CURS_Y_OFFSET, 0); | ||||
|         reset_buf(ctx); | ||||
|     } | ||||
| } | ||||
|  | ||||
| static void groupchat_onDraw(ToxWindow *self, Tox *m) | ||||
| { | ||||
|     int x2, y2; | ||||
|     getmaxyx(self->window, y2, x2); | ||||
|  | ||||
|     if (x2 <= 0 || y2 <= 0) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     ChatContext *ctx = self->chatwin; | ||||
|  | ||||
|     pthread_mutex_lock(&Winthread.lock); | ||||
|     line_info_print(self); | ||||
|     pthread_mutex_unlock(&Winthread.lock); | ||||
|  | ||||
|     wclear(ctx->linewin); | ||||
|  | ||||
|     curs_set(1); | ||||
|  | ||||
|     if (ctx->len > 0) { | ||||
|         mvwprintw(ctx->linewin, 1, 0, "%ls", &ctx->line[ctx->start]); | ||||
|     } | ||||
|  | ||||
|     wclear(ctx->sidebar); | ||||
|     mvwhline(self->window, y2 - CHATBOX_HEIGHT, 0, ACS_HLINE, x2); | ||||
|  | ||||
|     if (self->show_peerlist) { | ||||
|         mvwvline(ctx->sidebar, 0, 0, ACS_VLINE, y2 - CHATBOX_HEIGHT); | ||||
|         mvwaddch(ctx->sidebar, y2 - CHATBOX_HEIGHT, 0, ACS_BTEE); | ||||
|  | ||||
|         pthread_mutex_lock(&Winthread.lock); | ||||
|         int num_peers = groupchats[self->num].num_peers; | ||||
|         pthread_mutex_unlock(&Winthread.lock); | ||||
|  | ||||
|         wmove(ctx->sidebar, 0, 1); | ||||
|         wattron(ctx->sidebar, A_BOLD); | ||||
|         wprintw(ctx->sidebar, "Peers: %d\n", num_peers); | ||||
|         wattroff(ctx->sidebar, A_BOLD); | ||||
|  | ||||
|         mvwaddch(ctx->sidebar, 1, 0, ACS_LTEE); | ||||
|         mvwhline(ctx->sidebar, 1, 1, ACS_HLINE, SIDEBAR_WIDTH - 1); | ||||
|  | ||||
|         int maxlines = y2 - SDBAR_OFST - CHATBOX_HEIGHT; | ||||
|         int i; | ||||
|  | ||||
|         for (i = 0; i < num_peers && i < maxlines; ++i) { | ||||
|             wmove(ctx->sidebar, i + 2, 1); | ||||
|  | ||||
|             pthread_mutex_lock(&Winthread.lock); | ||||
|             uint32_t peer = i + groupchats[self->num].side_pos; | ||||
|             pthread_mutex_unlock(&Winthread.lock); | ||||
|  | ||||
|             /* truncate nick to fit in side panel without modifying list */ | ||||
|             char tmpnck[TOX_MAX_NAME_LENGTH]; | ||||
|             int maxlen = SIDEBAR_WIDTH - 2; | ||||
|  | ||||
|             pthread_mutex_lock(&Winthread.lock); | ||||
|             memcpy(tmpnck, &groupchats[self->num].name_list[peer * TOX_MAX_NAME_LENGTH], maxlen); | ||||
|             pthread_mutex_unlock(&Winthread.lock); | ||||
|  | ||||
|             tmpnck[maxlen] = '\0'; | ||||
|  | ||||
|             wprintw(ctx->sidebar, "%s\n", tmpnck); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     int y, x; | ||||
|     getyx(self->window, y, x); | ||||
|     (void) x; | ||||
|     int new_x = ctx->start ? x2 - 1 : MAX(0, wcswidth(ctx->line, ctx->pos)); | ||||
|     wmove(self->window, y + 1, new_x); | ||||
|  | ||||
|     wnoutrefresh(self->window); | ||||
|  | ||||
|     if (self->help->active) { | ||||
|         help_onDraw(self); | ||||
|     } | ||||
| } | ||||
|  | ||||
| static void groupchat_onInit(ToxWindow *self, Tox *m) | ||||
| { | ||||
|     int x2, y2; | ||||
|     getmaxyx(self->window, y2, x2); | ||||
|  | ||||
|     if (x2 <= 0 || y2 <= 0) { | ||||
|         exit_toxic_err("failed in groupchat_onInit", FATALERR_CURSES); | ||||
|     } | ||||
|  | ||||
|     ChatContext *ctx = self->chatwin; | ||||
|  | ||||
|     ctx->history = subwin(self->window, y2 - CHATBOX_HEIGHT + 1, x2 - SIDEBAR_WIDTH - 1, 0, 0); | ||||
|     ctx->linewin = subwin(self->window, CHATBOX_HEIGHT, x2, y2 - CHATBOX_HEIGHT, 0); | ||||
|     ctx->sidebar = subwin(self->window, y2 - CHATBOX_HEIGHT + 1, SIDEBAR_WIDTH, 0, x2 - SIDEBAR_WIDTH); | ||||
|  | ||||
|     ctx->hst = calloc(1, sizeof(struct history)); | ||||
|     ctx->log = calloc(1, sizeof(struct chatlog)); | ||||
|  | ||||
|     if (ctx->log == NULL || ctx->hst == NULL) { | ||||
|         exit_toxic_err("failed in groupchat_onInit", FATALERR_MEMORY); | ||||
|     } | ||||
|  | ||||
|     line_info_init(ctx->hst); | ||||
|  | ||||
|     if (user_settings->autolog == AUTOLOG_ON) { | ||||
|         char myid[TOX_ADDRESS_SIZE]; | ||||
|         tox_self_get_address(m, (uint8_t *) myid); | ||||
|  | ||||
|         if (log_enable(self->name, myid, NULL, ctx->log, LOG_GROUP) == -1) { | ||||
|             line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Warning: Log failed to initialize."); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     execute(ctx->history, self, m, "/log", GLOBAL_COMMAND_MODE); | ||||
|  | ||||
|     scrollok(ctx->history, 0); | ||||
|     wmove(self->window, y2 - CURS_Y_OFFSET, 0); | ||||
| } | ||||
|  | ||||
| ToxWindow *new_group_chat(Tox *m, uint32_t groupnum) | ||||
| { | ||||
|     ToxWindow *ret = calloc(1, sizeof(ToxWindow)); | ||||
|  | ||||
|     if (ret == NULL) { | ||||
|         exit_toxic_err("failed in new_group_chat", FATALERR_MEMORY); | ||||
|     } | ||||
|  | ||||
|     ret->is_groupchat = true; | ||||
|  | ||||
|     ret->onKey = &groupchat_onKey; | ||||
|     ret->onDraw = &groupchat_onDraw; | ||||
|     ret->onInit = &groupchat_onInit; | ||||
|     ret->onGroupMessage = &groupchat_onGroupMessage; | ||||
|     ret->onGroupNameListChange = &groupchat_onGroupNameListChange; | ||||
|     ret->onGroupPeerNameChange = &groupchat_onGroupPeerNameChange; | ||||
|     ret->onGroupTitleChange = &groupchat_onGroupTitleChange; | ||||
|  | ||||
|     snprintf(ret->name, sizeof(ret->name), "Group %u", groupnum); | ||||
|  | ||||
|     ChatContext *chatwin = calloc(1, sizeof(ChatContext)); | ||||
|     Help *help = calloc(1, sizeof(Help)); | ||||
|  | ||||
|     if (chatwin == NULL || help == NULL) { | ||||
|         exit_toxic_err("failed in new_group_chat", FATALERR_MEMORY); | ||||
|     } | ||||
|  | ||||
|     ret->chatwin = chatwin; | ||||
|     ret->help = help; | ||||
|  | ||||
|     ret->num = groupnum; | ||||
|     ret->show_peerlist = true; | ||||
|     ret->active_box = -1; | ||||
|  | ||||
|     return ret; | ||||
| } | ||||
| @@ -1,66 +0,0 @@ | ||||
| /*  groupchat.h | ||||
|  * | ||||
|  * | ||||
|  *  Copyright (C) 2014 Toxic All Rights Reserved. | ||||
|  * | ||||
|  *  This file is part of Toxic. | ||||
|  * | ||||
|  *  Toxic is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|  *  the Free Software Foundation, either version 3 of the License, or | ||||
|  *  (at your option) any later version. | ||||
|  * | ||||
|  *  Toxic is distributed in the hope that it will be useful, | ||||
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  *  GNU General Public License for more details. | ||||
|  * | ||||
|  *  You should have received a copy of the GNU General Public License | ||||
|  *  along with Toxic.  If not, see <http://www.gnu.org/licenses/>. | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #ifndef GROUPCHAT_H | ||||
| #define GROUPCHAT_H | ||||
|  | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| #define SIDEBAR_WIDTH 16 | ||||
| #define SDBAR_OFST 2    /* Offset for the peer number box at the top of the statusbar */ | ||||
| #define MAX_GROUPCHAT_NUM MAX_WINDOWS_NUM - 2 | ||||
| #define GROUP_EVENT_WAIT 3 | ||||
|  | ||||
| typedef struct GroupPeer { | ||||
|     bool       active; | ||||
|     char       name[TOX_MAX_NAME_LENGTH]; | ||||
|     size_t     name_length; | ||||
|     uint32_t   peernumber; | ||||
| } GroupPeer; | ||||
|  | ||||
| typedef struct { | ||||
|     int chatwin; | ||||
|     bool active; | ||||
|     uint8_t type; | ||||
|     int side_pos;    /* current position of the sidebar - used for scrolling up and down */ | ||||
|     time_t start_time; | ||||
|  | ||||
|     GroupPeer *peer_list; | ||||
|     size_t max_idx; | ||||
|  | ||||
|     char *name_list; | ||||
|     size_t num_peers; | ||||
|  | ||||
| } GroupChat; | ||||
|  | ||||
| /* Frees all Toxic associated data structures for a groupchat (does not call tox_conference_delete() ) */ | ||||
| void free_groupchat(ToxWindow *self, Tox *m, uint32_t groupnum); | ||||
|  | ||||
| int init_groupchat_win(ToxWindow *prompt, Tox *m, uint32_t groupnum, uint8_t type); | ||||
|  | ||||
| /* destroys and re-creates groupchat window with or without the peerlist */ | ||||
| void redraw_groupchat_win(ToxWindow *self); | ||||
|  | ||||
| ToxWindow *new_group_chat(Tox *m, uint32_t groupnum); | ||||
|  | ||||
| #endif /* GROUPCHAT_H */ | ||||
							
								
								
									
										78
									
								
								src/help.c
									
									
									
									
									
								
							
							
						
						
									
										78
									
								
								src/help.c
									
									
									
									
									
								
							| @@ -22,10 +22,10 @@ | ||||
|  | ||||
| #include <string.h> | ||||
|  | ||||
| #include "windows.h" | ||||
| #include "toxic.h" | ||||
| #include "help.h" | ||||
| #include "misc_tools.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| #ifdef PYTHON | ||||
| #include "api.h" | ||||
| @@ -59,7 +59,10 @@ void help_init_menu(ToxWindow *self) | ||||
| static void help_exit(ToxWindow *self) | ||||
| { | ||||
|     delwin(self->help->win); | ||||
|     memset(self->help, 0, sizeof(Help)); | ||||
|  | ||||
|     *(self->help) = (struct Help) { | ||||
|         0 | ||||
|     }; | ||||
| } | ||||
|  | ||||
| static void help_init_window(ToxWindow *self, int height, int width) | ||||
| @@ -101,11 +104,11 @@ static void help_draw_menu(ToxWindow *self) | ||||
|     wattroff(win, A_BOLD | COLOR_PAIR(BLUE)); | ||||
|     wprintw(win, "hat commands\n"); | ||||
|  | ||||
|     wprintw(win, " g"); | ||||
|     wprintw(win, " c"); | ||||
|     wattron(win, A_BOLD | COLOR_PAIR(BLUE)); | ||||
|     wprintw(win, "r"); | ||||
|     wprintw(win, "o"); | ||||
|     wattroff(win, A_BOLD | COLOR_PAIR(BLUE)); | ||||
|     wprintw(win, "oup commands\n"); | ||||
|     wprintw(win, "nference commands\n"); | ||||
|  | ||||
| #ifdef PYTHON | ||||
|     wattron(win, A_BOLD | COLOR_PAIR(BLUE)); | ||||
| @@ -138,7 +141,8 @@ static void help_draw_bottom_menu(WINDOW *win) | ||||
| { | ||||
|     int y2, x2; | ||||
|     getmaxyx(win, y2, x2); | ||||
|     (void) x2; | ||||
|  | ||||
|     UNUSED_VAR(x2); | ||||
|  | ||||
|     wmove(win, y2 - 2, 1); | ||||
|  | ||||
| @@ -175,7 +179,7 @@ static void help_draw_global(ToxWindow *self) | ||||
|     wprintw(win, "  /nick <nick>               : Set your nickname\n"); | ||||
|     wprintw(win, "  /nospam <value>            : Change part of your Tox ID to stop spam\n"); | ||||
|     wprintw(win, "  /log <on> or <off>         : Enable/disable logging\n"); | ||||
|     wprintw(win, "  /group <type>              : Create a group chat where type: text | audio\n"); | ||||
|     wprintw(win, "  /conference <type>         : Create a conference where type: text | audio\n"); | ||||
|     wprintw(win, "  /myid                      : Print your Tox ID\n"); | ||||
| #ifdef QRCODE | ||||
| #ifdef QRPNG | ||||
| @@ -230,8 +234,8 @@ static void help_draw_chat(ToxWindow *self) | ||||
|     wprintw(win, "Chat Commands:\n"); | ||||
|     wattroff(win, A_BOLD | COLOR_PAIR(RED)); | ||||
|  | ||||
|     wprintw(win, "  /invite <n>                : Invite contact to a group chat\n"); | ||||
|     wprintw(win, "  /join                      : Join a pending group chat\n"); | ||||
|     wprintw(win, "  /invite <n>                : Invite contact to a conference \n"); | ||||
|     wprintw(win, "  /join                      : Join a pending conference\n"); | ||||
|     wprintw(win, "  /sendfile <path>           : Send a file\n"); | ||||
|     wprintw(win, "  /savefile <id>             : Receive a file\n"); | ||||
|     wprintw(win, "  /cancel <type> <id>        : Cancel file transfer where type: in|out\n"); | ||||
| @@ -255,7 +259,9 @@ static void help_draw_chat(ToxWindow *self) | ||||
|     wattron(win, A_BOLD); | ||||
|     wprintw(win, "\n Video:\n"); | ||||
|     wattroff(win, A_BOLD); | ||||
|     wprintw(win, "  /video                     : Toggle video call\n"); | ||||
|     wprintw(win, "  /res <width> <height>      : Set video resolution\n"); | ||||
|     wprintw(win, "  /vcall                     : Video call\n"); | ||||
|     wprintw(win, "  /video                     : Toggle video in call\n"); | ||||
| #endif /* VIDEO */ | ||||
|  | ||||
|     help_draw_bottom_menu(win); | ||||
| @@ -278,8 +284,8 @@ static void help_draw_keys(ToxWindow *self) | ||||
|     wprintw(win, "  Page Up and Page Down     : Scroll window history one line\n"); | ||||
|     wprintw(win, "  Ctrl+F and Ctrl+V         : Scroll window history half a page\n"); | ||||
|     wprintw(win, "  Ctrl+H                    : Move to the bottom of window history\n"); | ||||
|     wprintw(win, "  Ctrl+[ and Ctrl+]         : Scroll peer list in groupchats\n"); | ||||
|     wprintw(win, "  Ctrl+B                    : Toggle the groupchat peerlist\n"); | ||||
|     wprintw(win, "  Ctrl+up and Ctrl+down     : Scroll peer list in conference\n"); | ||||
|     wprintw(win, "  Ctrl+B                    : Toggle the conference peerlist\n"); | ||||
|     wprintw(win, "  Ctrl+J                    : Insert new line\n"); | ||||
|     wprintw(win, "  Ctrl+T                    : Toggle paste mode\n\n"); | ||||
|     wprintw(win, "  (Note: Custom keybindings override these defaults.)\n\n"); | ||||
| @@ -290,17 +296,27 @@ static void help_draw_keys(ToxWindow *self) | ||||
|     wnoutrefresh(win); | ||||
| } | ||||
|  | ||||
| static void help_draw_group(ToxWindow *self) | ||||
| static void help_draw_conference(ToxWindow *self) | ||||
| { | ||||
|     WINDOW *win = self->help->win; | ||||
|  | ||||
|     wmove(win, 1, 1); | ||||
|  | ||||
|     wattron(win, A_BOLD | COLOR_PAIR(RED)); | ||||
|     wprintw(win, "Group commands:\n"); | ||||
|     wprintw(win, "Conference commands:\n"); | ||||
|     wattroff(win, A_BOLD | COLOR_PAIR(RED)); | ||||
|  | ||||
|     wprintw(win, "  /title <msg>               : Set group title (show current title if no msg)\n\n"); | ||||
|     wprintw(win, "  /title <msg>               : Show/set conference title\n"); | ||||
| #ifdef AUDIO | ||||
|     wattron(win, A_BOLD); | ||||
|     wprintw(win, "\n Audio:\n"); | ||||
|     wattroff(win, A_BOLD); | ||||
|     wprintw(win, "  /audio <on> or <off>       : Enable/disable audio in an audio conference\n"); | ||||
|     wprintw(win, "  /mute                      : Toggle self audio mute status\n"); | ||||
|     wprintw(win, "  /mute <nick> or <pubkey>   : Toggle peer audio mute status\n"); | ||||
|     wprintw(win, "  /ptt <on> or <off>         : Toggle audio input Push-To-Talk (F2 to activate)\n"); | ||||
|     wprintw(win, "  /sense <n>                 : VAD sensitivity threshold\n\n"); | ||||
| #endif | ||||
|  | ||||
|     help_draw_bottom_menu(win); | ||||
|  | ||||
| @@ -355,14 +371,14 @@ void help_onKey(ToxWindow *self, wint_t key) | ||||
|     int height; | ||||
|  | ||||
|     switch (key) { | ||||
|         case 'x': | ||||
|         case L'x': | ||||
|         case T_KEY_ESC: | ||||
|             help_exit(self); | ||||
|             break; | ||||
|  | ||||
|         case 'c': | ||||
|         case L'c': | ||||
| #ifdef VIDEO | ||||
|             help_init_window(self, 23, 80); | ||||
|             help_init_window(self, 25, 80); | ||||
| #elif AUDIO | ||||
|             help_init_window(self, 20, 80); | ||||
| #else | ||||
| @@ -371,7 +387,7 @@ void help_onKey(ToxWindow *self, wint_t key) | ||||
|             self->help->type = HELP_CHAT; | ||||
|             break; | ||||
|  | ||||
|         case 'g': | ||||
|         case L'g': | ||||
|             height = 22; | ||||
| #ifdef VIDEO | ||||
|             height += 8; | ||||
| @@ -385,30 +401,34 @@ void help_onKey(ToxWindow *self, wint_t key) | ||||
|             self->help->type = HELP_GLOBAL; | ||||
|             break; | ||||
|  | ||||
|         case 'r': | ||||
|             help_init_window(self, 6, 80); | ||||
|             self->help->type = HELP_GROUP; | ||||
|         case L'o': | ||||
|             height = 6; | ||||
| #ifdef AUDIO | ||||
|             height += 7; | ||||
| #endif | ||||
|             help_init_window(self, height, 80); | ||||
|             self->help->type = HELP_CONFERENCE; | ||||
|             break; | ||||
|  | ||||
| #ifdef PYTHON | ||||
|  | ||||
|         case 'p': | ||||
|         case L'p': | ||||
|             help_init_window(self, 4 + num_registered_handlers(), help_max_width()); | ||||
|             self->help->type = HELP_PLUGIN; | ||||
|             break; | ||||
| #endif /* PYTHON */ | ||||
|  | ||||
|         case 'f': | ||||
|         case L'f': | ||||
|             help_init_window(self, 10, 80); | ||||
|             self->help->type = HELP_CONTACTS; | ||||
|             break; | ||||
|  | ||||
|         case 'k': | ||||
|         case L'k': | ||||
|             help_init_window(self, 15, 80); | ||||
|             self->help->type = HELP_KEYS; | ||||
|             break; | ||||
|  | ||||
|         case 'm': | ||||
|         case L'm': | ||||
|             help_init_menu(self); | ||||
|             self->help->type = HELP_MENU; | ||||
|             break; | ||||
| @@ -438,8 +458,8 @@ void help_onDraw(ToxWindow *self) | ||||
|             help_draw_contacts(self); | ||||
|             break; | ||||
|  | ||||
|         case HELP_GROUP: | ||||
|             help_draw_group(self); | ||||
|         case HELP_CONFERENCE: | ||||
|             help_draw_conference(self); | ||||
|             break; | ||||
|  | ||||
| #ifdef PYTHON | ||||
|   | ||||
| @@ -30,7 +30,7 @@ typedef enum { | ||||
|     HELP_MENU, | ||||
|     HELP_GLOBAL, | ||||
|     HELP_CHAT, | ||||
|     HELP_GROUP, | ||||
|     HELP_CONFERENCE, | ||||
|     HELP_KEYS, | ||||
|     HELP_CONTACTS, | ||||
| #ifdef PYTHON | ||||
|   | ||||
							
								
								
									
										88
									
								
								src/input.c
									
									
									
									
									
								
							
							
						
						
									
										88
									
								
								src/input.c
									
									
									
									
									
								
							| @@ -26,19 +26,19 @@ | ||||
|  | ||||
| #include <wchar.h> | ||||
|  | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "misc_tools.h" | ||||
| #include "toxic_strings.h" | ||||
| #include "conference.h" | ||||
| #include "line_info.h" | ||||
| #include "misc_tools.h" | ||||
| #include "notify.h" | ||||
| #include "groupchat.h" | ||||
| #include "settings.h" | ||||
| #include "toxic.h" | ||||
| #include "toxic_strings.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| extern struct user_settings *user_settings; | ||||
|  | ||||
| /* add a char to input field and buffer */ | ||||
| void input_new_char(ToxWindow *self, wint_t key, int x, int y, int mx_x, int mx_y) | ||||
| void input_new_char(ToxWindow *self, wint_t key, int x, int mx_x) | ||||
| { | ||||
|     ChatContext *ctx = self->chatwin; | ||||
|  | ||||
| @@ -94,13 +94,12 @@ static void input_delete(ToxWindow *self) | ||||
| } | ||||
|  | ||||
| /* delete last typed word */ | ||||
| static void input_del_word(ToxWindow *self, int x, int mx_x) | ||||
| static void input_del_word(ToxWindow *self) | ||||
| { | ||||
|     ChatContext *ctx = self->chatwin; | ||||
|  | ||||
|     if (del_word_buf(ctx) == -1) { | ||||
|         sound_notify(self, notif_error, 0, NULL); | ||||
|         return; | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -139,7 +138,7 @@ static void input_yank(ToxWindow *self, int x, int mx_x) | ||||
| } | ||||
|  | ||||
| /* moves cursor/line position to end of line in input field and buffer */ | ||||
| static void input_mv_end(ToxWindow *self, int y, int mx_x) | ||||
| static void input_mv_end(ToxWindow *self, int mx_x) | ||||
| { | ||||
|     ChatContext *ctx = self->chatwin; | ||||
|  | ||||
| @@ -172,17 +171,41 @@ static void input_mv_left(ToxWindow *self, int x, int mx_x) | ||||
|     } | ||||
|  | ||||
|     int cur_len = ctx->pos > 0 ? wcwidth(ctx->line[ctx->pos - 1]) : 0; | ||||
|     int s_len = ctx->start > 0 ? wcwidth(ctx->line[ctx->start - 1]) : 0; | ||||
|  | ||||
|     --ctx->pos; | ||||
|  | ||||
|     if (ctx->start && (x >= mx_x - cur_len)) { | ||||
|     if (ctx->start > 0 && (x >= mx_x - cur_len)) { | ||||
|         int s_len = wcwidth(ctx->line[ctx->start - 1]); | ||||
|         ctx->start = MAX(0, ctx->start - 1 + (s_len - cur_len)); | ||||
|     } else if (ctx->start) { | ||||
|     } else if (ctx->start > 0) { | ||||
|         ctx->start = MAX(0, ctx->start - cur_len); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /* moves the cursor to the beginning of the previous word in input field and buffer */ | ||||
| static void input_skip_left(ToxWindow *self, int x, int mx_x) | ||||
| { | ||||
|     ChatContext *ctx = self->chatwin; | ||||
|  | ||||
|     if (ctx->pos <= 0) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     int count = 0; | ||||
|  | ||||
|     do { | ||||
|         --ctx->pos; | ||||
|         count += wcwidth(ctx->line[ctx->pos]); | ||||
|     } while (ctx->pos > 0 && (ctx->line[ctx->pos - 1] != L' ' || ctx->line[ctx->pos] == L' ')); | ||||
|  | ||||
|     if (ctx->start > 0 && (x >= mx_x - count)) { | ||||
|         int s_len = wcwidth(ctx->line[ctx->start - 1]); | ||||
|         ctx->start = MAX(0, ctx->start - 1 + (s_len - count)); | ||||
|     } else if (ctx->start > 0) { | ||||
|         ctx->start = MAX(0, ctx->start - count); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /* moves cursor/line position right in input field and buffer */ | ||||
| static void input_mv_right(ToxWindow *self, int x, int mx_x) | ||||
| { | ||||
| @@ -202,6 +225,29 @@ static void input_mv_right(ToxWindow *self, int x, int mx_x) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /* moves the cursor to the end of the next word in input field and buffer */ | ||||
| static void input_skip_right(ToxWindow *self, int x, int mx_x) | ||||
| { | ||||
|     ChatContext *ctx = self->chatwin; | ||||
|  | ||||
|     if (ctx->pos >= ctx->len) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     int count = 0; | ||||
|  | ||||
|     do { | ||||
|         count += wcwidth(ctx->line[ctx->pos]); | ||||
|         ++ctx->pos; | ||||
|     } while (ctx->pos < ctx->len && !(ctx->line[ctx->pos] == L' ' && ctx->line[ctx->pos - 1] != L' ')); | ||||
|  | ||||
|     int newpos = x + count; | ||||
|  | ||||
|     if (newpos >= mx_x) { | ||||
|         ctx->start += (1 + (newpos - mx_x)); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /* puts a line history item in input field and buffer */ | ||||
| static void input_history(ToxWindow *self, wint_t key, int mx_x) | ||||
| { | ||||
| @@ -214,7 +260,7 @@ static void input_history(ToxWindow *self, wint_t key, int mx_x) | ||||
|  | ||||
| /* Handles non-printable input keys that behave the same for all types of chat windows. | ||||
|    return true if key matches a function, false otherwise */ | ||||
| bool input_handle(ToxWindow *self, wint_t key, int x, int y, int mx_x, int mx_y) | ||||
| bool input_handle(ToxWindow *self, wint_t key, int x, int mx_x) | ||||
| { | ||||
|     bool match = true; | ||||
|  | ||||
| @@ -241,7 +287,7 @@ bool input_handle(ToxWindow *self, wint_t key, int x, int y, int mx_x, int mx_y) | ||||
|             break; | ||||
|  | ||||
|         case T_KEY_C_W: | ||||
|             input_del_word(self, x, mx_x); | ||||
|             input_del_word(self); | ||||
|             break; | ||||
|  | ||||
|         case KEY_HOME: | ||||
| @@ -251,7 +297,7 @@ bool input_handle(ToxWindow *self, wint_t key, int x, int y, int mx_x, int mx_y) | ||||
|  | ||||
|         case KEY_END: | ||||
|         case T_KEY_C_E: | ||||
|             input_mv_end(self, y, mx_x); | ||||
|             input_mv_end(self, mx_x); | ||||
|             break; | ||||
|  | ||||
|         case KEY_LEFT: | ||||
| @@ -271,6 +317,14 @@ bool input_handle(ToxWindow *self, wint_t key, int x, int y, int mx_x, int mx_y) | ||||
|             force_refresh(self->chatwin->history); | ||||
|             break; | ||||
|  | ||||
|         case T_KEY_C_LEFT: | ||||
|             input_skip_left(self, x, mx_x); | ||||
|             break; | ||||
|  | ||||
|         case T_KEY_C_RIGHT: | ||||
|             input_skip_right(self, x, mx_x); | ||||
|             break; | ||||
|  | ||||
|         default: | ||||
|             match = false; | ||||
|             break; | ||||
| @@ -280,9 +334,9 @@ bool input_handle(ToxWindow *self, wint_t key, int x, int y, int mx_x, int mx_y) | ||||
|        maybe convert entire function to if/else and make them all customizable keys? */ | ||||
|     if (!match) { | ||||
|         if (key == user_settings->key_toggle_peerlist) { | ||||
|             if (self->is_groupchat) { | ||||
|             if (self->type == WINDOW_TYPE_CONFERENCE) { | ||||
|                 self->show_peerlist ^= 1; | ||||
|                 redraw_groupchat_win(self); | ||||
|                 redraw_conference_win(self); | ||||
|             } | ||||
|  | ||||
|             match = true; | ||||
|   | ||||
| @@ -24,10 +24,10 @@ | ||||
| #define INPUT_H | ||||
|  | ||||
| /* add a char to input field and buffer for given chatcontext */ | ||||
| void input_new_char(ToxWindow *self, wint_t key, int x, int y, int mx_x, int mx_y); | ||||
| void input_new_char(ToxWindow *self, wint_t key, int x, int mx_x); | ||||
|  | ||||
| /* Handles non-printable input keys that behave the same for all types of chat windows. | ||||
|    return true if key matches a function, false otherwise */ | ||||
| bool input_handle(ToxWindow *self, wint_t key, int x, int y, int mx_x, int mx_y); | ||||
| bool input_handle(ToxWindow *self, wint_t key, int x, int mx_x); | ||||
|  | ||||
| #endif /* INPUT_H */ | ||||
|   | ||||
							
								
								
									
										525
									
								
								src/line_info.c
									
									
									
									
									
								
							
							
						
						
									
										525
									
								
								src/line_info.c
									
									
									
									
									
								
							| @@ -20,19 +20,19 @@ | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <stdarg.h> | ||||
| #include <stdbool.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <stdbool.h> | ||||
| #include <stdarg.h> | ||||
|  | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "conference.h" | ||||
| #include "line_info.h" | ||||
| #include "groupchat.h" | ||||
| #include "settings.h" | ||||
| #include "notify.h" | ||||
| #include "message_queue.h" | ||||
| #include "misc_tools.h" | ||||
| #include "notify.h" | ||||
| #include "settings.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| extern struct user_settings *user_settings; | ||||
|  | ||||
| @@ -46,7 +46,7 @@ void line_info_init(struct history *hst) | ||||
|  | ||||
|     hst->line_start = hst->line_root; | ||||
|     hst->line_end = hst->line_start; | ||||
|     hst->queue_sz = 0; | ||||
|     hst->queue_size = 0; | ||||
| } | ||||
|  | ||||
| /* resets line_start (moves to end of chat history) */ | ||||
| @@ -54,27 +54,28 @@ void line_info_reset_start(ToxWindow *self, struct history *hst) | ||||
| { | ||||
|     struct line_info *line = hst->line_end; | ||||
|  | ||||
|     if (line->prev == NULL) { | ||||
|     if (line == NULL || line->prev == NULL) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     int y2, x2; | ||||
|     int y2; | ||||
|     int x2; | ||||
|     getmaxyx(self->window, y2, x2); | ||||
|     UNUSED_VAR(x2); | ||||
|  | ||||
|     int side_offst = self->show_peerlist ? SIDEBAR_WIDTH : 0; | ||||
|     int top_offst = self->is_chat || self->is_prompt ? 2 : 0; | ||||
|     int max_y = (y2 - CHATBOX_HEIGHT - top_offst); | ||||
|     int top_offst = (self->type == WINDOW_TYPE_CHAT) || (self->type == WINDOW_TYPE_PROMPT) ? TOP_BAR_HEIGHT : 0; | ||||
|     int max_y = y2 - CHATBOX_HEIGHT - WINDOW_BAR_HEIGHT - top_offst; | ||||
|  | ||||
|     int curlines = 0; | ||||
|     int nxtlines = line->newlines + (line->len / (x2 - side_offst)); | ||||
|     uint16_t curlines = 0; | ||||
|  | ||||
|     do { | ||||
|         curlines += 1 + nxtlines; | ||||
|         curlines += line->format_lines; | ||||
|         line = line->prev; | ||||
|         nxtlines = line->newlines + (line->len / (x2 - side_offst)); | ||||
|     } while (line->prev && curlines + nxtlines < max_y); | ||||
|     } while (line->prev && curlines + line->format_lines <= max_y); | ||||
|  | ||||
|     hst->line_start = line; | ||||
|  | ||||
|     self->scroll_pause = false; | ||||
| } | ||||
|  | ||||
| void line_info_cleanup(struct history *hst) | ||||
| @@ -87,9 +88,7 @@ void line_info_cleanup(struct history *hst) | ||||
|         tmp1 = tmp2; | ||||
|     } | ||||
|  | ||||
|     int i; | ||||
|  | ||||
|     for (i = 0; i < hst->queue_sz; ++i) { | ||||
|     for (size_t i = 0; i < hst->queue_size; ++i) { | ||||
|         if (hst->queue[i]) { | ||||
|             free(hst->queue[i]); | ||||
|         } | ||||
| @@ -117,29 +116,230 @@ static void line_info_root_fwd(struct history *hst) | ||||
| /* returns ptr to queue item 0 and removes it from queue. Returns NULL if queue is empty. */ | ||||
| static struct line_info *line_info_ret_queue(struct history *hst) | ||||
| { | ||||
|     if (hst->queue_sz <= 0) { | ||||
|     if (hst->queue_size == 0) { | ||||
|         return NULL; | ||||
|     } | ||||
|  | ||||
|     struct line_info *line = hst->queue[0]; | ||||
|  | ||||
|     int i; | ||||
|  | ||||
|     for (i = 0; i < hst->queue_sz; ++i) { | ||||
|     for (size_t i = 0; i < hst->queue_size; ++i) { | ||||
|         hst->queue[i] = hst->queue[i + 1]; | ||||
|     } | ||||
|  | ||||
|     --hst->queue_sz; | ||||
|     --hst->queue_size; | ||||
|  | ||||
|     return line; | ||||
| } | ||||
|  | ||||
| /* Prints a maximum of `n` chars from `s` to `win`. | ||||
|  * | ||||
|  * Return 1 if the string contains a newline byte. | ||||
|  * Return 0 if string does not contain a newline byte. | ||||
|  * Return -1 if printing was aborted. | ||||
|  */ | ||||
| static int print_n_chars(WINDOW *win, const char *s, size_t n, int max_y) | ||||
| { | ||||
|     bool newline = false; | ||||
|     char ch; | ||||
|  | ||||
|     for (size_t i = 0; i < n && (ch = s[i]); ++i) { | ||||
|         if (ch == '\n') { | ||||
|             newline = true; | ||||
|  | ||||
|             int x; | ||||
|             int y; | ||||
|             UNUSED_VAR(x); | ||||
|             getyx(win, y, x); | ||||
|  | ||||
|             // make sure cursor will wrap correctly after newline to prevent display bugs | ||||
|             if (y + 1 >= max_y) { | ||||
|                 return -1; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (win) { | ||||
|             wprintw(win, "%c", ch); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return newline; | ||||
| } | ||||
|  | ||||
| /* Returns the index of the last space character in `s` found before `limit`. | ||||
|  * Returns -1 if no space is found. | ||||
|  */ | ||||
| static int rspace_index(const char *s, int limit) | ||||
| { | ||||
|     for (int i = limit; i >= 0; --i) { | ||||
|         if (s[i] == ' ') { | ||||
|             return i; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return -1; | ||||
| } | ||||
|  | ||||
| /* Returns the first index in `s` containing a newline byte found before `limit`. | ||||
|  * Returns -1 if no newline  is found. | ||||
|  */ | ||||
| static int newline_index(const char *s, int limit) | ||||
| { | ||||
|     char ch; | ||||
|  | ||||
|     for (int i = 0; i < limit && (ch = s[i]); ++i) { | ||||
|         if (ch == '\n') { | ||||
|             return i; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return -1; | ||||
| } | ||||
|  | ||||
| /* Returns the number of newline bytes in `s` */ | ||||
| static unsigned int newline_count(const char *s) | ||||
| { | ||||
|     char ch; | ||||
|     unsigned int count = 0; | ||||
|  | ||||
|     for (size_t i = 0; (ch = s[i]); ++i) { | ||||
|         if (ch == '\n') { | ||||
|             ++count; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return count; | ||||
| } | ||||
|  | ||||
| /* Prints `line` message to window, wrapping at the last word that fits on the current line. | ||||
|  * This function updates the `format_lines` field of `line` according to current window dimensions. | ||||
|  * | ||||
|  * If `win` is null nothing will be printed to the window. This is useful to set the | ||||
|  * `format_lines` field on initialization. | ||||
|  * | ||||
|  * Return 0 on success. | ||||
|  * Return -1 if not all characters in line's message were printed to screen. | ||||
|  */ | ||||
| static int print_wrap(WINDOW *win, struct line_info *line, int max_x, int max_y) | ||||
| { | ||||
|     int x; | ||||
|     int y; | ||||
|     UNUSED_VAR(y); | ||||
|  | ||||
|     const char *msg = line->msg; | ||||
|     uint16_t length = line->msg_len; | ||||
|     uint16_t lines = 0; | ||||
|     const int x_start = line->len - line->msg_len - 1;  // manually keep track of x position because ncurses sucks | ||||
|     int x_limit = max_x - x_start; | ||||
|  | ||||
|     if (x_limit <= 1) { | ||||
|         fprintf(stderr, "Warning: x_limit <= 0 in print_wrap(): %d\n", x_limit); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     while (msg) { | ||||
|         getyx(win, y, x); | ||||
|  | ||||
|         // next line would print past window limit so we abort; we don't want to update format_lines | ||||
|         if (x > x_start) { | ||||
|             return -1; | ||||
|         } | ||||
|  | ||||
|         if (length < x_limit) { | ||||
|             int p_ret = print_n_chars(win, msg, length, max_y); | ||||
|  | ||||
|             if (p_ret == 1) { | ||||
|                 lines += newline_count(msg); | ||||
|             } else if (p_ret == -1) { | ||||
|                 return -1; | ||||
|             } | ||||
|  | ||||
|             ++lines; | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         int newline_idx = newline_index(msg, x_limit - 1); | ||||
|  | ||||
|         if (newline_idx >= 0) { | ||||
|             if (print_n_chars(win, msg, newline_idx + 1, max_y) == -1) { | ||||
|                 return -1; | ||||
|             } | ||||
|  | ||||
|             msg += (newline_idx + 1); | ||||
|             length -= (newline_idx + 1); | ||||
|             x_limit = max_x; // if we find a newline we stop adding column padding for rest of message | ||||
|             ++lines; | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         int space_idx = rspace_index(msg, x_limit - 1); | ||||
|  | ||||
|         if (space_idx >= 1) { | ||||
|             if (print_n_chars(win, msg, space_idx, max_y) == -1) { | ||||
|                 return -1; | ||||
|             } | ||||
|  | ||||
|             msg += space_idx + 1; | ||||
|             length -= (space_idx + 1); | ||||
|  | ||||
|             if (win) { | ||||
|                 waddch(win, '\n'); | ||||
|             } | ||||
|         } else { | ||||
|             if (print_n_chars(win, msg, x_limit, max_y) == -1) { | ||||
|                 return -1; | ||||
|             } | ||||
|  | ||||
|             msg += x_limit; | ||||
|             length -= x_limit; | ||||
|         } | ||||
|  | ||||
|         // Add padding to the start of the next line | ||||
|         if (win && x_limit < max_x) { | ||||
|             for (size_t i = 0; i < x_start; ++i) { | ||||
|                 waddch(win, ' '); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         ++lines; | ||||
|     } | ||||
|  | ||||
|     if (win && line->noread_flag) { | ||||
|         getyx(win, y, x); | ||||
|  | ||||
|         if (x >= max_x - 1 || x == x_start) { | ||||
|             ++lines; | ||||
|         } | ||||
|  | ||||
|         wattron(win, COLOR_PAIR(RED)); | ||||
|         wprintw(win, " x"); | ||||
|         wattroff(win, COLOR_PAIR(RED)); | ||||
|     } | ||||
|  | ||||
|     line->format_lines = lines; | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| static void line_info_init_line(ToxWindow *self, struct line_info *line) | ||||
| { | ||||
|     int y2; | ||||
|     int x2; | ||||
|     UNUSED_VAR(y2); | ||||
|  | ||||
|     getmaxyx(self->window, y2, x2); | ||||
|  | ||||
|     const int max_y = y2 - CHATBOX_HEIGHT - WINDOW_BAR_HEIGHT; | ||||
|     const int max_x = self->show_peerlist ? x2 - 1 - SIDEBAR_WIDTH : x2; | ||||
|  | ||||
|     print_wrap(NULL, line, max_x, max_y); | ||||
| } | ||||
|  | ||||
| /* creates new line_info line and puts it in the queue. | ||||
|  * | ||||
|  * Returns the id of the new line. | ||||
|  * Returns -1 on failure. | ||||
|  */ | ||||
| int line_info_add(ToxWindow *self, const char *timestr, const char *name1, const char *name2, uint8_t type, | ||||
| int line_info_add(ToxWindow *self, bool show_timestamp, const char *name1, const char *name2, LINE_TYPE type, | ||||
|                   uint8_t bold, uint8_t colour, const char *msg, ...) | ||||
| { | ||||
|     if (!self) { | ||||
| @@ -148,7 +348,7 @@ int line_info_add(ToxWindow *self, const char *timestr, const char *name1, const | ||||
|  | ||||
|     struct history *hst = self->chatwin->hst; | ||||
|  | ||||
|     if (hst->queue_sz >= MAX_LINE_INFO_QUEUE) { | ||||
|     if (hst->queue_size >= MAX_LINE_INFO_QUEUE) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
| @@ -158,7 +358,8 @@ int line_info_add(ToxWindow *self, const char *timestr, const char *name1, const | ||||
|         exit_toxic_err("failed in line_info_add", FATALERR_MEMORY); | ||||
|     } | ||||
|  | ||||
|     char frmt_msg[MAX_LINE_INFO_MSG_SIZE] = {0}; | ||||
|     char frmt_msg[MAX_LINE_INFO_MSG_SIZE]; | ||||
|     frmt_msg[0] = 0; | ||||
|  | ||||
|     va_list args; | ||||
|     va_start(args, msg); | ||||
| @@ -173,33 +374,33 @@ int line_info_add(ToxWindow *self, const char *timestr, const char *name1, const | ||||
|  | ||||
|         /* fallthrough */ | ||||
|         case OUT_ACTION: | ||||
|             len += strlen(user_settings->line_normal) + 2; | ||||
|             len += strlen(user_settings->line_normal) + 2; // two spaces | ||||
|             break; | ||||
|  | ||||
|         case IN_MSG: | ||||
|  | ||||
|         /* fallthrough */ | ||||
|         case OUT_MSG: | ||||
|             len += strlen(user_settings->line_normal) + 3; | ||||
|             len += strlen(user_settings->line_normal) + 3; // two spaces and a ':' char | ||||
|             break; | ||||
|  | ||||
|         case CONNECTION: | ||||
|             len += strlen(user_settings->line_join) + 2; | ||||
|             len += strlen(user_settings->line_join) + 2;  // two spaces | ||||
|             break; | ||||
|  | ||||
|         case DISCONNECTION: | ||||
|             len += strlen(user_settings->line_quit) + 2; | ||||
|             len += strlen(user_settings->line_quit) + 2;  // two spaces | ||||
|             break; | ||||
|  | ||||
|         case SYS_MSG: | ||||
|             break; | ||||
|  | ||||
|         case NAME_CHANGE: | ||||
|             len += strlen(user_settings->line_alert) + 1; | ||||
|             len += strlen(user_settings->line_alert) + 2;  // two spaces | ||||
|             break; | ||||
|  | ||||
|         case PROMPT: | ||||
|             ++len; | ||||
|             len += 2;  // '$' char and a space | ||||
|             break; | ||||
|  | ||||
|         default: | ||||
| @@ -207,21 +408,16 @@ int line_info_add(ToxWindow *self, const char *timestr, const char *name1, const | ||||
|             break; | ||||
|     } | ||||
|  | ||||
|     uint16_t msg_len = 0; | ||||
|  | ||||
|     if (frmt_msg[0]) { | ||||
|         snprintf(new_line->msg, sizeof(new_line->msg), "%s", frmt_msg); | ||||
|         len += strlen(new_line->msg); | ||||
|  | ||||
|         int i; | ||||
|  | ||||
|         for (i = 0; frmt_msg[i]; ++i) { | ||||
|             if (frmt_msg[i] == '\n') { | ||||
|                 ++new_line->newlines; | ||||
|             } | ||||
|         } | ||||
|         msg_len = strlen(new_line->msg); | ||||
|         len += msg_len; | ||||
|     } | ||||
|  | ||||
|     if (timestr) { | ||||
|         snprintf(new_line->timestr, sizeof(new_line->timestr), "%s", timestr); | ||||
|     if (show_timestamp) { | ||||
|         get_time_str(new_line->timestr, sizeof(new_line->timestr)); | ||||
|         len += strlen(new_line->timestr) + 1; | ||||
|     } | ||||
|  | ||||
| @@ -235,15 +431,18 @@ int line_info_add(ToxWindow *self, const char *timestr, const char *name1, const | ||||
|         len += strlen(new_line->name2); | ||||
|     } | ||||
|  | ||||
|     new_line->id = (hst->line_end->id + 1 + hst->queue_sz) % INT_MAX; | ||||
|     new_line->id = (hst->line_end->id + 1 + hst->queue_size) % INT_MAX; | ||||
|     new_line->len = len; | ||||
|     new_line->msg_len = msg_len; | ||||
|     new_line->type = type; | ||||
|     new_line->bold = bold; | ||||
|     new_line->colour = colour; | ||||
|     new_line->noread_flag = false; | ||||
|     new_line->timestamp = get_unix_time(); | ||||
|  | ||||
|     hst->queue[hst->queue_sz++] = new_line; | ||||
|     line_info_init_line(self, new_line); | ||||
|  | ||||
|     hst->queue[hst->queue_size++] = new_line; | ||||
|  | ||||
|     return new_line->id; | ||||
| } | ||||
| @@ -267,26 +466,8 @@ static void line_info_check_queue(ToxWindow *self) | ||||
|     hst->line_end = line; | ||||
|     hst->line_end->id = line->id; | ||||
|  | ||||
|     int y, y2, x, x2; | ||||
|     getmaxyx(self->window, y2, x2); | ||||
|     getyx(self->chatwin->history, y, x); | ||||
|     (void) x; | ||||
|  | ||||
|     if (x2 <= SIDEBAR_WIDTH) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     int offst = self->show_peerlist ? SIDEBAR_WIDTH : 0;   /* offset width of groupchat sidebar */ | ||||
|     int lines = 1 + line->newlines + (line->len / (x2 - offst)); | ||||
|     int max_y = y2 - CHATBOX_HEIGHT; | ||||
|  | ||||
|     /* move line_start forward proportionate to the number of new lines */ | ||||
|     if (y + lines - 1 >= max_y) { | ||||
|         while (lines > 0 && hst->line_start->next) { | ||||
|             lines -= 1 + hst->line_start->next->newlines + (hst->line_start->next->len / (x2 - offst)); | ||||
|             hst->line_start = hst->line_start->next; | ||||
|             ++hst->start_id; | ||||
|         } | ||||
|     if (!self->scroll_pause) { | ||||
|         line_info_reset_start(self, hst); | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -309,25 +490,44 @@ void line_info_print(ToxWindow *self) | ||||
|  | ||||
|     wclear(win); | ||||
|  | ||||
|     int y2, x2; | ||||
|     int y2; | ||||
|  | ||||
|     int x2; | ||||
|  | ||||
|     getmaxyx(self->window, y2, x2); | ||||
|  | ||||
|     if (x2 <= SIDEBAR_WIDTH) { | ||||
|     if (x2 - 1 <= SIDEBAR_WIDTH) {  // leave room on x axis for sidebar padding | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (self->is_groupchat) { | ||||
|     if (self->type == WINDOW_TYPE_CONFERENCE) { | ||||
|         wmove(win, 0, 0); | ||||
|     } else { | ||||
|         wmove(win, 2, 0); | ||||
|         wmove(win, TOP_BAR_HEIGHT, 0); | ||||
|     } | ||||
|  | ||||
|     struct line_info *line = hst->line_start->next; | ||||
|  | ||||
|     int numlines = 0; | ||||
|     if (!line) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     const int max_y = y2 - CHATBOX_HEIGHT - WINDOW_BAR_HEIGHT; | ||||
|     const int max_x = self->show_peerlist ? x2 - 1 - SIDEBAR_WIDTH : x2; | ||||
|     uint16_t numlines = line->format_lines; | ||||
|     int print_ret = 0; | ||||
|  | ||||
|     while (line && numlines++ <= max_y && print_ret == 0) { | ||||
|         int y; | ||||
|         int x; | ||||
|         UNUSED_VAR(y); | ||||
|  | ||||
|         getyx(win, y, x); | ||||
|  | ||||
|         if (x > 0) { // Prevents us from printing off the screen | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|     while (line && numlines++ <= y2) { | ||||
|         uint8_t type = line->type; | ||||
|  | ||||
|         switch (type) { | ||||
| @@ -354,43 +554,32 @@ void line_info_print(ToxWindow *self) | ||||
|                 wprintw(win, "%s %s: ", user_settings->line_normal, line->name1); | ||||
|                 wattroff(win, COLOR_PAIR(nameclr)); | ||||
|  | ||||
|                 char *msg = line->msg; | ||||
|                 if (line->msg[0] == 0) { | ||||
|                     waddch(win, '\n'); | ||||
|                     break; | ||||
|                 } | ||||
|  | ||||
|                 while (msg) { | ||||
|                     char *line = strsep(&msg, "\n"); | ||||
|  | ||||
|                     if (line[0] == '>') { | ||||
|                 if (line->msg[0] == '>') { | ||||
|                     wattron(win, COLOR_PAIR(GREEN)); | ||||
|                     } else if (line[0] == '<') { | ||||
|                 } else if (line->msg[0] == '<') { | ||||
|                     wattron(win, COLOR_PAIR(RED)); | ||||
|                 } | ||||
|  | ||||
|                     wprintw(win, "%s%c", line, msg ? '\n' : '\0'); | ||||
|                 print_ret = print_wrap(win, line, max_x, max_y); | ||||
|  | ||||
|                     if (line[0] == '>') { | ||||
|                 if (line->msg[0] == '>') { | ||||
|                     wattroff(win, COLOR_PAIR(GREEN)); | ||||
|                     } else if (line[0] == '<') { | ||||
|                 } else if (line->msg[0] == '<') { | ||||
|                     wattroff(win, COLOR_PAIR(RED)); | ||||
|                 } | ||||
|  | ||||
|                     // change the \0 set by strsep back to \n | ||||
|                     if (msg) { | ||||
|                         msg[-1] = '\n'; | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 if (type == OUT_MSG && timed_out(line->timestamp, NOREAD_FLAG_TIMEOUT)) { | ||||
|                     wattron(win, COLOR_PAIR(RED)); | ||||
|                     wprintw(win, " x", line->msg); | ||||
|                     wattroff(win, COLOR_PAIR(RED)); | ||||
|  | ||||
|                     if (line->noread_flag == false) { | ||||
|                 if (type == OUT_MSG && !line->read_flag) { | ||||
|                     if (timed_out(line->timestamp, NOREAD_FLAG_TIMEOUT)) { | ||||
|                         line->noread_flag = true; | ||||
|                         line->len += 2; | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 wprintw(win, "\n", line->msg); | ||||
|                 waddch(win, '\n'); | ||||
|                 break; | ||||
|  | ||||
|             case OUT_ACTION_READ: | ||||
| @@ -405,21 +594,17 @@ void line_info_print(ToxWindow *self) | ||||
|                 wattroff(win, COLOR_PAIR(BLUE)); | ||||
|  | ||||
|                 wattron(win, COLOR_PAIR(YELLOW)); | ||||
|                 wprintw(win, "%s %s %s", user_settings->line_normal, line->name1, line->msg); | ||||
|                 wprintw(win, "%s %s ", user_settings->line_normal, line->name1); | ||||
|                 print_ret = print_wrap(win, line, max_x, max_y); | ||||
|                 wattroff(win, COLOR_PAIR(YELLOW)); | ||||
|  | ||||
|                 if (type == OUT_ACTION && timed_out(line->timestamp, NOREAD_FLAG_TIMEOUT)) { | ||||
|                     wattron(win, COLOR_PAIR(RED)); | ||||
|                     wprintw(win, " x", line->msg); | ||||
|                     wattroff(win, COLOR_PAIR(RED)); | ||||
|  | ||||
|                     if (line->noread_flag == false) { | ||||
|                 if (type == OUT_ACTION && !line->read_flag) { | ||||
|                     if (timed_out(line->timestamp, NOREAD_FLAG_TIMEOUT)) { | ||||
|                         line->noread_flag = true; | ||||
|                         line->len += 2; | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 wprintw(win, "\n", line->msg); | ||||
|                 waddch(win, '\n'); | ||||
|                 break; | ||||
|  | ||||
|             case SYS_MSG: | ||||
| @@ -437,7 +622,8 @@ void line_info_print(ToxWindow *self) | ||||
|                     wattron(win, COLOR_PAIR(line->colour)); | ||||
|                 } | ||||
|  | ||||
|                 wprintw(win, "%s\n", line->msg); | ||||
|                 print_ret = print_wrap(win, line, max_x, max_y); | ||||
|                 waddch(win, '\n'); | ||||
|  | ||||
|                 if (line->bold) { | ||||
|                     wattroff(win, A_BOLD); | ||||
| @@ -455,10 +641,10 @@ void line_info_print(ToxWindow *self) | ||||
|                 wattroff(win, COLOR_PAIR(GREEN)); | ||||
|  | ||||
|                 if (line->msg[0]) { | ||||
|                     wprintw(win, "%s", line->msg); | ||||
|                     print_ret = print_wrap(win, line, max_x, max_y); | ||||
|                 } | ||||
|  | ||||
|                 wprintw(win, "\n"); | ||||
|                 waddch(win, '\n'); | ||||
|                 break; | ||||
|  | ||||
|             case CONNECTION: | ||||
| @@ -473,7 +659,9 @@ void line_info_print(ToxWindow *self) | ||||
|                 wprintw(win, "%s ", line->name1); | ||||
|                 wattroff(win, A_BOLD); | ||||
|  | ||||
|                 wprintw(win, "%s\n", line->msg); | ||||
|                 print_ret = print_wrap(win, line, max_x, max_y); | ||||
|                 waddch(win, '\n'); | ||||
|  | ||||
|                 wattroff(win, COLOR_PAIR(line->colour)); | ||||
|  | ||||
|                 break; | ||||
| @@ -490,7 +678,9 @@ void line_info_print(ToxWindow *self) | ||||
|                 wprintw(win, "%s ", line->name1); | ||||
|                 wattroff(win, A_BOLD); | ||||
|  | ||||
|                 wprintw(win, "%s\n", line->msg); | ||||
|                 print_ret = print_wrap(win, line, max_x, max_y); | ||||
|                 waddch(win, '\n'); | ||||
|  | ||||
|                 wattroff(win, COLOR_PAIR(line->colour)); | ||||
|  | ||||
|                 break; | ||||
| @@ -506,7 +696,7 @@ void line_info_print(ToxWindow *self) | ||||
|                 wprintw(win, "%s", line->name1); | ||||
|                 wattroff(win, A_BOLD); | ||||
|  | ||||
|                 wprintw(win, "%s", line->msg); | ||||
|                 print_ret = print_wrap(win, line, max_x, max_y); | ||||
|  | ||||
|                 wattron(win, A_BOLD); | ||||
|                 wprintw(win, "%s\n", line->name2); | ||||
| @@ -520,11 +710,43 @@ void line_info_print(ToxWindow *self) | ||||
|     } | ||||
|  | ||||
|     /* keep calling until queue is empty */ | ||||
|     if (hst->queue_sz > 0) { | ||||
|     if (hst->queue_size > 0) { | ||||
|         line_info_print(self); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Return true if all lines starting from `line` can fit on the screen. | ||||
|  */ | ||||
| static bool line_info_screen_fit(ToxWindow *self, struct line_info *line) | ||||
| { | ||||
|     if (!line) { | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     int x2; | ||||
|     int y2; | ||||
|     getmaxyx(self->chatwin->history, y2, x2); | ||||
|  | ||||
|     UNUSED_VAR(x2); | ||||
|  | ||||
|     const int top_offset = (self->type == WINDOW_TYPE_CHAT) || (self->type == WINDOW_TYPE_PROMPT) ? TOP_BAR_HEIGHT : 0; | ||||
|     const int max_y = y2 - top_offset; | ||||
|  | ||||
|     uint16_t lines = line->format_lines; | ||||
|  | ||||
|     while (line) { | ||||
|         if (lines > max_y) { | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         lines += line->format_lines; | ||||
|         line = line->next; | ||||
|     } | ||||
|  | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| /* puts msg in specified line_info msg buffer */ | ||||
| void line_info_set(ToxWindow *self, uint32_t id, char *msg) | ||||
| { | ||||
| @@ -532,6 +754,9 @@ void line_info_set(ToxWindow *self, uint32_t id, char *msg) | ||||
|  | ||||
|     while (line) { | ||||
|         if (line->id == id) { | ||||
|             size_t new_len = strlen(msg); | ||||
|             line->len = line->len - line->msg_len + new_len; | ||||
|             line->msg_len = new_len; | ||||
|             snprintf(line->msg, sizeof(line->msg), "%s", msg); | ||||
|             return; | ||||
|         } | ||||
| @@ -540,52 +765,74 @@ void line_info_set(ToxWindow *self, uint32_t id, char *msg) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /* static void line_info_goto_root(struct history *hst) | ||||
| { | ||||
|     hst->line_start = hst->line_root; | ||||
| } */ | ||||
|  | ||||
| static void line_info_scroll_up(struct history *hst) | ||||
| static void line_info_scroll_up(ToxWindow *self, struct history *hst) | ||||
| { | ||||
|     if (hst->line_start->prev) { | ||||
|         hst->line_start = hst->line_start->prev; | ||||
|     } else { | ||||
|         sound_notify(NULL, notif_error, NT_ALWAYS, NULL); | ||||
|         self->scroll_pause = true; | ||||
|     } | ||||
| } | ||||
|  | ||||
| static void line_info_scroll_down(struct history *hst) | ||||
| static void line_info_scroll_down(ToxWindow *self, struct history *hst) | ||||
| { | ||||
|     if (hst->line_start->next) { | ||||
|         hst->line_start = hst->line_start->next; | ||||
|     struct line_info *next = hst->line_start->next; | ||||
|  | ||||
|     if (next && self->scroll_pause) { | ||||
|         if (line_info_screen_fit(self, next->next)) { | ||||
|             line_info_reset_start(self, hst); | ||||
|         } else { | ||||
|         sound_notify(NULL, notif_error, NT_ALWAYS, NULL); | ||||
|             hst->line_start = next; | ||||
|         } | ||||
|     } else { | ||||
|         line_info_reset_start(self, hst); | ||||
|     } | ||||
| } | ||||
|  | ||||
| static void line_info_page_up(ToxWindow *self, struct history *hst) | ||||
| { | ||||
|     int x2, y2; | ||||
|     int x2; | ||||
|     int y2; | ||||
|     getmaxyx(self->window, y2, x2); | ||||
|     (void) x2; | ||||
|     int jump_dist = y2 / 2; | ||||
|     int i; | ||||
|  | ||||
|     for (i = 0; i < jump_dist && hst->line_start->prev; ++i) { | ||||
|     UNUSED_VAR(x2); | ||||
|  | ||||
|     const int top_offset = (self->type == WINDOW_TYPE_CHAT) || (self->type == WINDOW_TYPE_PROMPT) ? TOP_BAR_HEIGHT : 0; | ||||
|     const int max_y = y2 - top_offset; | ||||
|     size_t jump_dist = max_y / 2; | ||||
|  | ||||
|     for (size_t i = 0; i < jump_dist && hst->line_start->prev; ++i) { | ||||
|         hst->line_start = hst->line_start->prev; | ||||
|     } | ||||
|  | ||||
|     self->scroll_pause = true; | ||||
| } | ||||
|  | ||||
| static void line_info_page_down(ToxWindow *self, struct history *hst) | ||||
| { | ||||
|     int x2, y2; | ||||
|     getmaxyx(self->window, y2, x2); | ||||
|     (void) x2; | ||||
|     int jump_dist = y2 / 2; | ||||
|     int i; | ||||
|     if (!self->scroll_pause) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     for (i = 0; i < jump_dist && hst->line_start->next; ++i) { | ||||
|         hst->line_start = hst->line_start->next; | ||||
|     int x2; | ||||
|     int y2; | ||||
|     getmaxyx(self->chatwin->history, y2, x2); | ||||
|  | ||||
|     UNUSED_VAR(x2); | ||||
|  | ||||
|     const int top_offset = (self->type == WINDOW_TYPE_CHAT) || (self->type == WINDOW_TYPE_PROMPT) ? TOP_BAR_HEIGHT : 0; | ||||
|     const int max_y = y2 - top_offset; | ||||
|     size_t jump_dist = max_y / 2; | ||||
|  | ||||
|     struct line_info *next = hst->line_start->next; | ||||
|  | ||||
|     for (size_t i = 0; i < jump_dist && next; ++i) { | ||||
|         if (line_info_screen_fit(self, next->next)) { | ||||
|             line_info_reset_start(self, hst); | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         hst->line_start = next; | ||||
|         next = hst->line_start->next; | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -599,9 +846,9 @@ bool line_info_onKey(ToxWindow *self, wint_t key) | ||||
|     } else if (key == user_settings->key_half_page_down) { | ||||
|         line_info_page_down(self, hst); | ||||
|     } else if (key == user_settings->key_scroll_line_up) { | ||||
|         line_info_scroll_up(hst); | ||||
|         line_info_scroll_up(self, hst); | ||||
|     } else if (key == user_settings->key_scroll_line_down) { | ||||
|         line_info_scroll_down(hst); | ||||
|         line_info_scroll_down(self, hst); | ||||
|     } else if (key == user_settings->key_page_bottom) { | ||||
|         line_info_reset_start(self, hst); | ||||
|     } else { | ||||
|   | ||||
| @@ -23,15 +23,15 @@ | ||||
| #ifndef LINE_INFO_H | ||||
| #define LINE_INFO_H | ||||
|  | ||||
| #include "windows.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| #define MAX_HISTORY 100000 | ||||
| #define MIN_HISTORY 40 | ||||
| #define MAX_LINE_INFO_QUEUE 1024 | ||||
| #define MAX_LINE_INFO_MSG_SIZE MAX_STR_SIZE + TOXIC_MAX_NAME_LENGTH + 32    /* needs extra room for log loading */ | ||||
| #define MAX_LINE_INFO_MSG_SIZE (MAX_STR_SIZE + TOXIC_MAX_NAME_LENGTH + 32) /* needs extra room for log loading */ | ||||
|  | ||||
| typedef enum { | ||||
| typedef enum LINE_TYPE { | ||||
|     SYS_MSG, | ||||
|     IN_MSG, | ||||
|     OUT_MSG, | ||||
| @@ -54,10 +54,12 @@ struct line_info { | ||||
|     uint8_t type; | ||||
|     uint8_t bold; | ||||
|     uint8_t colour; | ||||
|     uint8_t noread_flag;   /* true if a line should be flagged as unread */ | ||||
|     bool    noread_flag;   /* true if a line should be flagged as unread */ | ||||
|     bool    read_flag;     /* true if a message has been flagged as read */ | ||||
|     uint32_t id; | ||||
|     uint16_t len;   /* combined len of entire line */ | ||||
|     uint8_t newlines; | ||||
|     uint16_t len;        /* combined length of entire line */ | ||||
|     uint16_t msg_len;    /* length of the message */ | ||||
|     uint16_t format_lines;  /* number of lines the combined string takes up (dynamically set) */ | ||||
|  | ||||
|     struct line_info *prev; | ||||
|     struct line_info *next; | ||||
| @@ -71,7 +73,7 @@ struct history { | ||||
|     uint32_t start_id;    /* keeps track of where line_start should be when at bottom of history */ | ||||
|  | ||||
|     struct line_info *queue[MAX_LINE_INFO_QUEUE]; | ||||
|     int queue_sz; | ||||
|     size_t queue_size; | ||||
| }; | ||||
|  | ||||
| /* creates new line_info line and puts it in the queue. | ||||
| @@ -79,7 +81,7 @@ struct history { | ||||
|  * Returns the id of the new line. | ||||
|  * Returns -1 on failure. | ||||
|  */ | ||||
| int line_info_add(ToxWindow *self, const char *timestr, const char *name1, const char *name2, uint8_t type, | ||||
| int line_info_add(ToxWindow *self, bool show_timestamp, const char *name1, const char *name2, LINE_TYPE type, | ||||
|                   uint8_t bold, uint8_t colour, const char *msg, ...); | ||||
|  | ||||
| /* Prints a section of history starting at line_start */ | ||||
|   | ||||
							
								
								
									
										256
									
								
								src/log.c
									
									
									
									
									
								
							
							
						
						
									
										256
									
								
								src/log.c
									
									
									
									
									
								
							| @@ -22,42 +22,46 @@ | ||||
|  | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <time.h> | ||||
| #include <sys/stat.h> | ||||
| #include <time.h> | ||||
|  | ||||
| #include "configdir.h" | ||||
| #include "line_info.h" | ||||
| #include "log.h" | ||||
| #include "misc_tools.h" | ||||
| #include "settings.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "misc_tools.h" | ||||
| #include "log.h" | ||||
| #include "settings.h" | ||||
| #include "line_info.h" | ||||
|  | ||||
| extern struct user_settings *user_settings; | ||||
|  | ||||
| /* There are three types of logs: chat logs, groupchat logs, and prompt logs (see LOG_TYPE in log.h) | ||||
|    A prompt log is in the format: LOGDIR/selfkey-home.log | ||||
|    A chat log is in the format: LOGDIR/selfkey-friendname-otherkey.log | ||||
|    A groupchat log is in the format: LOGDIR/selfkey-groupname-date[time].log | ||||
|  | ||||
|    Only the first (KEY_IDENT_DIGITS * 2) numbers of the key are used. | ||||
|  | ||||
|    Returns 0 on success, -1 if the path is too long */ | ||||
| static int get_log_path(char *dest, int destsize, char *name, const char *selfkey, const char *otherkey, int logtype) | ||||
| /* Creates a log path and puts it in `dest. | ||||
|  * | ||||
|  * There are two types of logs: chat logs and prompt logs (see LOG_TYPE in log.h) | ||||
|  * A prompt log is in the format: LOGDIR/selfkey-home.log | ||||
|  * A chat log is in the format: LOGDIR/selfkey-name-otherkey.log | ||||
|  * | ||||
|  * For friend chats `otherkey` is the first 6 bytes of the friend's Tox ID. | ||||
|  * For Conferences/groups `otherkey` is the first 6 bytes of the group's unique ID. | ||||
|  * | ||||
|  * Return path length on success. | ||||
|  * Return -1 if the path is too long. | ||||
|  */ | ||||
| static int get_log_path(char *dest, int destsize, const char *name, const char *selfkey, const char *otherkey) | ||||
| { | ||||
|     if (!valid_nick(name)) { | ||||
|         name = UNKNOWN_NAME; | ||||
|     } | ||||
|  | ||||
|     const char *namedash = logtype == LOG_PROMPT ? "" : "-"; | ||||
|     const char *namedash = otherkey ? "-" : ""; | ||||
|     const char *set_path = user_settings->chatlogs_path; | ||||
|  | ||||
|     char *user_config_dir = get_user_config_dir(); | ||||
|     int path_len = strlen(name) + strlen(".log") + strlen("-") + strlen(namedash); | ||||
|     path_len += strlen(set_path) ? *set_path : strlen(user_config_dir) + strlen(LOGDIR); | ||||
|  | ||||
|     /* first 6 digits of selfkey */ | ||||
|     char self_id[32]; | ||||
|     /* first 6 bytes of selfkey */ | ||||
|     char self_id[32] = {0}; | ||||
|     path_len += KEY_IDENT_DIGITS * 2; | ||||
|     sprintf(&self_id[0], "%02X", selfkey[0] & 0xff); | ||||
|     sprintf(&self_id[2], "%02X", selfkey[1] & 0xff); | ||||
| @@ -66,19 +70,13 @@ static int get_log_path(char *dest, int destsize, char *name, const char *selfke | ||||
|  | ||||
|     char other_id[32] = {0}; | ||||
|  | ||||
|     switch (logtype) { | ||||
|         case LOG_CHAT: | ||||
|     if (otherkey) { | ||||
|         /* first 6 bytes of otherkey */ | ||||
|         path_len += KEY_IDENT_DIGITS * 2; | ||||
|         sprintf(&other_id[0], "%02X", otherkey[0] & 0xff); | ||||
|         sprintf(&other_id[2], "%02X", otherkey[1] & 0xff); | ||||
|         sprintf(&other_id[4], "%02X", otherkey[2] & 0xff); | ||||
|         other_id[KEY_IDENT_DIGITS * 2] = '\0'; | ||||
|             break; | ||||
|  | ||||
|         case LOG_GROUP: | ||||
|             strftime(other_id, sizeof(other_id), "%Y-%m-%d[%H:%M:%S]", get_time()); | ||||
|             path_len += strlen(other_id); | ||||
|             break; | ||||
|     } | ||||
|  | ||||
|     if (path_len >= destsize) { | ||||
| @@ -94,28 +92,35 @@ static int get_log_path(char *dest, int destsize, char *name, const char *selfke | ||||
|  | ||||
|     free(user_config_dir); | ||||
|  | ||||
|     return 0; | ||||
|     return path_len; | ||||
| } | ||||
|  | ||||
| /* Opens log file or creates a new one */ | ||||
| static int init_logging_session(char *name, const char *selfkey, const char *otherkey, struct chatlog *log, int logtype) | ||||
| /* Initializes log path for `log`. | ||||
|  * | ||||
|  * Return 0 on success. | ||||
|  * Return -1 on failure. | ||||
|  */ | ||||
| static int init_logging_session(const char *name, const char *selfkey, const char *otherkey, struct chatlog *log, | ||||
|                                 LOG_TYPE type) | ||||
| { | ||||
|     if (selfkey == NULL || (logtype == LOG_CHAT && otherkey == NULL)) { | ||||
|     if (log == NULL) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (selfkey == NULL || (type == LOG_TYPE_CHAT && otherkey == NULL)) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     char log_path[MAX_STR_SIZE]; | ||||
|  | ||||
|     if (get_log_path(log_path, sizeof(log_path), name, selfkey, otherkey, logtype) == -1) { | ||||
|     int path_len = get_log_path(log_path, sizeof(log_path), name, selfkey, otherkey); | ||||
|  | ||||
|     if (path_len == -1 || path_len >= sizeof(log->path)) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     log->file = fopen(log_path, "a+"); | ||||
|     snprintf(log->path, sizeof(log->path), "%s", log_path); | ||||
|  | ||||
|     if (log->file == NULL) { | ||||
|         return -1; | ||||
|     } | ||||
|     memcpy(log->path, log_path, path_len); | ||||
|     log->path[path_len] = 0; | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
| @@ -124,6 +129,10 @@ static int init_logging_session(char *name, const char *selfkey, const char *oth | ||||
|  | ||||
| void write_to_log(const char *msg, const char *name, struct chatlog *log, bool event) | ||||
| { | ||||
|     if (log == NULL) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (!log->log_on) { | ||||
|         return; | ||||
|     } | ||||
| @@ -154,92 +163,162 @@ void write_to_log(const char *msg, const char *name, struct chatlog *log, bool e | ||||
|  | ||||
| void log_disable(struct chatlog *log) | ||||
| { | ||||
|     if (log == NULL) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (log->file != NULL) { | ||||
|         fclose(log->file); | ||||
|         log->file = NULL; | ||||
|     } | ||||
|  | ||||
|     memset(log, 0, sizeof(struct chatlog)); | ||||
|     log->lastwrite = 0; | ||||
|     log->log_on = false; | ||||
| } | ||||
|  | ||||
| int log_enable(char *name, const char *selfkey, const char *otherkey, struct chatlog *log, int logtype) | ||||
| int log_enable(struct chatlog *log) | ||||
| { | ||||
|     log->log_on = true; | ||||
|  | ||||
|     if (log->file != NULL) { | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     if (init_logging_session(name, selfkey, otherkey, log, logtype) == -1) { | ||||
|         log_disable(log); | ||||
|     if (log == NULL) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (log->log_on) { | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     if (*log->path == 0) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (log->file != NULL) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     log->file = fopen(log->path, "a+"); | ||||
|  | ||||
|     if (log->file == NULL) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     log->log_on = true; | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| /* Loads previous history from chat log */ | ||||
| void load_chat_history(ToxWindow *self, struct chatlog *log) | ||||
| /* Initializes a log. This function must be called before any other logging operations. | ||||
|  * | ||||
|  * Return 0 on success. | ||||
|  * Return -1 on failure. | ||||
|  */ | ||||
| int log_init(struct chatlog *log, const char *name, const char *selfkey, const char *otherkey, LOG_TYPE type) | ||||
| { | ||||
|     if (log->file == NULL) { | ||||
|         return; | ||||
|     if (log == NULL) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (log->file != NULL || log->log_on) { | ||||
|         fprintf(stderr, "Warning: Called log_init() on an already initialized log\n"); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (init_logging_session(name, selfkey, otherkey, log, type) == -1) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     log_disable(log); | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| /* Loads chat log history and prints it to `self` window. | ||||
|  * | ||||
|  * Return 0 on success or if log file doesn't exist. | ||||
|  * Return -1 on failure. | ||||
|  */ | ||||
| int load_chat_history(ToxWindow *self, struct chatlog *log) | ||||
| { | ||||
|     if (log == NULL) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (*log->path == 0) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     off_t sz = file_size(log->path); | ||||
|  | ||||
|     if (sz <= 0) { | ||||
|         return; | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     char *hstbuf = malloc(sz + 1); | ||||
|     FILE *fp = fopen(log->path, "r"); | ||||
|  | ||||
|     if (hstbuf == NULL) { | ||||
|         exit_toxic_err("failed in load_chat_history", FATALERR_MEMORY); | ||||
|     if (fp == NULL) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (fseek(log->file, 0L, SEEK_SET) == -1) { | ||||
|         free(hstbuf); | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, RED, " * Failed to read log file"); | ||||
|         return; | ||||
|     char *buf = malloc(sz + 1); | ||||
|  | ||||
|     if (buf == NULL) { | ||||
|         fclose(fp); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (fread(hstbuf, sz, 1, log->file) != 1) { | ||||
|         free(hstbuf); | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, RED, " * Failed to read log file"); | ||||
|         return; | ||||
|     if (fseek(fp, 0L, SEEK_SET) == -1) { | ||||
|         free(buf); | ||||
|         fclose(fp); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     hstbuf[sz] = '\0'; | ||||
|     if (fread(buf, sz, 1, fp) != 1) { | ||||
|         free(buf); | ||||
|         fclose(fp); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     fclose(fp); | ||||
|  | ||||
|     buf[sz] = 0; | ||||
|  | ||||
|     /* Number of history lines to load: must not be larger than MAX_LINE_INFO_QUEUE - 2 */ | ||||
|     int L = MIN(MAX_LINE_INFO_QUEUE - 2, user_settings->history_size); | ||||
|     int start, count = 0; | ||||
|  | ||||
|     int start = 0; | ||||
|     int count = 0; | ||||
|  | ||||
|     /* start at end and backtrace L lines or to the beginning of buffer */ | ||||
|     for (start = sz - 1; start >= 0 && count < L; --start) { | ||||
|         if (hstbuf[start] == '\n') { | ||||
|         if (buf[start] == '\n') { | ||||
|             ++count; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     const char *line = strtok(&hstbuf[start + 1], "\n"); | ||||
|     char *tmp = NULL; | ||||
|     const char *line = strtok_r(&buf[start + 1], "\n", &tmp); | ||||
|  | ||||
|     if (line == NULL) { | ||||
|         free(hstbuf); | ||||
|         return; | ||||
|         free(buf); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     while (line != NULL && count--) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "%s", line); | ||||
|         line = strtok(NULL, "\n"); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "%s", line); | ||||
|         line = strtok_r(NULL, "\n", &tmp); | ||||
|     } | ||||
|  | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, ""); | ||||
|     free(hstbuf); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, YELLOW, "---"); | ||||
|  | ||||
|     free(buf); | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| /* renames chatlog file replacing src with dest. | ||||
|    Returns 0 on success or if no log exists, -1 on failure. */ | ||||
| int rename_logfile(char *src, char *dest, const char *selfkey, const char *otherkey, int winnum) | ||||
| /* Renames chatlog file `src` to `dest`. | ||||
|  * | ||||
|  * Return 0 on success or if no log exists. | ||||
|  * Return -1 on failure. | ||||
|  */ | ||||
| int rename_logfile(const char *src, const char *dest, const char *selfkey, const char *otherkey, int winnum) | ||||
| { | ||||
|     ToxWindow *toxwin = get_window_ptr(winnum); | ||||
|     struct chatlog *log = NULL; | ||||
| @@ -248,6 +327,11 @@ int rename_logfile(char *src, char *dest, const char *selfkey, const char *other | ||||
|     /* disable log if necessary and save its state */ | ||||
|     if (toxwin != NULL) { | ||||
|         log = toxwin->chatwin->log; | ||||
|  | ||||
|         if (log == NULL) { | ||||
|             return -1; | ||||
|         } | ||||
|  | ||||
|         log_on = log->log_on; | ||||
|     } | ||||
|  | ||||
| @@ -258,24 +342,36 @@ int rename_logfile(char *src, char *dest, const char *selfkey, const char *other | ||||
|     char newpath[MAX_STR_SIZE]; | ||||
|     char oldpath[MAX_STR_SIZE]; | ||||
|  | ||||
|     if (get_log_path(oldpath, sizeof(oldpath), src, selfkey, otherkey, LOG_CHAT) == -1) { | ||||
|     if (get_log_path(oldpath, sizeof(oldpath), src, selfkey, otherkey) == -1) { | ||||
|         goto on_error; | ||||
|     } | ||||
|  | ||||
|     if (!file_exists(oldpath)) { | ||||
|         init_logging_session(dest, selfkey, otherkey, log, LOG_TYPE_CHAT);  // still need to rename path | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     if (get_log_path(newpath, sizeof(newpath), dest, selfkey, otherkey, LOG_CHAT) == -1) { | ||||
|     int new_path_len = get_log_path(newpath, sizeof(newpath), dest, selfkey, otherkey); | ||||
|  | ||||
|     if (new_path_len == -1 || new_path_len >= MAX_STR_SIZE) { | ||||
|         goto on_error; | ||||
|     } | ||||
|  | ||||
|     if (rename(oldpath, newpath) != 0) { | ||||
|     if (file_exists(newpath)) { | ||||
|         if (remove(oldpath) != 0) { | ||||
|             fprintf(stderr, "Warning: remove() failed to remove log path `%s`\n", oldpath); | ||||
|         } | ||||
|     } else if (rename(oldpath, newpath) != 0) { | ||||
|         goto on_error; | ||||
|     } | ||||
|  | ||||
|     if (log != NULL) { | ||||
|         memcpy(log->path, newpath, new_path_len); | ||||
|         log->path[new_path_len] = 0; | ||||
|  | ||||
|         if (log_on) { | ||||
|         log_enable(dest, selfkey, otherkey, log, LOG_CHAT); | ||||
|             log_enable(log); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return 0; | ||||
| @@ -283,7 +379,7 @@ int rename_logfile(char *src, char *dest, const char *selfkey, const char *other | ||||
| on_error: | ||||
|  | ||||
|     if (log_on) { | ||||
|         log_enable(src, selfkey, otherkey, log, LOG_CHAT); | ||||
|         log_enable(log); | ||||
|     } | ||||
|  | ||||
|     return -1; | ||||
|   | ||||
							
								
								
									
										35
									
								
								src/log.h
									
									
									
									
									
								
							
							
						
						
									
										35
									
								
								src/log.h
									
									
									
									
									
								
							| @@ -30,30 +30,43 @@ struct chatlog { | ||||
|     bool log_on;    /* specific to current chat window */ | ||||
| }; | ||||
|  | ||||
| typedef enum { | ||||
|     LOG_GROUP, | ||||
|     LOG_PROMPT, | ||||
|     LOG_CHAT, | ||||
| typedef enum LOG_TYPE { | ||||
|     LOG_TYPE_PROMPT, | ||||
|     LOG_TYPE_CHAT, | ||||
| } LOG_TYPE; | ||||
|  | ||||
| /* Initializes a log. This function must be called before any other logging operations. | ||||
|  * | ||||
|  * Return 0 on success. | ||||
|  * Return -1 on failure. | ||||
|  */ | ||||
| int log_init(struct chatlog *log, const char *name, const char *selfkey, const char *otherkey, LOG_TYPE type); | ||||
|  | ||||
| /* formats/writes line to log file */ | ||||
| void write_to_log(const char *msg, const char *name, struct chatlog *log, bool event); | ||||
|  | ||||
| /* enables logging for specified log and creates/fetches file if necessary. | ||||
| /* enables logging for specified log. | ||||
|  * | ||||
|  * Returns 0 on success. | ||||
|  * Returns -1 on failure. | ||||
|  */ | ||||
| int log_enable(char *name, const char *selfkey, const char *otherkey, struct chatlog *log, int logtype); | ||||
| int log_enable(struct chatlog *log); | ||||
|  | ||||
| /* disables logging for specified log and closes file */ | ||||
| void log_disable(struct chatlog *log); | ||||
|  | ||||
| /* Loads previous history from chat log */ | ||||
| void load_chat_history(ToxWindow *self, struct chatlog *log); | ||||
| /* Loads chat log history and prints it to `self` window. | ||||
|  * | ||||
|  * Return 0 on success or if log file doesn't exist. | ||||
|  * Return -1 on failure. | ||||
|  */ | ||||
| int load_chat_history(ToxWindow *self, struct chatlog *log); | ||||
|  | ||||
| /* renames chatlog file replacing src with dest. | ||||
|    Returns 0 on success or if no log exists, -1 on failure. */ | ||||
| int rename_logfile(char *src, char *dest, const char *selfkey, const char *otherkey, int winnum); | ||||
| /* Renames chatlog file `src` to `dest`. | ||||
|  * | ||||
|  * Return 0 on success or if no log exists. | ||||
|  * Return -1 on failure. | ||||
|  */ | ||||
| int rename_logfile(const char *src, const char *dest, const char *selfkey, const char *otherkey, int winnum); | ||||
|  | ||||
| #endif /* LOG_H */ | ||||
|   | ||||
| @@ -22,12 +22,12 @@ | ||||
|  | ||||
| #include <stdlib.h> | ||||
|  | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "message_queue.h" | ||||
| #include "misc_tools.h" | ||||
| #include "line_info.h" | ||||
| #include "log.h" | ||||
| #include "message_queue.h" | ||||
| #include "misc_tools.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| void cqueue_cleanup(struct chat_queue *q) | ||||
| { | ||||
| @@ -59,7 +59,7 @@ void cqueue_add(struct chat_queue *q, const char *msg, size_t len, uint8_t type, | ||||
|     new_m->type = type; | ||||
|     new_m->line_id = line_id; | ||||
|     new_m->last_send_try = 0; | ||||
|     new_m->receipt = 0; | ||||
|     new_m->receipt = -1; | ||||
|     new_m->next = NULL; | ||||
|  | ||||
|     if (q->root == NULL) { | ||||
| @@ -86,9 +86,9 @@ static void cqueue_mark_read(ToxWindow *self, struct cqueue_msg *msg) | ||||
|  | ||||
|         line->type = msg->type == OUT_ACTION ? OUT_ACTION_READ : OUT_MSG_READ; | ||||
|  | ||||
|         if (line->noread_flag == true) { | ||||
|             line->len -= 2; | ||||
|         if (line->noread_flag) { | ||||
|             line->noread_flag = false; | ||||
|             line->read_flag = true; | ||||
|         } | ||||
|  | ||||
|         return; | ||||
| @@ -98,6 +98,7 @@ static void cqueue_mark_read(ToxWindow *self, struct cqueue_msg *msg) | ||||
| /* removes message with matching receipt from queue, writes to log and updates line to show the message was received. */ | ||||
| void cqueue_remove(ToxWindow *self, Tox *m, uint32_t receipt) | ||||
| { | ||||
|     struct chatlog *log = self->chatwin->log; | ||||
|     struct chat_queue *q = self->chatwin->cqueue; | ||||
|     struct cqueue_msg *msg = q->root; | ||||
|  | ||||
| @@ -107,13 +108,16 @@ void cqueue_remove(ToxWindow *self, Tox *m, uint32_t receipt) | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         if (log->log_on) { | ||||
|             char selfname[TOX_MAX_NAME_LENGTH]; | ||||
|             tox_self_get_name(m, (uint8_t *) selfname); | ||||
|  | ||||
|             size_t len = tox_self_get_name_size(m); | ||||
|         selfname[len] = '\0'; | ||||
|             selfname[len] = 0; | ||||
|  | ||||
|             write_to_log(msg->message, selfname, log, msg->type == OUT_ACTION); | ||||
|         } | ||||
|  | ||||
|         write_to_log(msg->message, selfname, self->chatwin->log, msg->type == OUT_ACTION); | ||||
|         cqueue_mark_read(self, msg); | ||||
|  | ||||
|         struct cqueue_msg *next = msg->next; | ||||
| @@ -123,35 +127,65 @@ void cqueue_remove(ToxWindow *self, Tox *m, uint32_t receipt) | ||||
|                 next->prev = NULL; | ||||
|             } | ||||
|  | ||||
|             free(msg); | ||||
|             q->root = next; | ||||
|         } else { | ||||
|             struct cqueue_msg *prev = msg->prev; | ||||
|             free(msg); | ||||
|             prev->next = next; | ||||
|             next->prev = prev; | ||||
|         } | ||||
|  | ||||
|         free(msg); | ||||
|  | ||||
|         return; | ||||
|     } | ||||
| } | ||||
|  | ||||
| #define CQUEUE_TRY_SEND_INTERVAL 10 | ||||
| // We use knowledge of toxcore internals (bad!) to determine that if we haven't received a read receipt for a | ||||
| // sent packet after this amount of time, the connection has been severed and the packet needs to be re-sent. | ||||
| #define TRY_SEND_TIMEOUT 32 | ||||
|  | ||||
| /* Tries to send the oldest unsent message in queue. */ | ||||
| /* | ||||
|  * Marks all timed out messages in queue as unsent. | ||||
|  */ | ||||
| static void cqueue_check_timeouts(struct cqueue_msg *msg) | ||||
| { | ||||
|     while (msg) { | ||||
|         if (timed_out(msg->last_send_try, TRY_SEND_TIMEOUT)) { | ||||
|             msg->receipt = -1; | ||||
|         } | ||||
|  | ||||
|         msg = msg->next; | ||||
|     } | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Tries to send all messages in the send queue in sequential order. | ||||
|  * If a message fails to send the function will immediately return. | ||||
|  */ | ||||
| void cqueue_try_send(ToxWindow *self, Tox *m) | ||||
| { | ||||
|     struct chat_queue *q = self->chatwin->cqueue; | ||||
|     struct cqueue_msg *msg = q->root; | ||||
|  | ||||
|     if (!msg) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (msg->receipt != 0 && !timed_out(msg->last_send_try, CQUEUE_TRY_SEND_INTERVAL)) { | ||||
|     while (msg) { | ||||
|         if (msg->receipt != -1) { | ||||
|             // we can no longer try to send unsent messages until we get receipts for our previous sent | ||||
|             // messages, but we continue to iterate the list, checking timestamps for any further | ||||
|             // successfully sent messages that have not yet gotten a receipt. | ||||
|             cqueue_check_timeouts(msg); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         TOX_ERR_FRIEND_SEND_MESSAGE err; | ||||
|         Tox_Message_Type type = msg->type == OUT_MSG ? TOX_MESSAGE_TYPE_NORMAL : TOX_MESSAGE_TYPE_ACTION; | ||||
|     msg->receipt = tox_friend_send_message(m, self->num, type, (uint8_t *) msg->message, msg->len, NULL); | ||||
|         uint32_t receipt = tox_friend_send_message(m, self->num, type, (uint8_t *) msg->message, msg->len, &err); | ||||
|  | ||||
|         if (err != TOX_ERR_FRIEND_SEND_MESSAGE_OK) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         msg->receipt = receipt; | ||||
|         msg->last_send_try = get_unix_time(); | ||||
|         msg = msg->next; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -27,9 +27,9 @@ struct cqueue_msg { | ||||
|     char message[MAX_STR_SIZE]; | ||||
|     size_t len; | ||||
|     int line_id; | ||||
|     uint8_t type; | ||||
|     uint32_t receipt; | ||||
|     time_t last_send_try; | ||||
|     uint8_t type; | ||||
|     int64_t receipt; | ||||
|     struct cqueue_msg *next; | ||||
|     struct cqueue_msg *prev; | ||||
| }; | ||||
| @@ -42,7 +42,10 @@ struct chat_queue { | ||||
| void cqueue_cleanup(struct chat_queue *q); | ||||
| void cqueue_add(struct chat_queue *q, const char *msg, size_t len, uint8_t type, int line_id); | ||||
|  | ||||
| /* Tries to send the oldest unsent message in queue. */ | ||||
| /* | ||||
|  * Tries to send all messages in the send queue in sequential order. | ||||
|  * If a message fails to send the function will immediately return. | ||||
|  */ | ||||
| void cqueue_try_send(ToxWindow *self, Tox *m); | ||||
|  | ||||
| /* removes message with matching receipt from queue, writes to log and updates line to show the message was received. */ | ||||
|   | ||||
							
								
								
									
										265
									
								
								src/misc_tools.c
									
									
									
									
									
								
							
							
						
						
									
										265
									
								
								src/misc_tools.c
									
									
									
									
									
								
							| @@ -20,39 +20,44 @@ | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <stdlib.h> | ||||
| #include <ctype.h> | ||||
| #include <string.h> | ||||
| #include <time.h> | ||||
| #include <limits.h> | ||||
| #include <dirent.h> | ||||
| #include <netinet/in.h> | ||||
| #include <sys/socket.h> | ||||
| #include <arpa/inet.h> | ||||
| #include <ctype.h> | ||||
| #include <limits.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <sys/stat.h> | ||||
| #include <time.h> | ||||
|  | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "file_transfers.h" | ||||
| #include "misc_tools.h" | ||||
| #include "settings.h" | ||||
| #include "file_transfers.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| extern ToxWindow *prompt; | ||||
| extern struct user_settings *user_settings; | ||||
|  | ||||
| void clear_screen(void) | ||||
| { | ||||
|     printf("\033[2J\033[1;1H"); | ||||
| } | ||||
|  | ||||
| void hst_to_net(uint8_t *num, uint16_t numbytes) | ||||
| { | ||||
| #ifndef WORDS_BIGENDIAN | ||||
|     uint32_t i; | ||||
|     uint8_t buff[numbytes]; | ||||
|     uint8_t *buff = malloc(numbytes); | ||||
|  | ||||
|     for (i = 0; i < numbytes; ++i) { | ||||
|     if (buff == NULL) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     for (uint32_t i = 0; i < numbytes; ++i) { | ||||
|         buff[i] = num[numbytes - i - 1]; | ||||
|     } | ||||
|  | ||||
|     memcpy(num, buff, numbytes); | ||||
|     free(buff); | ||||
| #endif | ||||
|     return; | ||||
| } | ||||
|  | ||||
| time_t get_unix_time(void) | ||||
| @@ -66,6 +71,22 @@ int timed_out(time_t timestamp, time_t timeout) | ||||
|     return timestamp + timeout <= get_unix_time(); | ||||
| } | ||||
|  | ||||
| /* Attempts to sleep the caller's thread for `usec` microseconds */ | ||||
| void sleep_thread(long int usec) | ||||
| { | ||||
|     struct timespec req; | ||||
|     struct timespec rem; | ||||
|  | ||||
|     req.tv_sec = 0; | ||||
|     req.tv_nsec = usec * 1000L; | ||||
|  | ||||
|     if (nanosleep(&req, &rem) == -1) { | ||||
|         if (nanosleep(&rem, NULL) == -1) { | ||||
|             fprintf(stderr, "nanosleep() returned -1\n"); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| /* Get the current local time */ | ||||
| struct tm *get_time(void) | ||||
| { | ||||
| @@ -75,11 +96,16 @@ struct tm *get_time(void) | ||||
|     return timeinfo; | ||||
| } | ||||
|  | ||||
| /*Puts the current time in buf in the format of [HH:mm:ss] */ | ||||
| void get_time_str(char *buf, int bufsize) | ||||
| /* Puts the current time in buf in the format of specified by the config */ | ||||
| void get_time_str(char *buf, size_t bufsize) | ||||
| { | ||||
|     if (buf == NULL || bufsize == 0) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     *buf = 0; | ||||
|  | ||||
|     if (user_settings->timestamps == TIMESTAMPS_OFF) { | ||||
|         buf[0] = '\0'; | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -137,11 +163,10 @@ int hex_string_to_bytes(char *buf, int size, const char *keystr) | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     int i, res; | ||||
|     const char *pos = keystr; | ||||
|  | ||||
|     for (i = 0; i < size; ++i) { | ||||
|         res = sscanf(pos, "%2hhx", (unsigned char *)&buf[i]); | ||||
|     for (size_t i = 0; i < size; ++i) { | ||||
|         int res = sscanf(pos, "%2hhx", (unsigned char *)&buf[i]); | ||||
|         pos += 2; | ||||
|  | ||||
|         if (res == EOF || res < 1) { | ||||
| @@ -163,15 +188,31 @@ int bin_id_to_string(const char *bin_id, size_t bin_id_size, char *output, size_ | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     size_t i; | ||||
|  | ||||
|     for (i = 0; i < TOX_ADDRESS_SIZE; ++i) { | ||||
|     for (size_t i = 0; i < TOX_ADDRESS_SIZE; ++i) { | ||||
|         snprintf(&output[i * 2], output_size - (i * 2), "%02X", bin_id[i] & 0xff); | ||||
|     } | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| /* Converts a binary representation of a Tox public key into a string. | ||||
|  * | ||||
|  * Returns 0 on success. | ||||
|  * Returns -1 on failure. | ||||
|  */ | ||||
| int bin_pubkey_to_string(const uint8_t *bin_pubkey, size_t bin_pubkey_size, char *output, size_t output_size) | ||||
| { | ||||
|     if (bin_pubkey_size != TOX_PUBLIC_KEY_SIZE || output_size < (TOX_PUBLIC_KEY_SIZE * 2 + 1)) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     for (size_t i = 0; i < TOX_PUBLIC_KEY_SIZE; ++i) { | ||||
|         snprintf(&output[i * 2], output_size - (i * 2), "%02X", bin_pubkey[i] & 0xff); | ||||
|     } | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| /* Returns 1 if the string is empty, 0 otherwise */ | ||||
| int string_is_empty(const char *string) | ||||
| { | ||||
| @@ -230,43 +271,65 @@ int qsort_strcasecmp_hlpr(const void *str1, const void *str2) | ||||
|     return strcasecmp((const char *) str1, (const char *) str2); | ||||
| } | ||||
|  | ||||
| /* Returns 1 if nick is valid, 0 if not. A valid toxic nick: | ||||
|       - cannot be empty | ||||
|       - cannot start with a space | ||||
|       - must not contain a forward slash (for logfile naming purposes) | ||||
|       - must not contain contiguous spaces | ||||
|       - must not contain a newline or tab seqeunce */ | ||||
| int valid_nick(const char *nick) | ||||
| /* case-insensitive string compare function for use with qsort */ | ||||
| int qsort_ptr_char_array_helper(const void *str1, const void *str2) | ||||
| { | ||||
|     return strcasecmp(*(char **)str1, *(char **)str2); | ||||
| } | ||||
|  | ||||
| static const char invalid_chars[] = {'/', '\n', '\t', '\v', '\r', '\0'}; | ||||
|  | ||||
| /* | ||||
|  * Helper function for `valid_nick()`. | ||||
|  * | ||||
|  * Returns true if `ch` is not in the `invalid_chars` array. | ||||
|  */ | ||||
| static bool is_valid_char(char ch) | ||||
| { | ||||
|     char tmp; | ||||
|  | ||||
|     for (size_t i = 0; (tmp = invalid_chars[i]); ++i) { | ||||
|         if (tmp == ch) { | ||||
|             return false; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| /* Returns true if nick is valid. | ||||
|  * | ||||
|  * A valid toxic nick: | ||||
|  * - cannot be empty | ||||
|  * - cannot start with a space | ||||
|  * - must not contain a forward slash (for logfile naming purposes) | ||||
|  * - must not contain contiguous spaces | ||||
|  * - must not contain a newline or tab seqeunce | ||||
|  */ | ||||
| bool valid_nick(const char *nick) | ||||
| { | ||||
|     if (!nick[0] || nick[0] == ' ') { | ||||
|         return 0; | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     int i; | ||||
|     for (size_t i = 0; nick[i]; ++i) { | ||||
|         char ch = nick[i]; | ||||
|  | ||||
|     for (i = 0; nick[i]; ++i) { | ||||
|         if ((nick[i] == ' ' && nick[i + 1] == ' ') | ||||
|                 || nick[i] == '/' | ||||
|                 || nick[i] == '\n' | ||||
|                 || nick[i] == '\t' | ||||
|                 || nick[i] == '\v' | ||||
|                 || nick[i] == '\r') | ||||
|  | ||||
|         { | ||||
|             return 0; | ||||
|         if ((ch == ' ' && nick[i + 1] == ' ') || !is_valid_char(ch)) { | ||||
|             return false; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return 1; | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| /* Converts all newline/tab chars to spaces (use for strings that should be contained to a single line) */ | ||||
| void filter_str(char *str, size_t len) | ||||
| { | ||||
|     size_t i; | ||||
|     for (size_t i = 0; i < len; ++i) { | ||||
|         char ch = str[i]; | ||||
|  | ||||
|     for (i = 0; i < len; ++i) { | ||||
|         if (str[i] == '\n' || str[i] == '\r' || str[i] == '\t' || str[i] == '\v' || str[i] == '\0') { | ||||
|         if (!is_valid_char(ch) || str[i] == '\0') { | ||||
|             str[i] = ' '; | ||||
|         } | ||||
|     } | ||||
| @@ -370,16 +433,16 @@ on_error: | ||||
|     return len; | ||||
| } | ||||
|  | ||||
| /* same as get_nick_truncate but for groupchats */ | ||||
| int get_group_nick_truncate(Tox *m, char *buf, uint32_t peernum, uint32_t groupnum) | ||||
| /* same as get_nick_truncate but for conferences */ | ||||
| int get_conference_nick_truncate(Tox *m, char *buf, uint32_t peernum, uint32_t conferencenum) | ||||
| { | ||||
|     Tox_Err_Conference_Peer_Query err; | ||||
|     size_t len = tox_conference_peer_get_name_size(m, groupnum, peernum, &err); | ||||
|     size_t len = tox_conference_peer_get_name_size(m, conferencenum, peernum, &err); | ||||
|  | ||||
|     if (err != TOX_ERR_CONFERENCE_PEER_QUERY_OK) { | ||||
|         goto on_error; | ||||
|     } else { | ||||
|         if (!tox_conference_peer_get_name(m, groupnum, peernum, (uint8_t *) buf, NULL)) { | ||||
|         if (!tox_conference_peer_get_name(m, conferencenum, peernum, (uint8_t *) buf, NULL)) { | ||||
|             goto on_error; | ||||
|         } | ||||
|     } | ||||
| @@ -491,13 +554,18 @@ bool file_exists(const char *path) | ||||
| File_Type file_type(const char *path) | ||||
| { | ||||
|     struct stat s; | ||||
|     stat(path, &s); | ||||
|  | ||||
|     if (stat(path, &s) == -1) { | ||||
|         return FILE_TYPE_OTHER; | ||||
|     } | ||||
|  | ||||
|     switch (s.st_mode & S_IFMT) { | ||||
|         case S_IFDIR: | ||||
|             return FILE_TYPE_DIRECTORY; | ||||
|  | ||||
|         case S_IFREG: | ||||
|             return FILE_TYPE_REGULAR; | ||||
|  | ||||
|         default: | ||||
|             return FILE_TYPE_OTHER; | ||||
|     } | ||||
| @@ -515,34 +583,17 @@ off_t file_size(const char *path) | ||||
|     return st.st_size; | ||||
| } | ||||
|  | ||||
| /* compares the first size bytes of fp to signature. | ||||
|    Returns 0 if they are the same, 1 if they differ, and -1 on error. | ||||
|  | ||||
|    On success this function will seek back to the beginning of fp */ | ||||
| int check_file_signature(const char *signature, size_t size, FILE *fp) | ||||
| { | ||||
|     char buf[size]; | ||||
|  | ||||
|     if (fread(buf, size, 1, fp) != 1) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     int ret = memcmp(signature, buf, size); | ||||
|  | ||||
|     if (fseek(fp, 0L, SEEK_SET) == -1) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     return ret == 0 ? 0 : 1; | ||||
| } | ||||
|  | ||||
| /* sets window title in tab bar. */ | ||||
| void set_window_title(ToxWindow *self, const char *title, int len) | ||||
| { | ||||
|     if (len <= 0 || !title) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     char cpy[TOXIC_MAX_NAME_LENGTH + 1]; | ||||
|  | ||||
|     if (self->is_groupchat) { /* keep groupnumber in title */ | ||||
|         snprintf(cpy, sizeof(cpy), "%d %s", self->num, title); | ||||
|     if (self->type == WINDOW_TYPE_CONFERENCE) { /* keep conferencenumber in title for invites */ | ||||
|         snprintf(cpy, sizeof(cpy), "%u %s", self->num, title); | ||||
|     } else { | ||||
|         snprintf(cpy, sizeof(cpy), "%s", title); | ||||
|     } | ||||
| @@ -555,35 +606,49 @@ void set_window_title(ToxWindow *self, const char *title, int len) | ||||
|     snprintf(self->name, sizeof(self->name), "%s", cpy); | ||||
| } | ||||
|  | ||||
| /* Return true if address appears to be a valid ipv4 address. */ | ||||
| bool is_ip4_address(const char *address) | ||||
| { | ||||
|     struct sockaddr_in s_addr; | ||||
|     return inet_pton(AF_INET, address, &(s_addr.sin_addr)) != 0; | ||||
| } | ||||
|  | ||||
| /* Return true if address roughly appears to be a valid ipv6 address. | ||||
|  * | ||||
|  * TODO: Improve this function (inet_pton behaves strangely with ipv6). | ||||
|  * for now the only guarantee is that it won't return true if the | ||||
|  * address is a domain or ipv4 address, and should only be used if you're | ||||
|  * reasonably sure that the address is one of the three (ipv4, ipv6 or a domain). | ||||
| /* | ||||
|  * Frees all members of a pointer array plus `arr`. | ||||
|  */ | ||||
| bool is_ip6_address(const char *address) | ||||
| void free_ptr_array(void **arr) | ||||
| { | ||||
|     size_t i; | ||||
|     size_t num_colons = 0; | ||||
|     char ch = 0; | ||||
|  | ||||
|     for (i = 0; (ch = address[i]); ++i) { | ||||
|         if (ch == '.') { | ||||
|             return false; | ||||
|     if (arr == NULL) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|         if (ch == ':') { | ||||
|             ++num_colons; | ||||
|         } | ||||
|     void **tmp = arr; | ||||
|  | ||||
|     while (*arr) { | ||||
|         free(*arr); | ||||
|         ++arr; | ||||
|     } | ||||
|  | ||||
|     return num_colons > 1 && num_colons < 8; | ||||
|     free(tmp); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Returns a null terminated array of `length` pointers. Each pointer is allocated `bytes` bytes. | ||||
|  * Returns NULL on failure. | ||||
|  * | ||||
|  * The caller is responsible for freeing the array with `free_ptr_array`. | ||||
|  */ | ||||
| void **malloc_ptr_array(size_t length, size_t bytes) | ||||
| { | ||||
|     void **arr = malloc((length + 1) * sizeof(void *)); | ||||
|  | ||||
|     if (arr == NULL) { | ||||
|         return NULL; | ||||
|     } | ||||
|  | ||||
|     for (size_t i = 0; i < length; ++i) { | ||||
|         arr[i] = malloc(bytes); | ||||
|  | ||||
|         if (arr[i] == NULL) { | ||||
|             free_ptr_array(arr); | ||||
|             return NULL; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     arr[length] = NULL; | ||||
|  | ||||
|     return arr; | ||||
| } | ||||
|   | ||||
| @@ -24,8 +24,8 @@ | ||||
|  | ||||
| #include <sys/stat.h> | ||||
|  | ||||
| #include "windows.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| #ifndef MIN | ||||
| #define MIN(x, y) (((x) < (y)) ? (x) : (y)) | ||||
| @@ -39,14 +39,17 @@ | ||||
| #define net_to_host(x, y) hst_to_net(x, y) | ||||
| #endif | ||||
|  | ||||
| typedef enum File_Type | ||||
| { | ||||
| #define UNUSED_VAR(x) ((void) x) | ||||
|  | ||||
| typedef enum File_Type { | ||||
|     FILE_TYPE_REGULAR, | ||||
|     FILE_TYPE_DIRECTORY, | ||||
|     FILE_TYPE_OTHER, | ||||
| } File_Type; | ||||
|  | ||||
|  | ||||
| void clear_screen(void); | ||||
|  | ||||
| void hst_to_net(uint8_t *num, uint16_t numbytes); | ||||
|  | ||||
| /* | ||||
| @@ -68,11 +71,18 @@ int hex_string_to_bytes(char *buf, int size, const char *keystr); | ||||
|  */ | ||||
| int bin_id_to_string(const char *bin_id, size_t bin_id_size, char *output, size_t output_size); | ||||
|  | ||||
| /* Converts a binary representation of a Tox public key into a string. | ||||
|  * | ||||
|  * Returns 0 on success. | ||||
|  * Returns -1 on failure. | ||||
|  */ | ||||
| int bin_pubkey_to_string(const uint8_t *bin_pubkey, size_t bin_pubkey_size, char *output, size_t output_size); | ||||
|  | ||||
| /* get the current unix time (not thread safe) */ | ||||
| time_t get_unix_time(void); | ||||
|  | ||||
| /* Puts the current time in buf in the format of [HH:mm:ss] (not thread safe) */ | ||||
| void get_time_str(char *buf, int bufsize); | ||||
| /* Puts the current time in buf in the format of specified by the config */ | ||||
| void get_time_str(char *buf, size_t bufsize); | ||||
|  | ||||
| /* Converts seconds to string in format HH:mm:ss; truncates hours and minutes when necessary */ | ||||
| void get_elapsed_time_str(char *buf, int bufsize, time_t secs); | ||||
| @@ -98,19 +108,28 @@ int mbs_to_wcs_buf(wchar_t *buf, const char *string, size_t n); | ||||
| /* Returns 1 if connection has timed out, 0 otherwise */ | ||||
| int timed_out(time_t timestamp, time_t timeout); | ||||
|  | ||||
| /* Attempts to sleep the caller's thread for `usec` microseconds */ | ||||
| void sleep_thread(long int usec); | ||||
|  | ||||
| /* Colours the window tab according to type. Beeps if is_beep is true */ | ||||
| void alert_window(ToxWindow *self, int type, bool is_beep); | ||||
|  | ||||
| /* case-insensitive string compare function for use with qsort */ | ||||
| int qsort_strcasecmp_hlpr(const void *str1, const void *str2); | ||||
|  | ||||
| /* Returns 1 if nick is valid, 0 if not. A valid toxic nick: | ||||
|       - cannot be empty | ||||
|       - cannot start with a space | ||||
|       - must not contain a forward slash (for logfile naming purposes) | ||||
|       - must not contain contiguous spaces | ||||
|       - must not contain a newline or tab seqeunce */ | ||||
| int valid_nick(const char *nick); | ||||
| /* case-insensitive string compare function for use with qsort */ | ||||
| int qsort_ptr_char_array_helper(const void *str1, const void *str2); | ||||
|  | ||||
| /* Returns true if nick is valid. | ||||
|  * | ||||
|  * A valid toxic nick: | ||||
|  * - cannot be empty | ||||
|  * - cannot start with a space | ||||
|  * - must not contain a forward slash (for logfile naming purposes) | ||||
|  * - must not contain contiguous spaces | ||||
|  * - must not contain a newline or tab seqeunce | ||||
|  */ | ||||
| bool valid_nick(const char *nick); | ||||
|  | ||||
| /* Converts all newline/tab chars to spaces (use for strings that should be contained to a single line) */ | ||||
| void filter_str(char *str, size_t len); | ||||
| @@ -133,8 +152,8 @@ void str_to_lower(char *str); | ||||
|    Returns nick len on success, -1 on failure */ | ||||
| size_t get_nick_truncate(Tox *m, char *buf, uint32_t friendnum); | ||||
|  | ||||
| /* same as get_nick_truncate but for groupchats */ | ||||
| int get_group_nick_truncate(Tox *m, char *buf, uint32_t peernum, uint32_t groupnum); | ||||
| /* same as get_nick_truncate but for conferences */ | ||||
| int get_conference_nick_truncate(Tox *m, char *buf, uint32_t peernum, uint32_t conferencenum); | ||||
|  | ||||
| /* copies data to msg buffer. | ||||
|    returns length of msg, which will be no larger than size-1 */ | ||||
| @@ -166,25 +185,20 @@ File_Type file_type(const char *path); | ||||
| /* returns file size. If file doesn't exist returns 0. */ | ||||
| off_t file_size(const char *path); | ||||
|  | ||||
| /* compares the first size bytes of fp and signature. | ||||
|    Returns 0 if they are the same, 1 if they differ, and -1 on error. | ||||
|  | ||||
|    On success this function will seek back to the beginning of fp */ | ||||
| int check_file_signature(const char *signature, size_t size, FILE *fp); | ||||
|  | ||||
| /* sets window title in tab bar. */ | ||||
| void set_window_title(ToxWindow *self, const char *title, int len); | ||||
|  | ||||
| /* Return true if address appears to be a valid ipv4 address. */ | ||||
| bool is_ip4_address(const char *address); | ||||
|  | ||||
| /* Return true if address roughly appears to be a valid ipv6 address. | ||||
|  * | ||||
|  * TODO: Improve this function (inet_pton behaves strangely with ipv6). | ||||
|  * for now the only guarantee is that it won't return true if the | ||||
|  * address is a domain or ipv4 address, and should only be used if you're | ||||
|  * reasonably sure that the address is one of the three (ipv4, ipv6 or a domain). | ||||
| /* | ||||
|  * Frees all members of a pointer array plus `arr`. | ||||
|  */ | ||||
| bool is_ip6_address(const char *address); | ||||
| void free_ptr_array(void **arr); | ||||
|  | ||||
| /* | ||||
|  * Returns a null terminated array of `length` pointers. Each pointer is allocated `bytes` bytes. | ||||
|  * Returns NULL on failure. | ||||
|  * | ||||
|  * The caller is responsible for freeing the array with `free_ptr_array`. | ||||
|  */ | ||||
| void **malloc_ptr_array(size_t length, size_t bytes); | ||||
|  | ||||
| #endif /* MISC_TOOLS_H */ | ||||
|   | ||||
| @@ -20,18 +20,18 @@ | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <stdlib.h> | ||||
| #include <stdarg.h> | ||||
| #include <string.h> | ||||
| #include <curl/curl.h> | ||||
| #include <stdarg.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
|  | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "line_info.h" | ||||
| #include "global_commands.h" | ||||
| #include "misc_tools.h" | ||||
| #include "configdir.h" | ||||
| #include "curl_util.h" | ||||
| #include "global_commands.h" | ||||
| #include "line_info.h" | ||||
| #include "misc_tools.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| extern struct arg_opts arg_opts; | ||||
| extern struct Winthread Winthread; | ||||
| @@ -63,6 +63,13 @@ static struct lookup_thread { | ||||
|     pthread_attr_t attr; | ||||
| } lookup_thread; | ||||
|  | ||||
| static void clear_thread_data(void) | ||||
| { | ||||
|     t_data = (struct thread_data) { | ||||
|         0 | ||||
|     }; | ||||
| } | ||||
|  | ||||
| static int lookup_error(ToxWindow *self, const char *errmsg, ...) | ||||
| { | ||||
|     char frmt_msg[MAX_STR_SIZE]; | ||||
| @@ -73,7 +80,7 @@ static int lookup_error(ToxWindow *self, const char *errmsg, ...) | ||||
|     va_end(args); | ||||
|  | ||||
|     pthread_mutex_lock(&Winthread.lock); | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "name lookup failed: %s", frmt_msg); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "name lookup failed: %s", frmt_msg); | ||||
|     pthread_mutex_unlock(&Winthread.lock); | ||||
|  | ||||
|     return -1; | ||||
| @@ -81,7 +88,7 @@ static int lookup_error(ToxWindow *self, const char *errmsg, ...) | ||||
|  | ||||
| static void kill_lookup_thread(void) | ||||
| { | ||||
|     memset(&t_data, 0, sizeof(struct thread_data)); | ||||
|     clear_thread_data(); | ||||
|     pthread_attr_destroy(&lookup_thread.attr); | ||||
|     pthread_exit(NULL); | ||||
| } | ||||
| @@ -233,6 +240,8 @@ static int process_response(struct Recv_Curl_Data *recv_data) | ||||
|  | ||||
| void *lookup_thread_func(void *data) | ||||
| { | ||||
|     UNUSED_VAR(data); | ||||
|  | ||||
|     ToxWindow *self = t_data.self; | ||||
|  | ||||
|     char input_domain[MAX_STR_SIZE]; | ||||
| @@ -263,11 +272,14 @@ void *lookup_thread_func(void *data) | ||||
|         kill_lookup_thread(); | ||||
|     } | ||||
|  | ||||
|     struct Recv_Curl_Data recv_data; | ||||
|     struct Recv_Curl_Data *recv_data = calloc(1, sizeof(struct Recv_Curl_Data)); | ||||
|  | ||||
|     memset(&recv_data, 0, sizeof(struct Recv_Curl_Data)); | ||||
|     if (recv_data == NULL) { | ||||
|         lookup_error(self, "memory allocation error"); | ||||
|         kill_lookup_thread(); | ||||
|     } | ||||
|  | ||||
|     char post_data[MAX_STR_SIZE]; | ||||
|     char post_data[MAX_STR_SIZE + 30]; | ||||
|  | ||||
|     snprintf(post_data, sizeof(post_data), "{\"action\": 3, \"name\": \"%s\"}", name); | ||||
|  | ||||
| @@ -283,7 +295,7 @@ void *lookup_thread_func(void *data) | ||||
|  | ||||
|     curl_easy_setopt(c_handle, CURLOPT_WRITEFUNCTION, curl_cb_write_data); | ||||
|  | ||||
|     curl_easy_setopt(c_handle, CURLOPT_WRITEDATA, &recv_data); | ||||
|     curl_easy_setopt(c_handle, CURLOPT_WRITEDATA, recv_data); | ||||
|  | ||||
|     curl_easy_setopt(c_handle, CURLOPT_USERAGENT, "libcurl-agent/1.0"); | ||||
|  | ||||
| @@ -292,7 +304,7 @@ void *lookup_thread_func(void *data) | ||||
|     int proxy_ret = set_curl_proxy(c_handle, arg_opts.proxy_address, arg_opts.proxy_port, arg_opts.proxy_type); | ||||
|  | ||||
|     if (proxy_ret != 0) { | ||||
|         lookup_error(self, "Failed to set proxy (error %d)\n"); | ||||
|         lookup_error(self, "Failed to set proxy (error %d)\n", proxy_ret); | ||||
|         goto on_exit; | ||||
|     } | ||||
|  | ||||
| @@ -332,7 +344,7 @@ void *lookup_thread_func(void *data) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (process_response(&recv_data) == -1) { | ||||
|     if (process_response(recv_data) == -1) { | ||||
|         lookup_error(self, "Bad response."); | ||||
|         goto on_exit; | ||||
|     } | ||||
| @@ -342,6 +354,7 @@ void *lookup_thread_func(void *data) | ||||
|     pthread_mutex_unlock(&Winthread.lock); | ||||
|  | ||||
| on_exit: | ||||
|     free(recv_data); | ||||
|     curl_slist_free_all(headers); | ||||
|     curl_easy_cleanup(c_handle); | ||||
|     kill_lookup_thread(); | ||||
| @@ -352,12 +365,12 @@ on_exit: | ||||
| void name_lookup(ToxWindow *self, Tox *m, const char *id_bin, const char *addr, const char *message) | ||||
| { | ||||
|     if (t_data.disabled) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "name lookups are disabled."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "name lookups are disabled."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (t_data.busy) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Please wait for previous name lookup to finish."); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Please wait for previous name lookup to finish."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -369,22 +382,22 @@ void name_lookup(ToxWindow *self, Tox *m, const char *id_bin, const char *addr, | ||||
|     t_data.busy = true; | ||||
|  | ||||
|     if (pthread_attr_init(&lookup_thread.attr) != 0) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, RED, "Error: lookup thread attr failed to init"); | ||||
|         memset(&t_data, 0, sizeof(struct thread_data)); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, RED, "Error: lookup thread attr failed to init"); | ||||
|         clear_thread_data(); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (pthread_attr_setdetachstate(&lookup_thread.attr, PTHREAD_CREATE_DETACHED) != 0) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, RED, "Error: lookup thread attr failed to set"); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, RED, "Error: lookup thread attr failed to set"); | ||||
|         pthread_attr_destroy(&lookup_thread.attr); | ||||
|         memset(&t_data, 0, sizeof(struct thread_data)); | ||||
|         clear_thread_data(); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (pthread_create(&lookup_thread.tid, &lookup_thread.attr, lookup_thread_func, NULL) != 0) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, RED, "Error: lookup thread failed to init"); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, RED, "Error: lookup thread failed to init"); | ||||
|         pthread_attr_destroy(&lookup_thread.attr); | ||||
|         memset(&t_data, 0, sizeof(struct thread_data)); | ||||
|         clear_thread_data(); | ||||
|         return; | ||||
|     } | ||||
| } | ||||
|   | ||||
							
								
								
									
										147
									
								
								src/notify.c
									
									
									
									
									
								
							
							
						
						
									
										147
									
								
								src/notify.c
									
									
									
									
									
								
							| @@ -20,22 +20,22 @@ | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <assert.h> | ||||
| #include <stdarg.h> | ||||
| #include <stdbool.h> | ||||
| #include <stdio.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <stdio.h> | ||||
| #include <stdbool.h> | ||||
| #include <unistd.h> | ||||
| #include <stdarg.h> | ||||
| #include <time.h> | ||||
| #include <assert.h> | ||||
| #include <sys/stat.h> | ||||
| #include <time.h> | ||||
| #include <unistd.h> | ||||
|  | ||||
| #include "notify.h" | ||||
| #include "audio_device.h" | ||||
| #include "settings.h" | ||||
| #include "line_info.h" | ||||
| #include "misc_tools.h" | ||||
| #include "xtra.h" | ||||
| #include "notify.h" | ||||
| #include "settings.h" | ||||
| #include "x11focus.h" | ||||
|  | ||||
| #if defined(AUDIO) || defined(SOUND_NOTIFY) | ||||
| #ifdef __APPLE__ | ||||
| @@ -60,7 +60,7 @@ | ||||
|  | ||||
| #define MAX_BOX_MSG_LEN 127 | ||||
| #define SOUNDS_SIZE 10 | ||||
| #define ACTIVE_NOTIFS_MAX 50 | ||||
| #define ACTIVE_NOTIFS_MAX 10 | ||||
|  | ||||
| extern struct user_settings *user_settings; | ||||
|  | ||||
| @@ -101,6 +101,17 @@ static struct _ActiveNotifications { | ||||
| /**********************************************************************************/ | ||||
| /**********************************************************************************/ | ||||
|  | ||||
| static void clear_actives_index(size_t idx) | ||||
| { | ||||
|     if (actives[idx].id_indicator) { | ||||
|         *actives[idx].id_indicator = -1; | ||||
|     } | ||||
|  | ||||
|     actives[idx] = (struct _ActiveNotifications) { | ||||
|         0 | ||||
|     }; | ||||
| } | ||||
|  | ||||
| /* coloured tab notifications: primary notification type */ | ||||
| static void tab_notify(ToxWindow *self, uint64_t flags) | ||||
| { | ||||
| @@ -115,6 +126,8 @@ static void tab_notify(ToxWindow *self, uint64_t flags) | ||||
|     } else if ((flags & NT_WNDALERT_2) && (!self->alert || self->alert > WINDOW_ALERT_1)) { | ||||
|         self->alert = WINDOW_ALERT_2; | ||||
|     } | ||||
|  | ||||
|     ++self->pending_messages; | ||||
| } | ||||
|  | ||||
| static bool notifications_are_disabled(uint64_t flags) | ||||
| @@ -169,7 +182,7 @@ void m_open_device(void) | ||||
|     } | ||||
|  | ||||
|     /* Blah error check */ | ||||
|     open_primary_device(output, &Control.device_idx, 48000, 20, 1); | ||||
|     open_output_device(&Control.device_idx, 48000, 20, 1); | ||||
|  | ||||
|     device_opened = true; | ||||
| } | ||||
| @@ -193,7 +206,7 @@ void graceful_clear(void) | ||||
|     while (1) { | ||||
|         int i; | ||||
|  | ||||
|         for (i = 0; i < ACTIVE_NOTIFS_MAX; i ++) { | ||||
|         for (i = 0; i < ACTIVE_NOTIFS_MAX; ++i) { | ||||
|             if (actives[i].active) { | ||||
| #ifdef BOX_NOTIFY | ||||
|  | ||||
| @@ -213,7 +226,7 @@ void graceful_clear(void) | ||||
|                     stop_sound(i); | ||||
|                 } else { | ||||
|                     if (!is_playing(actives[i].source)) { | ||||
|                         memset(&actives[i], 0, sizeof(struct _ActiveNotifications)); | ||||
|                         clear_actives_index(i); | ||||
|                     } else { | ||||
|                         break; | ||||
|                     } | ||||
| @@ -227,7 +240,7 @@ void graceful_clear(void) | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         usleep(1000); | ||||
|         sleep_thread(1000L); | ||||
|     } | ||||
|  | ||||
|     control_unlock(); | ||||
| @@ -235,7 +248,7 @@ void graceful_clear(void) | ||||
|  | ||||
| void *do_playing(void *_p) | ||||
| { | ||||
|     (void)_p; | ||||
|     UNUSED_VAR(_p); | ||||
|  | ||||
|     while (true) { | ||||
|         control_lock(); | ||||
| @@ -249,7 +262,7 @@ void *do_playing(void *_p) | ||||
|         bool test_active_notify = false; | ||||
|         int i; | ||||
|  | ||||
|         for (i = 0; i < ACTIVE_NOTIFS_MAX; i ++) { | ||||
|         for (i = 0; i < ACTIVE_NOTIFS_MAX; ++i) { | ||||
|  | ||||
|             if (actives[i].looping) { | ||||
|                 has_looping = true; | ||||
| @@ -270,7 +283,7 @@ void *do_playing(void *_p) | ||||
|                     alSourceStop(actives[i].source); | ||||
|                     alDeleteSources(1, &actives[i].source); | ||||
|                     alDeleteBuffers(1, &actives[i].buffer); | ||||
|                     memset(&actives[i], 0, sizeof(struct _ActiveNotifications)); | ||||
|                     clear_actives_index(i); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
| @@ -289,7 +302,7 @@ void *do_playing(void *_p) | ||||
|                     alSourceStop(actives[i].source); | ||||
|                     alDeleteSources(1, &actives[i].source); | ||||
|                     alDeleteBuffers(1, &actives[i].buffer); | ||||
|                     memset(&actives[i], 0, sizeof(struct _ActiveNotifications)); | ||||
|                     clear_actives_index(i); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
| @@ -305,7 +318,7 @@ void *do_playing(void *_p) | ||||
|         has_looping = false; | ||||
|  | ||||
|         control_unlock(); | ||||
|         usleep(10000); | ||||
|         sleep_thread(10000L); | ||||
|     } | ||||
|  | ||||
|     pthread_exit(NULL); | ||||
| @@ -315,7 +328,7 @@ int play_source(uint32_t source, uint32_t buffer, bool looping) | ||||
| { | ||||
|     int i = 0; | ||||
|  | ||||
|     for (; i < ACTIVE_NOTIFS_MAX && actives[i].active; i ++); | ||||
|     for (; i < ACTIVE_NOTIFS_MAX && actives[i].active; ++i); | ||||
|  | ||||
|     if (i == ACTIVE_NOTIFS_MAX) { | ||||
|         return -1; /* Full */ | ||||
| @@ -334,7 +347,7 @@ int play_source(uint32_t source, uint32_t buffer, bool looping) | ||||
| #elif BOX_NOTIFY | ||||
| void *do_playing(void *_p) | ||||
| { | ||||
|     (void)_p; | ||||
|     UNUSED_VAR(_p); | ||||
|  | ||||
|     while (true) { | ||||
|         control_lock(); | ||||
| @@ -344,24 +357,16 @@ void *do_playing(void *_p) | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         int i; | ||||
|  | ||||
|         for (i = 0; i < ACTIVE_NOTIFS_MAX; i ++) { | ||||
|         for (size_t i = 0; i < ACTIVE_NOTIFS_MAX; ++i) { | ||||
|             if (actives[i].box && time(NULL) >= actives[i].n_timeout) { | ||||
|                 GError *ignore; | ||||
|                 notify_notification_close(actives[i].box, &ignore); | ||||
|                 actives[i].box = NULL; | ||||
|  | ||||
|                 if (actives[i].id_indicator) { | ||||
|                     *actives[i].id_indicator = -1;    /* reset indicator value */ | ||||
|                 } | ||||
|  | ||||
|                 memset(&actives[i], 0, sizeof(struct _ActiveNotifications)); | ||||
|                 clear_actives_index(i); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         control_unlock(); | ||||
|         usleep(10000); | ||||
|         sleep_thread(10000L); | ||||
|     } | ||||
|  | ||||
|     pthread_exit(NULL); | ||||
| @@ -369,27 +374,48 @@ void *do_playing(void *_p) | ||||
|  | ||||
| void graceful_clear(void) | ||||
| { | ||||
|     int i; | ||||
|     control_lock(); | ||||
|  | ||||
|     for (i = 0; i < ACTIVE_NOTIFS_MAX; i ++) { | ||||
|     for (size_t i = 0; i < ACTIVE_NOTIFS_MAX; ++i) { | ||||
|         if (actives[i].box) { | ||||
|             GError *ignore; | ||||
|             notify_notification_close(actives[i].box, &ignore); | ||||
|             actives[i].box = NULL; | ||||
|         } | ||||
|  | ||||
|         if (actives[i].id_indicator) { | ||||
|             *actives[i].id_indicator = -1;    /* reset indicator value */ | ||||
|         } | ||||
|  | ||||
|         memset(&actives[i], 0, sizeof(struct _ActiveNotifications)); | ||||
|         clear_actives_index(i); | ||||
|     } | ||||
|  | ||||
|     control_unlock(); | ||||
| } | ||||
|  | ||||
| #endif /* SOUND_NOTIFY */ | ||||
|  | ||||
| /* Kills all notifications for `id`. This must be called before freeing a ToxWindow. */ | ||||
| void kill_notifs(int id) | ||||
| { | ||||
|     control_lock(); | ||||
|  | ||||
|     for (size_t i = 0; i < ACTIVE_NOTIFS_MAX; ++i) { | ||||
|         if (!actives[i].id_indicator) { | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         if (*actives[i].id_indicator == id) { | ||||
| #ifdef BOX_NOTIFY | ||||
|  | ||||
|             if (actives[i].box) { | ||||
|                 GError *ignore; | ||||
|                 notify_notification_close(actives[i].box, &ignore); | ||||
|             } | ||||
|  | ||||
| #endif // BOX_NOTIFY | ||||
|             clear_actives_index(i); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     control_unlock(); | ||||
| } | ||||
|  | ||||
| /**********************************************************************************/ | ||||
| /**********************************************************************************/ | ||||
| /**********************************************************************************/ | ||||
| @@ -450,7 +476,7 @@ void terminate_notify(void) | ||||
| #ifdef SOUND_NOTIFY | ||||
|     int i = 0; | ||||
|  | ||||
|     for (; i < SOUNDS_SIZE; i ++) { | ||||
|     for (; i < SOUNDS_SIZE; ++i) { | ||||
|         free(Control.sounds[i]); | ||||
|     } | ||||
|  | ||||
| @@ -523,7 +549,6 @@ int play_notify_sound(Notification notif, uint64_t flags) | ||||
|     return rc; | ||||
| } | ||||
|  | ||||
|  | ||||
| void stop_sound(int id) | ||||
| { | ||||
|     if (id >= 0 && id < ACTIVE_NOTIFS_MAX && actives[id].looping && actives[id].active) { | ||||
| @@ -536,15 +561,11 @@ void stop_sound(int id) | ||||
|  | ||||
| #endif /* BOX_NOTIFY */ | ||||
|  | ||||
|         if (actives[id].id_indicator) { | ||||
|             *actives[id].id_indicator = -1; | ||||
|         } | ||||
|  | ||||
|         // alSourcei(actives[id].source, AL_LOOPING, false); | ||||
|         alSourceStop(actives[id].source); | ||||
|         alDeleteSources(1, &actives[id].source); | ||||
|         alDeleteBuffers(1, &actives[id].buffer); | ||||
|         memset(&actives[id], 0, sizeof(struct _ActiveNotifications)); | ||||
|         clear_actives_index(id); | ||||
|     } | ||||
| } | ||||
| #endif /* SOUND_NOTIFY */ | ||||
| @@ -754,20 +775,18 @@ int box_notify2(ToxWindow *self, Notification notif, uint64_t flags, int id, con | ||||
|     actives[id].size++; | ||||
|     actives[id].n_timeout = get_unix_time() + Control.notif_timeout / 1000; | ||||
|  | ||||
|     char formated[128 * 129] = {'\0'}; | ||||
|     char *formatted = calloc(1, sizeof(char) * ((MAX_BOX_MSG_LEN + 1) * (MAX_BOX_MSG_LEN + 2))); | ||||
|  | ||||
|     int i = 0; | ||||
|  | ||||
|     for (; i < actives[id].size; i ++) { | ||||
|         strcat(formated, actives[id].messages[i]); | ||||
|         strcat(formated, "\n"); | ||||
|     for (size_t i = 0; i < actives[id].size; ++i) { | ||||
|         strcat(formatted, actives[id].messages[i]); | ||||
|         strcat(formatted, "\n"); | ||||
|     } | ||||
|  | ||||
|     formated[strlen(formated) - 1] = '\0'; | ||||
|  | ||||
|     notify_notification_update(actives[id].box, actives[id].title, formated, NULL); | ||||
|     notify_notification_update(actives[id].box, actives[id].title, formatted, NULL); | ||||
|     notify_notification_show(actives[id].box, NULL); | ||||
|  | ||||
|     free(formatted); | ||||
|  | ||||
|     control_unlock(); | ||||
|  | ||||
|     return id; | ||||
| @@ -863,20 +882,18 @@ int box_silent_notify2(ToxWindow *self, uint64_t flags, int id, const char *form | ||||
|     actives[id].size ++; | ||||
|     actives[id].n_timeout = get_unix_time() + Control.notif_timeout / 1000; | ||||
|  | ||||
|     char formated[128 * 129] = {'\0'}; | ||||
|     char *formatted = calloc(1, sizeof(char) * ((MAX_BOX_MSG_LEN + 1) * (MAX_BOX_MSG_LEN + 2))); | ||||
|  | ||||
|     int i = 0; | ||||
|  | ||||
|     for (; i < actives[id].size; i ++) { | ||||
|         strcat(formated, actives[id].messages[i]); | ||||
|         strcat(formated, "\n"); | ||||
|     for (size_t i = 0; i < actives[id].size; ++i) { | ||||
|         strcat(formatted, actives[id].messages[i]); | ||||
|         strcat(formatted, "\n"); | ||||
|     } | ||||
|  | ||||
|     formated[strlen(formated) - 1] = '\0'; | ||||
|  | ||||
|     notify_notification_update(actives[id].box, actives[id].title, formated, NULL); | ||||
|     notify_notification_update(actives[id].box, actives[id].title, formatted, NULL); | ||||
|     notify_notification_show(actives[id].box, NULL); | ||||
|  | ||||
|     free(formatted); | ||||
|  | ||||
|     control_unlock(); | ||||
|  | ||||
|     return id; | ||||
|   | ||||
| @@ -23,7 +23,7 @@ | ||||
| #ifndef NOTIFY_H | ||||
| #define NOTIFY_H | ||||
|  | ||||
| #include <inttypes.h> | ||||
| #include <stdint.h> | ||||
| #include "windows.h" | ||||
|  | ||||
| typedef enum _Notification { | ||||
| @@ -62,6 +62,9 @@ typedef enum _Flags { | ||||
| int init_notify(int login_cooldown, int notification_timeout); | ||||
| void terminate_notify(void); | ||||
|  | ||||
| /* Kills all notifications for `id`. This must be called before freeing a ToxWindow. */ | ||||
| void kill_notifs(int id); | ||||
|  | ||||
| int sound_notify(ToxWindow *self, Notification notif, uint64_t flags, int *id_indicator); | ||||
| int sound_notify2(ToxWindow *self, Notification notif, uint64_t flags, int id); | ||||
|  | ||||
|   | ||||
							
								
								
									
										321
									
								
								src/prompt.c
									
									
									
									
									
								
							
							
						
						
									
										321
									
								
								src/prompt.c
									
									
									
									
									
								
							| @@ -28,20 +28,20 @@ | ||||
| #include <string.h> | ||||
| #include <wchar.h> | ||||
|  | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "prompt.h" | ||||
| #include "friendlist.h" | ||||
| #include "execute.h" | ||||
| #include "misc_tools.h" | ||||
| #include "toxic_strings.h" | ||||
| #include "log.h" | ||||
| #include "line_info.h" | ||||
| #include "settings.h" | ||||
| #include "input.h" | ||||
| #include "help.h" | ||||
| #include "notify.h" | ||||
| #include "autocomplete.h" | ||||
| #include "execute.h" | ||||
| #include "friendlist.h" | ||||
| #include "help.h" | ||||
| #include "input.h" | ||||
| #include "line_info.h" | ||||
| #include "log.h" | ||||
| #include "misc_tools.h" | ||||
| #include "notify.h" | ||||
| #include "prompt.h" | ||||
| #include "settings.h" | ||||
| #include "toxic.h" | ||||
| #include "toxic_strings.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| extern ToxWindow *prompt; | ||||
| extern struct user_settings *user_settings; | ||||
| @@ -49,68 +49,47 @@ extern struct Winthread Winthread; | ||||
|  | ||||
| extern FriendsList Friends; | ||||
| FriendRequests FrndRequests; | ||||
| #ifdef AUDIO | ||||
| #define AC_NUM_GLOB_COMMANDS_AUDIO 2 | ||||
| #else | ||||
| #define AC_NUM_GLOB_COMMANDS_AUDIO 0 | ||||
| #endif /* AUDIO */ | ||||
| #ifdef VIDEO | ||||
| #define AC_NUM_GLOB_COMMANDS_VIDEO 2 | ||||
| #else | ||||
| #define AC_NUM_GLOB_COMMANDS_VIDEO 0 | ||||
| #endif /* VIDEO */ | ||||
| #ifdef PYTHON | ||||
| #define AC_NUM_GLOB_COMMANDS_PYTHON 1 | ||||
| #else | ||||
| #define AC_NUM_GLOB_COMMANDS_PYTHON 0 | ||||
| #endif /* PYTHON */ | ||||
| #ifdef QRCODE | ||||
| #define AC_NUM_GLOB_COMMANDS_QRCODE 1 | ||||
| #else | ||||
| #define AC_NUM_GLOB_COMMANDS_QRCODE 0 | ||||
| #endif /* QRCODE */ | ||||
| #define AC_NUM_GLOB_COMMANDS (17 + AC_NUM_GLOB_COMMANDS_AUDIO + AC_NUM_GLOB_COMMANDS_VIDEO + AC_NUM_GLOB_COMMANDS_PYTHON + AC_NUM_GLOB_COMMANDS_QRCODE) | ||||
|  | ||||
| /* Array of global command names used for tab completion. */ | ||||
| static const char glob_cmd_list[AC_NUM_GLOB_COMMANDS][MAX_CMDNAME_SIZE] = { | ||||
|     { "/accept"     }, | ||||
|     { "/add"        }, | ||||
|     { "/avatar"     }, | ||||
|     { "/clear"      }, | ||||
|     { "/connect"    }, | ||||
|     { "/decline"    }, | ||||
|     { "/exit"       }, | ||||
|     { "/group"      }, | ||||
|     { "/help"       }, | ||||
|     { "/log"        }, | ||||
|     { "/myid"       }, | ||||
| static const char *glob_cmd_list[] = { | ||||
|     "/accept", | ||||
|     "/add", | ||||
|     "/avatar", | ||||
|     "/clear", | ||||
|     "/connect", | ||||
|     "/decline", | ||||
|     "/exit", | ||||
|     "/conference", | ||||
|     "/help", | ||||
|     "/log", | ||||
|     "/myid", | ||||
| #ifdef QRCODE | ||||
|     { "/myqr"       }, | ||||
|     "/myqr", | ||||
| #endif /* QRCODE */ | ||||
|     { "/nick"       }, | ||||
|     { "/note"       }, | ||||
|     { "/nospam"     }, | ||||
|     { "/quit"       }, | ||||
|     { "/requests"   }, | ||||
|     { "/status"     }, | ||||
|     "/nick", | ||||
|     "/note", | ||||
|     "/nospam", | ||||
|     "/quit", | ||||
|     "/requests", | ||||
|     "/status", | ||||
|  | ||||
| #ifdef AUDIO | ||||
|  | ||||
|     { "/lsdev"       }, | ||||
|     { "/sdev"        }, | ||||
|     "/lsdev", | ||||
|     "/sdev", | ||||
|  | ||||
| #endif /* AUDIO */ | ||||
|  | ||||
| #ifdef VIDEO | ||||
|  | ||||
|     { "/lsvdev"      }, | ||||
|     { "/svdev"       }, | ||||
|     "/lsvdev", | ||||
|     "/svdev", | ||||
|  | ||||
| #endif /* VIDEO */ | ||||
|  | ||||
| #ifdef PYTHON | ||||
|  | ||||
|     { "/run"         }, | ||||
|     "/run", | ||||
|  | ||||
| #endif /* PYTHON */ | ||||
|  | ||||
| @@ -139,6 +118,8 @@ void kill_prompt_window(ToxWindow *self) | ||||
| /* callback: Updates own connection status in prompt statusbar */ | ||||
| void on_self_connection_status(Tox *m, Tox_Connection connection_status, void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(m); | ||||
|     UNUSED_VAR(userdata); | ||||
|     StatusBar *statusbar = prompt->stb; | ||||
|     statusbar->connection = connection_status; | ||||
| } | ||||
| @@ -163,7 +144,7 @@ void prompt_update_statusmessage(ToxWindow *prompt, Tox *m, const char *statusms | ||||
|     tox_self_set_status_message(m, (const uint8_t *) statusmsg, len, &err); | ||||
|  | ||||
|     if (err != TOX_ERR_SET_INFO_OK) { | ||||
|         line_info_add(prompt, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to set note (error %d)\n", err); | ||||
|         line_info_add(prompt, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to set note (error %d)\n", err); | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -210,7 +191,10 @@ static int add_friend_request(const char *public_key, const char *data) | ||||
|     return -1; | ||||
| } | ||||
|  | ||||
| static void prompt_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr) | ||||
| /* | ||||
|  * Return true if input is recognized by handler | ||||
|  */ | ||||
| static bool prompt_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr) | ||||
| { | ||||
|     ChatContext *ctx = self->chatwin; | ||||
|  | ||||
| @@ -218,8 +202,10 @@ static void prompt_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr) | ||||
|     getyx(self->window, y, x); | ||||
|     getmaxyx(self->window, y2, x2); | ||||
|  | ||||
|     UNUSED_VAR(y); | ||||
|  | ||||
|     if (x2 <= 0 || y2 <= 0) { | ||||
|         return; | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     if (ctx->pastemode && key == '\r') { | ||||
| @@ -229,21 +215,27 @@ static void prompt_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr) | ||||
|     /* ignore non-menu related input if active */ | ||||
|     if (self->help->active) { | ||||
|         help_onKey(self, key); | ||||
|         return; | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     if (ltr || key == '\n') {    /* char is printable */ | ||||
|         input_new_char(self, key, x, y, x2, y2); | ||||
|         return; | ||||
|         input_new_char(self, key, x, x2); | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     if (line_info_onKey(self, key)) { | ||||
|         return; | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     input_handle(self, key, x, y, x2, y2); | ||||
|     if (input_handle(self, key, x, x2)) { | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     int input_ret = false; | ||||
|  | ||||
|     if (key == '\t') {    /* TAB key: auto-completes command */ | ||||
|         input_ret = true; | ||||
|  | ||||
|         if (ctx->len > 1 && ctx->line[0] == '/') { | ||||
|             int diff = -1; | ||||
|  | ||||
| @@ -259,14 +251,14 @@ static void prompt_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr) | ||||
| #endif | ||||
|  | ||||
|             else if (wcsncmp(ctx->line, L"/status ", wcslen(L"/status ")) == 0) { | ||||
|                 const char status_cmd_list[3][8] = { | ||||
|                     {"online"}, | ||||
|                     {"away"}, | ||||
|                     {"busy"}, | ||||
|                 const char *status_cmd_list[] = { | ||||
|                     "online", | ||||
|                     "away", | ||||
|                     "busy", | ||||
|                 }; | ||||
|                 diff = complete_line(self, status_cmd_list, 3, 8); | ||||
|                 diff = complete_line(self, status_cmd_list, sizeof(status_cmd_list) / sizeof(char *)); | ||||
|             } else { | ||||
|                 diff = complete_line(self, glob_cmd_list, AC_NUM_GLOB_COMMANDS, MAX_CMDNAME_SIZE); | ||||
|                 diff = complete_line(self, glob_cmd_list, sizeof(glob_cmd_list) / sizeof(char *)); | ||||
|             } | ||||
|  | ||||
|             if (diff != -1) { | ||||
| @@ -281,31 +273,36 @@ static void prompt_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr) | ||||
|             sound_notify(self, notif_error, 0, NULL); | ||||
|         } | ||||
|     } else if (key == '\r') { | ||||
|         input_ret = true; | ||||
|  | ||||
|         rm_trailing_spaces_buf(ctx); | ||||
|  | ||||
|         if (!wstring_is_empty(ctx->line)) { | ||||
|             add_line_to_hist(ctx); | ||||
|             wstrsubst(ctx->line, L'¶', L'\n'); | ||||
|  | ||||
|             char line[MAX_STR_SIZE] = {0}; | ||||
|             char line[MAX_STR_SIZE]; | ||||
|  | ||||
|             if (wcs_to_mbs_buf(line, ctx->line, MAX_STR_SIZE) == -1) { | ||||
|                 memset(&line, 0, sizeof(line)); | ||||
|             } | ||||
|  | ||||
|             line_info_add(self, NULL, NULL, NULL, PROMPT, 0, 0, "%s", line); | ||||
|                 line_info_add(self, false, NULL, NULL, SYS_MSG, 0, RED, " * Failed to parse message."); | ||||
|             } else { | ||||
|                 line_info_add(self, false, NULL, NULL, PROMPT, 0, 0, "%s", line); | ||||
|                 execute(ctx->history, self, m, line, GLOBAL_COMMAND_MODE); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         wclear(ctx->linewin); | ||||
|         wmove(self->window, y2 - CURS_Y_OFFSET, 0); | ||||
|         wmove(self->window, y2, 0); | ||||
|         reset_buf(ctx); | ||||
|     } | ||||
|  | ||||
|     return input_ret; | ||||
| } | ||||
|  | ||||
| static void prompt_onDraw(ToxWindow *self, Tox *m) | ||||
| { | ||||
|     int x2, y2; | ||||
|     int x2; | ||||
|     int y2; | ||||
|     getmaxyx(self->window, y2, x2); | ||||
|  | ||||
|     if (y2 <= 0 || x2 <= 0) { | ||||
| @@ -320,64 +317,88 @@ static void prompt_onDraw(ToxWindow *self, Tox *m) | ||||
|  | ||||
|     wclear(ctx->linewin); | ||||
|  | ||||
|     curs_set(1); | ||||
|  | ||||
|     if (ctx->len > 0) { | ||||
|         mvwprintw(ctx->linewin, 1, 0, "%ls", &ctx->line[ctx->start]); | ||||
|         mvwprintw(ctx->linewin, 0, 0, "%ls", &ctx->line[ctx->start]); | ||||
|     } | ||||
|  | ||||
|     mvwhline(ctx->linewin, 0, ctx->len, ' ', x2 - ctx->len); | ||||
|  | ||||
|     curs_set(1); | ||||
|  | ||||
|     StatusBar *statusbar = self->stb; | ||||
|  | ||||
|     mvwhline(statusbar->topline, 1, 0, ACS_HLINE, x2); | ||||
|     wmove(statusbar->topline, 0, 0); | ||||
|  | ||||
|     pthread_mutex_lock(&Winthread.lock); | ||||
|     Tox_Connection connection = statusbar->connection; | ||||
|     Tox_User_Status status = statusbar->status; | ||||
|     pthread_mutex_unlock(&Winthread.lock); | ||||
|  | ||||
|     if (connection != TOX_CONNECTION_NONE) { | ||||
|         int colour = MAGENTA; | ||||
|         const char *status_text = "ERROR"; | ||||
|  | ||||
|         pthread_mutex_lock(&Winthread.lock); | ||||
|         Tox_User_Status status = statusbar->status; | ||||
|         pthread_mutex_unlock(&Winthread.lock); | ||||
|  | ||||
|         switch (status) { | ||||
|             case TOX_USER_STATUS_NONE: | ||||
|                 status_text = "Online"; | ||||
|                 colour = GREEN; | ||||
|                 colour = STATUS_ONLINE; | ||||
|                 break; | ||||
|  | ||||
|             case TOX_USER_STATUS_AWAY: | ||||
|                 status_text = "Away"; | ||||
|                 colour = YELLOW; | ||||
|                 colour = STATUS_AWAY; | ||||
|                 break; | ||||
|  | ||||
|             case TOX_USER_STATUS_BUSY: | ||||
|                 status_text = "Busy"; | ||||
|                 colour = RED; | ||||
|                 colour = STATUS_BUSY; | ||||
|                 break; | ||||
|         } | ||||
|  | ||||
|         wattron(statusbar->topline, COLOR_PAIR(colour) | A_BOLD); | ||||
|         wprintw(statusbar->topline, " [%s]", status_text); | ||||
|         wattroff(statusbar->topline, COLOR_PAIR(colour) | A_BOLD); | ||||
|         wattron(statusbar->topline, COLOR_PAIR(BAR_ACCENT)); | ||||
|         wprintw(statusbar->topline, " ["); | ||||
|         wattroff(statusbar->topline, COLOR_PAIR(BAR_ACCENT)); | ||||
|  | ||||
|         wattron(statusbar->topline, A_BOLD | COLOR_PAIR(colour)); | ||||
|         wprintw(statusbar->topline, "%s", status_text); | ||||
|         wattroff(statusbar->topline, A_BOLD | COLOR_PAIR(colour)); | ||||
|  | ||||
|         wattron(statusbar->topline, COLOR_PAIR(BAR_ACCENT)); | ||||
|         wprintw(statusbar->topline, "]"); | ||||
|         wattroff(statusbar->topline, COLOR_PAIR(BAR_ACCENT)); | ||||
|  | ||||
|         wattron(statusbar->topline, COLOR_PAIR(BAR_TEXT)); | ||||
|  | ||||
|         wattron(statusbar->topline, A_BOLD); | ||||
|         pthread_mutex_lock(&Winthread.lock); | ||||
|         wprintw(statusbar->topline, " %s", statusbar->nick); | ||||
|         pthread_mutex_unlock(&Winthread.lock); | ||||
|         wattroff(statusbar->topline, A_BOLD); | ||||
|     } else { | ||||
|         wprintw(statusbar->topline, " [Offline]"); | ||||
|         wattron(statusbar->topline, A_BOLD); | ||||
|         wattron(statusbar->topline, COLOR_PAIR(BAR_ACCENT)); | ||||
|         wprintw(statusbar->topline, " ["); | ||||
|         wattroff(statusbar->topline, COLOR_PAIR(BAR_ACCENT)); | ||||
|  | ||||
|         wattron(statusbar->topline, COLOR_PAIR(BAR_TEXT)); | ||||
|         wprintw(statusbar->topline, "Offline"); | ||||
|         wattroff(statusbar->topline, COLOR_PAIR(BAR_TEXT)); | ||||
|  | ||||
|         wattron(statusbar->topline, COLOR_PAIR(BAR_ACCENT)); | ||||
|         wprintw(statusbar->topline, "]"); | ||||
|         wattroff(statusbar->topline, COLOR_PAIR(BAR_ACCENT)); | ||||
|  | ||||
|         wattron(statusbar->topline, COLOR_PAIR(BAR_TEXT)); | ||||
|  | ||||
|         pthread_mutex_lock(&Winthread.lock); | ||||
|         wprintw(statusbar->topline, " %s", statusbar->nick); | ||||
|         pthread_mutex_unlock(&Winthread.lock); | ||||
|         wattroff(statusbar->topline, A_BOLD); | ||||
|     } | ||||
|  | ||||
|     int s_y; | ||||
|     int s_x; | ||||
|     getyx(statusbar->topline, s_y, s_x); | ||||
|  | ||||
|     mvwhline(statusbar->topline, s_y, s_x, ' ', x2 - s_x); | ||||
|     wattroff(statusbar->topline, COLOR_PAIR(BAR_TEXT)); | ||||
|  | ||||
|     /* Reset statusbar->statusmsg on window resize */ | ||||
|     if (x2 != self->x) { | ||||
|         char statusmsg[TOX_MAX_STATUS_MESSAGE_LENGTH]; | ||||
| @@ -405,19 +426,27 @@ static void prompt_onDraw(ToxWindow *self, Tox *m) | ||||
|     } | ||||
|  | ||||
|     if (statusbar->statusmsg[0]) { | ||||
|         wprintw(statusbar->topline, " : %s", statusbar->statusmsg); | ||||
|         wattron(statusbar->topline, COLOR_PAIR(BAR_ACCENT)); | ||||
|         wprintw(statusbar->topline, " | "); | ||||
|         wattroff(statusbar->topline, COLOR_PAIR(BAR_ACCENT)); | ||||
|  | ||||
|         wattron(statusbar->topline, COLOR_PAIR(BAR_TEXT)); | ||||
|         wprintw(statusbar->topline, "%s", statusbar->statusmsg); | ||||
|         wattroff(statusbar->topline, COLOR_PAIR(BAR_TEXT)); | ||||
|     } | ||||
|  | ||||
|     pthread_mutex_unlock(&Winthread.lock); | ||||
|  | ||||
|     mvwhline(self->window, y2 - CHATBOX_HEIGHT, 0, ACS_HLINE, x2); | ||||
|  | ||||
|     int y, x; | ||||
|     int y; | ||||
|     int x; | ||||
|     getyx(self->window, y, x); | ||||
|     (void) x; | ||||
|  | ||||
|     UNUSED_VAR(x); | ||||
|  | ||||
|     int new_x = ctx->start ? x2 - 1 : MAX(0, wcswidth(ctx->line, ctx->pos)); | ||||
|     wmove(self->window, y + 1, new_x); | ||||
|     wmove(self->window, y, new_x); | ||||
|  | ||||
|     draw_window_bar(self); | ||||
|  | ||||
|     wnoutrefresh(self->window); | ||||
|  | ||||
| @@ -437,8 +466,6 @@ static void prompt_onConnectionChange(ToxWindow *self, Tox *m, uint32_t friendnu | ||||
|         snprintf(nick, sizeof(nick), "%s", UNKNOWN_NAME); | ||||
|     } | ||||
|  | ||||
|     char timefrmt[TIME_STR_SIZE]; | ||||
|     get_time_str(timefrmt, sizeof(timefrmt)); | ||||
|     const char *msg; | ||||
|  | ||||
|     if (user_settings->show_connection_msg == SHOW_WELCOME_MSG_OFF) { | ||||
| @@ -447,7 +474,7 @@ static void prompt_onConnectionChange(ToxWindow *self, Tox *m, uint32_t friendnu | ||||
|  | ||||
|     if (connection_status != TOX_CONNECTION_NONE && Friends.list[friendnum].connection_status == TOX_CONNECTION_NONE) { | ||||
|         msg = "has come online"; | ||||
|         line_info_add(self, timefrmt, nick, NULL, CONNECTION, 0, GREEN, msg); | ||||
|         line_info_add(self, true, nick, NULL, CONNECTION, 0, GREEN, msg); | ||||
|         write_to_log(msg, nick, ctx->log, true); | ||||
|  | ||||
|         if (self->active_box != -1) { | ||||
| @@ -459,7 +486,7 @@ static void prompt_onConnectionChange(ToxWindow *self, Tox *m, uint32_t friendnu | ||||
|         } | ||||
|     } else if (connection_status == TOX_CONNECTION_NONE) { | ||||
|         msg = "has gone offline"; | ||||
|         line_info_add(self, timefrmt, nick, NULL, DISCONNECTION, 0, RED, msg); | ||||
|         line_info_add(self, true, nick, NULL, DISCONNECTION, 0, RED, msg); | ||||
|         write_to_log(msg, nick, ctx->log, true); | ||||
|  | ||||
|         if (self->active_box != -1) { | ||||
| @@ -474,23 +501,23 @@ static void prompt_onConnectionChange(ToxWindow *self, Tox *m, uint32_t friendnu | ||||
|  | ||||
| static void prompt_onFriendRequest(ToxWindow *self, Tox *m, const char *key, const char *data, size_t length) | ||||
| { | ||||
|     UNUSED_VAR(m); | ||||
|     UNUSED_VAR(length); | ||||
|  | ||||
|     ChatContext *ctx = self->chatwin; | ||||
|  | ||||
|     char timefrmt[TIME_STR_SIZE]; | ||||
|     get_time_str(timefrmt, sizeof(timefrmt)); | ||||
|  | ||||
|     line_info_add(self, timefrmt, NULL, NULL, SYS_MSG, 0, 0, "Friend request with the message '%s'", data); | ||||
|     line_info_add(self, true, NULL, NULL, SYS_MSG, 0, 0, "Friend request with the message '%s'", data); | ||||
|     write_to_log("Friend request with the message '%s'", "", ctx->log, true); | ||||
|  | ||||
|     int n = add_friend_request(key, data); | ||||
|  | ||||
|     if (n == -1) { | ||||
|         const char *errmsg = "Friend request queue is full. Discarding request."; | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, errmsg); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, errmsg); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Type \"/accept %d\" or \"/decline %d\"", n, n); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Type \"/accept %d\" or \"/decline %d\"", n, n); | ||||
|     sound_notify(self, generic_message, NT_WNDALERT_1 | NT_NOTIFWND, NULL); | ||||
| } | ||||
|  | ||||
| @@ -503,7 +530,7 @@ void prompt_init_statusbar(ToxWindow *self, Tox *m, bool first_time_run) | ||||
|         exit_toxic_err("failed in prompt_init_statusbar", FATALERR_CURSES); | ||||
|     } | ||||
|  | ||||
|     (void) y2; | ||||
|     UNUSED_VAR(y2); | ||||
|  | ||||
|     /* Init statusbar info */ | ||||
|     StatusBar *statusbar = self->stb; | ||||
| @@ -535,29 +562,50 @@ void prompt_init_statusbar(ToxWindow *self, Tox *m, bool first_time_run) | ||||
|     prompt_update_nick(prompt, nick); | ||||
|  | ||||
|     /* Init statusbar subwindow */ | ||||
|     statusbar->topline = subwin(self->window, 2, x2, 0, 0); | ||||
|     statusbar->topline = subwin(self->window, TOP_BAR_HEIGHT, x2, 0, 0); | ||||
| } | ||||
|  | ||||
| static void print_welcome_msg(ToxWindow *self) | ||||
| { | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 1, BLUE, "    _____ _____  _____ ____ "); | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 1, BLUE, "   |_   _/ _ \\ \\/ /_ _/ ___|"); | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 1, BLUE, "     | || | | \\  / | | |    "); | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 1, BLUE, "     | || |_| /  \\ | | |___ "); | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 1, BLUE, "     |_| \\___/_/\\_\\___\\____| v." TOXICVER); | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, ""); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 1, BLUE, "    _____ _____  _____ ____ "); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 1, BLUE, "   |_   _/ _ \\ \\/ /_ _/ ___|"); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 1, BLUE, "     | || | | \\  / | | |    "); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 1, BLUE, "     | || |_| /  \\ | | |___ "); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 1, BLUE, "     |_| \\___/_/\\_\\___\\____| v." TOXICVER); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, ""); | ||||
|  | ||||
|     const char *msg = "Welcome to Toxic, a free, open source Tox-based instant messenging client."; | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 1, CYAN, msg); | ||||
|     const char *msg = "Welcome to Toxic, a free, open source Tox-based instant messaging client."; | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 1, CYAN, msg); | ||||
|     msg = "Type \"/help\" for assistance. Further help may be found via the man page."; | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 1, CYAN, msg); | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, ""); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 1, CYAN, msg); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, ""); | ||||
| } | ||||
|  | ||||
| static void prompt_init_log(ToxWindow *self, Tox *m, const char *self_name) | ||||
| { | ||||
|     ChatContext *ctx = self->chatwin; | ||||
|  | ||||
|     char myid[TOX_ADDRESS_SIZE]; | ||||
|     tox_self_get_address(m, (uint8_t *) myid); | ||||
|  | ||||
|     if (log_init(ctx->log, self->name, myid, NULL, LOG_TYPE_PROMPT) != 0) { | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Warning: Log failed to initialize."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (user_settings->autolog == AUTOLOG_ON) { | ||||
|         if (log_enable(ctx->log) == -1) { | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Warning: Failed to enable log."); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| static void prompt_onInit(ToxWindow *self, Tox *m) | ||||
| { | ||||
|     curs_set(1); | ||||
|     int y2, x2; | ||||
|  | ||||
|     int y2; | ||||
|     int x2; | ||||
|     getmaxyx(self->window, y2, x2); | ||||
|  | ||||
|     if (y2 <= 0 || x2 <= 0) { | ||||
| @@ -565,8 +613,10 @@ static void prompt_onInit(ToxWindow *self, Tox *m) | ||||
|     } | ||||
|  | ||||
|     ChatContext *ctx = self->chatwin; | ||||
|     ctx->history = subwin(self->window, y2 - CHATBOX_HEIGHT + 1, x2, 0, 0); | ||||
|     ctx->linewin = subwin(self->window, CHATBOX_HEIGHT, x2, y2 - CHATBOX_HEIGHT, 0); | ||||
|  | ||||
|     ctx->history = subwin(self->window, y2 - CHATBOX_HEIGHT - WINDOW_BAR_HEIGHT, x2, 0, 0); | ||||
|     self->window_bar = subwin(self->window, WINDOW_BAR_HEIGHT, x2, y2 - (CHATBOX_HEIGHT + WINDOW_BAR_HEIGHT), 0); | ||||
|     ctx->linewin = subwin(self->window, CHATBOX_HEIGHT, x2, y2 - WINDOW_BAR_HEIGHT, 0); | ||||
|  | ||||
|     ctx->log = calloc(1, sizeof(struct chatlog)); | ||||
|     ctx->hst = calloc(1, sizeof(struct history)); | ||||
| @@ -577,14 +627,7 @@ static void prompt_onInit(ToxWindow *self, Tox *m) | ||||
|  | ||||
|     line_info_init(ctx->hst); | ||||
|  | ||||
|     if (user_settings->autolog == AUTOLOG_ON) { | ||||
|         char myid[TOX_ADDRESS_SIZE]; | ||||
|         tox_self_get_address(m, (uint8_t *) myid); | ||||
|  | ||||
|         if (log_enable(self->name, myid, NULL, ctx->log, LOG_PROMPT) == -1) { | ||||
|             line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Warning: Log failed to initialize."); | ||||
|         } | ||||
|     } | ||||
|     prompt_init_log(self, m, self->name); | ||||
|  | ||||
|     scrollok(ctx->history, 0); | ||||
|     wmove(self->window, y2 - CURS_Y_OFFSET, 0); | ||||
| @@ -603,7 +646,7 @@ ToxWindow *new_prompt(void) | ||||
|     } | ||||
|  | ||||
|     ret->num = -1; | ||||
|     ret->is_prompt = true; | ||||
|     ret->type = WINDOW_TYPE_PROMPT; | ||||
|  | ||||
|     ret->onKey = &prompt_onKey; | ||||
|     ret->onDraw = &prompt_onDraw; | ||||
| @@ -611,7 +654,7 @@ ToxWindow *new_prompt(void) | ||||
|     ret->onConnectionChange = &prompt_onConnectionChange; | ||||
|     ret->onFriendRequest = &prompt_onFriendRequest; | ||||
|  | ||||
|     strcpy(ret->name, "home"); | ||||
|     strcpy(ret->name, "Home"); | ||||
|  | ||||
|     ChatContext *chatwin = calloc(1, sizeof(ChatContext)); | ||||
|     StatusBar *stb = calloc(1, sizeof(StatusBar)); | ||||
|   | ||||
| @@ -26,22 +26,20 @@ | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| #define MAX_FRIEND_REQUESTS 32 | ||||
| #define MAX_FRIEND_REQUESTS 20 | ||||
|  | ||||
| struct friend_request { | ||||
|     bool active; | ||||
|     char msg[MAX_STR_SIZE]; | ||||
|     char msg[TOX_MAX_FRIEND_REQUEST_LENGTH + 1]; | ||||
|     uint8_t key[TOX_PUBLIC_KEY_SIZE]; | ||||
| }; | ||||
|  | ||||
| typedef struct { | ||||
| typedef struct FriendRequests { | ||||
|     int max_idx; | ||||
|     int num_requests; | ||||
|     struct friend_request request[MAX_FRIEND_REQUESTS]; | ||||
| } FriendRequests; | ||||
|  | ||||
| extern FriendRequests FrndRequests; | ||||
|  | ||||
| ToxWindow *new_prompt(void); | ||||
|  | ||||
| void prep_prompt_win(void); | ||||
|   | ||||
| @@ -114,9 +114,7 @@ static PyObject *python_api_get_status_message(PyObject *self, PyObject *args) | ||||
|  | ||||
| static PyObject *python_api_get_all_friends(PyObject *self, PyObject *args) | ||||
| { | ||||
|     size_t       i, ii; | ||||
|     FriendsList  friends; | ||||
|     PyObject    *cur, *ret; | ||||
|     char pubkey_buf[TOX_PUBLIC_KEY_SIZE * 2 + 1]; | ||||
|  | ||||
|     if (!PyArg_ParseTuple(args, "")) { | ||||
| @@ -124,15 +122,15 @@ static PyObject *python_api_get_all_friends(PyObject *self, PyObject *args) | ||||
|     } | ||||
|  | ||||
|     friends = api_get_friendslist(); | ||||
|     ret     = PyList_New(0); | ||||
|     PyObject *ret = PyList_New(0); | ||||
|  | ||||
|     for (i = 0; i < friends.num_friends; i++) { | ||||
|         for (ii = 0; ii < TOX_PUBLIC_KEY_SIZE; ii++) { | ||||
|     for (size_t i = 0; i < friends.num_friends; i++) { | ||||
|         for (size_t ii = 0; ii < TOX_PUBLIC_KEY_SIZE; ii++) { | ||||
|             snprintf(pubkey_buf + ii * 2, 3, "%02X", friends.list[i].pub_key[ii] & 0xff); | ||||
|         } | ||||
|  | ||||
|         pubkey_buf[TOX_PUBLIC_KEY_SIZE * 2] = '\0'; | ||||
|         cur = Py_BuildValue("(s,s)", friends.list[i].name, pubkey_buf); | ||||
|         PyObject *cur = Py_BuildValue("(s,s)", friends.list[i].name, pubkey_buf); | ||||
|         PyList_Append(ret, cur); | ||||
|     } | ||||
|  | ||||
| @@ -252,26 +250,26 @@ PyMODINIT_FUNC PyInit_toxic_api(void) | ||||
|     PyObject *m = PyModule_Create(&toxic_api_module); | ||||
|     PyObject *global_command_const    = Py_BuildValue("i", GLOBAL_COMMAND_MODE); | ||||
|     PyObject *chat_command_const      = Py_BuildValue("i", CHAT_COMMAND_MODE); | ||||
|     PyObject *groupchat_command_const = Py_BuildValue("i", GROUPCHAT_COMMAND_MODE); | ||||
|     PyObject *conference_command_const = Py_BuildValue("i", CONFERENCE_COMMAND_MODE); | ||||
|     PyObject_SetAttrString(m, "GLOBAL_COMMAND",    global_command_const); | ||||
|     PyObject_SetAttrString(m, "CHAT_COMMAND",      chat_command_const); | ||||
|     PyObject_SetAttrString(m, "GROUPCHAT_COMMAND", groupchat_command_const); | ||||
|     PyObject_SetAttrString(m, "CONFERENCE_COMMAND", conference_command_const); | ||||
|     Py_DECREF(global_command_const); | ||||
|     Py_DECREF(chat_command_const); | ||||
|     Py_DECREF(groupchat_command_const); | ||||
|     Py_DECREF(conference_command_const); | ||||
|     return m; | ||||
| } | ||||
|  | ||||
| void terminate_python(void) | ||||
| { | ||||
|     struct python_registered_func *cur, *old; | ||||
|  | ||||
|     if (python_commands.name != NULL) { | ||||
|         free(python_commands.name); | ||||
|     } | ||||
|  | ||||
|     struct python_registered_func *cur = NULL; | ||||
|  | ||||
|     for (cur = python_commands.next; cur != NULL;) { | ||||
|         old = cur; | ||||
|         struct python_registered_func *old = cur; | ||||
|         cur = cur->next; | ||||
|         free(old->name); | ||||
|         free(old); | ||||
|   | ||||
| @@ -22,14 +22,14 @@ | ||||
|  | ||||
| #ifdef QRCODE | ||||
|  | ||||
| #include <stdlib.h> | ||||
| #include <qrencode.h> | ||||
| #include <stdbool.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
|  | ||||
| #include "qr_code.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "qr_code.h" | ||||
|  | ||||
| #ifdef QRPNG | ||||
| #include <png.h> | ||||
| @@ -138,12 +138,19 @@ int ID_to_QRcode_png(const char *tox_id, const char *outfile) | ||||
|  | ||||
|     real_width = (qr_obj->width + BORDER_LEN * 2) * SQUARE_SIZE; | ||||
|     size_t row_size = real_width * 4; | ||||
|     unsigned char row[row_size]; | ||||
|     unsigned char *row = malloc(row_size); | ||||
|  | ||||
|     if (row == NULL) { | ||||
|         fclose(fp); | ||||
|         QRcode_free(qr_obj); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL); | ||||
|  | ||||
|     if (png_ptr == NULL) { | ||||
|         fclose(fp); | ||||
|         free(row); | ||||
|         QRcode_free(qr_obj); | ||||
|         return -1; | ||||
|     } | ||||
| @@ -152,12 +159,14 @@ int ID_to_QRcode_png(const char *tox_id, const char *outfile) | ||||
|  | ||||
|     if (info_ptr == NULL) { | ||||
|         fclose(fp); | ||||
|         free(row); | ||||
|         QRcode_free(qr_obj); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (setjmp(png_jmpbuf(png_ptr))) { | ||||
|         fclose(fp); | ||||
|         free(row); | ||||
|         QRcode_free(qr_obj); | ||||
|         png_destroy_write_struct(&png_ptr, &info_ptr); | ||||
|         return -1; | ||||
| @@ -206,10 +215,12 @@ int ID_to_QRcode_png(const char *tox_id, const char *outfile) | ||||
|         png_write_row(png_ptr, row); | ||||
|     } | ||||
|  | ||||
|     free(row); | ||||
|     fclose(fp); | ||||
|  | ||||
|     png_write_end(png_ptr, info_ptr); | ||||
|     png_destroy_write_struct(&png_ptr, &info_ptr); | ||||
|  | ||||
|     fclose(fp); | ||||
|     QRcode_free(qr_obj); | ||||
|  | ||||
|     return 0; | ||||
|   | ||||
| @@ -20,23 +20,23 @@ | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <ctype.h> | ||||
| #include <libconfig.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <libconfig.h> | ||||
| #include <ctype.h> | ||||
|  | ||||
| #include "configdir.h" | ||||
| #include "misc_tools.h" | ||||
| #include "notify.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "configdir.h" | ||||
| #include "notify.h" | ||||
| #include "misc_tools.h" | ||||
|  | ||||
| #ifdef AUDIO | ||||
| #include "audio_device.h" | ||||
| #endif /* AUDIO */ | ||||
|  | ||||
| #include "settings.h" | ||||
| #include "line_info.h" | ||||
| #include "settings.h" | ||||
|  | ||||
| #ifndef PACKAGE_DATADIR | ||||
| #define PACKAGE_DATADIR "." | ||||
| @@ -58,11 +58,13 @@ static struct ui_strings { | ||||
|     const char *native_colors; | ||||
|     const char *autolog; | ||||
|     const char *history_size; | ||||
|     const char *notification_timeout; | ||||
|     const char *show_typing_self; | ||||
|     const char *show_typing_other; | ||||
|     const char *show_welcome_msg; | ||||
|     const char *show_connection_msg; | ||||
|     const char *nodeslist_update_freq; | ||||
|     const char *autosave_freq; | ||||
|  | ||||
|     const char *line_join; | ||||
|     const char *line_quit; | ||||
| @@ -71,6 +73,11 @@ static struct ui_strings { | ||||
|  | ||||
|     const char *mplex_away; | ||||
|     const char *mplex_away_note; | ||||
|  | ||||
|     const char *color_bar_bg; | ||||
|     const char *color_bar_fg; | ||||
|     const char *color_bar_accent; | ||||
|     const char *color_bar_notify; | ||||
| } ui_strings = { | ||||
|     "ui", | ||||
|     "timestamps", | ||||
| @@ -85,17 +92,23 @@ static struct ui_strings { | ||||
|     "native_colors", | ||||
|     "autolog", | ||||
|     "history_size", | ||||
|     "notification_timeout", | ||||
|     "show_typing_self", | ||||
|     "show_typing_other", | ||||
|     "show_welcome_msg", | ||||
|     "show_connection_msg", | ||||
|     "nodeslist_update_freq", | ||||
|     "autosave_freq", | ||||
|     "line_join", | ||||
|     "line_quit", | ||||
|     "line_alert", | ||||
|     "line_normal", | ||||
|     "mplex_away", | ||||
|     "mplex_away_note", | ||||
|     "color_bar_bg", | ||||
|     "color_bar_fg", | ||||
|     "color_bar_accent", | ||||
|     "color_bar_notify", | ||||
| }; | ||||
|  | ||||
| static void ui_defaults(struct user_settings *settings) | ||||
| @@ -112,11 +125,13 @@ static void ui_defaults(struct user_settings *settings) | ||||
|     settings->bell_on_invite = 0; | ||||
|     settings->colour_theme = DFLT_COLS; | ||||
|     settings->history_size = 700; | ||||
|     settings->notification_timeout = 6000; | ||||
|     settings->show_typing_self = SHOW_TYPING_ON; | ||||
|     settings->show_typing_other = SHOW_TYPING_ON; | ||||
|     settings->show_welcome_msg = SHOW_WELCOME_MSG_ON; | ||||
|     settings->show_connection_msg = SHOW_CONNECTION_MSG_ON; | ||||
|     settings->nodeslist_update_freq = 7; | ||||
|     settings->autosave_freq = 600; | ||||
|  | ||||
|     snprintf(settings->line_join, LINE_HINT_MAX + 1, "%s", LINE_JOIN); | ||||
|     snprintf(settings->line_quit, LINE_HINT_MAX + 1, "%s", LINE_QUIT); | ||||
| @@ -124,10 +139,7 @@ static void ui_defaults(struct user_settings *settings) | ||||
|     snprintf(settings->line_normal, LINE_HINT_MAX + 1, "%s", LINE_NORMAL); | ||||
|  | ||||
|     settings->mplex_away = MPLEX_ON; | ||||
|     snprintf(settings->mplex_away_note, | ||||
|              sizeof(settings->mplex_away_note), | ||||
|              "%s", | ||||
|              MPLEX_AWAY_NOTE); | ||||
|     snprintf(settings->mplex_away_note, sizeof(settings->mplex_away_note), "%s", MPLEX_AWAY_NOTE); | ||||
| } | ||||
|  | ||||
| static const struct keys_strings { | ||||
| @@ -139,8 +151,6 @@ static const struct keys_strings { | ||||
|     const char *half_page_up; | ||||
|     const char *half_page_down; | ||||
|     const char *page_bottom; | ||||
|     const char *peer_list_up; | ||||
|     const char *peer_list_down; | ||||
|     const char *toggle_peerlist; | ||||
|     const char *toggle_pastemode; | ||||
| } key_strings = { | ||||
| @@ -152,8 +162,6 @@ static const struct keys_strings { | ||||
|     "half_page_up", | ||||
|     "half_page_down", | ||||
|     "page_bottom", | ||||
|     "peer_list_up", | ||||
|     "peer_list_down", | ||||
|     "toggle_peerlist", | ||||
|     "toggle_paste_mode", | ||||
| }; | ||||
| @@ -168,8 +176,6 @@ static void key_defaults(struct user_settings *settings) | ||||
|     settings->key_half_page_up = T_KEY_C_F; | ||||
|     settings->key_half_page_down = T_KEY_C_V; | ||||
|     settings->key_page_bottom = T_KEY_C_H; | ||||
|     settings->key_peer_list_up = T_KEY_C_LB; | ||||
|     settings->key_peer_list_down = T_KEY_C_RB; | ||||
|     settings->key_toggle_peerlist = T_KEY_C_B; | ||||
|     settings->key_toggle_pastemode = T_KEY_C_T; | ||||
| } | ||||
| @@ -204,19 +210,28 @@ static const struct audio_strings { | ||||
|     const char *self; | ||||
|     const char *input_device; | ||||
|     const char *output_device; | ||||
|     const char *VAD_treshold; | ||||
|     const char *VAD_threshold; | ||||
|     const char *conference_audio_channels; | ||||
|     const char *chat_audio_channels; | ||||
|     const char *push_to_talk; | ||||
| } audio_strings = { | ||||
|     "audio", | ||||
|     "input_device", | ||||
|     "output_device", | ||||
|     "VAD_treshold", | ||||
|     "VAD_threshold", | ||||
|     "conference_audio_channels", | ||||
|     "chat_audio_channels", | ||||
|     "push_to_talk", | ||||
| }; | ||||
|  | ||||
| static void audio_defaults(struct user_settings *settings) | ||||
| { | ||||
|     settings->audio_in_dev = 0; | ||||
|     settings->audio_out_dev = 0; | ||||
|     settings->VAD_treshold = 40.0; | ||||
|     settings->VAD_threshold = 5.0; | ||||
|     settings->conference_audio_channels = 1; | ||||
|     settings->chat_audio_channels = 2; | ||||
|     settings->push_to_talk = 0; | ||||
| } | ||||
| #endif | ||||
|  | ||||
| @@ -330,7 +345,7 @@ int settings_load(struct user_settings *s, const char *patharg) | ||||
|  | ||||
|         if (config_setting_lookup_int(setting, ui_strings.time_format, &time)) { | ||||
|             if (time == 12) { | ||||
|                 snprintf(s->timestamp_format, sizeof(s->timestamp_format), "%s", "%I:%M:%S %p"); | ||||
|                 snprintf(s->timestamp_format, sizeof(s->timestamp_format), "%s", "%I:%M %p"); | ||||
|                 snprintf(s->log_timestamp_format, sizeof(s->log_timestamp_format), "%s", "%Y/%m/%d [%I:%M:%S %p]"); | ||||
|             } | ||||
|         } | ||||
| @@ -339,6 +354,22 @@ int settings_load(struct user_settings *s, const char *patharg) | ||||
|             snprintf(s->timestamp_format, sizeof(s->timestamp_format), "%s", str); | ||||
|         } | ||||
|  | ||||
|         if (config_setting_lookup_string(setting, ui_strings.color_bar_bg, &str)) { | ||||
|             snprintf(s->color_bar_bg, sizeof(s->color_bar_bg), "%s", str); | ||||
|         } | ||||
|  | ||||
|         if (config_setting_lookup_string(setting, ui_strings.color_bar_fg, &str)) { | ||||
|             snprintf(s->color_bar_fg, sizeof(s->color_bar_fg), "%s", str); | ||||
|         } | ||||
|  | ||||
|         if (config_setting_lookup_string(setting, ui_strings.color_bar_accent, &str)) { | ||||
|             snprintf(s->color_bar_accent, sizeof(s->color_bar_accent), "%s", str); | ||||
|         } | ||||
|  | ||||
|         if (config_setting_lookup_string(setting, ui_strings.color_bar_notify, &str)) { | ||||
|             snprintf(s->color_bar_notify, sizeof(s->color_bar_notify), "%s", str); | ||||
|         } | ||||
|  | ||||
|         if (config_setting_lookup_string(setting, ui_strings.log_timestamp_format, &str)) { | ||||
|             snprintf(s->log_timestamp_format, sizeof(s->log_timestamp_format), "%s", str); | ||||
|         } | ||||
| @@ -369,7 +400,9 @@ int settings_load(struct user_settings *s, const char *patharg) | ||||
|         config_setting_lookup_bool(setting, ui_strings.show_connection_msg, &s->show_connection_msg); | ||||
|  | ||||
|         config_setting_lookup_int(setting, ui_strings.history_size, &s->history_size); | ||||
|         config_setting_lookup_int(setting, ui_strings.notification_timeout, &s->notification_timeout); | ||||
|         config_setting_lookup_int(setting, ui_strings.nodeslist_update_freq, &s->nodeslist_update_freq); | ||||
|         config_setting_lookup_int(setting, ui_strings.autosave_freq, &s->autosave_freq); | ||||
|  | ||||
|         if (config_setting_lookup_string(setting, ui_strings.line_join, &str)) { | ||||
|             snprintf(s->line_join, sizeof(s->line_join), "%s", str); | ||||
| @@ -485,14 +518,6 @@ int settings_load(struct user_settings *s, const char *patharg) | ||||
|             set_key_binding(&s->key_page_bottom, &tmp); | ||||
|         } | ||||
|  | ||||
|         if (config_setting_lookup_string(setting, key_strings.peer_list_up, &tmp)) { | ||||
|             set_key_binding(&s->key_peer_list_up, &tmp); | ||||
|         } | ||||
|  | ||||
|         if (config_setting_lookup_string(setting, key_strings.peer_list_down, &tmp)) { | ||||
|             set_key_binding(&s->key_peer_list_down, &tmp); | ||||
|         } | ||||
|  | ||||
|         if (config_setting_lookup_string(setting, key_strings.toggle_peerlist, &tmp)) { | ||||
|             set_key_binding(&s->key_toggle_peerlist, &tmp); | ||||
|         } | ||||
| @@ -511,7 +536,16 @@ int settings_load(struct user_settings *s, const char *patharg) | ||||
|         config_setting_lookup_int(setting, audio_strings.output_device, &s->audio_out_dev); | ||||
|         s->audio_out_dev = s->audio_out_dev < 0 || s->audio_out_dev > MAX_DEVICES ? 0 : s->audio_out_dev; | ||||
|  | ||||
|         config_setting_lookup_float(setting, audio_strings.VAD_treshold, &s->VAD_treshold); | ||||
|         config_setting_lookup_float(setting, audio_strings.VAD_threshold, &s->VAD_threshold); | ||||
|  | ||||
|         config_setting_lookup_int(setting, audio_strings.conference_audio_channels, &s->conference_audio_channels); | ||||
|         s->conference_audio_channels = s->conference_audio_channels <= 0 | ||||
|                                        || s->conference_audio_channels > 2 ? 1 : s->conference_audio_channels; | ||||
|  | ||||
|         config_setting_lookup_int(setting, audio_strings.chat_audio_channels, &s->chat_audio_channels); | ||||
|         s->chat_audio_channels = s->chat_audio_channels <= 0 || s->chat_audio_channels > 2 ? 2 : s->chat_audio_channels; | ||||
|  | ||||
|         config_setting_lookup_bool(setting, audio_strings.push_to_talk, &s->push_to_talk); | ||||
|     } | ||||
|  | ||||
| #endif | ||||
|   | ||||
| @@ -49,11 +49,13 @@ struct user_settings { | ||||
|  | ||||
|     int colour_theme;      /* boolean (0 for default toxic colours) */ | ||||
|     int history_size;      /* int between MIN_HISTORY and MAX_HISTORY */ | ||||
|     int notification_timeout; | ||||
|     int show_typing_self;  /* boolean */ | ||||
|     int show_typing_other; /* boolean */ | ||||
|     int show_welcome_msg;  /* boolean */ | ||||
|     int show_connection_msg;  /* boolean */ | ||||
|     int nodeslist_update_freq;  /* int (<= 0 to disable updates) */ | ||||
|     int autosave_freq; /* int (<= 0 to disable autosave) */ | ||||
|  | ||||
|     char line_join[LINE_HINT_MAX + 1]; | ||||
|     char line_quit[LINE_HINT_MAX + 1]; | ||||
| @@ -66,6 +68,11 @@ struct user_settings { | ||||
|     char autorun_path[PATH_MAX]; | ||||
|     char password_eval[PASSWORD_EVAL_MAX]; | ||||
|  | ||||
|     char color_bar_bg[COLOR_STR_SIZE]; | ||||
|     char color_bar_fg[COLOR_STR_SIZE]; | ||||
|     char color_bar_accent[COLOR_STR_SIZE]; | ||||
|     char color_bar_notify[COLOR_STR_SIZE]; | ||||
|  | ||||
|     int key_next_tab; | ||||
|     int key_prev_tab; | ||||
|     int key_scroll_line_up; | ||||
| @@ -73,8 +80,6 @@ struct user_settings { | ||||
|     int key_half_page_up; | ||||
|     int key_half_page_down; | ||||
|     int key_page_bottom; | ||||
|     int key_peer_list_up; | ||||
|     int key_peer_list_down; | ||||
|     int key_toggle_peerlist; | ||||
|     int key_toggle_pastemode; | ||||
|  | ||||
| @@ -84,7 +89,10 @@ struct user_settings { | ||||
| #ifdef AUDIO | ||||
|     int audio_in_dev; | ||||
|     int audio_out_dev; | ||||
|     double VAD_treshold; | ||||
|     double VAD_threshold; | ||||
|     int conference_audio_channels; | ||||
|     int chat_audio_channels; | ||||
|     int push_to_talk;      /* boolean */ | ||||
| #endif | ||||
| }; | ||||
|  | ||||
| @@ -119,10 +127,10 @@ enum settings_values { | ||||
| #define LINE_JOIN    "-->" | ||||
| #define LINE_QUIT    "<--" | ||||
| #define LINE_ALERT   "-!-" | ||||
| #define LINE_NORMAL  "---" | ||||
| #define TIMESTAMP_DEFAULT      "%H:%M:%S" | ||||
| #define LINE_NORMAL  "-" | ||||
| #define TIMESTAMP_DEFAULT      "%H:%M" | ||||
| #define LOG_TIMESTAMP_DEFAULT  "%Y/%m/%d [%H:%M:%S]" | ||||
| #define MPLEX_AWAY_NOTE "Detached from screen" | ||||
| #define MPLEX_AWAY_NOTE "Away from keyboard, be back soon!" | ||||
|  | ||||
| int settings_load(struct user_settings *s, const char *patharg); | ||||
|  | ||||
|   | ||||
| @@ -26,18 +26,18 @@ | ||||
| #include <string.h> /* strlen, strcpy, strstr, strchr, strrchr, strcat, strncmp */ | ||||
|  | ||||
| #include <pthread.h> | ||||
| #include <sys/types.h> | ||||
| #include <sys/stat.h> | ||||
| #include <sys/types.h> | ||||
| #include <time.h> | ||||
| #include <unistd.h> | ||||
|  | ||||
| #include <tox/tox.h> | ||||
|  | ||||
| #include "execute.h" | ||||
| #include "windows.h" | ||||
| #include "settings.h" | ||||
| #include "term_mplex.h" | ||||
| #include "toxic.h" | ||||
| #include "settings.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| extern struct ToxWindow *prompt; | ||||
| extern struct user_settings *user_settings; | ||||
| @@ -100,9 +100,19 @@ static char *read_into_dyn_buffer(FILE *stream) | ||||
|         int length = dyn_buffer_size + strlen(input_ptr); | ||||
|  | ||||
|         if (dyn_buffer) { | ||||
|             dyn_buffer = (char *) realloc(dyn_buffer, length); | ||||
|             char *tmp = realloc(dyn_buffer, length); | ||||
|  | ||||
|             if (tmp == NULL) { | ||||
|                 return NULL; | ||||
|             } | ||||
|  | ||||
|             dyn_buffer = tmp; | ||||
|         } else { | ||||
|             dyn_buffer = (char *) malloc(length); | ||||
|             dyn_buffer = malloc(length); | ||||
|  | ||||
|             if (dyn_buffer == NULL) { | ||||
|                 return NULL; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         strcpy(dyn_buffer + dyn_buffer_size - 1, input_ptr); | ||||
| @@ -143,7 +153,12 @@ static char *extract_socket_path(const char *info) | ||||
|         return NULL; | ||||
|     } | ||||
|  | ||||
|     path = (char *) malloc(end - pos + 1); | ||||
|     path = malloc(end - pos + 1); | ||||
|  | ||||
|     if (path == NULL) { | ||||
|         return NULL; | ||||
|     } | ||||
|  | ||||
|     *end = '\0'; | ||||
|     return strcpy(path, pos); | ||||
| } | ||||
| @@ -300,7 +315,12 @@ static int tmux_is_detached(void) | ||||
|     session_info_stream = NULL; | ||||
|  | ||||
|     /* prepare search string, for finding the current session's entry */ | ||||
|     search_str = (char *) malloc(numstr_len + 2); | ||||
|     search_str = malloc(numstr_len + 2); | ||||
|  | ||||
|     if (search_str == NULL) { | ||||
|         goto fail; | ||||
|     } | ||||
|  | ||||
|     search_str[0] = '\n'; | ||||
|     strcpy(search_str + 1, mplex_data); | ||||
|  | ||||
|   | ||||
							
								
								
									
										594
									
								
								src/toxic.c
									
									
									
									
									
								
							
							
						
						
									
										594
									
								
								src/toxic.c
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										19
									
								
								src/toxic.h
									
									
									
									
									
								
							
							
						
						
									
										19
									
								
								src/toxic.h
									
									
									
									
									
								
							| @@ -48,6 +48,7 @@ | ||||
| #define TOXIC_MAX_NAME_LENGTH 32   /* Must be <= TOX_MAX_NAME_LENGTH */ | ||||
| #define KEY_IDENT_DIGITS 3    /* number of hex digits to display for the pub-key based identifier */ | ||||
| #define TIME_STR_SIZE 32 | ||||
| #define COLOR_STR_SIZE 10 /* should fit every color option */ | ||||
|  | ||||
| #ifndef MAX_PORT_RANGE | ||||
| #define MAX_PORT_RANGE 65535 | ||||
| @@ -61,8 +62,6 @@ | ||||
| #define T_KEY_PREV       0x0F     /* ctrl-o */ | ||||
| #define T_KEY_C_E        0x05     /* ctrl-e */ | ||||
| #define T_KEY_C_A        0x01     /* ctrl-a */ | ||||
| #define T_KEY_C_RB       0x1D     /* ctrl-] */ | ||||
| #define T_KEY_C_LB       0x1B     /* ctrl-[ */ | ||||
| #define T_KEY_C_V        0x16     /* ctrl-v */ | ||||
| #define T_KEY_C_F        0x06     /* ctrl-f */ | ||||
| #define T_KEY_C_H        0x08     /* ctrl-h */ | ||||
| @@ -71,9 +70,13 @@ | ||||
| #define T_KEY_C_W        0x17     /* ctrl-w */ | ||||
| #define T_KEY_C_B        0x02     /* ctrl-b */ | ||||
| #define T_KEY_C_T        0x14     /* ctrl-t */ | ||||
| #define T_KEY_C_LEFT     0x221    /* ctrl-left arrow */ | ||||
| #define T_KEY_C_RIGHT    0x230    /* ctrl-right arrow */ | ||||
| #define T_KEY_C_UP       0x236    /* ctrl-up arrow */ | ||||
| #define T_KEY_C_DOWN     0x20D    /* ctrl-down arrow */ | ||||
| #define T_KEY_TAB        0x09     /* TAB key */ | ||||
|  | ||||
| #define ONLINE_CHAR "*" | ||||
| #define ONLINE_CHAR  "o" | ||||
| #define OFFLINE_CHAR "o" | ||||
|  | ||||
| typedef enum _FATAL_ERRS { | ||||
| @@ -113,14 +116,14 @@ void on_friend_name(Tox *m, uint32_t friendnumber, const uint8_t *string, size_t | ||||
| void on_friend_status(Tox *m, uint32_t friendnumber, Tox_User_Status status, void *userdata); | ||||
| void on_friend_status_message(Tox *m, uint32_t friendnumber, const uint8_t *string, size_t length, void *userdata); | ||||
| void on_friend_added(Tox *m, uint32_t friendnumber, bool sort); | ||||
| void on_conference_message(Tox *m, uint32_t groupnumber, uint32_t peernumber, Tox_Message_Type type, | ||||
| void on_conference_message(Tox *m, uint32_t conferencenumber, uint32_t peernumber, Tox_Message_Type type, | ||||
|                            const uint8_t *message, size_t length, void *userdata); | ||||
| void on_conference_invite(Tox *m, uint32_t friendnumber, Tox_Conference_Type type, const uint8_t *group_pub_key, | ||||
| void on_conference_invite(Tox *m, uint32_t friendnumber, Tox_Conference_Type type, const uint8_t *conference_pub_key, | ||||
|                           size_t length, void *userdata); | ||||
| void on_conference_peer_list_changed(Tox *m, uint32_t groupnumber, void *userdata); | ||||
| void on_conference_peer_name(Tox *m, uint32_t groupnumber, uint32_t peernumber, const uint8_t *name, | ||||
| void on_conference_peer_list_changed(Tox *m, uint32_t conferencenumber, void *userdata); | ||||
| void on_conference_peer_name(Tox *m, uint32_t conferencenumber, uint32_t peernumber, const uint8_t *name, | ||||
|                              size_t length, void *userdata); | ||||
| void on_conference_title(Tox *m, uint32_t groupnumber, uint32_t peernumber, const uint8_t *title, size_t length, | ||||
| void on_conference_title(Tox *m, uint32_t conferencenumber, uint32_t peernumber, const uint8_t *title, size_t length, | ||||
|                          void *userdata); | ||||
| void on_file_chunk_request(Tox *m, uint32_t friendnumber, uint32_t filenumber, uint64_t position, size_t length, | ||||
|                            void *userdata); | ||||
|   | ||||
| @@ -24,11 +24,11 @@ | ||||
| #include <string.h> | ||||
| #include <wchar.h> | ||||
|  | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "misc_tools.h" | ||||
| #include "toxic_strings.h" | ||||
| #include "notify.h" | ||||
| #include "toxic.h" | ||||
| #include "toxic_strings.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| /* Adds char to line at pos. Return 0 on success, -1 if line buffer is full */ | ||||
| int add_char_to_buf(ChatContext *ctx, wint_t ch) | ||||
| @@ -236,6 +236,12 @@ void add_line_to_hist(ChatContext *ctx) | ||||
|    resets line if at end of history */ | ||||
| void fetch_hist_item(ChatContext *ctx, int key_dir) | ||||
| { | ||||
|     if (wcscmp(ctx->line, L"\0") != 0 | ||||
|             && ctx->hst_pos == ctx->hst_tot) { | ||||
|         add_line_to_hist(ctx); | ||||
|         ctx->hst_pos--; | ||||
|     } | ||||
|  | ||||
|     if (key_dir == KEY_UP) { | ||||
|         if (--ctx->hst_pos < 0) { | ||||
|             ctx->hst_pos = 0; | ||||
|   | ||||
							
								
								
									
										335
									
								
								src/video_call.c
									
									
									
									
									
								
							
							
						
						
									
										335
									
								
								src/video_call.c
									
									
									
									
									
								
							| @@ -20,26 +20,27 @@ | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "video_call.h" | ||||
| #include "video_device.h" | ||||
| #include "chat_commands.h" | ||||
| #include "global_commands.h" | ||||
| #include "line_info.h" | ||||
| #include "misc_tools.h" | ||||
| #include "notify.h" | ||||
| #include "toxic.h" | ||||
| #include "video_call.h" | ||||
| #include "video_device.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| #include <stdbool.h> | ||||
| #include <curses.h> | ||||
| #include <string.h> | ||||
| #include <pthread.h> | ||||
| #include <unistd.h> | ||||
| #include <stdlib.h> | ||||
| #include <assert.h> | ||||
| #include <curses.h> | ||||
| #include <pthread.h> | ||||
| #include <stdbool.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <unistd.h> | ||||
|  | ||||
| #ifdef VIDEO | ||||
|  | ||||
| #define default_video_bit_rate 5000 | ||||
| #define DEFAULT_VIDEO_BIT_RATE 5000 | ||||
|  | ||||
| void on_video_receive_frame(ToxAV *av, uint32_t friend_number, | ||||
|                             uint16_t width, uint16_t height, | ||||
| @@ -51,25 +52,27 @@ void on_video_bit_rate(ToxAV *av, uint32_t friend_number, uint32_t video_bit_rat | ||||
|  | ||||
| static void print_err(ToxWindow *self, const char *error_str) | ||||
| { | ||||
|     line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "%s", error_str); | ||||
|     line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "%s", error_str); | ||||
| } | ||||
|  | ||||
| ToxAV *init_video(ToxWindow *self, Tox *tox) | ||||
| { | ||||
|     UNUSED_VAR(tox); | ||||
|  | ||||
|     CallControl.video_errors = ve_None; | ||||
|  | ||||
|     CallControl.video_enabled = true; | ||||
|     CallControl.video_bit_rate = 0; | ||||
|     CallControl.default_video_bit_rate = 0; | ||||
|     CallControl.video_frame_duration = 10; | ||||
|  | ||||
|     if (!CallControl.av) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Video failed to init with ToxAV instance"); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Video failed to init with ToxAV instance"); | ||||
|  | ||||
|         return NULL; | ||||
|     } | ||||
|  | ||||
|     if (init_video_devices(CallControl.av) == vde_InternalError) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to init video devices"); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to init video devices"); | ||||
|  | ||||
|         return NULL; | ||||
|     } | ||||
| @@ -89,8 +92,9 @@ void terminate_video(void) | ||||
|  | ||||
|         stop_video_transmission(this_call, i); | ||||
|  | ||||
|         if (this_call->vout_idx != -1) { | ||||
|         if (this_call->status == cs_Active && this_call->vout_idx != -1) { | ||||
|             close_video_device(vdt_output, this_call->vout_idx); | ||||
|             this_call->vout_idx = -1; | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -105,18 +109,18 @@ void read_video_device_callback(int16_t width, int16_t height, const uint8_t *y, | ||||
|     Toxav_Err_Send_Frame error; | ||||
|  | ||||
|     /* Drop frame if video sending is disabled */ | ||||
|     if (CallControl.video_bit_rate == 0 || this_call->vin_idx == -1) { | ||||
|         line_info_add(CallControl.prompt, NULL, NULL, NULL, SYS_MSG, 0, 0, "Video frame dropped."); | ||||
|     if (this_call->video_bit_rate == 0 || this_call->status != cs_Active || this_call->vin_idx == -1) { | ||||
|         line_info_add(CallControl.prompt, false, NULL, NULL, SYS_MSG, 0, 0, "Video frame dropped."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (toxav_video_send_frame(CallControl.av, friend_number, width, height, y, u, v, &error) == false) { | ||||
|         line_info_add(CallControl.prompt, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to send video frame"); | ||||
|         line_info_add(CallControl.prompt, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to send video frame"); | ||||
|  | ||||
|         if (error == TOXAV_ERR_SEND_FRAME_NULL) { | ||||
|             line_info_add(CallControl.prompt, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to capture video frame"); | ||||
|             line_info_add(CallControl.prompt, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to capture video frame"); | ||||
|         } else if (error == TOXAV_ERR_SEND_FRAME_INVALID) { | ||||
|             line_info_add(CallControl.prompt, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to prepare video frame"); | ||||
|             line_info_add(CallControl.prompt, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to prepare video frame"); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -126,30 +130,30 @@ void write_video_device_callback(uint32_t friend_number, uint16_t width, uint16_ | ||||
|                                  int32_t ystride, int32_t ustride, int32_t vstride, | ||||
|                                  void *user_data) | ||||
| { | ||||
|     UNUSED_VAR(friend_number); | ||||
|  | ||||
|     write_video_out(width, height, y, u, v, ystride, ustride, vstride, user_data); | ||||
| } | ||||
|  | ||||
| int start_video_transmission(ToxWindow *self, ToxAV *av, Call *call) | ||||
| { | ||||
|     if (!self || !av) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to prepare transmission"); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to prepare video transmission"); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     CallControl.video_bit_rate = default_video_bit_rate; | ||||
|  | ||||
|     if (toxav_video_set_bit_rate(CallControl.av, self->num, CallControl.video_bit_rate, NULL) == false) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to set video bit rate"); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (open_primary_video_device(vdt_input, &call->vin_idx) != vde_None) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to open input video device!"); | ||||
|     if (open_primary_video_device(vdt_input, &call->vin_idx, &call->video_width, &call->video_height) != vde_None) { | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to open input video device!"); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (register_video_device_callback(self->num, call->vin_idx, read_video_device_callback, &self->num) != vde_None) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to register input video handler!"); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to register input video handler!"); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (!toxav_video_set_bit_rate(CallControl.av, self->num, call->video_bit_rate, NULL)) { | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to set video bit rate"); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
| @@ -158,8 +162,12 @@ int start_video_transmission(ToxWindow *self, ToxAV *av, Call *call) | ||||
|  | ||||
| int stop_video_transmission(Call *call, int friend_number) | ||||
| { | ||||
|     CallControl.video_bit_rate = 0; | ||||
|     toxav_video_set_bit_rate(CallControl.av, friend_number, CallControl.video_bit_rate, NULL); | ||||
|     if (call->status != cs_Active) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     call->video_bit_rate = 0; | ||||
|     toxav_video_set_bit_rate(CallControl.av, friend_number, call->video_bit_rate, NULL); | ||||
|  | ||||
|     if (call->vin_idx != -1) { | ||||
|         close_video_device(vdt_input, call->vin_idx); | ||||
| @@ -185,35 +193,48 @@ void on_video_receive_frame(ToxAV *av, uint32_t friend_number, | ||||
|                             int32_t ystride, int32_t ustride, int32_t vstride, | ||||
|                             void *user_data) | ||||
| { | ||||
|     UNUSED_VAR(av); | ||||
|  | ||||
|     write_video_device_callback(friend_number, width, height, y, u, v, ystride, ustride, vstride, user_data); | ||||
| } | ||||
|  | ||||
| void on_video_bit_rate(ToxAV *av, uint32_t friend_number, uint32_t video_bit_rate, void *user_data) | ||||
| { | ||||
|     CallControl.video_bit_rate = video_bit_rate; | ||||
|     toxav_video_set_bit_rate(CallControl.av, friend_number, CallControl.video_bit_rate, NULL); | ||||
|     UNUSED_VAR(av); | ||||
|     UNUSED_VAR(user_data); | ||||
|  | ||||
|     Call *call = &CallControl.calls[friend_number]; | ||||
|     call->video_bit_rate = video_bit_rate; | ||||
|  | ||||
|     /* TODO: with current toxav using one-pass VP8, the value of | ||||
|      * video_bit_rate has no effect, except to disable video if it is 0. | ||||
|      * Automatically change resolution instead? */ | ||||
|     toxav_video_set_bit_rate(CallControl.av, friend_number, call->video_bit_rate, NULL); | ||||
| } | ||||
|  | ||||
| void callback_recv_video_starting(uint32_t friend_number) | ||||
| { | ||||
|     Call *this_call = &CallControl.calls[friend_number]; | ||||
|  | ||||
|     if (this_call->vout_idx != -1) { | ||||
|     if (this_call->status != cs_Active || this_call->vout_idx != -1) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     open_primary_video_device(vdt_output, &this_call->vout_idx); | ||||
|     open_primary_video_device(vdt_output, &this_call->vout_idx, NULL, NULL); | ||||
| } | ||||
| void callback_recv_video_end(uint32_t friend_number) | ||||
| { | ||||
|     Call *this_call = &CallControl.calls[friend_number]; | ||||
|  | ||||
|     if (this_call->status != cs_Active || this_call->vout_idx == -1) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     close_video_device(vdt_output, this_call->vout_idx); | ||||
|     this_call->vout_idx = -1; | ||||
| } | ||||
| void callback_video_starting(uint32_t friend_number) | ||||
| { | ||||
|     ToxWindow *windows = CallControl.prompt; | ||||
|     Call *this_call = &CallControl.calls[friend_number]; | ||||
|  | ||||
|     Toxav_Err_Call_Control error = TOXAV_ERR_CALL_CONTROL_OK; | ||||
| @@ -223,13 +244,12 @@ void callback_video_starting(uint32_t friend_number) | ||||
|         size_t i; | ||||
|  | ||||
|         for (i = 0; i < MAX_WINDOWS_NUM; ++i) { | ||||
|             if (windows[i].is_call && windows[i].num == friend_number) { | ||||
|                 if (0 != start_video_transmission(&windows[i], CallControl.av, this_call)) { | ||||
|                     line_info_add(&windows[i], NULL, NULL, NULL, SYS_MSG, 0, 0, "Error starting transmission!"); | ||||
|                     return; | ||||
|                 } | ||||
|             ToxWindow *window = get_window_ptr(i); | ||||
|  | ||||
|                 line_info_add(&windows[i], NULL, NULL, NULL, SYS_MSG, 0, 0, "Video capture starting."); | ||||
|             if (window != NULL && window->is_call && window->num == friend_number) { | ||||
|                 if (start_video_transmission(window, CallControl.av, this_call) == 0) { | ||||
|                     line_info_add(window, NULL, NULL, NULL, SYS_MSG, 0, 0, "Video capture starting."); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| @@ -247,54 +267,137 @@ void callback_video_end(uint32_t friend_number) | ||||
| /* | ||||
|  * Commands from chat_commands.h | ||||
|  */ | ||||
| void cmd_video(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| void cmd_vcall(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     const char *error_str; | ||||
|     Call *this_call = &CallControl.calls[self->num]; | ||||
|     UNUSED_VAR(window); | ||||
|     UNUSED_VAR(m); | ||||
|     UNUSED_VAR(argv); | ||||
|  | ||||
|     if (argc != 0) { | ||||
|         error_str = "Unknown arguments."; | ||||
|         goto on_error; | ||||
|         print_err(self, "Unknown arguments."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (!CallControl.av) { | ||||
|         error_str = "ToxAV not supported!"; | ||||
|         goto on_error; | ||||
|         print_err(self, "ToxAV not supported!"); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (!self->stb->connection) { | ||||
|         error_str = "Friend is offline."; | ||||
|         goto on_error; | ||||
|         print_err(self, "Friend is offline."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (!self->is_call) { | ||||
|         error_str = "Not in call!"; | ||||
|         goto on_error; | ||||
|     Call *call = &CallControl.calls[self->num]; | ||||
|  | ||||
|     if (call->status != cs_None) { | ||||
|         print_err(self, "Already calling."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     init_call(call); | ||||
|  | ||||
|     call->video_bit_rate = DEFAULT_VIDEO_BIT_RATE; | ||||
|  | ||||
|     place_call(self); | ||||
| } | ||||
|  | ||||
| void cmd_video(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|     UNUSED_VAR(m); | ||||
|     UNUSED_VAR(argv); | ||||
|  | ||||
|     Call *this_call = &CallControl.calls[self->num]; | ||||
|  | ||||
|     if (argc != 0) { | ||||
|         print_err(self, "Unknown arguments."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (!CallControl.av) { | ||||
|         print_err(self, "ToxAV not supported!"); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (!self->stb->connection) { | ||||
|         print_err(self, "Friend is offline."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (this_call->status != cs_Active) { | ||||
|         print_err(self, "Not in call!"); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (this_call->vin_idx == -1) { | ||||
|         this_call->video_bit_rate = DEFAULT_VIDEO_BIT_RATE; | ||||
|         callback_video_starting(self->num); | ||||
|     } else { | ||||
|         callback_video_end(self->num); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void cmd_res(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     UNUSED_VAR(window); | ||||
|     UNUSED_VAR(m); | ||||
|  | ||||
|     Call *call = &CallControl.calls[self->num]; | ||||
|  | ||||
|     if (argc == 0) { | ||||
|         if (call->status == cs_Active && call->vin_idx != -1) { | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, | ||||
|                           "Resolution of current call: %u x %u", | ||||
|                           call->video_width, call->video_height); | ||||
|         } else { | ||||
|             line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, | ||||
|                           "Initial resolution for video calls: %u x %u", | ||||
|                           CallControl.default_video_width, CallControl.default_video_height); | ||||
|         } | ||||
|  | ||||
|         return; | ||||
| on_error: | ||||
|     print_err(self, error_str); | ||||
|     } | ||||
|  | ||||
|     if (argc != 2) { | ||||
|         print_err(self, "Require 0 or 2 arguments."); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     char *endw, *endh; | ||||
|     const long int width = strtol(argv[1], &endw, 10); | ||||
|     const long int height = strtol(argv[2], &endh, 10); | ||||
|  | ||||
|     if (*endw || *endh || width < 0 || height < 0) { | ||||
|         print_err(self, "Invalid input"); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (call->status == cs_Active && call->vin_idx != -1) { | ||||
|         stop_video_transmission(call, self->num); | ||||
|         call->video_width = width; | ||||
|         call->video_height = height; | ||||
|         call->video_bit_rate = DEFAULT_VIDEO_BIT_RATE; | ||||
|         start_video_transmission(self, CallControl.av, call); | ||||
|     } else { | ||||
|         CallControl.default_video_width = width; | ||||
|         CallControl.default_video_height = height; | ||||
|     } | ||||
| } | ||||
|  | ||||
| void cmd_list_video_devices(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     const char *error_str; | ||||
|     UNUSED_VAR(window); | ||||
|     UNUSED_VAR(m); | ||||
|  | ||||
|     if (argc != 1) { | ||||
|         if (argc < 1) { | ||||
|             error_str = "Type must be specified!"; | ||||
|             print_err(self, "Type must be specified!"); | ||||
|         } else { | ||||
|             error_str = "Only one argument allowed!"; | ||||
|             print_err(self, "Only one argument allowed!"); | ||||
|         } | ||||
|  | ||||
|         goto on_error; | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     VideoDeviceType type; | ||||
| @@ -308,32 +411,29 @@ void cmd_list_video_devices(WINDOW *window, ToxWindow *self, Tox *m, int argc, c | ||||
|     } | ||||
|  | ||||
|     else { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invalid type: %s", argv[1]); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid type: %s", argv[1]); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     print_video_devices(self, type); | ||||
|  | ||||
|     return; | ||||
| on_error: | ||||
|     print_err(self, error_str); | ||||
| } | ||||
|  | ||||
| /* This changes primary video device only */ | ||||
| void cmd_change_video_device(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     const char *error_str; | ||||
|     UNUSED_VAR(window); | ||||
|     UNUSED_VAR(m); | ||||
|  | ||||
|     if (argc != 2) { | ||||
|         if (argc < 1) { | ||||
|             error_str = "Type must be specified!"; | ||||
|             print_err(self, "Type must be specified!"); | ||||
|         } else if (argc < 2) { | ||||
|             error_str = "Must have id!"; | ||||
|             print_err(self, "Must have id!"); | ||||
|         } else { | ||||
|             error_str = "Only two arguments allowed!"; | ||||
|             print_err(self, "Only two arguments allowed!"); | ||||
|         } | ||||
|  | ||||
|         goto on_error; | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     VideoDeviceType type; | ||||
| @@ -347,7 +447,7 @@ void cmd_change_video_device(WINDOW *window, ToxWindow *self, Tox *m, int argc, | ||||
|     } | ||||
|  | ||||
|     else { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invalid type: %s", argv[1]); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid type: %s", argv[1]); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -356,94 +456,13 @@ void cmd_change_video_device(WINDOW *window, ToxWindow *self, Tox *m, int argc, | ||||
|     long int selection = strtol(argv[2], &end, 10); | ||||
|  | ||||
|     if (*end) { | ||||
|         error_str = "Invalid input"; | ||||
|         goto on_error; | ||||
|         print_err(self, "Invalid input"); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (set_primary_video_device(type, selection) == vde_InvalidSelection) { | ||||
|         error_str = "Invalid selection!"; | ||||
|         goto on_error; | ||||
|     } | ||||
|  | ||||
|         print_err(self, "Invalid selection!"); | ||||
|         return; | ||||
| on_error: | ||||
|     print_err(self, error_str); | ||||
| } | ||||
|  | ||||
| void cmd_ccur_video_device(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]) | ||||
| { | ||||
|     const char *error_str; | ||||
|  | ||||
|     if (argc != 2) { | ||||
|         if (argc < 1) { | ||||
|             error_str = "Type must be specified!"; | ||||
|         } else if (argc < 2) { | ||||
|             error_str = "Must have id!"; | ||||
|         } else { | ||||
|             error_str = "Only two arguments allowed!"; | ||||
|         } | ||||
|  | ||||
|         goto on_error; | ||||
|     } | ||||
|  | ||||
|     VideoDeviceType type; | ||||
|  | ||||
|     if (strcmp(argv[1], "in") == 0) { /* Input devices */ | ||||
|         type = vdt_input; | ||||
|     } | ||||
|  | ||||
|     else if (strcmp(argv[1], "out") == 0) { /* Output devices */ | ||||
|         type = vdt_output; | ||||
|     } | ||||
|  | ||||
|     else { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invalid type: %s", argv[1]); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|  | ||||
|     char *end; | ||||
|     long int selection = strtol(argv[2], &end, 10); | ||||
|  | ||||
|     if (*end) { | ||||
|         error_str = "Invalid input"; | ||||
|         goto on_error; | ||||
|     } | ||||
|  | ||||
|     if (video_selection_valid(type, selection) == vde_InvalidSelection) { | ||||
|         error_str = "Invalid selection!"; | ||||
|         goto on_error; | ||||
|     } | ||||
|  | ||||
|     /* If call is active, change device */ | ||||
|     if (self->is_call) { | ||||
|         Call *this_call = &CallControl.calls[self->num]; | ||||
|  | ||||
|         if (this_call->ttas) { | ||||
|  | ||||
|             if (type == vdt_output) { | ||||
|             } else { | ||||
|                 /* TODO: check for failure */ | ||||
|                 close_video_device(vdt_input, this_call->vin_idx); | ||||
|                 open_video_device(vdt_input, selection, &this_call->vin_idx); | ||||
|                 register_video_device_callback(self->num, this_call->vin_idx, read_video_device_callback, &self->num); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     self->video_device_selection[type] = selection; | ||||
|  | ||||
|     return; | ||||
| on_error: | ||||
|     print_err(self, error_str); | ||||
| } | ||||
|  | ||||
| void stop_video_stream(ToxWindow *self) | ||||
| { | ||||
|     Call *this_call = &CallControl.calls[self->num]; | ||||
|  | ||||
|     if (this_call && this_call->vin_idx != -1) { | ||||
|         stop_video_transmission(this_call, self->num); | ||||
|     } | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -34,7 +34,6 @@ ToxAV *init_video(ToxWindow *self, Tox *tox); | ||||
| void terminate_video(void); | ||||
| int start_video_transmission(ToxWindow *self, ToxAV *av, Call *call); | ||||
| int stop_video_transmission(Call *call, int friend_number); | ||||
| void stop_video_stream(ToxWindow *self); | ||||
|  | ||||
| void callback_recv_video_starting(uint32_t friend_number); | ||||
| void callback_recv_video_end(uint32_t friend_number); | ||||
|   | ||||
| @@ -20,40 +20,40 @@ | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include "video_device.h" | ||||
| #include "video_call.h" | ||||
| #include "video_device.h" | ||||
|  | ||||
| #include <sys/ioctl.h> | ||||
| #include <X11/Xlib.h> | ||||
| #include <X11/Xutil.h> | ||||
| #include <X11/Xos.h> | ||||
|  | ||||
| #include <vpx/vpx_image.h> | ||||
|  | ||||
| #if defined(__OSX__) | ||||
| #if defined(__OSX__) || defined(__APPLE__) | ||||
| #import "osx_video.h" | ||||
| #else | ||||
| #include <sys/types.h> | ||||
| #include <sys/stat.h> | ||||
| #include <sys/mman.h> | ||||
| #include <fcntl.h> | ||||
| #include <sys/mman.h> | ||||
| #include <sys/stat.h> | ||||
| #include <sys/types.h> | ||||
| #include <X11/Xlib.h> | ||||
| #include <X11/Xos.h> | ||||
| #include <X11/Xutil.h> | ||||
| #if defined(__OpenBSD__) || defined(__NetBSD__) | ||||
| #include <sys/videoio.h> | ||||
| #else | ||||
| #include <linux/videodev2.h> | ||||
| #endif /* defined(__OpenBSD__) || defined(__NetBSD__) */ | ||||
| #endif /* __OSX__ */ | ||||
| #endif /* __OSX__ || __APPLE__ */ | ||||
|  | ||||
| #include "line_info.h" | ||||
| #include "misc_tools.h" | ||||
| #include "settings.h" | ||||
|  | ||||
| #include <errno.h> | ||||
|  | ||||
| #include <stdbool.h> | ||||
| #include <string.h> | ||||
| #include <pthread.h> | ||||
| #include <unistd.h> | ||||
| #include <stdbool.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <unistd.h> | ||||
|  | ||||
| #ifdef VIDEO | ||||
|  | ||||
| @@ -71,7 +71,7 @@ typedef struct VideoDevice { | ||||
|     void *cb_data;                          /* Data to be passed to callback */ | ||||
|     int32_t friend_number;                  /* ToxAV friend number */ | ||||
|  | ||||
| #if !defined(__OSX__) | ||||
| #if !(defined(__OSX__) || defined(__APPLE__)) | ||||
|     int fd;                                 /* File descriptor of video device selected/opened */ | ||||
|     struct v4l2_format fmt; | ||||
|     struct VideoBuffer *buffers; | ||||
| @@ -101,8 +101,8 @@ uint32_t primary_video_device[2];          /* Primary device */ | ||||
| static ToxAV *av = NULL; | ||||
|  | ||||
| /* q_mutex */ | ||||
| #define lock pthread_mutex_lock(&video_mutex); | ||||
| #define unlock pthread_mutex_unlock(&video_mutex); | ||||
| #define lock pthread_mutex_lock(&video_mutex) | ||||
| #define unlock pthread_mutex_unlock(&video_mutex) | ||||
| pthread_mutex_t video_mutex; | ||||
|  | ||||
| bool video_thread_running = true, | ||||
| @@ -136,7 +136,7 @@ static void yuv420tobgr(uint16_t width, uint16_t height, const uint8_t *y, | ||||
|     } | ||||
| } | ||||
|  | ||||
| #if !defined(__OSX__) | ||||
| #if !(defined(__OSX__) || defined(__APPLE__)) | ||||
| static void yuv422to420(uint8_t *plane_y, uint8_t *plane_u, uint8_t *plane_v, | ||||
|                         uint8_t *input, uint16_t width, uint16_t height) | ||||
| { | ||||
| @@ -180,18 +180,18 @@ static int xioctl(int fh, unsigned long request, void *arg) | ||||
| #ifdef VIDEO | ||||
| VideoDeviceError init_video_devices(ToxAV *av_) | ||||
| #else | ||||
| VideoDeviceError init_video_devices() | ||||
| VideoDeviceError init_video_devices(void) | ||||
| #endif /* VIDEO */ | ||||
| { | ||||
|     size[vdt_input] = 0; | ||||
|  | ||||
| #if defined(__OSX__) | ||||
| #if defined(__OSX__) || defined(__APPLE__) | ||||
|  | ||||
|     if (osx_video_init((char **)video_devices_names[vdt_input], &size[vdt_input]) != 0) { | ||||
|         return vde_InternalError; | ||||
|     } | ||||
|  | ||||
| #else /* not __OSX__*/ | ||||
| #else /* not __OSX__ || __APPLE__ */ | ||||
|  | ||||
|     for (; size[vdt_input] <= MAX_DEVICES; ++size[vdt_input]) { | ||||
|         int fd; | ||||
| @@ -257,7 +257,7 @@ VideoDeviceError terminate_video_devices(void) | ||||
|     video_thread_running = false; | ||||
|     unlock; | ||||
|  | ||||
|     usleep(20000); | ||||
|     sleep_thread(20000L); | ||||
|  | ||||
|     int i; | ||||
|  | ||||
| @@ -269,9 +269,9 @@ VideoDeviceError terminate_video_devices(void) | ||||
|         return (VideoDeviceError) vde_InternalError; | ||||
|     } | ||||
|  | ||||
| #ifdef __OSX__ | ||||
| #if defined(__OSX__) || defined(__APPLE__) | ||||
|     osx_video_release(); | ||||
| #endif /* __OSX__ */ | ||||
| #endif /* __OSX__ || __APPLE__ */ | ||||
|  | ||||
|     return (VideoDeviceError) vde_None; | ||||
| } | ||||
| @@ -279,13 +279,13 @@ VideoDeviceError terminate_video_devices(void) | ||||
| VideoDeviceError register_video_device_callback(int32_t friend_number, uint32_t device_idx, | ||||
|         VideoDataHandleCallback callback, void *data) | ||||
| { | ||||
| #if defined(__OSX__) | ||||
| #if defined(__OSX__) || defined(__APPLE__) | ||||
|  | ||||
|     if (size[vdt_input] <= device_idx || !video_devices_running[vdt_input][device_idx]) { | ||||
|         return vde_InvalidSelection; | ||||
|     } | ||||
|  | ||||
| #else /* not __OSX__ */ | ||||
| #else /* not __OSX__ || __APPLE__ */ | ||||
|  | ||||
|     if (size[vdt_input] <= device_idx || !video_devices_running[vdt_input][device_idx] | ||||
|             || !video_devices_running[vdt_input][device_idx]->fd) { | ||||
| @@ -314,9 +314,10 @@ VideoDeviceError set_primary_video_device(VideoDeviceType type, int32_t selectio | ||||
|     return vde_None; | ||||
| } | ||||
|  | ||||
| VideoDeviceError open_primary_video_device(VideoDeviceType type, uint32_t *device_idx) | ||||
| VideoDeviceError open_primary_video_device(VideoDeviceType type, uint32_t *device_idx, | ||||
|         uint32_t *width, uint32_t *height) | ||||
| { | ||||
|     return open_video_device(type, primary_video_device[type], device_idx); | ||||
|     return open_video_device(type, primary_video_device[type], device_idx, width, height); | ||||
| } | ||||
|  | ||||
| void get_primary_video_device_name(VideoDeviceType type, char *buf, int size) | ||||
| @@ -324,7 +325,8 @@ void get_primary_video_device_name(VideoDeviceType type, char *buf, int size) | ||||
|     memcpy(buf, dvideo_device_names[type], size); | ||||
| } | ||||
|  | ||||
| VideoDeviceError open_video_device(VideoDeviceType type, int32_t selection, uint32_t *device_idx) | ||||
| VideoDeviceError open_video_device(VideoDeviceType type, int32_t selection, uint32_t *device_idx, | ||||
|                                    uint32_t *width, uint32_t *height) | ||||
| { | ||||
|     if (size[type] <= selection || selection < 0) { | ||||
|         return vde_InvalidSelection; | ||||
| @@ -369,15 +371,16 @@ VideoDeviceError open_video_device(VideoDeviceType type, int32_t selection, uint | ||||
|     if (type == vdt_input) { | ||||
|         video_thread_paused = true; | ||||
|  | ||||
| #if defined(__OSX__) | ||||
| #if defined(__OSX__) || defined(__APPLE__) | ||||
|  | ||||
|         /* TODO: use requested resolution */ | ||||
|         if (osx_video_open_device(selection, &device->video_width, &device->video_height) != 0) { | ||||
|             free(device); | ||||
|             unlock; | ||||
|             return vde_FailedStart; | ||||
|         } | ||||
|  | ||||
| #else /* not __OSX__*/ | ||||
| #else /* not __OSX__ || __APPLE__ */ | ||||
|         /* Open selected device */ | ||||
|         char device_address[] = "/dev/videoXX"; | ||||
|         snprintf(device_address + 10, sizeof(device_address) - 10, "%i", selection); | ||||
| @@ -400,11 +403,12 @@ VideoDeviceError open_video_device(VideoDeviceType type, int32_t selection, uint | ||||
|         } | ||||
|  | ||||
|         /* Setup video format */ | ||||
|         struct v4l2_format fmt; | ||||
|         memset(&(fmt), 0, sizeof(fmt)); | ||||
|         struct v4l2_format fmt = {0}; | ||||
|  | ||||
|         fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; | ||||
|         fmt.fmt.pix.pixelformat = V4L2_PIX_FMT_YUYV; | ||||
|         fmt.fmt.pix.width = width == NULL ? 0 : *width; | ||||
|         fmt.fmt.pix.height = height == NULL ? 0 : *height; | ||||
|  | ||||
|         if (-1 == xioctl(device->fd, VIDIOC_S_FMT, &fmt)) { | ||||
|             close(device->fd); | ||||
| @@ -417,8 +421,8 @@ VideoDeviceError open_video_device(VideoDeviceType type, int32_t selection, uint | ||||
|         device->video_height = fmt.fmt.pix.height; | ||||
|  | ||||
|         /* Request buffers */ | ||||
|         struct v4l2_requestbuffers req; | ||||
|         memset(&(req), 0, sizeof(req)); | ||||
|         struct v4l2_requestbuffers req = {0}; | ||||
|  | ||||
|         req.count = 4; | ||||
|         req.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; | ||||
|         req.memory = V4L2_MEMORY_MMAP; | ||||
| @@ -440,8 +444,7 @@ VideoDeviceError open_video_device(VideoDeviceType type, int32_t selection, uint | ||||
|         device->buffers = calloc(req.count, sizeof(struct VideoBuffer)); | ||||
|  | ||||
|         for (i = 0; i < req.count; ++i) { | ||||
|             struct v4l2_buffer buf; | ||||
|             memset(&(buf), 0, sizeof(buf)); | ||||
|             struct v4l2_buffer buf = {0}; | ||||
|  | ||||
|             buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; | ||||
|             buf.memory = V4L2_MEMORY_MMAP; | ||||
| @@ -478,8 +481,7 @@ VideoDeviceError open_video_device(VideoDeviceType type, int32_t selection, uint | ||||
|         enum v4l2_buf_type type; | ||||
|  | ||||
|         for (i = 0; i < device->n_buffers; ++i) { | ||||
|             struct v4l2_buffer buf; | ||||
|             memset(&(buf), 0, sizeof(buf)); | ||||
|             struct v4l2_buffer buf = {0}; | ||||
|  | ||||
|             buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; | ||||
|             buf.memory = V4L2_MEMORY_MMAP; | ||||
| @@ -545,6 +547,14 @@ VideoDeviceError open_video_device(VideoDeviceType type, int32_t selection, uint | ||||
|  | ||||
|         vpx_img_alloc(&device->input, VPX_IMG_FMT_I420, device->video_width, device->video_height, 1); | ||||
|  | ||||
|         if (width != NULL) { | ||||
|             *width = device->video_width; | ||||
|         } | ||||
|  | ||||
|         if (height != NULL) { | ||||
|             *height = device->video_height; | ||||
|         } | ||||
|  | ||||
|         video_thread_paused = false; | ||||
|     } else { /* vdt_output */ | ||||
|  | ||||
| @@ -596,6 +606,8 @@ VideoDeviceError write_video_out(uint16_t width, uint16_t height, | ||||
|                                  int32_t ystride, int32_t ustride, int32_t vstride, | ||||
|                                  void *user_data) | ||||
| { | ||||
|     UNUSED_VAR(user_data); | ||||
|  | ||||
|     VideoDevice *device = video_devices_running[vdt_output][0]; | ||||
|  | ||||
|     if (!device) { | ||||
| @@ -659,7 +671,7 @@ void *video_thread_poll(void *arg)  // TODO: maybe use thread for every input so | ||||
|     /* | ||||
|      * NOTE: We only need to poll input devices for data. | ||||
|      */ | ||||
|     (void)arg; | ||||
|     UNUSED_VAR(arg); | ||||
|     uint32_t i; | ||||
|  | ||||
|     while (1) { | ||||
| @@ -673,7 +685,7 @@ void *video_thread_poll(void *arg)  // TODO: maybe use thread for every input so | ||||
|         unlock; | ||||
|  | ||||
|         if (video_thread_paused) { | ||||
|             usleep(10000);    /* Wait for unpause. */ | ||||
|             sleep_thread(10000L);    /* Wait for unpause. */ | ||||
|         } else { | ||||
|             for (i = 0; i < size[vdt_input]; ++i) { | ||||
|                 lock; | ||||
| @@ -687,16 +699,15 @@ void *video_thread_poll(void *arg)  // TODO: maybe use thread for every input so | ||||
|                     uint8_t *u = device->input.planes[1]; | ||||
|                     uint8_t *v = device->input.planes[2]; | ||||
|  | ||||
| #if defined(__OSX__) | ||||
| #if defined(__OSX__) || defined(__APPLE__) | ||||
|  | ||||
|                     if (osx_video_read_device(y, u, v, &video_width, &video_height) != 0) { | ||||
|                         unlock; | ||||
|                         continue; | ||||
|                     } | ||||
|  | ||||
| #else /* not __OSX__*/ | ||||
|                     struct v4l2_buffer buf; | ||||
|                     memset(&(buf), 0, sizeof(buf)); | ||||
| #else /* not __OSX__ || __APPLE__ */ | ||||
|                     struct v4l2_buffer buf = {0}; | ||||
|  | ||||
|                     buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; | ||||
|                     buf.memory = V4L2_MEMORY_MMAP; | ||||
| @@ -748,7 +759,7 @@ void *video_thread_poll(void *arg)  // TODO: maybe use thread for every input so | ||||
|                     XFlush(device->x_display); | ||||
|                     free(img_data); | ||||
|  | ||||
| #if !defined(__OSX__) | ||||
| #if !(defined(__OSX__) || defined(__APPLE__)) | ||||
|  | ||||
|                     if (-1 == xioctl(device->fd, VIDIOC_QBUF, &buf)) { | ||||
|                         unlock; | ||||
| @@ -762,7 +773,8 @@ void *video_thread_poll(void *arg)  // TODO: maybe use thread for every input so | ||||
|                 unlock; | ||||
|             } | ||||
|  | ||||
|             usleep(1000 * 1000 / 24); | ||||
|             long int sleep_duration = 1000 * 1000 / 24; | ||||
|             sleep_thread(sleep_duration); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -789,10 +801,10 @@ VideoDeviceError close_video_device(VideoDeviceType type, uint32_t device_idx) | ||||
|     if (!device->ref_count) { | ||||
|  | ||||
|         if (type == vdt_input) { | ||||
| #if defined(__OSX__) | ||||
| #if defined(__OSX__) || defined(__APPLE__) | ||||
|  | ||||
|             osx_video_close_device(device_idx); | ||||
| #else /* not __OSX__ */ | ||||
| #else /* not __OSX__ || __APPLE__ */ | ||||
|             enum v4l2_buf_type buf_type = V4L2_BUF_TYPE_VIDEO_CAPTURE; | ||||
|  | ||||
|             if (-1 == xioctl(device->fd, VIDIOC_STREAMOFF, &buf_type)) {} | ||||
| @@ -813,9 +825,9 @@ VideoDeviceError close_video_device(VideoDeviceType type, uint32_t device_idx) | ||||
|             XCloseDisplay(device->x_display); | ||||
|             pthread_mutex_destroy(device->mutex); | ||||
|  | ||||
| #if !defined(__OSX__) | ||||
| #if !(defined(__OSX__) || defined(__APPLE__)) | ||||
|             free(device->buffers); | ||||
| #endif /* not __OSX__ */ | ||||
| #endif /* not __OSX__ || __APPLE__ */ | ||||
|  | ||||
|             free(device); | ||||
|         } else { | ||||
| @@ -840,7 +852,7 @@ void print_video_devices(ToxWindow *self, VideoDeviceType type) | ||||
|     int i; | ||||
|  | ||||
|     for (i = 0; i < size[type]; ++i) { | ||||
|         line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "%d: %s", i, video_devices_names[type][i]); | ||||
|         line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "%d: %s", i, video_devices_names[type][i]); | ||||
|     } | ||||
|  | ||||
|     return; | ||||
|   | ||||
| @@ -62,9 +62,11 @@ VideoDeviceError register_video_device_callback(int32_t call_idx, uint32_t devic | ||||
| void *get_video_device_callback_data(uint32_t device_idx); | ||||
|  | ||||
| VideoDeviceError set_primary_video_device(VideoDeviceType type, int32_t selection); | ||||
| VideoDeviceError open_primary_video_device(VideoDeviceType type, uint32_t *device_idx); | ||||
| VideoDeviceError open_primary_video_device(VideoDeviceType type, uint32_t *device_idx, | ||||
|         uint32_t *width, uint32_t *height); | ||||
| /* Start device */ | ||||
| VideoDeviceError open_video_device(VideoDeviceType type, int32_t selection, uint32_t *device_idx); | ||||
| VideoDeviceError open_video_device(VideoDeviceType type, int32_t selection, uint32_t *device_idx, | ||||
|                                    uint32_t *width, uint32_t *height); | ||||
| /* Stop device */ | ||||
| VideoDeviceError close_video_device(VideoDeviceType type, uint32_t device_idx); | ||||
|  | ||||
|   | ||||
							
								
								
									
										412
									
								
								src/windows.c
									
									
									
									
									
								
							
							
						
						
									
										412
									
								
								src/windows.c
									
									
									
									
									
								
							| @@ -20,22 +20,22 @@ | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include <ctype.h> | ||||
| #include <pthread.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include <pthread.h> | ||||
| #include <ctype.h> | ||||
|  | ||||
| #include "friendlist.h" | ||||
| #include "prompt.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
| #include "groupchat.h" | ||||
| #include "avatars.h" | ||||
| #include "chat.h" | ||||
| #include "conference.h" | ||||
| #include "file_transfers.h" | ||||
| #include "friendlist.h" | ||||
| #include "line_info.h" | ||||
| #include "misc_tools.h" | ||||
| #include "avatars.h" | ||||
| #include "prompt.h" | ||||
| #include "settings.h" | ||||
| #include "file_transfers.h" | ||||
| #include "toxic.h" | ||||
| #include "windows.h" | ||||
|  | ||||
| extern char *DATA_FILE; | ||||
| extern struct Winthread Winthread; | ||||
| @@ -50,6 +50,8 @@ extern struct user_settings *user_settings; | ||||
| /* CALLBACKS START */ | ||||
| void on_friend_request(Tox *m, const uint8_t *public_key, const uint8_t *data, size_t length, void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(userdata); | ||||
|  | ||||
|     char msg[MAX_STR_SIZE + 1]; | ||||
|     length = copy_tox_str(msg, sizeof(msg), (const char *) data, length); | ||||
|  | ||||
| @@ -62,6 +64,10 @@ void on_friend_request(Tox *m, const uint8_t *public_key, const uint8_t *data, s | ||||
|  | ||||
| void on_friend_connection_status(Tox *m, uint32_t friendnumber, Tox_Connection connection_status, void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(userdata); | ||||
|  | ||||
|     on_avatar_friend_connection_status(m, friendnumber, connection_status); | ||||
|  | ||||
|     for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) { | ||||
|         if (windows[i] != NULL && windows[i]->onConnectionChange != NULL) { | ||||
|             windows[i]->onConnectionChange(windows[i], m, friendnumber, connection_status); | ||||
| @@ -71,6 +77,8 @@ void on_friend_connection_status(Tox *m, uint32_t friendnumber, Tox_Connection c | ||||
|  | ||||
| void on_friend_typing(Tox *m, uint32_t friendnumber, bool is_typing, void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(userdata); | ||||
|  | ||||
|     if (user_settings->show_typing_other == SHOW_TYPING_OFF) { | ||||
|         return; | ||||
|     } | ||||
| @@ -85,6 +93,8 @@ void on_friend_typing(Tox *m, uint32_t friendnumber, bool is_typing, void *userd | ||||
| void on_friend_message(Tox *m, uint32_t friendnumber, Tox_Message_Type type, const uint8_t *string, size_t length, | ||||
|                        void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(userdata); | ||||
|  | ||||
|     char msg[MAX_STR_SIZE + 1]; | ||||
|     length = copy_tox_str(msg, sizeof(msg), (const char *) string, length); | ||||
|  | ||||
| @@ -97,6 +107,8 @@ void on_friend_message(Tox *m, uint32_t friendnumber, Tox_Message_Type type, con | ||||
|  | ||||
| void on_friend_name(Tox *m, uint32_t friendnumber, const uint8_t *string, size_t length, void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(userdata); | ||||
|  | ||||
|     char nick[TOXIC_MAX_NAME_LENGTH + 1]; | ||||
|     length = copy_tox_str(nick, sizeof(nick), (const char *) string, length); | ||||
|     filter_str(nick, length); | ||||
| @@ -112,6 +124,9 @@ void on_friend_name(Tox *m, uint32_t friendnumber, const uint8_t *string, size_t | ||||
|  | ||||
| void on_friend_status_message(Tox *m, uint32_t friendnumber, const uint8_t *string, size_t length, void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(userdata); | ||||
|     UNUSED_VAR(m); | ||||
|  | ||||
|     char msg[TOX_MAX_STATUS_MESSAGE_LENGTH + 1]; | ||||
|     length = copy_tox_str(msg, sizeof(msg), (const char *) string, length); | ||||
|     filter_str(msg, length); | ||||
| @@ -125,6 +140,8 @@ void on_friend_status_message(Tox *m, uint32_t friendnumber, const uint8_t *stri | ||||
|  | ||||
| void on_friend_status(Tox *m, uint32_t friendnumber, Tox_User_Status status, void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(userdata); | ||||
|  | ||||
|     for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) { | ||||
|         if (windows[i] != NULL && windows[i]->onStatusChange != NULL) { | ||||
|             windows[i]->onStatusChange(windows[i], m, friendnumber, status); | ||||
| @@ -143,61 +160,71 @@ void on_friend_added(Tox *m, uint32_t friendnumber, bool sort) | ||||
|     store_data(m, DATA_FILE); | ||||
| } | ||||
|  | ||||
| void on_conference_message(Tox *m, uint32_t groupnumber, uint32_t peernumber, Tox_Message_Type type, | ||||
| void on_conference_message(Tox *m, uint32_t conferencenumber, uint32_t peernumber, Tox_Message_Type type, | ||||
|                            const uint8_t *message, size_t length, void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(userdata); | ||||
|  | ||||
|     char msg[MAX_STR_SIZE + 1]; | ||||
|     length = copy_tox_str(msg, sizeof(msg), (const char *) message, length); | ||||
|  | ||||
|     for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) { | ||||
|         if (windows[i] != NULL && windows[i]->onGroupMessage != NULL) { | ||||
|             windows[i]->onGroupMessage(windows[i], m, groupnumber, peernumber, type, msg, length); | ||||
|         if (windows[i] != NULL && windows[i]->onConferenceMessage != NULL) { | ||||
|             windows[i]->onConferenceMessage(windows[i], m, conferencenumber, peernumber, type, msg, length); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| void on_conference_invite(Tox *m, uint32_t friendnumber, Tox_Conference_Type type, const uint8_t *group_pub_key, | ||||
| void on_conference_invite(Tox *m, uint32_t friendnumber, Tox_Conference_Type type, const uint8_t *conference_pub_key, | ||||
|                           size_t length, void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(userdata); | ||||
|  | ||||
|     for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) { | ||||
|         if (windows[i] != NULL && windows[i]->onGroupInvite != NULL) { | ||||
|             windows[i]->onGroupInvite(windows[i], m, friendnumber, type, (char *) group_pub_key, length); | ||||
|         if (windows[i] != NULL && windows[i]->onConferenceInvite != NULL) { | ||||
|             windows[i]->onConferenceInvite(windows[i], m, friendnumber, type, (char *) conference_pub_key, length); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| void on_conference_peer_list_changed(Tox *m, uint32_t groupnumber, void *userdata) | ||||
| void on_conference_peer_list_changed(Tox *m, uint32_t conferencenumber, void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(userdata); | ||||
|  | ||||
|     for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) { | ||||
|         if (windows[i] != NULL && windows[i]->onGroupNameListChange != NULL) { | ||||
|             windows[i]->onGroupNameListChange(windows[i], m, groupnumber); | ||||
|         if (windows[i] != NULL && windows[i]->onConferenceNameListChange != NULL) { | ||||
|             windows[i]->onConferenceNameListChange(windows[i], m, conferencenumber); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| void on_conference_peer_name(Tox *m, uint32_t groupnumber, uint32_t peernumber, const uint8_t *name, | ||||
| void on_conference_peer_name(Tox *m, uint32_t conferencenumber, uint32_t peernumber, const uint8_t *name, | ||||
|                              size_t length, void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(userdata); | ||||
|  | ||||
|     char nick[TOXIC_MAX_NAME_LENGTH + 1]; | ||||
|     length = copy_tox_str(nick, sizeof(nick), (const char *) name, length); | ||||
|     filter_str(nick, length); | ||||
|  | ||||
|     for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) { | ||||
|         if (windows[i] != NULL && windows[i]->onGroupPeerNameChange != NULL) { | ||||
|             windows[i]->onGroupPeerNameChange(windows[i], m, groupnumber, peernumber, nick, length); | ||||
|         if (windows[i] != NULL && windows[i]->onConferencePeerNameChange != NULL) { | ||||
|             windows[i]->onConferencePeerNameChange(windows[i], m, conferencenumber, peernumber, nick, length); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| void on_conference_title(Tox *m, uint32_t groupnumber, uint32_t peernumber, const uint8_t *title, size_t length, | ||||
| void on_conference_title(Tox *m, uint32_t conferencenumber, uint32_t peernumber, const uint8_t *title, size_t length, | ||||
|                          void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(userdata); | ||||
|  | ||||
|     char data[MAX_STR_SIZE + 1]; | ||||
|     length = copy_tox_str(data, sizeof(data), (const char *) title, length); | ||||
|  | ||||
|     for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) { | ||||
|         if (windows[i] != NULL && windows[i]->onGroupTitleChange != NULL) { | ||||
|             windows[i]->onGroupTitleChange(windows[i], m, groupnumber, peernumber, data, length); | ||||
|         if (windows[i] != NULL && windows[i]->onConferenceTitleChange != NULL) { | ||||
|             windows[i]->onConferenceTitleChange(windows[i], m, conferencenumber, peernumber, data, length); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -205,6 +232,8 @@ void on_conference_title(Tox *m, uint32_t groupnumber, uint32_t peernumber, cons | ||||
| void on_file_chunk_request(Tox *m, uint32_t friendnumber, uint32_t filenumber, uint64_t position, | ||||
|                            size_t length, void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(userdata); | ||||
|  | ||||
|     struct FileTransfer *ft = get_file_transfer_struct(friendnumber, filenumber); | ||||
|  | ||||
|     if (!ft) { | ||||
| @@ -224,8 +253,10 @@ void on_file_chunk_request(Tox *m, uint32_t friendnumber, uint32_t filenumber, u | ||||
| } | ||||
|  | ||||
| void on_file_recv_chunk(Tox *m, uint32_t friendnumber, uint32_t filenumber, uint64_t position, | ||||
|                         const uint8_t *data, size_t length, void *user_data) | ||||
|                         const uint8_t *data, size_t length, void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(userdata); | ||||
|  | ||||
|     struct FileTransfer *ft = get_file_transfer_struct(friendnumber, filenumber); | ||||
|  | ||||
|     if (!ft) { | ||||
| @@ -242,6 +273,8 @@ void on_file_recv_chunk(Tox *m, uint32_t friendnumber, uint32_t filenumber, uint | ||||
| void on_file_recv_control(Tox *m, uint32_t friendnumber, uint32_t filenumber, Tox_File_Control control, | ||||
|                           void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(userdata); | ||||
|  | ||||
|     struct FileTransfer *ft = get_file_transfer_struct(friendnumber, filenumber); | ||||
|  | ||||
|     if (!ft) { | ||||
| @@ -263,6 +296,8 @@ void on_file_recv_control(Tox *m, uint32_t friendnumber, uint32_t filenumber, To | ||||
| void on_file_recv(Tox *m, uint32_t friendnumber, uint32_t filenumber, uint32_t kind, uint64_t file_size, | ||||
|                   const uint8_t *filename, size_t filename_length, void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(userdata); | ||||
|  | ||||
|     /* We don't care about receiving avatars */ | ||||
|     if (kind != TOX_FILE_KIND_DATA) { | ||||
|         tox_file_control(m, friendnumber, filenumber, TOX_FILE_CONTROL_CANCEL, NULL); | ||||
| @@ -279,6 +314,8 @@ void on_file_recv(Tox *m, uint32_t friendnumber, uint32_t filenumber, uint32_t k | ||||
|  | ||||
| void on_friend_read_receipt(Tox *m, uint32_t friendnumber, uint32_t receipt, void *userdata) | ||||
| { | ||||
|     UNUSED_VAR(userdata); | ||||
|  | ||||
|     for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) { | ||||
|         if (windows[i] != NULL && windows[i]->onReadReceipt != NULL) { | ||||
|             windows[i]->onReadReceipt(windows[i], m, friendnumber, receipt); | ||||
| @@ -299,7 +336,7 @@ int add_window(Tox *m, ToxWindow *w) | ||||
|         } | ||||
|  | ||||
|         w->index = i; | ||||
|         w->window = newwin(LINES - 2, COLS, 0, 0); | ||||
|         w->window = newwin(LINES, COLS, 0, 0); | ||||
|  | ||||
|         if (w->window == NULL) { | ||||
|             return -1; | ||||
| @@ -360,8 +397,9 @@ void del_window(ToxWindow *w) | ||||
|     set_active_window_index(0); | ||||
|  | ||||
|     uint8_t idx = w->index; | ||||
|     delwin(w->window_bar); | ||||
|     delwin(w->window); | ||||
|     free(windows[idx]); | ||||
|     free(w); | ||||
|     windows[idx] = NULL; | ||||
|  | ||||
|     clear(); | ||||
| @@ -371,7 +409,12 @@ void del_window(ToxWindow *w) | ||||
|  | ||||
| ToxWindow *init_windows(Tox *m) | ||||
| { | ||||
|     if (COLS <= CHATBOX_HEIGHT + WINDOW_BAR_HEIGHT) { | ||||
|         exit_toxic_err("add_window() for prompt failed in init_windows", FATALERR_WININIT); | ||||
|     } | ||||
|  | ||||
|     prompt = new_prompt(); | ||||
|  | ||||
|     int n_prompt = add_window(m, prompt); | ||||
|  | ||||
|     if (n_prompt < 0) { | ||||
| @@ -393,25 +436,18 @@ void on_window_resize(void) | ||||
|     refresh(); | ||||
|     clear(); | ||||
|  | ||||
|     /* equivalent to LINES and COLS */ | ||||
|     int x2, y2; | ||||
|     getmaxyx(stdscr, y2, x2); | ||||
|     y2 -= 2; | ||||
|  | ||||
|     if (y2 <= 0 || x2 <= 0) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) { | ||||
|         if (windows[i] == NULL) { | ||||
|         ToxWindow *w = windows[i]; | ||||
|  | ||||
|         if (w == NULL) { | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         ToxWindow *w = windows[i]; | ||||
|  | ||||
|         if (windows[i]->is_friendlist)  { | ||||
|         if (w->type == WINDOW_TYPE_FRIEND_LIST)  { | ||||
|             delwin(w->window_bar); | ||||
|             delwin(w->window); | ||||
|             w->window = newwin(y2, x2, 0, 0); | ||||
|             w->window = newwin(LINES, COLS, 0, 0); | ||||
|             w->window_bar = subwin(w->window, WINDOW_BAR_HEIGHT, COLS, LINES - 2, 0); | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
| @@ -419,7 +455,7 @@ void on_window_resize(void) | ||||
|             wclear(w->help->win); | ||||
|         } | ||||
|  | ||||
|         if (w->is_groupchat) { | ||||
|         if (w->type == WINDOW_TYPE_CONFERENCE) { | ||||
|             delwin(w->chatwin->sidebar); | ||||
|             w->chatwin->sidebar = NULL; | ||||
|         } else { | ||||
| @@ -428,22 +464,35 @@ void on_window_resize(void) | ||||
|  | ||||
|         delwin(w->chatwin->linewin); | ||||
|         delwin(w->chatwin->history); | ||||
|         delwin(w->window_bar); | ||||
|         delwin(w->window); | ||||
|  | ||||
|         w->window = newwin(y2, x2, 0, 0); | ||||
|         w->chatwin->linewin = subwin(w->window, CHATBOX_HEIGHT, x2, y2 - CHATBOX_HEIGHT, 0); | ||||
|         w->window = newwin(LINES, COLS, 0, 0); | ||||
|  | ||||
|         int x2; | ||||
|         int y2; | ||||
|         getmaxyx(w->window, y2, x2); | ||||
|  | ||||
|         if (y2 <= 0 || x2 <= 0) { | ||||
|             fprintf(stderr, "Failed to resize window: max_x: %d, max_y: %d\n", x2, y2); | ||||
|             delwin(w->window); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (w->show_peerlist) { | ||||
|             w->chatwin->history = subwin(w->window, y2 - CHATBOX_HEIGHT + 1, x2 - SIDEBAR_WIDTH - 1, 0, 0); | ||||
|             w->chatwin->sidebar = subwin(w->window, y2 - CHATBOX_HEIGHT + 1, SIDEBAR_WIDTH, 0, x2 - SIDEBAR_WIDTH); | ||||
|             w->chatwin->history = subwin(w->window, y2 - CHATBOX_HEIGHT - WINDOW_BAR_HEIGHT, x2 - SIDEBAR_WIDTH - 1, 0, 0); | ||||
|             w->chatwin->sidebar = subwin(w->window, y2 - CHATBOX_HEIGHT - WINDOW_BAR_HEIGHT, SIDEBAR_WIDTH, 0, x2 - SIDEBAR_WIDTH); | ||||
|         } else { | ||||
|             w->chatwin->history = subwin(w->window, y2 - CHATBOX_HEIGHT + 1, x2, 0, 0); | ||||
|             w->chatwin->history =  subwin(w->window, y2 - CHATBOX_HEIGHT - WINDOW_BAR_HEIGHT, x2, 0, 0); | ||||
|  | ||||
|             if (!w->is_groupchat) { | ||||
|                 w->stb->topline = subwin(w->window, 2, x2, 0, 0); | ||||
|             if (w->type != WINDOW_TYPE_CONFERENCE) { | ||||
|                 w->stb->topline = subwin(w->window, TOP_BAR_HEIGHT, x2, 0, 0); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         w->window_bar = subwin(w->window, WINDOW_BAR_HEIGHT, x2, y2 - (CHATBOX_HEIGHT + WINDOW_BAR_HEIGHT), 0); | ||||
|         w->chatwin->linewin = subwin(w->window, CHATBOX_HEIGHT, x2, y2 - WINDOW_BAR_HEIGHT, 0); | ||||
|  | ||||
| #ifdef AUDIO | ||||
|  | ||||
|         if (w->chatwin->infobox.active) { | ||||
| @@ -454,82 +503,190 @@ void on_window_resize(void) | ||||
| #endif /* AUDIO */ | ||||
|  | ||||
|         scrollok(w->chatwin->history, 0); | ||||
|         wmove(w->window, y2 - CURS_Y_OFFSET, 0); | ||||
|     } | ||||
| } | ||||
|  | ||||
| static void draw_window_tab(ToxWindow *toxwin) | ||||
| static void draw_window_tab(WINDOW *win, ToxWindow *toxwin, bool active_window) | ||||
| { | ||||
|     pthread_mutex_lock(&Winthread.lock); | ||||
|  | ||||
|     if (toxwin->alert != WINDOW_ALERT_NONE) { | ||||
|         attron(COLOR_PAIR(toxwin->alert)); | ||||
|     } | ||||
|     bool has_alert = toxwin->alert != WINDOW_ALERT_NONE; | ||||
|     unsigned int pending_messages = toxwin->pending_messages; | ||||
|  | ||||
|     pthread_mutex_unlock(&Winthread.lock); | ||||
|  | ||||
|     clrtoeol(); | ||||
|     printw(" [%s]", toxwin->name); | ||||
|     WINDOW_TYPE type = toxwin->type; | ||||
|  | ||||
|     pthread_mutex_lock(&Winthread.lock); | ||||
|  | ||||
|     if (toxwin->alert != WINDOW_ALERT_NONE) { | ||||
|         attroff(COLOR_PAIR(toxwin->alert)); | ||||
|     if (active_window) { | ||||
|         wattron(win, A_BOLD | COLOR_PAIR(BAR_ACCENT)); | ||||
|         wprintw(win, " ["); | ||||
|         wattroff(win, COLOR_PAIR(BAR_ACCENT)); | ||||
|         wattron(win, COLOR_PAIR(BAR_TEXT)); | ||||
|     } else { | ||||
|         if (has_alert) { | ||||
|             wattron(win, COLOR_PAIR(BAR_ACCENT)); | ||||
|             wprintw(win, " ["); | ||||
|             wattroff(win, COLOR_PAIR(BAR_ACCENT)); | ||||
|             wattron(win, A_BOLD | COLOR_PAIR(toxwin->alert)); | ||||
|         } else { | ||||
|             wattron(win, COLOR_PAIR(BAR_ACCENT)); | ||||
|             wprintw(win, " ["); | ||||
|             wattroff(win, COLOR_PAIR(BAR_ACCENT)); | ||||
|             wattron(win, COLOR_PAIR(BAR_TEXT)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pthread_mutex_unlock(&Winthread.lock); | ||||
|     if (active_window || (type == WINDOW_TYPE_PROMPT || type == WINDOW_TYPE_FRIEND_LIST)) { | ||||
|         wprintw(win, "%s", toxwin->name); | ||||
|     } else { | ||||
|         if (pending_messages > 0) { | ||||
|             wprintw(win, "%u", pending_messages); | ||||
|         } else { | ||||
|             wprintw(win, "-"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (active_window) { | ||||
|         wattroff(win, COLOR_PAIR(BAR_TEXT)); | ||||
|         wattron(win, COLOR_PAIR(BAR_ACCENT)); | ||||
|         wprintw(win, "]"); | ||||
|         wattroff(win, A_BOLD | COLOR_PAIR(BAR_ACCENT)); | ||||
|     } else { | ||||
|         if (has_alert) { | ||||
|             wattroff(win, A_BOLD | COLOR_PAIR(toxwin->alert)); | ||||
|             wattron(win, COLOR_PAIR(BAR_ACCENT)); | ||||
|             wprintw(win, "]"); | ||||
|             wattroff(win, COLOR_PAIR(BAR_ACCENT)); | ||||
|         } else { | ||||
|             wattroff(win, COLOR_PAIR(BAR_TEXT)); | ||||
|             wattron(win, COLOR_PAIR(BAR_ACCENT)); | ||||
|             wprintw(win, "]"); | ||||
|             wattroff(win, COLOR_PAIR(BAR_ACCENT)); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| static void draw_bar(void) | ||||
| void draw_window_bar(ToxWindow *self) | ||||
| { | ||||
|     int y, x; | ||||
|     WINDOW *win = self->window_bar; | ||||
|     wclear(win); | ||||
|  | ||||
|     ToxWindow *w = windows[active_window_index]; | ||||
|  | ||||
|     if (w == NULL) { | ||||
|         return; | ||||
|     if (self->scroll_pause) { | ||||
|         wattron(win, A_BLINK | A_BOLD | COLOR_PAIR(BAR_NOTIFY)); | ||||
|         wprintw(win, "^"); | ||||
|         wattroff(win, A_BLINK | A_BOLD | COLOR_PAIR(BAR_NOTIFY)); | ||||
|     } else { | ||||
|         wattron(win, COLOR_PAIR(BAR_TEXT)); | ||||
|         wprintw(win, " "); | ||||
|         wattroff(win, COLOR_PAIR(BAR_TEXT)); | ||||
|     } | ||||
|  | ||||
|     // save current cursor position | ||||
|     getyx(w->window, y, x); | ||||
|  | ||||
|     attron(COLOR_PAIR(BLUE)); | ||||
|     mvhline(LINES - 2, 0, '_', COLS); | ||||
|     attroff(COLOR_PAIR(BLUE)); | ||||
|  | ||||
|     move(LINES - 1, 0); | ||||
|  | ||||
|     for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) { | ||||
|         if (windows[i] == NULL) { | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         if (i == active_window_index) { | ||||
|  | ||||
| #ifdef URXVT_FIX | ||||
|             attron(A_BOLD | COLOR_PAIR(GREEN)); | ||||
|         } else { | ||||
| #endif | ||||
|  | ||||
|             attron(A_BOLD); | ||||
|         bool active_window = i == active_window_index; | ||||
|         draw_window_tab(win, windows[i], active_window); | ||||
|     } | ||||
|  | ||||
|         draw_window_tab(windows[i]); | ||||
|     int cur_x; | ||||
|     int cur_y; | ||||
|     UNUSED_VAR(cur_y); | ||||
|  | ||||
|         if (i == active_window_index) { | ||||
|     getyx(win, cur_y, cur_x); | ||||
|  | ||||
| #ifdef URXVT_FIX | ||||
|             attroff(A_BOLD | COLOR_PAIR(GREEN)); | ||||
|         } else { | ||||
| #endif | ||||
|     wattron(win, COLOR_PAIR(BAR_TEXT)); | ||||
|     mvwhline(win, 0, cur_x, ' ', COLS - cur_x); | ||||
|     wattroff(win, COLOR_PAIR(BAR_TEXT)); | ||||
| } | ||||
|  | ||||
|             attroff(A_BOLD); | ||||
| /* | ||||
|  * Gets current char from stdscr and puts it in ch. | ||||
|  * | ||||
|  * Return 1 if char is printable. | ||||
|  * Return 0 if char is not printable. | ||||
|  * Return -1 on error. | ||||
|  */ | ||||
| static int get_current_char(wint_t *ch) | ||||
| { | ||||
|     wint_t tmpchar = 0; | ||||
|     bool is_printable = false; | ||||
|  | ||||
| #ifdef HAVE_WIDECHAR | ||||
|     int status = wget_wch(stdscr, &tmpchar); | ||||
|  | ||||
|     if (status == ERR) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     if (status == OK) { | ||||
|         is_printable = iswprint(tmpchar); | ||||
|     } | ||||
|  | ||||
| #else | ||||
|     tmpchar = getch(); | ||||
|  | ||||
|     if (tmpchar == ERR) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     is_printable = isprint(tmpchar); | ||||
| #endif /* HAVE_WIDECHAR */ | ||||
|  | ||||
|     *ch = tmpchar; | ||||
|  | ||||
|     return (int) is_printable; | ||||
| } | ||||
|  | ||||
| static struct key_sequence_codes { | ||||
|     wchar_t *code; | ||||
|     wint_t   key; | ||||
| } Keys[] = { | ||||
|     { L"[1;5A", T_KEY_C_UP    }, | ||||
|     { L"[1;5B", T_KEY_C_DOWN  }, | ||||
|     { L"[1;5C", T_KEY_C_RIGHT }, | ||||
|     { L"[1;5D", T_KEY_C_LEFT  }, | ||||
|     { NULL, 0 } | ||||
| }; | ||||
|  | ||||
| /* | ||||
|  * Return key code corresponding to character sequence queued in stdscr. | ||||
|  * Return -1 if sequence is unknown. | ||||
|  */ | ||||
| #define MAX_SEQUENCE_SIZE 5 | ||||
| static wint_t get_input_sequence_code(void) | ||||
| { | ||||
|     wchar_t code[MAX_SEQUENCE_SIZE + 1]; | ||||
|  | ||||
|     size_t length = 0; | ||||
|     wint_t ch = 0; | ||||
|  | ||||
|     for (size_t i = 0; i < MAX_SEQUENCE_SIZE; ++i) { | ||||
|         int res = get_current_char(&ch); | ||||
|  | ||||
|         if (res < 0) { | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         ++length; | ||||
|         code[i] = (wchar_t) ch; | ||||
|     } | ||||
|  | ||||
|     if (length == 0) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     code[length] = L'\0'; | ||||
|  | ||||
|     for (size_t i = 0; Keys[i].key != 0; ++i) { | ||||
|         if (wcscmp(code, Keys[i].code) == 0) { | ||||
|             return Keys[i].key; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // restore cursor position after drawing | ||||
|     move(y, x); | ||||
|  | ||||
|     refresh(); | ||||
|     return -1; | ||||
| } | ||||
|  | ||||
| void draw_active_window(Tox *m) | ||||
| @@ -542,53 +699,48 @@ void draw_active_window(Tox *m) | ||||
|  | ||||
|     pthread_mutex_lock(&Winthread.lock); | ||||
|     a->alert = WINDOW_ALERT_NONE; | ||||
|     a->pending_messages = 0; | ||||
|     pthread_mutex_unlock(&Winthread.lock); | ||||
|  | ||||
|     wint_t ch = 0; | ||||
|  | ||||
|     draw_bar(); | ||||
|  | ||||
|     touchwin(a->window); | ||||
|     a->onDraw(a, m); | ||||
|     wrefresh(a->window); | ||||
|  | ||||
|     /* Handle input */ | ||||
|     bool ltr; | ||||
| #ifdef HAVE_WIDECHAR | ||||
|     int status = wget_wch(stdscr, &ch); | ||||
|     wint_t ch = 0; | ||||
|     int printable = get_current_char(&ch); | ||||
|  | ||||
|     if (status == ERR) { | ||||
|     if (printable < 0) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (status == OK) { | ||||
|         ltr = iswprint(ch); | ||||
|     } else { /* if (status == KEY_CODE_YES) */ | ||||
|         ltr = false; | ||||
|     } | ||||
|  | ||||
| #else | ||||
|     ch = getch(); | ||||
|  | ||||
|     if (ch == ERR) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     /* TODO verify if this works */ | ||||
|     ltr = isprint(ch); | ||||
| #endif /* HAVE_WIDECHAR */ | ||||
|  | ||||
|     if (!ltr && (ch == user_settings->key_next_tab || ch == user_settings->key_prev_tab)) { | ||||
|     if (printable == 0 && (ch == user_settings->key_next_tab || ch == user_settings->key_prev_tab)) { | ||||
|         set_next_window((int) ch); | ||||
|     } else { | ||||
|         return; | ||||
|     } else if ((printable == 0) && (a->type != WINDOW_TYPE_FRIEND_LIST)) { | ||||
|         pthread_mutex_lock(&Winthread.lock); | ||||
|         a->onKey(a, m, ch, ltr); | ||||
|         bool input_ret = a->onKey(a, m, ch, (bool) printable); | ||||
|         pthread_mutex_unlock(&Winthread.lock); | ||||
|  | ||||
|         if (input_ret) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // if an unprintable key code is unrecognized by input handler we attempt to manually decode char sequence | ||||
|         wint_t tmp = get_input_sequence_code(); | ||||
|  | ||||
|         if (tmp != (wint_t) -1) { | ||||
|             ch = tmp; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pthread_mutex_lock(&Winthread.lock); | ||||
|     a->onKey(a, m, ch, (bool) printable); | ||||
|     pthread_mutex_unlock(&Winthread.lock); | ||||
| } | ||||
|  | ||||
| /* refresh inactive windows to prevent scrolling bugs. | ||||
|    call at least once per second */ | ||||
| /* Refresh inactive windows to prevent scrolling bugs. | ||||
|  * Call at least once per second. | ||||
|  */ | ||||
| void refresh_inactive_windows(void) | ||||
| { | ||||
|     for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) { | ||||
| @@ -598,7 +750,7 @@ void refresh_inactive_windows(void) | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         if (i != active_window_index && !toxwin->is_friendlist) { | ||||
|         if ((i != active_window_index) && (toxwin->type != WINDOW_TYPE_FRIEND_LIST)) { | ||||
|             pthread_mutex_lock(&Winthread.lock); | ||||
|             line_info_print(toxwin); | ||||
|             pthread_mutex_unlock(&Winthread.lock); | ||||
| @@ -636,7 +788,7 @@ int get_num_active_windows(void) | ||||
|     return num_active_windows; | ||||
| } | ||||
|  | ||||
| /* destroys all chat and groupchat windows (should only be called on shutdown) */ | ||||
| /* destroys all chat and conference windows (should only be called on shutdown) */ | ||||
| void kill_all_windows(Tox *m) | ||||
| { | ||||
|     for (uint8_t i = 2; i < MAX_WINDOWS_NUM; ++i) { | ||||
| @@ -644,10 +796,10 @@ void kill_all_windows(Tox *m) | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         if (windows[i]->is_chat) { | ||||
|         if (windows[i]->type == WINDOW_TYPE_CHAT) { | ||||
|             kill_chat_window(windows[i], m); | ||||
|         } else if (windows[i]->is_groupchat) { | ||||
|             free_groupchat(windows[i], m, windows[i]->num); | ||||
|         } else if (windows[i]->type == WINDOW_TYPE_CONFERENCE) { | ||||
|             free_conference(windows[i], windows[i]->num); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -24,9 +24,9 @@ | ||||
| #define WINDOWS_H | ||||
|  | ||||
| #include <pthread.h> | ||||
| #include <wctype.h> | ||||
| #include <wchar.h> | ||||
| #include <signal.h> | ||||
| #include <wchar.h> | ||||
| #include <wctype.h> | ||||
|  | ||||
| #include <tox/tox.h> | ||||
|  | ||||
| @@ -36,12 +36,14 @@ | ||||
|  | ||||
| #include "toxic.h" | ||||
|  | ||||
| #define MAX_WINDOWS_NUM 16 | ||||
| #define MAX_WINDOWS_NUM 20 | ||||
| #define MAX_WINDOW_NAME_LENGTH 22 | ||||
| #define CURS_Y_OFFSET 1    /* y-axis cursor offset for chat contexts */ | ||||
| #define CHATBOX_HEIGHT 2 | ||||
| #define CHATBOX_HEIGHT 1 | ||||
| #define TOP_BAR_HEIGHT 1 | ||||
| #define WINDOW_BAR_HEIGHT 1 | ||||
|  | ||||
| /* Curses foreground colours (background is black) */ | ||||
| /* ncurses colour pairs as FOREGROUND_BACKGROUND. No background defaults to black. */ | ||||
| typedef enum { | ||||
|     WHITE, | ||||
|     GREEN, | ||||
| @@ -51,16 +53,33 @@ typedef enum { | ||||
|     YELLOW, | ||||
|     MAGENTA, | ||||
|     BLACK, | ||||
|     BLUE_BLACK, | ||||
|     BLACK_WHITE, | ||||
|     BAR_TEXT, | ||||
|     STATUS_ONLINE, | ||||
|     BAR_ACCENT, | ||||
|     PURPLE_BG, | ||||
|     BLACK_BG, | ||||
|     STATUS_BUSY, | ||||
|     STATUS_AWAY, | ||||
|     BAR_NOTIFY, | ||||
| } C_COLOURS; | ||||
|  | ||||
| /* tab alert types: lower types take priority (this relies on the order of C_COLOURS) */ | ||||
| typedef enum { | ||||
|     WINDOW_ALERT_NONE = 0, | ||||
|     WINDOW_ALERT_0 = GREEN, | ||||
|     WINDOW_ALERT_1 = RED, | ||||
|     WINDOW_ALERT_2 = MAGENTA, | ||||
|     WINDOW_ALERT_0 = STATUS_ONLINE, | ||||
|     WINDOW_ALERT_1 = BAR_ACCENT, | ||||
|     WINDOW_ALERT_2 = PURPLE_BG, | ||||
| } WINDOW_ALERTS; | ||||
|  | ||||
| typedef enum { | ||||
|     WINDOW_TYPE_PROMPT, | ||||
|     WINDOW_TYPE_CHAT, | ||||
|     WINDOW_TYPE_CONFERENCE, | ||||
|     WINDOW_TYPE_FRIEND_LIST, | ||||
| } WINDOW_TYPE; | ||||
|  | ||||
| /* Fixes text color problem on some terminals. | ||||
|    Uncomment if necessary */ | ||||
| /* #define URXVT_FIX */ | ||||
| @@ -83,6 +102,7 @@ struct av_thread { | ||||
| struct arg_opts { | ||||
|     bool use_ipv4; | ||||
|     bool force_tcp; | ||||
|     bool disable_local_discovery; | ||||
|     bool debug; | ||||
|     bool default_locale; | ||||
|     bool use_custom_data; | ||||
| @@ -94,6 +114,9 @@ struct arg_opts { | ||||
|     char config_path[MAX_STR_SIZE]; | ||||
|     char nodes_path[MAX_STR_SIZE]; | ||||
|  | ||||
|     bool logging; | ||||
|     FILE *log_fp; | ||||
|  | ||||
|     char proxy_address[256]; | ||||
|     uint8_t proxy_type; | ||||
|     uint16_t proxy_port; | ||||
| @@ -109,7 +132,7 @@ typedef struct Help Help; | ||||
|  | ||||
| struct ToxWindow { | ||||
|     /* ncurses */ | ||||
|     void(*onKey)(ToxWindow *, Tox *, wint_t, bool); | ||||
|     bool(*onKey)(ToxWindow *, Tox *, wint_t, bool); | ||||
|     void(*onDraw)(ToxWindow *, Tox *); | ||||
|     void(*onInit)(ToxWindow *, Tox *); | ||||
|  | ||||
| @@ -121,11 +144,11 @@ struct ToxWindow { | ||||
|     void(*onNickChange)(ToxWindow *, Tox *, uint32_t, const char *, size_t); | ||||
|     void(*onStatusChange)(ToxWindow *, Tox *, uint32_t, Tox_User_Status); | ||||
|     void(*onStatusMessageChange)(ToxWindow *, uint32_t, const char *, size_t); | ||||
|     void(*onGroupMessage)(ToxWindow *, Tox *, uint32_t, uint32_t, Tox_Message_Type, const char *, size_t); | ||||
|     void(*onGroupInvite)(ToxWindow *, Tox *, int32_t, uint8_t, const char *, uint16_t); | ||||
|     void(*onGroupNameListChange)(ToxWindow *, Tox *, uint32_t); | ||||
|     void(*onGroupPeerNameChange)(ToxWindow *, Tox *, uint32_t, uint32_t, const char *, size_t); | ||||
|     void(*onGroupTitleChange)(ToxWindow *, Tox *, uint32_t, uint32_t, const char *, size_t); | ||||
|     void(*onConferenceMessage)(ToxWindow *, Tox *, uint32_t, uint32_t, Tox_Message_Type, const char *, size_t); | ||||
|     void(*onConferenceInvite)(ToxWindow *, Tox *, int32_t, uint8_t, const char *, uint16_t); | ||||
|     void(*onConferenceNameListChange)(ToxWindow *, Tox *, uint32_t); | ||||
|     void(*onConferencePeerNameChange)(ToxWindow *, Tox *, uint32_t, uint32_t, const char *, size_t); | ||||
|     void(*onConferenceTitleChange)(ToxWindow *, Tox *, uint32_t, uint32_t, const char *, size_t); | ||||
|     void(*onFileChunkRequest)(ToxWindow *, Tox *, uint32_t, uint32_t, uint64_t, size_t); | ||||
|     void(*onFileRecvChunk)(ToxWindow *, Tox *, uint32_t, uint32_t, uint64_t, const char *, size_t); | ||||
|     void(*onFileControl)(ToxWindow *, Tox *, uint32_t, uint32_t, Tox_File_Control); | ||||
| @@ -146,16 +169,9 @@ struct ToxWindow { | ||||
|     void(*onEnd)(ToxWindow *, ToxAV *, uint32_t, int); | ||||
|     void(*onWriteDevice)(ToxWindow *, Tox *, uint32_t, int, const int16_t *, unsigned int, uint8_t, unsigned int); | ||||
|  | ||||
|     int device_selection[2]; /* -1 if not set, if set uses these selections instead of primary device */ | ||||
|     bool is_call; | ||||
|     int ringing_sound; | ||||
|  | ||||
| #ifdef VIDEO | ||||
|  | ||||
|     int video_device_selection[2]; /* -1 if not set, if set uses these selections instead of primary video device */ | ||||
|  | ||||
| #endif /* VIDEO */ | ||||
|  | ||||
| #endif /* AUDIO */ | ||||
|  | ||||
|     int active_box; /* For box notify */ | ||||
| @@ -163,13 +179,13 @@ struct ToxWindow { | ||||
|     char name[TOXIC_MAX_NAME_LENGTH + 1]; | ||||
|     uint32_t num;    /* corresponds to friendnumber in chat windows */ | ||||
|     uint8_t index; /* This window's index in the windows array */ | ||||
|     bool scroll_pause; /* true if this window is not scrolled to the bottom */ | ||||
|     unsigned int pending_messages;  /* # of new messages in this window since the last time it was focused */ | ||||
|     int x; | ||||
|  | ||||
|     bool is_chat; | ||||
|     bool is_prompt; | ||||
|     bool is_friendlist; | ||||
|     bool is_groupchat; | ||||
|     int show_peerlist;    /* used to toggle groupchat peerlist */ | ||||
|     WINDOW_TYPE type; | ||||
|  | ||||
|     int show_peerlist;    /* used to toggle conference peerlist */ | ||||
|  | ||||
|     WINDOW_ALERTS alert; | ||||
|  | ||||
| @@ -178,6 +194,7 @@ struct ToxWindow { | ||||
|     Help *help; | ||||
|  | ||||
|     WINDOW *window; | ||||
|     WINDOW *window_bar; | ||||
| }; | ||||
|  | ||||
| /* statusbar info holder */ | ||||
| @@ -214,7 +231,7 @@ struct infobox { | ||||
|  | ||||
| #define MAX_LINE_HIST 128 | ||||
|  | ||||
| /* chat and groupchat window/buffer holder */ | ||||
| /* chat and conference window/buffer holder */ | ||||
| struct ChatContext { | ||||
|     wchar_t line[MAX_STR_SIZE]; | ||||
|     int pos; | ||||
| @@ -261,6 +278,7 @@ void on_window_resize(void); | ||||
| void force_refresh(WINDOW *w); | ||||
| ToxWindow *get_window_ptr(size_t i); | ||||
| ToxWindow *get_active_window(void); | ||||
| void draw_window_bar(ToxWindow *self); | ||||
|  | ||||
| /* refresh inactive windows to prevent scrolling bugs. | ||||
|    call at least once per second */ | ||||
|   | ||||
							
								
								
									
										87
									
								
								src/x11focus.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								src/x11focus.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | ||||
| /*  x11focus.c | ||||
|  * | ||||
|  * | ||||
|  *  Copyright (C) 2020 Toxic All Rights Reserved. | ||||
|  * | ||||
|  *  This file is part of Toxic. | ||||
|  * | ||||
|  *  Toxic is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|  *  the Free Software Foundation, either version 3 of the License, or | ||||
|  *  (at your option) any later version. | ||||
|  * | ||||
|  *  Toxic is distributed in the hope that it will be useful, | ||||
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  *  GNU General Public License for more details. | ||||
|  * | ||||
|  *  You should have received a copy of the GNU General Public License | ||||
|  *  along with Toxic.  If not, see <http://www.gnu.org/licenses/>. | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include "x11focus.h" | ||||
|  | ||||
| #ifndef __APPLE__ | ||||
|  | ||||
| #include <X11/Xlib.h> | ||||
|  | ||||
| static struct Focus { | ||||
|     Display *display; | ||||
|     Window terminal_window; | ||||
| } Focus; | ||||
|  | ||||
| static long unsigned int focused_window_id(void) | ||||
| { | ||||
|     if (!Focus.display) { | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     Window focus; | ||||
|     int revert; | ||||
|  | ||||
|     XLockDisplay(Focus.display); | ||||
|     XGetInputFocus(Focus.display, &focus, &revert); | ||||
|     XUnlockDisplay(Focus.display); | ||||
|  | ||||
|     return focus; | ||||
| } | ||||
|  | ||||
| bool is_focused(void) | ||||
| { | ||||
|     if (!Focus.display) { | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     return Focus.terminal_window == focused_window_id(); | ||||
| } | ||||
|  | ||||
| int init_x11focus(void) | ||||
| { | ||||
|     if (XInitThreads() == 0) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     Focus.display = XOpenDisplay(NULL); | ||||
|  | ||||
|     if (!Focus.display) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     Focus.terminal_window = focused_window_id(); | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| void terminate_x11focus(void) | ||||
| { | ||||
|     if (!Focus.display || !Focus.terminal_window) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     XLockDisplay(Focus.display); | ||||
|     XCloseDisplay(Focus.display); | ||||
|     XUnlockDisplay(Focus.display); | ||||
| } | ||||
|  | ||||
| #endif /* !__APPLE__ */ | ||||
| @@ -1,7 +1,7 @@ | ||||
| /*  group_commands.h
 | ||||
| /*  xtra.h
 | ||||
|  * | ||||
|  * | ||||
|  *  Copyright (C) 2014 Toxic All Rights Reserved. | ||||
|  *  Copyright (C) 2020 Toxic All Rights Reserved. | ||||
|  * | ||||
|  *  This file is part of Toxic. | ||||
|  * | ||||
| @@ -20,12 +20,15 @@ | ||||
|  * | ||||
|  */ | ||||
| 
 | ||||
| #ifndef GROUP_COMMANDS_H | ||||
| #define GROUP_COMMANDS_H | ||||
| #ifndef X11FOCUS_H | ||||
| #define X11FOCUS_H | ||||
| 
 | ||||
| #include "windows.h" | ||||
| #include "toxic.h" | ||||
| #include <stdbool.h> | ||||
| 
 | ||||
| void cmd_set_title(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE]); | ||||
| /* NOTE: If no xlib present don't compile */ | ||||
| 
 | ||||
| #endif /* GROUP_COMMANDS_H */ | ||||
| int               init_x11focus(void); | ||||
| void              terminate_x11focus(void); | ||||
| bool              is_focused(void); | ||||
| 
 | ||||
| #endif /* X11FOCUS */ | ||||
							
								
								
									
										418
									
								
								src/xtra.c
									
									
									
									
									
								
							
							
						
						
									
										418
									
								
								src/xtra.c
									
									
									
									
									
								
							| @@ -1,418 +0,0 @@ | ||||
| /*  xtra.c | ||||
|  * | ||||
|  * | ||||
|  *  Copyright (C) 2014 Toxic All Rights Reserved. | ||||
|  * | ||||
|  *  This file is part of Toxic. | ||||
|  * | ||||
|  *  Toxic is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|  *  the Free Software Foundation, either version 3 of the License, or | ||||
|  *  (at your option) any later version. | ||||
|  * | ||||
|  *  Toxic is distributed in the hope that it will be useful, | ||||
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  *  GNU General Public License for more details. | ||||
|  * | ||||
|  *  You should have received a copy of the GNU General Public License | ||||
|  *  along with Toxic.  If not, see <http://www.gnu.org/licenses/>. | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #include "xtra.h" | ||||
|  | ||||
| #include <X11/Xlib.h> | ||||
| #include <X11/Xatom.h> | ||||
|  | ||||
| #include <pthread.h> | ||||
| #include <assert.h> | ||||
| #include <string.h> | ||||
| #include <unistd.h> | ||||
| #include <stdio.h> | ||||
|  | ||||
|  | ||||
| const  Atom XtraTerminate = 1; | ||||
| const  Atom XtraNil = 0; | ||||
|  | ||||
| static Atom XdndAware; | ||||
| static Atom XdndEnter; | ||||
| static Atom XdndLeave; | ||||
| static Atom XdndPosition; | ||||
| static Atom XdndStatus; | ||||
| static Atom XdndDrop; | ||||
| static Atom XdndSelection; | ||||
| static Atom XdndDATA; | ||||
| static Atom XdndTypeList; | ||||
| static Atom XdndActionCopy; | ||||
| static Atom XdndFinished; | ||||
|  | ||||
| struct Xtra { | ||||
|     drop_callback on_drop; | ||||
|     Display *display; | ||||
|     Window terminal_window; | ||||
|     Window proxy_window; | ||||
|     Window source_window; /* When we have a drop */ | ||||
|     Atom handling_version; | ||||
|     Atom expecting_type; | ||||
| } Xtra; | ||||
|  | ||||
| typedef struct Property { | ||||
|     unsigned char *data; | ||||
|     int            read_format; | ||||
|     unsigned long  read_num; | ||||
|     Atom           read_type; | ||||
| } Property; | ||||
|  | ||||
| static Property read_property(Window s, Atom p) | ||||
| { | ||||
|     Atom read_type; | ||||
|     int  read_format; | ||||
|     unsigned long read_num; | ||||
|     unsigned long left_bytes; | ||||
|     unsigned char *data = NULL; | ||||
|  | ||||
|     int read_bytes = 1024; | ||||
|  | ||||
|     /* Keep trying to read the property until there are no bytes unread */ | ||||
|     do { | ||||
|         if (data) { | ||||
|             XFree(data); | ||||
|         } | ||||
|  | ||||
|         XGetWindowProperty(Xtra.display, s, | ||||
|                            p, 0, | ||||
|                            read_bytes, | ||||
|                            False, AnyPropertyType, | ||||
|                            &read_type, &read_format, | ||||
|                            &read_num, &left_bytes, | ||||
|                            &data); | ||||
|  | ||||
|         read_bytes *= 2; | ||||
|     } while (left_bytes != 0); | ||||
|  | ||||
|     Property property = {data, read_format, read_num, read_type}; | ||||
|     return property; | ||||
| } | ||||
|  | ||||
| static Atom get_dnd_type(long *a, int l) | ||||
| { | ||||
|     int i = 0; | ||||
|  | ||||
|     for (; i < l; i ++) { | ||||
|         if (a[i] != XtraNil) { | ||||
|             return a[i];    /* Get first valid */ | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return XtraNil; | ||||
| } | ||||
|  | ||||
| /* TODO maybe support only certain types in the future */ | ||||
| static void handle_xdnd_enter(XClientMessageEvent *e) | ||||
| { | ||||
|     Xtra.handling_version = (e->data.l[1] >> 24); | ||||
|  | ||||
|     if ((e->data.l[1] & 1)) { | ||||
|         // Fetch the list of possible conversions | ||||
|         Property p = read_property(e->data.l[0], XdndTypeList); | ||||
|         Xtra.expecting_type = get_dnd_type((long *)p.data, p.read_num); | ||||
|         XFree(p.data); | ||||
|     } else { | ||||
|         // Use the available list | ||||
|         Xtra.expecting_type = get_dnd_type(e->data.l + 2, 3); | ||||
|     } | ||||
| } | ||||
|  | ||||
| static void handle_xdnd_position(XClientMessageEvent *e) | ||||
| { | ||||
|     XEvent ev = { | ||||
|         .xclient = { | ||||
|             .type = ClientMessage, | ||||
|             .display = e->display, | ||||
|             .window = e->data.l[0], | ||||
|             .message_type = XdndStatus, | ||||
|             .format = 32, | ||||
|             .data = { | ||||
|                 .l = { | ||||
|                     Xtra.proxy_window, | ||||
|                     (Xtra.expecting_type != XtraNil), | ||||
|                     0, 0, | ||||
|                     XdndActionCopy | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     XSendEvent(Xtra.display, e->data.l[0], False, NoEventMask, &ev); | ||||
|     XFlush(Xtra.display); | ||||
| } | ||||
|  | ||||
| static void handle_xdnd_drop(XClientMessageEvent *e) | ||||
| { | ||||
|     /* Not expecting any type */ | ||||
|     if (Xtra.expecting_type == XtraNil) { | ||||
|         XEvent ev = { | ||||
|             .xclient = { | ||||
|                 .type = ClientMessage, | ||||
|                 .display = e->display, | ||||
|                 .window = e->data.l[0], | ||||
|                 .message_type = XdndFinished, | ||||
|                 .format = 32, | ||||
|                 .data = { | ||||
|                     .l = {Xtra.proxy_window, 0, 0} | ||||
|                 } | ||||
|             } | ||||
|         }; | ||||
|  | ||||
|         XSendEvent(Xtra.display, e->data.l[0], False, NoEventMask, &ev); | ||||
|     } else { | ||||
|         Xtra.source_window = e->data.l[0]; | ||||
|         XConvertSelection(Xtra.display, | ||||
|                           XdndSelection, | ||||
|                           Xtra.expecting_type, | ||||
|                           XdndSelection, | ||||
|                           Xtra.proxy_window, | ||||
|                           Xtra.handling_version >= 1 ? e->data.l[2] : CurrentTime); | ||||
|     } | ||||
| } | ||||
|  | ||||
| static void handle_xdnd_selection(XSelectionEvent *e) | ||||
| { | ||||
|     /* DnD succesfully finished, send finished and call callback */ | ||||
|     XEvent ev = { | ||||
|         .xclient = { | ||||
|             .type = ClientMessage, | ||||
|             .display = Xtra.display, | ||||
|             .window = Xtra.source_window, | ||||
|             .message_type = XdndFinished, | ||||
|             .format = 32, | ||||
|             .data = { | ||||
|                 .l = {Xtra.proxy_window, 1, XdndActionCopy} | ||||
|             } | ||||
|         } | ||||
|     }; | ||||
|     XSendEvent(Xtra.display, Xtra.source_window, False, NoEventMask, &ev); | ||||
|  | ||||
|     Property p = read_property(Xtra.proxy_window, XdndSelection); | ||||
|     DropType dt; | ||||
|  | ||||
|     if (strcmp(XGetAtomName(Xtra.display, p.read_type), "text/uri-list") == 0) { | ||||
|         dt = DT_file_list; | ||||
|     } else { /* text/uri-list */ | ||||
|         dt = DT_plain; | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /* Call callback for every entry */ | ||||
|     if (Xtra.on_drop && p.read_num) { | ||||
|         char *sptr; | ||||
|         char *str = strtok_r((char *) p.data, "\n\r", &sptr); | ||||
|  | ||||
|         if (str) { | ||||
|             Xtra.on_drop(str, dt); | ||||
|         } | ||||
|  | ||||
|         while ((str = strtok_r(NULL, "\n\r", &sptr))) { | ||||
|             Xtra.on_drop(str, dt); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (p.data) { | ||||
|         XFree(p.data); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void *event_loop(void *p) | ||||
| { | ||||
|     /* Handle events like a real nigga */ | ||||
|  | ||||
|     (void) p; /* DINDUNOTHIN */ | ||||
|  | ||||
|     XEvent event; | ||||
|     int pending; | ||||
|  | ||||
|     while (Xtra.display) { | ||||
|         /* NEEDMOEVENTSFODEMPROGRAMS */ | ||||
|  | ||||
|         XLockDisplay(Xtra.display); | ||||
|  | ||||
|         if ((pending = XPending(Xtra.display))) { | ||||
|             XNextEvent(Xtra.display, &event); | ||||
|         } | ||||
|  | ||||
|         if (!pending) { | ||||
|             XUnlockDisplay(Xtra.display); | ||||
|             usleep(10000); | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         if (event.type == ClientMessage) { | ||||
|             Atom type = event.xclient.message_type; | ||||
|  | ||||
|             if (type == XdndEnter) { | ||||
|                 handle_xdnd_enter(&event.xclient); | ||||
|             } else if (type == XdndPosition) { | ||||
|                 handle_xdnd_position(&event.xclient); | ||||
|             } else if (type == XdndDrop) { | ||||
|                 handle_xdnd_drop(&event.xclient); | ||||
|             } else if (type == XtraTerminate) { | ||||
|                 break; | ||||
|             } | ||||
|         } else if (event.type == SelectionNotify) { | ||||
|             handle_xdnd_selection(&event.xselection); | ||||
|         } | ||||
|         /* AINNOBODYCANHANDLEDEMEVENTS*/ | ||||
|         else { | ||||
|             XSendEvent(Xtra.display, Xtra.terminal_window, 0, 0, &event); | ||||
|         } | ||||
|  | ||||
|         XUnlockDisplay(Xtra.display); | ||||
|     } | ||||
|  | ||||
|     /* Actual XTRA termination | ||||
|      * Please call xtra_terminate() at exit | ||||
|      * otherwise HEWUSAGUDBOI happens | ||||
|      */ | ||||
|     if (Xtra.display) { | ||||
|         XCloseDisplay(Xtra.display); | ||||
|     } | ||||
|  | ||||
|     return (Xtra.display = NULL); | ||||
| } | ||||
|  | ||||
| static long unsigned int focused_window_id(void) | ||||
| { | ||||
|     if (!Xtra.display) { | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     Window focus; | ||||
|     int revert; | ||||
|     XLockDisplay(Xtra.display); | ||||
|     XGetInputFocus(Xtra.display, &focus, &revert); | ||||
|     XUnlockDisplay(Xtra.display); | ||||
|     return focus; | ||||
| } | ||||
|  | ||||
| int is_focused(void) | ||||
| { | ||||
|     return Xtra.proxy_window == focused_window_id() || Xtra.terminal_window == focused_window_id(); | ||||
| } | ||||
|  | ||||
| int init_xtra(drop_callback d) | ||||
| { | ||||
|     if (!d) { | ||||
|         return -1; | ||||
|     } else { | ||||
|         Xtra.on_drop = d; | ||||
|     } | ||||
|  | ||||
|     XInitThreads(); | ||||
|  | ||||
|     if (!(Xtra.display = XOpenDisplay(NULL))) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     Xtra.terminal_window = focused_window_id(); | ||||
|  | ||||
|     /* OSX: if focused window is 0, it means toxic is ran from | ||||
|      * native terminal and not X11 terminal window, silently exit */ | ||||
|     if (!Xtra.terminal_window) { | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     { | ||||
|         /* Create an invisible window which will act as proxy for the DnD operation. */ | ||||
|         XSetWindowAttributes attr  = {0}; | ||||
|         attr.event_mask            = EnterWindowMask | | ||||
|                                      LeaveWindowMask | | ||||
|                                      ButtonMotionMask | | ||||
|                                      ButtonPressMask | | ||||
|                                      ButtonReleaseMask | | ||||
|                                      ResizeRedirectMask; | ||||
|  | ||||
|         attr.do_not_propagate_mask = NoEventMask; | ||||
|  | ||||
|         Window root; | ||||
|         int x, y; | ||||
|         unsigned int wht, hht, b, d; | ||||
|  | ||||
|         /* Since we cannot capture resize events for parent window we will have to create | ||||
|          * this window to have maximum size as defined in root window | ||||
|          */ | ||||
|         XGetGeometry(Xtra.display, | ||||
|                      XDefaultRootWindow(Xtra.display), | ||||
|                      &root, &x, &y, &wht, &hht, &b, &d); | ||||
|  | ||||
|         if (!(Xtra.proxy_window = XCreateWindow | ||||
|                                   (Xtra.display, Xtra.terminal_window,       /* Parent */ | ||||
|                                    0, 0,                                     /* Position */ | ||||
|                                    wht, hht,                                 /* Width + height */ | ||||
|                                    0,                                        /* Border width */ | ||||
|                                    CopyFromParent,                           /* Depth */ | ||||
|                                    InputOnly,                                /* Class */ | ||||
|                                    CopyFromParent,                           /* Visual */ | ||||
|                                    CWEventMask | CWCursor,                   /* Value mask */ | ||||
|                                    &attr))) {                                /* Attributes for value mask */ | ||||
|             return -1; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     XMapWindow(Xtra.display, Xtra.proxy_window);   /* Show window (sandwich) */ | ||||
|     XLowerWindow(Xtra.display, Xtra.proxy_window); /* Don't interfere with parent lmao */ | ||||
|  | ||||
|     XdndAware = XInternAtom(Xtra.display, "XdndAware", False); | ||||
|     XdndEnter = XInternAtom(Xtra.display, "XdndEnter", False); | ||||
|     XdndLeave = XInternAtom(Xtra.display, "XdndLeave", False); | ||||
|     XdndPosition = XInternAtom(Xtra.display, "XdndPosition", False); | ||||
|     XdndStatus = XInternAtom(Xtra.display, "XdndStatus", False); | ||||
|     XdndDrop = XInternAtom(Xtra.display, "XdndDrop", False); | ||||
|     XdndSelection = XInternAtom(Xtra.display, "XdndSelection", False); | ||||
|     XdndDATA = XInternAtom(Xtra.display, "XdndDATA", False); | ||||
|     XdndTypeList = XInternAtom(Xtra.display, "XdndTypeList", False); | ||||
|     XdndActionCopy = XInternAtom(Xtra.display, "XdndActionCopy", False); | ||||
|     XdndFinished = XInternAtom(Xtra.display, "XdndFinished", False); | ||||
|  | ||||
|     /* Inform my nigga windows that we are aware of dnd */ | ||||
|     Atom XdndVersion = 3; | ||||
|     XChangeProperty(Xtra.display, | ||||
|                     Xtra.proxy_window, | ||||
|                     XdndAware, | ||||
|                     XA_ATOM, | ||||
|                     32, | ||||
|                     PropModeReplace, | ||||
|                     (unsigned char *)&XdndVersion, 1); | ||||
|  | ||||
|     pthread_t id; | ||||
|  | ||||
|     if (pthread_create(&id, NULL, event_loop, NULL) != 0) { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     pthread_detach(id); | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| void terminate_xtra(void) | ||||
| { | ||||
|     if (!Xtra.display || !Xtra.terminal_window) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     XEvent terminate = { | ||||
|         .xclient = { | ||||
|             .type = ClientMessage, | ||||
|             .display = Xtra.display, | ||||
|             .message_type = XtraTerminate, | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     XLockDisplay(Xtra.display); | ||||
|     XDeleteProperty(Xtra.display, Xtra.proxy_window, XdndAware); | ||||
|     XSendEvent(Xtra.display, Xtra.proxy_window, 0, NoEventMask, &terminate); | ||||
|     XUnlockDisplay(Xtra.display); | ||||
|  | ||||
|     while (Xtra.display); /* Wait for termination */ | ||||
| } | ||||
		Reference in New Issue
	
	Block a user