mirror of
https://github.com/Tha14/toxic.git
synced 2025-06-28 22:46:45 +02:00
Compare commits
54 Commits
Author | SHA1 | Date | |
---|---|---|---|
b1d8ab102f | |||
c387df35f8 | |||
351a50c214 | |||
93175314b5 | |||
b905a1a3c5 | |||
c4386b195f | |||
ed1e617380 | |||
1382adb1f6 | |||
ecf1c317b7 | |||
cf0b99f1e5 | |||
3605a296a9 | |||
9375d220f9 | |||
8f94b0a218 | |||
85a0becbf9 | |||
fec36ad9e6 | |||
ecdf6f01d2 | |||
e1bfa30769 | |||
ebcbc7497b | |||
e844ece28b | |||
8508451ba6 | |||
5cc83a7cb5 | |||
febc725763 | |||
f2c116feb3 | |||
52dd60dc86 | |||
80c0500299 | |||
ab490d28b4 | |||
a9f7f85617 | |||
1bfc1ba371 | |||
2ede39369a | |||
922c184195 | |||
56a9571509 | |||
0136f22076 | |||
c4ace288af | |||
6d3fbfee59 | |||
be5e7906da | |||
369f26932e | |||
22ea522baf | |||
4f60d546e6 | |||
76d1eafdc0 | |||
37912f2d88 | |||
09710327c5 | |||
acee4615f8 | |||
5ed26eda9b | |||
6d2b90ac9f | |||
02ea0fac44 | |||
7d3d129624 | |||
b3ed8bc35c | |||
90210daca7 | |||
0e13a1f1bc | |||
09e2690211 | |||
e65e3af274 | |||
c6c60d018e | |||
451d4ced80 | |||
7a7402ff86 |
1
.gitignore
vendored
1
.gitignore
vendored
@ -16,3 +16,4 @@ stamp-h1
|
|||||||
build/toxic
|
build/toxic
|
||||||
build/*.o
|
build/*.o
|
||||||
build/*.d
|
build/*.d
|
||||||
|
apidoc/python/build
|
||||||
|
26
INSTALL.md
26
INSTALL.md
@ -1,30 +1,30 @@
|
|||||||
# Installation
|
# Installation
|
||||||
* [Dependencies](#deps)
|
* [Dependencies](#dependencies)
|
||||||
* [OS X Notes](#deps_osx)
|
* [OS X Notes](#os-x-notes)
|
||||||
* [Compiling](#compiling)
|
* [Compiling](#compiling)
|
||||||
* [Documentation](#docs)
|
* [Documentation](#documentation)
|
||||||
* [Notes](#notes)
|
* [Notes](#notes)
|
||||||
* [Compilation variables](#comp_vars)
|
* [Compilation variables](#compilation-variables)
|
||||||
* [Packaging](#packaging)
|
* [Packaging](#packaging)
|
||||||
|
|
||||||
<a name="deps" />
|
|
||||||
## Dependencies
|
## Dependencies
|
||||||
| Name | Needed by | Debian package |
|
| Name | Needed by | Debian package |
|
||||||
|------------------------------------------------------|----------------------------|---------------------|
|
|------------------------------------------------------|----------------------------|---------------------|
|
||||||
| [Tox Core](https://github.com/irungentoo/toxcore) | BASE | *None* |
|
| [Tox Core](https://github.com/toktok/c-toxcore) | BASE | *None* |
|
||||||
| [NCurses](https://www.gnu.org/software/ncurses) | BASE | libncursesw5-dev |
|
| [NCurses](https://www.gnu.org/software/ncurses) | BASE | libncursesw5-dev |
|
||||||
| [LibConfig](http://www.hyperrealm.com/libconfig) | BASE | libconfig-dev |
|
| [LibConfig](http://www.hyperrealm.com/libconfig) | BASE | libconfig-dev |
|
||||||
| [GNUmake](https://www.gnu.org/software/make) | BASE | make |
|
| [GNUmake](https://www.gnu.org/software/make) | BASE | make |
|
||||||
| [libcurl](http://curl.haxx.se/) | BASE | libcurl4-openssl-dev|
|
| [libcurl](http://curl.haxx.se/) | BASE | libcurl4-openssl-dev|
|
||||||
| [libqrencode](https://fukuchi.org/works/qrencode/) | BASE | libqrencode-dev |
|
| [libqrencode](https://fukuchi.org/works/qrencode/) | BASE | libqrencode-dev |
|
||||||
| [Tox Core AV](https://github.com/irungentoo/toxcore) | AUDIO | *None* |
|
| [Tox Core AV](https://github.com/toktok/c-toxcore) | AUDIO | *None* |
|
||||||
| [OpenAL](http://openal.org) | AUDIO, SOUND NOTIFICATIONS | libopenal-dev |
|
| [OpenAL](http://openal.org) | AUDIO, SOUND NOTIFICATIONS | libopenal-dev |
|
||||||
| [OpenALUT](http://openal.org) | SOUND NOTIFICATIONS | libalut-dev |
|
| [OpenALUT](http://openal.org) | SOUND NOTIFICATIONS | libalut-dev |
|
||||||
| [LibNotify](https://developer.gnome.org/libnotify) | DESKTOP NOTIFICATIONS | libnotify-dev |
|
| [LibNotify](https://developer.gnome.org/libnotify) | DESKTOP NOTIFICATIONS | libnotify-dev |
|
||||||
|
| [Python 3](http://www.python.org/) | PYTHON | python3-dev |
|
||||||
| [AsciiDoc](http://asciidoc.org/index.html) | DOCUMENTATION<sup>1</sup> | asciidoc |
|
| [AsciiDoc](http://asciidoc.org/index.html) | DOCUMENTATION<sup>1</sup> | asciidoc |
|
||||||
<sup>1</sup>: see [Documentation](#docs)
|
|
||||||
|
|
||||||
<a name="deps_osx" />
|
<sup>1</sup>: see [Documentation](#documentation)
|
||||||
|
|
||||||
#### OS X Notes
|
#### OS X Notes
|
||||||
Using [Homebrew](http://brew.sh):
|
Using [Homebrew](http://brew.sh):
|
||||||
```
|
```
|
||||||
@ -35,23 +35,19 @@ brew install libnotify
|
|||||||
|
|
||||||
You can omit `libnotify` if you intend to build without desktop notifications enabled.
|
You can omit `libnotify` if you intend to build without desktop notifications enabled.
|
||||||
|
|
||||||
<a name="Compiling">
|
|
||||||
## Compiling
|
## Compiling
|
||||||
```
|
```
|
||||||
make PREFIX="/where/to/install"
|
make PREFIX="/where/to/install"
|
||||||
sudo make install PREFIX="/where/to/install"
|
sudo make install PREFIX="/where/to/install"
|
||||||
```
|
```
|
||||||
|
|
||||||
<a name="docs" />
|
|
||||||
#### Documentation
|
#### Documentation
|
||||||
Run `make doc` in the build directory after editing the asciidoc files to regenerate the manpages.<br />
|
Run `make doc` in the build directory after editing the asciidoc files to regenerate the manpages.<br />
|
||||||
**NOTE FOR DEVELOPERS**: asciidoc files and generated manpages will need to be commited together.<br />
|
**NOTE FOR DEVELOPERS**: asciidoc files and generated manpages will need to be commited together.<br />
|
||||||
**NOTE FOR EVERYONE**: [asciidoc](http://asciidoc.org/index.html) (and this step) is only required for regenerating manpages when you modify them.
|
**NOTE FOR EVERYONE**: [asciidoc](http://asciidoc.org/index.html) (and this step) is only required for regenerating manpages when you modify them.
|
||||||
|
|
||||||
<a name="notes" />
|
|
||||||
## Notes
|
## Notes
|
||||||
|
|
||||||
<a name="comp_vars" />
|
|
||||||
#### Compilation variables
|
#### Compilation variables
|
||||||
* You can add specific flags to the Makefile with `USER_CFLAGS=""` and/or `USER_LDFLAGS=""`
|
* You can add specific flags to the Makefile with `USER_CFLAGS=""` and/or `USER_LDFLAGS=""`
|
||||||
* You can pass your own flags to the Makefile with `CFLAGS=""` and/or `LDFLAGS=""` (this will supersede the default ones)
|
* You can pass your own flags to the Makefile with `CFLAGS=""` and/or `LDFLAGS=""` (this will supersede the default ones)
|
||||||
@ -60,11 +56,11 @@ Run `make doc` in the build directory after editing the asciidoc files to regene
|
|||||||
* `DISABLE_AV=1` → build toxic without audio call support
|
* `DISABLE_AV=1` → build toxic without audio call support
|
||||||
* `DISABLE_SOUND_NOTIFY=1` → build toxic without sound notifications support
|
* `DISABLE_SOUND_NOTIFY=1` → build toxic without sound notifications support
|
||||||
* `DISABLE_DESKTOP_NOTIFY=1` → build toxic without desktop notifications support
|
* `DISABLE_DESKTOP_NOTIFY=1` → build toxic without desktop notifications support
|
||||||
|
* Features excluded from the default build must be explicitly enabled using special variables:
|
||||||
|
* `ENABLE_PYTHON=1` → build toxic with Python scripting support
|
||||||
|
|
||||||
<a name="packaging" />
|
|
||||||
#### Packaging
|
#### Packaging
|
||||||
* For packaging purpose, you can use `DESTDIR=""` to specify a directory where to store installed files
|
* For packaging purpose, you can use `DESTDIR=""` to specify a directory where to store installed files
|
||||||
* `DESTDIR=""` can be used in addition to `PREFIX=""`:
|
* `DESTDIR=""` can be used in addition to `PREFIX=""`:
|
||||||
* `DESTDIR=""` is meant to specify a directory where to store installed files (ex: "/tmp/build/pkg")
|
* `DESTDIR=""` is meant to specify a directory where to store installed files (ex: "/tmp/build/pkg")
|
||||||
* `PREFIX=""` is meant to specify a prefix directory for binaries and data files (ex: "/usr/local")
|
* `PREFIX=""` is meant to specify a prefix directory for binaries and data files (ex: "/usr/local")
|
||||||
|
|
||||||
|
3
Makefile
3
Makefile
@ -30,6 +30,9 @@ endif
|
|||||||
ifeq ($(UNAME_S), OpenBSD)
|
ifeq ($(UNAME_S), OpenBSD)
|
||||||
-include $(CFG_DIR)/systems/FreeBSD.mk
|
-include $(CFG_DIR)/systems/FreeBSD.mk
|
||||||
endif
|
endif
|
||||||
|
ifeq ($(UNAME_S), NetBSD)
|
||||||
|
-include $(CFG_DIR)/systems/FreeBSD.mk
|
||||||
|
endif
|
||||||
ifeq ($(UNAME_S), Darwin)
|
ifeq ($(UNAME_S), Darwin)
|
||||||
-include $(CFG_DIR)/systems/Darwin.mk
|
-include $(CFG_DIR)/systems/Darwin.mk
|
||||||
endif
|
endif
|
||||||
|
@ -3,7 +3,7 @@
|
|||||||
src="https://scan.coverity.com/projects/4975/badge.svg"/>
|
src="https://scan.coverity.com/projects/4975/badge.svg"/>
|
||||||
</a>
|
</a>
|
||||||
|
|
||||||
Toxic is a [Tox](https://tox.chat)-based instant messenging client which formerly resided in the [Tox core repository](https://github.com/irungentoo/toxcore), and is now available as a standalone application.
|
Toxic is a [Tox](https://tox.chat)-based instant messenging client which formerly resided in the [Tox core repository](https://github.com/toktok/c-toxcore), and is now available as a standalone application.
|
||||||
|
|
||||||
[](https://i.imgur.com/san99Z2.png)
|
[](https://i.imgur.com/san99Z2.png)
|
||||||
|
|
||||||
|
20
apidoc/python/Makefile
Normal file
20
apidoc/python/Makefile
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
# Minimal makefile for Sphinx documentation
|
||||||
|
#
|
||||||
|
|
||||||
|
# You can set these variables from the command line.
|
||||||
|
SPHINXOPTS =
|
||||||
|
SPHINXBUILD = sphinx-build
|
||||||
|
SPHINXPROJ = toxic_api
|
||||||
|
SOURCEDIR = source
|
||||||
|
BUILDDIR = build
|
||||||
|
|
||||||
|
# Put it first so that "make" without argument is like "make help".
|
||||||
|
help:
|
||||||
|
@$(SPHINXBUILD) -M help "$(SOURCEDIR)" "$(BUILDDIR)" $(SPHINXOPTS) $(O)
|
||||||
|
|
||||||
|
.PHONY: help Makefile
|
||||||
|
|
||||||
|
# Catch-all target: route all unknown targets to Sphinx using the new
|
||||||
|
# "make mode" option. $(O) is meant as a shortcut for $(SPHINXOPTS).
|
||||||
|
%: Makefile
|
||||||
|
@$(SPHINXBUILD) -M $@ "$(SOURCEDIR)" "$(BUILDDIR)" $(SPHINXOPTS) $(O)
|
157
apidoc/python/source/conf.py
Normal file
157
apidoc/python/source/conf.py
Normal file
@ -0,0 +1,157 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
#
|
||||||
|
# toxic_api documentation build configuration file, created by
|
||||||
|
# sphinx-quickstart on Tue May 16 08:58:24 2017.
|
||||||
|
#
|
||||||
|
# This file is execfile()d with the current directory set to its
|
||||||
|
# containing dir.
|
||||||
|
#
|
||||||
|
# Note that not all possible configuration values are present in this
|
||||||
|
# autogenerated file.
|
||||||
|
#
|
||||||
|
# All configuration values have a default; values that are commented out
|
||||||
|
# serve to show the default.
|
||||||
|
|
||||||
|
# If extensions (or modules to document with autodoc) are in another directory,
|
||||||
|
# add these directories to sys.path here. If the directory is relative to the
|
||||||
|
# documentation root, use os.path.abspath to make it absolute, like shown here.
|
||||||
|
#
|
||||||
|
# import os
|
||||||
|
# import sys
|
||||||
|
# sys.path.insert(0, os.path.abspath('.'))
|
||||||
|
|
||||||
|
|
||||||
|
# -- General configuration ------------------------------------------------
|
||||||
|
|
||||||
|
# If your documentation needs a minimal Sphinx version, state it here.
|
||||||
|
#
|
||||||
|
# needs_sphinx = '1.0'
|
||||||
|
|
||||||
|
# Add any Sphinx extension module names here, as strings. They can be
|
||||||
|
# extensions coming with Sphinx (named 'sphinx.ext.*') or your custom
|
||||||
|
# ones.
|
||||||
|
extensions = []
|
||||||
|
|
||||||
|
# Add any paths that contain templates here, relative to this directory.
|
||||||
|
templates_path = ['_templates']
|
||||||
|
|
||||||
|
# The suffix(es) of source filenames.
|
||||||
|
# You can specify multiple suffix as a list of string:
|
||||||
|
#
|
||||||
|
# source_suffix = ['.rst', '.md']
|
||||||
|
source_suffix = '.rst'
|
||||||
|
|
||||||
|
# The master toctree document.
|
||||||
|
master_doc = 'index'
|
||||||
|
|
||||||
|
# General information about the project.
|
||||||
|
project = 'toxic_api'
|
||||||
|
copyright = '2017, Jakob Kreuze'
|
||||||
|
author = 'Jakob Kreuze'
|
||||||
|
|
||||||
|
# The version info for the project you're documenting, acts as replacement for
|
||||||
|
# |version| and |release|, also used in various other places throughout the
|
||||||
|
# built documents.
|
||||||
|
#
|
||||||
|
# The short X.Y version.
|
||||||
|
version = '0.8.1'
|
||||||
|
# The full version, including alpha/beta/rc tags.
|
||||||
|
release = '0.8.1'
|
||||||
|
|
||||||
|
# The language for content autogenerated by Sphinx. Refer to documentation
|
||||||
|
# for a list of supported languages.
|
||||||
|
#
|
||||||
|
# This is also used if you do content translation via gettext catalogs.
|
||||||
|
# Usually you set "language" from the command line for these cases.
|
||||||
|
language = None
|
||||||
|
|
||||||
|
# List of patterns, relative to source directory, that match files and
|
||||||
|
# directories to ignore when looking for source files.
|
||||||
|
# This patterns also effect to html_static_path and html_extra_path
|
||||||
|
exclude_patterns = []
|
||||||
|
|
||||||
|
# The name of the Pygments (syntax highlighting) style to use.
|
||||||
|
pygments_style = 'sphinx'
|
||||||
|
|
||||||
|
# If true, `todo` and `todoList` produce output, else they produce nothing.
|
||||||
|
todo_include_todos = False
|
||||||
|
|
||||||
|
|
||||||
|
# -- Options for HTML output ----------------------------------------------
|
||||||
|
|
||||||
|
# The theme to use for HTML and HTML Help pages. See the documentation for
|
||||||
|
# a list of builtin themes.
|
||||||
|
#
|
||||||
|
html_theme = 'alabaster'
|
||||||
|
|
||||||
|
# Theme options are theme-specific and customize the look and feel of a theme
|
||||||
|
# further. For a list of options available for each theme, see the
|
||||||
|
# documentation.
|
||||||
|
#
|
||||||
|
# html_theme_options = {}
|
||||||
|
|
||||||
|
# Add any paths that contain custom static files (such as style sheets) here,
|
||||||
|
# relative to this directory. They are copied after the builtin static files,
|
||||||
|
# so a file named "default.css" will overwrite the builtin "default.css".
|
||||||
|
html_static_path = ['_static']
|
||||||
|
|
||||||
|
|
||||||
|
# -- Options for HTMLHelp output ------------------------------------------
|
||||||
|
|
||||||
|
# Output file base name for HTML help builder.
|
||||||
|
htmlhelp_basename = 'toxic_apidoc'
|
||||||
|
|
||||||
|
|
||||||
|
# -- Options for LaTeX output ---------------------------------------------
|
||||||
|
|
||||||
|
latex_elements = {
|
||||||
|
# The paper size ('letterpaper' or 'a4paper').
|
||||||
|
#
|
||||||
|
# 'papersize': 'letterpaper',
|
||||||
|
|
||||||
|
# The font size ('10pt', '11pt' or '12pt').
|
||||||
|
#
|
||||||
|
# 'pointsize': '10pt',
|
||||||
|
|
||||||
|
# Additional stuff for the LaTeX preamble.
|
||||||
|
#
|
||||||
|
# 'preamble': '',
|
||||||
|
|
||||||
|
# Latex figure (float) alignment
|
||||||
|
#
|
||||||
|
# 'figure_align': 'htbp',
|
||||||
|
}
|
||||||
|
|
||||||
|
# Grouping the document tree into LaTeX files. List of tuples
|
||||||
|
# (source start file, target name, title,
|
||||||
|
# author, documentclass [howto, manual, or own class]).
|
||||||
|
latex_documents = [
|
||||||
|
(master_doc, 'toxic_api.tex', 'toxic\\_api Documentation',
|
||||||
|
'Jakob Kreuze', 'manual'),
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
# -- Options for manual page output ---------------------------------------
|
||||||
|
|
||||||
|
# One entry per manual page. List of tuples
|
||||||
|
# (source start file, name, description, authors, manual section).
|
||||||
|
man_pages = [
|
||||||
|
(master_doc, 'toxic_api', 'toxic_api Documentation',
|
||||||
|
[author], 1)
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
# -- Options for Texinfo output -------------------------------------------
|
||||||
|
|
||||||
|
# Grouping the document tree into Texinfo files. List of tuples
|
||||||
|
# (source start file, target name, title, author,
|
||||||
|
# dir menu entry, description, category)
|
||||||
|
texinfo_documents = [
|
||||||
|
(master_doc, 'toxic_api', 'toxic_api Documentation',
|
||||||
|
author, 'toxic_api', 'One line description of project.',
|
||||||
|
'Miscellaneous'),
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
|
8
apidoc/python/source/examples.rst
Normal file
8
apidoc/python/source/examples.rst
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
============
|
||||||
|
API Examples
|
||||||
|
============
|
||||||
|
|
||||||
|
Fortune
|
||||||
|
=======
|
||||||
|
.. literalinclude:: fortune.py
|
||||||
|
:language: python
|
37
apidoc/python/source/fortune.py
Normal file
37
apidoc/python/source/fortune.py
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
import toxic_api
|
||||||
|
import random
|
||||||
|
|
||||||
|
FORTUNES = [
|
||||||
|
"A bug in the code is worth two in the documentation.",
|
||||||
|
"A bug in the hand is better than one as yet undetected.",
|
||||||
|
"\"A debugged program is one for which you have not yet found the "
|
||||||
|
"conditions that make it fail.\" -- Jerry Ogdin"
|
||||||
|
]
|
||||||
|
|
||||||
|
def send_fortune(args):
|
||||||
|
"""Callback function that sends the contact of the current window a
|
||||||
|
given number of random fortunes.
|
||||||
|
"""
|
||||||
|
if len(args) != 1:
|
||||||
|
toxic_api.display("Only one argument allowed!")
|
||||||
|
return
|
||||||
|
|
||||||
|
try:
|
||||||
|
count = int(args[0])
|
||||||
|
except ValueError:
|
||||||
|
toxic_api.display("Argument must be a number!")
|
||||||
|
return
|
||||||
|
|
||||||
|
if count < 0 or count > 20:
|
||||||
|
toxic_api.display("Argument is too large!")
|
||||||
|
return
|
||||||
|
|
||||||
|
name = toxic_api.get_nick()
|
||||||
|
|
||||||
|
toxic_api.send("%s has decided to send you %d fortunes:" % (name, count))
|
||||||
|
for _ in range(count):
|
||||||
|
toxic_api.send(random.choice(FORTUNES))
|
||||||
|
|
||||||
|
|
||||||
|
toxic_api.register("/fortune", "Send a fortune to the contact of the current "
|
||||||
|
"window", send_fortune)
|
9
apidoc/python/source/index.rst
Normal file
9
apidoc/python/source/index.rst
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
Toxic Scripting Interface Documentation
|
||||||
|
=======================================
|
||||||
|
|
||||||
|
.. toctree::
|
||||||
|
:maxdepth: 2
|
||||||
|
|
||||||
|
intro
|
||||||
|
reference
|
||||||
|
examples
|
12
apidoc/python/source/intro.rst
Normal file
12
apidoc/python/source/intro.rst
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
=========================
|
||||||
|
Toxic Scripting Interface
|
||||||
|
=========================
|
||||||
|
|
||||||
|
A Python scripting interface to `Toxic <https://github.com/JFreegman/toxic>`_.
|
||||||
|
|
||||||
|
|
||||||
|
Getting Started
|
||||||
|
===============
|
||||||
|
Toxic is compiled with Python support by default. To access the scripting interface, simply import "toxic_api" in your script.
|
||||||
|
|
||||||
|
Scripts can be run by issuing "/run <path>" from toxic, or placing them in the "autorun_path" from your toxic configuration file.
|
73
apidoc/python/source/reference.rst
Normal file
73
apidoc/python/source/reference.rst
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
=============
|
||||||
|
API Reference
|
||||||
|
=============
|
||||||
|
|
||||||
|
Messages
|
||||||
|
========
|
||||||
|
.. function:: display(msg)
|
||||||
|
|
||||||
|
Display a message to the user through the current window.
|
||||||
|
|
||||||
|
:param msg: The message to display.
|
||||||
|
:type msg: string
|
||||||
|
:rtype: none
|
||||||
|
|
||||||
|
.. function:: send(msg)
|
||||||
|
|
||||||
|
Send a message to the user specified by the currently open conversation.
|
||||||
|
|
||||||
|
:param msg: The message to display.
|
||||||
|
:type msg: string
|
||||||
|
:rtype: none
|
||||||
|
|
||||||
|
|
||||||
|
State
|
||||||
|
=====
|
||||||
|
.. function:: get_nick()
|
||||||
|
|
||||||
|
Return the user's current nickname.
|
||||||
|
|
||||||
|
:rtype: string
|
||||||
|
|
||||||
|
.. function:: get_status()
|
||||||
|
|
||||||
|
Return a string representing the user's current status. Can be either "online", "away", or "busy".
|
||||||
|
|
||||||
|
:rtype: string
|
||||||
|
|
||||||
|
.. function:: get_status_message()
|
||||||
|
|
||||||
|
Return the user's current status message.
|
||||||
|
|
||||||
|
:rtype: string
|
||||||
|
|
||||||
|
.. function:: get_all_friends()
|
||||||
|
|
||||||
|
Return a list of all the user's friends.
|
||||||
|
|
||||||
|
:rtype: list of (string, string) tuples containing the nickname followed by their public key
|
||||||
|
|
||||||
|
|
||||||
|
Commands
|
||||||
|
========
|
||||||
|
.. function:: execute(command, class)
|
||||||
|
|
||||||
|
Executes the given command. The API exports three constants for the class parameter; GLOBAL_COMMAND, CHAT_COMMAND, and GROUPCHAT_COMMAND.
|
||||||
|
|
||||||
|
:param command: The command to execute.
|
||||||
|
:type command: string
|
||||||
|
:param class: The class of the command.
|
||||||
|
:type class: int
|
||||||
|
:rtype: none
|
||||||
|
|
||||||
|
.. function:: register(command, help, callback)
|
||||||
|
|
||||||
|
Register a callback to be executed whenever command is run. The callback function will be called with one argument, a list of arguments from when the user calls the command.
|
||||||
|
|
||||||
|
:param command: The command to listen for.
|
||||||
|
:type command: string
|
||||||
|
:param help: A description of the command to be shown in the help menu.
|
||||||
|
:type help: string
|
||||||
|
:param callback: The function to be called.
|
||||||
|
:type callback: callable
|
||||||
|
:rtype: none
|
@ -18,4 +18,4 @@ else ifneq ($(MAKECMDGOALS), clean)
|
|||||||
$(warning WARNING -- Toxic will be compiled without audio support)
|
$(warning WARNING -- Toxic will be compiled without audio support)
|
||||||
$(warning WARNING -- You need these libraries for audio support)
|
$(warning WARNING -- You need these libraries for audio support)
|
||||||
$(warning WARNING -- $(MISSING_AUDIO_LIBS))
|
$(warning WARNING -- $(MISSING_AUDIO_LIBS))
|
||||||
endif
|
endif
|
||||||
|
@ -40,6 +40,12 @@ ifneq ($(QR_PNG), disabled)
|
|||||||
-include $(CHECKS_DIR)/qr_png.mk
|
-include $(CHECKS_DIR)/qr_png.mk
|
||||||
endif
|
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)
|
||||||
|
ifneq ($(PYTHON), disabled)
|
||||||
|
-include $(CHECKS_DIR)/python.mk
|
||||||
|
endif
|
||||||
|
|
||||||
# Check if we can build Toxic
|
# 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)
|
ifneq ($(CHECK_LIBS), error)
|
||||||
|
15
cfg/checks/python.mk
Normal file
15
cfg/checks/python.mk
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
# Variables for Python scripting support
|
||||||
|
PYTHON3_LIBS = python3
|
||||||
|
PYTHON_CFLAGS = -DPYTHON
|
||||||
|
PYTHON_OBJ = api.o python_api.o
|
||||||
|
|
||||||
|
# Check if we can build Python scripting support
|
||||||
|
CHECK_PYTHON3_LIBS = $(shell $(PKG_CONFIG) --exists $(PYTHON3_LIBS) || echo -n "error")
|
||||||
|
ifneq ($(CHECK_PYTHON3_LIBS), error)
|
||||||
|
LDFLAGS += $(shell python3-config --ldflags)
|
||||||
|
CFLAGS += $(PYTHON_CFLAGS) $(shell python3-config --includes)
|
||||||
|
OBJ += $(PYTHON_OBJ)
|
||||||
|
else ifneq ($(MAKECMDGOALS), clean)
|
||||||
|
$(warning WARNING -- Toxic will be compiled without Python scripting support)
|
||||||
|
$(warning WARNING -- You need python3 installed for Python scripting support)
|
||||||
|
endif
|
@ -1,5 +1,5 @@
|
|||||||
# Version
|
# Version
|
||||||
TOXIC_VERSION = 0.7.2
|
TOXIC_VERSION = 0.8.1
|
||||||
REV = $(shell git rev-list HEAD --count 2>/dev/null || echo -n "error")
|
REV = $(shell git rev-list HEAD --count 2>/dev/null || echo -n "error")
|
||||||
ifneq (, $(findstring error, $(REV)))
|
ifneq (, $(findstring error, $(REV)))
|
||||||
VERSION = $(TOXIC_VERSION)
|
VERSION = $(TOXIC_VERSION)
|
||||||
|
@ -15,6 +15,7 @@ help:
|
|||||||
@echo " DISABLE_SOUND_NOTIFY: Set to \"1\" to force building without sound notification support"
|
@echo " DISABLE_SOUND_NOTIFY: Set to \"1\" to force building without sound notification support"
|
||||||
@echo " DISABLE_DESKTOP_NOTIFY: Set to \"1\" to force building without desktop notifications support"
|
@echo " DISABLE_DESKTOP_NOTIFY: Set to \"1\" to force building without desktop notifications support"
|
||||||
@echo " DISABLE_QRPNG: Set to \"1\" to force building without QR exported as PNG 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 " USER_CFLAGS: Add custom flags to default CFLAGS"
|
@echo " USER_CFLAGS: Add custom flags to default CFLAGS"
|
||||||
@echo " USER_LDFLAGS: Add custom flags to default LDFLAGS"
|
@echo " USER_LDFLAGS: Add custom flags to default LDFLAGS"
|
||||||
@echo " PREFIX: Specify a prefix directory for binaries, data files,... (default is \"$(abspath $(PREFIX))\")"
|
@echo " PREFIX: Specify a prefix directory for binaries, data files,... (default is \"$(abspath $(PREFIX))\")"
|
||||||
|
@ -35,7 +35,7 @@ install: $(BUILD_DIR)/toxic
|
|||||||
mv temp_file $$file ;\
|
mv temp_file $$file ;\
|
||||||
sed -e 's:__DATADIR__:'$(abspath $(DATADIR))':g' $$file > temp_file && \
|
sed -e 's:__DATADIR__:'$(abspath $(DATADIR))':g' $$file > temp_file && \
|
||||||
mv temp_file $$file ;\
|
mv temp_file $$file ;\
|
||||||
gzip -f -9 $$file ;\
|
gzip -f -n -9 $$file ;\
|
||||||
done
|
done
|
||||||
|
|
||||||
.PHONY: install
|
.PHONY: install
|
||||||
|
@ -1,13 +1,13 @@
|
|||||||
'\" t
|
'\" t
|
||||||
.\" Title: toxic.conf
|
.\" Title: toxic.conf
|
||||||
.\" Author: [see the "AUTHORS" section]
|
.\" Author: [see the "AUTHORS" section]
|
||||||
.\" Generator: DocBook XSL Stylesheets v1.78.1 <http://docbook.sf.net/>
|
.\" Generator: DocBook XSL Stylesheets v1.79.1 <http://docbook.sf.net/>
|
||||||
.\" Date: 2016-07-21
|
.\" Date: 2016-09-20
|
||||||
.\" Manual: Toxic Manual
|
.\" Manual: Toxic Manual
|
||||||
.\" Source: toxic __VERSION__
|
.\" Source: toxic __VERSION__
|
||||||
.\" Language: English
|
.\" Language: English
|
||||||
.\"
|
.\"
|
||||||
.TH "TOXIC\&.CONF" "5" "2016\-07\-21" "toxic __VERSION__" "Toxic Manual"
|
.TH "TOXIC\&.CONF" "5" "2016\-09\-20" "toxic __VERSION__" "Toxic Manual"
|
||||||
.\" -----------------------------------------------------------------
|
.\" -----------------------------------------------------------------
|
||||||
.\" * Define some portability stuff
|
.\" * Define some portability stuff
|
||||||
.\" -----------------------------------------------------------------
|
.\" -----------------------------------------------------------------
|
||||||
@ -227,6 +227,11 @@ Default path for downloads\&. String value\&. Absolute path for downloaded files
|
|||||||
Path for your avatar (file must be a \&.png and cannot exceed 16\&.3 KiB)
|
Path for your avatar (file must be a \&.png and cannot exceed 16\&.3 KiB)
|
||||||
.RE
|
.RE
|
||||||
.PP
|
.PP
|
||||||
|
\fBautorun_path\fR
|
||||||
|
.RS 4
|
||||||
|
Path for any scripts that should be run on startup
|
||||||
|
.RE
|
||||||
|
.PP
|
||||||
\fBchatlogs_path\fR
|
\fBchatlogs_path\fR
|
||||||
.RS 4
|
.RS 4
|
||||||
Default path for chatlogs\&. String value\&. Absolute path for chatlog files\&.
|
Default path for chatlogs\&. String value\&. Absolute path for chatlog files\&.
|
||||||
|
@ -143,6 +143,9 @@ OPTIONS
|
|||||||
*avatar_path*;;
|
*avatar_path*;;
|
||||||
Path for your avatar (file must be a .png and cannot exceed 16.3 KiB)
|
Path for your avatar (file must be a .png and cannot exceed 16.3 KiB)
|
||||||
|
|
||||||
|
*autorun_path*;;
|
||||||
|
Path for any scripts that should be run on startup
|
||||||
|
|
||||||
*chatlogs_path*;;
|
*chatlogs_path*;;
|
||||||
Default path for chatlogs. String value. Absolute path for chatlog files.
|
Default path for chatlogs. String value. Absolute path for chatlog files.
|
||||||
|
|
||||||
|
@ -87,6 +87,9 @@ tox = {
|
|||||||
// Path for your avatar (file must be a .png and cannot exceed 64 KiB)
|
// Path for your avatar (file must be a .png and cannot exceed 64 KiB)
|
||||||
// avatar_path="/home/USERNAME/Pictures/youravatar.png";
|
// avatar_path="/home/USERNAME/Pictures/youravatar.png";
|
||||||
|
|
||||||
|
// Path for scripts that should be run on startup
|
||||||
|
// autorun_path="/home/USERNAME/toxic_scripts/";
|
||||||
|
|
||||||
// Path for chatlogs
|
// Path for chatlogs
|
||||||
// chatlogs_path="/home/USERNAME/toxic_chatlogs/";
|
// chatlogs_path="/home/USERNAME/toxic_chatlogs/";
|
||||||
};
|
};
|
||||||
@ -118,4 +121,3 @@ keys = {
|
|||||||
toggle_peerlist="Ctrl+b";
|
toggle_peerlist="Ctrl+b";
|
||||||
toggle_paste_mode="Ctrl+T";
|
toggle_paste_mode="Ctrl+T";
|
||||||
};
|
};
|
||||||
|
|
||||||
|
210
src/api.c
Normal file
210
src/api.c
Normal file
@ -0,0 +1,210 @@
|
|||||||
|
/* api.c
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* Copyright (C) 2017 Jakob Kreuze <jakob@memeware.net>
|
||||||
|
*
|
||||||
|
* 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 <dirent.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
|
#include <tox/tox.h>
|
||||||
|
|
||||||
|
#include "execute.h"
|
||||||
|
#include "friendlist.h"
|
||||||
|
#include "line_info.h"
|
||||||
|
#include "message_queue.h"
|
||||||
|
#include "misc_tools.h"
|
||||||
|
#include "settings.h"
|
||||||
|
#include "toxic_strings.h"
|
||||||
|
#include "windows.h"
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
#include "python_api.h"
|
||||||
|
#endif /* PYTHON */
|
||||||
|
|
||||||
|
Tox *user_tox;
|
||||||
|
static WINDOW *cur_window;
|
||||||
|
static ToxWindow *self_window;
|
||||||
|
|
||||||
|
extern FriendsList Friends;
|
||||||
|
extern struct user_settings *user_settings;
|
||||||
|
|
||||||
|
void api_display(const char *const msg)
|
||||||
|
{
|
||||||
|
if (msg == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
self_window = get_active_window();
|
||||||
|
line_info_add(self_window, NULL, NULL, NULL, SYS_MSG, 0, 0, msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
FriendsList api_get_friendslist(void)
|
||||||
|
{
|
||||||
|
return Friends;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *api_get_nick(void)
|
||||||
|
{
|
||||||
|
size_t len = tox_self_get_name_size(user_tox);
|
||||||
|
uint8_t *name = malloc(len + 1);
|
||||||
|
|
||||||
|
if (name == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
tox_self_get_name(user_tox, name);
|
||||||
|
name[len] = '\0';
|
||||||
|
return (char *) name;
|
||||||
|
}
|
||||||
|
|
||||||
|
TOX_USER_STATUS api_get_status(void)
|
||||||
|
{
|
||||||
|
return tox_self_get_status(user_tox);
|
||||||
|
}
|
||||||
|
|
||||||
|
char *api_get_status_message(void)
|
||||||
|
{
|
||||||
|
size_t len = tox_self_get_status_message_size(user_tox);
|
||||||
|
uint8_t *status = malloc(len + 1);
|
||||||
|
|
||||||
|
if (status == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
tox_self_get_status_message(user_tox, status);
|
||||||
|
status[len] = '\0';
|
||||||
|
return (char *) status;
|
||||||
|
}
|
||||||
|
|
||||||
|
void api_send(const char *msg)
|
||||||
|
{
|
||||||
|
if (msg == NULL || self_window->chatwin->cqueue == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
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);
|
||||||
|
cqueue_add(self_window->chatwin->cqueue, msg, strlen(msg), OUT_MSG, id);
|
||||||
|
free(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
void api_execute(const char *input, int mode)
|
||||||
|
{
|
||||||
|
self_window = get_active_window();
|
||||||
|
execute(cur_window, self_window, user_tox, input, mode);
|
||||||
|
}
|
||||||
|
|
||||||
|
int do_plugin_command(int num_args, char (*args)[MAX_STR_SIZE])
|
||||||
|
{
|
||||||
|
return do_python_command(num_args, args);
|
||||||
|
}
|
||||||
|
|
||||||
|
int num_registered_handlers(void)
|
||||||
|
{
|
||||||
|
return python_num_registered_handlers();
|
||||||
|
}
|
||||||
|
|
||||||
|
int help_max_width(void)
|
||||||
|
{
|
||||||
|
return python_help_max_width();
|
||||||
|
}
|
||||||
|
|
||||||
|
void draw_handler_help(WINDOW *win)
|
||||||
|
{
|
||||||
|
python_draw_handler_help(win);
|
||||||
|
}
|
||||||
|
|
||||||
|
void cmd_run(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE])
|
||||||
|
{
|
||||||
|
FILE *fp;
|
||||||
|
const char *error_str;
|
||||||
|
|
||||||
|
cur_window = window;
|
||||||
|
self_window = self;
|
||||||
|
|
||||||
|
if ( argc != 1 ) {
|
||||||
|
if ( argc < 1 ) error_str = "Path must be specified!";
|
||||||
|
else error_str = "Only one argument allowed!";
|
||||||
|
|
||||||
|
line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, error_str);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
fp = fopen(argv[1], "r");
|
||||||
|
|
||||||
|
if ( fp == NULL ) {
|
||||||
|
error_str = "Path does not exist!";
|
||||||
|
|
||||||
|
line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, error_str);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
run_python(fp, argv[1]);
|
||||||
|
fclose(fp);
|
||||||
|
}
|
||||||
|
|
||||||
|
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;
|
||||||
|
|
||||||
|
if (user_settings->autorun_path[0] == '\0')
|
||||||
|
return;
|
||||||
|
|
||||||
|
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);
|
||||||
|
api_display(err_buf);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
cur_window = window;
|
||||||
|
self_window = self;
|
||||||
|
|
||||||
|
while ((dir = readdir(d)) != NULL) {
|
||||||
|
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");
|
||||||
|
|
||||||
|
if (fp == NULL) {
|
||||||
|
snprintf(err_buf, PATH_MAX + 1, "Invalid path: %s", abspath_buf);
|
||||||
|
api_display(err_buf);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
run_python(fp, abspath_buf);
|
||||||
|
fclose(fp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
closedir(d);
|
||||||
|
}
|
42
src/api.h
Normal file
42
src/api.h
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
/* api.h
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* Copyright (C) 2017 Jakob Kreuze <jakob@memeware.net>
|
||||||
|
*
|
||||||
|
* 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 API_H
|
||||||
|
#define API_H
|
||||||
|
|
||||||
|
#include "friendlist.h"
|
||||||
|
#include "windows.h"
|
||||||
|
|
||||||
|
void api_display(const char *const msg);
|
||||||
|
FriendsList api_get_friendslist(void);
|
||||||
|
char *api_get_nick(void);
|
||||||
|
TOX_USER_STATUS api_get_status(void);
|
||||||
|
char *api_get_status_message(void);
|
||||||
|
void api_send(const char *msg);
|
||||||
|
void api_execute(const char *input, int mode);
|
||||||
|
int do_plugin_command(int num_args, char (*args)[MAX_STR_SIZE]);
|
||||||
|
int num_registered_handlers(void);
|
||||||
|
int help_max_width(void);
|
||||||
|
void draw_handler_help(WINDOW *win);
|
||||||
|
void invoke_autoruns(WINDOW *w, ToxWindow *self);
|
||||||
|
|
||||||
|
#endif /* #define API_H */
|
@ -342,6 +342,7 @@ void audio_bit_rate_status_cb(ToxAV *av, uint32_t friend_number, uint32_t audio_
|
|||||||
uint32_t video_bit_rate, void *user_data)
|
uint32_t video_bit_rate, void *user_data)
|
||||||
{
|
{
|
||||||
CallControl.audio_bit_rate = audio_bit_rate;
|
CallControl.audio_bit_rate = audio_bit_rate;
|
||||||
|
toxav_bit_rate_set(av, friend_number, audio_bit_rate, video_bit_rate, user_data);
|
||||||
}
|
}
|
||||||
|
|
||||||
void callback_recv_invite(Tox *m, uint32_t friend_number)
|
void callback_recv_invite(Tox *m, uint32_t friend_number)
|
||||||
@ -387,7 +388,7 @@ void callback_recv_starting(uint32_t friend_number)
|
|||||||
windows[i].onStarting(&windows[i], CallControl.av, friend_number, CallControl.call_state);
|
windows[i].onStarting(&windows[i], CallControl.av, friend_number, CallControl.call_state);
|
||||||
|
|
||||||
if ( 0 != start_transmission(&windows[i], &CallControl.calls[friend_number]) ) /* YEAH! */
|
if ( 0 != start_transmission(&windows[i], &CallControl.calls[friend_number]) ) /* YEAH! */
|
||||||
line_info_add(&windows[i], NULL, NULL, NULL, SYS_MSG, 0, 0 , "Error starting transmission!");
|
line_info_add(&windows[i], NULL, NULL, NULL, SYS_MSG, 0, 0, "Error starting transmission!");
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -632,6 +633,9 @@ void cmd_list_devices(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Refresh device list.
|
||||||
|
get_devices_names();
|
||||||
|
|
||||||
print_devices(self, type);
|
print_devices(self, type);
|
||||||
|
|
||||||
return;
|
return;
|
||||||
@ -834,6 +838,55 @@ on_error:
|
|||||||
print_err (self, error_str);
|
print_err (self, error_str);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void cmd_bitrate(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX_STR_SIZE])
|
||||||
|
{
|
||||||
|
char *error_str;
|
||||||
|
|
||||||
|
if ( argc != 1 ) {
|
||||||
|
error_str = "Must have value!";
|
||||||
|
goto on_error;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( self->is_call == false ) {
|
||||||
|
error_str = "Must be in a call";
|
||||||
|
goto on_error;
|
||||||
|
}
|
||||||
|
|
||||||
|
const uint32_t bitrate = strtol(argv[1], NULL, 10);
|
||||||
|
|
||||||
|
TOXAV_ERR_BIT_RATE_SET error;
|
||||||
|
audio_bit_rate_status_cb(CallControl.av, self->num, bitrate, -1, &error);
|
||||||
|
|
||||||
|
if (error != TOXAV_ERR_BIT_RATE_SET_OK) {
|
||||||
|
switch (error) {
|
||||||
|
case TOXAV_ERR_BIT_RATE_SET_SYNC:
|
||||||
|
error_str = "Syncronization error occured";
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TOXAV_ERR_BIT_RATE_SET_INVALID_AUDIO_BIT_RATE:
|
||||||
|
error_str = "Invalid audio bit rate value (valid is 6-510)";
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TOXAV_ERR_BIT_RATE_SET_FRIEND_NOT_FOUND:
|
||||||
|
error_str = "Friend not found";
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TOXAV_ERR_BIT_RATE_SET_FRIEND_NOT_IN_CALL:
|
||||||
|
error_str = "Friend is not in the call";
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
error_str = "Unknown error";
|
||||||
|
}
|
||||||
|
|
||||||
|
goto on_error;
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
|
||||||
|
on_error:
|
||||||
|
print_err (self, error_str);
|
||||||
|
}
|
||||||
|
|
||||||
void stop_current_call(ToxWindow *self)
|
void stop_current_call(ToxWindow *self)
|
||||||
{
|
{
|
||||||
|
@ -100,34 +100,7 @@ DeviceError init_devices(ToxAV *av_)
|
|||||||
DeviceError init_devices()
|
DeviceError init_devices()
|
||||||
#endif /* AUDIO */
|
#endif /* AUDIO */
|
||||||
{
|
{
|
||||||
const char *stringed_device_list;
|
get_devices_names();
|
||||||
|
|
||||||
size[input] = 0;
|
|
||||||
|
|
||||||
if ( (stringed_device_list = alcGetString(NULL, ALC_CAPTURE_DEVICE_SPECIFIER)) ) {
|
|
||||||
ddevice_names[input] = alcGetString(NULL, ALC_CAPTURE_DEFAULT_DEVICE_SPECIFIER);
|
|
||||||
|
|
||||||
for ( ; *stringed_device_list && size[input] < MAX_DEVICES; ++size[input] ) {
|
|
||||||
devices_names[input][size[input]] = stringed_device_list;
|
|
||||||
stringed_device_list += strlen( stringed_device_list ) + 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
size[output] = 0;
|
|
||||||
|
|
||||||
if (alcIsExtensionPresent(NULL, "ALC_ENUMERATE_ALL_EXT") != AL_FALSE)
|
|
||||||
stringed_device_list = alcGetString(NULL, ALC_ALL_DEVICES_SPECIFIER);
|
|
||||||
else
|
|
||||||
stringed_device_list = alcGetString(NULL, ALC_DEVICE_SPECIFIER);
|
|
||||||
|
|
||||||
if (stringed_device_list) {
|
|
||||||
ddevice_names[output] = alcGetString(NULL, ALC_DEFAULT_DEVICE_SPECIFIER);
|
|
||||||
|
|
||||||
for ( ; *stringed_device_list && size[output] < MAX_DEVICES; ++size[output] ) {
|
|
||||||
devices_names[output][size[output]] = stringed_device_list;
|
|
||||||
stringed_device_list += strlen( stringed_device_list ) + 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Start poll thread
|
// Start poll thread
|
||||||
if (pthread_mutex_init(&mutex, NULL) != 0)
|
if (pthread_mutex_init(&mutex, NULL) != 0)
|
||||||
@ -160,6 +133,38 @@ DeviceError terminate_devices()
|
|||||||
return (DeviceError) de_None;
|
return (DeviceError) de_None;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void get_devices_names() {
|
||||||
|
|
||||||
|
const char *stringed_device_list;
|
||||||
|
|
||||||
|
size[input] = 0;
|
||||||
|
|
||||||
|
if ( (stringed_device_list = alcGetString(NULL, ALC_CAPTURE_DEVICE_SPECIFIER)) ) {
|
||||||
|
ddevice_names[input] = alcGetString(NULL, ALC_CAPTURE_DEFAULT_DEVICE_SPECIFIER);
|
||||||
|
|
||||||
|
for ( ; *stringed_device_list && size[input] < MAX_DEVICES; ++size[input] ) {
|
||||||
|
devices_names[input][size[input]] = stringed_device_list;
|
||||||
|
stringed_device_list += strlen( stringed_device_list ) + 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
size[output] = 0;
|
||||||
|
|
||||||
|
if (alcIsExtensionPresent(NULL, "ALC_ENUMERATE_ALL_EXT") != AL_FALSE)
|
||||||
|
stringed_device_list = alcGetString(NULL, ALC_ALL_DEVICES_SPECIFIER);
|
||||||
|
else
|
||||||
|
stringed_device_list = alcGetString(NULL, ALC_DEVICE_SPECIFIER);
|
||||||
|
|
||||||
|
if (stringed_device_list) {
|
||||||
|
ddevice_names[output] = alcGetString(NULL, ALC_DEFAULT_DEVICE_SPECIFIER);
|
||||||
|
|
||||||
|
for ( ; *stringed_device_list && size[output] < MAX_DEVICES; ++size[output] ) {
|
||||||
|
devices_names[output][size[output]] = stringed_device_list;
|
||||||
|
stringed_device_list += strlen( stringed_device_list ) + 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
DeviceError device_mute(DeviceType type, uint32_t device_idx)
|
DeviceError device_mute(DeviceType type, uint32_t device_idx)
|
||||||
{
|
{
|
||||||
if (device_idx >= MAX_DEVICES) return de_InvalidSelection;
|
if (device_idx >= MAX_DEVICES) return de_InvalidSelection;
|
||||||
@ -481,8 +486,9 @@ void print_devices(ToxWindow *self, DeviceType type)
|
|||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
for (i = 0; i < size[type]; ++i)
|
for (i = 0; i < size[type]; ++i) {
|
||||||
line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "%d: %s", i, devices_names[type][i]);
|
line_info_add(self, NULL, NULL, NULL, SYS_MSG, i == primary_device[type] ? 1 : 0, 0, "%d: %s", i, devices_names[type][i]);
|
||||||
|
}
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -61,6 +61,7 @@ DeviceError init_devices(ToxAV *av);
|
|||||||
DeviceError init_devices();
|
DeviceError init_devices();
|
||||||
#endif /* AUDIO */
|
#endif /* AUDIO */
|
||||||
|
|
||||||
|
void get_devices_names();
|
||||||
DeviceError terminate_devices();
|
DeviceError terminate_devices();
|
||||||
|
|
||||||
/* Callback handles ready data from INPUT device */
|
/* Callback handles ready data from INPUT device */
|
||||||
|
@ -110,6 +110,10 @@ int complete_line(ToxWindow *self, const void *list, int n_items, int size)
|
|||||||
bool dir_search = !strncmp(ubuf, "/sendfile", strlen("/sendfile"))
|
bool dir_search = !strncmp(ubuf, "/sendfile", strlen("/sendfile"))
|
||||||
|| !strncmp(ubuf, "/avatar", strlen("/avatar"));
|
|| !strncmp(ubuf, "/avatar", strlen("/avatar"));
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
dir_search = dir_search || !strncmp(ubuf, "/run", strlen("/run"));
|
||||||
|
#endif
|
||||||
|
|
||||||
/* isolate substring from space behind pos to pos */
|
/* isolate substring from space behind pos to pos */
|
||||||
char tmp[MAX_STR_SIZE];
|
char tmp[MAX_STR_SIZE];
|
||||||
snprintf(tmp, sizeof(tmp), "%s", ubuf);
|
snprintf(tmp, sizeof(tmp), "%s", ubuf);
|
||||||
@ -283,7 +287,7 @@ int dir_match(ToxWindow *self, Tox *m, const wchar_t *line, const wchar_t *cmd)
|
|||||||
if (dp == NULL)
|
if (dp == NULL)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
char dirnames[MAX_DIRS][NAME_MAX];
|
char dirnames[MAX_DIRS][NAME_MAX + 1];
|
||||||
struct dirent *entry;
|
struct dirent *entry;
|
||||||
int dircount = 0;
|
int dircount = 0;
|
||||||
|
|
||||||
|
36
src/chat.c
36
src/chat.c
@ -65,8 +65,12 @@ static void init_infobox(ToxWindow *self);
|
|||||||
static void kill_infobox(ToxWindow *self);
|
static void kill_infobox(ToxWindow *self);
|
||||||
#endif /* AUDIO */
|
#endif /* AUDIO */
|
||||||
|
|
||||||
#ifdef AUDIO
|
#if defined(AUDIO) && defined(PYTHON)
|
||||||
#define AC_NUM_CHAT_COMMANDS 30
|
#define AC_NUM_CHAT_COMMANDS 32
|
||||||
|
#elif AUDIO
|
||||||
|
#define AC_NUM_CHAT_COMMANDS 31
|
||||||
|
#elif PYTHON
|
||||||
|
#define AC_NUM_CHAT_COMMANDS 23
|
||||||
#else
|
#else
|
||||||
#define AC_NUM_CHAT_COMMANDS 22
|
#define AC_NUM_CHAT_COMMANDS 22
|
||||||
#endif /* AUDIO */
|
#endif /* AUDIO */
|
||||||
@ -106,8 +110,15 @@ static const char chat_cmd_list[AC_NUM_CHAT_COMMANDS][MAX_CMDNAME_SIZE] = {
|
|||||||
{ "/mute" },
|
{ "/mute" },
|
||||||
{ "/sense" },
|
{ "/sense" },
|
||||||
{ "/video" },
|
{ "/video" },
|
||||||
|
{ "/bitrate" },
|
||||||
|
|
||||||
#endif /* AUDIO */
|
#endif /* AUDIO */
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
|
||||||
|
{ "/run" },
|
||||||
|
|
||||||
|
#endif /* PYTHON */
|
||||||
};
|
};
|
||||||
|
|
||||||
static void set_self_typingstatus(ToxWindow *self, Tox *m, bool is_typing)
|
static void set_self_typingstatus(ToxWindow *self, Tox *m, bool is_typing)
|
||||||
@ -550,7 +561,7 @@ static void chat_onFileRecv(ToxWindow *self, Tox *m, uint32_t friendnum, uint32_
|
|||||||
bytes_convert_str(sizestr, sizeof(sizestr), file_size);
|
bytes_convert_str(sizestr, sizeof(sizestr), file_size);
|
||||||
line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "File transfer request for '%s' (%s)", filename, sizestr);
|
line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "File transfer request for '%s' (%s)", filename, sizestr);
|
||||||
|
|
||||||
char file_path[MAX_STR_SIZE];
|
char file_path[PATH_MAX + name_length + 1];
|
||||||
size_t path_len = name_length;
|
size_t path_len = name_length;
|
||||||
|
|
||||||
/* use specified download path in config if possible */
|
/* use specified download path in config if possible */
|
||||||
@ -885,8 +896,8 @@ static void send_action(ToxWindow *self, ChatContext *ctx, Tox *m, char *action)
|
|||||||
char timefrmt[TIME_STR_SIZE];
|
char timefrmt[TIME_STR_SIZE];
|
||||||
get_time_str(timefrmt, sizeof(timefrmt));
|
get_time_str(timefrmt, sizeof(timefrmt));
|
||||||
|
|
||||||
line_info_add(self, timefrmt, selfname, NULL, OUT_ACTION, 0, 0, "%s", action);
|
int id = line_info_add(self, timefrmt, selfname, NULL, OUT_ACTION, 0, 0, "%s", action);
|
||||||
cqueue_add(ctx->cqueue, action, strlen(action), OUT_ACTION, ctx->hst->line_end->id + 1);
|
cqueue_add(ctx->cqueue, action, strlen(action), OUT_ACTION, id);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void chat_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr)
|
static void chat_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr)
|
||||||
@ -931,7 +942,16 @@ static void chat_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr)
|
|||||||
diff = dir_match(self, m, ctx->line, L"/sendfile");
|
diff = dir_match(self, m, ctx->line, L"/sendfile");
|
||||||
} else if (wcsncmp(ctx->line, L"/avatar \"", wcslen(L"/avatar \"")) == 0) {
|
} else if (wcsncmp(ctx->line, L"/avatar \"", wcslen(L"/avatar \"")) == 0) {
|
||||||
diff = dir_match(self, m, ctx->line, L"/avatar");
|
diff = dir_match(self, m, ctx->line, L"/avatar");
|
||||||
} else if (wcsncmp(ctx->line, L"/status ", wcslen(L"/status ")) == 0) {
|
}
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
else if (wcsncmp(ctx->line, L"/run \"", wcslen(L"/run \"")) == 0) {
|
||||||
|
diff = dir_match(self, m, ctx->line, L"/run");
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
else if (wcsncmp(ctx->line, L"/status ", wcslen(L"/status ")) == 0) {
|
||||||
const char status_cmd_list[3][8] = {
|
const char status_cmd_list[3][8] = {
|
||||||
{"online"},
|
{"online"},
|
||||||
{"away"},
|
{"away"},
|
||||||
@ -983,8 +1003,8 @@ static void chat_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr)
|
|||||||
char timefrmt[TIME_STR_SIZE];
|
char timefrmt[TIME_STR_SIZE];
|
||||||
get_time_str(timefrmt, sizeof(timefrmt));
|
get_time_str(timefrmt, sizeof(timefrmt));
|
||||||
|
|
||||||
line_info_add(self, timefrmt, selfname, NULL, OUT_MSG, 0, 0, "%s", line);
|
int id = line_info_add(self, timefrmt, selfname, NULL, OUT_MSG, 0, 0, "%s", line);
|
||||||
cqueue_add(ctx->cqueue, line, strlen(line), OUT_MSG, ctx->hst->line_end->id + 1);
|
cqueue_add(ctx->cqueue, line, strlen(line), OUT_MSG, id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -41,6 +41,7 @@ void cmd_cancel(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZ
|
|||||||
void cmd_ccur_device(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]);
|
void cmd_ccur_device(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]);
|
||||||
void cmd_mute(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]);
|
void cmd_mute(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]);
|
||||||
void cmd_sense(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]);
|
void cmd_sense(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]);
|
||||||
|
void cmd_bitrate(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]);
|
||||||
#endif /* AUDIO */
|
#endif /* AUDIO */
|
||||||
|
|
||||||
#ifdef VIDEO
|
#ifdef VIDEO
|
||||||
|
@ -33,6 +33,7 @@
|
|||||||
#include "line_info.h"
|
#include "line_info.h"
|
||||||
#include "misc_tools.h"
|
#include "misc_tools.h"
|
||||||
#include "notify.h"
|
#include "notify.h"
|
||||||
|
#include "api.h"
|
||||||
|
|
||||||
struct cmd_func {
|
struct cmd_func {
|
||||||
const char *name;
|
const char *name;
|
||||||
@ -65,8 +66,11 @@ static struct cmd_func global_commands[] = {
|
|||||||
#endif /* AUDIO */
|
#endif /* AUDIO */
|
||||||
#ifdef VIDEO
|
#ifdef VIDEO
|
||||||
{ "/lsvdev", cmd_list_video_devices },
|
{ "/lsvdev", cmd_list_video_devices },
|
||||||
{ "/svdev" , cmd_change_video_device },
|
{ "/svdev", cmd_change_video_device },
|
||||||
#endif /* VIDEO */
|
#endif /* VIDEO */
|
||||||
|
#ifdef PYTHON
|
||||||
|
{ "/run", cmd_run },
|
||||||
|
#endif /* PYTHON */
|
||||||
{ NULL, NULL },
|
{ NULL, NULL },
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -83,6 +87,7 @@ static struct cmd_func chat_commands[] = {
|
|||||||
{ "/hangup", cmd_hangup },
|
{ "/hangup", cmd_hangup },
|
||||||
{ "/mute", cmd_mute },
|
{ "/mute", cmd_mute },
|
||||||
{ "/sense", cmd_sense },
|
{ "/sense", cmd_sense },
|
||||||
|
{ "/bitrate", cmd_bitrate },
|
||||||
#endif /* AUDIO */
|
#endif /* AUDIO */
|
||||||
#ifdef VIDEO
|
#ifdef VIDEO
|
||||||
{ "/video", cmd_video },
|
{ "/video", cmd_video },
|
||||||
@ -193,5 +198,12 @@ void execute(WINDOW *w, ToxWindow *self, Tox *m, const char *input, int mode)
|
|||||||
if (do_command(w, self, m, num_args, global_commands, args) == 0)
|
if (do_command(w, self, m, num_args, global_commands, args) == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
|
||||||
|
if (do_plugin_command(num_args, args) == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invalid command.");
|
line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Invalid command.");
|
||||||
}
|
}
|
||||||
|
@ -153,10 +153,14 @@ static int save_blocklist(char *path)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (Blocked.list[i].active) {
|
if (Blocked.list[i].active) {
|
||||||
|
if (Blocked.list[i].namelength > TOXIC_MAX_NAME_LENGTH) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
BlockedFriend tmp;
|
BlockedFriend tmp;
|
||||||
memset(&tmp, 0, sizeof(BlockedFriend));
|
memset(&tmp, 0, sizeof(BlockedFriend));
|
||||||
tmp.namelength = htons(Blocked.list[i].namelength);
|
tmp.namelength = htons(Blocked.list[i].namelength);
|
||||||
memcpy(tmp.name, Blocked.list[i].name, Blocked.list[i].namelength + 1);
|
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);
|
memcpy(tmp.pub_key, Blocked.list[i].pub_key, TOX_PUBLIC_KEY_SIZE);
|
||||||
|
|
||||||
uint8_t lastonline[sizeof(uint64_t)];
|
uint8_t lastonline[sizeof(uint64_t)];
|
||||||
@ -250,10 +254,15 @@ int load_blocklist(char *path)
|
|||||||
memset(&Blocked.list[i], 0, sizeof(BlockedFriend));
|
memset(&Blocked.list[i], 0, sizeof(BlockedFriend));
|
||||||
|
|
||||||
memcpy(&tmp, data + i * sizeof(BlockedFriend), sizeof(BlockedFriend));
|
memcpy(&tmp, data + i * sizeof(BlockedFriend), sizeof(BlockedFriend));
|
||||||
|
Blocked.list[i].namelength = ntohs(tmp.namelength);
|
||||||
|
|
||||||
|
if (Blocked.list[i].namelength > TOXIC_MAX_NAME_LENGTH) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
Blocked.list[i].active = true;
|
Blocked.list[i].active = true;
|
||||||
Blocked.list[i].num = i;
|
Blocked.list[i].num = i;
|
||||||
Blocked.list[i].namelength = MIN(TOXIC_MAX_NAME_LENGTH, ntohs(tmp.namelength));
|
memcpy(Blocked.list[i].name, tmp.name, Blocked.list[i].namelength + 1); // copy null byte
|
||||||
memcpy(Blocked.list[i].name, tmp.name, Blocked.list[i].namelength + 1);
|
|
||||||
memcpy(Blocked.list[i].pub_key, tmp.pub_key, TOX_PUBLIC_KEY_SIZE);
|
memcpy(Blocked.list[i].pub_key, tmp.pub_key, TOX_PUBLIC_KEY_SIZE);
|
||||||
|
|
||||||
uint8_t lastonline[sizeof(uint64_t)];
|
uint8_t lastonline[sizeof(uint64_t)];
|
||||||
@ -657,7 +666,7 @@ static void draw_del_popup(void)
|
|||||||
wattroff(PendingDelete.popup, A_BOLD);
|
wattroff(PendingDelete.popup, A_BOLD);
|
||||||
wprintw(PendingDelete.popup, "? y/n");
|
wprintw(PendingDelete.popup, "? y/n");
|
||||||
|
|
||||||
wrefresh(PendingDelete.popup);
|
wnoutrefresh(PendingDelete.popup);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* deletes contact from blocked list */
|
/* deletes contact from blocked list */
|
||||||
@ -886,7 +895,7 @@ static void blocklist_onDraw(ToxWindow *self, Tox *m, int y2, int x2)
|
|||||||
wprintw(self->window, "%02X", Blocked.list[selected_num].pub_key[i] & 0xff);
|
wprintw(self->window, "%02X", Blocked.list[selected_num].pub_key[i] & 0xff);
|
||||||
}
|
}
|
||||||
|
|
||||||
wrefresh(self->window);
|
wnoutrefresh(self->window);
|
||||||
draw_del_popup();
|
draw_del_popup();
|
||||||
|
|
||||||
if (self->help->active)
|
if (self->help->active)
|
||||||
@ -1103,7 +1112,7 @@ static void friendlist_onDraw(ToxWindow *self, Tox *m)
|
|||||||
wprintw(self->window, "%02X", Friends.list[selected_num].pub_key[i] & 0xff);
|
wprintw(self->window, "%02X", Friends.list[selected_num].pub_key[i] & 0xff);
|
||||||
}
|
}
|
||||||
|
|
||||||
wrefresh(self->window);
|
wnoutrefresh(self->window);
|
||||||
draw_del_popup();
|
draw_del_popup();
|
||||||
|
|
||||||
if (self->help->active)
|
if (self->help->active)
|
||||||
|
@ -44,7 +44,7 @@ struct GroupChatInvite {
|
|||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
char name[TOXIC_MAX_NAME_LENGTH + 1];
|
char name[TOXIC_MAX_NAME_LENGTH + 1];
|
||||||
int namelength;
|
uint16_t namelength;
|
||||||
char statusmsg[TOX_MAX_STATUS_MESSAGE_LENGTH + 1];
|
char statusmsg[TOX_MAX_STATUS_MESSAGE_LENGTH + 1];
|
||||||
size_t statusmsg_len;
|
size_t statusmsg_len;
|
||||||
char pub_key[TOX_PUBLIC_KEY_SIZE];
|
char pub_key[TOX_PUBLIC_KEY_SIZE];
|
||||||
@ -65,7 +65,7 @@ typedef struct {
|
|||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
char name[TOXIC_MAX_NAME_LENGTH + 1];
|
char name[TOXIC_MAX_NAME_LENGTH + 1];
|
||||||
int namelength;
|
uint16_t namelength;
|
||||||
char pub_key[TOX_PUBLIC_KEY_SIZE];
|
char pub_key[TOX_PUBLIC_KEY_SIZE];
|
||||||
uint32_t num;
|
uint32_t num;
|
||||||
bool active;
|
bool active;
|
||||||
|
@ -655,6 +655,9 @@ void cmd_status(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[
|
|||||||
msg[len] = '\0';
|
msg[len] = '\0';
|
||||||
|
|
||||||
prompt_update_statusmessage(prompt, m, msg);
|
prompt_update_statusmessage(prompt, m, msg);
|
||||||
|
line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Your status has been changed to %s: \"%s\".", status_str, msg);
|
||||||
|
} else {
|
||||||
|
line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "Your status has been changed to %s.", status_str);
|
||||||
}
|
}
|
||||||
|
|
||||||
finish:
|
finish:
|
||||||
|
@ -56,4 +56,8 @@ void cmd_list_video_devices(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)
|
|||||||
void cmd_change_video_device(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]);
|
void cmd_change_video_device(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]);
|
||||||
#endif /* VIDEO */
|
#endif /* VIDEO */
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
void cmd_run(WINDOW *, ToxWindow *, Tox *, int argc, char (*argv)[MAX_STR_SIZE]);
|
||||||
|
#endif
|
||||||
|
|
||||||
#endif /* #define GLOBAL_COMMANDS_H */
|
#endif /* #define GLOBAL_COMMANDS_H */
|
||||||
|
@ -69,8 +69,12 @@ static int max_groupchat_index = 0;
|
|||||||
extern struct user_settings *user_settings;
|
extern struct user_settings *user_settings;
|
||||||
extern struct Winthread Winthread;
|
extern struct Winthread Winthread;
|
||||||
|
|
||||||
#ifdef AUDIO
|
#if defined(AUDIO) && defined(PYTHON)
|
||||||
|
#define AC_NUM_GROUP_COMMANDS 25
|
||||||
|
#elif AUDIO
|
||||||
#define AC_NUM_GROUP_COMMANDS 24
|
#define AC_NUM_GROUP_COMMANDS 24
|
||||||
|
#elif PYTHON
|
||||||
|
#define AC_NUM_GROUP_COMMANDS 21
|
||||||
#else
|
#else
|
||||||
#define AC_NUM_GROUP_COMMANDS 20
|
#define AC_NUM_GROUP_COMMANDS 20
|
||||||
#endif /* AUDIO */
|
#endif /* AUDIO */
|
||||||
@ -97,6 +101,12 @@ static const char group_cmd_list[AC_NUM_GROUP_COMMANDS][MAX_CMDNAME_SIZE] = {
|
|||||||
{ "/requests" },
|
{ "/requests" },
|
||||||
{ "/status" },
|
{ "/status" },
|
||||||
{ "/title" },
|
{ "/title" },
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
|
||||||
|
{ "/run" },
|
||||||
|
|
||||||
|
#endif /* PYTHON */
|
||||||
};
|
};
|
||||||
|
|
||||||
int init_groupchat_win(ToxWindow *prompt, Tox *m, uint32_t groupnum, uint8_t type)
|
int init_groupchat_win(ToxWindow *prompt, Tox *m, uint32_t groupnum, uint8_t type)
|
||||||
@ -255,7 +265,8 @@ static void groupchat_onGroupMessage(ToxWindow *self, Tox *m, uint32_t groupnum,
|
|||||||
write_to_log(msg, nick, ctx->log, false);
|
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,
|
static void groupchat_onGroupTitleChange(ToxWindow *self, Tox *m, uint32_t groupnum, uint32_t peernum,
|
||||||
|
const char *title,
|
||||||
size_t length)
|
size_t length)
|
||||||
{
|
{
|
||||||
ChatContext *ctx = self->chatwin;
|
ChatContext *ctx = self->chatwin;
|
||||||
@ -378,7 +389,7 @@ void *group_add_wait(void *data)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void groupchat_onGroupNamelistChange(ToxWindow *self, Tox *m, uint32_t groupnum, uint32_t peernum,
|
static void groupchat_onGroupNamelistChange(ToxWindow *self, Tox *m, uint32_t groupnum, uint32_t peernum,
|
||||||
TOX_CONFERENCE_STATE_CHANGE change)
|
TOX_CONFERENCE_STATE_CHANGE change)
|
||||||
{
|
{
|
||||||
if (self->num != groupnum)
|
if (self->num != groupnum)
|
||||||
return;
|
return;
|
||||||
@ -543,7 +554,16 @@ static void groupchat_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr)
|
|||||||
TOX_MAX_NAME_LENGTH);
|
TOX_MAX_NAME_LENGTH);
|
||||||
} else if (wcsncmp(ctx->line, L"/avatar \"", wcslen(L"/avatar \"")) == 0) {
|
} else if (wcsncmp(ctx->line, L"/avatar \"", wcslen(L"/avatar \"")) == 0) {
|
||||||
diff = dir_match(self, m, ctx->line, L"/avatar");
|
diff = dir_match(self, m, ctx->line, L"/avatar");
|
||||||
} else {
|
}
|
||||||
|
|
||||||
|
#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);
|
diff = complete_line(self, group_cmd_list, AC_NUM_GROUP_COMMANDS, MAX_CMDNAME_SIZE);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -673,7 +693,7 @@ static void groupchat_onDraw(ToxWindow *self, Tox *m)
|
|||||||
int new_x = ctx->start ? x2 - 1 : MAX(0, wcswidth(ctx->line, ctx->pos));
|
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 + 1, new_x);
|
||||||
|
|
||||||
wrefresh(self->window);
|
wnoutrefresh(self->window);
|
||||||
|
|
||||||
if (self->help->active)
|
if (self->help->active)
|
||||||
help_onDraw(self);
|
help_onDraw(self);
|
||||||
|
90
src/help.c
90
src/help.c
@ -27,7 +27,15 @@
|
|||||||
#include "help.h"
|
#include "help.h"
|
||||||
#include "misc_tools.h"
|
#include "misc_tools.h"
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
#include "api.h"
|
||||||
|
#endif /* PYTHON */
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
#define HELP_MENU_HEIGHT 10
|
||||||
|
#else
|
||||||
#define HELP_MENU_HEIGHT 9
|
#define HELP_MENU_HEIGHT 9
|
||||||
|
#endif /* PYTHON */
|
||||||
#define HELP_MENU_WIDTH 26
|
#define HELP_MENU_WIDTH 26
|
||||||
|
|
||||||
void help_init_menu(ToxWindow *self)
|
void help_init_menu(ToxWindow *self)
|
||||||
@ -95,6 +103,13 @@ static void help_draw_menu(ToxWindow *self)
|
|||||||
wattroff(win, A_BOLD | COLOR_PAIR(BLUE));
|
wattroff(win, A_BOLD | COLOR_PAIR(BLUE));
|
||||||
wprintw(win, "oup commands\n");
|
wprintw(win, "oup commands\n");
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
wattron(win, A_BOLD | COLOR_PAIR(BLUE));
|
||||||
|
wprintw(win, " p");
|
||||||
|
wattroff(win, A_BOLD | COLOR_PAIR(BLUE));
|
||||||
|
wprintw(win, "lugin commands\n");
|
||||||
|
#endif /* PYTHON */
|
||||||
|
|
||||||
wattron(win, A_BOLD | COLOR_PAIR(BLUE));
|
wattron(win, A_BOLD | COLOR_PAIR(BLUE));
|
||||||
wprintw(win, " f");
|
wprintw(win, " f");
|
||||||
wattroff(win, A_BOLD | COLOR_PAIR(BLUE));
|
wattroff(win, A_BOLD | COLOR_PAIR(BLUE));
|
||||||
@ -112,7 +127,7 @@ static void help_draw_menu(ToxWindow *self)
|
|||||||
wprintw(win, "it menu\n");
|
wprintw(win, "it menu\n");
|
||||||
|
|
||||||
box(win, ACS_VLINE, ACS_HLINE);
|
box(win, ACS_VLINE, ACS_HLINE);
|
||||||
wrefresh(win);
|
wnoutrefresh(win);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void help_draw_bottom_menu(WINDOW *win)
|
static void help_draw_bottom_menu(WINDOW *win)
|
||||||
@ -185,10 +200,18 @@ static void help_draw_global(ToxWindow *self)
|
|||||||
wprintw(win, " /svdev <type> <id> : Set active video device\n");
|
wprintw(win, " /svdev <type> <id> : Set active video device\n");
|
||||||
#endif /* VIDEO */
|
#endif /* VIDEO */
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
wattron(win, A_BOLD);
|
||||||
|
wprintw(win, "\n Scripting:\n");
|
||||||
|
wattroff(win, A_BOLD);
|
||||||
|
|
||||||
|
wprintw(win, " /run <path> : Load and run the script at path\n");
|
||||||
|
#endif /* PYTHON */
|
||||||
|
|
||||||
help_draw_bottom_menu(win);
|
help_draw_bottom_menu(win);
|
||||||
|
|
||||||
box(win, ACS_VLINE, ACS_HLINE);
|
box(win, ACS_VLINE, ACS_HLINE);
|
||||||
wrefresh(win);
|
wnoutrefresh(win);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void help_draw_chat(ToxWindow *self)
|
static void help_draw_chat(ToxWindow *self)
|
||||||
@ -219,6 +242,7 @@ static void help_draw_chat(ToxWindow *self)
|
|||||||
wprintw(win, " /sdev <type> <id> : Change active device\n");
|
wprintw(win, " /sdev <type> <id> : Change active device\n");
|
||||||
wprintw(win, " /mute <type> : Mute active device if in call\n");
|
wprintw(win, " /mute <type> : Mute active device if in call\n");
|
||||||
wprintw(win, " /sense <n> : VAD sensitivity threshold\n");
|
wprintw(win, " /sense <n> : VAD sensitivity threshold\n");
|
||||||
|
wprintw(win, " /bitrate <n> : Set the audio encoding bitrate\n");
|
||||||
#endif /* AUDIO */
|
#endif /* AUDIO */
|
||||||
|
|
||||||
#ifdef VIDEO
|
#ifdef VIDEO
|
||||||
@ -231,7 +255,7 @@ static void help_draw_chat(ToxWindow *self)
|
|||||||
help_draw_bottom_menu(win);
|
help_draw_bottom_menu(win);
|
||||||
|
|
||||||
box(win, ACS_VLINE, ACS_HLINE);
|
box(win, ACS_VLINE, ACS_HLINE);
|
||||||
wrefresh(win);
|
wnoutrefresh(win);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void help_draw_keys(ToxWindow *self)
|
static void help_draw_keys(ToxWindow *self)
|
||||||
@ -257,7 +281,7 @@ static void help_draw_keys(ToxWindow *self)
|
|||||||
help_draw_bottom_menu(win);
|
help_draw_bottom_menu(win);
|
||||||
|
|
||||||
box(win, ACS_VLINE, ACS_HLINE);
|
box(win, ACS_VLINE, ACS_HLINE);
|
||||||
wrefresh(win);
|
wnoutrefresh(win);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void help_draw_group(ToxWindow *self)
|
static void help_draw_group(ToxWindow *self)
|
||||||
@ -275,9 +299,29 @@ static void help_draw_group(ToxWindow *self)
|
|||||||
help_draw_bottom_menu(win);
|
help_draw_bottom_menu(win);
|
||||||
|
|
||||||
box(win, ACS_VLINE, ACS_HLINE);
|
box(win, ACS_VLINE, ACS_HLINE);
|
||||||
wrefresh(win);
|
wnoutrefresh(win);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
static void help_draw_plugin(ToxWindow *self)
|
||||||
|
{
|
||||||
|
WINDOW *win = self->help->win;
|
||||||
|
|
||||||
|
wmove(win, 1, 1);
|
||||||
|
|
||||||
|
wattron(win, A_BOLD | COLOR_PAIR(RED));
|
||||||
|
wprintw(win, "Plugin commands:\n");
|
||||||
|
wattroff(win, A_BOLD | COLOR_PAIR(RED));
|
||||||
|
|
||||||
|
draw_handler_help(win);
|
||||||
|
|
||||||
|
help_draw_bottom_menu(win);
|
||||||
|
|
||||||
|
box(win, ACS_VLINE, ACS_HLINE);
|
||||||
|
wnoutrefresh(win);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
static void help_draw_contacts(ToxWindow *self)
|
static void help_draw_contacts(ToxWindow *self)
|
||||||
{
|
{
|
||||||
WINDOW *win = self->help->win;
|
WINDOW *win = self->help->win;
|
||||||
@ -297,11 +341,13 @@ static void help_draw_contacts(ToxWindow *self)
|
|||||||
help_draw_bottom_menu(win);
|
help_draw_bottom_menu(win);
|
||||||
|
|
||||||
box(win, ACS_VLINE, ACS_HLINE);
|
box(win, ACS_VLINE, ACS_HLINE);
|
||||||
wrefresh(win);
|
wnoutrefresh(win);
|
||||||
}
|
}
|
||||||
|
|
||||||
void help_onKey(ToxWindow *self, wint_t key)
|
void help_onKey(ToxWindow *self, wint_t key)
|
||||||
{
|
{
|
||||||
|
int height;
|
||||||
|
|
||||||
switch (key) {
|
switch (key) {
|
||||||
case 'x':
|
case 'x':
|
||||||
case T_KEY_ESC:
|
case T_KEY_ESC:
|
||||||
@ -310,9 +356,9 @@ void help_onKey(ToxWindow *self, wint_t key)
|
|||||||
|
|
||||||
case 'c':
|
case 'c':
|
||||||
#ifdef VIDEO
|
#ifdef VIDEO
|
||||||
help_init_window(self, 22, 80);
|
help_init_window(self, 23, 80);
|
||||||
#elif AUDIO
|
#elif AUDIO
|
||||||
help_init_window(self, 19, 80);
|
help_init_window(self, 20, 80);
|
||||||
#else
|
#else
|
||||||
help_init_window(self, 10, 80);
|
help_init_window(self, 10, 80);
|
||||||
#endif
|
#endif
|
||||||
@ -320,13 +366,16 @@ void help_onKey(ToxWindow *self, wint_t key)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case 'g':
|
case 'g':
|
||||||
|
height = 22;
|
||||||
#ifdef VIDEO
|
#ifdef VIDEO
|
||||||
help_init_window(self, 30, 80);
|
height += 8;
|
||||||
#elif AUDIO
|
#elif AUDIO
|
||||||
help_init_window(self, 26, 80);
|
height += 4;
|
||||||
#else
|
|
||||||
help_init_window(self, 22, 80);
|
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef PYTHON
|
||||||
|
height += 2;
|
||||||
|
#endif
|
||||||
|
help_init_window(self, height, 80);
|
||||||
self->help->type = HELP_GLOBAL;
|
self->help->type = HELP_GLOBAL;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -335,6 +384,14 @@ void help_onKey(ToxWindow *self, wint_t key)
|
|||||||
self->help->type = HELP_GROUP;
|
self->help->type = HELP_GROUP;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
|
||||||
|
case 'p':
|
||||||
|
help_init_window(self, 4 + num_registered_handlers(), help_max_width());
|
||||||
|
self->help->type = HELP_PLUGIN;
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
|
||||||
case 'f':
|
case 'f':
|
||||||
help_init_window(self, 10, 80);
|
help_init_window(self, 10, 80);
|
||||||
self->help->type = HELP_CONTACTS;
|
self->help->type = HELP_CONTACTS;
|
||||||
@ -354,8 +411,6 @@ void help_onKey(ToxWindow *self, wint_t key)
|
|||||||
|
|
||||||
void help_onDraw(ToxWindow *self)
|
void help_onDraw(ToxWindow *self)
|
||||||
{
|
{
|
||||||
curs_set(0);
|
|
||||||
|
|
||||||
switch (self->help->type) {
|
switch (self->help->type) {
|
||||||
case HELP_MENU:
|
case HELP_MENU:
|
||||||
help_draw_menu(self);
|
help_draw_menu(self);
|
||||||
@ -380,5 +435,12 @@ void help_onDraw(ToxWindow *self)
|
|||||||
case HELP_GROUP:
|
case HELP_GROUP:
|
||||||
help_draw_group(self);
|
help_draw_group(self);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
|
||||||
|
case HELP_PLUGIN:
|
||||||
|
help_draw_plugin(self);
|
||||||
|
break;
|
||||||
|
#endif /* PYTHON */
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -33,6 +33,9 @@ typedef enum {
|
|||||||
HELP_GROUP,
|
HELP_GROUP,
|
||||||
HELP_KEYS,
|
HELP_KEYS,
|
||||||
HELP_CONTACTS,
|
HELP_CONTACTS,
|
||||||
|
#ifdef PYTHON
|
||||||
|
HELP_PLUGIN,
|
||||||
|
#endif
|
||||||
} HELP_TYPES;
|
} HELP_TYPES;
|
||||||
|
|
||||||
void help_onDraw(ToxWindow *self);
|
void help_onDraw(ToxWindow *self);
|
||||||
|
@ -129,17 +129,21 @@ static struct line_info *line_info_ret_queue(struct history *hst)
|
|||||||
return line;
|
return line;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* creates new line_info line and puts it in the queue. */
|
/* creates new line_info line and puts it in the queue.
|
||||||
void line_info_add(ToxWindow *self, const char *timestr, const char *name1, const char *name2, uint8_t type,
|
*
|
||||||
uint8_t bold, uint8_t colour, const char *msg, ...)
|
* 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,
|
||||||
|
uint8_t bold, uint8_t colour, const char *msg, ...)
|
||||||
{
|
{
|
||||||
if (!self)
|
if (!self)
|
||||||
return;
|
return -1;
|
||||||
|
|
||||||
struct history *hst = self->chatwin->hst;
|
struct history *hst = self->chatwin->hst;
|
||||||
|
|
||||||
if (hst->queue_sz >= MAX_LINE_INFO_QUEUE)
|
if (hst->queue_sz >= MAX_LINE_INFO_QUEUE)
|
||||||
return;
|
return -1;
|
||||||
|
|
||||||
struct line_info *new_line = calloc(1, sizeof(struct line_info));
|
struct line_info *new_line = calloc(1, sizeof(struct line_info));
|
||||||
|
|
||||||
@ -222,6 +226,7 @@ void line_info_add(ToxWindow *self, const char *timestr, const char *name1, cons
|
|||||||
len += strlen(new_line->name2);
|
len += strlen(new_line->name2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
new_line->id = (hst->line_end->id + 1 + hst->queue_sz) % INT_MAX;
|
||||||
new_line->len = len;
|
new_line->len = len;
|
||||||
new_line->type = type;
|
new_line->type = type;
|
||||||
new_line->bold = bold;
|
new_line->bold = bold;
|
||||||
@ -230,6 +235,8 @@ void line_info_add(ToxWindow *self, const char *timestr, const char *name1, cons
|
|||||||
new_line->timestamp = get_unix_time();
|
new_line->timestamp = get_unix_time();
|
||||||
|
|
||||||
hst->queue[hst->queue_sz++] = new_line;
|
hst->queue[hst->queue_sz++] = new_line;
|
||||||
|
|
||||||
|
return new_line->id;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* adds a single queue item to hst if possible. only called once per call to line_info_print() */
|
/* adds a single queue item to hst if possible. only called once per call to line_info_print() */
|
||||||
@ -244,10 +251,10 @@ static void line_info_check_queue(ToxWindow *self)
|
|||||||
if (hst->start_id > user_settings->history_size)
|
if (hst->start_id > user_settings->history_size)
|
||||||
line_info_root_fwd(hst);
|
line_info_root_fwd(hst);
|
||||||
|
|
||||||
line->id = hst->line_end->id + 1;
|
|
||||||
line->prev = hst->line_end;
|
line->prev = hst->line_end;
|
||||||
hst->line_end->next = line;
|
hst->line_end->next = line;
|
||||||
hst->line_end = line;
|
hst->line_end = line;
|
||||||
|
hst->line_end->id = line->id;
|
||||||
|
|
||||||
int y, y2, x, x2;
|
int y, y2, x, x2;
|
||||||
getmaxyx(self->window, y2, x2);
|
getmaxyx(self->window, y2, x2);
|
||||||
|
@ -74,9 +74,13 @@ struct history {
|
|||||||
int queue_sz;
|
int queue_sz;
|
||||||
};
|
};
|
||||||
|
|
||||||
/* creates new line_info line and puts it in the queue. */
|
/* creates new line_info line and puts it in the queue.
|
||||||
void line_info_add(ToxWindow *self, const char *timestr, const char *name1, const char *name2, uint8_t type,
|
*
|
||||||
uint8_t bold, uint8_t colour, const char *msg, ...);
|
* 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,
|
||||||
|
uint8_t bold, uint8_t colour, const char *msg, ...);
|
||||||
|
|
||||||
/* Prints a section of history starting at line_start */
|
/* Prints a section of history starting at line_start */
|
||||||
void line_info_print(ToxWindow *self);
|
void line_info_print(ToxWindow *self);
|
||||||
|
@ -42,8 +42,12 @@ void cqueue_cleanup(struct chat_queue *q)
|
|||||||
free(q);
|
free(q);
|
||||||
}
|
}
|
||||||
|
|
||||||
void cqueue_add(struct chat_queue *q, const char *msg, size_t len, uint8_t type, uint32_t line_id)
|
void cqueue_add(struct chat_queue *q, const char *msg, size_t len, uint8_t type, int line_id)
|
||||||
{
|
{
|
||||||
|
if (line_id < 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
struct cqueue_msg *new_m = malloc(sizeof(struct cqueue_msg));
|
struct cqueue_msg *new_m = malloc(sizeof(struct cqueue_msg));
|
||||||
|
|
||||||
if (new_m == NULL)
|
if (new_m == NULL)
|
||||||
|
@ -40,7 +40,7 @@ struct chat_queue {
|
|||||||
};
|
};
|
||||||
|
|
||||||
void cqueue_cleanup(struct chat_queue *q);
|
void cqueue_cleanup(struct chat_queue *q);
|
||||||
void cqueue_add(struct chat_queue *q, const char *msg, size_t len, uint8_t type, uint32_t line_id);
|
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 the oldest unsent message in queue. */
|
||||||
void cqueue_try_send(ToxWindow *self, Tox *m);
|
void cqueue_try_send(ToxWindow *self, Tox *m);
|
||||||
|
@ -26,12 +26,11 @@
|
|||||||
#include <time.h>
|
#include <time.h>
|
||||||
#include <limits.h>
|
#include <limits.h>
|
||||||
#include <dirent.h>
|
#include <dirent.h>
|
||||||
#if defined(__FreeBSD__)
|
#if SYSTEM == BSD
|
||||||
#include <netinet/in.h>
|
#include <netinet/in.h>
|
||||||
#include <sys/socket.h>
|
#include <sys/socket.h>
|
||||||
#else
|
#endif /* BSD! */
|
||||||
#include <arpa/inet.h>
|
#include <arpa/inet.h>
|
||||||
#endif
|
|
||||||
#include <sys/stat.h>
|
#include <sys/stat.h>
|
||||||
|
|
||||||
#include "toxic.h"
|
#include "toxic.h"
|
||||||
@ -372,14 +371,22 @@ on_error:
|
|||||||
return len;
|
return len;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* copies data to msg buffer.
|
/* copies data to msg buffer, removing return characters.
|
||||||
returns length of msg, which will be no larger than size-1 */
|
returns length of msg, which will be no larger than size-1 */
|
||||||
size_t copy_tox_str(char *msg, size_t size, const char *data, size_t length)
|
size_t copy_tox_str(char *msg, size_t size, const char *data, size_t length)
|
||||||
{
|
{
|
||||||
size_t len = MIN(length, size - 1);
|
size_t i;
|
||||||
memcpy(msg, data, len);
|
size_t j = 0;
|
||||||
msg[len] = '\0';
|
|
||||||
return len;
|
for (i = 0; (i < length) && (j < size - 1); ++i) {
|
||||||
|
if (data[i] != '\r') {
|
||||||
|
msg[j++] = data[i];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
msg[j] = '\0';
|
||||||
|
|
||||||
|
return j;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* returns index of the first instance of ch in s starting at idx.
|
/* returns index of the first instance of ch in s starting at idx.
|
||||||
|
@ -21,6 +21,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
#include <stdarg.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <curl/curl.h>
|
#include <curl/curl.h>
|
||||||
|
|
||||||
|
@ -208,10 +208,10 @@ void bgrxtoyuv420(uint8_t *plane_y, uint8_t *plane_u, uint8_t *plane_v, uint8_t
|
|||||||
// TODO possibly get a better pixel format
|
// TODO possibly get a better pixel format
|
||||||
if (_shouldMangleDimensions) {
|
if (_shouldMangleDimensions) {
|
||||||
[_linkerVideo setVideoSettings: @ {
|
[_linkerVideo setVideoSettings: @ {
|
||||||
(id)kCVPixelBufferPixelFormatTypeKey: @(kCVPixelFormatType_32BGRA),
|
(id)kCVPixelBufferPixelFormatTypeKey: @(kCVPixelFormatType_32BGRA),
|
||||||
(id)kCVPixelBufferWidthKey: @640,
|
(id)kCVPixelBufferWidthKey: @640,
|
||||||
(id)kCVPixelBufferHeightKey: @480
|
(id)kCVPixelBufferHeightKey: @480
|
||||||
}];
|
}];
|
||||||
} else {
|
} else {
|
||||||
[_linkerVideo setVideoSettings: @ {(id)kCVPixelBufferPixelFormatTypeKey: @(kCVPixelFormatType_32BGRA)}];
|
[_linkerVideo setVideoSettings: @ {(id)kCVPixelBufferPixelFormatTypeKey: @(kCVPixelFormatType_32BGRA)}];
|
||||||
}
|
}
|
||||||
|
23
src/prompt.c
23
src/prompt.c
@ -49,10 +49,16 @@ extern struct Winthread Winthread;
|
|||||||
|
|
||||||
extern FriendsList Friends;
|
extern FriendsList Friends;
|
||||||
FriendRequests FrndRequests;
|
FriendRequests FrndRequests;
|
||||||
#ifdef VIDEO
|
#if defined(PYTHON) && defined(VIDEO)
|
||||||
|
#define AC_NUM_GLOB_COMMANDS 23
|
||||||
|
#elif defined(PYTHON) && defined(AUDIO)
|
||||||
|
#define AC_NUM_GLOB_COMMANDS 21
|
||||||
|
#elif VIDEO
|
||||||
#define AC_NUM_GLOB_COMMANDS 22
|
#define AC_NUM_GLOB_COMMANDS 22
|
||||||
#elif AUDIO
|
#elif AUDIO
|
||||||
#define AC_NUM_GLOB_COMMANDS 20
|
#define AC_NUM_GLOB_COMMANDS 20
|
||||||
|
#elif PYTHON
|
||||||
|
#define AC_NUM_GLOB_COMMANDS 19
|
||||||
#else
|
#else
|
||||||
#define AC_NUM_GLOB_COMMANDS 18
|
#define AC_NUM_GLOB_COMMANDS 18
|
||||||
#endif
|
#endif
|
||||||
@ -92,6 +98,12 @@ static const char glob_cmd_list[AC_NUM_GLOB_COMMANDS][MAX_CMDNAME_SIZE] = {
|
|||||||
|
|
||||||
#endif /* VIDEO */
|
#endif /* VIDEO */
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
|
||||||
|
{ "/run" },
|
||||||
|
|
||||||
|
#endif /* PYTHON */
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
void kill_prompt_window(ToxWindow *self)
|
void kill_prompt_window(ToxWindow *self)
|
||||||
@ -214,6 +226,13 @@ static void prompt_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr)
|
|||||||
|
|
||||||
if (wcsncmp(ctx->line, L"/avatar \"", wcslen(L"/avatar \"")) == 0)
|
if (wcsncmp(ctx->line, L"/avatar \"", wcslen(L"/avatar \"")) == 0)
|
||||||
diff = dir_match(self, m, ctx->line, L"/avatar");
|
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 if (wcsncmp(ctx->line, L"/status ", wcslen(L"/status ")) == 0) {
|
else if (wcsncmp(ctx->line, L"/status ", wcslen(L"/status ")) == 0) {
|
||||||
const char status_cmd_list[3][8] = {
|
const char status_cmd_list[3][8] = {
|
||||||
{"online"},
|
{"online"},
|
||||||
@ -376,7 +395,7 @@ static void prompt_onDraw(ToxWindow *self, Tox *m)
|
|||||||
help_onDraw(self);
|
help_onDraw(self);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void prompt_onConnectionChange(ToxWindow *self, Tox *m, uint32_t friendnum , TOX_CONNECTION connection_status)
|
static void prompt_onConnectionChange(ToxWindow *self, Tox *m, uint32_t friendnum, TOX_CONNECTION connection_status)
|
||||||
{
|
{
|
||||||
ChatContext *ctx = self->chatwin;
|
ChatContext *ctx = self->chatwin;
|
||||||
|
|
||||||
|
347
src/python_api.c
Normal file
347
src/python_api.c
Normal file
@ -0,0 +1,347 @@
|
|||||||
|
/* python_api.c
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* Copyright (C) 2017 Jakob Kreuze <jakob@memeware.net>
|
||||||
|
*
|
||||||
|
* 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/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
#include <Python.h>
|
||||||
|
#include "api.h"
|
||||||
|
#endif /* PYTHON */
|
||||||
|
|
||||||
|
#include "execute.h"
|
||||||
|
|
||||||
|
extern Tox *user_tox;
|
||||||
|
|
||||||
|
static struct python_registered_func {
|
||||||
|
char *name;
|
||||||
|
char *help;
|
||||||
|
PyObject *callback;
|
||||||
|
struct python_registered_func *next;
|
||||||
|
} python_commands = {0};
|
||||||
|
|
||||||
|
static PyObject *python_api_display(PyObject *self, PyObject *args)
|
||||||
|
{
|
||||||
|
const char *msg;
|
||||||
|
|
||||||
|
if (!PyArg_ParseTuple(args, "s", &msg))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
api_display(msg);
|
||||||
|
return Py_None;
|
||||||
|
}
|
||||||
|
|
||||||
|
static PyObject *python_api_get_nick(PyObject *self, PyObject *args)
|
||||||
|
{
|
||||||
|
char *name;
|
||||||
|
PyObject *ret;
|
||||||
|
|
||||||
|
if (!PyArg_ParseTuple(args, ""))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
name = api_get_nick();
|
||||||
|
|
||||||
|
if (name == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
ret = Py_BuildValue("s", name);
|
||||||
|
free(name);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static PyObject *python_api_get_status(PyObject *self, PyObject *args)
|
||||||
|
{
|
||||||
|
PyObject *ret;
|
||||||
|
|
||||||
|
if (!PyArg_ParseTuple(args, ""))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
switch (api_get_status()) {
|
||||||
|
case TOX_USER_STATUS_NONE:
|
||||||
|
ret = Py_BuildValue("s", "online");
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TOX_USER_STATUS_AWAY:
|
||||||
|
ret = Py_BuildValue("s", "away");
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TOX_USER_STATUS_BUSY:
|
||||||
|
ret = Py_BuildValue("s", "busy");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static PyObject *python_api_get_status_message(PyObject *self, PyObject *args)
|
||||||
|
{
|
||||||
|
char *status;
|
||||||
|
PyObject *ret;
|
||||||
|
|
||||||
|
if (!PyArg_ParseTuple(args, ""))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
status = api_get_status_message();
|
||||||
|
|
||||||
|
if (status == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
ret = Py_BuildValue("s", status);
|
||||||
|
free(status);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
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, ""))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
friends = api_get_friendslist();
|
||||||
|
ret = PyList_New(0);
|
||||||
|
|
||||||
|
for (i = 0; i < friends.num_friends; i++) {
|
||||||
|
for (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);
|
||||||
|
PyList_Append(ret, cur);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static PyObject *python_api_send(PyObject *self, PyObject *args)
|
||||||
|
{
|
||||||
|
const char *msg;
|
||||||
|
|
||||||
|
if (!PyArg_ParseTuple(args, "s", &msg))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
api_send(msg);
|
||||||
|
return Py_None;
|
||||||
|
}
|
||||||
|
|
||||||
|
static PyObject *python_api_execute(PyObject *self, PyObject *args)
|
||||||
|
{
|
||||||
|
int mode;
|
||||||
|
const char *command;
|
||||||
|
|
||||||
|
if (!PyArg_ParseTuple(args, "si", &command, &mode))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
api_execute(command, mode);
|
||||||
|
return Py_None;
|
||||||
|
}
|
||||||
|
|
||||||
|
static PyObject *python_api_register(PyObject *self, PyObject *args)
|
||||||
|
{
|
||||||
|
struct python_registered_func *cur;
|
||||||
|
size_t command_len, help_len;
|
||||||
|
const char *command, *help;
|
||||||
|
PyObject *callback;
|
||||||
|
|
||||||
|
if (!PyArg_ParseTuple(args, "ssO:register_command", &command, &help, &callback))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
if (!PyCallable_Check(callback)) {
|
||||||
|
PyErr_SetString(PyExc_TypeError, "Calback parameter must be callable");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (command[0] != '/') {
|
||||||
|
PyErr_SetString(PyExc_TypeError, "Command must be prefixed with a '/'");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (cur = &python_commands; ; cur = cur->next) {
|
||||||
|
if (cur->name != NULL && !strcmp(command, cur->name)) {
|
||||||
|
Py_XDECREF(cur->callback);
|
||||||
|
Py_XINCREF(callback);
|
||||||
|
cur->callback = callback;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cur->next == NULL) {
|
||||||
|
Py_XINCREF(callback);
|
||||||
|
cur->next = malloc(sizeof(struct python_registered_func));
|
||||||
|
|
||||||
|
if (cur->next == NULL)
|
||||||
|
return PyErr_NoMemory();
|
||||||
|
|
||||||
|
command_len = strlen(command);
|
||||||
|
cur->next->name = malloc(command_len + 1);
|
||||||
|
|
||||||
|
if (cur->next->name == NULL)
|
||||||
|
return PyErr_NoMemory();
|
||||||
|
|
||||||
|
strncpy(cur->next->name, command, command_len + 1);
|
||||||
|
help_len = strlen(help);
|
||||||
|
cur->next->help = malloc(help_len + 1);
|
||||||
|
|
||||||
|
if (cur->next->help == NULL)
|
||||||
|
return PyErr_NoMemory();
|
||||||
|
|
||||||
|
strncpy(cur->next->help, help, help_len + 1);
|
||||||
|
cur->next->callback = callback;
|
||||||
|
cur->next->next = NULL;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Py_INCREF(Py_None);
|
||||||
|
return Py_None;
|
||||||
|
}
|
||||||
|
|
||||||
|
static PyMethodDef ToxicApiMethods[] = {
|
||||||
|
{"display", python_api_display, METH_VARARGS, "Display a message to the current prompt"},
|
||||||
|
{"get_nick", python_api_get_nick, METH_VARARGS, "Return the user's current nickname"},
|
||||||
|
{"get_status", python_api_get_status, METH_VARARGS, "Returns the user's current status"},
|
||||||
|
{"get_status_message", python_api_get_status_message, METH_VARARGS, "Return the user's current status message"},
|
||||||
|
{"get_all_friends", python_api_get_all_friends, METH_VARARGS, "Return all of the user's friends"},
|
||||||
|
{"send", python_api_send, METH_VARARGS, "Send the message to the current user"},
|
||||||
|
{"execute", python_api_execute, METH_VARARGS, "Execute a command like `/nick`"},
|
||||||
|
{"register", python_api_register, METH_VARARGS, "Register a command like `/nick` to a Python function"},
|
||||||
|
{NULL, NULL, 0, NULL},
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct PyModuleDef toxic_api_module = {
|
||||||
|
PyModuleDef_HEAD_INIT,
|
||||||
|
"toxic_api",
|
||||||
|
NULL,
|
||||||
|
-1,
|
||||||
|
ToxicApiMethods
|
||||||
|
};
|
||||||
|
|
||||||
|
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_SetAttrString(m, "GLOBAL_COMMAND", global_command_const);
|
||||||
|
PyObject_SetAttrString(m, "CHAT_COMMAND", chat_command_const);
|
||||||
|
PyObject_SetAttrString(m, "GROUPCHAT_COMMAND", groupchat_command_const);
|
||||||
|
Py_DECREF(global_command_const);
|
||||||
|
Py_DECREF(chat_command_const);
|
||||||
|
Py_DECREF(groupchat_command_const);
|
||||||
|
return m;
|
||||||
|
}
|
||||||
|
|
||||||
|
void terminate_python(void)
|
||||||
|
{
|
||||||
|
struct python_registered_func *cur, *old;
|
||||||
|
|
||||||
|
if (python_commands.name != NULL)
|
||||||
|
free(python_commands.name);
|
||||||
|
|
||||||
|
for (cur = python_commands.next; cur != NULL;) {
|
||||||
|
old = cur;
|
||||||
|
cur = cur->next;
|
||||||
|
free(old->name);
|
||||||
|
free(old);
|
||||||
|
}
|
||||||
|
|
||||||
|
Py_Finalize();
|
||||||
|
}
|
||||||
|
|
||||||
|
void init_python(Tox *m)
|
||||||
|
{
|
||||||
|
user_tox = m;
|
||||||
|
PyImport_AppendInittab("toxic_api", PyInit_toxic_api);
|
||||||
|
Py_Initialize();
|
||||||
|
}
|
||||||
|
|
||||||
|
void run_python(FILE *fp, char *path)
|
||||||
|
{
|
||||||
|
PyRun_SimpleFile(fp, path);
|
||||||
|
}
|
||||||
|
|
||||||
|
int do_python_command(int num_args, char (*args)[MAX_STR_SIZE])
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
PyObject *callback_args, *args_strings;
|
||||||
|
struct python_registered_func *cur;
|
||||||
|
|
||||||
|
for (cur = &python_commands; cur != NULL; cur = cur->next) {
|
||||||
|
if (cur->name == NULL)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (!strcmp(args[0], cur->name)) {
|
||||||
|
args_strings = PyList_New(0);
|
||||||
|
|
||||||
|
for (i = 1; i < num_args; i++)
|
||||||
|
PyList_Append(args_strings, Py_BuildValue("s", args[i]));
|
||||||
|
|
||||||
|
callback_args = PyTuple_Pack(1, args_strings);
|
||||||
|
|
||||||
|
if (PyObject_CallObject(cur->callback, callback_args) == NULL)
|
||||||
|
api_display("Exception raised in callback function");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int python_num_registered_handlers(void)
|
||||||
|
{
|
||||||
|
int n = 0;
|
||||||
|
struct python_registered_func *cur;
|
||||||
|
|
||||||
|
for (cur = &python_commands; cur != NULL; cur = cur->next) {
|
||||||
|
if (cur->name != NULL)
|
||||||
|
n++;
|
||||||
|
}
|
||||||
|
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
int python_help_max_width(void)
|
||||||
|
{
|
||||||
|
size_t tmp;
|
||||||
|
int max = 0;
|
||||||
|
struct python_registered_func *cur;
|
||||||
|
|
||||||
|
for (cur = &python_commands; cur != NULL; cur = cur->next) {
|
||||||
|
if (cur->name != NULL) {
|
||||||
|
tmp = strlen(cur->help);
|
||||||
|
max = tmp > max ? tmp : max;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
max = max > 50 ? 50 : max;
|
||||||
|
return 37 + max;
|
||||||
|
}
|
||||||
|
|
||||||
|
void python_draw_handler_help(WINDOW *win)
|
||||||
|
{
|
||||||
|
struct python_registered_func *cur;
|
||||||
|
|
||||||
|
for (cur = &python_commands; cur != NULL; cur = cur->next) {
|
||||||
|
if (cur->name != NULL)
|
||||||
|
wprintw(win, " %-29s: %.50s\n", cur->name, cur->help);
|
||||||
|
}
|
||||||
|
}
|
39
src/python_api.h
Normal file
39
src/python_api.h
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
/* python_api.h
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* Copyright (C) 2017 Jakob Kreuze <jakob@memeware.net>
|
||||||
|
*
|
||||||
|
* 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 PYTHON_API_H
|
||||||
|
#define PYTHON_API_H
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
#include <Python.h>
|
||||||
|
#endif /* PYTHON */
|
||||||
|
|
||||||
|
PyMODINIT_FUNC PyInit_toxic_api(void);
|
||||||
|
void terminate_python(void);
|
||||||
|
void init_python(Tox *m);
|
||||||
|
void run_python(FILE *fp, char *path);
|
||||||
|
int do_python_command(int num_args, char (*args)[MAX_STR_SIZE]);
|
||||||
|
int python_num_registered_handlers(void);
|
||||||
|
int python_help_max_width(void);
|
||||||
|
void python_draw_handler_help(WINDOW *win);
|
||||||
|
|
||||||
|
#endif /* #define PYTHON_API_H */
|
@ -179,12 +179,14 @@ static const struct tox_strings {
|
|||||||
const char *download_path;
|
const char *download_path;
|
||||||
const char *chatlogs_path;
|
const char *chatlogs_path;
|
||||||
const char *avatar_path;
|
const char *avatar_path;
|
||||||
|
const char *autorun_path;
|
||||||
const char *password_eval;
|
const char *password_eval;
|
||||||
} tox_strings = {
|
} tox_strings = {
|
||||||
"tox",
|
"tox",
|
||||||
"download_path",
|
"download_path",
|
||||||
"chatlogs_path",
|
"chatlogs_path",
|
||||||
"avatar_path",
|
"avatar_path",
|
||||||
|
"autorun_path",
|
||||||
"password_eval",
|
"password_eval",
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -193,6 +195,7 @@ static void tox_defaults(struct user_settings *settings)
|
|||||||
strcpy(settings->download_path, "");
|
strcpy(settings->download_path, "");
|
||||||
strcpy(settings->chatlogs_path, "");
|
strcpy(settings->chatlogs_path, "");
|
||||||
strcpy(settings->avatar_path, "");
|
strcpy(settings->avatar_path, "");
|
||||||
|
strcpy(settings->autorun_path, "");
|
||||||
strcpy(settings->password_eval, "");
|
strcpy(settings->password_eval, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -418,6 +421,20 @@ int settings_load(struct user_settings *s, const char *patharg)
|
|||||||
s->avatar_path[0] = '\0';
|
s->avatar_path[0] = '\0';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
|
||||||
|
if ( config_setting_lookup_string(setting, tox_strings.autorun_path, &str) ) {
|
||||||
|
snprintf(s->autorun_path, sizeof(s->autorun_path), "%s", str);
|
||||||
|
int len = strlen(str);
|
||||||
|
|
||||||
|
if (len >= sizeof(s->autorun_path) - 2)
|
||||||
|
s->autorun_path[0] = '\0';
|
||||||
|
else if (s->autorun_path[len - 1] != '/')
|
||||||
|
strcat(&s->autorun_path[len - 1], "/");
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
if ( config_setting_lookup_string(setting, tox_strings.password_eval, &str) ) {
|
if ( config_setting_lookup_string(setting, tox_strings.password_eval, &str) ) {
|
||||||
snprintf(s->password_eval, sizeof(s->password_eval), "%s", str);
|
snprintf(s->password_eval, sizeof(s->password_eval), "%s", str);
|
||||||
int len = strlen(str);
|
int len = strlen(str);
|
||||||
|
@ -63,6 +63,7 @@ struct user_settings {
|
|||||||
char download_path[PATH_MAX];
|
char download_path[PATH_MAX];
|
||||||
char chatlogs_path[PATH_MAX];
|
char chatlogs_path[PATH_MAX];
|
||||||
char avatar_path[PATH_MAX];
|
char avatar_path[PATH_MAX];
|
||||||
|
char autorun_path[PATH_MAX];
|
||||||
char password_eval[PASSWORD_EVAL_MAX];
|
char password_eval[PASSWORD_EVAL_MAX];
|
||||||
|
|
||||||
int key_next_tab;
|
int key_next_tab;
|
||||||
|
@ -214,8 +214,8 @@ static int detect_tmux ()
|
|||||||
if (!pos)
|
if (!pos)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/* store the session number string for later use */
|
/* store the session id for later use */
|
||||||
snprintf (mplex_data, sizeof(mplex_data), "%s", pos + 1);
|
snprintf (mplex_data, sizeof(mplex_data), "$%s", pos + 1);
|
||||||
mplex = MPLEX_TMUX;
|
mplex = MPLEX_TMUX;
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
@ -252,14 +252,8 @@ static int gnu_screen_is_detached ()
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Detects tmux attached/detached by getting session data and finding the
|
/* Detects tmux attached/detached by getting session data and finding the
|
||||||
current session's entry. An attached entry ends with "(attached)". Example:
|
current session's entry.
|
||||||
|
*/
|
||||||
$ tmux list-sessions
|
|
||||||
0: 1 windows (created Mon Mar 2 21:48:29 2015) [80x23] (attached)
|
|
||||||
1: 2 windows (created Mon Mar 2 21:48:43 2015) [80x23]
|
|
||||||
|
|
||||||
In this example, session 0 is attached and session 1 is detached.
|
|
||||||
*/
|
|
||||||
static int tmux_is_detached ()
|
static int tmux_is_detached ()
|
||||||
{
|
{
|
||||||
if (mplex != MPLEX_TMUX)
|
if (mplex != MPLEX_TMUX)
|
||||||
@ -267,10 +261,12 @@ static int tmux_is_detached ()
|
|||||||
|
|
||||||
FILE *session_info_stream = NULL;
|
FILE *session_info_stream = NULL;
|
||||||
char *dyn_buffer = NULL, *search_str = NULL;
|
char *dyn_buffer = NULL, *search_str = NULL;
|
||||||
char *entry_pos, *nl_pos, *attached_pos;
|
char *entry_pos;
|
||||||
|
int detached;
|
||||||
const int numstr_len = strlen (mplex_data);
|
const int numstr_len = strlen (mplex_data);
|
||||||
|
|
||||||
session_info_stream = popen ("env LC_ALL=C tmux list-sessions", "r");
|
/* get the number of attached clients for each session */
|
||||||
|
session_info_stream = popen ("tmux list-sessions -F \"#{session_id} #{session_attached}\"", "r");
|
||||||
|
|
||||||
if (!session_info_stream)
|
if (!session_info_stream)
|
||||||
goto fail;
|
goto fail;
|
||||||
@ -284,13 +280,12 @@ static int tmux_is_detached ()
|
|||||||
session_info_stream = NULL;
|
session_info_stream = NULL;
|
||||||
|
|
||||||
/* prepare search string, for finding the current session's entry */
|
/* prepare search string, for finding the current session's entry */
|
||||||
search_str = (char *) malloc (numstr_len + 4);
|
search_str = (char *) malloc (numstr_len + 2);
|
||||||
search_str[0] = '\n';
|
search_str[0] = '\n';
|
||||||
strcpy (search_str + 1, mplex_data);
|
strcpy (search_str + 1, mplex_data);
|
||||||
strcat (search_str, ": ");
|
|
||||||
|
|
||||||
/* do the search */
|
/* do the search */
|
||||||
if (strncmp (dyn_buffer, search_str + 1, numstr_len + 2) == 0)
|
if (strncmp (dyn_buffer, search_str + 1, numstr_len) == 0)
|
||||||
entry_pos = dyn_buffer;
|
entry_pos = dyn_buffer;
|
||||||
else
|
else
|
||||||
entry_pos = strstr (dyn_buffer, search_str);
|
entry_pos = strstr (dyn_buffer, search_str);
|
||||||
@ -298,9 +293,8 @@ static int tmux_is_detached ()
|
|||||||
if (! entry_pos)
|
if (! entry_pos)
|
||||||
goto fail;
|
goto fail;
|
||||||
|
|
||||||
/* find the next \n and look for the "(attached)" before it */
|
entry_pos = strchr (entry_pos, ' ') + 1;
|
||||||
nl_pos = strchr (entry_pos + 1, '\n');
|
detached = strncmp (entry_pos, "0\n", 2) == 0;
|
||||||
attached_pos = strstr (entry_pos + 1, "(attached)\n");
|
|
||||||
|
|
||||||
free (search_str);
|
free (search_str);
|
||||||
search_str = NULL;
|
search_str = NULL;
|
||||||
@ -308,7 +302,7 @@ static int tmux_is_detached ()
|
|||||||
free (dyn_buffer);
|
free (dyn_buffer);
|
||||||
dyn_buffer = NULL;
|
dyn_buffer = NULL;
|
||||||
|
|
||||||
return attached_pos == NULL || attached_pos > nl_pos;
|
return detached;
|
||||||
|
|
||||||
fail:
|
fail:
|
||||||
|
|
||||||
|
49
src/toxic.c
49
src/toxic.c
@ -26,6 +26,7 @@
|
|||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
#include <stdarg.h>
|
||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
#include <locale.h>
|
#include <locale.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
@ -75,6 +76,11 @@
|
|||||||
ToxAV *av;
|
ToxAV *av;
|
||||||
#endif /* AUDIO */
|
#endif /* AUDIO */
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
#include "api.h"
|
||||||
|
#include "python_api.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifndef PACKAGE_DATADIR
|
#ifndef PACKAGE_DATADIR
|
||||||
#define PACKAGE_DATADIR "."
|
#define PACKAGE_DATADIR "."
|
||||||
#endif
|
#endif
|
||||||
@ -104,9 +110,17 @@ static struct user_password {
|
|||||||
int len;
|
int len;
|
||||||
} user_password;
|
} user_password;
|
||||||
|
|
||||||
|
static time_t last_signal_time;
|
||||||
|
|
||||||
static void catch_SIGINT(int sig)
|
static void catch_SIGINT(int sig)
|
||||||
{
|
{
|
||||||
Winthread.sig_exit_toxic = 1;
|
time_t cur_time = get_unix_time();
|
||||||
|
|
||||||
|
if (difftime(cur_time, last_signal_time) <= 1) {
|
||||||
|
Winthread.sig_exit_toxic = 1;
|
||||||
|
} else {
|
||||||
|
last_signal_time = cur_time;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void catch_SIGSEGV(int sig)
|
static void catch_SIGSEGV(int sig)
|
||||||
@ -162,6 +176,10 @@ void exit_toxic_success(Tox *m)
|
|||||||
terminate_audio();
|
terminate_audio();
|
||||||
#endif /* AUDIO */
|
#endif /* AUDIO */
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
terminate_python();
|
||||||
|
#endif /* PYTHON */
|
||||||
|
|
||||||
free_global_data();
|
free_global_data();
|
||||||
tox_kill(m);
|
tox_kill(m);
|
||||||
endwin();
|
endwin();
|
||||||
@ -382,6 +400,7 @@ static void first_time_encrypt(const char *msg)
|
|||||||
do {
|
do {
|
||||||
system("clear");
|
system("clear");
|
||||||
printf("%s ", msg);
|
printf("%s ", msg);
|
||||||
|
fflush(stdout);
|
||||||
|
|
||||||
if (!strcasecmp(ch, "y\n") || !strcasecmp(ch, "n\n") || !strcasecmp(ch, "yes\n")
|
if (!strcasecmp(ch, "y\n") || !strcasecmp(ch, "n\n") || !strcasecmp(ch, "yes\n")
|
||||||
|| !strcasecmp(ch, "no\n") || !strcasecmp(ch, "q\n"))
|
|| !strcasecmp(ch, "no\n") || !strcasecmp(ch, "q\n"))
|
||||||
@ -402,6 +421,7 @@ static void first_time_encrypt(const char *msg)
|
|||||||
printf("Enter a new password (must be at least %d characters) ", MIN_PASSWORD_LEN);
|
printf("Enter a new password (must be at least %d characters) ", MIN_PASSWORD_LEN);
|
||||||
|
|
||||||
while (valid_password == false) {
|
while (valid_password == false) {
|
||||||
|
fflush(stdout); // Flush all before user input
|
||||||
len = password_prompt(user_password.pass, sizeof(user_password.pass));
|
len = password_prompt(user_password.pass, sizeof(user_password.pass));
|
||||||
user_password.len = len;
|
user_password.len = len;
|
||||||
|
|
||||||
@ -559,7 +579,7 @@ static void init_tox_options(struct Tox_Options *tox_opts)
|
|||||||
tox_opts->proxy_host = arg_opts.proxy_address;
|
tox_opts->proxy_host = arg_opts.proxy_address;
|
||||||
const char *ps = tox_opts->proxy_type == TOX_PROXY_TYPE_SOCKS5 ? "SOCKS5" : "HTTP";
|
const char *ps = tox_opts->proxy_type == TOX_PROXY_TYPE_SOCKS5 ? "SOCKS5" : "HTTP";
|
||||||
|
|
||||||
char tmp[48];
|
char tmp[sizeof(arg_opts.proxy_address) + MAX_STR_SIZE];
|
||||||
snprintf(tmp, sizeof(tmp), "Using %s proxy %s : %d", ps, arg_opts.proxy_address, arg_opts.proxy_port);
|
snprintf(tmp, sizeof(tmp), "Using %s proxy %s : %d", ps, arg_opts.proxy_address, arg_opts.proxy_port);
|
||||||
queue_init_message("%s", tmp);
|
queue_init_message("%s", tmp);
|
||||||
}
|
}
|
||||||
@ -627,6 +647,8 @@ static Tox *load_tox(char *data_path, struct Tox_Options *tox_opts, TOX_ERR_NEW
|
|||||||
char plain[plain_len];
|
char plain[plain_len];
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
|
fflush(stdout); // Flush before prompts so the user sees the question/message
|
||||||
|
|
||||||
if (pweval) {
|
if (pweval) {
|
||||||
pwlen = password_eval(user_password.pass, sizeof(user_password.pass));
|
pwlen = password_eval(user_password.pass, sizeof(user_password.pass));
|
||||||
} else {
|
} else {
|
||||||
@ -835,6 +857,13 @@ static void print_usage(void)
|
|||||||
fprintf(stderr, " -t, --force-tcp Force toxic to use a TCP connection (use with proxies)\n");
|
fprintf(stderr, " -t, --force-tcp Force toxic to use a TCP connection (use with proxies)\n");
|
||||||
fprintf(stderr, " -T, --tcp-server Act as a TCP relay server: Requires [port]\n");
|
fprintf(stderr, " -T, --tcp-server Act as a TCP relay server: Requires [port]\n");
|
||||||
fprintf(stderr, " -u, --unencrypt-data Unencrypt an encrypted data file\n");
|
fprintf(stderr, " -u, --unencrypt-data Unencrypt an encrypted data file\n");
|
||||||
|
fprintf(stderr, " -v, --version Print the version\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
static void print_version(void)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "Toxic version %s\n", TOXICVER);
|
||||||
|
fprintf(stderr, "Toxcore version %d.%d.%d\n", tox_version_major(), tox_version_minor(), tox_version_patch());
|
||||||
}
|
}
|
||||||
|
|
||||||
static void set_default_opts(void)
|
static void set_default_opts(void)
|
||||||
@ -865,10 +894,11 @@ static void parse_args(int argc, char *argv[])
|
|||||||
{"SOCKS5-proxy", required_argument, 0, 'p'},
|
{"SOCKS5-proxy", required_argument, 0, 'p'},
|
||||||
{"HTTP-proxy", required_argument, 0, 'P'},
|
{"HTTP-proxy", required_argument, 0, 'P'},
|
||||||
{"unencrypt-data", no_argument, 0, 'u'},
|
{"unencrypt-data", no_argument, 0, 'u'},
|
||||||
|
{"version", no_argument, 0, 'v'},
|
||||||
{NULL, no_argument, NULL, 0},
|
{NULL, no_argument, NULL, 0},
|
||||||
};
|
};
|
||||||
|
|
||||||
const char *opts_str = "4bdehotuxc:f:n:r:p:P:T:";
|
const char *opts_str = "4bdehotuxvc:f:n:r:p:P:T:";
|
||||||
int opt, indexptr;
|
int opt, indexptr;
|
||||||
long int port = 0;
|
long int port = 0;
|
||||||
|
|
||||||
@ -991,6 +1021,10 @@ static void parse_args(int argc, char *argv[])
|
|||||||
arg_opts.unencrypt_data = 1;
|
arg_opts.unencrypt_data = 1;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case 'v':
|
||||||
|
print_version();
|
||||||
|
exit(EXIT_SUCCESS);
|
||||||
|
|
||||||
case 'h':
|
case 'h':
|
||||||
default:
|
default:
|
||||||
print_usage();
|
print_usage();
|
||||||
@ -1197,6 +1231,13 @@ int main(int argc, char **argv)
|
|||||||
|
|
||||||
#endif /* AUDIO */
|
#endif /* AUDIO */
|
||||||
|
|
||||||
|
#ifdef PYTHON
|
||||||
|
|
||||||
|
init_python(m);
|
||||||
|
invoke_autoruns(prompt->chatwin->history, prompt);
|
||||||
|
|
||||||
|
#endif /* PYTHON */
|
||||||
|
|
||||||
init_notify(60, 3000);
|
init_notify(60, 3000);
|
||||||
|
|
||||||
/* screen/tmux auto-away timer */
|
/* screen/tmux auto-away timer */
|
||||||
@ -1216,7 +1257,7 @@ int main(int argc, char **argv)
|
|||||||
cleanup_init_messages();
|
cleanup_init_messages();
|
||||||
|
|
||||||
/* set user avatar from config file. if no path is supplied tox_unset_avatar is called */
|
/* set user avatar from config file. if no path is supplied tox_unset_avatar is called */
|
||||||
char avatarstr[MAX_STR_SIZE];
|
char avatarstr[PATH_MAX + 11];
|
||||||
snprintf(avatarstr, sizeof(avatarstr), "/avatar \"%s\"", user_settings->avatar_path);
|
snprintf(avatarstr, sizeof(avatarstr), "/avatar \"%s\"", user_settings->avatar_path);
|
||||||
execute(prompt->chatwin->history, prompt, m, avatarstr, GLOBAL_COMMAND_MODE);
|
execute(prompt->chatwin->history, prompt, m, avatarstr, GLOBAL_COMMAND_MODE);
|
||||||
|
|
||||||
|
@ -30,14 +30,18 @@
|
|||||||
|
|
||||||
#include <vpx/vpx_image.h>
|
#include <vpx/vpx_image.h>
|
||||||
|
|
||||||
#if defined(__linux__) || defined(__FreeBSD__)
|
#if defined(__OSX__)
|
||||||
|
#import "osx_video.h"
|
||||||
|
#else
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
#include <sys/stat.h>
|
#include <sys/stat.h>
|
||||||
#include <sys/mman.h>
|
#include <sys/mman.h>
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
|
#if defined(__linux__)
|
||||||
#include <linux/videodev2.h>
|
#include <linux/videodev2.h>
|
||||||
#else /* __OSX__ */
|
#else
|
||||||
#import "osx_video.h"
|
#include <sys/videoio.h>
|
||||||
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "line_info.h"
|
#include "line_info.h"
|
||||||
@ -65,7 +69,7 @@ typedef struct VideoDevice {
|
|||||||
void *cb_data; /* Data to be passed to callback */
|
void *cb_data; /* Data to be passed to callback */
|
||||||
int32_t friend_number; /* ToxAV friend number */
|
int32_t friend_number; /* ToxAV friend number */
|
||||||
|
|
||||||
#if defined(__linux__) || defined(__FreeBSD__)
|
#if defined(__linux__) || SYSTEM == BSD
|
||||||
int fd; /* File descriptor of video device selected/opened */
|
int fd; /* File descriptor of video device selected/opened */
|
||||||
struct v4l2_format fmt;
|
struct v4l2_format fmt;
|
||||||
struct VideoBuffer *buffers;
|
struct VideoBuffer *buffers;
|
||||||
@ -132,7 +136,7 @@ static void yuv420tobgr(uint16_t width, uint16_t height, const uint8_t *y,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#if defined(__linux__) || defined(__FreeBSD__)
|
#if defined(__linux__) || SYSTEM == BSD
|
||||||
static void yuv422to420(uint8_t *plane_y, uint8_t *plane_u, uint8_t *plane_v,
|
static void yuv422to420(uint8_t *plane_y, uint8_t *plane_u, uint8_t *plane_v,
|
||||||
uint8_t *input, uint16_t width, uint16_t height)
|
uint8_t *input, uint16_t width, uint16_t height)
|
||||||
{
|
{
|
||||||
@ -170,7 +174,7 @@ static int xioctl(int fh, unsigned long request, void *arg)
|
|||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif /* __linux__ */
|
#endif
|
||||||
|
|
||||||
/* Meet devices */
|
/* Meet devices */
|
||||||
#ifdef VIDEO
|
#ifdef VIDEO
|
||||||
@ -181,7 +185,10 @@ VideoDeviceError init_video_devices()
|
|||||||
{
|
{
|
||||||
size[vdt_input] = 0;
|
size[vdt_input] = 0;
|
||||||
|
|
||||||
#if defined(__linux__) || defined(__FreeBSD__)
|
#if defined(__OSX__)
|
||||||
|
if( osx_video_init((char**)video_devices_names[vdt_input], &size[vdt_input]) != 0 )
|
||||||
|
return vde_InternalError;
|
||||||
|
#else /* not __OSX__*/
|
||||||
|
|
||||||
for (; size[vdt_input] <= MAX_DEVICES; ++size[vdt_input]) {
|
for (; size[vdt_input] <= MAX_DEVICES; ++size[vdt_input]) {
|
||||||
int fd;
|
int fd;
|
||||||
@ -216,11 +223,6 @@ VideoDeviceError init_video_devices()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#else /* __OSX__ */
|
|
||||||
|
|
||||||
if ( osx_video_init((char **)video_devices_names[vdt_input], &size[vdt_input]) != 0 )
|
|
||||||
return vde_InternalError;
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
size[vdt_output] = 1;
|
size[vdt_output] = 1;
|
||||||
@ -271,17 +273,17 @@ VideoDeviceError terminate_video_devices()
|
|||||||
VideoDeviceError register_video_device_callback(int32_t friend_number, uint32_t device_idx,
|
VideoDeviceError register_video_device_callback(int32_t friend_number, uint32_t device_idx,
|
||||||
VideoDataHandleCallback callback, void *data)
|
VideoDataHandleCallback callback, void *data)
|
||||||
{
|
{
|
||||||
#if defined(__linux__) || defined(__FreeBSD__)
|
#if defined(__OSX__)
|
||||||
|
|
||||||
|
if ( size[vdt_input] <= device_idx || !video_devices_running[vdt_input][device_idx] )
|
||||||
|
return vde_InvalidSelection;
|
||||||
|
|
||||||
|
#else /* not __OSX__ */
|
||||||
|
|
||||||
if ( size[vdt_input] <= device_idx || !video_devices_running[vdt_input][device_idx]
|
if ( size[vdt_input] <= device_idx || !video_devices_running[vdt_input][device_idx]
|
||||||
|| !video_devices_running[vdt_input][device_idx]->fd )
|
|| !video_devices_running[vdt_input][device_idx]->fd )
|
||||||
return vde_InvalidSelection;
|
return vde_InvalidSelection;
|
||||||
|
|
||||||
#else /* __OSX__ */
|
|
||||||
|
|
||||||
if ( size[vdt_input] <= device_idx || !video_devices_running[vdt_input][device_idx] )
|
|
||||||
return vde_InvalidSelection;
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
lock;
|
lock;
|
||||||
@ -355,10 +357,16 @@ VideoDeviceError open_video_device(VideoDeviceType type, int32_t selection, uint
|
|||||||
if ( type == vdt_input ) {
|
if ( type == vdt_input ) {
|
||||||
video_thread_paused = true;
|
video_thread_paused = true;
|
||||||
|
|
||||||
#if defined(__linux__) || defined(__FreeBSD__)
|
#if defined(__OSX__)
|
||||||
|
if ( osx_video_open_device(selection, &device->video_width, &device->video_height) != 0 ) {
|
||||||
|
free(device);
|
||||||
|
unlock;
|
||||||
|
return vde_FailedStart;
|
||||||
|
}
|
||||||
|
#else /* not __OSX__*/
|
||||||
/* Open selected device */
|
/* Open selected device */
|
||||||
char device_address[] = "/dev/videoXX";
|
char device_address[] = "/dev/videoXX";
|
||||||
snprintf(device_address + 10 , sizeof(device_address) - 10, "%i", selection);
|
snprintf(device_address + 10, sizeof(device_address) - 10, "%i", selection);
|
||||||
|
|
||||||
device->fd = open(device_address, O_RDWR);
|
device->fd = open(device_address, O_RDWR);
|
||||||
|
|
||||||
@ -482,14 +490,6 @@ VideoDeviceError open_video_device(VideoDeviceType type, int32_t selection, uint
|
|||||||
return vde_FailedStart;
|
return vde_FailedStart;
|
||||||
}
|
}
|
||||||
|
|
||||||
#else /* __OSX__ */
|
|
||||||
|
|
||||||
if ( osx_video_open_device(selection, &device->video_width, &device->video_height) != 0 ) {
|
|
||||||
free(device);
|
|
||||||
unlock;
|
|
||||||
return vde_FailedStart;
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Create X11 window associated to device */
|
/* Create X11 window associated to device */
|
||||||
@ -666,7 +666,12 @@ void *video_thread_poll (void *arg) // TODO: maybe use thread for every input so
|
|||||||
uint8_t *u = device->input.planes[1];
|
uint8_t *u = device->input.planes[1];
|
||||||
uint8_t *v = device->input.planes[2];
|
uint8_t *v = device->input.planes[2];
|
||||||
|
|
||||||
#if defined(__linux__) || defined(__FreeBSD__)
|
#if defined(__OSX__)
|
||||||
|
if ( osx_video_read_device(y, u, v, &video_width, &video_height) != 0 ) {
|
||||||
|
unlock;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
#else /* not __OSX__*/
|
||||||
struct v4l2_buffer buf;
|
struct v4l2_buffer buf;
|
||||||
memset(&(buf), 0, sizeof(buf));
|
memset(&(buf), 0, sizeof(buf));
|
||||||
|
|
||||||
@ -683,13 +688,6 @@ void *video_thread_poll (void *arg) // TODO: maybe use thread for every input so
|
|||||||
/* Convert frame image data to YUV420 for ToxAV */
|
/* Convert frame image data to YUV420 for ToxAV */
|
||||||
yuv422to420(y, u, v, data, video_width, video_height);
|
yuv422to420(y, u, v, data, video_width, video_height);
|
||||||
|
|
||||||
#else /* __OSX__*/
|
|
||||||
|
|
||||||
if ( osx_video_read_device(y, u, v, &video_width, &video_height) != 0 ) {
|
|
||||||
unlock;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Send frame data to friend through ToxAV */
|
/* Send frame data to friend through ToxAV */
|
||||||
@ -726,14 +724,14 @@ void *video_thread_poll (void *arg) // TODO: maybe use thread for every input so
|
|||||||
XFlush(device->x_display);
|
XFlush(device->x_display);
|
||||||
free(img_data);
|
free(img_data);
|
||||||
|
|
||||||
#if defined(__linux__) || defined(__FreeBSD__)
|
#if defined(__linux__) || SYSTEM == BSD
|
||||||
|
|
||||||
if ( -1 == xioctl(device->fd, VIDIOC_QBUF, &buf) ) {
|
if ( -1 == xioctl(device->fd, VIDIOC_QBUF, &buf) ) {
|
||||||
unlock;
|
unlock;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif /* __linux__ */
|
#endif /* __linux__ / BSD */
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -765,7 +763,10 @@ VideoDeviceError close_video_device(VideoDeviceType type, uint32_t device_idx)
|
|||||||
if ( !device->ref_count ) {
|
if ( !device->ref_count ) {
|
||||||
|
|
||||||
if ( type == vdt_input ) {
|
if ( type == vdt_input ) {
|
||||||
#if defined(__linux__) || defined(__FreeBSD__)
|
#if defined(__OSX__)
|
||||||
|
|
||||||
|
osx_video_close_device(device_idx);
|
||||||
|
#else /* not __OSX__ */
|
||||||
enum v4l2_buf_type buf_type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
|
enum v4l2_buf_type buf_type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
|
||||||
|
|
||||||
if ( -1 == xioctl(device->fd, VIDIOC_STREAMOFF, &buf_type) ) {}
|
if ( -1 == xioctl(device->fd, VIDIOC_STREAMOFF, &buf_type) ) {}
|
||||||
@ -779,8 +780,6 @@ VideoDeviceError close_video_device(VideoDeviceType type, uint32_t device_idx)
|
|||||||
|
|
||||||
close(device->fd);
|
close(device->fd);
|
||||||
|
|
||||||
#else /* __OSX__ */
|
|
||||||
osx_video_close_device(device_idx);
|
|
||||||
#endif
|
#endif
|
||||||
vpx_img_free(&device->input);
|
vpx_img_free(&device->input);
|
||||||
XDestroyWindow(device->x_display, device->x_window);
|
XDestroyWindow(device->x_display, device->x_window);
|
||||||
@ -788,9 +787,9 @@ VideoDeviceError close_video_device(VideoDeviceType type, uint32_t device_idx)
|
|||||||
XCloseDisplay(device->x_display);
|
XCloseDisplay(device->x_display);
|
||||||
pthread_mutex_destroy(device->mutex);
|
pthread_mutex_destroy(device->mutex);
|
||||||
|
|
||||||
#if defined(__linux__) || defined(__FreeBSD__)
|
#if defined(__linux__) || SYSTEM == BSD
|
||||||
free(device->buffers);
|
free(device->buffers);
|
||||||
#endif /* __linux__ */
|
#endif /* __linux__ / BSD */
|
||||||
|
|
||||||
free(device);
|
free(device);
|
||||||
} else {
|
} else {
|
||||||
|
@ -475,10 +475,6 @@ static void draw_bar(void)
|
|||||||
|
|
||||||
move(LINES - 1, 0);
|
move(LINES - 1, 0);
|
||||||
|
|
||||||
attron(COLOR_PAIR(BLUE) | A_BOLD);
|
|
||||||
printw(" TOXIC " TOXICVER " |");
|
|
||||||
attroff(COLOR_PAIR(BLUE) | A_BOLD);
|
|
||||||
|
|
||||||
size_t i;
|
size_t i;
|
||||||
|
|
||||||
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
||||||
@ -588,6 +584,12 @@ ToxWindow *get_window_ptr(int i)
|
|||||||
return toxwin;
|
return toxwin;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* returns a pointer to the currently open ToxWindow. */
|
||||||
|
ToxWindow *get_active_window(void)
|
||||||
|
{
|
||||||
|
return active_window;
|
||||||
|
}
|
||||||
|
|
||||||
void force_refresh(WINDOW *w)
|
void force_refresh(WINDOW *w)
|
||||||
{
|
{
|
||||||
wclear(w);
|
wclear(w);
|
||||||
|
@ -259,6 +259,7 @@ void kill_all_windows(Tox *m); /* should only be called on shutdown */
|
|||||||
void on_window_resize(void);
|
void on_window_resize(void);
|
||||||
void force_refresh(WINDOW *w);
|
void force_refresh(WINDOW *w);
|
||||||
ToxWindow *get_window_ptr(int i);
|
ToxWindow *get_window_ptr(int i);
|
||||||
|
ToxWindow *get_active_window(void);
|
||||||
|
|
||||||
/* refresh inactive windows to prevent scrolling bugs.
|
/* refresh inactive windows to prevent scrolling bugs.
|
||||||
call at least once per second */
|
call at least once per second */
|
||||||
|
Reference in New Issue
Block a user