grim/pidgin
Clone
Summary
Browse
Changes
Graph
Add an escaped-name property to PurpleSavedPresence
11 months ago, Gary Kramlich
0c86a3fa4617
Add an escaped-name property to PurpleSavedPresence
This is mean to be used for serialization, name in GSettings paths. We still
need to determine what to do if the name is NULL, but maybe the serialization
routine can handle that.
Testing Done:
Ran the unit tests
Reviewed at https://reviews.imfreedom.org/r/2492/
/*
* Pidgin - Internet Messenger
* Copyright (C) Pidgin Developers <devel@pidgin.im>
*
* Pidgin is the legal property of its developers, whose names are too numerous
* to list here. Please refer to the COPYRIGHT file distributed with this
* source distribution.
*
* 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, see <https://www.gnu.org/licenses/>.
*/
#include
<glib/gi18n-lib.h>
#include
<gtk/gtk.h>
#include
<purple.h>
#include
"pidginstatusbox.h"
#include
"pidginiconname.h"
#include
"pidginstatusdisplay.h"
#define SAVEDSTATUS_FORMAT "savedstatus_%lu"
struct
_PidginStatusBox
{
GtkBox
parent
;
GtkWidget
*
button
;
PidginStatusDisplay
*
display
;
GtkPopover
*
popover
;
GMenu
*
primitives
;
GMenu
*
saved_statuses
;
GList
*
custom_widgets
;
};
G_DEFINE_TYPE
(
PidginStatusBox
,
pidgin_status_box
,
GTK_TYPE_BOX
)
/******************************************************************************
* Helpers
*****************************************************************************/
static
GtkWidget
*
pidgin_status_box_make_primitive_widget
(
const
char
*
action
,
const
char
*
id
)
{
GtkWidget
*
button
=
NULL
;
GtkWidget
*
display
=
NULL
;
PurpleStatusPrimitive
primitive
=
PURPLE_STATUS_UNSET
;
primitive
=
purple_primitive_get_type_from_id
(
id
);
display
=
pidgin_status_display_new_for_primitive
(
primitive
);
button
=
gtk_button_new
();
gtk_widget_add_css_class
(
button
,
"flat"
);
gtk_button_set_child
(
GTK_BUTTON
(
button
),
display
);
gtk_actionable_set_action_name
(
GTK_ACTIONABLE
(
button
),
action
);
gtk_actionable_set_action_target
(
GTK_ACTIONABLE
(
button
),
(
const
char
*
)
G_VARIANT_TYPE_INT32
,
primitive
);
return
button
;
}
static
void
pidgin_status_box_populate_primitives
(
PidginStatusBox
*
status_box
)
{
GtkPopoverMenu
*
popover
=
GTK_POPOVER_MENU
(
status_box
->
popover
);
GMenuModel
*
menu
=
G_MENU_MODEL
(
status_box
->
primitives
);
gint
n_items
=
0
;
n_items
=
g_menu_model_get_n_items
(
menu
);
for
(
gint
index
=
0
;
index
<
n_items
;
index
++
)
{
GtkWidget
*
button
=
NULL
;
char
*
action
=
NULL
;
char
*
target
=
NULL
;
char
*
custom_id
=
NULL
;
g_menu_model_get_item_attribute
(
menu
,
index
,
G_MENU_ATTRIBUTE_ACTION
,
"s"
,
&
action
);
g_menu_model_get_item_attribute
(
menu
,
index
,
G_MENU_ATTRIBUTE_TARGET
,
"s"
,
&
target
);
g_menu_model_get_item_attribute
(
menu
,
index
,
"custom"
,
"s"
,
&
custom_id
);
button
=
pidgin_status_box_make_primitive_widget
(
action
,
target
);
gtk_popover_menu_add_child
(
popover
,
button
,
custom_id
);
g_free
(
action
);
g_free
(
target
);
g_free
(
custom_id
);
}
}
static
char
*
pidgin_status_box_make_savedstatus_widget
(
PurpleSavedStatus
*
saved_status
,
GtkWidget
**
widget
)
{
GtkWidget
*
button
=
NULL
;
GtkWidget
*
display
=
NULL
;
time_t
creation_time
=
0
;
display
=
pidgin_status_display_new_for_saved_status
(
saved_status
);
if
(
!
purple_savedstatus_is_transient
(
saved_status
))
{
GtkWidget
*
image
=
gtk_image_new_from_icon_name
(
"document-save"
);
gtk_widget_set_halign
(
image
,
GTK_ALIGN_END
);
gtk_widget_set_hexpand
(
image
,
TRUE
);
gtk_box_append
(
GTK_BOX
(
display
),
image
);
}
button
=
gtk_button_new
();
gtk_widget_add_css_class
(
button
,
"flat"
);
gtk_button_set_child
(
GTK_BUTTON
(
button
),
display
);
creation_time
=
purple_savedstatus_get_creation_time
(
saved_status
);
gtk_actionable_set_action_name
(
GTK_ACTIONABLE
(
button
),
"status.set-saved"
);
gtk_actionable_set_action_target
(
GTK_ACTIONABLE
(
button
),
(
const
char
*
)
G_VARIANT_TYPE_INT64
,
creation_time
);
*
widget
=
button
;
return
g_strdup_printf
(
SAVEDSTATUS_FORMAT
,
creation_time
);
}
static
void
pidgin_status_box_populate_saved_statuses
(
PidginStatusBox
*
status_box
)
{
GtkPopoverMenu
*
popover_menu
=
NULL
;
GMenu
*
menu
=
NULL
;
GList
*
list
,
*
cur
;
list
=
purple_savedstatuses_get_popular
(
6
);
if
(
list
==
NULL
)
{
/* Odd... oh well, nothing we can do about it. */
return
;
}
popover_menu
=
GTK_POPOVER_MENU
(
status_box
->
popover
);
menu
=
status_box
->
saved_statuses
;
for
(
cur
=
list
;
cur
!=
NULL
;
cur
=
cur
->
next
)
{
PurpleSavedStatus
*
saved
=
cur
->
data
;
GtkWidget
*
widget
=
NULL
;
char
*
id
=
NULL
;
GMenuItem
*
item
=
NULL
;
id
=
pidgin_status_box_make_savedstatus_widget
(
saved
,
&
widget
);
item
=
g_menu_item_new
(
NULL
,
NULL
);
g_menu_item_set_attribute
(
item
,
"custom"
,
"s"
,
id
);
g_menu_append_item
(
menu
,
item
);
gtk_popover_menu_add_child
(
popover_menu
,
widget
,
id
);
status_box
->
custom_widgets
=
g_list_prepend
(
status_box
->
custom_widgets
,
widget
);
g_free
(
id
);
}
g_list_free
(
list
);
}
/******************************************************************************
* Callbacks
*****************************************************************************/
static
void
pidgin_status_box_set_primitive
(
G_GNUC_UNUSED
GSimpleAction
*
action
,
GVariant
*
parameter
,
gpointer
data
)
{
PidginStatusBox
*
status_box
=
data
;
PurpleSavedStatus
*
saved_status
=
NULL
;
PurpleStatusPrimitive
primitive
;
gtk_menu_button_popdown
(
GTK_MENU_BUTTON
(
status_box
->
button
));
if
(
!
g_variant_is_of_type
(
parameter
,
G_VARIANT_TYPE_INT32
))
{
g_critical
(
"status.set-primitive action parameter is of incorrect type %s"
,
g_variant_get_type_string
(
parameter
));
return
;
}
primitive
=
g_variant_get_int32
(
parameter
);
saved_status
=
purple_savedstatus_find_transient_by_type_and_message
(
primitive
,
NULL
);
if
(
saved_status
==
NULL
)
{
saved_status
=
purple_savedstatus_new
(
NULL
,
primitive
);
}
if
(
saved_status
!=
NULL
)
{
if
(
saved_status
!=
purple_savedstatus_get_current
())
{
purple_savedstatus_activate
(
saved_status
);
}
}
}
static
void
pidgin_status_box_set_saved_status
(
G_GNUC_UNUSED
GSimpleAction
*
action
,
GVariant
*
parameter
,
gpointer
data
)
{
PidginStatusBox
*
status_box
=
data
;
PurpleSavedStatus
*
saved_status
=
NULL
;
time_t
creation_time
=
0
;
gtk_menu_button_popdown
(
GTK_MENU_BUTTON
(
status_box
->
button
));
if
(
!
g_variant_is_of_type
(
parameter
,
G_VARIANT_TYPE_INT64
))
{
g_critical
(
"status.set-saved action parameter is of incorrect type %s"
,
g_variant_get_type_string
(
parameter
));
return
;
}
creation_time
=
(
time_t
)
g_variant_get_int64
(
parameter
);
saved_status
=
purple_savedstatus_find_by_creation_time
(
creation_time
);
if
(
saved_status
!=
NULL
)
{
if
(
saved_status
!=
purple_savedstatus_get_current
())
{
purple_savedstatus_activate
(
saved_status
);
}
}
}
static
void
pidgin_status_box_savedstatus_changed_cb
(
PurpleSavedStatus
*
now
,
G_GNUC_UNUSED
PurpleSavedStatus
*
old
,
gpointer
data
)
{
PidginStatusBox
*
status_box
=
data
;
/* If we don't have a status, we have to bail. */
if
(
now
==
NULL
)
{
return
;
}
pidgin_status_display_set_saved_status
(
status_box
->
display
,
now
);
}
static
void
pidgin_status_box_remove_custom_widget
(
GtkWidget
*
widget
,
gpointer
user_data
)
{
GtkPopoverMenu
*
popover_menu
=
user_data
;
gtk_popover_menu_remove_child
(
popover_menu
,
widget
);
}
static
void
pidgin_status_box_savedstatus_updated_cb
(
G_GNUC_UNUSED
PurpleSavedStatus
*
status
,
gpointer
data
)
{
PidginStatusBox
*
status_box
=
data
;
g_list_foreach
(
status_box
->
custom_widgets
,
(
GFunc
)
pidgin_status_box_remove_custom_widget
,
status_box
->
popover
);
g_clear_list
(
&
status_box
->
custom_widgets
,
NULL
);
g_menu_remove_all
(
status_box
->
saved_statuses
);
pidgin_status_box_populate_saved_statuses
(
status_box
);
}
/******************************************************************************
* GObject Implementation
*****************************************************************************/
static
void
pidgin_status_box_finalize
(
GObject
*
obj
)
{
PidginStatusBox
*
status_box
=
PIDGIN_STATUS_BOX
(
obj
);
purple_signals_disconnect_by_handle
(
status_box
);
g_clear_list
(
&
status_box
->
custom_widgets
,
NULL
);
G_OBJECT_CLASS
(
pidgin_status_box_parent_class
)
->
finalize
(
obj
);
}
static
void
pidgin_status_box_init
(
PidginStatusBox
*
status_box
)
{
gpointer
handle
;
GSimpleActionGroup
*
action_group
=
NULL
;
GActionEntry
actions
[]
=
{
{
.
name
=
"set-primitive"
,
.
activate
=
pidgin_status_box_set_primitive
,
.
parameter_type
=
(
const
char
*
)
G_VARIANT_TYPE_INT32
,
},
{
.
name
=
"set-saved"
,
.
activate
=
pidgin_status_box_set_saved_status
,
.
parameter_type
=
(
const
char
*
)
G_VARIANT_TYPE_INT64
,
},
};
gtk_widget_init_template
(
GTK_WIDGET
(
status_box
));
action_group
=
g_simple_action_group_new
();
g_action_map_add_action_entries
(
G_ACTION_MAP
(
action_group
),
actions
,
G_N_ELEMENTS
(
actions
),
status_box
);
gtk_widget_insert_action_group
(
GTK_WIDGET
(
status_box
),
"status"
,
G_ACTION_GROUP
(
action_group
));
status_box
->
popover
=
gtk_menu_button_get_popover
(
GTK_MENU_BUTTON
(
status_box
->
button
));
gtk_popover_set_has_arrow
(
status_box
->
popover
,
FALSE
);
pidgin_status_box_populate_primitives
(
status_box
);
pidgin_status_box_populate_saved_statuses
(
status_box
);
handle
=
purple_savedstatuses_get_handle
();
purple_signal_connect
(
handle
,
"savedstatus-changed"
,
status_box
,
G_CALLBACK
(
pidgin_status_box_savedstatus_changed_cb
),
status_box
);
purple_signal_connect
(
handle
,
"savedstatus-added"
,
status_box
,
G_CALLBACK
(
pidgin_status_box_savedstatus_updated_cb
),
status_box
);
purple_signal_connect
(
handle
,
"savedstatus-deleted"
,
status_box
,
G_CALLBACK
(
pidgin_status_box_savedstatus_updated_cb
),
status_box
);
purple_signal_connect
(
handle
,
"savedstatus-modified"
,
status_box
,
G_CALLBACK
(
pidgin_status_box_savedstatus_updated_cb
),
status_box
);
}
static
void
pidgin_status_box_constructed
(
GObject
*
obj
)
{
PidginStatusBox
*
status_box
=
PIDGIN_STATUS_BOX
(
obj
);
PurpleSavedStatus
*
status
=
NULL
;
G_OBJECT_CLASS
(
pidgin_status_box_parent_class
)
->
constructed
(
obj
);
status
=
purple_savedstatus_get_current
();
pidgin_status_display_set_saved_status
(
status_box
->
display
,
status
);
}
static
void
pidgin_status_box_class_init
(
PidginStatusBoxClass
*
klass
)
{
GObjectClass
*
obj_class
=
G_OBJECT_CLASS
(
klass
);
GtkWidgetClass
*
widget_class
=
GTK_WIDGET_CLASS
(
klass
);
obj_class
->
finalize
=
pidgin_status_box_finalize
;
obj_class
->
constructed
=
pidgin_status_box_constructed
;
gtk_widget_class_set_template_from_resource
(
widget_class
,
"/im/pidgin/Pidgin3/Status/box.ui"
);
gtk_widget_class_bind_template_child
(
widget_class
,
PidginStatusBox
,
button
);
gtk_widget_class_bind_template_child
(
widget_class
,
PidginStatusBox
,
display
);
gtk_widget_class_bind_template_child
(
widget_class
,
PidginStatusBox
,
primitives
);
gtk_widget_class_bind_template_child
(
widget_class
,
PidginStatusBox
,
saved_statuses
);
}
/******************************************************************************
* Public API
*****************************************************************************/
GtkWidget
*
pidgin_status_box_new
(
void
)
{
return
g_object_new
(
PIDGIN_TYPE_STATUS_BOX
,
NULL
);
}