1 Commits

Author SHA1 Message Date
Ghislain MARY
e44404fe32 Add Java tutorials. 2024-09-06 16:37:20 +02:00
85 changed files with 3382 additions and 1817 deletions

View File

@@ -27,6 +27,10 @@ Linphone-SDK binaries are fetched from our [Cocoapods repository](https://gitlab
Desktop tutorials are in C#, leveraging on our Nuget packaging.
## Java
Java tutorials targeting the Desktop. If targeting Android, take a look at the Android tutorials instead.
## Additional resources
All tutorials require a SIP account to function, and if you don't have one you can create as many as you want and for free using our [free SIP service](https://subscribe.linphone.org/).

6
java/.gitignore vendored Normal file
View File

@@ -0,0 +1,6 @@
.gradle
build
linphone-sdk*.jar
linphone-sdk*.zip
.*.db
.*.sqlite3

View File

@@ -0,0 +1,8 @@
Hello World tutorial
====================
The purpose of this tutorial is to explain how to add our SDK as a dependency of an Java project and how to create the `Core` object that all our APIs depends on.
Start by taking a look at the `build.gradle` file to see how we reference the Linphone SDK and add it as a dependency of our project.
The user interface will only display the `Core`'s version, but in the next tutorial you will learn how to use it to login your SIP account.

View File

@@ -0,0 +1,18 @@
plugins {
id 'application'
id 'org.openjfx.javafxplugin' version '0.1.0'
}
repositories {
mavenCentral()
}
dependencies {
implementation files('linphone-sdk.jar')
}
javafx {
modules = [ 'javafx.controls', 'javafx.fxml' ]
}
mainClassName = 'LinphoneJavaFX'

Binary file not shown.

View File

@@ -0,0 +1,7 @@
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-8.5-bin.zip
networkTimeout=10000
validateDistributionUrl=true
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists

249
java/0-HelloWorld/gradlew vendored Executable file
View File

@@ -0,0 +1,249 @@
#!/bin/sh
#
# Copyright © 2015-2021 the original authors.
#
# 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
#
# https://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.
#
##############################################################################
#
# Gradle start up script for POSIX generated by Gradle.
#
# Important for running:
#
# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is
# noncompliant, but you have some other compliant shell such as ksh or
# bash, then to run this script, type that shell name before the whole
# command line, like:
#
# ksh Gradle
#
# Busybox and similar reduced shells will NOT work, because this script
# requires all of these POSIX shell features:
# * functions;
# * expansions «$var», «${var}», «${var:-default}», «${var+SET}»,
# «${var#prefix}», «${var%suffix}», and «$( cmd )»;
# * compound commands having a testable exit status, especially «case»;
# * various built-in commands including «command», «set», and «ulimit».
#
# Important for patching:
#
# (2) This script targets any POSIX shell, so it avoids extensions provided
# by Bash, Ksh, etc; in particular arrays are avoided.
#
# The "traditional" practice of packing multiple parameters into a
# space-separated string is a well documented source of bugs and security
# problems, so this is (mostly) avoided, by progressively accumulating
# options in "$@", and eventually passing that to Java.
#
# Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS,
# and GRADLE_OPTS) rely on word-splitting, this is performed explicitly;
# see the in-line comments for details.
#
# There are tweaks for specific operating systems such as AIX, CygWin,
# Darwin, MinGW, and NonStop.
#
# (3) This script is generated from the Groovy template
# https://github.com/gradle/gradle/blob/HEAD/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt
# within the Gradle project.
#
# You can find Gradle at https://github.com/gradle/gradle/.
#
##############################################################################
# Attempt to set APP_HOME
# Resolve links: $0 may be a link
app_path=$0
# Need this for daisy-chained symlinks.
while
APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path
[ -h "$app_path" ]
do
ls=$( ls -ld "$app_path" )
link=${ls#*' -> '}
case $link in #(
/*) app_path=$link ;; #(
*) app_path=$APP_HOME$link ;;
esac
done
# This is normally unused
# shellcheck disable=SC2034
APP_BASE_NAME=${0##*/}
# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036)
APP_HOME=$( cd "${APP_HOME:-./}" > /dev/null && pwd -P ) || exit
# Use the maximum available, or set MAX_FD != -1 to use that value.
MAX_FD=maximum
warn () {
echo "$*"
} >&2
die () {
echo
echo "$*"
echo
exit 1
} >&2
# OS specific support (must be 'true' or 'false').
cygwin=false
msys=false
darwin=false
nonstop=false
case "$( uname )" in #(
CYGWIN* ) cygwin=true ;; #(
Darwin* ) darwin=true ;; #(
MSYS* | MINGW* ) msys=true ;; #(
NONSTOP* ) nonstop=true ;;
esac
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
# Determine the Java command to use to start the JVM.
if [ -n "$JAVA_HOME" ] ; then
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
# IBM's JDK on AIX uses strange locations for the executables
JAVACMD=$JAVA_HOME/jre/sh/java
else
JAVACMD=$JAVA_HOME/bin/java
fi
if [ ! -x "$JAVACMD" ] ; then
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
else
JAVACMD=java
if ! command -v java >/dev/null 2>&1
then
die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
fi
# Increase the maximum file descriptors if we can.
if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then
case $MAX_FD in #(
max*)
# In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked.
# shellcheck disable=SC2039,SC3045
MAX_FD=$( ulimit -H -n ) ||
warn "Could not query maximum file descriptor limit"
esac
case $MAX_FD in #(
'' | soft) :;; #(
*)
# In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked.
# shellcheck disable=SC2039,SC3045
ulimit -n "$MAX_FD" ||
warn "Could not set maximum file descriptor limit to $MAX_FD"
esac
fi
# Collect all arguments for the java command, stacking in reverse order:
# * args from the command line
# * the main class name
# * -classpath
# * -D...appname settings
# * --module-path (only if needed)
# * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables.
# For Cygwin or MSYS, switch paths to Windows format before running java
if "$cygwin" || "$msys" ; then
APP_HOME=$( cygpath --path --mixed "$APP_HOME" )
CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" )
JAVACMD=$( cygpath --unix "$JAVACMD" )
# Now convert the arguments - kludge to limit ourselves to /bin/sh
for arg do
if
case $arg in #(
-*) false ;; # don't mess with options #(
/?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath
[ -e "$t" ] ;; #(
*) false ;;
esac
then
arg=$( cygpath --path --ignore --mixed "$arg" )
fi
# Roll the args list around exactly as many times as the number of
# args, so each arg winds up back in the position where it started, but
# possibly modified.
#
# NB: a `for` loop captures its iteration list before it begins, so
# changing the positional parameters here affects neither the number of
# iterations, nor the values presented in `arg`.
shift # remove old arg
set -- "$@" "$arg" # push replacement arg
done
fi
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
# Collect all arguments for the java command:
# * DEFAULT_JVM_OPTS, JAVA_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments,
# and any embedded shellness will be escaped.
# * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be
# treated as '${Hostname}' itself on the command line.
set -- \
"-Dorg.gradle.appname=$APP_BASE_NAME" \
-classpath "$CLASSPATH" \
org.gradle.wrapper.GradleWrapperMain \
"$@"
# Stop when "xargs" is not available.
if ! command -v xargs >/dev/null 2>&1
then
die "xargs is not available"
fi
# Use "xargs" to parse quoted args.
#
# With -n1 it outputs one arg per line, with the quotes and backslashes removed.
#
# In Bash we could simply go:
#
# readarray ARGS < <( xargs -n1 <<<"$var" ) &&
# set -- "${ARGS[@]}" "$@"
#
# but POSIX shell has neither arrays nor command substitution, so instead we
# post-process each arg (as a line of input to sed) to backslash-escape any
# character that might be a shell metacharacter, then use eval to reverse
# that process (while maintaining the separation between arguments), and wrap
# the whole thing up as a single "set" statement.
#
# This will of course break if any of these variables contains a newline or
# an unmatched quote.
#
eval "set -- $(
printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" |
xargs -n1 |
sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' |
tr '\n' ' '
)" '"$@"'
exec "$JAVACMD" "$@"

92
java/0-HelloWorld/gradlew.bat vendored Normal file
View File

@@ -0,0 +1,92 @@
@rem
@rem Copyright 2015 the original author or authors.
@rem
@rem Licensed under the Apache License, Version 2.0 (the "License");
@rem you may not use this file except in compliance with the License.
@rem You may obtain a copy of the License at
@rem
@rem https://www.apache.org/licenses/LICENSE-2.0
@rem
@rem Unless required by applicable law or agreed to in writing, software
@rem distributed under the License is distributed on an "AS IS" BASIS,
@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
@rem See the License for the specific language governing permissions and
@rem limitations under the License.
@rem
@if "%DEBUG%"=="" @echo off
@rem ##########################################################################
@rem
@rem Gradle startup script for Windows
@rem
@rem ##########################################################################
@rem Set local scope for the variables with windows NT shell
if "%OS%"=="Windows_NT" setlocal
set DIRNAME=%~dp0
if "%DIRNAME%"=="" set DIRNAME=.
@rem This is normally unused
set APP_BASE_NAME=%~n0
set APP_HOME=%DIRNAME%
@rem Resolve any "." and ".." in APP_HOME to make it shorter.
for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
@rem Find java.exe
if defined JAVA_HOME goto findJavaFromJavaHome
set JAVA_EXE=java.exe
%JAVA_EXE% -version >NUL 2>&1
if %ERRORLEVEL% equ 0 goto execute
echo.
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:findJavaFromJavaHome
set JAVA_HOME=%JAVA_HOME:"=%
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
if exist "%JAVA_EXE%" goto execute
echo.
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:execute
@rem Setup the command line
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
@rem Execute Gradle
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %*
:end
@rem End local scope for the variables with windows NT shell
if %ERRORLEVEL% equ 0 goto mainEnd
:fail
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
rem the _cmd.exe /c_ return code!
set EXIT_CODE=%ERRORLEVEL%
if %EXIT_CODE% equ 0 set EXIT_CODE=1
if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE%
exit /b %EXIT_CODE%
:mainEnd
if "%OS%"=="Windows_NT" endlocal
:omega

View File

@@ -0,0 +1,63 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* This file is part of Linphone Java Tutorial.
*
* 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 javafx.application.Application;
import javafx.scene.Scene;
import javafx.scene.control.Label;
import javafx.scene.layout.StackPane;
import javafx.stage.Stage;
import org.linphone.core.Core;
import org.linphone.core.Factory;
import org.linphone.core.LogLevel;
import org.linphone.core.LoggingService;
public class LinphoneJavaFX extends Application {
@Override
public void start(Stage stage) throws Exception {
// Core is the main object of the SDK. You can't do much without it
// To create a Core, we need the instance of the Factory.
Factory factory = Factory.instance();
factory.setDataDir(".");
// Some configuration can be done before the Core is created, for example enable logs.
LoggingService loggingService = factory.getLoggingService();
loggingService.setLogLevel(LogLevel.Message);
// Your Core can use up to 2 configuration files, but that isn't mandatory.
// The third parameter is the application context, which is *not* mandatory when working with Java.
// You can now create your Core object.
Core core = factory.createCore("", "", null);
// Once you have your core you can start to do a lot of things, get its version for example.
String version = core.getVersion();
Label l = new Label("Hello world, Linphone core version is " + version);
Scene scene = new Scene(new StackPane(l), 640, 480);
stage.setScene(scene);
stage.show();
}
public static void main(String[] args) {
launch();
}
}

View File

@@ -0,0 +1,10 @@
Account login tutorial
====================
Now that you have set up Linphone-SDK in a Java project, let's start using it.
We will see how to login on a SIP server using the `Core` object instanciated in the previous tutorial.
If you don't have a SIP server yet, you can create an account for free using our [free SIP service](https://subscribe.linphone.org/).
Once you'll be logged-in, you'll be able to continue to the next tutorials to make calls.

View File

@@ -0,0 +1,18 @@
plugins {
id 'application'
id 'org.openjfx.javafxplugin' version '0.1.0'
}
repositories {
mavenCentral()
}
dependencies {
implementation files('linphone-sdk.jar')
}
javafx {
modules = [ 'javafx.controls', 'javafx.fxml' ]
}
mainClassName = 'LinphoneJavaFX'

Binary file not shown.

View File

@@ -0,0 +1,7 @@
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-8.5-bin.zip
networkTimeout=10000
validateDistributionUrl=true
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists

249
java/1-AccountLogin/gradlew vendored Executable file
View File

@@ -0,0 +1,249 @@
#!/bin/sh
#
# Copyright © 2015-2021 the original authors.
#
# 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
#
# https://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.
#
##############################################################################
#
# Gradle start up script for POSIX generated by Gradle.
#
# Important for running:
#
# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is
# noncompliant, but you have some other compliant shell such as ksh or
# bash, then to run this script, type that shell name before the whole
# command line, like:
#
# ksh Gradle
#
# Busybox and similar reduced shells will NOT work, because this script
# requires all of these POSIX shell features:
# * functions;
# * expansions «$var», «${var}», «${var:-default}», «${var+SET}»,
# «${var#prefix}», «${var%suffix}», and «$( cmd )»;
# * compound commands having a testable exit status, especially «case»;
# * various built-in commands including «command», «set», and «ulimit».
#
# Important for patching:
#
# (2) This script targets any POSIX shell, so it avoids extensions provided
# by Bash, Ksh, etc; in particular arrays are avoided.
#
# The "traditional" practice of packing multiple parameters into a
# space-separated string is a well documented source of bugs and security
# problems, so this is (mostly) avoided, by progressively accumulating
# options in "$@", and eventually passing that to Java.
#
# Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS,
# and GRADLE_OPTS) rely on word-splitting, this is performed explicitly;
# see the in-line comments for details.
#
# There are tweaks for specific operating systems such as AIX, CygWin,
# Darwin, MinGW, and NonStop.
#
# (3) This script is generated from the Groovy template
# https://github.com/gradle/gradle/blob/HEAD/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt
# within the Gradle project.
#
# You can find Gradle at https://github.com/gradle/gradle/.
#
##############################################################################
# Attempt to set APP_HOME
# Resolve links: $0 may be a link
app_path=$0
# Need this for daisy-chained symlinks.
while
APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path
[ -h "$app_path" ]
do
ls=$( ls -ld "$app_path" )
link=${ls#*' -> '}
case $link in #(
/*) app_path=$link ;; #(
*) app_path=$APP_HOME$link ;;
esac
done
# This is normally unused
# shellcheck disable=SC2034
APP_BASE_NAME=${0##*/}
# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036)
APP_HOME=$( cd "${APP_HOME:-./}" > /dev/null && pwd -P ) || exit
# Use the maximum available, or set MAX_FD != -1 to use that value.
MAX_FD=maximum
warn () {
echo "$*"
} >&2
die () {
echo
echo "$*"
echo
exit 1
} >&2
# OS specific support (must be 'true' or 'false').
cygwin=false
msys=false
darwin=false
nonstop=false
case "$( uname )" in #(
CYGWIN* ) cygwin=true ;; #(
Darwin* ) darwin=true ;; #(
MSYS* | MINGW* ) msys=true ;; #(
NONSTOP* ) nonstop=true ;;
esac
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
# Determine the Java command to use to start the JVM.
if [ -n "$JAVA_HOME" ] ; then
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
# IBM's JDK on AIX uses strange locations for the executables
JAVACMD=$JAVA_HOME/jre/sh/java
else
JAVACMD=$JAVA_HOME/bin/java
fi
if [ ! -x "$JAVACMD" ] ; then
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
else
JAVACMD=java
if ! command -v java >/dev/null 2>&1
then
die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
fi
# Increase the maximum file descriptors if we can.
if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then
case $MAX_FD in #(
max*)
# In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked.
# shellcheck disable=SC2039,SC3045
MAX_FD=$( ulimit -H -n ) ||
warn "Could not query maximum file descriptor limit"
esac
case $MAX_FD in #(
'' | soft) :;; #(
*)
# In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked.
# shellcheck disable=SC2039,SC3045
ulimit -n "$MAX_FD" ||
warn "Could not set maximum file descriptor limit to $MAX_FD"
esac
fi
# Collect all arguments for the java command, stacking in reverse order:
# * args from the command line
# * the main class name
# * -classpath
# * -D...appname settings
# * --module-path (only if needed)
# * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables.
# For Cygwin or MSYS, switch paths to Windows format before running java
if "$cygwin" || "$msys" ; then
APP_HOME=$( cygpath --path --mixed "$APP_HOME" )
CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" )
JAVACMD=$( cygpath --unix "$JAVACMD" )
# Now convert the arguments - kludge to limit ourselves to /bin/sh
for arg do
if
case $arg in #(
-*) false ;; # don't mess with options #(
/?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath
[ -e "$t" ] ;; #(
*) false ;;
esac
then
arg=$( cygpath --path --ignore --mixed "$arg" )
fi
# Roll the args list around exactly as many times as the number of
# args, so each arg winds up back in the position where it started, but
# possibly modified.
#
# NB: a `for` loop captures its iteration list before it begins, so
# changing the positional parameters here affects neither the number of
# iterations, nor the values presented in `arg`.
shift # remove old arg
set -- "$@" "$arg" # push replacement arg
done
fi
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
# Collect all arguments for the java command:
# * DEFAULT_JVM_OPTS, JAVA_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments,
# and any embedded shellness will be escaped.
# * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be
# treated as '${Hostname}' itself on the command line.
set -- \
"-Dorg.gradle.appname=$APP_BASE_NAME" \
-classpath "$CLASSPATH" \
org.gradle.wrapper.GradleWrapperMain \
"$@"
# Stop when "xargs" is not available.
if ! command -v xargs >/dev/null 2>&1
then
die "xargs is not available"
fi
# Use "xargs" to parse quoted args.
#
# With -n1 it outputs one arg per line, with the quotes and backslashes removed.
#
# In Bash we could simply go:
#
# readarray ARGS < <( xargs -n1 <<<"$var" ) &&
# set -- "${ARGS[@]}" "$@"
#
# but POSIX shell has neither arrays nor command substitution, so instead we
# post-process each arg (as a line of input to sed) to backslash-escape any
# character that might be a shell metacharacter, then use eval to reverse
# that process (while maintaining the separation between arguments), and wrap
# the whole thing up as a single "set" statement.
#
# This will of course break if any of these variables contains a newline or
# an unmatched quote.
#
eval "set -- $(
printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" |
xargs -n1 |
sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' |
tr '\n' ' '
)" '"$@"'
exec "$JAVACMD" "$@"

92
java/1-AccountLogin/gradlew.bat vendored Normal file
View File

@@ -0,0 +1,92 @@
@rem
@rem Copyright 2015 the original author or authors.
@rem
@rem Licensed under the Apache License, Version 2.0 (the "License");
@rem you may not use this file except in compliance with the License.
@rem You may obtain a copy of the License at
@rem
@rem https://www.apache.org/licenses/LICENSE-2.0
@rem
@rem Unless required by applicable law or agreed to in writing, software
@rem distributed under the License is distributed on an "AS IS" BASIS,
@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
@rem See the License for the specific language governing permissions and
@rem limitations under the License.
@rem
@if "%DEBUG%"=="" @echo off
@rem ##########################################################################
@rem
@rem Gradle startup script for Windows
@rem
@rem ##########################################################################
@rem Set local scope for the variables with windows NT shell
if "%OS%"=="Windows_NT" setlocal
set DIRNAME=%~dp0
if "%DIRNAME%"=="" set DIRNAME=.
@rem This is normally unused
set APP_BASE_NAME=%~n0
set APP_HOME=%DIRNAME%
@rem Resolve any "." and ".." in APP_HOME to make it shorter.
for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
@rem Find java.exe
if defined JAVA_HOME goto findJavaFromJavaHome
set JAVA_EXE=java.exe
%JAVA_EXE% -version >NUL 2>&1
if %ERRORLEVEL% equ 0 goto execute
echo.
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:findJavaFromJavaHome
set JAVA_HOME=%JAVA_HOME:"=%
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
if exist "%JAVA_EXE%" goto execute
echo.
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:execute
@rem Setup the command line
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
@rem Execute Gradle
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %*
:end
@rem End local scope for the variables with windows NT shell
if %ERRORLEVEL% equ 0 goto mainEnd
:fail
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
rem the _cmd.exe /c_ return code!
set EXIT_CODE=%ERRORLEVEL%
if %EXIT_CODE% equ 0 set EXIT_CODE=1
if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE%
exit /b %EXIT_CODE%
:mainEnd
if "%OS%"=="Windows_NT" endlocal
:omega

View File

@@ -0,0 +1,47 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* This file is part of Linphone Java Tutorial.
*
* 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 javafx.application.Application;
import javafx.scene.Parent;
import javafx.scene.Scene;
import javafx.stage.Stage;
import service.CoreService;
import javafx.fxml.FXMLLoader;
public class LinphoneJavaFX extends Application {
@Override
public void start(Stage stage) throws Exception {
Parent root = FXMLLoader.load(getClass().getResource("/view/LoginPage.fxml"));
stage.setTitle("Linphone SDK example");
stage.setScene(new Scene(root));
stage.show();
}
@Override
public void stop() {
CoreService.instance().stop();
System.exit(0);
}
public static void main(String[] args) {
launch();
}
}

View File

@@ -0,0 +1,154 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* This file is part of Linphone Java Tutorial.
*
* 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/>.
*/
package controller;
import java.net.URL;
import java.util.ResourceBundle;
import javafx.fxml.FXML;
import javafx.fxml.Initializable;
import javafx.scene.control.Button;
import javafx.scene.control.Label;
import javafx.scene.control.PasswordField;
import javafx.scene.control.RadioButton;
import javafx.scene.control.TextField;
import service.CoreService;
import org.linphone.core.Account;
import org.linphone.core.Core;
import org.linphone.core.CoreListenerStub;
import org.linphone.core.RegistrationState;
import org.linphone.core.TransportType;
public class LoginController extends CoreListenerStub implements Initializable {
@FXML
private Button loginButton;
@FXML
private Button logoutButton;
@FXML
private TextField identityField;
@FXML
private PasswordField passwordField;
@FXML
private RadioButton tlsRadio;
@FXML
private RadioButton tcpRadio;
@FXML
private RadioButton udpRadio;
@FXML
private Label loginLabel;
@FXML
private Label registrationLabel;
private CoreService coreService = CoreService.instance();
@FXML
public void initialize(URL location, ResourceBundle resourceBundle) {
// The Core is the main object of the SDK. You can't do much without it.
// If you're not familiar with Linphone Core creation, see the 0-HelloWorld project.
Core core = CoreService.instance().core;
// In this tutorial we are going to log in and our registration state will change.
// To get callbacks from the Core and be notified of the registration state change,
// we need to register ourself as a listener of the core.
core.addListener(this);
// Start the core after setup, and before everything else.
coreService.start();
// Setup GUI.
logoutGuiChanges();
// Now use the GUI to log in, and see LogInClick to see how to handle login.
}
// Called when you click on the "Login" button.
@FXML
private void onLoginClicked() {
if (loginButton.isDisabled())
return;
loginButton.setDisable(true);
TransportType transport = TransportType.Udp;
if (tlsRadio.isSelected()) {
transport = TransportType.Tls;
} else if (tcpRadio.isSelected()) {
transport = TransportType.Tcp;
}
coreService.login(identityField.getText(), passwordField.getText(), transport);
}
// Called when you click on the "Logout" button.
@FXML
private void onLogoutClicked() {
if (logoutButton.isDisabled())
return;
logoutButton.setDisable(true);
coreService.logout();
}
private void logoutGuiChanges() {
loginButton.setDisable(false);
logoutButton.setDisable(true);
loginLabel.setText("You are logged out");
}
private void loginGuiChanges() {
loginButton.setDisable(true);
logoutButton.setDisable(false);
loginLabel.setText("You are logged in, with identity " + CoreService.instance().core.getIdentity() + ".");
}
private void loginInProgressGuiChanges() {
loginButton.setDisable(true);
logoutButton.setDisable(true);
loginLabel.setText("Login in progress, with identity " + CoreService.instance().core.getIdentity() + ".");
}
private void loginFailedGuiChanges() {
loginButton.setDisable(false);
logoutButton.setDisable(true);
loginLabel.setText("Login failed, try again.");
}
// This method is called every time the RegistrationState is updated by background core's actions.
// In this example we use this to update the GUI.
@Override
public void onAccountRegistrationStateChanged(Core core, Account account, RegistrationState state, String message) {
registrationLabel.setText("Your registration state is: " + state.toString());
switch (state) {
case Cleared:
case None:
coreService.clearCoreAfterLogout();
logoutGuiChanges();
break;
case Ok:
loginGuiChanges();
break;
case Progress:
loginInProgressGuiChanges();
break;
case Failed:
loginFailedGuiChanges();
break;
default:
break;
}
}
}

View File

@@ -0,0 +1,159 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* This file is part of Linphone Java Tutorial.
*
* 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/>.
*/
package service;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.Timer;
import java.util.TimerTask;
import org.linphone.core.Account;
import org.linphone.core.AccountParams;
import org.linphone.core.Address;
import org.linphone.core.AuthInfo;
import org.linphone.core.Call;
import org.linphone.core.CallParams;
import org.linphone.core.Core;
import org.linphone.core.CoreListenerStub;
import org.linphone.core.Factory;
import org.linphone.core.GlobalState;
import org.linphone.core.LogLevel;
import org.linphone.core.LoggingService;
import org.linphone.core.TransportType;
import org.linphone.core.VideoActivationPolicy;
import javafx.application.Platform;
public class CoreService extends CoreListenerStub {
private static CoreService instance;
public Core core;
public IterateRunnable iterateRunnable;
private Timer iterateTimer;
private CoreService() {
Factory factory = Factory.instance();
factory.setDataDir(".");
LoggingService loggingService = factory.getLoggingService();
loggingService.setLogLevel(LogLevel.Message);
core = factory.createCore("", "", null);
core.setAudioPort(7666);
core.setUserCertificatesPath(Paths.get("").toString());
}
public static CoreService instance() {
if (instance == null) {
instance = new CoreService();
}
return instance;
}
public void start() {
this.core.start();
// The iterate method of the Core is to be called regularly, typically at 20 ms interval.
// So we create an IterateRunnable and a Timer to execute this call every 20 ms.
iterateRunnable = new IterateRunnable();
iterateTimer = new Timer();
iterateTimer.schedule(new IterateTimerTask(), 20, 20);
}
public void stop() {
core.stop();
}
public void login(String identity, String password, TransportType transport) {
// To configure a SIP account, we need an Account object and an AuthInfo object.
// The first one is how to connect to the proxy server, the second one stores the credentials.
// Here we are creating an AuthInfo object from the identity Address and password provided by the user.
Address address = Factory.instance().createAddress(identity);
// The AuthInfo can be created from the Factory as it's only a data class.
// userID is set to "" as it's the same as the username in our case.
// ha1 is set to "" as we are using the clear text password. Upon first register, the hash will be computed automatically.
// The realm will be determined automatically from the first register, as well as the algorithm.
AuthInfo authInfo = Factory.instance().createAuthInfo(address.getUsername(), "", password, "", "",
address.getDomain());
// And we add it to the Core.
core.addAuthInfo(authInfo);
// Then we create an AccountParams object.
// It contains the account informations needed by the core.
AccountParams accountParams = core.createAccountParams();
// A SIP account is identified by an identity address that we can construct from the username and domain.
accountParams.setIdentityAddress(address);
// We also need to configure where the proxy server is located.
Address serverAddress = Factory.instance().createAddress("sip:" + address.getDomain());
// We use the Address object to easily set the transport protocol.
serverAddress.setTransport(transport);
accountParams.setServerAddress(serverAddress);
// If setRegisterEnabled(true), when this account will be added to the core it will automatically try to connect.
accountParams.setRegisterEnabled(true);
// We can now create an Account object from the AccountParams...
Account account = core.createAccount(accountParams);
// ... and add it to the core, launching the connection process.
core.addAccount(account);
// Also set the newly added account as default.
core.setDefaultAccount(account);
}
public void logout() {
// setRegisterEnabled(false) on a connected Account object will launch the logout action.
Account account = core.getDefaultAccount();
if (account != null) {
// BUT BE CAREFUL : the Params attribute of an account is read-only, you MUST Clone it.
AccountParams accountParams = account.getParams().clone();
// Then you can modify the clone.
accountParams.setRegisterEnabled(false);
// And finally setting the new Params value triggers the changes, here the logout.
account.setParams(accountParams);
}
}
public void clearCoreAfterLogout() {
core.clearAllAuthInfo();
core.clearAccounts();
}
@Override
public void onGlobalStateChanged(Core core, GlobalState state, String message) {
if (state == GlobalState.Off) {
iterateTimer.cancel();
}
}
}
class IterateTimerTask extends TimerTask {
@Override
public void run() {
Platform.runLater(CoreService.instance().iterateRunnable);
}
}
class IterateRunnable implements Runnable {
@Override
public void run() {
CoreService.instance().core.iterate();
}
}

View File

@@ -0,0 +1,78 @@
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.geometry.Insets?>
<?import javafx.scene.control.Button?>
<?import javafx.scene.control.ButtonBar?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.control.PasswordField?>
<?import javafx.scene.control.RadioButton?>
<?import javafx.scene.control.Separator?>
<?import javafx.scene.control.TextField?>
<?import javafx.scene.control.ToggleGroup?>
<?import javafx.scene.layout.ColumnConstraints?>
<?import javafx.scene.layout.GridPane?>
<?import javafx.scene.layout.HBox?>
<?import javafx.scene.layout.Pane?>
<?import javafx.scene.layout.RowConstraints?>
<?import javafx.scene.layout.VBox?>
<Pane maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" prefHeight="400.0" prefWidth="600.0" xmlns="http://javafx.com/javafx/17" xmlns:fx="http://javafx.com/fxml/1" fx:controller="controller.LoginController">
<children>
<VBox prefHeight="400.0" prefWidth="600.0">
<children>
<GridPane vgap="20.0">
<columnConstraints>
<ColumnConstraints hgrow="SOMETIMES" minWidth="10.0" prefWidth="100.0" />
<ColumnConstraints hgrow="SOMETIMES" maxWidth="350.0" minWidth="350.0" prefWidth="350.0" />
</columnConstraints>
<rowConstraints>
<RowConstraints minHeight="10.0" prefHeight="30.0" vgrow="SOMETIMES" />
<RowConstraints minHeight="10.0" prefHeight="30.0" vgrow="SOMETIMES" />
<RowConstraints minHeight="10.0" prefHeight="30.0" vgrow="SOMETIMES" />
</rowConstraints>
<children>
<Label text="Identity:" GridPane.halignment="CENTER" />
<TextField id="Identity" fx:id="identityField" maxWidth="350.0" minWidth="350.0" prefWidth="350.0" promptText="sip:" GridPane.columnIndex="1" GridPane.halignment="CENTER" />
<Label text="Password:" GridPane.halignment="CENTER" GridPane.rowIndex="1" />
<PasswordField fx:id="passwordField" promptText="password" GridPane.columnIndex="1" GridPane.rowIndex="1" />
<HBox alignment="CENTER" prefHeight="60.0" prefWidth="200.0" spacing="50.0" GridPane.columnIndex="1" GridPane.rowIndex="2">
<children>
<RadioButton id="TlsRadio" fx:id="tlsRadio" mnemonicParsing="false" selected="true" text="TLS">
<toggleGroup>
<ToggleGroup fx:id="TransportGroup" />
</toggleGroup>
</RadioButton>
<RadioButton id="TcpRadio" fx:id="tcpRadio" mnemonicParsing="false" text="TCP" toggleGroup="$TransportGroup" />
<RadioButton id="UdpRadio" fx:id="udpRadio" mnemonicParsing="false" text="UDP" toggleGroup="$TransportGroup" />
</children>
</HBox>
<Label text="Transport:" GridPane.halignment="CENTER" GridPane.rowIndex="2" />
</children>
</GridPane>
<GridPane VBox.vgrow="ALWAYS">
<columnConstraints>
<ColumnConstraints hgrow="SOMETIMES" minWidth="10.0" prefWidth="100.0" />
</columnConstraints>
<rowConstraints>
<RowConstraints minHeight="10.0" prefHeight="30.0" vgrow="SOMETIMES" />
<RowConstraints minHeight="10.0" prefHeight="30.0" vgrow="SOMETIMES" />
</rowConstraints>
<children>
<Label id="LoginText" fx:id="loginLabel" alignment="TOP_LEFT" GridPane.halignment="CENTER" />
<Label fx:id="registrationLabel" alignment="TOP_LEFT" GridPane.halignment="CENTER" GridPane.rowIndex="1" />
</children>
</GridPane>
<Separator prefWidth="200.0" />
<ButtonBar prefHeight="40.0" prefWidth="200.0">
<buttons>
<Button fx:id="loginButton" defaultButton="true" mnemonicParsing="false" onMouseClicked="#onLoginClicked" text="Login" />
<Button fx:id="logoutButton" cancelButton="true" disable="true" mnemonicParsing="false" onMouseClicked="#onLogoutClicked" text="Logout" />
</buttons>
</ButtonBar>
</children>
<padding>
<Insets bottom="20.0" left="20.0" right="20.0" top="20.0" />
</padding>
</VBox>
</children>
</Pane>

View File

@@ -0,0 +1,8 @@
Incoming call tutorial
====================
This tutorial will focus on how the app will be notified when a call is being received and how to either accept it or terminate it.
We'll also cover how to toggle the microphone and the speakerphone during an active call.
If you want to test it on either a device, you'll need another SIP client to make the call. If you don't, you can use the [outgoing call tutorial](https://gitlab.linphone.org/BC/public/tutorials/-/tree/master/java/4-OutgoingCall) to do it.

View File

@@ -0,0 +1,18 @@
plugins {
id 'application'
id 'org.openjfx.javafxplugin' version '0.1.0'
}
repositories {
mavenCentral()
}
dependencies {
implementation files('linphone-sdk.jar')
}
javafx {
modules = [ 'javafx.controls', 'javafx.fxml' ]
}
mainClassName = 'LinphoneJavaFX'

Binary file not shown.

View File

@@ -0,0 +1,7 @@
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-8.5-bin.zip
networkTimeout=10000
validateDistributionUrl=true
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists

249
java/2-IncomingCall/gradlew vendored Executable file
View File

@@ -0,0 +1,249 @@
#!/bin/sh
#
# Copyright © 2015-2021 the original authors.
#
# 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
#
# https://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.
#
##############################################################################
#
# Gradle start up script for POSIX generated by Gradle.
#
# Important for running:
#
# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is
# noncompliant, but you have some other compliant shell such as ksh or
# bash, then to run this script, type that shell name before the whole
# command line, like:
#
# ksh Gradle
#
# Busybox and similar reduced shells will NOT work, because this script
# requires all of these POSIX shell features:
# * functions;
# * expansions «$var», «${var}», «${var:-default}», «${var+SET}»,
# «${var#prefix}», «${var%suffix}», and «$( cmd )»;
# * compound commands having a testable exit status, especially «case»;
# * various built-in commands including «command», «set», and «ulimit».
#
# Important for patching:
#
# (2) This script targets any POSIX shell, so it avoids extensions provided
# by Bash, Ksh, etc; in particular arrays are avoided.
#
# The "traditional" practice of packing multiple parameters into a
# space-separated string is a well documented source of bugs and security
# problems, so this is (mostly) avoided, by progressively accumulating
# options in "$@", and eventually passing that to Java.
#
# Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS,
# and GRADLE_OPTS) rely on word-splitting, this is performed explicitly;
# see the in-line comments for details.
#
# There are tweaks for specific operating systems such as AIX, CygWin,
# Darwin, MinGW, and NonStop.
#
# (3) This script is generated from the Groovy template
# https://github.com/gradle/gradle/blob/HEAD/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt
# within the Gradle project.
#
# You can find Gradle at https://github.com/gradle/gradle/.
#
##############################################################################
# Attempt to set APP_HOME
# Resolve links: $0 may be a link
app_path=$0
# Need this for daisy-chained symlinks.
while
APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path
[ -h "$app_path" ]
do
ls=$( ls -ld "$app_path" )
link=${ls#*' -> '}
case $link in #(
/*) app_path=$link ;; #(
*) app_path=$APP_HOME$link ;;
esac
done
# This is normally unused
# shellcheck disable=SC2034
APP_BASE_NAME=${0##*/}
# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036)
APP_HOME=$( cd "${APP_HOME:-./}" > /dev/null && pwd -P ) || exit
# Use the maximum available, or set MAX_FD != -1 to use that value.
MAX_FD=maximum
warn () {
echo "$*"
} >&2
die () {
echo
echo "$*"
echo
exit 1
} >&2
# OS specific support (must be 'true' or 'false').
cygwin=false
msys=false
darwin=false
nonstop=false
case "$( uname )" in #(
CYGWIN* ) cygwin=true ;; #(
Darwin* ) darwin=true ;; #(
MSYS* | MINGW* ) msys=true ;; #(
NONSTOP* ) nonstop=true ;;
esac
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
# Determine the Java command to use to start the JVM.
if [ -n "$JAVA_HOME" ] ; then
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
# IBM's JDK on AIX uses strange locations for the executables
JAVACMD=$JAVA_HOME/jre/sh/java
else
JAVACMD=$JAVA_HOME/bin/java
fi
if [ ! -x "$JAVACMD" ] ; then
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
else
JAVACMD=java
if ! command -v java >/dev/null 2>&1
then
die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
fi
# Increase the maximum file descriptors if we can.
if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then
case $MAX_FD in #(
max*)
# In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked.
# shellcheck disable=SC2039,SC3045
MAX_FD=$( ulimit -H -n ) ||
warn "Could not query maximum file descriptor limit"
esac
case $MAX_FD in #(
'' | soft) :;; #(
*)
# In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked.
# shellcheck disable=SC2039,SC3045
ulimit -n "$MAX_FD" ||
warn "Could not set maximum file descriptor limit to $MAX_FD"
esac
fi
# Collect all arguments for the java command, stacking in reverse order:
# * args from the command line
# * the main class name
# * -classpath
# * -D...appname settings
# * --module-path (only if needed)
# * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables.
# For Cygwin or MSYS, switch paths to Windows format before running java
if "$cygwin" || "$msys" ; then
APP_HOME=$( cygpath --path --mixed "$APP_HOME" )
CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" )
JAVACMD=$( cygpath --unix "$JAVACMD" )
# Now convert the arguments - kludge to limit ourselves to /bin/sh
for arg do
if
case $arg in #(
-*) false ;; # don't mess with options #(
/?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath
[ -e "$t" ] ;; #(
*) false ;;
esac
then
arg=$( cygpath --path --ignore --mixed "$arg" )
fi
# Roll the args list around exactly as many times as the number of
# args, so each arg winds up back in the position where it started, but
# possibly modified.
#
# NB: a `for` loop captures its iteration list before it begins, so
# changing the positional parameters here affects neither the number of
# iterations, nor the values presented in `arg`.
shift # remove old arg
set -- "$@" "$arg" # push replacement arg
done
fi
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
# Collect all arguments for the java command:
# * DEFAULT_JVM_OPTS, JAVA_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments,
# and any embedded shellness will be escaped.
# * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be
# treated as '${Hostname}' itself on the command line.
set -- \
"-Dorg.gradle.appname=$APP_BASE_NAME" \
-classpath "$CLASSPATH" \
org.gradle.wrapper.GradleWrapperMain \
"$@"
# Stop when "xargs" is not available.
if ! command -v xargs >/dev/null 2>&1
then
die "xargs is not available"
fi
# Use "xargs" to parse quoted args.
#
# With -n1 it outputs one arg per line, with the quotes and backslashes removed.
#
# In Bash we could simply go:
#
# readarray ARGS < <( xargs -n1 <<<"$var" ) &&
# set -- "${ARGS[@]}" "$@"
#
# but POSIX shell has neither arrays nor command substitution, so instead we
# post-process each arg (as a line of input to sed) to backslash-escape any
# character that might be a shell metacharacter, then use eval to reverse
# that process (while maintaining the separation between arguments), and wrap
# the whole thing up as a single "set" statement.
#
# This will of course break if any of these variables contains a newline or
# an unmatched quote.
#
eval "set -- $(
printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" |
xargs -n1 |
sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' |
tr '\n' ' '
)" '"$@"'
exec "$JAVACMD" "$@"

92
java/2-IncomingCall/gradlew.bat vendored Normal file
View File

@@ -0,0 +1,92 @@
@rem
@rem Copyright 2015 the original author or authors.
@rem
@rem Licensed under the Apache License, Version 2.0 (the "License");
@rem you may not use this file except in compliance with the License.
@rem You may obtain a copy of the License at
@rem
@rem https://www.apache.org/licenses/LICENSE-2.0
@rem
@rem Unless required by applicable law or agreed to in writing, software
@rem distributed under the License is distributed on an "AS IS" BASIS,
@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
@rem See the License for the specific language governing permissions and
@rem limitations under the License.
@rem
@if "%DEBUG%"=="" @echo off
@rem ##########################################################################
@rem
@rem Gradle startup script for Windows
@rem
@rem ##########################################################################
@rem Set local scope for the variables with windows NT shell
if "%OS%"=="Windows_NT" setlocal
set DIRNAME=%~dp0
if "%DIRNAME%"=="" set DIRNAME=.
@rem This is normally unused
set APP_BASE_NAME=%~n0
set APP_HOME=%DIRNAME%
@rem Resolve any "." and ".." in APP_HOME to make it shorter.
for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
@rem Find java.exe
if defined JAVA_HOME goto findJavaFromJavaHome
set JAVA_EXE=java.exe
%JAVA_EXE% -version >NUL 2>&1
if %ERRORLEVEL% equ 0 goto execute
echo.
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:findJavaFromJavaHome
set JAVA_HOME=%JAVA_HOME:"=%
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
if exist "%JAVA_EXE%" goto execute
echo.
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:execute
@rem Setup the command line
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
@rem Execute Gradle
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %*
:end
@rem End local scope for the variables with windows NT shell
if %ERRORLEVEL% equ 0 goto mainEnd
:fail
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
rem the _cmd.exe /c_ return code!
set EXIT_CODE=%ERRORLEVEL%
if %EXIT_CODE% equ 0 set EXIT_CODE=1
if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE%
exit /b %EXIT_CODE%
:mainEnd
if "%OS%"=="Windows_NT" endlocal
:omega

View File

@@ -0,0 +1,54 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* This file is part of Linphone Java Tutorial.
*
* 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 controller.ScreenController;
import javafx.application.Application;
import javafx.scene.Parent;
import javafx.scene.Scene;
import javafx.stage.Stage;
import service.CoreService;
import javafx.fxml.FXMLLoader;
public class LinphoneJavaFX extends Application {
@Override
public void start(Stage stage) throws Exception {
Parent root = FXMLLoader.load(getClass().getResource("/view/LoginPage.fxml"));
stage.setTitle("Linphone SDK example");
stage.setScene(new Scene(root));
ScreenController screenController = new ScreenController(stage.getScene());
screenController.add("login", "/view/LoginPage.fxml");
screenController.add("call", "/view/CallPage.fxml");
screenController.activate("login");
CoreService.instance().screenController = screenController;
stage.show();
}
@Override
public void stop() {
CoreService.instance().stop();
System.exit(0);
}
public static void main(String[] args) {
launch();
}
}

View File

@@ -0,0 +1,159 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* This file is part of Linphone Java Tutorial.
*
* 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/>.
*/
package controller;
import org.linphone.core.Call;
import org.linphone.core.Core;
import org.linphone.core.CoreListenerStub;
import org.linphone.core.Reason;
import javafx.fxml.FXML;
import javafx.scene.control.Button;
import javafx.scene.control.Label;
import javafx.scene.control.TextField;
import javafx.scene.control.TitledPane;
import javafx.scene.layout.VBox;
import service.CoreService;
public class CallController extends CoreListenerStub implements ScreenInterface {
private CoreService coreService = CoreService.instance();
private Call incomingCall;
@FXML
private TitledPane callPane;
@FXML
private Button hangupButton;
@FXML
private Button muteSoundButton;
@FXML
private Button muteMicrophoneButton;
@FXML
private VBox incomingCallVBox;
@FXML
private Label incomingCallLabel;
@FXML
private Button answerButton;
@FXML
private Button declineButton;
@FXML
private Label callLabel;
@FXML
private void onHangUpClicked() {
coreService.core.terminateAllCalls();
}
@FXML
private void onMuteSoundClicked() {
if (coreService.toggleSpeaker()) {
muteSoundButton.setText("Activate sound");
} else {
muteSoundButton.setText("Mute sound");
}
}
@FXML
private void onMuteMicrophoneClicked() {
if (coreService.toggleMicrophone()) {
muteMicrophoneButton.setText("Activate microphone");
} else {
muteMicrophoneButton.setText("Mute microphone");
}
}
@FXML
private void onAnswerClicked() {
if (incomingCall == null)
return;
incomingCall.accept();
incomingCall = null;
}
@FXML
private void onDeclineClicked() {
if (incomingCall == null)
return;
incomingCall.decline(Reason.Declined);
incomingCall = null;
}
private void callInProgressGuiUpdates() {
incomingCallVBox.setVisible(false);
hangupButton.setDisable(false);
muteSoundButton.setDisable(false);
muteMicrophoneButton.setDisable(false);
}
private void endingCallGuiUpdates() {
incomingCallVBox.setVisible(false);
hangupButton.setDisable(true);
muteSoundButton.setDisable(true);
muteSoundButton.setText("Mute sound");
muteMicrophoneButton.setDisable(true);
muteMicrophoneButton.setText("Mute microphone");
}
public void onCallStateChanged(Core core, Call call, Call.State state, String message) {
callLabel.setText("Your call state is: " + state.toString());
switch (state) {
case IncomingReceived:
incomingCall = call;
incomingCallVBox.setVisible(true);
incomingCallLabel.setText(incomingCall.getRemoteAddress().asString());
break;
case StreamsRunning:
callInProgressGuiUpdates();
break;
case Error:
case End:
case Released:
incomingCall = null;
endingCallGuiUpdates();
break;
default:
break;
}
}
@Override
public void onNavigatedFrom() {
coreService.core.removeListener(this);
}
@Override
public void onNavigatedTo() {
callPane.setText("Hello " + coreService.core.getDefaultProxyConfig().findAuthInfo().getUsername());
coreService.core.addListener(this);
if (coreService.core.getCurrentCall() != null) {
onCallStateChanged(coreService.core, coreService.core.getCurrentCall(),
coreService.core.getCurrentCall().getState(), null);
}
}
}

View File

@@ -0,0 +1,96 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* This file is part of Linphone Java Tutorial.
*
* 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/>.
*/
package controller;
import java.net.URL;
import java.util.ResourceBundle;
import javafx.fxml.FXML;
import javafx.fxml.Initializable;
import javafx.scene.control.Button;
import javafx.scene.control.Label;
import javafx.scene.control.PasswordField;
import javafx.scene.control.TextField;
import service.CoreService;
import org.linphone.core.Account;
import org.linphone.core.Core;
import org.linphone.core.CoreListenerStub;
import org.linphone.core.RegistrationState;
public class LoginController extends CoreListenerStub implements Initializable, ScreenInterface {
@FXML
private Button loginButton;
@FXML
private TextField identityField;
@FXML
private PasswordField passwordField;
@FXML
private Label registrationLabel;
private CoreService coreService = CoreService.instance();
@FXML
public void initialize(URL location, ResourceBundle resourceBundle) {
coreService.start();
}
@FXML
private void onLoginClicked() {
if (loginButton.isDisabled())
return;
loginButton.setDisable(true);
coreService.login(identityField.getText(), passwordField.getText());
}
@Override
public void onAccountRegistrationStateChanged(Core core, Account account, RegistrationState state, String message) {
registrationLabel.setText("Your registration state is: " + state.toString());
switch (state) {
case Cleared:
case None:
coreService.clearCoreAfterLogout();
loginButton.setDisable(false);
break;
case Ok:
loginButton.setDisable(true);
coreService.screenController.activate("call");
break;
case Progress:
loginButton.setDisable(true);
break;
case Failed:
loginButton.setDisable(false);
break;
default:
break;
}
}
@Override
public void onNavigatedFrom() {
coreService.core.removeListener(this);
}
@Override
public void onNavigatedTo() {
coreService.core.addListener(this);
}
}

View File

@@ -0,0 +1,66 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* This file is part of Linphone Java Tutorial.
*
* 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/>.
*/
package controller;
import java.io.IOException;
import java.util.HashMap;
import javafx.fxml.FXMLLoader;
import javafx.scene.Parent;
import javafx.scene.Scene;
class Screen {
public Parent root;
public FXMLLoader loader;
Screen(Parent root, FXMLLoader loader) {
this.root = root;
this.loader = loader;
}
}
public class ScreenController {
private HashMap<String, Screen> screenMap = new HashMap<>();
private Scene main;
private Screen currentScreen;
public ScreenController(Scene main) {
this.main = main;
}
public void add(String name, String viewFile) throws IOException {
FXMLLoader loader = new FXMLLoader(getClass().getResource(viewFile));
Parent root = loader.load();
screenMap.put(name, new Screen(root, loader));
}
public void remove(String name) {
screenMap.remove(name);
}
public void activate(String name) {
if (currentScreen != null) {
currentScreen.loader.<ScreenInterface>getController().onNavigatedFrom();
}
currentScreen = screenMap.get(name);
main.setRoot(currentScreen.root);
currentScreen.loader.<ScreenInterface>getController().onNavigatedTo();
}
}

View File

@@ -1,6 +1,8 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* This file is part of Linphone Java Tutorial.
*
* 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
@@ -15,15 +17,9 @@
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include <QCoreApplication>
package controller;
#include "App.hpp"
int main(int argc, char *argv[])
{
QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
App app(argc, argv);
app.init();
app.exec();
app.stop();
public interface ScreenInterface {
public void onNavigatedFrom();
public void onNavigatedTo();
}

View File

@@ -0,0 +1,148 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* This file is part of Linphone Java Tutorial.
*
* 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/>.
*/
package service;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.Timer;
import java.util.TimerTask;
import org.linphone.core.Account;
import org.linphone.core.AccountParams;
import org.linphone.core.Address;
import org.linphone.core.AuthInfo;
import org.linphone.core.Call;
import org.linphone.core.CallParams;
import org.linphone.core.Core;
import org.linphone.core.CoreListenerStub;
import org.linphone.core.Factory;
import org.linphone.core.GlobalState;
import org.linphone.core.LogLevel;
import org.linphone.core.LoggingService;
import org.linphone.core.VideoActivationPolicy;
import controller.ScreenController;
import javafx.application.Platform;
public class CoreService extends CoreListenerStub {
private static CoreService instance;
public Core core;
public ScreenController screenController;
public IterateRunnable iterateRunnable;
private Timer iterateTimer;
private CoreService() {
Factory factory = Factory.instance();
factory.setDataDir(".");
LoggingService loggingService = factory.getLoggingService();
loggingService.setLogLevel(LogLevel.Message);
core = factory.createCore("", "", null);
core.setAudioPort(7666);
core.setUserCertificatesPath(Paths.get("").toString());
}
public static CoreService instance() {
if (instance == null) {
instance = new CoreService();
}
return instance;
}
public void start() {
this.core.start();
iterateRunnable = new IterateRunnable();
iterateTimer = new Timer();
iterateTimer.schedule(new IterateTimerTask(), 20, 20);
}
public void stop() {
core.stop();
}
public void login(String identity, String password) {
Address address = Factory.instance().createAddress(identity);
AuthInfo authInfo = Factory.instance().createAuthInfo(address.getUsername(), "", password, "", "",
address.getDomain());
core.addAuthInfo(authInfo);
AccountParams accountParams = core.createAccountParams();
accountParams.setIdentityAddress(address);
Address serverAddress = Factory.instance().createAddress("sip:" + address.getDomain() + ";transport=tls");
accountParams.setServerAddress(serverAddress);
accountParams.setRegisterEnabled(true);
Account account = core.createAccount(accountParams);
core.addAccount(account);
core.setDefaultAccount(account);
}
public void logout() {
Account account = core.getDefaultAccount();
if (account != null) {
AccountParams accountParams = account.getParams().clone();
accountParams.setRegisterEnabled(false);
account.setParams(accountParams);
}
}
/// Mute/Unmute your microphone.
/// Setting to false on the Core mutes your microphone globally.
public Boolean toggleMicrophone() {
core.setMicEnabled(!core.isMicEnabled());
return core.isMicEnabled();
}
/// Enable/Disable the speaker sound.
/// setSpeakerMuted(true) on a Call object disables the sound output of this call.
public Boolean toggleSpeaker() {
core.getCurrentCall().setSpeakerMuted(!core.getCurrentCall().getSpeakerMuted());
return core.getCurrentCall().getSpeakerMuted();
}
public void clearCoreAfterLogout() {
core.clearAllAuthInfo();
core.clearAccounts();
}
@Override
public void onGlobalStateChanged(Core core, GlobalState state, String message) {
if (state == GlobalState.Off) {
iterateTimer.cancel();
}
}
}
class IterateTimerTask extends TimerTask {
@Override
public void run() {
Platform.runLater(CoreService.instance().iterateRunnable);
}
}
class IterateRunnable implements Runnable {
@Override
public void run() {
CoreService.instance().core.iterate();
}
}

View File

@@ -0,0 +1,52 @@
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.geometry.Insets?>
<?import javafx.scene.control.Button?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.control.TextField?>
<?import javafx.scene.control.TitledPane?>
<?import javafx.scene.layout.AnchorPane?>
<?import javafx.scene.layout.ColumnConstraints?>
<?import javafx.scene.layout.GridPane?>
<?import javafx.scene.layout.HBox?>
<?import javafx.scene.layout.RowConstraints?>
<?import javafx.scene.layout.VBox?>
<TitledPane fx:id="callPane" alignment="TOP_CENTER" animated="false" collapsible="false" maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" prefHeight="400.0" prefWidth="600.0" text="Hello" xmlns="http://javafx.com/javafx/17" xmlns:fx="http://javafx.com/fxml/1" fx:controller="controller.CallController">
<content>
<AnchorPane minHeight="0.0" minWidth="0.0" prefHeight="180.0" prefWidth="200.0">
<children>
<VBox alignment="CENTER" prefHeight="200.0" prefWidth="100.0" AnchorPane.bottomAnchor="0.0" AnchorPane.leftAnchor="0.0" AnchorPane.rightAnchor="0.0" AnchorPane.topAnchor="0.0">
<children>
<VBox alignment="CENTER" prefHeight="200.0" prefWidth="100.0">
<children>
<Label fx:id="callLabel" alignment="CENTER" text="Your call state is: Idle" />
<HBox alignment="CENTER" prefHeight="100.0" prefWidth="200.0" spacing="35.0">
<children>
<Button fx:id="hangupButton" disable="true" mnemonicParsing="false" onMouseClicked="#onHangUpClicked" text="Hang up" />
<Button fx:id="muteSoundButton" disable="true" mnemonicParsing="false" onMouseClicked="#onMuteSoundClicked" text="Mute sound" />
<Button fx:id="muteMicrophoneButton" disable="true" mnemonicParsing="false" onMouseClicked="#onMuteMicrophoneClicked" text="Mute microphone" />
</children>
</HBox>
</children>
<VBox.margin>
<Insets top="20.0" />
</VBox.margin>
</VBox>
<VBox fx:id="incomingCallVBox" alignment="CENTER" prefHeight="200.0" prefWidth="100.0" visible="false">
<children>
<Label text="You have a call from:" />
<Label fx:id="incomingCallLabel" />
<HBox alignment="CENTER" prefHeight="100.0" prefWidth="200.0" spacing="35.0">
<children>
<Button fx:id="answerButton" mnemonicParsing="false" onMouseClicked="#onAnswerClicked" text="Answer" />
<Button fx:id="declineButton" mnemonicParsing="false" onMouseClicked="#onDeclineClicked" text="Decline" />
</children>
</HBox>
</children>
</VBox>
</children>
</VBox>
</children></AnchorPane>
</content>
</TitledPane>

View File

@@ -0,0 +1,57 @@
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.geometry.Insets?>
<?import javafx.scene.control.Button?>
<?import javafx.scene.control.ButtonBar?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.control.PasswordField?>
<?import javafx.scene.control.Separator?>
<?import javafx.scene.control.TextField?>
<?import javafx.scene.control.TitledPane?>
<?import javafx.scene.layout.AnchorPane?>
<?import javafx.scene.layout.ColumnConstraints?>
<?import javafx.scene.layout.FlowPane?>
<?import javafx.scene.layout.GridPane?>
<?import javafx.scene.layout.RowConstraints?>
<TitledPane alignment="TOP_CENTER" animated="false" collapsible="false" maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" prefHeight="400.0" prefWidth="600.0" text="Login Form" xmlns:fx="http://javafx.com/fxml/1" xmlns="http://javafx.com/javafx/17" fx:controller="controller.LoginController">
<content>
<AnchorPane minHeight="0.0" minWidth="0.0" prefHeight="180.0" prefWidth="200.0">
<children>
<FlowPane alignment="TOP_CENTER" columnHalignment="CENTER" layoutX="199.0" layoutY="54.0" orientation="VERTICAL" prefHeight="200.0" prefWidth="200.0" AnchorPane.leftAnchor="0.0" AnchorPane.rightAnchor="0.0" AnchorPane.topAnchor="0.0">
<children>
<GridPane>
<columnConstraints>
<ColumnConstraints hgrow="SOMETIMES" minWidth="10.0" prefWidth="100.0" />
<ColumnConstraints hgrow="ALWAYS" minWidth="10.0" prefWidth="350.0" />
</columnConstraints>
<rowConstraints>
<RowConstraints minHeight="10.0" prefHeight="30.0" vgrow="SOMETIMES" />
<RowConstraints minHeight="10.0" prefHeight="30.0" vgrow="SOMETIMES" />
</rowConstraints>
<children>
<Label text="Identity:" />
<TextField fx:id="identityField" maxWidth="350.0" minWidth="350.0" prefWidth="350.0" GridPane.columnIndex="1" />
<Label text="Password:" GridPane.rowIndex="1" />
<PasswordField fx:id="passwordField" maxWidth="350.0" minWidth="350.0" prefWidth="350.0" GridPane.columnIndex="1" GridPane.rowIndex="1" />
</children>
<FlowPane.margin>
<Insets bottom="20.0" top="20.0" />
</FlowPane.margin>
</GridPane>
<Label fx:id="registrationLabel" prefWidth="0.0">
<FlowPane.margin>
<Insets />
</FlowPane.margin>
</Label>
<Separator prefWidth="200.0" />
<ButtonBar prefHeight="40.0" prefWidth="200.0">
<buttons>
<Button fx:id="loginButton" mnemonicParsing="false" onMouseClicked="#onLoginClicked" text="Login" />
</buttons>
</ButtonBar>
</children>
</FlowPane>
</children></AnchorPane>
</content>
</TitledPane>

View File

@@ -0,0 +1,4 @@
Outgoing call tutorial
====================
In the previous tutorial we saw how to handle an incoming call, now let's start one.

View File

@@ -0,0 +1,18 @@
plugins {
id 'application'
id 'org.openjfx.javafxplugin' version '0.1.0'
}
repositories {
mavenCentral()
}
dependencies {
implementation files('linphone-sdk.jar')
}
javafx {
modules = [ 'javafx.controls', 'javafx.fxml' ]
}
mainClassName = 'LinphoneJavaFX'

Binary file not shown.

View File

@@ -0,0 +1,7 @@
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-8.5-bin.zip
networkTimeout=10000
validateDistributionUrl=true
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists

249
java/3-OutgoingCall/gradlew vendored Executable file
View File

@@ -0,0 +1,249 @@
#!/bin/sh
#
# Copyright © 2015-2021 the original authors.
#
# 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
#
# https://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.
#
##############################################################################
#
# Gradle start up script for POSIX generated by Gradle.
#
# Important for running:
#
# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is
# noncompliant, but you have some other compliant shell such as ksh or
# bash, then to run this script, type that shell name before the whole
# command line, like:
#
# ksh Gradle
#
# Busybox and similar reduced shells will NOT work, because this script
# requires all of these POSIX shell features:
# * functions;
# * expansions «$var», «${var}», «${var:-default}», «${var+SET}»,
# «${var#prefix}», «${var%suffix}», and «$( cmd )»;
# * compound commands having a testable exit status, especially «case»;
# * various built-in commands including «command», «set», and «ulimit».
#
# Important for patching:
#
# (2) This script targets any POSIX shell, so it avoids extensions provided
# by Bash, Ksh, etc; in particular arrays are avoided.
#
# The "traditional" practice of packing multiple parameters into a
# space-separated string is a well documented source of bugs and security
# problems, so this is (mostly) avoided, by progressively accumulating
# options in "$@", and eventually passing that to Java.
#
# Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS,
# and GRADLE_OPTS) rely on word-splitting, this is performed explicitly;
# see the in-line comments for details.
#
# There are tweaks for specific operating systems such as AIX, CygWin,
# Darwin, MinGW, and NonStop.
#
# (3) This script is generated from the Groovy template
# https://github.com/gradle/gradle/blob/HEAD/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt
# within the Gradle project.
#
# You can find Gradle at https://github.com/gradle/gradle/.
#
##############################################################################
# Attempt to set APP_HOME
# Resolve links: $0 may be a link
app_path=$0
# Need this for daisy-chained symlinks.
while
APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path
[ -h "$app_path" ]
do
ls=$( ls -ld "$app_path" )
link=${ls#*' -> '}
case $link in #(
/*) app_path=$link ;; #(
*) app_path=$APP_HOME$link ;;
esac
done
# This is normally unused
# shellcheck disable=SC2034
APP_BASE_NAME=${0##*/}
# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036)
APP_HOME=$( cd "${APP_HOME:-./}" > /dev/null && pwd -P ) || exit
# Use the maximum available, or set MAX_FD != -1 to use that value.
MAX_FD=maximum
warn () {
echo "$*"
} >&2
die () {
echo
echo "$*"
echo
exit 1
} >&2
# OS specific support (must be 'true' or 'false').
cygwin=false
msys=false
darwin=false
nonstop=false
case "$( uname )" in #(
CYGWIN* ) cygwin=true ;; #(
Darwin* ) darwin=true ;; #(
MSYS* | MINGW* ) msys=true ;; #(
NONSTOP* ) nonstop=true ;;
esac
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
# Determine the Java command to use to start the JVM.
if [ -n "$JAVA_HOME" ] ; then
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
# IBM's JDK on AIX uses strange locations for the executables
JAVACMD=$JAVA_HOME/jre/sh/java
else
JAVACMD=$JAVA_HOME/bin/java
fi
if [ ! -x "$JAVACMD" ] ; then
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
else
JAVACMD=java
if ! command -v java >/dev/null 2>&1
then
die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
fi
# Increase the maximum file descriptors if we can.
if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then
case $MAX_FD in #(
max*)
# In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked.
# shellcheck disable=SC2039,SC3045
MAX_FD=$( ulimit -H -n ) ||
warn "Could not query maximum file descriptor limit"
esac
case $MAX_FD in #(
'' | soft) :;; #(
*)
# In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked.
# shellcheck disable=SC2039,SC3045
ulimit -n "$MAX_FD" ||
warn "Could not set maximum file descriptor limit to $MAX_FD"
esac
fi
# Collect all arguments for the java command, stacking in reverse order:
# * args from the command line
# * the main class name
# * -classpath
# * -D...appname settings
# * --module-path (only if needed)
# * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables.
# For Cygwin or MSYS, switch paths to Windows format before running java
if "$cygwin" || "$msys" ; then
APP_HOME=$( cygpath --path --mixed "$APP_HOME" )
CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" )
JAVACMD=$( cygpath --unix "$JAVACMD" )
# Now convert the arguments - kludge to limit ourselves to /bin/sh
for arg do
if
case $arg in #(
-*) false ;; # don't mess with options #(
/?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath
[ -e "$t" ] ;; #(
*) false ;;
esac
then
arg=$( cygpath --path --ignore --mixed "$arg" )
fi
# Roll the args list around exactly as many times as the number of
# args, so each arg winds up back in the position where it started, but
# possibly modified.
#
# NB: a `for` loop captures its iteration list before it begins, so
# changing the positional parameters here affects neither the number of
# iterations, nor the values presented in `arg`.
shift # remove old arg
set -- "$@" "$arg" # push replacement arg
done
fi
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
# Collect all arguments for the java command:
# * DEFAULT_JVM_OPTS, JAVA_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments,
# and any embedded shellness will be escaped.
# * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be
# treated as '${Hostname}' itself on the command line.
set -- \
"-Dorg.gradle.appname=$APP_BASE_NAME" \
-classpath "$CLASSPATH" \
org.gradle.wrapper.GradleWrapperMain \
"$@"
# Stop when "xargs" is not available.
if ! command -v xargs >/dev/null 2>&1
then
die "xargs is not available"
fi
# Use "xargs" to parse quoted args.
#
# With -n1 it outputs one arg per line, with the quotes and backslashes removed.
#
# In Bash we could simply go:
#
# readarray ARGS < <( xargs -n1 <<<"$var" ) &&
# set -- "${ARGS[@]}" "$@"
#
# but POSIX shell has neither arrays nor command substitution, so instead we
# post-process each arg (as a line of input to sed) to backslash-escape any
# character that might be a shell metacharacter, then use eval to reverse
# that process (while maintaining the separation between arguments), and wrap
# the whole thing up as a single "set" statement.
#
# This will of course break if any of these variables contains a newline or
# an unmatched quote.
#
eval "set -- $(
printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" |
xargs -n1 |
sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' |
tr '\n' ' '
)" '"$@"'
exec "$JAVACMD" "$@"

92
java/3-OutgoingCall/gradlew.bat vendored Normal file
View File

@@ -0,0 +1,92 @@
@rem
@rem Copyright 2015 the original author or authors.
@rem
@rem Licensed under the Apache License, Version 2.0 (the "License");
@rem you may not use this file except in compliance with the License.
@rem You may obtain a copy of the License at
@rem
@rem https://www.apache.org/licenses/LICENSE-2.0
@rem
@rem Unless required by applicable law or agreed to in writing, software
@rem distributed under the License is distributed on an "AS IS" BASIS,
@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
@rem See the License for the specific language governing permissions and
@rem limitations under the License.
@rem
@if "%DEBUG%"=="" @echo off
@rem ##########################################################################
@rem
@rem Gradle startup script for Windows
@rem
@rem ##########################################################################
@rem Set local scope for the variables with windows NT shell
if "%OS%"=="Windows_NT" setlocal
set DIRNAME=%~dp0
if "%DIRNAME%"=="" set DIRNAME=.
@rem This is normally unused
set APP_BASE_NAME=%~n0
set APP_HOME=%DIRNAME%
@rem Resolve any "." and ".." in APP_HOME to make it shorter.
for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
@rem Find java.exe
if defined JAVA_HOME goto findJavaFromJavaHome
set JAVA_EXE=java.exe
%JAVA_EXE% -version >NUL 2>&1
if %ERRORLEVEL% equ 0 goto execute
echo.
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:findJavaFromJavaHome
set JAVA_HOME=%JAVA_HOME:"=%
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
if exist "%JAVA_EXE%" goto execute
echo.
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:execute
@rem Setup the command line
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
@rem Execute Gradle
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %*
:end
@rem End local scope for the variables with windows NT shell
if %ERRORLEVEL% equ 0 goto mainEnd
:fail
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
rem the _cmd.exe /c_ return code!
set EXIT_CODE=%ERRORLEVEL%
if %EXIT_CODE% equ 0 set EXIT_CODE=1
if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE%
exit /b %EXIT_CODE%
:mainEnd
if "%OS%"=="Windows_NT" endlocal
:omega

View File

@@ -0,0 +1,54 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* This file is part of Linphone Java Tutorial.
*
* 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 controller.ScreenController;
import javafx.application.Application;
import javafx.scene.Parent;
import javafx.scene.Scene;
import javafx.stage.Stage;
import service.CoreService;
import javafx.fxml.FXMLLoader;
public class LinphoneJavaFX extends Application {
@Override
public void start(Stage stage) throws Exception {
Parent root = FXMLLoader.load(getClass().getResource("/view/LoginPage.fxml"));
stage.setTitle("Linphone SDK example");
stage.setScene(new Scene(root));
ScreenController screenController = new ScreenController(stage.getScene());
screenController.add("login", "/view/LoginPage.fxml");
screenController.add("call", "/view/CallPage.fxml");
screenController.activate("login");
CoreService.instance().screenController = screenController;
stage.show();
}
@Override
public void stop() {
CoreService.instance().stop();
System.exit(0);
}
public static void main(String[] args) {
launch();
}
}

View File

@@ -0,0 +1,206 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* This file is part of Linphone Java Tutorial.
*
* 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/>.
*/
package controller;
import org.linphone.core.Call;
import org.linphone.core.Core;
import org.linphone.core.CoreListenerStub;
import org.linphone.core.Reason;
import javafx.fxml.FXML;
import javafx.scene.control.Button;
import javafx.scene.control.Label;
import javafx.scene.control.TextField;
import javafx.scene.control.TitledPane;
import javafx.scene.layout.VBox;
import service.CoreService;
public class CallController extends CoreListenerStub implements ScreenInterface {
private CoreService coreService = CoreService.instance();
private Call incomingCall;
@FXML
private TitledPane callPane;
@FXML
TextField uriToCallField;
@FXML
private Button callButton;
@FXML
private Button hangupButton;
@FXML
private Button cameraButton;
@FXML
private Button muteSoundButton;
@FXML
private Button muteMicrophoneButton;
@FXML
private VBox incomingCallVBox;
@FXML
private Label incomingCallLabel;
@FXML
private Button answerButton;
@FXML
private Button declineButton;
@FXML
private Label callLabel;
@FXML
private void onCallClicked() {
coreService.call(uriToCallField.getText());
}
@FXML
private void onHangUpClicked() {
coreService.core.terminateAllCalls();
}
@FXML
private void onCameraClicked() {
coreService.toggleCamera();
cameraButton.setText("Waiting...");
cameraButton.setDisable(true);
}
@FXML
private void onMuteSoundClicked() {
if (coreService.toggleSpeaker()) {
muteSoundButton.setText("Activate sound");
} else {
muteSoundButton.setText("Mute sound");
}
}
@FXML
private void onMuteMicrophoneClicked() {
if (coreService.toggleMicrophone()) {
muteMicrophoneButton.setText("Activate microphone");
} else {
muteMicrophoneButton.setText("Mute microphone");
}
}
@FXML
private void onAnswerClicked() {
if (incomingCall == null)
return;
incomingCall.accept();
incomingCall = null;
}
@FXML
private void onDeclineClicked() {
if (incomingCall == null)
return;
incomingCall.decline(Reason.Declined);
incomingCall = null;
}
private void callInProgressGuiUpdates() {
incomingCallVBox.setVisible(false);
callButton.setDisable(true);
hangupButton.setDisable(false);
cameraButton.setDisable(false);
muteSoundButton.setDisable(false);
muteMicrophoneButton.setDisable(false);
}
private void endingCallGuiUpdates() {
incomingCallVBox.setVisible(false);
callButton.setDisable(false);
hangupButton.setDisable(true);
cameraButton.setDisable(true);
cameraButton.setText("Activate camera");
muteSoundButton.setDisable(true);
muteSoundButton.setText("Mute sound");
muteMicrophoneButton.setDisable(true);
muteMicrophoneButton.setText("Mute microphone");
}
private void startVideoAndUpdateGui() {
cameraButton.setText("Deactivate camera");
callButton.setDisable(false);
}
private void stopVideoAndUpdateGui() {
cameraButton.setText("Activate camera");
cameraButton.setDisable(false);
}
public void onCallStateChanged(Core core, Call call, Call.State state, String message) {
callLabel.setText("Your call state is: " + state.toString());
switch (state) {
case IncomingReceived:
incomingCall = call;
incomingCallVBox.setVisible(true);
incomingCallLabel.setText(incomingCall.getRemoteAddress().asString());
break;
case OutgoingInit:
case OutgoingProgress:
case OutgoingRinging:
hangupButton.setDisable(false);
break;
case StreamsRunning:
case UpdatedByRemote:
callInProgressGuiUpdates();
if (call.getCurrentParams().isVideoEnabled()) {
startVideoAndUpdateGui();
} else {
stopVideoAndUpdateGui();
}
break;
case Error:
case End:
case Released:
incomingCall = null;
endingCallGuiUpdates();
break;
default:
break;
}
}
@Override
public void onNavigatedFrom() {
coreService.core.removeListener(this);
}
@Override
public void onNavigatedTo() {
callPane.setText("Hello " + coreService.core.getDefaultProxyConfig().findAuthInfo().getUsername());
coreService.core.addListener(this);
if (coreService.core.getCurrentCall() != null) {
onCallStateChanged(coreService.core, coreService.core.getCurrentCall(),
coreService.core.getCurrentCall().getState(), null);
}
}
}

View File

@@ -0,0 +1,96 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* This file is part of Linphone Java Tutorial.
*
* 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/>.
*/
package controller;
import java.net.URL;
import java.util.ResourceBundle;
import javafx.fxml.FXML;
import javafx.fxml.Initializable;
import javafx.scene.control.Button;
import javafx.scene.control.Label;
import javafx.scene.control.PasswordField;
import javafx.scene.control.TextField;
import service.CoreService;
import org.linphone.core.Account;
import org.linphone.core.Core;
import org.linphone.core.CoreListenerStub;
import org.linphone.core.RegistrationState;
public class LoginController extends CoreListenerStub implements Initializable, ScreenInterface {
@FXML
private Button loginButton;
@FXML
private TextField identityField;
@FXML
private PasswordField passwordField;
@FXML
private Label registrationLabel;
private CoreService coreService = CoreService.instance();
@FXML
public void initialize(URL location, ResourceBundle resourceBundle) {
coreService.start();
}
@FXML
private void onLoginClicked() {
if (loginButton.isDisabled())
return;
loginButton.setDisable(true);
coreService.login(identityField.getText(), passwordField.getText());
}
@Override
public void onAccountRegistrationStateChanged(Core core, Account account, RegistrationState state, String message) {
registrationLabel.setText("Your registration state is: " + state.toString());
switch (state) {
case Cleared:
case None:
coreService.clearCoreAfterLogout();
loginButton.setDisable(false);
break;
case Ok:
loginButton.setDisable(true);
coreService.screenController.activate("call");
break;
case Progress:
loginButton.setDisable(true);
break;
case Failed:
loginButton.setDisable(false);
break;
default:
break;
}
}
@Override
public void onNavigatedFrom() {
coreService.core.removeListener(this);
}
@Override
public void onNavigatedTo() {
coreService.core.addListener(this);
}
}

View File

@@ -0,0 +1,66 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* This file is part of Linphone Java Tutorial.
*
* 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/>.
*/
package controller;
import java.io.IOException;
import java.util.HashMap;
import javafx.fxml.FXMLLoader;
import javafx.scene.Parent;
import javafx.scene.Scene;
class Screen {
public Parent root;
public FXMLLoader loader;
Screen(Parent root, FXMLLoader loader) {
this.root = root;
this.loader = loader;
}
}
public class ScreenController {
private HashMap<String, Screen> screenMap = new HashMap<>();
private Scene main;
private Screen currentScreen;
public ScreenController(Scene main) {
this.main = main;
}
public void add(String name, String viewFile) throws IOException {
FXMLLoader loader = new FXMLLoader(getClass().getResource(viewFile));
Parent root = loader.load();
screenMap.put(name, new Screen(root, loader));
}
public void remove(String name) {
screenMap.remove(name);
}
public void activate(String name) {
if (currentScreen != null) {
currentScreen.loader.<ScreenInterface>getController().onNavigatedFrom();
}
currentScreen = screenMap.get(name);
main.setRoot(currentScreen.root);
currentScreen.loader.<ScreenInterface>getController().onNavigatedTo();
}
}

View File

@@ -1,6 +1,8 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* This file is part of Linphone Java Tutorial.
*
* 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
@@ -15,15 +17,9 @@
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include <QCoreApplication>
package controller;
#include "App.hpp"
int main(int argc, char *argv[])
{
QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
App app(argc, argv);
app.init();
app.exec();
app.stop();
public interface ScreenInterface {
public void onNavigatedFrom();
public void onNavigatedTo();
}

View File

@@ -0,0 +1,172 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* This file is part of Linphone Java Tutorial.
*
* 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/>.
*/
package service;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.Timer;
import java.util.TimerTask;
import org.linphone.core.Account;
import org.linphone.core.AccountParams;
import org.linphone.core.Address;
import org.linphone.core.AuthInfo;
import org.linphone.core.Call;
import org.linphone.core.CallParams;
import org.linphone.core.Core;
import org.linphone.core.CoreListenerStub;
import org.linphone.core.Factory;
import org.linphone.core.GlobalState;
import org.linphone.core.LogLevel;
import org.linphone.core.LoggingService;
import org.linphone.core.VideoActivationPolicy;
import controller.ScreenController;
import javafx.application.Platform;
public class CoreService extends CoreListenerStub {
private static CoreService instance;
public Core core;
public ScreenController screenController;
public IterateRunnable iterateRunnable;
private Timer iterateTimer;
private CoreService() {
Factory factory = Factory.instance();
factory.setDataDir(".");
LoggingService loggingService = factory.getLoggingService();
loggingService.setLogLevel(LogLevel.Message);
core = factory.createCore("", "", null);
core.setAudioPort(7666);
core.setVideoPort(9666);
core.setUserCertificatesPath(Paths.get("").toString());
VideoActivationPolicy videoActivationPolicy = factory.createVideoActivationPolicy();
videoActivationPolicy.setAutomaticallyAccept(true);
videoActivationPolicy.setAutomaticallyInitiate(false);
core.setVideoActivationPolicy(videoActivationPolicy);
core.setVideoCaptureEnabled(true); // TODO: core.VideoSupported()
core.usePreviewWindow(true);
core.setVideoDisplayFilter("MSGLXVideo");
}
public static CoreService instance() {
if (instance == null) {
instance = new CoreService();
}
return instance;
}
public void start() {
this.core.start();
iterateRunnable = new IterateRunnable();
iterateTimer = new Timer();
iterateTimer.schedule(new IterateTimerTask(), 20, 20);
}
public void stop() {
core.stop();
}
public void login(String identity, String password) {
Address address = Factory.instance().createAddress(identity);
AuthInfo authInfo = Factory.instance().createAuthInfo(address.getUsername(), "", password, "", "",
address.getDomain());
core.addAuthInfo(authInfo);
AccountParams accountParams = core.createAccountParams();
accountParams.setIdentityAddress(address);
Address serverAddress = Factory.instance().createAddress("sip:" + address.getDomain() + ";transport=tls");
accountParams.setServerAddress(serverAddress);
accountParams.setRegisterEnabled(true);
Account account = core.createAccount(accountParams);
core.addAccount(account);
core.setDefaultAccount(account);
}
public void logout() {
Account account = core.getDefaultAccount();
if (account != null) {
AccountParams accountParams = account.getParams().clone();
accountParams.setRegisterEnabled(false);
account.setParams(accountParams);
}
}
// Make an outgoing call.
public void call(String uriToCall) {
// We create an Address object from the URI.
// This method can create a SIP Address from a username or phone number only.
Address address = core.interpretUrl(uriToCall);
// Initiate an outgoing call to the given destination Address.
core.inviteAddress(address);
}
public Boolean toggleCamera() {
Call call = core.getCurrentCall();
CallParams params = core.createCallParams(call);
Boolean newValue = !params.isVideoEnabled();
params.setVideoEnabled(newValue);
call.update(params);
return newValue;
}
public Boolean toggleMicrophone() {
core.setMicEnabled(!core.isMicEnabled());
return core.isMicEnabled();
}
public Boolean toggleSpeaker() {
core.getCurrentCall().setSpeakerMuted(!core.getCurrentCall().getSpeakerMuted());
return core.getCurrentCall().getSpeakerMuted();
}
public void clearCoreAfterLogout() {
core.clearAllAuthInfo();
core.clearAccounts();
}
@Override
public void onGlobalStateChanged(Core core, GlobalState state, String message) {
if (state == GlobalState.Off) {
iterateTimer.cancel();
}
}
}
class IterateTimerTask extends TimerTask {
@Override
public void run() {
Platform.runLater(CoreService.instance().iterateRunnable);
}
}
class IterateRunnable implements Runnable {
@Override
public void run() {
CoreService.instance().core.iterate();
}
}

View File

@@ -0,0 +1,71 @@
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.geometry.Insets?>
<?import javafx.scene.control.Button?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.control.TextField?>
<?import javafx.scene.control.TitledPane?>
<?import javafx.scene.layout.AnchorPane?>
<?import javafx.scene.layout.ColumnConstraints?>
<?import javafx.scene.layout.GridPane?>
<?import javafx.scene.layout.HBox?>
<?import javafx.scene.layout.RowConstraints?>
<?import javafx.scene.layout.VBox?>
<TitledPane fx:id="callPane" alignment="TOP_CENTER" animated="false" collapsible="false" maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" prefHeight="400.0" prefWidth="600.0" text="Hello" xmlns="http://javafx.com/javafx/17" xmlns:fx="http://javafx.com/fxml/1" fx:controller="controller.CallController">
<content>
<AnchorPane minHeight="0.0" minWidth="0.0" prefHeight="180.0" prefWidth="200.0">
<children>
<VBox alignment="CENTER" prefHeight="200.0" prefWidth="100.0" AnchorPane.bottomAnchor="0.0" AnchorPane.leftAnchor="0.0" AnchorPane.rightAnchor="0.0" AnchorPane.topAnchor="0.0">
<children>
<VBox alignment="CENTER" prefHeight="200.0" prefWidth="100.0">
<children>
<GridPane>
<columnConstraints>
<ColumnConstraints hgrow="SOMETIMES" minWidth="10.0" prefWidth="100.0" />
<ColumnConstraints hgrow="ALWAYS" minWidth="10.0" prefWidth="100.0" />
</columnConstraints>
<rowConstraints>
<RowConstraints minHeight="10.0" prefHeight="30.0" vgrow="SOMETIMES" />
</rowConstraints>
<children>
<Label alignment="CENTER" text="URI to call:" GridPane.halignment="CENTER" />
<TextField fx:id="uriToCallField" maxWidth="350.0" minWidth="350.0" prefWidth="350.0" promptText="sip:" GridPane.columnIndex="1" GridPane.halignment="CENTER" />
</children>
</GridPane>
<Button fx:id="callButton" mnemonicParsing="false" onMouseClicked="#onCallClicked" text="Call">
<VBox.margin>
<Insets bottom="30.0" top="20.0" />
</VBox.margin>
</Button>
<Label fx:id="callLabel" alignment="CENTER" text="Your call state is: Idle" />
<HBox alignment="CENTER" prefHeight="100.0" prefWidth="200.0" spacing="35.0">
<children>
<Button fx:id="hangupButton" disable="true" mnemonicParsing="false" onMouseClicked="#onHangUpClicked" text="Hang up" />
<Button fx:id="cameraButton" disable="true" mnemonicParsing="false" onMouseClicked="#onCameraClicked" text="Activate camera" />
<Button fx:id="muteSoundButton" disable="true" mnemonicParsing="false" onMouseClicked="#onMuteSoundClicked" text="Mute sound" />
<Button fx:id="muteMicrophoneButton" disable="true" mnemonicParsing="false" onMouseClicked="#onMuteMicrophoneClicked" text="Mute microphone" />
</children>
</HBox>
</children>
<VBox.margin>
<Insets top="20.0" />
</VBox.margin>
</VBox>
<VBox fx:id="incomingCallVBox" alignment="CENTER" prefHeight="200.0" prefWidth="100.0" visible="false">
<children>
<Label text="You have a call from:" />
<Label fx:id="incomingCallLabel" />
<HBox alignment="CENTER" prefHeight="100.0" prefWidth="200.0" spacing="35.0">
<children>
<Button fx:id="answerButton" mnemonicParsing="false" onMouseClicked="#onAnswerClicked" text="Answer" />
<Button fx:id="declineButton" mnemonicParsing="false" onMouseClicked="#onDeclineClicked" text="Decline" />
</children>
</HBox>
</children>
</VBox>
</children>
</VBox>
</children></AnchorPane>
</content>
</TitledPane>

View File

@@ -0,0 +1,57 @@
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.geometry.Insets?>
<?import javafx.scene.control.Button?>
<?import javafx.scene.control.ButtonBar?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.control.PasswordField?>
<?import javafx.scene.control.Separator?>
<?import javafx.scene.control.TextField?>
<?import javafx.scene.control.TitledPane?>
<?import javafx.scene.layout.AnchorPane?>
<?import javafx.scene.layout.ColumnConstraints?>
<?import javafx.scene.layout.FlowPane?>
<?import javafx.scene.layout.GridPane?>
<?import javafx.scene.layout.RowConstraints?>
<TitledPane alignment="TOP_CENTER" animated="false" collapsible="false" maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" prefHeight="400.0" prefWidth="600.0" text="Login Form" xmlns:fx="http://javafx.com/fxml/1" xmlns="http://javafx.com/javafx/17" fx:controller="controller.LoginController">
<content>
<AnchorPane minHeight="0.0" minWidth="0.0" prefHeight="180.0" prefWidth="200.0">
<children>
<FlowPane alignment="TOP_CENTER" columnHalignment="CENTER" layoutX="199.0" layoutY="54.0" orientation="VERTICAL" prefHeight="200.0" prefWidth="200.0" AnchorPane.leftAnchor="0.0" AnchorPane.rightAnchor="0.0" AnchorPane.topAnchor="0.0">
<children>
<GridPane>
<columnConstraints>
<ColumnConstraints hgrow="SOMETIMES" minWidth="10.0" prefWidth="100.0" />
<ColumnConstraints hgrow="ALWAYS" minWidth="10.0" prefWidth="350.0" />
</columnConstraints>
<rowConstraints>
<RowConstraints minHeight="10.0" prefHeight="30.0" vgrow="SOMETIMES" />
<RowConstraints minHeight="10.0" prefHeight="30.0" vgrow="SOMETIMES" />
</rowConstraints>
<children>
<Label text="Identity:" />
<TextField fx:id="identityField" maxWidth="350.0" minWidth="350.0" prefWidth="350.0" GridPane.columnIndex="1" />
<Label text="Password:" GridPane.rowIndex="1" />
<PasswordField fx:id="passwordField" maxWidth="350.0" minWidth="350.0" prefWidth="350.0" GridPane.columnIndex="1" GridPane.rowIndex="1" />
</children>
<FlowPane.margin>
<Insets bottom="20.0" top="20.0" />
</FlowPane.margin>
</GridPane>
<Label fx:id="registrationLabel" prefWidth="0.0">
<FlowPane.margin>
<Insets />
</FlowPane.margin>
</Label>
<Separator prefWidth="200.0" />
<ButtonBar prefHeight="40.0" prefWidth="200.0">
<buttons>
<Button fx:id="loginButton" mnemonicParsing="false" onMouseClicked="#onLoginClicked" text="Login" />
</buttons>
</ButtonBar>
</children>
</FlowPane>
</children></AnchorPane>
</content>
</TitledPane>

11
java/README.md Normal file
View File

@@ -0,0 +1,11 @@
Java tutorials
====================
Tutorials are numbered 0 to 3, and we recommend you to read them in that order as features from previous tutorials are used in the next ones, such as account login.
Each tutorial is a full project, so you should build it and run it independently.
For each tutorial, you need to extract the Linphone Java SDK zip file in the tutorial directory and run `./gradlew run` to execute it.
Code is being kept as short and simple as possible, and comments explain how and why things are being done.
Full Java API is available [here](http://linphone.org/snapshots/docs/liblinphone/latest/java).

View File

@@ -1 +0,0 @@
build/

View File

@@ -1,44 +0,0 @@
cmake_minimum_required(VERSION 3.22)
project(00_HelloWorld LANGUAGES CXX)
set(CMAKE_CXX_STANDARD 17)
set(CMAKE_CXX_STANDARD_REQUIRED ON)
find_package(Qt5 REQUIRED COMPONENTS Core Gui Qml Quick)
find_package(LinphoneCxx REQUIRED)
set(CMAKE_AUTOMOC ON)
SET(CMAKE_AUTOUIC ON)
set(SOURCES
"src/main.cpp"
"src/CoreManager.cpp"
)
set(QRC_RESOURCES resources.qrc)
set(QML_SOURCES)
file(STRINGS ${QRC_RESOURCES} QRC_RESOURCES_CONTENT)
foreach(line ${QRC_RESOURCES_CONTENT})
set(result)
string(REGEX REPLACE
"^[ \t]*<[ \t]*file[ \t]*>[ \t]*(.+\\.[a-z]+)[ \t]*<[ \t]*/[ \t]*file[ \t]*>[ \t]*$"
"\\1"
result
"${line}"
)
string(REGEX MATCH "\\.[a-z]+$" is_ui ${result})
if(NOT ${is_ui} STREQUAL "")
list(APPEND QML_SOURCES "${CMAKE_CURRENT_SOURCE_DIR}/${result}")
endif()
endforeach()
add_executable(00_HelloWorld ${SOURCES} ${QML_SOURCES} ${QRC_RESOURCES})
target_include_directories(00_HelloWorld PRIVATE ${LINPHONECXX_INCLUDE_DIRS})
target_link_libraries(00_HelloWorld PRIVATE Qt5::Core Qt5::Gui Qt5::Qml Qt5::Quick ${LINPHONECXX_LIBRARIES})
set_target_properties(00_HelloWorld PROPERTIES AUTORCC ON)
set_target_properties(00_HelloWorld PROPERTIES
WIN32_EXECUTABLE ON
MACOSX_BUNDLE ON
)

View File

@@ -1,15 +0,0 @@
# Hello World tutorial
The purpose of this tutorial is to explain how to build a Qt app depending on the Linphone SDK and to create the `Core` object that all our APIs depends on.
The user interface will only display the `Core`'s version, but in the next tutorial you will learn how to use it to login your SIP account.
## How to build
In the following instructions, replace **<PATH-TO-SDK>** by the real path where your SDK is located, e.g. *~/projects/linphone-sdk/build-default/linphone-sdk/desktop/*
mkdir build
cd build
cmake .. -DCMAKE_PREFIX_PATH=<PATH-TO-SDK>
cmake --build .

View File

@@ -1,5 +0,0 @@
<RCC>
<qresource prefix="/">
<file>ui/MainPage.qml</file>
</qresource>
</RCC>

View File

@@ -1,32 +0,0 @@
#include "CoreManager.hpp"
CoreManager *CoreManager::mInstance = nullptr;
CoreManager::CoreManager()
{
// Create a core from the factory.
mCore = linphone::Factory::get()->createCore("", "", nullptr);
}
CoreManager::~CoreManager()
{
mCore = nullptr;
}
void CoreManager::init()
{
if (mInstance)
return;
mInstance = new CoreManager();
}
CoreManager *CoreManager::getInstance()
{
return mInstance;
}
QString CoreManager::getVersion() const
{
// Get the version from the core.
return QString::fromStdString(mCore->getVersion());
}

View File

@@ -1,25 +0,0 @@
#pragma once
#include <QtCore>
#include <linphone++/linphone.hh>
class CoreManager : public QObject
{
Q_OBJECT
Q_PROPERTY(QString version READ getVersion CONSTANT)
public:
static void init();
static CoreManager *getInstance();
QString getVersion() const;
private:
CoreManager();
~CoreManager();
std::shared_ptr<linphone::Core> mCore = nullptr;
static CoreManager *mInstance;
};

View File

@@ -1,28 +0,0 @@
#include <QDir>
#include <QGuiApplication>
#include <QQmlApplicationEngine>
#include <QQmlContext>
#include "CoreManager.hpp"
int main(int argc, char *argv[])
{
QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
QGuiApplication app(argc, argv);
app.setOrganizationName("Belledonne Communications");
app.setOrganizationDomain("belledonne-communications.com");
app.setApplicationName(QFileInfo(app.applicationFilePath()).baseName());
QQmlApplicationEngine engine;
engine.load(QUrl("qrc:/ui/MainPage.qml"));
if (engine.rootObjects().isEmpty())
qFatal("Unable to open main window.");
// Initialize the CoreManager singleton and add it to the Qml context.
CoreManager::init();
auto coreManager = CoreManager::getInstance();
QQmlContext *ctx = engine.rootContext();
ctx->setContextProperty("coreManager", coreManager);
return app.exec();
}

View File

@@ -1,18 +0,0 @@
import QtQuick 2.9
import QtQuick.Controls 2.2
ApplicationWindow {
id: window
visible: true
title: "Hello World"
width: 640
height: 480
Text {
id: versionText
text: "Hello world, Linphone core version is " + coreManager.version // Get the version from the core manager.
anchors.horizontalCenter: parent.horizontalCenter
anchors.verticalCenter: parent.verticalCenter
}
}

View File

@@ -1 +0,0 @@
build/

View File

@@ -1,52 +0,0 @@
cmake_minimum_required(VERSION 3.22)
project(01_AccountLogin LANGUAGES CXX)
set(CMAKE_CXX_STANDARD 17)
set(CMAKE_CXX_STANDARD_REQUIRED ON)
find_package(Qt5 REQUIRED COMPONENTS Core Gui Qml Quick)
find_package(LinphoneCxx REQUIRED)
set(CMAKE_AUTOMOC ON)
SET(CMAKE_AUTOUIC ON)
set(SOURCES
"src/main.cpp"
"src/App.cpp"
"src/CoreHandler.cpp"
"src/CoreListener.cpp"
"src/CoreManager.cpp"
)
set(QRC_RESOURCES resources.qrc)
set(QML_SOURCES)
file(STRINGS ${QRC_RESOURCES} QRC_RESOURCES_CONTENT)
foreach(line ${QRC_RESOURCES_CONTENT})
set(result)
string(REGEX REPLACE
"^[ \t]*<[ \t]*file[ \t]*>[ \t]*(.+\\.[a-z]+)[ \t]*<[ \t]*/[ \t]*file[ \t]*>[ \t]*$"
"\\1"
result
"${line}"
)
string(REGEX MATCH "\\.[a-z]+$" is_ui ${result})
if(NOT ${is_ui} STREQUAL "")
list(APPEND QML_SOURCES "${CMAKE_CURRENT_SOURCE_DIR}/${result}")
endif()
endforeach()
get_filename_component(SDK_PATH "${CMAKE_PREFIX_PATH}" REALPATH)
find_path(MSPLUGINS_PATH "plugins" PATH_SUFFIXES "lib/mediastreamer" "lib64/mediastreamer" REQUIRED)
set(MSPLUGINS_PATH "${MSPLUGINS_PATH}/plugins")
add_executable(01_AccountLogin ${SOURCES} ${QML_SOURCES} ${QRC_RESOURCES})
target_compile_definitions(01_AccountLogin PRIVATE "SDK_PATH=\"${SDK_PATH}\"" "MSPLUGINS_PATH=\"${MSPLUGINS_PATH}\"")
target_include_directories(01_AccountLogin PRIVATE ${LINPHONECXX_INCLUDE_DIRS})
target_link_libraries(01_AccountLogin PRIVATE Qt5::Core Qt5::Gui Qt5::Qml Qt5::Quick ${LINPHONECXX_LIBRARIES})
set_target_properties(01_AccountLogin PROPERTIES AUTORCC ON)
set_target_properties(01_AccountLogin PROPERTIES
WIN32_EXECUTABLE ON
MACOSX_BUNDLE ON
)

View File

@@ -1,15 +0,0 @@
# Account Login tutorial
This project will walk you through the different steps of logging in and out of a SIP account.
If you do not yet have a SIP account, please create one here : https://www.linphone.org/freesip/home
## How to build
In the following instructions, replace **<PATH-TO-SDK>** by the real path where your SDK is located, e.g. *~/projects/linphone-sdk/build-default/linphone-sdk/desktop/*
mkdir build
cd build
cmake .. -DCMAKE_PREFIX_PATH=<PATH-TO-SDK>
cmake --build .

View File

@@ -1,6 +0,0 @@
<RCC>
<qresource prefix="/">
<file>ui/MainPage.qml</file>
<file>ui/RegistrationPage.qml</file>
</qresource>
</RCC>

View File

@@ -1,47 +0,0 @@
#include <QDir>
#include <QQmlContext>
#include "App.hpp"
#include "CoreManager.hpp"
using namespace std;
using namespace linphone;
App::App(int &argc, char *argv[]) : QGuiApplication(argc, argv)
{
setOrganizationName("Belledonne Communications");
setOrganizationDomain("belledonne-communications.com");
setApplicationName(QFileInfo(applicationFilePath()).baseName());
}
App::~App()
{
}
void App::init()
{
registerTypes();
mEngine = new QQmlApplicationEngine();
mEngine->load(QUrl("qrc:/ui/MainPage.qml"));
if (mEngine->rootObjects().isEmpty())
qFatal("Unable to open main window.");
// Initialize the CoreManager singleton and add it to the Qml context.
CoreManager::init(this);
auto coreManager = CoreManager::getInstance();
QQmlContext *ctx = mEngine->rootContext();
ctx->setContextProperty("coreManager", coreManager);
}
void App::stop()
{
CoreManager::uninit();
}
void App::registerTypes()
{
qRegisterMetaType<string>();
qRegisterMetaType<RegistrationState>();
qRegisterMetaType<shared_ptr<Account>>();
}

View File

@@ -1,25 +0,0 @@
#include <QGuiApplication>
#include <QQmlApplicationEngine>
#include <linphone++/linphone.hh>
class App : public QGuiApplication
{
Q_OBJECT
public:
App(int &argc, char *argv[]);
~App();
void init();
void stop();
private:
void registerTypes();
QQmlApplicationEngine *mEngine = nullptr;
};
Q_DECLARE_METATYPE(std::string);
Q_DECLARE_METATYPE(linphone::RegistrationState);
Q_DECLARE_METATYPE(std::shared_ptr<linphone::Account>);

View File

@@ -1,69 +0,0 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* 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/>.
*/
#include <QDebug>
#include "CoreHandler.hpp"
#include "CoreListener.hpp"
#include "CoreManager.hpp"
CoreHandler::CoreHandler()
{
mCoreListener = std::make_shared<CoreListener>();
connectTo(mCoreListener.get());
}
void CoreHandler::setListener(std::shared_ptr<linphone::Core> core)
{
core->addListener(mCoreListener);
}
void CoreHandler::removeListener(std::shared_ptr<linphone::Core> core)
{
core->removeListener(mCoreListener);
}
void CoreHandler::onAccountRegistrationStateChanged(const std::shared_ptr<linphone::Core> &, const std::shared_ptr<linphone::Account> &account, linphone::RegistrationState state, const std::string &message)
{
emit registrationStateChanged(account, state, message);
}
void CoreHandler::onGlobalStateChanged(const std::shared_ptr<linphone::Core> &, linphone::GlobalState state, const std::string &message)
{
switch (state)
{
case linphone::GlobalState::On:
qInfo() << "Core is running " << QString::fromStdString(message);
break;
case linphone::GlobalState::Off:
qInfo() << "Core is stopped " << QString::fromStdString(message);
emit coreStopped();
break;
case linphone::GlobalState::Startup:
qInfo() << "Core is starting" << QString::fromStdString(message);
emit coreStarting();
break;
default:
break;
}
}
void CoreHandler::connectTo(CoreListener *listener)
{
connect(listener, &CoreListener::accountRegistrationStateChanged, this, &CoreHandler::onAccountRegistrationStateChanged);
connect(listener, &CoreListener::globalStateChanged, this, &CoreHandler::onGlobalStateChanged);
}

View File

@@ -1,47 +0,0 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* 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/>.
*/
#pragma once
#include <linphone++/linphone.hh>
#include <QObject>
class CoreListener;
class CoreHandler : public QObject
{
Q_OBJECT
public:
CoreHandler();
void setListener(std::shared_ptr<linphone::Core> core);
void removeListener(std::shared_ptr<linphone::Core> core);
signals:
void coreStarting();
void coreStopped();
void registrationStateChanged(const std::shared_ptr<linphone::Account> &account, linphone::RegistrationState state, const std::string &message);
public slots:
void onAccountRegistrationStateChanged(const std::shared_ptr<linphone::Core> &core, const std::shared_ptr<linphone::Account> &account, linphone::RegistrationState state, const std::string &message);
void onGlobalStateChanged(const std::shared_ptr<linphone::Core> &core, linphone::GlobalState gstate, const std::string &message);
private:
void connectTo(CoreListener *listener);
std::shared_ptr<CoreListener> mCoreListener;
};

View File

@@ -1,32 +0,0 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* 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/>.
*/
#include "CoreListener.hpp"
CoreListener::CoreListener(QObject *parent) : QObject(parent)
{
}
void CoreListener::onAccountRegistrationStateChanged(const std::shared_ptr<linphone::Core> &core, const std::shared_ptr<linphone::Account> &account, linphone::RegistrationState state, const std::string &message)
{
emit accountRegistrationStateChanged(core, account, state, message);
}
void CoreListener::onGlobalStateChanged(const std::shared_ptr<linphone::Core> &core, linphone::GlobalState gstate, const std::string &message)
{
emit globalStateChanged(core, gstate, message);
}

View File

@@ -1,37 +0,0 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* 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/>.
*/
#pragma once
#include <linphone++/linphone.hh>
#include <QObject>
class CoreListener : public QObject, public linphone::CoreListener
{
Q_OBJECT
public:
CoreListener(QObject *parent = nullptr);
virtual ~CoreListener() = default;
virtual void onAccountRegistrationStateChanged(const std::shared_ptr<linphone::Core> &core, const std::shared_ptr<linphone::Account> &account, linphone::RegistrationState state, const std::string &message) override;
virtual void onGlobalStateChanged(const std::shared_ptr<linphone::Core> &core, linphone::GlobalState gstate, const std::string &message) override;
signals:
void accountRegistrationStateChanged(const std::shared_ptr<linphone::Core> &core, const std::shared_ptr<linphone::Account> &account, linphone::RegistrationState state, const std::string &message);
void globalStateChanged(const std::shared_ptr<linphone::Core> &core, linphone::GlobalState gstate, const std::string &message);
};

View File

@@ -1,235 +0,0 @@
#include <QDebug>
#include "CoreHandler.hpp"
#include "CoreManager.hpp"
using namespace std;
using namespace linphone;
CoreManager *CoreManager::mInstance = nullptr;
CoreManager::CoreManager(QObject *parent) : QObject(parent)
{
mHandler = QSharedPointer<CoreHandler>::create();
CoreHandler *coreHandler = mHandler.get();
QObject::connect(coreHandler, &CoreHandler::coreStarting, this, &CoreManager::startIterate, Qt::QueuedConnection);
QObject::connect(coreHandler, &CoreHandler::coreStopped, this, &CoreManager::stopIterate, Qt::QueuedConnection);
QObject::connect(coreHandler, &CoreHandler::registrationStateChanged, this, &CoreManager::onRegistrationStateChanged, Qt::QueuedConnection);
// Delay the creation of the core so that the CoreManager instance is
// already set.
QTimer::singleShot(10, [this]()
{ createLinphoneCore(); });
}
CoreManager::~CoreManager()
{
mHandler->removeListener(mCore);
mHandler = nullptr;
mCore = nullptr;
}
void CoreManager::init(QObject *parent)
{
if (mInstance)
return;
mInstance = new CoreManager(parent);
}
void CoreManager::uninit()
{
if (mInstance)
{
mInstance->stopIterate();
auto core = mInstance->mCore;
delete mInstance;
mInstance = nullptr;
core->stop();
}
}
CoreManager *CoreManager::getInstance()
{
return mInstance;
}
void CoreManager::startIterate()
{
// Start a timer to call the core iterate every 20 ms.
mIterateTimer = new QTimer(this);
mIterateTimer->setInterval(20);
QObject::connect(mIterateTimer, &QTimer::timeout, this, &CoreManager::iterate);
qInfo() << QStringLiteral("Start iterate");
mIterateTimer->start();
}
void CoreManager::stopIterate()
{
qInfo() << QStringLiteral("Stop iterate");
mIterateTimer->stop();
mIterateTimer->deleteLater(); // Allow the timer to continue its stuff
mIterateTimer = nullptr;
}
void CoreManager::login(QString identity, QString password, QString transport)
{
if (mLoginButtonEnabled)
{
setProperty("loginButtonEnabled", false);
// To configure a SIP account, we need an Account object and an AuthInfo object
// The first one is how to connect to the proxy server, the second one stores the credentials
// Here we are creating an AuthInfo object from the identity Address and password provided by the user.
shared_ptr<Address> address = Factory::get()->createAddress(identity.toStdString());
// The AuthInfo can be created from the Factory as it's only a data class
// userID is set to null as it's the same as the username in our case
// ha1 is set to null as we are using the clear text password. Upon first register, the hash will be computed automatically.
// The realm will be determined automatically from the first register, as well as the algorithm
shared_ptr<AuthInfo> authInfo = Factory::get()->createAuthInfo(address->getUsername(), "", password.toStdString(), "", "", address->getDomain());
// And we add it to the Core
mCore->addAuthInfo(authInfo);
// Then we create an AccountParams object.
// It contains the account informations needed by the core
shared_ptr<AccountParams> accountParams = mCore->createAccountParams();
// A SIP account is identified by an identity address that we can construct from the username and domain
accountParams->setIdentityAddress(address);
// We also need to configure where the proxy server is located
shared_ptr<Address> serverAddr = Factory::get()->createAddress("sip:" + address->getDomain());
// We use the Address object to easily set the transport protocol
if (transport == "tls")
{
serverAddr->setTransport(TransportType::Tls);
}
else if (transport == "tcp")
{
serverAddr->setTransport(TransportType::Tcp);
}
else
{
serverAddr->setTransport(TransportType::Udp);
}
accountParams->setServerAddress(serverAddr);
// If enableRegister is set to true, when this account will be added to the core it will
// automatically try to connect.
accountParams->enableRegister(true);
// We can now create an Account object from the AccountParams ...
shared_ptr<Account> account = mCore->createAccount(accountParams);
// ... and add it to the core, launching the connection process.
mCore->addAccount(account);
// Also set the newly added account as default
mCore->setDefaultAccount(account);
}
}
void CoreManager::logout()
{
if (mLogoutButtonEnabled)
{
setProperty("logoutButtonEnabled", false);
// Setting enableRegister to false on a connected Account object will
// launch the logout action.
shared_ptr<Account> account = mCore->getDefaultAccount();
if (account)
{
// BUT BE CAREFUL : the Params of an account are read-only
// You MUST Clone it :
shared_ptr<AccountParams> accountParams = account->getParams()->clone();
// Then you can modify the clone :
accountParams->enableRegister(false);
// And finally setting the new Params value triggers the changes, here the logout.
account->setParams(accountParams);
}
}
}
void CoreManager::onRegistrationStateChanged(const shared_ptr<Account> &account, RegistrationState state, const string &message)
{
setProperty("registerText", QString::fromStdString("Your registration state is : " + message));
switch (state)
{
// If the Account was logged out, we clear the Core.
case RegistrationState::Cleared:
case RegistrationState::None:
mCore->clearAllAuthInfo();
mCore->clearAccounts();
logoutGuiChanges();
break;
case RegistrationState::Ok:
loginGuiChanges();
break;
case RegistrationState::Progress:
loginInProgressGuiChanges();
break;
case RegistrationState::Failed:
loginFailedGuiChanges();
break;
default:
break;
}
}
void CoreManager::logoutGuiChanges()
{
setProperty("loginButtonEnabled", true);
setProperty("logoutButtonEnabled", false);
setProperty("loginText", "You are logged out");
}
void CoreManager::loginFailedGuiChanges()
{
setProperty("loginButtonEnabled", true);
setProperty("logoutButtonEnabled", false);
setProperty("loginText", "Login failed, try again");
}
void CoreManager::loginGuiChanges()
{
setProperty("loginButtonEnabled", false);
setProperty("logoutButtonEnabled", true);
setProperty("loginText", QString::fromStdString("You are logged in, with identity " + mCore->getIdentity() + "."));
}
void CoreManager::loginInProgressGuiChanges()
{
setProperty("loginButtonEnabled", false);
setProperty("logoutButtonEnabled", false);
setProperty("loginText", QString::fromStdString("Login in progress, with identity " + mCore->getIdentity() + "."));
}
void CoreManager::createLinphoneCore()
{
// Setting linphone log level to message.
auto loggingService = LoggingService::get();
loggingService->setLogLevel(LogLevel::Message);
// Configure paths.
string assetsPath = string(SDK_PATH) + "/share";
Factory::get()->setTopResourcesDir(assetsPath);
Factory::get()->setDataResourcesDir(assetsPath);
Factory::get()->setSoundResourcesDir(assetsPath + "/sounds/linphone");
Factory::get()->setRingResourcesDir(Factory::get()->getSoundResourcesDir() + "/rings");
Factory::get()->setImageResourcesDir(assetsPath + "/images");
Factory::get()->setMspluginsDir(MSPLUGINS_PATH);
// Create a core from the factory.
mCore = Factory::get()->createCore("", "", nullptr);
mCore->setRootCa(assetsPath + "/linphone/rootca.pem");
// Listen for core events.
mHandler->setListener(mCore);
// Start the core.
mCore->start();
}
void CoreManager::iterate()
{
if (mCore)
mCore->iterate();
}

View File

@@ -1,60 +0,0 @@
#pragma once
#include <QObject>
#include <QSharedPointer>
#include <QTimer>
#include <linphone++/linphone.hh>
class CoreHandler;
class CoreManager : public QObject
{
Q_OBJECT
Q_PROPERTY(QString loginText MEMBER mLoginText NOTIFY loginTextChanged)
Q_PROPERTY(QString registerText MEMBER mRegisterText NOTIFY registerTextChanged)
Q_PROPERTY(bool loginButtonEnabled MEMBER mLoginButtonEnabled NOTIFY loginButtonEnabledChanged)
Q_PROPERTY(bool logoutButtonEnabled MEMBER mLogoutButtonEnabled NOTIFY logoutButtonEnabledChanged)
public:
static void init(QObject *parent);
static void uninit();
static CoreManager *getInstance();
signals:
void loginTextChanged(QString);
void registerTextChanged(QString);
void loginButtonEnabledChanged(bool);
void logoutButtonEnabledChanged(bool);
public slots:
void startIterate();
void stopIterate();
void login(QString identity, QString password, QString transport);
void logout();
void onRegistrationStateChanged(const std::shared_ptr<linphone::Account> &account, linphone::RegistrationState state, const std::string &message);
private:
CoreManager(QObject *parent);
~CoreManager();
void logoutGuiChanges();
void loginFailedGuiChanges();
void loginGuiChanges();
void loginInProgressGuiChanges();
void createLinphoneCore();
void iterate();
std::shared_ptr<linphone::Core> mCore = nullptr;
QSharedPointer<CoreHandler> mHandler;
QTimer *mIterateTimer = nullptr;
QString mLoginText;
QString mRegisterText;
bool mLoginButtonEnabled = true;
bool mLogoutButtonEnabled = false;
static CoreManager *mInstance;
};

View File

@@ -1,19 +0,0 @@
import QtQuick 2.9
import QtQuick.Controls 2.2
import QtQuick.Layouts 1.3
ApplicationWindow {
id: window
visible: true
title: "Account Login"
width: 640
height: 480
// Main content
Loader {
id: contentLoader
anchors.fill: parent
source: 'qrc:/ui/RegistrationPage.qml'
}
}

View File

@@ -1,95 +0,0 @@
import QtQuick 2.9
import QtQuick.Controls 2.2
import QtQuick.Layouts 1.3
ColumnLayout {
Layout.alignment: Qt.AlignHCenter
Layout.fillWidth: true
Layout.fillHeight: true
GridLayout {
Layout.fillWidth: true
Layout.margins: 20
columnSpacing: 20
columns: 2
Text {
text: "Identity:"
}
TextField {
id: identityTextField
Layout.fillWidth: true
text: "sip:"
}
Text {
text: "Password:"
}
TextField {
id: passwordTextField
echoMode: TextInput.Password
Layout.fillWidth: true
placeholderText: "my password"
}
Text {
text: "Transport:"
}
RowLayout {
Layout.fillWidth: true
RadioButton {
id: tlsButton
text: "TLS"
checked: true
}
RadioButton {
id: tcpButton
text: "TCP"
}
RadioButton {
id: udpButton
text: "UDP"
}
}
}
RowLayout {
Layout.fillWidth: true
Layout.alignment: Qt.AlignHCenter
spacing: 20
Button {
id: loginButton
text: "Login"
enabled: coreManager.loginButtonEnabled && identityTextField.text.length != 0 && passwordTextField.text.length != 0
onClicked: {
var transport = "tls"
if (tcpButton.checked) { transport = "tcp" }
else if (udpButton.checked) { transport = "udp" }
coreManager.login(identityTextField.text, passwordTextField.text, transport)
}
}
Button {
id: logoutButton
text: "Logout"
enabled: coreManager.logoutButtonEnabled
onClicked: {
coreManager.logout()
}
}
}
ColumnLayout {
Layout.fillHeight: true
Layout.alignment: Qt.AlignHCenter
Text {
id: loginText
text: coreManager.loginText
}
Text {
id: registrationText
text: coreManager.registerText
}
}
}

View File

@@ -1 +0,0 @@
build/

View File

@@ -1,52 +0,0 @@
cmake_minimum_required(VERSION 3.22)
project(02_IncomingCall LANGUAGES CXX)
set(CMAKE_CXX_STANDARD 17)
set(CMAKE_CXX_STANDARD_REQUIRED ON)
find_package(Qt5 REQUIRED COMPONENTS Core Gui Qml Quick)
find_package(LinphoneCxx REQUIRED)
set(CMAKE_AUTOMOC ON)
SET(CMAKE_AUTOUIC ON)
set(SOURCES
"src/main.cpp"
"src/App.cpp"
"src/CoreHandler.cpp"
"src/CoreListener.cpp"
"src/CoreManager.cpp"
)
set(QRC_RESOURCES resources.qrc)
set(QML_SOURCES)
file(STRINGS ${QRC_RESOURCES} QRC_RESOURCES_CONTENT)
foreach(line ${QRC_RESOURCES_CONTENT})
set(result)
string(REGEX REPLACE
"^[ \t]*<[ \t]*file[ \t]*>[ \t]*(.+\\.[a-z]+)[ \t]*<[ \t]*/[ \t]*file[ \t]*>[ \t]*$"
"\\1"
result
"${line}"
)
string(REGEX MATCH "\\.[a-z]+$" is_ui ${result})
if(NOT ${is_ui} STREQUAL "")
list(APPEND QML_SOURCES "${CMAKE_CURRENT_SOURCE_DIR}/${result}")
endif()
endforeach()
get_filename_component(SDK_PATH "${CMAKE_PREFIX_PATH}" REALPATH)
find_path(MSPLUGINS_PATH "plugins" PATH_SUFFIXES "lib/mediastreamer" "lib64/mediastreamer" REQUIRED)
set(MSPLUGINS_PATH "${MSPLUGINS_PATH}/plugins")
add_executable(02_IncomingCall ${SOURCES} ${QML_SOURCES} ${QRC_RESOURCES})
target_compile_definitions(02_IncomingCall PRIVATE "SDK_PATH=\"${SDK_PATH}\"" "MSPLUGINS_PATH=\"${MSPLUGINS_PATH}\"")
target_include_directories(02_IncomingCall PRIVATE ${LINPHONECXX_INCLUDE_DIRS})
target_link_libraries(02_IncomingCall PRIVATE Qt5::Core Qt5::Gui Qt5::Qml Qt5::Quick ${LINPHONECXX_LIBRARIES})
set_target_properties(02_IncomingCall PROPERTIES AUTORCC ON)
set_target_properties(02_IncomingCall PROPERTIES
WIN32_EXECUTABLE ON
MACOSX_BUNDLE ON
)

View File

@@ -1,15 +0,0 @@
# Incoming call tutorial
This time we are going to receive our first calls!
If you don't have SIP friends to test with, you can also install Linphone on your mobile device (Android or iOS) and call yourself with a different account.
## How to build
In the following instructions, replace **<PATH-TO-SDK>** by the real path where your SDK is located, e.g. *~/projects/linphone-sdk/build-default/linphone-sdk/desktop/*
mkdir build
cd build
cmake .. -DCMAKE_PREFIX_PATH=<PATH-TO-SDK>
cmake --build .

View File

@@ -1,7 +0,0 @@
<RCC>
<qresource prefix="/">
<file>ui/MainPage.qml</file>
<file>ui/CallPage.qml</file>
<file>ui/RegistrationPage.qml</file>
</qresource>
</RCC>

View File

@@ -1,49 +0,0 @@
#include <QDir>
#include <QQmlContext>
#include "App.hpp"
#include "CoreManager.hpp"
using namespace std;
using namespace linphone;
App::App(int &argc, char *argv[]) : QGuiApplication(argc, argv)
{
setOrganizationName("Belledonne Communications");
setOrganizationDomain("belledonne-communications.com");
setApplicationName(QFileInfo(applicationFilePath()).baseName());
}
App::~App()
{
}
void App::init()
{
registerTypes();
mEngine = new QQmlApplicationEngine();
mEngine->load(QUrl("qrc:/ui/MainPage.qml"));
if (mEngine->rootObjects().isEmpty())
qFatal("Unable to open main window.");
// Initialize the CoreManager singleton and add it to the Qml context.
CoreManager::init(this);
auto coreManager = CoreManager::getInstance();
QQmlContext *ctx = mEngine->rootContext();
ctx->setContextProperty("coreManager", coreManager);
}
void App::stop()
{
CoreManager::uninit();
}
void App::registerTypes()
{
qRegisterMetaType<string>();
qRegisterMetaType<RegistrationState>();
qRegisterMetaType<shared_ptr<Account>>();
qRegisterMetaType<Call::State>();
qRegisterMetaType<shared_ptr<Call>>();
}

View File

@@ -1,27 +0,0 @@
#include <QGuiApplication>
#include <QQmlApplicationEngine>
#include <linphone++/linphone.hh>
class App : public QGuiApplication
{
Q_OBJECT
public:
App(int &argc, char *argv[]);
~App();
void init();
void stop();
private:
void registerTypes();
QQmlApplicationEngine *mEngine = nullptr;
};
Q_DECLARE_METATYPE(std::string);
Q_DECLARE_METATYPE(linphone::RegistrationState);
Q_DECLARE_METATYPE(std::shared_ptr<linphone::Account>);
Q_DECLARE_METATYPE(linphone::Call::State);
Q_DECLARE_METATYPE(std::shared_ptr<linphone::Call>);

View File

@@ -1,75 +0,0 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* 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/>.
*/
#include <QDebug>
#include "CoreHandler.hpp"
#include "CoreListener.hpp"
#include "CoreManager.hpp"
CoreHandler::CoreHandler()
{
mCoreListener = std::make_shared<CoreListener>();
connectTo(mCoreListener.get());
}
void CoreHandler::setListener(std::shared_ptr<linphone::Core> core)
{
core->addListener(mCoreListener);
}
void CoreHandler::removeListener(std::shared_ptr<linphone::Core> core)
{
core->removeListener(mCoreListener);
}
void CoreHandler::onAccountRegistrationStateChanged(const std::shared_ptr<linphone::Core> &, const std::shared_ptr<linphone::Account> &account, linphone::RegistrationState state, const std::string &message)
{
emit registrationStateChanged(account, state, message);
}
void CoreHandler::onCallStateStateChanged(const std::shared_ptr<linphone::Core> &, const std::shared_ptr<linphone::Call> &call, linphone::Call::State state, const std::string &message)
{
emit callStateChanged(call, state, message);
}
void CoreHandler::onGlobalStateChanged(const std::shared_ptr<linphone::Core> &, linphone::GlobalState state, const std::string &message)
{
switch (state)
{
case linphone::GlobalState::On:
qInfo() << "Core is running " << QString::fromStdString(message);
break;
case linphone::GlobalState::Off:
qInfo() << "Core is stopped " << QString::fromStdString(message);
emit coreStopped();
break;
case linphone::GlobalState::Startup:
qInfo() << "Core is starting" << QString::fromStdString(message);
emit coreStarting();
break;
default:
break;
}
}
void CoreHandler::connectTo(CoreListener *listener)
{
connect(listener, &CoreListener::accountRegistrationStateChanged, this, &CoreHandler::onAccountRegistrationStateChanged);
connect(listener, &CoreListener::callStateChanged, this, &CoreHandler::onCallStateStateChanged);
connect(listener, &CoreListener::globalStateChanged, this, &CoreHandler::onGlobalStateChanged);
}

View File

@@ -1,49 +0,0 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* 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/>.
*/
#pragma once
#include <linphone++/linphone.hh>
#include <QObject>
class CoreListener;
class CoreHandler : public QObject
{
Q_OBJECT
public:
CoreHandler();
void setListener(std::shared_ptr<linphone::Core> core);
void removeListener(std::shared_ptr<linphone::Core> core);
signals:
void coreStarting();
void coreStopped();
void callStateChanged(const std::shared_ptr<linphone::Call>, linphone::Call::State state, const std::string &message);
void registrationStateChanged(const std::shared_ptr<linphone::Account> &account, linphone::RegistrationState state, const std::string &message);
public slots:
void onAccountRegistrationStateChanged(const std::shared_ptr<linphone::Core> &core, const std::shared_ptr<linphone::Account> &account, linphone::RegistrationState state, const std::string &message);
void onCallStateStateChanged(const std::shared_ptr<linphone::Core> &core, const std::shared_ptr<linphone::Call> &call, linphone::Call::State state, const std::string &message);
void onGlobalStateChanged(const std::shared_ptr<linphone::Core> &core, linphone::GlobalState gstate, const std::string &message);
private:
void connectTo(CoreListener *listener);
std::shared_ptr<CoreListener> mCoreListener;
};

View File

@@ -1,37 +0,0 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* 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/>.
*/
#include "CoreListener.hpp"
CoreListener::CoreListener(QObject *parent) : QObject(parent)
{
}
void CoreListener::onAccountRegistrationStateChanged(const std::shared_ptr<linphone::Core> &core, const std::shared_ptr<linphone::Account> &account, linphone::RegistrationState state, const std::string &message)
{
emit accountRegistrationStateChanged(core, account, state, message);
}
void CoreListener::onCallStateChanged(const std::shared_ptr<linphone::Core> &core, const std::shared_ptr<linphone::Call> &call, linphone::Call::State state, const std::string &message)
{
emit callStateChanged(core, call, state, message);
}
void CoreListener::onGlobalStateChanged(const std::shared_ptr<linphone::Core> &core, linphone::GlobalState gstate, const std::string &message)
{
emit globalStateChanged(core, gstate, message);
}

View File

@@ -1,39 +0,0 @@
/*
* Copyright (c) 2010-2024 Belledonne Communications SARL.
*
* 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/>.
*/
#pragma once
#include <linphone++/linphone.hh>
#include <QObject>
class CoreListener : public QObject, public linphone::CoreListener
{
Q_OBJECT
public:
CoreListener(QObject *parent = nullptr);
virtual ~CoreListener() = default;
virtual void onAccountRegistrationStateChanged(const std::shared_ptr<linphone::Core> &core, const std::shared_ptr<linphone::Account> &account, linphone::RegistrationState state, const std::string &message) override;
virtual void onCallStateChanged(const std::shared_ptr<linphone::Core> &core, const std::shared_ptr<linphone::Call> &call, linphone::Call::State state, const std::string &message) override;
virtual void onGlobalStateChanged(const std::shared_ptr<linphone::Core> &core, linphone::GlobalState gstate, const std::string &message) override;
signals:
void accountRegistrationStateChanged(const std::shared_ptr<linphone::Core> &core, const std::shared_ptr<linphone::Account> &account, linphone::RegistrationState state, const std::string &message);
void callStateChanged(const std::shared_ptr<linphone::Core> &core, const std::shared_ptr<linphone::Call> &call, linphone::Call::State state, const std::string &message);
void globalStateChanged(const std::shared_ptr<linphone::Core> &core, linphone::GlobalState gstate, const std::string &message);
};

View File

@@ -1,285 +0,0 @@
#include <QDebug>
#include "CoreHandler.hpp"
#include "CoreManager.hpp"
using namespace std;
using namespace linphone;
CoreManager *CoreManager::mInstance = nullptr;
CoreManager::CoreManager(QObject *parent) : QObject(parent)
{
mPage = QString("qrc:/ui/RegistrationPage.qml");
mHeaderText = QString("Registration Form");
mSoundButtonText = QString("Switch off sound");
mMicrophoneButtonText = QString("Mute");
mHandler = QSharedPointer<CoreHandler>::create();
CoreHandler *coreHandler = mHandler.get();
QObject::connect(coreHandler, &CoreHandler::coreStarting, this, &CoreManager::startIterate, Qt::QueuedConnection);
QObject::connect(coreHandler, &CoreHandler::coreStopped, this, &CoreManager::stopIterate, Qt::QueuedConnection);
QObject::connect(coreHandler, &CoreHandler::callStateChanged, this, &CoreManager::onCallStateChanged, Qt::QueuedConnection);
QObject::connect(coreHandler, &CoreHandler::registrationStateChanged, this, &CoreManager::onRegistrationStateChanged, Qt::QueuedConnection);
// Delay the creation of the core so that the CoreManager instance is
// already set.
QTimer::singleShot(10, [this]()
{ createLinphoneCore(); });
}
CoreManager::~CoreManager()
{
mHandler->removeListener(mCore);
mHandler = nullptr;
mCore = nullptr;
}
void CoreManager::init(QObject *parent)
{
if (mInstance)
return;
mInstance = new CoreManager(parent);
}
void CoreManager::uninit()
{
if (mInstance)
{
mInstance->stopIterate();
auto core = mInstance->mCore;
delete mInstance;
mInstance = nullptr;
core->stop();
}
}
CoreManager *CoreManager::getInstance()
{
return mInstance;
}
void CoreManager::startIterate()
{
// Start a timer to call the core iterate every 20 ms.
mIterateTimer = new QTimer(this);
mIterateTimer->setInterval(20);
QObject::connect(mIterateTimer, &QTimer::timeout, this, &CoreManager::iterate);
qInfo() << QStringLiteral("Start iterate");
mIterateTimer->start();
}
void CoreManager::stopIterate()
{
qInfo() << QStringLiteral("Stop iterate");
mIterateTimer->stop();
mIterateTimer->deleteLater(); // Allow the timer to continue its stuff
mIterateTimer = nullptr;
}
void CoreManager::login(QString identity, QString password)
{
if (mLoginButtonEnabled)
{
setProperty("loginButtonEnabled", false);
shared_ptr<Address> address = Factory::get()->createAddress(identity.toStdString());
shared_ptr<AuthInfo> authInfo = Factory::get()->createAuthInfo(address->getUsername(), "", password.toStdString(), "", "", address->getDomain());
mCore->addAuthInfo(authInfo);
shared_ptr<AccountParams> accountParams = mCore->createAccountParams();
accountParams->setIdentityAddress(address);
string serverAddr = "sip:" + address->getDomain() + ";transport=tls";
accountParams->setServerAddr(serverAddr);
accountParams->enableRegister(true);
shared_ptr<Account> account = mCore->createAccount(accountParams);
mCore->addAccount(account);
mCore->setDefaultAccount(account);
}
}
void CoreManager::logout()
{
shared_ptr<Account> account = mCore->getDefaultAccount();
if (account)
{
shared_ptr<AccountParams> accountParams = account->getParams()->clone();
accountParams->enableRegister(false);
account->setParams(accountParams);
}
}
void CoreManager::onCallStateChanged(const std::shared_ptr<linphone::Call> &call, linphone::Call::State state, const std::string &message)
{
setProperty("callStateText", QString::fromStdString("Your call state is: " + message));
switch (state)
{
case Call::State::IncomingReceived:
// When you receive a call the Call::State is incoming receive. By default you can only have one current call,
// so if a call is in progress or one is already ringing the second remote call will be decline with the reason
// "Busy". If you want to implement a multi call app you can increase Core::setMaxCalls.
// Here we store the incoming call reference so we can accept or decline the call on user input.
mIncomingCall = call;
// And we update the GUI to notify the user of the incoming call.
setProperty("incomingCallVisible", true);
setProperty("incomingCallText", QString::fromStdString(mIncomingCall->getRemoteAddress()->asString()));
break;
case Call::State::StreamsRunning:
// The StreamsRunning state is the default one during a call.
callInProgressGuiUpdates();
break;
case Call::State::Error:
case Call::State::End:
case Call::State::Released:
// By default after 30 seconds of ringing without accept or decline a call is
// automatically ended.
mIncomingCall = nullptr;
endingCallGuiUpdates();
break;
}
}
void CoreManager::onRegistrationStateChanged(const shared_ptr<Account> &account, RegistrationState state, const string &message)
{
setProperty("registerText", QString::fromStdString("Your registration state is : " + message));
switch (state)
{
// If the Account was logged out, we clear the Core.
case RegistrationState::Cleared:
case RegistrationState::None:
mCore->clearAllAuthInfo();
mCore->clearAccounts();
setProperty("loginButtonEnabled", true);
break;
case RegistrationState::Ok:
setProperty("loginButtonEnabled", false);
setProperty("headerText", QString::fromStdString("Hello " + mCore->getDefaultProxyConfig()->findAuthInfo()->getUsername()));
setProperty("page", QString("qrc:/ui/CallPage.qml"));
break;
case RegistrationState::Progress:
setProperty("loginButtonEnabled", false);
break;
case RegistrationState::Failed:
setProperty("loginButtonEnabled", true);
break;
default:
break;
}
}
void CoreManager::hangup()
{
// Simply call terminateAllCalls to hang out.
mCore->terminateAllCalls();
}
void CoreManager::soundButtonClicked()
{
if (toggleSpeaker())
{
setProperty("soundButtonText", QString("Switch on Sound"));
}
else
{
setProperty("soundButtonText", QString("Switch off Sound"));
}
}
void CoreManager::microphoneButtonClicked()
{
if (toggleMicrophone())
{
setProperty("microphoneButtonText", QString("Mute"));
}
else
{
setProperty("microphoneButtonText", QString("Unmute"));
}
}
void CoreManager::answer()
{
if (mIncomingCall)
{
// To accept a call only use the accept() method on the call object.
// If we wanted, we could create a CallParams object and answer using this object to make changes to the call configuration.
mIncomingCall->accept();
mIncomingCall = nullptr;
}
}
void CoreManager::decline()
{
if (mIncomingCall)
{
// You have to give a Reason to decline a call. This info is sent to the remote.
mIncomingCall->decline(Reason::Declined);
mIncomingCall = nullptr;
}
}
void CoreManager::createLinphoneCore()
{
// Setting linphone log level to message.
auto loggingService = LoggingService::get();
loggingService->setLogLevel(LogLevel::Message);
// Configure paths.
string assetsPath = string(SDK_PATH) + "/share";
Factory::get()->setTopResourcesDir(assetsPath);
Factory::get()->setDataResourcesDir(assetsPath);
Factory::get()->setSoundResourcesDir(assetsPath + "/sounds/linphone");
Factory::get()->setRingResourcesDir(Factory::get()->getSoundResourcesDir() + "/rings");
Factory::get()->setImageResourcesDir(assetsPath + "/images");
Factory::get()->setMspluginsDir(MSPLUGINS_PATH);
// Create a core from the factory.
mCore = Factory::get()->createCore("", "", nullptr);
mCore->setRootCa(assetsPath + "/linphone/rootca.pem");
// Listen for core events.
mHandler->setListener(mCore);
// Start the core.
mCore->start();
}
void CoreManager::iterate()
{
if (mCore)
mCore->iterate();
}
bool CoreManager::toggleSpeaker()
{
// Calling setSpeakerMuted(true) on a Call object disables the sound output of this call.
bool newValue = !mCore->getCurrentCall()->getSpeakerMuted();
mCore->getCurrentCall()->setSpeakerMuted(newValue);
return newValue;
}
bool CoreManager::toggleMicrophone()
{
// The following toggles the microphone, disabling completely / enabling the sound capture from the device microphone
bool newValue = !mCore->micEnabled();
mCore->enableMic(newValue);
return newValue;
}
void CoreManager::callInProgressGuiUpdates()
{
setProperty("incomingCallVisible", false);
setProperty("inCallButtonsEnabled", true);
}
void CoreManager::endingCallGuiUpdates()
{
setProperty("incomingCallVisible", false);
setProperty("inCallButtonsEnabled", false);
setProperty("soundButtonText", "Switch off Sound");
setProperty("microphoneButtonText", "Mute");
}

View File

@@ -1,84 +0,0 @@
#pragma once
#include <QObject>
#include <QSharedPointer>
#include <QTimer>
#include <linphone++/linphone.hh>
class CoreHandler;
class CoreManager : public QObject
{
Q_OBJECT
Q_PROPERTY(QString page MEMBER mPage NOTIFY pageChanged);
Q_PROPERTY(QString headerText MEMBER mHeaderText NOTIFY headerTextChanged)
Q_PROPERTY(QString registerText MEMBER mRegisterText NOTIFY registerTextChanged)
Q_PROPERTY(bool loginButtonEnabled MEMBER mLoginButtonEnabled NOTIFY loginButtonEnabledChanged)
Q_PROPERTY(QString callStateText MEMBER mCallStateText NOTIFY callStateTextChanged)
Q_PROPERTY(bool incomingCallVisible MEMBER mIncomingCallVisible NOTIFY incomingCallVisibleChanged)
Q_PROPERTY(QString incomingCallText MEMBER mIncomingCallText NOTIFY incomingCallTextChanged)
Q_PROPERTY(bool inCallButtonsEnabled MEMBER mInCallButtonsEnabled NOTIFY inCallButtonsEnabledChanged)
Q_PROPERTY(QString soundButtonText MEMBER mSoundButtonText NOTIFY soundButtonTextChanged)
Q_PROPERTY(QString microphoneButtonText MEMBER mMicrophoneButtonText NOTIFY microphoneButtonTextChanged)
public:
static void init(QObject *parent);
static void uninit();
static CoreManager *getInstance();
signals:
void pageChanged(QString);
void headerTextChanged(QString);
void registerTextChanged(QString);
void loginButtonEnabledChanged(bool);
void callStateTextChanged(QString);
void incomingCallVisibleChanged(bool);
void incomingCallTextChanged(QString);
void inCallButtonsEnabledChanged(bool);
void soundButtonTextChanged(QString);
void microphoneButtonTextChanged(QString);
public slots:
void startIterate();
void stopIterate();
void login(QString identity, QString password);
void logout();
void onCallStateChanged(const std::shared_ptr<linphone::Call> &call, linphone::Call::State state, const std::string &message);
void onRegistrationStateChanged(const std::shared_ptr<linphone::Account> &account, linphone::RegistrationState state, const std::string &message);
void hangup();
void soundButtonClicked();
void microphoneButtonClicked();
void answer();
void decline();
private:
CoreManager(QObject *parent);
~CoreManager();
void createLinphoneCore();
void iterate();
bool toggleSpeaker();
bool toggleMicrophone();
void callInProgressGuiUpdates();
void endingCallGuiUpdates();
std::shared_ptr<linphone::Core> mCore = nullptr;
std::shared_ptr<linphone::Call> mIncomingCall = nullptr;
QSharedPointer<CoreHandler> mHandler;
QTimer *mIterateTimer = nullptr;
QString mPage;
QString mHeaderText;
QString mRegisterText;
bool mLoginButtonEnabled = true;
QString mCallStateText;
bool mIncomingCallVisible = false;
QString mIncomingCallText;
bool mInCallButtonsEnabled = false;
QString mSoundButtonText;
QString mMicrophoneButtonText;
static CoreManager *mInstance;
};

View File

@@ -1,77 +0,0 @@
import QtQuick 2.9
import QtQuick.Controls 2.2
import QtQuick.Layouts 1.3
GridLayout {
Layout.alignment: Qt.AlignHCenter
Layout.fillWidth: true
Layout.fillHeight: true
columns: 1
ColumnLayout {
Layout.fillHeight: true
Layout.alignment: Qt.AlignHCenter
Text {
text: coreManager.callStateText
}
}
RowLayout {
Layout.fillWidth: true
Layout.alignment: Qt.AlignHCenter
spacing: 20
Button {
text: "Hang up"
enabled: coreManager.inCallButtonsEnabled
onClicked: {
coreManager.hangup()
}
}
Button {
text: coreManager.soundButtonText
enabled: coreManager.inCallButtonsEnabled
onClicked: {
coreManager.soundButtonClicked()
}
}
Button {
text: coreManager.microphoneButtonText
enabled: coreManager.inCallButtonsEnabled
onClicked: {
coreManager.microphoneButtonClicked()
}
}
}
GridLayout {
Layout.fillWidth: true
Layout.alignment: Qt.AlignHCenter
Layout.margins: 20
columnSpacing: 20
columns: 2
visible: coreManager.incomingCallVisible
Text {
Layout.alignment: Qt.AlignHCenter
Layout.columnSpan: 2
text: "You have a call from: " + coreManager.incomingCallText
}
Button {
text: "Answer"
onClicked: {
coreManager.answer()
}
}
Button {
text: "Decline"
onClicked: {
coreManager.decline()
}
}
}
}

View File

@@ -1,35 +0,0 @@
import QtQuick 2.9
import QtQuick.Controls 2.2
import QtQuick.Layouts 1.3
ApplicationWindow {
id: window
visible: true
title: "Account Login"
width: 640
height: 480
header: Rectangle {
color: "lightgray";
height: 40;
width: window.width
Text {
text: coreManager.headerText
font.bold: true
font.capitalization: Font.AllUppercase
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
width: parent.width
height: parent.height
}
}
// Main content
Loader {
id: contentLoader
anchors.fill: parent
source: coreManager.page
}
}

View File

@@ -1,57 +0,0 @@
import QtQuick 2.9
import QtQuick.Controls 2.2
import QtQuick.Layouts 1.3
ColumnLayout {
//Layout.alignment: Qt.AlignHCenter | Qt.AlignTop
Layout.fillWidth: true
Layout.fillHeight: true
GridLayout {
Layout.fillWidth: true
Layout.margins: 20
columnSpacing: 20
columns: 2
Text {
text: "Identity:"
}
TextField {
id: identityTextField
Layout.fillWidth: true
text: "sip:"
}
Text {
text: "Password:"
}
TextField {
id: passwordTextField
echoMode: TextInput.Password
Layout.fillWidth: true
placeholderText: "my password"
}
}
RowLayout {
Layout.fillWidth: true
Layout.alignment: Qt.AlignHCenter
Button {
text: "Login"
enabled: coreManager.loginButtonEnabled && identityTextField.text.length !== 0 && passwordTextField.text.length !== 0
onClicked: {
coreManager.login(identityTextField.text, passwordTextField.text)
}
}
}
ColumnLayout {
Layout.fillHeight: true
Layout.alignment: Qt.AlignHCenter
Text {
text: coreManager.registerText
}
}
}