mirror of
https://github.com/qTox/qTox.git
synced 2024-03-22 14:00:36 +08:00
Italian translation: udpate
This commit is contained in:
commit
4caef8743b
14
bootstrap.sh
14
bootstrap.sh
@ -15,6 +15,7 @@ SODIUM_VER=1.0.0
|
||||
# directory names of cloned repositories
|
||||
SODIUM_DIR=libsodium-$SODIUM_VER
|
||||
TOX_CORE_DIR=libtoxcore-latest
|
||||
FILTER_AUDIO_DIR=filter_audio
|
||||
|
||||
# this boolean describes whether the installation of
|
||||
# libsodium should be skipped or not
|
||||
@ -42,6 +43,11 @@ if [ -z "$TOX_CORE_DIR" ]; then
|
||||
exit 1
|
||||
fi
|
||||
|
||||
if [ -z "$FILTER_AUDIO_DIR" ]; then
|
||||
echo "internal error detected!"
|
||||
echo "FILTER_AUDIO_DIR should not be empty... aborting"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
|
||||
########## check input parameters ##########
|
||||
@ -95,7 +101,7 @@ mkdir -p ${BASE_DIR}
|
||||
# if exists, otherwise cloning them may fail
|
||||
rm -rf ${BASE_DIR}/${SODIUM_DIR}
|
||||
rm -rf ${BASE_DIR}/${TOX_CORE_DIR}
|
||||
|
||||
rm -rf ${BASE_DIR}/${FILTER_AUDIO_DIR}
|
||||
|
||||
|
||||
############### install step ###############
|
||||
@ -122,6 +128,12 @@ if [[ $TOX_ONLY = "false" ]]; then
|
||||
fi
|
||||
|
||||
popd
|
||||
|
||||
if [[ $GLOBAL = "false" ]]; then
|
||||
./install_libfilteraudio.sh ${BASE_DIR}/${FILTER_AUDIO_DIR} ${BASE_DIR}
|
||||
else
|
||||
./install_libfilteraudio.sh ${BASE_DIR}/${FILTER_AUDIO_DIR}
|
||||
fi
|
||||
fi
|
||||
|
||||
# clone current master of libtoxcore
|
||||
|
40
install_libfilteraudio.sh
Executable file
40
install_libfilteraudio.sh
Executable file
@ -0,0 +1,40 @@
|
||||
#!/bin/sh
|
||||
|
||||
if [ -z $1 ]; then
|
||||
SOURCE_DIR="filter_audio/"
|
||||
else
|
||||
SOURCE_DIR="$1/"
|
||||
fi
|
||||
|
||||
if [ -z "$2" ]; then
|
||||
LIB_DIR="/usr/local/lib/"
|
||||
INCLUDE_DIR="/usr/local/include/"
|
||||
else
|
||||
LIB_DIR="$2/lib/"
|
||||
INCLUDE_DIR="$2/include/"
|
||||
fi
|
||||
|
||||
echo "Cloning filter_audio from GitHub.com"
|
||||
git clone https://github.com/irungentoo/filter_audio.git $SOURCE_DIR
|
||||
|
||||
echo "Compiling filter_audio"
|
||||
cd $SOURCE_DIR
|
||||
gcc -c -fPIC filter_audio.c aec/*.c agc/*.c ns/*.c other/*.c -lm -lpthread
|
||||
|
||||
echo "Creating shared object file"
|
||||
gcc *.o -shared -o libfilteraudio.so
|
||||
|
||||
echo "Cleaning up"
|
||||
rm *.o
|
||||
|
||||
muhcmd="cp libfilteraudio.so $LIB_DIR"
|
||||
[ -z "$2" ] && muhcmd="sudo $muhcmd"
|
||||
echo "Installing libfilteraudio.so with $muhcmd"
|
||||
$muhcmd
|
||||
|
||||
muhcmd="cp *.h $INCLUDE_DIR"
|
||||
[ -z "$2" ] && muhcmd="sudo $muhcmd"
|
||||
echo "Installing include files with $muhcmd"
|
||||
$muhcmd
|
||||
|
||||
echo "Finished."
|
23
osx/updater/.gitignore
vendored
Normal file
23
osx/updater/.gitignore
vendored
Normal file
@ -0,0 +1,23 @@
|
||||
# Compiled Object files, Static and Dynamic libs (Shared Objects)
|
||||
*.o
|
||||
*.a
|
||||
*.so
|
||||
|
||||
# Folders
|
||||
_obj
|
||||
_test
|
||||
|
||||
# Architecture specific extensions/prefixes
|
||||
*.[568vq]
|
||||
[568vq].out
|
||||
|
||||
*.cgo1.go
|
||||
*.cgo2.c
|
||||
_cgo_defun.c
|
||||
_cgo_gotypes.go
|
||||
_cgo_export.*
|
||||
|
||||
_testmain.go
|
||||
|
||||
*.exe
|
||||
*.test
|
13
osx/updater/README.md
Normal file
13
osx/updater/README.md
Normal file
@ -0,0 +1,13 @@
|
||||
The qTox OS X updater is a mix of objective C and Go compiled as static binaries used do effortless updates in the background.
|
||||
|
||||
It uses Objective C to access Apples own security framework and call some long dead APIs in order to give the statically linked go updater permissions to install the latest build without prompting the user for every file.
|
||||
|
||||
* Release commits: ``https://github.com/Tox/qTox_updater``
|
||||
* Development commits: ``https://github.mit.edu/sean-2/updater``
|
||||
|
||||
Compiling:
|
||||
|
||||
* ```clang qtox_sudo.m -framework corefoundation -framework security -framework cocoa -Os -o qtox_sudo```
|
||||
* ```go build updater.go```
|
||||
|
||||
(Starting with this commit all commits will be signed with [this key](http://pgp.mit.edu/pks/lookup?op=get&search=0x13D2043169D25DF4).)
|
273
osx/updater/qtox_sudo.m
Normal file
273
osx/updater/qtox_sudo.m
Normal file
@ -0,0 +1,273 @@
|
||||
// Modifications listed here GPLv3: https://gist.githubusercontent.com/stqism/2e82352026915f8f6ab3/raw/fca6f6f16fb8d61a64b6053dacf50c3433c2e0af/gistfile1.txt
|
||||
//
|
||||
// Copyright 2009 Performant Design, LLC. All rights reserved.
|
||||
// Copyright (C) 2014 Tox Foundation
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
//
|
||||
// This program 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.
|
||||
//
|
||||
// This program 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
|
||||
// this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
//
|
||||
|
||||
#import <Cocoa/Cocoa.h>
|
||||
|
||||
#include <sys/stat.h>
|
||||
#include <unistd.h>
|
||||
#include <Security/Authorization.h>
|
||||
#include <Security/AuthorizationTags.h>
|
||||
|
||||
char *addFileToPath(const char *path, const char *filename) {
|
||||
char *outbuf;
|
||||
char *lc;
|
||||
|
||||
lc = (char *)path + strlen(path) - 1;
|
||||
|
||||
if (lc < path || *lc != '/') {
|
||||
lc = NULL;
|
||||
}
|
||||
|
||||
while (*filename == '/') {
|
||||
filename++;
|
||||
}
|
||||
|
||||
outbuf = malloc(strlen(path) + strlen(filename) + 1 + (lc == NULL ? 1 : 0));
|
||||
|
||||
sprintf(outbuf, "%s%s%s", path, (lc == NULL) ? "/" : "", filename);
|
||||
|
||||
return outbuf;
|
||||
}
|
||||
|
||||
int isExecFile(const char *name) {
|
||||
struct stat s;
|
||||
|
||||
return (!access(name, X_OK) && !stat(name, &s) && S_ISREG(s.st_mode));
|
||||
}
|
||||
|
||||
char *which(const char *filename)
|
||||
{
|
||||
char *path, *p, *n;
|
||||
|
||||
path = getenv("PATH");
|
||||
|
||||
if (!path) {
|
||||
return NULL;
|
||||
}
|
||||
|
||||
p = path = strdup(path);
|
||||
|
||||
while (p) {
|
||||
n = strchr(p, ':');
|
||||
|
||||
if (n) {
|
||||
*n++ = '\0';
|
||||
}
|
||||
|
||||
if (*p != '\0') {
|
||||
p = addFileToPath(p, filename);
|
||||
|
||||
if (isExecFile(p)) {
|
||||
free(path);
|
||||
|
||||
return p;
|
||||
}
|
||||
|
||||
free(p);
|
||||
}
|
||||
|
||||
p = n;
|
||||
}
|
||||
|
||||
free(path);
|
||||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
int cocoaSudo(char *executable, char *commandArgs[], char *icon, char *prompt) {
|
||||
int retVal = 1;
|
||||
OSStatus status;
|
||||
AuthorizationRef authRef;
|
||||
|
||||
AuthorizationItem right = {kAuthorizationRightExecute, 0, NULL, 0};
|
||||
AuthorizationRights rightSet = {1, &right};
|
||||
|
||||
AuthorizationEnvironment myAuthorizationEnvironment;
|
||||
AuthorizationItem kAuthEnv[2];
|
||||
myAuthorizationEnvironment.items = kAuthEnv;
|
||||
|
||||
AuthorizationFlags flags = kAuthorizationFlagDefaults;
|
||||
|
||||
|
||||
if (prompt && icon) {
|
||||
kAuthEnv[0].name = kAuthorizationEnvironmentPrompt;
|
||||
kAuthEnv[0].valueLength = strlen(prompt);
|
||||
kAuthEnv[0].value = prompt;
|
||||
kAuthEnv[0].flags = 0;
|
||||
|
||||
kAuthEnv[1].name = kAuthorizationEnvironmentIcon;
|
||||
kAuthEnv[1].valueLength = strlen(icon);
|
||||
kAuthEnv[1].value = icon;
|
||||
kAuthEnv[1].flags = 0;
|
||||
|
||||
myAuthorizationEnvironment.count = 2;
|
||||
}
|
||||
else if (prompt) {
|
||||
kAuthEnv[0].name = kAuthorizationEnvironmentPrompt;
|
||||
kAuthEnv[0].valueLength = strlen(prompt);
|
||||
kAuthEnv[0].value = prompt;
|
||||
kAuthEnv[0].flags = 0;
|
||||
|
||||
myAuthorizationEnvironment.count = 1;
|
||||
}
|
||||
else if (icon) {
|
||||
kAuthEnv[0].name = kAuthorizationEnvironmentIcon;
|
||||
kAuthEnv[0].valueLength = strlen(icon);
|
||||
kAuthEnv[0].value = icon;
|
||||
kAuthEnv[0].flags = 0;
|
||||
|
||||
myAuthorizationEnvironment.count = 1;
|
||||
}
|
||||
else {
|
||||
myAuthorizationEnvironment.count = 0;
|
||||
}
|
||||
|
||||
status = AuthorizationCreate(NULL, &myAuthorizationEnvironment, flags, &authRef);
|
||||
|
||||
if (status != errAuthorizationSuccess) {
|
||||
NSLog(@"Could not create authorization reference object.");
|
||||
status = errAuthorizationBadAddress;
|
||||
}
|
||||
else {
|
||||
flags = kAuthorizationFlagDefaults |
|
||||
kAuthorizationFlagInteractionAllowed |
|
||||
kAuthorizationFlagPreAuthorize |
|
||||
kAuthorizationFlagExtendRights;
|
||||
|
||||
status = AuthorizationCopyRights(authRef, &rightSet, &myAuthorizationEnvironment, flags, NULL);
|
||||
}
|
||||
|
||||
if (status == errAuthorizationSuccess) {
|
||||
FILE *ioPipe;
|
||||
char buffer[1024];
|
||||
int bytesRead;
|
||||
|
||||
flags = kAuthorizationFlagDefaults;
|
||||
status = AuthorizationExecuteWithPrivileges(authRef, executable, flags, commandArgs, &ioPipe);
|
||||
|
||||
/* Just pipe processes' stdout to our stdout for now; hopefully can add stdin pipe later as well */
|
||||
|
||||
for (;;) {
|
||||
bytesRead = fread(buffer, sizeof(char), 1024, ioPipe);
|
||||
|
||||
if (bytesRead < 1) {
|
||||
break;
|
||||
}
|
||||
|
||||
write(STDOUT_FILENO, buffer, bytesRead * sizeof(char));
|
||||
}
|
||||
|
||||
pid_t pid;
|
||||
int pidStatus;
|
||||
|
||||
do {
|
||||
pid = wait(&pidStatus);
|
||||
}
|
||||
while (pid != -1);
|
||||
|
||||
if (status == errAuthorizationSuccess) {
|
||||
retVal = 0;
|
||||
}
|
||||
}
|
||||
else {
|
||||
AuthorizationFree(authRef, kAuthorizationFlagDestroyRights);
|
||||
authRef = NULL;
|
||||
|
||||
if (status != errAuthorizationCanceled) {
|
||||
// pre-auth failed
|
||||
NSLog(@"Pre-auth failed");
|
||||
}
|
||||
}
|
||||
|
||||
return retVal;
|
||||
}
|
||||
|
||||
void usage(char *appNameFull) {
|
||||
char *appName = strrchr(appNameFull, '/');
|
||||
|
||||
if (appName == NULL) {
|
||||
appName = appNameFull;
|
||||
}
|
||||
else {
|
||||
appName++;
|
||||
}
|
||||
|
||||
fprintf(stderr, "usage: %s command\n", appName);
|
||||
}
|
||||
|
||||
int main(int argc, char *argv[]) {
|
||||
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
|
||||
|
||||
int retVal = 1;
|
||||
int programArgsStartAt = 1;
|
||||
char *icon = NULL;
|
||||
char *prompt = NULL;
|
||||
|
||||
if (programArgsStartAt >= argc) {
|
||||
usage(argv[0]);
|
||||
}
|
||||
else {
|
||||
char *executable;
|
||||
|
||||
if (strchr(argv[programArgsStartAt], '/')) {
|
||||
executable = isExecFile(argv[programArgsStartAt]) ? strdup(argv[programArgsStartAt]) : NULL;
|
||||
}
|
||||
else {
|
||||
executable = which(argv[programArgsStartAt]);
|
||||
}
|
||||
|
||||
if (executable) {
|
||||
char **commandArgs = malloc((argc - programArgsStartAt) * sizeof(char**));
|
||||
|
||||
memcpy(commandArgs, argv + programArgsStartAt + 1, (argc - programArgsStartAt - 1) * sizeof(char**));
|
||||
|
||||
commandArgs[argc - programArgsStartAt - 1] = NULL;
|
||||
|
||||
retVal = cocoaSudo(executable, commandArgs, icon, prompt);
|
||||
|
||||
free(commandArgs);
|
||||
free(executable);
|
||||
}
|
||||
else {
|
||||
fprintf(stderr, "Unable to find %s\n", argv[programArgsStartAt]);
|
||||
|
||||
usage(argv[0]);
|
||||
}
|
||||
}
|
||||
|
||||
if (prompt) {
|
||||
free(prompt);
|
||||
}
|
||||
|
||||
[pool release];
|
||||
|
||||
return retVal;
|
||||
}
|
@ -7,8 +7,13 @@ import (
|
||||
"os"
|
||||
"os/exec"
|
||||
"os/user"
|
||||
"syscall"
|
||||
|
||||
"bitbucket.org/kardianos/osext"
|
||||
)
|
||||
|
||||
var custom_user string
|
||||
|
||||
func fs_type(path string) int {
|
||||
//name := "FileOrDir"
|
||||
f, err := os.Open(path)
|
||||
@ -68,25 +73,61 @@ func install(path string, pathlen int) int {
|
||||
}
|
||||
|
||||
func main() {
|
||||
syscall.Setuid(0)
|
||||
usr, e := user.Current()
|
||||
if e != nil {
|
||||
log.Fatal(e)
|
||||
}
|
||||
|
||||
update_dir := usr.HomeDir + "/Library/Preferences/tox/update/"
|
||||
if _, err := os.Stat(update_dir); os.IsNotExist(err) {
|
||||
fmt.Println("Error: No update folder, is check for updates enabled?")
|
||||
CHECK:
|
||||
if usr.Name != "System Administrator" {
|
||||
fmt.Println("Not running as root, relaunching")
|
||||
|
||||
appdir, _ := osext.Executable()
|
||||
appdir_len := len(appdir)
|
||||
sudo_path := appdir[0:(appdir_len-7)] + "qtox_sudo" //qtox_sudo is a fork of cocoasudo with all of its flags and other features stripped out
|
||||
|
||||
if _, err := os.Stat(sudo_path); os.IsNotExist(err) {
|
||||
fmt.Println("Error: No qtox_sudo binary installed, falling back")
|
||||
custom_user = usr.Name
|
||||
usr.Name = "System Administrator"
|
||||
goto CHECK
|
||||
}
|
||||
|
||||
relaunch := exec.Command(sudo_path, appdir, usr.Name)
|
||||
relaunch.Stdout = os.Stdout
|
||||
relaunch.Stderr = os.Stderr
|
||||
relaunch.Run()
|
||||
return
|
||||
|
||||
} else {
|
||||
|
||||
if len(os.Args) > 1 || custom_user != "" {
|
||||
|
||||
if custom_user == "" {
|
||||
custom_user = os.Args[1]
|
||||
}
|
||||
|
||||
update_dir := "/Users/" + custom_user + "/Library/Preferences/tox/update/"
|
||||
if _, err := os.Stat(update_dir); os.IsNotExist(err) {
|
||||
fmt.Println("Error: No update folder, is check for updates enabled?")
|
||||
return
|
||||
}
|
||||
fmt.Println("qTox Updater")
|
||||
|
||||
killqtox := exec.Command("/usr/bin/killall", "qtox")
|
||||
_ = killqtox.Run()
|
||||
|
||||
install(update_dir, len(update_dir))
|
||||
|
||||
os.RemoveAll(update_dir)
|
||||
fmt.Println("Update metadata wiped, launching qTox")
|
||||
launchqtox := exec.Command("/usr/bin/open", "-b", "im.tox.qtox")
|
||||
launchqtox.Run()
|
||||
|
||||
} else {
|
||||
fmt.Println("Error: no user passed")
|
||||
}
|
||||
|
||||
}
|
||||
fmt.Println("qTox Updater")
|
||||
|
||||
killqtox := exec.Command("/usr/bin/killall", "qtox")
|
||||
_ = killqtox.Run()
|
||||
|
||||
install(update_dir, len(update_dir))
|
||||
|
||||
os.RemoveAll(update_dir)
|
||||
fmt.Println("Update metadata wiped, launching qTox")
|
||||
launchqtox := exec.Command("/usr/bin/open", "-b", "im.tox.qtox")
|
||||
launchqtox.Run()
|
||||
}
|
22
qtox.pro
22
qtox.pro
@ -59,6 +59,12 @@ contains(DISABLE_PLATFORM_EXT, YES) {
|
||||
DEFINES += QTOX_PLATFORM_EXT
|
||||
}
|
||||
|
||||
contains(DISABLE_FILTER_AUDIO, YES) {
|
||||
|
||||
} else {
|
||||
DEFINES += QTOX_FILTER_AUDIO
|
||||
}
|
||||
|
||||
contains(JENKINS,YES) {
|
||||
INCLUDEPATH += ./libs/include/
|
||||
} else {
|
||||
@ -78,6 +84,7 @@ win32 {
|
||||
QMAKE_INFO_PLIST = osx/info.plist
|
||||
LIBS += -L$$PWD/libs/lib/ -ltoxcore -ltoxav -ltoxencryptsave -ltoxdns -lsodium -lvpx -framework OpenAL -lopencv_core -lopencv_highgui
|
||||
contains(DEFINES, QTOX_PLATFORM_EXT) { LIBS += -framework IOKit -framework CoreFoundation }
|
||||
contains(DEFINES, QTOX_FILTER_AUDIO) { }
|
||||
} else {
|
||||
# If we're building a package, static link libtox[core,av] and libsodium, since they are not provided by any package
|
||||
contains(STATICPKG, YES) {
|
||||
@ -94,8 +101,16 @@ win32 {
|
||||
LIBS += -lX11 -lXss
|
||||
}
|
||||
|
||||
contains(DEFINES, QTOX_FILTER_AUDIO) {
|
||||
contains(STATICPKG, YES) {
|
||||
LIBS += -Wl,-Bstatic -lfilteraudio
|
||||
} else {
|
||||
LIBS += -lfilteraudio
|
||||
}
|
||||
}
|
||||
|
||||
contains(JENKINS, YES) {
|
||||
LIBS = ./libs/lib/libtoxav.a ./libs/lib/libvpx.a ./libs/lib/libopus.a ./libs/lib/libtoxdns.a ./libs/lib/libtoxencryptsave.a ./libs/lib/libtoxcore.a ./libs/lib/libsodium.a /usr/lib/libopencv_core.so /usr/lib/libopencv_highgui.so /usr/lib/libopencv_imgproc.so -lopenal -lX11 -lXss -s
|
||||
LIBS = ./libs/lib/libtoxav.a ./libs/lib/libvpx.a ./libs/lib/libopus.a ./libs/lib/libtoxdns.a ./libs/lib/libtoxencryptsave.a ./libs/lib/libtoxcore.a ./libs/lib/libsodium.a ./libs/lib/libfilteraudio.a /usr/lib/libopencv_core.so /usr/lib/libopencv_highgui.so /usr/lib/libopencv_imgproc.so -lopenal -lX11 -lXss -s
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -234,6 +249,11 @@ SOURCES += \
|
||||
src/widget/form/settings/advancedform.cpp \
|
||||
src/audio.cpp
|
||||
|
||||
contains(DEFINES, QTOX_FILTER_AUDIO) {
|
||||
HEADERS += src/audiofilterer.h
|
||||
SOURCES += src/audiofilterer.cpp
|
||||
}
|
||||
|
||||
contains(DEFINES, QTOX_PLATFORM_EXT) {
|
||||
HEADERS += src/platform/timer.h
|
||||
SOURCES += src/platform/timer_osx.cpp \
|
||||
|
53
src/audiofilterer.cpp
Normal file
53
src/audiofilterer.cpp
Normal file
@ -0,0 +1,53 @@
|
||||
/*
|
||||
Copyright (C) 2014 by Project Tox <https://tox.im>
|
||||
|
||||
This file is part of qTox, a Qt-based graphical interface for Tox.
|
||||
|
||||
This program is libre 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.
|
||||
This program 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 COPYING file for more details.
|
||||
*/
|
||||
|
||||
|
||||
#ifdef QTOX_FILTER_AUDIO
|
||||
|
||||
#include "audiofilterer.h"
|
||||
extern "C"{
|
||||
#include <filter_audio.h>
|
||||
}
|
||||
|
||||
void AudioFilterer::startFilter(unsigned int fs)
|
||||
{
|
||||
closeFilter();
|
||||
filter = new_filter_audio(fs);
|
||||
}
|
||||
|
||||
void AudioFilterer::closeFilter()
|
||||
{
|
||||
if (filter)
|
||||
kill_filter_audio(filter);
|
||||
filter = nullptr;
|
||||
}
|
||||
|
||||
|
||||
void AudioFilterer::filterAudio(int16_t* data, int framesize)
|
||||
{
|
||||
if (!filter)
|
||||
return;
|
||||
|
||||
filter_audio(filter, (int16_t*) data, framesize);
|
||||
}
|
||||
|
||||
|
||||
AudioFilterer::~AudioFilterer()
|
||||
{
|
||||
closeFilter();
|
||||
}
|
||||
|
||||
#endif // QTOX_FILTER_AUDIO
|
41
src/audiofilterer.h
Normal file
41
src/audiofilterer.h
Normal file
@ -0,0 +1,41 @@
|
||||
/*
|
||||
Copyright (C) 2014 by Project Tox <https://tox.im>
|
||||
|
||||
This file is part of qTox, a Qt-based graphical interface for Tox.
|
||||
|
||||
This program is libre 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.
|
||||
This program 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 COPYING file for more details.
|
||||
*/
|
||||
|
||||
#ifdef QTOX_FILTER_AUDIO
|
||||
#ifndef AUDIOFILTERER_H
|
||||
#define AUDIOFILTERER_H
|
||||
#include <cstdint>
|
||||
|
||||
#ifndef _FILTER_AUDIO
|
||||
typedef struct Filter_Audio Filter_Audio;
|
||||
#endif
|
||||
|
||||
class AudioFilterer
|
||||
{
|
||||
public:
|
||||
explicit AudioFilterer() = default;
|
||||
~AudioFilterer();
|
||||
|
||||
void startFilter(unsigned int fs);
|
||||
void filterAudio(int16_t* data, int framesize);
|
||||
void closeFilter();
|
||||
|
||||
private:
|
||||
struct Filter_Audio* filter{nullptr};
|
||||
};
|
||||
|
||||
#endif // AUDIOFILTERER_H
|
||||
#endif // QTOX_FILTER_AUDIO
|
@ -33,6 +33,9 @@ class QTimer;
|
||||
class QString;
|
||||
class CString;
|
||||
class VideoSource;
|
||||
#ifdef QTOX_FILTER_AUDIO
|
||||
class AudioFilterer;
|
||||
#endif
|
||||
|
||||
class Core : public QObject
|
||||
{
|
||||
@ -283,6 +286,9 @@ private:
|
||||
int dhtServerId;
|
||||
static QList<ToxFile> fileSendQueue, fileRecvQueue;
|
||||
static ToxCall calls[TOXAV_MAX_CALLS];
|
||||
#ifdef QTOX_FILTER_AUDIO
|
||||
static AudioFilterer * filterer[TOXAV_MAX_CALLS];
|
||||
#endif
|
||||
static QHash<int, ToxGroupCall> groupCalls; // Maps group IDs to ToxGroupCalls
|
||||
QMutex fileSendMutex, messageSendMutex;
|
||||
bool ready;
|
||||
|
@ -17,10 +17,17 @@
|
||||
#include "core.h"
|
||||
#include "video/camera.h"
|
||||
#include "audio.h"
|
||||
#ifdef QTOX_FILTER_AUDIO
|
||||
#include "audiofilterer.h"
|
||||
#endif
|
||||
#include "misc/settings.h"
|
||||
#include <QDebug>
|
||||
#include <QTimer>
|
||||
|
||||
ToxCall Core::calls[TOXAV_MAX_CALLS];
|
||||
#ifdef QTOX_FILTER_AUDIO
|
||||
AudioFilterer * Core::filterer[TOXAV_MAX_CALLS] { nullptr};
|
||||
#endif
|
||||
const int Core::videobufsize{TOXAV_MAX_VIDEO_WIDTH * TOXAV_MAX_VIDEO_HEIGHT * 4};
|
||||
uint8_t* Core::videobuf;
|
||||
|
||||
@ -65,6 +72,19 @@ void Core::prepareCall(int friendId, int callId, ToxAv* toxav, bool videoEnabled
|
||||
calls[callId].sendVideoTimer->start();
|
||||
Camera::getInstance()->subscribe();
|
||||
}
|
||||
|
||||
#ifdef QTOX_FILTER_AUDIO
|
||||
if (Settings::getInstance().getFilterAudio())
|
||||
{
|
||||
Core::filterer[callId] = new AudioFilterer();
|
||||
filterer[callId]->startFilter(48000);
|
||||
}
|
||||
else
|
||||
{
|
||||
delete filterer[callId];
|
||||
filterer[callId] = nullptr;
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
void Core::onAvMediaChange(void* toxav, int32_t callId, void* core)
|
||||
@ -246,8 +266,16 @@ void Core::sendCallAudio(int callId, ToxAv* toxav)
|
||||
return;
|
||||
}
|
||||
|
||||
#ifdef QTOX_FILTER_AUDIO
|
||||
if (filterer[callId])
|
||||
{
|
||||
filterer[callId]->filterAudio((int16_t*) buf, framesize);
|
||||
}
|
||||
#endif
|
||||
if ((r = toxav_send_audio(toxav, callId, dest, r)) < 0)
|
||||
{
|
||||
qDebug() << "Core: toxav_send_audio error";
|
||||
}
|
||||
}
|
||||
calls[callId].sendAudioTimer->start();
|
||||
}
|
||||
@ -294,14 +322,16 @@ void Core::sendCallVideo(int callId)
|
||||
|
||||
void Core::micMuteToggle(int callId)
|
||||
{
|
||||
if (calls[callId].active) {
|
||||
if (calls[callId].active)
|
||||
{
|
||||
calls[callId].muteMic = !calls[callId].muteMic;
|
||||
}
|
||||
}
|
||||
|
||||
void Core::volMuteToggle(int callId)
|
||||
{
|
||||
if (calls[callId].active) {
|
||||
if (calls[callId].active)
|
||||
{
|
||||
calls[callId].muteVol = !calls[callId].muteVol;
|
||||
alSourcef(calls[callId].alSource, AL_GAIN, calls[callId].muteVol ? 0.f : 1.f);
|
||||
}
|
||||
@ -321,6 +351,15 @@ void Core::onAvCancel(void* _toxav, int32_t callId, void* core)
|
||||
|
||||
calls[callId].active = false;
|
||||
|
||||
#ifdef QTOX_FILTER_AUDIO
|
||||
if (filterer[callId])
|
||||
{
|
||||
filterer[callId]->closeFilter();
|
||||
delete filterer[callId];
|
||||
filterer[callId] = nullptr;
|
||||
}
|
||||
#endif
|
||||
|
||||
emit static_cast<Core*>(core)->avCancel(friendId, callId);
|
||||
}
|
||||
|
||||
|
@ -198,6 +198,7 @@ void Settings::load()
|
||||
s.beginGroup("Audio");
|
||||
inDev = s.value("inDev", "").toString();
|
||||
outDev = s.value("outDev", "").toString();
|
||||
filterAudio = s.value("filterAudio", false).toBool();
|
||||
s.endGroup();
|
||||
|
||||
// Read the embedded DHT bootsrap nodes list if needed
|
||||
@ -340,6 +341,7 @@ void Settings::save(QString path, bool writeFriends)
|
||||
s.beginGroup("Audio");
|
||||
s.setValue("inDev", inDev);
|
||||
s.setValue("outDev", outDev);
|
||||
s.setValue("filterAudio", filterAudio);
|
||||
s.endGroup();
|
||||
|
||||
if (!writeFriends || currentProfile.isEmpty()) // Core::switchConfiguration
|
||||
@ -898,6 +900,16 @@ void Settings::setOutDev(const QString& deviceSpecifier)
|
||||
outDev = deviceSpecifier;
|
||||
}
|
||||
|
||||
bool Settings::getFilterAudio() const
|
||||
{
|
||||
return filterAudio;
|
||||
}
|
||||
|
||||
void Settings::setFilterAudio(bool newValue)
|
||||
{
|
||||
filterAudio = newValue;
|
||||
}
|
||||
|
||||
QString Settings::getFriendAdress(const QString &publicKey) const
|
||||
{
|
||||
QString key = ToxID::fromString(publicKey).publicKey;
|
||||
|
@ -128,6 +128,9 @@ public:
|
||||
QString getOutDev() const;
|
||||
void setOutDev(const QString& deviceSpecifier);
|
||||
|
||||
bool getFilterAudio() const;
|
||||
void setFilterAudio(bool newValue);
|
||||
|
||||
// Assume all widgets have unique names
|
||||
// Don't use it to save every single thing you want to save, use it
|
||||
// for some general purpose widgets, such as MainWindows or Splitters,
|
||||
@ -298,6 +301,7 @@ private:
|
||||
// Audio
|
||||
QString inDev;
|
||||
QString outDev;
|
||||
bool filterAudio;
|
||||
|
||||
struct friendProp
|
||||
{
|
||||
|
@ -33,12 +33,19 @@ AVForm::AVForm() :
|
||||
bodyUI = new Ui::AVSettings;
|
||||
bodyUI->setupUi(this);
|
||||
|
||||
#ifdef QTOX_FILTER_AUDIO
|
||||
bodyUI->filterAudio->setChecked(Settings::getInstance().getFilterAudio());
|
||||
#else
|
||||
bodyUI->filterAudio->setDisabled(true);
|
||||
#endif
|
||||
|
||||
connect(Camera::getInstance(), &Camera::propProbingFinished, this, &AVForm::onPropProbingFinished);
|
||||
connect(Camera::getInstance(), &Camera::resolutionProbingFinished, this, &AVForm::onResProbingFinished);
|
||||
|
||||
auto qcomboboxIndexChanged = (void(QComboBox::*)(const QString&)) &QComboBox::currentIndexChanged;
|
||||
connect(bodyUI->inDevCombobox, qcomboboxIndexChanged, this, &AVForm::onInDevChanged);
|
||||
connect(bodyUI->outDevCombobox, qcomboboxIndexChanged, this, &AVForm::onOutDevChanged);
|
||||
connect(bodyUI->filterAudio, SIGNAL(toggled(bool)), this, SLOT(onFilterAudioToggled(bool)));
|
||||
connect(bodyUI->rescanButton, &QPushButton::clicked, this, [=](){getAudioInDevices(); getAudioOutDevices();});
|
||||
}
|
||||
|
||||
@ -189,3 +196,8 @@ void AVForm::onOutDevChanged(const QString& deviceDescriptor)
|
||||
Settings::getInstance().setOutDev(deviceDescriptor);
|
||||
Audio::openOutput(deviceDescriptor);
|
||||
}
|
||||
|
||||
void AVForm::onFilterAudioToggled(bool filterAudio)
|
||||
{
|
||||
Settings::getInstance().setFilterAudio(filterAudio);
|
||||
}
|
||||
|
@ -52,6 +52,7 @@ private slots:
|
||||
// audio
|
||||
void onInDevChanged(const QString& deviceDescriptor);
|
||||
void onOutDevChanged(const QString& deviceDescriptor);
|
||||
void onFilterAudioToggled(bool filterAudio);
|
||||
|
||||
// camera
|
||||
void onPropProbingFinished(Camera::Prop prop, double val);
|
||||
|
@ -30,8 +30,8 @@
|
||||
<rect>
|
||||
<x>0</x>
|
||||
<y>0</y>
|
||||
<width>810</width>
|
||||
<height>496</height>
|
||||
<width>808</width>
|
||||
<height>618</height>
|
||||
</rect>
|
||||
</property>
|
||||
<layout class="QVBoxLayout" name="verticalLayout_5">
|
||||
@ -96,6 +96,13 @@
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="8" column="0">
|
||||
<widget class="QCheckBox" name="filterAudio">
|
||||
<property name="text">
|
||||
<string>Filter audio</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
|
17
translations/it.ts
vendored
17
translations/it.ts
vendored
@ -12,12 +12,12 @@
|
||||
<context>
|
||||
<name>AVSettings</name>
|
||||
<message>
|
||||
<location filename="../src/widget/form/settings/avsettings.ui" line="105"/>
|
||||
<location filename="../src/widget/form/settings/avsettings.ui" line="112"/>
|
||||
<source>Video Settings</source>
|
||||
<translation>Impostazioni Video</translation>
|
||||
</message>
|
||||
<message>
|
||||
<location filename="../src/widget/form/settings/avsettings.ui" line="116"/>
|
||||
<location filename="../src/widget/form/settings/avsettings.ui" line="123"/>
|
||||
<source>Resolution</source>
|
||||
<translation>Risoluzione</translation>
|
||||
</message>
|
||||
@ -52,22 +52,27 @@
|
||||
<translation>Cerca dispositivi audio</translation>
|
||||
</message>
|
||||
<message>
|
||||
<location filename="../src/widget/form/settings/avsettings.ui" line="133"/>
|
||||
<location filename="../src/widget/form/settings/avsettings.ui" line="102"/>
|
||||
<source>Filter audio</source>
|
||||
<translation>Filtra audio</translation>
|
||||
</message>
|
||||
<message>
|
||||
<location filename="../src/widget/form/settings/avsettings.ui" line="140"/>
|
||||
<source>Hue</source>
|
||||
<translation>Colore</translation>
|
||||
</message>
|
||||
<message>
|
||||
<location filename="../src/widget/form/settings/avsettings.ui" line="147"/>
|
||||
<location filename="../src/widget/form/settings/avsettings.ui" line="154"/>
|
||||
<source>Brightness</source>
|
||||
<translation>Luminoistà</translation>
|
||||
</message>
|
||||
<message>
|
||||
<location filename="../src/widget/form/settings/avsettings.ui" line="161"/>
|
||||
<location filename="../src/widget/form/settings/avsettings.ui" line="168"/>
|
||||
<source>Saturation</source>
|
||||
<translation>Saturazione</translation>
|
||||
</message>
|
||||
<message>
|
||||
<location filename="../src/widget/form/settings/avsettings.ui" line="175"/>
|
||||
<location filename="../src/widget/form/settings/avsettings.ui" line="182"/>
|
||||
<source>Contrast</source>
|
||||
<translation>Contrasto</translation>
|
||||
</message>
|
||||
|
Loading…
x
Reference in New Issue
Block a user