main.c 62.3 KB
Newer Older
aymeric's avatar
aymeric committed
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
/*
linphone, gtk-glade interface.
Copyright (C) 2008  Simon MORLAT (simon.morlat@linphone.org)

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 2
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, write to the Free Software
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
*/


21
#define VIDEOSELFVIEW_DEFAULT 0
22

23
#include "linphone.h"
aymeric's avatar
aymeric committed
24 25
#include "lpconfig.h"

26

aymeric's avatar
aymeric committed
27 28 29 30
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>

jehan's avatar
jehan committed
31
#ifdef HAVE_GTK_OSX
32 33 34
#include <gtkosxapplication.h>
#endif

35 36 37 38
#ifdef WIN32
#define chdir _chdir
#endif

39
#if defined(HAVE_NOTIFY1) || defined(HAVE_NOTIFY4)
40 41 42
#define HAVE_NOTIFY
#endif

43 44 45 46
#ifdef HAVE_NOTIFY
#include <libnotify/notify.h>
#endif

47
#define LINPHONE_ICON "linphone.png"
aymeric's avatar
aymeric committed
48

smorlat's avatar
smorlat committed
49 50
const char *this_program_ident_string="linphone_ident_string=" LINPHONE_VERSION;

aymeric's avatar
aymeric committed
51 52
static LinphoneCore *the_core=NULL;
static GtkWidget *the_ui=NULL;
Sylvain Berfini's avatar
Wizard  
Sylvain Berfini committed
53
GtkWidget *the_wizard=NULL;
aymeric's avatar
aymeric committed
54

Simon Morlat's avatar
Simon Morlat committed
55
static void linphone_gtk_registration_state_changed(LinphoneCore *lc, LinphoneProxyConfig *cfg, LinphoneRegistrationState rs, const char *msg);
56
static void linphone_gtk_notify_recv(LinphoneCore *lc, LinphoneFriend * fid);
aymeric's avatar
aymeric committed
57 58 59 60 61 62 63
static void linphone_gtk_new_unknown_subscriber(LinphoneCore *lc, LinphoneFriend *lf, const char *url);
static void linphone_gtk_auth_info_requested(LinphoneCore *lc, const char *realm, const char *username);
static void linphone_gtk_display_status(LinphoneCore *lc, const char *status);
static void linphone_gtk_display_message(LinphoneCore *lc, const char *msg);
static void linphone_gtk_display_warning(LinphoneCore *lc, const char *warning);
static void linphone_gtk_display_url(LinphoneCore *lc, const char *msg, const char *url);
static void linphone_gtk_call_log_updated(LinphoneCore *lc, LinphoneCallLog *cl);
Simon Morlat's avatar
Simon Morlat committed
64
static void linphone_gtk_call_state_changed(LinphoneCore *lc, LinphoneCall *call, LinphoneCallState cs, const char *msg);
Simon Morlat's avatar
Simon Morlat committed
65
static void linphone_gtk_call_encryption_changed(LinphoneCore *lc, LinphoneCall *call, bool_t enabled, const char *token);
66
static void linphone_gtk_transfer_state_changed(LinphoneCore *lc, LinphoneCall *call, LinphoneCallState cstate);
67
static gboolean linphone_gtk_auto_answer(LinphoneCall *call);
68
static void linphone_gtk_status_icon_set_blinking(gboolean val);
69
void _linphone_gtk_enable_video(gboolean val);
aymeric's avatar
aymeric committed
70 71 72


static gboolean verbose=0;
73 74
static gboolean auto_answer = 0;
static gchar * addr_to_call = NULL;
75
static gboolean no_video=FALSE;
smorlat's avatar
smorlat committed
76
static gboolean iconified=FALSE;
77
static gchar *workingdir=NULL;
78
static char *progpath=NULL;
79
gchar *linphone_logfile=NULL;
80
static gboolean workaround_gtk_entry_chinese_bug=FALSE;
smorlat's avatar
smorlat committed
81

82
static GOptionEntry linphone_options[]={
aymeric's avatar
aymeric committed
83 84 85 86 87
	{
		.long_name="verbose",
		.short_name= '\0',
		.arg=G_OPTION_ARG_NONE,
		.arg_data= (gpointer)&verbose,
88
		.description=N_("log to stdout some debug information while running.")
89
	},
90 91 92 93 94 95 96
	{
	    .long_name = "logfile",
	    .short_name = 'l',
	    .arg = G_OPTION_ARG_STRING,
	    .arg_data = &linphone_logfile,
	    .description = N_("path to a file to write logs into.")
	},
97 98 99 100 101 102 103
	{
	    .long_name = "no-video",
	    .short_name = '\0',
	    .arg = G_OPTION_ARG_NONE,
	    .arg_data = (gpointer)&no_video,
	    .description = N_("Start linphone with video disabled.")
	},
smorlat's avatar
smorlat committed
104 105 106 107 108 109 110
	{
		.long_name="iconified",
		.short_name= '\0',
		.arg=G_OPTION_ARG_NONE,
		.arg_data= (gpointer)&iconified,
		.description=N_("Start only in the system tray, do not show the main interface.")
	},
111
	{
112 113 114
	    .long_name = "call",
	    .short_name = 'c',
	    .arg = G_OPTION_ARG_STRING,
115 116
	    .arg_data = &addr_to_call,
	    .description = N_("address to call right now")
117
	},
118
	{
119 120 121 122
	    .long_name = "auto-answer",
	    .short_name = 'a',
	    .arg = G_OPTION_ARG_NONE,
	    .arg_data = (gpointer) & auto_answer,
123
	    .description = N_("if set automatically answer incoming calls")
124
	},
Simon Morlat's avatar
Simon Morlat committed
125
	{
126 127 128 129 130 131
	    .long_name = "workdir",
	    .short_name = '\0',
	    .arg = G_OPTION_ARG_STRING,
	    .arg_data = (gpointer) & workingdir,
	    .description = N_("Specifiy a working directory (should be the base of the installation, eg: c:\\Program Files\\Linphone)")
	},
132
	{0}
aymeric's avatar
aymeric committed
133 134 135
};

#define INSTALLED_XML_DIR PACKAGE_DATA_DIR "/linphone"
Simon Morlat's avatar
Simon Morlat committed
136
#define RELATIVE_XML_DIR 
137
#define BUILD_TREE_XML_DIR "gtk"
smorlat's avatar
smorlat committed
138 139

#ifndef WIN32
aymeric's avatar
aymeric committed
140
#define CONFIG_FILE ".linphonerc"
Simon Morlat's avatar
Simon Morlat committed
141
#define SECRETS_FILE ".linphone-zidcache"
smorlat's avatar
smorlat committed
142 143
#else
#define CONFIG_FILE "linphonerc"
Simon Morlat's avatar
Simon Morlat committed
144
#define SECRETS_FILE "linphone-zidcache"
smorlat's avatar
smorlat committed
145 146 147
#endif


Simon Morlat's avatar
Simon Morlat committed
148 149 150 151
char *linphone_gtk_get_config_file(const char *filename){
	const int path_max=1024;
	char *config_file=g_malloc0(path_max);
	if (filename==NULL) filename=CONFIG_FILE;
aymeric's avatar
aymeric committed
152 153
	/*try accessing a local file first if exists*/
	if (access(CONFIG_FILE,F_OK)==0){
154
		snprintf(config_file,path_max,"%s",filename);
aymeric's avatar
aymeric committed
155 156 157 158
	}else{
#ifdef WIN32
		const char *appdata=getenv("APPDATA");
		if (appdata){
Simon Morlat's avatar
Simon Morlat committed
159 160 161
			snprintf(config_file,path_max,"%s\\%s",appdata,LINPHONE_CONFIG_DIR);
			CreateDirectory(config_file,NULL);
			snprintf(config_file,path_max,"%s\\%s\\%s",appdata,LINPHONE_CONFIG_DIR,filename);
aymeric's avatar
aymeric committed
162 163
		}
#else
164 165
		const char *home=getenv("HOME");
		if (home==NULL) home=".";
Simon Morlat's avatar
Simon Morlat committed
166
		snprintf(config_file,path_max,"%s/%s",home,filename);
aymeric's avatar
aymeric committed
167 168
#endif
	}
Simon Morlat's avatar
Simon Morlat committed
169
	return config_file;
smorlat's avatar
smorlat committed
170 171
}

172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220

#define FACTORY_CONFIG_FILE "linphonerc.factory"
static char _factory_config_file[1024];
static const char *linphone_gtk_get_factory_config_file(){
	/*try accessing a local file first if exists*/
	if (access(FACTORY_CONFIG_FILE,F_OK)==0){
		snprintf(_factory_config_file,sizeof(_factory_config_file),
						 "%s",FACTORY_CONFIG_FILE);
	} else {
		char *progdir;
		
		if (progpath != NULL) {
			char *basename;
			progdir = strdup(progpath);
#ifdef WIN32
			basename = strrchr(progdir, '\\');
			if (basename != NULL) {
				basename ++;
				*basename = '\0';
				snprintf(_factory_config_file, sizeof(_factory_config_file),
								 "%s\\..\\%s", progdir, FACTORY_CONFIG_FILE);
			} else {
				if (workingdir!=NULL) {
					snprintf(_factory_config_file, sizeof(_factory_config_file),
									 "%s\\%s", workingdir, FACTORY_CONFIG_FILE);
				} else {
					free(progdir);
					return NULL;
				}
			}
#else
			basename = strrchr(progdir, '/');
			if (basename != NULL) {
				basename ++;
				*basename = '\0';
				snprintf(_factory_config_file, sizeof(_factory_config_file),
								 "%s/../share/Linphone/%s", progdir, FACTORY_CONFIG_FILE);
			} else {
				free(progdir);
				return NULL;
			}
#endif
			free(progdir);
		}
	}
	return _factory_config_file;
}

static void linphone_gtk_init_liblinphone(const char *config_file,
221
		const char *factory_config_file) {
Simon Morlat's avatar
Simon Morlat committed
222
	LinphoneCoreVTable vtable={0};
Simon Morlat's avatar
Simon Morlat committed
223
	gchar *secrets_file=linphone_gtk_get_config_file(SECRETS_FILE);
Simon Morlat's avatar
Simon Morlat committed
224 225

	vtable.call_state_changed=linphone_gtk_call_state_changed;
Simon Morlat's avatar
Simon Morlat committed
226
	vtable.registration_state_changed=linphone_gtk_registration_state_changed;
Simon Morlat's avatar
Simon Morlat committed
227
	vtable.notify_presence_recv=linphone_gtk_notify_recv;
jehan's avatar
jehan committed
228
	vtable.new_subscription_request=linphone_gtk_new_unknown_subscriber;
Simon Morlat's avatar
Simon Morlat committed
229 230 231 232 233 234 235 236 237
	vtable.auth_info_requested=linphone_gtk_auth_info_requested;
	vtable.display_status=linphone_gtk_display_status;
	vtable.display_message=linphone_gtk_display_message;
	vtable.display_warning=linphone_gtk_display_warning;
	vtable.display_url=linphone_gtk_display_url;
	vtable.call_log_updated=linphone_gtk_call_log_updated;
	vtable.text_received=linphone_gtk_text_received;
	vtable.refer_received=linphone_gtk_refer_received;
	vtable.buddy_info_updated=linphone_gtk_buddy_info_updated;
Simon Morlat's avatar
Simon Morlat committed
238
	vtable.call_encryption_changed=linphone_gtk_call_encryption_changed;
239
	vtable.transfer_state_changed=linphone_gtk_transfer_state_changed;
Simon Morlat's avatar
Simon Morlat committed
240

smorlat's avatar
smorlat committed
241
	linphone_core_set_user_agent("Linphone", LINPHONE_VERSION);
242
	the_core=linphone_core_new(&vtable,config_file,factory_config_file,NULL);
243
	linphone_core_set_waiting_callback(the_core,linphone_gtk_wait,NULL);
Simon Morlat's avatar
Simon Morlat committed
244 245
	linphone_core_set_zrtp_secrets_file(the_core,secrets_file);
	g_free(secrets_file);
246
	linphone_core_enable_video(the_core,TRUE,TRUE);
247 248 249 250
	if (no_video) {
		_linphone_gtk_enable_video(FALSE);
		linphone_gtk_set_ui_config_int("videoselfview",0);
	}
aymeric's avatar
aymeric committed
251 252 253 254 255 256 257 258 259 260 261 262
}



LinphoneCore *linphone_gtk_get_core(void){
	return the_core;
}

GtkWidget *linphone_gtk_get_main_window(){
	return the_ui;
}

smorlat's avatar
smorlat committed
263
static void linphone_gtk_configure_window(GtkWidget *w, const char *window_name){
smorlat's avatar
smorlat committed
264 265 266
	static const char *icon_path=NULL;
	static const char *hiddens=NULL;
	static const char *shown=NULL;
267
	static bool_t config_loaded=FALSE;
smorlat's avatar
smorlat committed
268
	if (linphone_gtk_get_core()==NULL) return;
269 270
	if (config_loaded==FALSE){
		hiddens=linphone_gtk_get_ui_config("hidden_widgets",NULL);
smorlat's avatar
smorlat committed
271
		shown=linphone_gtk_get_ui_config("shown_widgets",NULL);
272
		icon_path=linphone_gtk_get_ui_config("icon",LINPHONE_ICON);
273 274
		config_loaded=TRUE;
	}
smorlat's avatar
smorlat committed
275
	if (hiddens)
Simon Morlat's avatar
Simon Morlat committed
276
		linphone_gtk_visibility_set(hiddens,window_name,w,FALSE);
smorlat's avatar
smorlat committed
277
	if (shown)
Simon Morlat's avatar
Simon Morlat committed
278
		linphone_gtk_visibility_set(shown,window_name,w,TRUE);
279 280 281 282 283
	if (icon_path) {
		GdkPixbuf *pbuf=create_pixbuf(icon_path);
		gtk_window_set_icon(GTK_WINDOW(w),pbuf);
		g_object_unref(G_OBJECT(pbuf));
	}
smorlat's avatar
smorlat committed
284 285
}

286 287
static int get_ui_file(const char *name, char *path, int pathsize){
	snprintf(path,pathsize,"%s/%s.ui",BUILD_TREE_XML_DIR,name);
288
	if (access(path,F_OK)!=0){
289
		snprintf(path,pathsize,"%s/%s.ui",INSTALLED_XML_DIR,name);
290
		if (access(path,F_OK)!=0){
Simon Morlat's avatar
Simon Morlat committed
291
			g_error("Could not locate neither %s/%s.ui nor %s/%s.ui",BUILD_TREE_XML_DIR,name,
292 293
				INSTALLED_XML_DIR,name);
			return -1;
294 295
		}
	}
296 297 298 299 300 301 302 303 304 305 306
	return 0;
}

GtkWidget *linphone_gtk_create_window(const char *window_name){
	GError* error = NULL;
	GtkBuilder* builder = gtk_builder_new ();
	char path[512];
	GtkWidget *w;

	if (get_ui_file(window_name,path,sizeof(path))==-1) return NULL;
	
307 308 309
	if (!gtk_builder_add_from_file (builder, path, &error)){
		g_error("Couldn't load builder file: %s", error->message);
		g_error_free (error);
310
		return NULL;
311 312 313 314 315 316 317 318 319 320
	}
	w=GTK_WIDGET(gtk_builder_get_object (builder,window_name));
	if (w==NULL){
		g_error("Could not retrieve '%s' window from xml file",window_name);
		return NULL;
	}
	g_object_set_data(G_OBJECT(w),"builder",builder);
	gtk_builder_connect_signals(builder,w);
	linphone_gtk_configure_window(w,window_name);
	return w;
aymeric's avatar
aymeric committed
321 322
}

323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345
GtkWidget *linphone_gtk_create_widget(const char *filename, const char *widget_name){
	char path[2048];
	GtkWidget *w;
	GtkBuilder* builder = gtk_builder_new ();
	GError *error=NULL;
	gchar *object_ids[2];
	object_ids[0]=g_strdup(widget_name);
	object_ids[1]=NULL;
	
	if (get_ui_file(filename,path,sizeof(path))==-1) return NULL;
	if (!gtk_builder_add_objects_from_file(builder,path,object_ids,&error)){
		g_error("Couldn't load %s from builder file %s: %s", widget_name,path,error->message);
		g_error_free (error);
		g_free(object_ids[0]);
		return NULL;
	}
	g_free(object_ids[0]);
	w=GTK_WIDGET(gtk_builder_get_object (builder,widget_name));
	if (w==NULL){
		g_error("Could not retrieve '%s' window from xml file",widget_name);
		return NULL;
	}
	g_object_set_data(G_OBJECT(w),"builder",builder);
346
	g_signal_connect_swapped(G_OBJECT(w),"destroy",(GCallback)g_object_unref,builder);
347 348 349 350
	gtk_builder_connect_signals(builder,w);
	return w;
}

351 352 353 354 355
static void entry_unmapped(GtkWidget *entry){
	g_message("Entry is unmapped, calling unrealize to workaround chinese bug.");
	gtk_widget_unrealize(entry);
}

aymeric's avatar
aymeric committed
356
GtkWidget *linphone_gtk_get_widget(GtkWidget *window, const char *name){
357
	GtkBuilder *builder;
358
	GObject *w;
359 360
	if (window==NULL) return NULL;
	builder=(GtkBuilder*)g_object_get_data(G_OBJECT(window),"builder");
361 362 363 364 365
	if (builder==NULL){
		g_error("Fail to retrieve builder from window !");
		return NULL;
	}
	w=gtk_builder_get_object(builder,name);
aymeric's avatar
aymeric committed
366 367 368
	if (w==NULL){
		g_error("No widget named %s found in xml interface.",name);
	}
369 370 371 372 373 374 375 376 377
	if (workaround_gtk_entry_chinese_bug){
		if (strcmp(G_OBJECT_TYPE_NAME(w),"GtkEntry")==0){
			if (g_object_get_data(G_OBJECT(w),"entry_bug_workaround")==NULL){
				g_object_set_data(G_OBJECT(w),"entry_bug_workaround",GINT_TO_POINTER(1));
				g_message("%s is a GtkEntry",name);
				g_signal_connect(G_OBJECT(w),"unmap",(GCallback)entry_unmapped,NULL);
			}
		}
	}
aymeric's avatar
aymeric committed
378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394
	return GTK_WIDGET(w);
}


void linphone_gtk_display_something(GtkMessageType type,const gchar *message){
	GtkWidget *dialog;
	GtkWidget *main_window=linphone_gtk_get_main_window();
	
	gtk_widget_show(main_window);
	if (type==GTK_MESSAGE_QUESTION)
	{
		/* draw a question box. link to dialog_click callback */
		dialog = gtk_message_dialog_new (
				GTK_WINDOW(main_window),
                                GTK_DIALOG_DESTROY_WITH_PARENT,
				GTK_MESSAGE_QUESTION,
                                GTK_BUTTONS_YES_NO,
smorlat's avatar
smorlat committed
395
                                "%s",
aymeric's avatar
aymeric committed
396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411
				(const gchar*)message);
		/* connect to some callback : REVISIT */
		/*
		g_signal_connect_swapped (G_OBJECT (dialog), "response",
                           G_CALLBACK (dialog_click),
                           G_OBJECT (dialog));
		*/
		/* actually show the box */
		gtk_widget_show(dialog);
	}
	else
	{
		dialog = gtk_message_dialog_new (GTK_WINDOW(main_window),
                                  GTK_DIALOG_DESTROY_WITH_PARENT,
                                  type,
                                  GTK_BUTTONS_CLOSE,
smorlat's avatar
smorlat committed
412
                                  "%s",
aymeric's avatar
aymeric committed
413 414 415 416 417 418 419 420 421 422 423 424 425 426 427
                                  (const gchar*)message);
		/* Destroy the dialog when the user responds to it (e.g. clicks a button) */
		g_signal_connect_swapped (G_OBJECT (dialog), "response",
                           G_CALLBACK (gtk_widget_destroy),
                           G_OBJECT (dialog));
		gtk_widget_show(dialog);
	}
}

void linphone_gtk_about_response(GtkDialog *dialog, gint id){
	if (id==GTK_RESPONSE_CANCEL){
		gtk_widget_destroy(GTK_WIDGET(dialog));
	}
}

428 429 430 431 432
static void about_url_clicked(GtkAboutDialog *dialog, const char *url, gpointer data){
	g_message("About url clicked");
	linphone_gtk_open_browser(url);
}

aymeric's avatar
aymeric committed
433 434
void linphone_gtk_show_about(){
	struct stat filestat;
435
	const char *license_file=PACKAGE_DATA_DIR "/linphone/COPYING";
aymeric's avatar
aymeric committed
436
	GtkWidget *about;
437
	const char *tmp;
438 439
	GdkPixbuf *logo=create_pixbuf(
	    linphone_gtk_get_ui_config("logo","linphone-banner.png"));
440
	static const char *defcfg="defcfg";
441
	
aymeric's avatar
aymeric committed
442
	about=linphone_gtk_create_window("about");
443
	gtk_about_dialog_set_url_hook(about_url_clicked,NULL,NULL);
aymeric's avatar
aymeric committed
444 445 446 447 448 449 450 451 452 453 454 455 456 457 458
	memset(&filestat,0,sizeof(filestat));
	if (stat(license_file,&filestat)!=0){
		license_file="COPYING";
		stat(license_file,&filestat);
	}
	if (filestat.st_size>0){
		char *license=g_malloc(filestat.st_size+1);
		FILE *f=fopen(license_file,"r");
		if (f && fread(license,filestat.st_size,1,f)==1){
			license[filestat.st_size]='\0';
			gtk_about_dialog_set_license(GTK_ABOUT_DIALOG(about),license);
		}
		g_free(license);
	}
	gtk_about_dialog_set_version(GTK_ABOUT_DIALOG(about),LINPHONE_VERSION);
459 460
	gtk_about_dialog_set_program_name(GTK_ABOUT_DIALOG(about),linphone_gtk_get_ui_config("title","Linphone"));
	gtk_about_dialog_set_website(GTK_ABOUT_DIALOG(about),linphone_gtk_get_ui_config("home","http://www.linphone.org"));
461
	if (logo)	gtk_about_dialog_set_logo(GTK_ABOUT_DIALOG(about),logo);
462 463 464 465 466 467 468 469 470 471 472 473 474
	tmp=linphone_gtk_get_ui_config("artists",defcfg);
	if (tmp!=defcfg){
		const char *tmp2[2];
		tmp2[0]=tmp;
		tmp2[1]=NULL;
		gtk_about_dialog_set_artists(GTK_ABOUT_DIALOG(about),tmp2);
	}
	tmp=linphone_gtk_get_ui_config("translators",defcfg);
	if (tmp!=defcfg)
		gtk_about_dialog_set_translator_credits (GTK_ABOUT_DIALOG(about),tmp);
	tmp=linphone_gtk_get_ui_config("comments",defcfg);
	if (tmp!=defcfg)
		gtk_about_dialog_set_comments(GTK_ABOUT_DIALOG(about),tmp);
aymeric's avatar
aymeric committed
475 476 477
	gtk_widget_show(about);
}

478 479
static void set_video_window_decorations(GdkWindow *w){
	const char *title=linphone_gtk_get_ui_config("title","Linphone");
480
	const char *icon_path=linphone_gtk_get_ui_config("icon",LINPHONE_ICON);
481 482
	char video_title[256];
	GdkPixbuf *pbuf=create_pixbuf(icon_path);
smorlat's avatar
smorlat committed
483
	if (!linphone_core_in_call(linphone_gtk_get_core())){
484 485 486
		snprintf(video_title,sizeof(video_title),"%s video",title);
		/* When not in call, treat the video as a normal window */
		gdk_window_set_keep_above(w, FALSE);
smorlat's avatar
smorlat committed
487
	}else{
488
		LinphoneAddress *uri =
489
			linphone_address_clone(linphone_core_get_current_call_remote_address(linphone_gtk_get_core()));
490
		char *display_name;
491 492 493

		linphone_address_clean(uri);
		if (linphone_address_get_display_name(uri)!=NULL){
494
			display_name=ms_strdup(linphone_address_get_display_name(uri));
495
		}else{
496
			display_name=linphone_address_as_string(uri);
497 498
		}
		snprintf(video_title,sizeof(video_title),_("Call with %s"),display_name);
499
		linphone_address_destroy(uri);
500
		ms_free(display_name);
501 502 503 504 505 506 507 508

		/* During calls, bring up the video window, arrange so that
		it is above all the other windows */
		gdk_window_deiconify(w);
		gdk_window_set_keep_above(w,TRUE);
		/* Maybe we should have the following, but then we want to
		have a timer that turns it off after a little while. */
		/* gdk_window_set_urgency_hint(w,TRUE); */
smorlat's avatar
smorlat committed
509
	}
510
	gdk_window_set_title(w,video_title);
511 512 513 514
	/* Refrain the video window to be closed at all times. */
	gdk_window_set_functions(w,
				 GDK_FUNC_RESIZE|GDK_FUNC_MOVE|
				 GDK_FUNC_MINIMIZE|GDK_FUNC_MAXIMIZE);
515 516 517 518 519 520 521 522 523
	if (pbuf){
		GList *l=NULL;
		l=g_list_append(l,pbuf);
		gdk_window_set_icon_list(w,l);
		g_list_free(l);
		g_object_unref(G_OBJECT(pbuf));
	}
}

smorlat's avatar
smorlat committed
524 525 526 527 528
static gboolean video_needs_update=FALSE;

static void update_video_title(){
	video_needs_update=TRUE;
}
529

aymeric's avatar
aymeric committed
530
static gboolean linphone_gtk_iterate(LinphoneCore *lc){
531
	static gboolean first_time=TRUE;
532 533
	unsigned long id;
	static unsigned long previd=0;
534
	static unsigned long preview_previd=0;
smorlat's avatar
smorlat committed
535
	static gboolean in_iterate=FALSE;
536

smorlat's avatar
smorlat committed
537 538 539
	/*avoid reentrancy*/
	if (in_iterate) return TRUE;
	in_iterate=TRUE;
aymeric's avatar
aymeric committed
540
	linphone_core_iterate(lc);
541 542 543 544 545 546
	if (first_time){
		/*after the first call to iterate, SipSetupContexts should be ready, so take actions:*/
		linphone_gtk_show_directory_search();
		first_time=FALSE;
	}

547
	id=linphone_core_get_native_video_window_id(lc);
smorlat's avatar
smorlat committed
548
	if (id!=previd || video_needs_update){
549 550 551
		GdkWindow *w;
		previd=id;
		if (id!=0){
smorlat's avatar
smorlat committed
552
			ms_message("Updating window decorations");
553
#ifndef WIN32
554
			w=gdk_window_foreign_new(id);
555 556
#else
			w=gdk_window_foreign_new((HANDLE)id);
557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575
#endif
			if (w) {
				set_video_window_decorations(w);
				g_object_unref(G_OBJECT(w));
			}
			else ms_error("gdk_window_foreign_new() failed");
			if (video_needs_update) video_needs_update=FALSE;
		}
	}
	id=linphone_core_get_native_preview_window_id (lc);
	if (id!=preview_previd ){
		GdkWindow *w;
		preview_previd=id;
		if (id!=0){
			ms_message("Updating window decorations for preview");
#ifndef WIN32
			w=gdk_window_foreign_new(id);
#else
			w=gdk_window_foreign_new((HANDLE)id);
576
#endif
577 578 579 580 581
			if (w) {
				set_video_window_decorations(w);
				g_object_unref(G_OBJECT(w));
			}
			else ms_error("gdk_window_foreign_new() failed");
smorlat's avatar
smorlat committed
582
			if (video_needs_update) video_needs_update=FALSE;
583 584
		}
	}
585
	if (addr_to_call!=NULL){
smorlat's avatar
smorlat committed
586 587 588 589 590 591 592 593 594
		/*make sure we are not showing the login screen*/
		GtkWidget *mw=linphone_gtk_get_main_window();
		GtkWidget *login_frame=linphone_gtk_get_widget(mw,"login_frame");
		if (!GTK_WIDGET_VISIBLE(login_frame)){
			GtkWidget *uri_bar=linphone_gtk_get_widget(mw,"uribar");
			gtk_entry_set_text(GTK_ENTRY(uri_bar),addr_to_call);
			addr_to_call=NULL;
			linphone_gtk_start_call(uri_bar);
		}
595
	}
smorlat's avatar
smorlat committed
596
	in_iterate=FALSE;
aymeric's avatar
aymeric committed
597 598 599 600 601 602 603 604 605 606 607 608
	return TRUE;
}

static void load_uri_history(){
	GtkEntry *uribar=GTK_ENTRY(linphone_gtk_get_widget(linphone_gtk_get_main_window(),"uribar"));
	char key[20];
	int i;
	GtkEntryCompletion *gep=gtk_entry_completion_new();
	GtkListStore *model=gtk_list_store_new(1,G_TYPE_STRING);
	for (i=0;;i++){
		const char *uri;
		snprintf(key,sizeof(key),"uri%i",i);
609
		uri=linphone_gtk_get_ui_config(key,NULL);
aymeric's avatar
aymeric committed
610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671
		if (uri!=NULL) {
			GtkTreeIter iter;
			gtk_list_store_append(model,&iter);
			gtk_list_store_set(model,&iter,0,uri,-1);
			if (i==0) gtk_entry_set_text(uribar,uri);
		}
		else break;
	}
	gtk_entry_completion_set_model(gep,GTK_TREE_MODEL(model));
	gtk_entry_completion_set_text_column(gep,0);
	gtk_entry_set_completion(uribar,gep);
}

static void save_uri_history(){
	LinphoneCore *lc=linphone_gtk_get_core();
	LpConfig *cfg=linphone_core_get_config(lc);
	GtkEntry *uribar=GTK_ENTRY(linphone_gtk_get_widget(linphone_gtk_get_main_window(),"uribar"));
	char key[20];
	int i=0;
	char *uri=NULL;
	GtkTreeIter iter;
	GtkTreeModel *model=gtk_entry_completion_get_model(gtk_entry_get_completion(uribar));

	if (!gtk_tree_model_get_iter_first(model,&iter)) return;
	do {
		gtk_tree_model_get(model,&iter,0,&uri,-1);
		if (uri) {
			snprintf(key,sizeof(key),"uri%i",i);
			lp_config_set_string(cfg,"GtkUi",key,uri);
			g_free(uri);
		}else break;
		i++;
		if (i>5) break;
	}while(gtk_tree_model_iter_next(model,&iter));
	lp_config_sync(cfg);
}

static void completion_add_text(GtkEntry *entry, const char *text){
	GtkTreeIter iter;
	GtkTreeModel *model=gtk_entry_completion_get_model(gtk_entry_get_completion(entry));
	
	if (gtk_tree_model_get_iter_first(model,&iter)){ 
		do {
			gchar *uri=NULL;
			gtk_tree_model_get(model,&iter,0,&uri,-1);
			if (uri!=NULL){
				if (strcmp(uri,text)==0) {
					/*remove text */
					gtk_list_store_remove(GTK_LIST_STORE(model),&iter);
					g_free(uri);
					break;
				}
				g_free(uri);
			}
		}while (gtk_tree_model_iter_next(model,&iter));
	}
	/* and prepend it on top of the list */
	gtk_list_store_prepend(GTK_LIST_STORE(model),&iter);
	gtk_list_store_set(GTK_LIST_STORE(model),&iter,0,text,-1);
	save_uri_history();
}

672

673 674 675 676 677
bool_t linphone_gtk_video_enabled(void){
	const LinphoneVideoPolicy *vpol=linphone_core_get_video_policy(linphone_gtk_get_core());
	return vpol->automatically_accept && vpol->automatically_initiate;
}

678
void linphone_gtk_show_main_window(){
679 680
	GtkWidget *w=linphone_gtk_get_main_window();
	LinphoneCore *lc=linphone_gtk_get_core();
681
	linphone_core_enable_video_preview(lc,linphone_gtk_get_ui_config_int("videoselfview",
682 683 684 685 686
	    	VIDEOSELFVIEW_DEFAULT));
	gtk_widget_show(w);
	gtk_window_present(GTK_WINDOW(w));
}

687
void linphone_gtk_call_terminated(LinphoneCall *call, const char *error){
smorlat's avatar
smorlat committed
688
	GtkWidget *mw=linphone_gtk_get_main_window();
689 690 691 692
	if (linphone_core_get_calls(linphone_gtk_get_core())==NULL){
	    gtk_widget_set_sensitive(linphone_gtk_get_widget(mw,"terminate_call"),FALSE);
	    gtk_widget_set_sensitive(linphone_gtk_get_widget(mw,"start_call"),TRUE);
	}
693 694
	if (linphone_gtk_use_in_call_view() && call)
		linphone_gtk_in_call_view_terminate(call,error);
smorlat's avatar
smorlat committed
695
	update_video_title();
smorlat's avatar
smorlat committed
696 697
}

698 699 700 701 702 703 704 705
static void linphone_gtk_update_call_buttons(LinphoneCall *call){
	LinphoneCore *lc=linphone_gtk_get_core();
	GtkWidget *mw=linphone_gtk_get_main_window();
	const MSList *calls=linphone_core_get_calls(lc);
	GtkWidget *button;
	bool_t start_active=TRUE;
	bool_t stop_active=FALSE;
	bool_t add_call=FALSE;
706
	int call_list_size=ms_list_size(calls);
707 708 709 710
	
	if (calls==NULL){
		start_active=TRUE;
		stop_active=FALSE;
Simon Morlat's avatar
Simon Morlat committed
711
	}else{
712 713 714
		stop_active=TRUE;	
		start_active=TRUE;
		add_call=TRUE;
715 716 717 718 719 720
	}
	button=linphone_gtk_get_widget(mw,"start_call");
	gtk_widget_set_sensitive(button,start_active);
	gtk_widget_set_visible(button,!add_call);
	
	button=linphone_gtk_get_widget(mw,"add_call");
721 722 723 724 725
	if (linphone_core_sound_resources_locked(lc) || (call && linphone_call_get_state(call)==LinphoneCallIncomingReceived)) {
		gtk_widget_set_sensitive(button,FALSE);
	} else {
		gtk_widget_set_sensitive(button,start_active);
	}
726 727 728
	gtk_widget_set_visible(button,add_call);
	
	gtk_widget_set_sensitive(linphone_gtk_get_widget(mw,"terminate_call"),stop_active);
729

730
	linphone_gtk_enable_transfer_button(lc,call_list_size>1);
731
	linphone_gtk_enable_conference_button(lc,call_list_size>1);
smorlat's avatar
smorlat committed
732
	update_video_title();
733
	if (call) linphone_gtk_update_video_button(call);
smorlat's avatar
smorlat committed
734 735 736 737
}

static gboolean linphone_gtk_start_call_do(GtkWidget *uri_bar){
	const char *entered=gtk_entry_get_text(GTK_ENTRY(uri_bar));
738
	if (linphone_core_invite(linphone_gtk_get_core(),entered)!=NULL) {
smorlat's avatar
smorlat committed
739 740
		completion_add_text(GTK_ENTRY(uri_bar),entered);
	}else{
741
		linphone_gtk_call_terminated(NULL,NULL);
smorlat's avatar
smorlat committed
742 743
	}
	return FALSE;
aymeric's avatar
aymeric committed
744 745
}

746 747 748 749
static gboolean linphone_gtk_auto_answer(LinphoneCall *call){
	if (linphone_call_get_state(call)==LinphoneCallIncomingReceived){
		linphone_core_accept_call (linphone_gtk_get_core(),call);
		linphone_call_unref(call);
smorlat's avatar
smorlat committed
750
	}
751
	return FALSE;
smorlat's avatar
smorlat committed
752 753
}

754

smorlat's avatar
smorlat committed
755
void linphone_gtk_start_call(GtkWidget *w){
aymeric's avatar
aymeric committed
756
	LinphoneCore *lc=linphone_gtk_get_core();
757 758 759 760 761
	LinphoneCall *call;
	/*change into in-call mode, then do the work later as it might block a bit */
	GtkWidget *mw=gtk_widget_get_toplevel(w);
	GtkWidget *uri_bar=linphone_gtk_get_widget(mw,"uribar");

762
	call=linphone_gtk_get_currently_displayed_call(NULL);
763 764
	if (call!=NULL && linphone_call_get_state(call)==LinphoneCallIncomingReceived){
		linphone_core_accept_call(lc,call);
aymeric's avatar
aymeric committed
765
	}else{
766 767 768
		/*immediately disable the button and delay a bit the execution the linphone_core_invite()
		so that we don't freeze the button. linphone_core_invite() might block for some hundreds of milliseconds*/
		gtk_widget_set_sensitive(linphone_gtk_get_widget(mw,"start_call"),FALSE);
smorlat's avatar
smorlat committed
769
		g_timeout_add(100,(GSourceFunc)linphone_gtk_start_call_do,uri_bar);
aymeric's avatar
aymeric committed
770
	}
771
	
aymeric's avatar
aymeric committed
772 773
}

smorlat's avatar
smorlat committed
774 775 776 777 778
void linphone_gtk_uri_bar_activate(GtkWidget *w){
	linphone_gtk_start_call(w);
}


aymeric's avatar
aymeric committed
779
void linphone_gtk_terminate_call(GtkWidget *button){
780 781 782
	gboolean is_conf;
	LinphoneCall *call=linphone_gtk_get_currently_displayed_call(&is_conf);
	if (call){
783
		linphone_core_terminate_call(linphone_gtk_get_core(),call);
784 785 786
	}else if (is_conf){
		linphone_core_terminate_conference(linphone_gtk_get_core());
	}
aymeric's avatar
aymeric committed
787 788
}

789
void linphone_gtk_decline_clicked(GtkWidget *button){
790
	LinphoneCall *call=linphone_gtk_get_currently_displayed_call(NULL);
791 792
	if (call)
		linphone_core_terminate_call(linphone_gtk_get_core(),call);
aymeric's avatar
aymeric committed
793 794
}

795
void linphone_gtk_answer_clicked(GtkWidget *button){
796
	LinphoneCall *call=linphone_gtk_get_currently_displayed_call(NULL);
Simon Morlat's avatar
Simon Morlat committed
797
	if (call){
798
		linphone_core_accept_call(linphone_gtk_get_core(),call);
799
		linphone_gtk_show_main_window(); /* useful when the button is clicked on a notification */
Simon Morlat's avatar
Simon Morlat committed
800
	}
801 802
}

803
void _linphone_gtk_enable_video(gboolean val){
804 805 806 807 808
	LinphoneVideoPolicy policy={0};
	policy.automatically_initiate=policy.automatically_accept=val;
	linphone_core_enable_video(linphone_gtk_get_core(),TRUE,TRUE);
	linphone_core_set_video_policy(linphone_gtk_get_core(),&policy);
	
809 810
	if (val){
		linphone_core_enable_video_preview(linphone_gtk_get_core(),
811
		linphone_gtk_get_ui_config_int("videoselfview",VIDEOSELFVIEW_DEFAULT));
812 813 814
	}else{
		linphone_core_enable_video_preview(linphone_gtk_get_core(),FALSE);
	}
aymeric's avatar
aymeric committed
815 816
}

817 818 819 820 821 822
void linphone_gtk_enable_video(GtkWidget *w){
	gboolean val=gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(w));
	//GtkWidget *selfview_item=linphone_gtk_get_widget(linphone_gtk_get_main_window(),"selfview_item");
	_linphone_gtk_enable_video(val);
}

smorlat's avatar
smorlat committed
823
void linphone_gtk_enable_self_view(GtkWidget *w){
824 825 826 827 828
	gboolean val=gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(w));
	LinphoneCore *lc=linphone_gtk_get_core();
	linphone_core_enable_video_preview(lc,val);
	linphone_core_enable_self_view(lc,val);
	linphone_gtk_set_ui_config_int("videoselfview",val);
smorlat's avatar
smorlat committed
829 830
}

aymeric's avatar
aymeric committed
831 832 833
void linphone_gtk_used_identity_changed(GtkWidget *w){
	int active=gtk_combo_box_get_active(GTK_COMBO_BOX(w));
	char *sel=gtk_combo_box_get_active_text(GTK_COMBO_BOX(w));
834
	if (sel && strlen(sel)>0){ //avoid a dummy "changed" at gui startup
aymeric's avatar
aymeric committed
835
		linphone_core_set_default_proxy_index(linphone_gtk_get_core(),(active==0) ? -1 : (active-1));
836 837
		linphone_gtk_show_directory_search();
	}
smorlat's avatar
smorlat committed
838
	if (sel) g_free(sel);
aymeric's avatar
aymeric committed
839 840
}

841 842 843 844 845 846 847 848 849 850 851 852

void on_proxy_refresh_button_clicked(GtkWidget *w){
	LinphoneCore *lc=linphone_gtk_get_core();
	MSList const *item=linphone_core_get_proxy_config_list(lc);
	while (item != NULL) {
		LinphoneProxyConfig *lpc=(LinphoneProxyConfig*)item->data;
		linphone_proxy_config_edit(lpc);
		linphone_proxy_config_done(lpc);
		item = item->next;
	}
}

853
static void linphone_gtk_notify_recv(LinphoneCore *lc, LinphoneFriend * fid){
854
	linphone_gtk_show_friends();
aymeric's avatar
aymeric committed
855 856 857 858 859 860 861 862 863 864 865 866 867 868 869
}

static void linphone_gtk_new_subscriber_response(GtkWidget *dialog, guint response_id, LinphoneFriend *lf){
	switch(response_id){
		case GTK_RESPONSE_YES:
			linphone_gtk_show_contact(lf);
		break;
		default:
			linphone_core_reject_subscriber(linphone_gtk_get_core(),lf);
	}
	gtk_widget_destroy(dialog);
}

static void linphone_gtk_new_unknown_subscriber(LinphoneCore *lc, LinphoneFriend *lf, const char *url){
	GtkWidget *dialog;
smorlat's avatar
smorlat committed
870 871 872 873 874 875

	if (linphone_gtk_get_ui_config_int("subscribe_deny_all",0)){
		linphone_core_reject_subscriber(linphone_gtk_get_core(),lf);
		return;
	}

aymeric's avatar
aymeric committed
876 877 878 879 880 881
	gchar *message=g_strdup_printf(_("%s would like to add you to his contact list.\nWould you allow him to see your presence status or add him to your contact list ?\nIf you answer no, this person will be temporarily blacklisted."),url);
	dialog = gtk_message_dialog_new (
				GTK_WINDOW(linphone_gtk_get_main_window()),
                                GTK_DIALOG_DESTROY_WITH_PARENT,
				GTK_MESSAGE_QUESTION,
                                GTK_BUTTONS_YES_NO,
smorlat's avatar
smorlat committed
882
                                "%s",
aymeric's avatar
aymeric committed
883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934
				message);
	g_free(message);
	g_signal_connect(G_OBJECT (dialog), "response",
		G_CALLBACK (linphone_gtk_new_subscriber_response),lf);
	/* actually show the box */
	gtk_widget_show(dialog);
}

typedef struct _AuthTimeout{
	GtkWidget *w;
} AuthTimeout;


static void auth_timeout_clean(AuthTimeout *tout){
	tout->w=NULL;
}

static gboolean auth_timeout_destroy(AuthTimeout *tout){
	if (tout->w)  {
		g_object_weak_unref(G_OBJECT(tout->w),(GWeakNotify)auth_timeout_clean,tout);
		gtk_widget_destroy(tout->w);
	}
	g_free(tout);
	return FALSE;
}

static AuthTimeout * auth_timeout_new(GtkWidget *w){
	AuthTimeout *tout=g_new(AuthTimeout,1);
	tout->w=w;
	/*so that the timeout no more references the widget when it is destroyed:*/
	g_object_weak_ref(G_OBJECT(w),(GWeakNotify)auth_timeout_clean,tout);
	/*so that the widget is automatically destroyed after some time */
	g_timeout_add(30000,(GtkFunction)auth_timeout_destroy,tout);
	return tout;
}

void linphone_gtk_password_cancel(GtkWidget *w){
	LinphoneAuthInfo *info;
	GtkWidget *window=gtk_widget_get_toplevel(w);
	info=(LinphoneAuthInfo*)g_object_get_data(G_OBJECT(window),"auth_info");
	linphone_core_abort_authentication(linphone_gtk_get_core(),info);
	gtk_widget_destroy(window);
}

void linphone_gtk_password_ok(GtkWidget *w){
	GtkWidget *entry;
	GtkWidget *window=gtk_widget_get_toplevel(w);
	LinphoneAuthInfo *info;
	info=(LinphoneAuthInfo*)g_object_get_data(G_OBJECT(window),"auth_info");
	g_object_weak_unref(G_OBJECT(window),(GWeakNotify)linphone_auth_info_destroy,info);
	entry=linphone_gtk_get_widget(window,"password_entry");
	linphone_auth_info_set_passwd(info,gtk_entry_get_text(GTK_ENTRY(entry)));
935 936
	linphone_auth_info_set_userid(info,
		gtk_entry_get_text(GTK_ENTRY(linphone_gtk_get_widget(window,"userid_entry"))));
aymeric's avatar
aymeric committed
937 938 939 940 941 942 943 944 945
	linphone_core_add_auth_info(linphone_gtk_get_core(),info);
	gtk_widget_destroy(window);
}

static void linphone_gtk_auth_info_requested(LinphoneCore *lc, const char *realm, const char *username){
	GtkWidget *w=linphone_gtk_create_window("password");
	GtkWidget *label=linphone_gtk_get_widget(w,"message");
	LinphoneAuthInfo *info;
	gchar *msg;
946 947 948 949 950 951 952 953
	GtkWidget *mw=linphone_gtk_get_main_window();
	
	if (mw && GTK_WIDGET_VISIBLE(linphone_gtk_get_widget(mw,"login_frame"))){
		/*don't prompt for authentication when login frame is visible*/
		linphone_core_abort_authentication(lc,NULL);
		return;
	}

954 955 956
	msg=g_strdup_printf(_("Please enter your password for username <i>%s</i>\n at domain <i>%s</i>:"),
		username,realm);
	gtk_label_set_markup(GTK_LABEL(label),msg);
aymeric's avatar
aymeric committed
957
	g_free(msg);
958
	gtk_entry_set_text(GTK_ENTRY(linphone_gtk_get_widget(w,"userid_entry")),username);
aymeric's avatar
aymeric committed
959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990
	info=linphone_auth_info_new(username, NULL, NULL, NULL,realm);
	g_object_set_data(G_OBJECT(w),"auth_info",info);
	g_object_weak_ref(G_OBJECT(w),(GWeakNotify)linphone_auth_info_destroy,info);
	gtk_widget_show(w);
	auth_timeout_new(w);
}

static void linphone_gtk_display_status(LinphoneCore *lc, const char *status){
	GtkWidget *w=linphone_gtk_get_main_window();
	GtkWidget *status_bar=linphone_gtk_get_widget(w,"status_bar");
	gtk_statusbar_push(GTK_STATUSBAR(status_bar),
			gtk_statusbar_get_context_id(GTK_STATUSBAR(status_bar),""),
			status);
}

static void linphone_gtk_display_message(LinphoneCore *lc, const char *msg){
	linphone_gtk_display_something(GTK_MESSAGE_INFO,msg);
}

static void linphone_gtk_display_warning(LinphoneCore *lc, const char *warning){
	linphone_gtk_display_something(GTK_MESSAGE_WARNING,warning);
}

static void linphone_gtk_display_url(LinphoneCore *lc, const char *msg, const char *url){
	char richtext[4096];
	snprintf(richtext,sizeof(richtext),"%s %s",msg,url);
	linphone_gtk_display_something(GTK_MESSAGE_INFO,richtext);
}

static void linphone_gtk_call_log_updated(LinphoneCore *lc, LinphoneCallLog *cl){
	GtkWidget *w=(GtkWidget*)g_object_get_data(G_OBJECT(linphone_gtk_get_main_window()),"call_logs");
	if (w) linphone_gtk_call_log_update(w);
991
	linphone_gtk_call_log_update(linphone_gtk_get_main_window());
aymeric's avatar
aymeric committed
992 993
}

994
#ifdef HAVE_NOTIFY
995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020
static bool_t notify_actions_supported() {
	bool_t accepts_actions = FALSE;
	GList *capabilities = notify_get_server_caps();
	GList *c;
	if(capabilities != NULL) {
		for(c = capabilities; c != NULL; c = c->next) {
			if(strcmp((char*)c->data, "actions") == 0 ) {
				accepts_actions = TRUE;
				break;
			}
		}
		g_list_foreach(capabilities, (GFunc)g_free, NULL);
		g_list_free(capabilities);
	}
	return accepts_actions;
}

static NotifyNotification* build_notification(const char *title, const char *body){
	 return notify_notification_new(title,body,linphone_gtk_get_ui_config("icon",LINPHONE_ICON)
#ifdef HAVE_NOTIFY1
        ,NULL
#endif
	);
}

static void show_notification(NotifyNotification* n){
1021
	if (n && !notify_notification_show(n,NULL))
1022 1023 1024 1025 1026
		ms_error("Failed to send notification.");
}

static void make_notification(const char *title, const char *body){
	show_notification(build_notification(title,body));
1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037
}

#endif

static void linphone_gtk_notify(LinphoneCall *call, const char *msg){
#ifdef HAVE_NOTIFY
	if (!notify_is_initted())
		if (!notify_init ("Linphone")) ms_error("Libnotify failed to init.");
#endif
	if (!call) {
#ifdef HAVE_NOTIFY
1038 1039 1040 1041 1042
		if (!notify_notification_show(notify_notification_new("Linphone",msg,NULL
#ifdef HAVE_NOTIFY1
	,NULL
#endif
),NULL))
1043 1044 1045 1046 1047 1048 1049 1050
				ms_error("Failed to send notification.");
#else
		linphone_gtk_show_main_window();
#endif
	} else if (!gtk_window_is_active((GtkWindow*)linphone_gtk_get_main_window())) {
#ifdef HAVE_NOTIFY
		char *body=NULL;
		char *remote=call!=NULL ? linphone_call_get_remote_address_as_string(call) : NULL;
1051
		NotifyNotification *n;
1052 1053 1054 1055 1056 1057 1058 1059
		switch(linphone_call_get_state(call)){
			case LinphoneCallError:
				make_notification(_("Call error"),body=g_markup_printf_escaped("<span size=\"large\">%s</span>\n%s",msg,remote));
			break;
			case LinphoneCallEnd:
				make_notification(_("Call ended"),body=g_markup_printf_escaped("<span size=\"large\">%s</span>",remote));
			break;
			case LinphoneCallIncomingReceived:
1060 1061 1062 1063 1064 1065 1066 1067
				n=build_notification(_("Incoming call"),body=g_markup_printf_escaped("<span size=\"large\">%s</span>",remote));
				if (notify_actions_supported()) {
					notify_notification_add_action (n,"answer", _("Answer"),
						NOTIFY_ACTION_CALLBACK(linphone_gtk_answer_clicked),NULL,NULL);
					notify_notification_add_action (n,"decline",_("Decline"),
						NOTIFY_ACTION_CALLBACK(linphone_gtk_decline_clicked),NULL,NULL);
				}
				show_notification(n);
1068 1069
			break;
			case LinphoneCallPausedByRemote:
1070
				make_notification(_("Call paused"),body=g_markup_printf_escaped(_("<span size=\"large\">by %s</span>"),remote));
1071 1072 1073 1074 1075 1076 1077 1078 1079 1080
			break;
			default:
			break;
		}
		if (body) g_free(body);
		if (remote) g_free(remote);
#endif
	}
}

1081 1082 1083 1084 1085 1086 1087 1088 </