libpurple/purpleui.c

Mon, 12 May 2025 20:25:16 -0500

author
Gary Kramlich <grim@reaperworld.com>
date
Mon, 12 May 2025 20:25:16 -0500
changeset 43246
0e9bbe9b4da8
parent 42995
006b071042ed
permissions
-rw-r--r--

Handle formatting in server messages

Testing Done:
Used the default motd on my local ergo server to verify that formatting was working.

Reviewed at https://reviews.imfreedom.org/r/3991/

/*
 * Purple - Internet Messaging Library
 * Copyright (C) Pidgin Developers <devel@pidgin.im>
 *
 * Purple 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 library 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 library 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 library; if not, see <https://www.gnu.org/licenses/>.
 */

#include "purpleui.h"

#include "util.h"

typedef struct {
	char *id;
	char *name;
	char *version;
	char *website;
	char *support_website;
	char *client_type;
} PurpleUiPrivate;

enum {
	PROP_0,
	PROP_ID,
	PROP_NAME,
	PROP_VERSION,
	PROP_WEBSITE,
	PROP_SUPPORT_WEBSITE,
	PROP_CLIENT_TYPE,
	N_PROPERTIES,
};
static GParamSpec *properties[N_PROPERTIES] = {NULL, };

enum {
	SIG_STARTED,
	N_SIGNALS,
};
static guint signals[N_SIGNALS] = {0, };

G_DEFINE_ABSTRACT_TYPE_WITH_PRIVATE(PurpleUi, purple_ui, G_TYPE_OBJECT)

/******************************************************************************
 * Helpers
 *****************************************************************************/
static void
purple_ui_set_id(PurpleUi *ui, const char *id) {
	PurpleUiPrivate *priv = NULL;

	g_return_if_fail(PURPLE_IS_UI(ui));

	priv = purple_ui_get_instance_private(ui);

	if(g_set_str(&priv->id, id)) {
		g_object_notify_by_pspec(G_OBJECT(ui), properties[PROP_ID]);
	}
}

static void
purple_ui_set_name(PurpleUi *ui, const char *name) {
	PurpleUiPrivate *priv = NULL;

	g_return_if_fail(PURPLE_IS_UI(ui));

	priv = purple_ui_get_instance_private(ui);

	if(g_set_str(&priv->name, name)) {
		g_object_notify_by_pspec(G_OBJECT(ui), properties[PROP_NAME]);
	}
}

static void
purple_ui_set_version(PurpleUi *ui, const char *version) {
	PurpleUiPrivate *priv = NULL;

	g_return_if_fail(PURPLE_IS_UI(ui));

	priv = purple_ui_get_instance_private(ui);

	if(g_set_str(&priv->version, version)) {
		g_object_notify_by_pspec(G_OBJECT(ui), properties[PROP_VERSION]);
	}
}

static void
purple_ui_set_website(PurpleUi *ui, const char *website) {
	PurpleUiPrivate *priv = NULL;

	g_return_if_fail(PURPLE_IS_UI(ui));

	priv = purple_ui_get_instance_private(ui);

	if(g_set_str(&priv->website, website)) {
		g_object_notify_by_pspec(G_OBJECT(ui), properties[PROP_WEBSITE]);
	}
}

static void
purple_ui_set_support_website(PurpleUi *ui, const char *support_website) {
	PurpleUiPrivate *priv = NULL;

	g_return_if_fail(PURPLE_IS_UI(ui));

	priv = purple_ui_get_instance_private(ui);

	if(g_set_str(&priv->support_website, support_website)) {
		g_object_notify_by_pspec(G_OBJECT(ui),
		                         properties[PROP_SUPPORT_WEBSITE]);
	}
}

static void
purple_ui_set_client_type(PurpleUi *ui, const char *client_type) {
	PurpleUiPrivate *priv = NULL;

	g_return_if_fail(PURPLE_IS_UI(ui));

	priv = purple_ui_get_instance_private(ui);

	if(g_set_str(&priv->client_type, client_type)) {
		g_object_notify_by_pspec(G_OBJECT(ui), properties[PROP_CLIENT_TYPE]);
	}
}

/******************************************************************************
 * GObject Implementation
 *****************************************************************************/
static void
purple_ui_get_property(GObject *obj, guint param_id, GValue *value,
                       GParamSpec *pspec)
{
	PurpleUi *ui = PURPLE_UI(obj);

	switch(param_id) {
	case PROP_ID:
		g_value_set_string(value, purple_ui_get_id(ui));
		break;
	case PROP_NAME:
		g_value_set_string(value, purple_ui_get_name(ui));
		break;
	case PROP_VERSION:
		g_value_set_string(value, purple_ui_get_version(ui));
		break;
	case PROP_WEBSITE:
		g_value_set_string(value, purple_ui_get_website(ui));
		break;
	case PROP_SUPPORT_WEBSITE:
		g_value_set_string(value, purple_ui_get_support_website(ui));
		break;
	case PROP_CLIENT_TYPE:
		g_value_set_string(value, purple_ui_get_client_type(ui));
		break;
	default:
		G_OBJECT_WARN_INVALID_PROPERTY_ID(obj, param_id, pspec);
		break;
	}
}

static void
purple_ui_set_property(GObject *obj, guint param_id, const GValue *value,
                       GParamSpec *pspec)
{
	PurpleUi *ui = PURPLE_UI(obj);

	switch(param_id) {
	case PROP_ID:
		purple_ui_set_id(ui, g_value_get_string(value));
		break;
	case PROP_NAME:
		purple_ui_set_name(ui, g_value_get_string(value));
		break;
	case PROP_VERSION:
		purple_ui_set_version(ui, g_value_get_string(value));
		break;
	case PROP_WEBSITE:
		purple_ui_set_website(ui, g_value_get_string(value));
		break;
	case PROP_SUPPORT_WEBSITE:
		purple_ui_set_support_website(ui, g_value_get_string(value));
		break;
	case PROP_CLIENT_TYPE:
		purple_ui_set_client_type(ui, g_value_get_string(value));
		break;
	default:
		G_OBJECT_WARN_INVALID_PROPERTY_ID(obj, param_id, pspec);
		break;
	}
}

static void
purple_ui_finalize(GObject *obj) {
	PurpleUiPrivate *priv = purple_ui_get_instance_private(PURPLE_UI(obj));

	g_clear_pointer(&priv->id, g_free);
	g_clear_pointer(&priv->name, g_free);
	g_clear_pointer(&priv->version, g_free);
	g_clear_pointer(&priv->website, g_free);
	g_clear_pointer(&priv->support_website, g_free);
	g_clear_pointer(&priv->client_type, g_free);

	G_OBJECT_CLASS(purple_ui_parent_class)->finalize(obj);
}

static void
purple_ui_init(G_GNUC_UNUSED PurpleUi *ui) {
}

static void
purple_ui_class_init(PurpleUiClass *klass) {
	GObjectClass *obj_class = G_OBJECT_CLASS(klass);

	obj_class->get_property = purple_ui_get_property;
	obj_class->set_property = purple_ui_set_property;
	obj_class->finalize = purple_ui_finalize;

	/**
	 * PurpleUi:id:
	 *
	 * The identifier of the user interface. This is used in places where a
	 * constant string is need to represent the user interface.
	 *
	 * Since: 3.0
	 */
	properties[PROP_ID] = g_param_spec_string(
		"id", NULL, NULL,
		NULL,
		G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS);

	/**
	 * PurpleUi:name:
	 *
	 * The name of the user interface. This is used in places where it will be
	 * displayed to users, so it should be translated.
	 *
	 * Since: 3.0
	 */
	properties[PROP_NAME] = g_param_spec_string(
		"name", NULL, NULL,
		NULL,
		G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS);

	/**
	 * PurpleUi:version:
	 *
	 * The version number of the user interface.
	 *
	 * Since: 3.0
	 */
	properties[PROP_VERSION] = g_param_spec_string(
		"version", NULL, NULL,
		NULL,
		G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS);

	/**
	 * PurpleUi:website:
	 *
	 * The website of the user interface. This should be the main website.
	 *
	 * Since: 3.0
	 */
	properties[PROP_WEBSITE] = g_param_spec_string(
		"website", NULL, NULL,
		NULL,
		G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS);

	/**
	 * PurpleUi:support-website:
	 *
	 * The support website of the user interface. This should link to a page
	 * that specifically directs users how to get support for your user
	 * interface.
	 *
	 * Since: 3.0
	 */
	properties[PROP_SUPPORT_WEBSITE] = g_param_spec_string(
		"support-website", NULL, NULL,
		NULL,
		G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS);

	/**
	 * PurpleUi:client-type:
	 *
	 * The client type of the user interface. Common values include `bot`,
	 * `console`, `mobile`, `pc`, `web`, etc.
	 *
	 * Since: 3.0
	 */
	properties[PROP_CLIENT_TYPE] = g_param_spec_string(
		"client-type", NULL, NULL,
		NULL,
		G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS);

	g_object_class_install_properties(obj_class, N_PROPERTIES, properties);

	/**
	 * PurpleUi::started:
	 *
	 * Emitted if [method@Ui.start] completed successfully.
	 *
	 * Since: 3.0
	 */
	signals[SIG_STARTED] = g_signal_new_class_handler(
		"started",
		G_OBJECT_CLASS_TYPE(klass),
		G_SIGNAL_RUN_LAST,
		NULL,
		NULL,
		NULL,
		NULL,
		G_TYPE_NONE,
		0);
}

/******************************************************************************
 * Public API
 *****************************************************************************/
PurpleConversationManager *
purple_ui_get_conversation_manager(PurpleUi *ui) {
	PurpleUiClass *klass = NULL;

	g_return_val_if_fail(PURPLE_IS_UI(ui), NULL);

	klass = PURPLE_UI_GET_CLASS(ui);
	if(klass != NULL && klass->get_conversation_manager != NULL) {
		return klass->get_conversation_manager(ui);
	}

	return NULL;
}

const char *
purple_ui_get_id(PurpleUi *ui) {
	PurpleUiPrivate *priv = NULL;

	g_return_val_if_fail(PURPLE_IS_UI(ui), NULL);

	priv = purple_ui_get_instance_private(ui);

	return priv->id;
}

const char *
purple_ui_get_name(PurpleUi *ui) {
	PurpleUiPrivate *priv = NULL;

	g_return_val_if_fail(PURPLE_IS_UI(ui), NULL);

	priv = purple_ui_get_instance_private(ui);

	return priv->name;
}

const char *
purple_ui_get_version(PurpleUi *ui) {
	PurpleUiPrivate *priv = NULL;

	g_return_val_if_fail(PURPLE_IS_UI(ui), NULL);

	priv = purple_ui_get_instance_private(ui);

	return priv->version;
}

const char *
purple_ui_get_website(PurpleUi *ui) {
	PurpleUiPrivate *priv = NULL;

	g_return_val_if_fail(PURPLE_IS_UI(ui), NULL);

	priv = purple_ui_get_instance_private(ui);

	return priv->website;
}

const char *
purple_ui_get_support_website(PurpleUi *ui) {
	PurpleUiPrivate *priv = NULL;

	g_return_val_if_fail(PURPLE_IS_UI(ui), NULL);

	priv = purple_ui_get_instance_private(ui);

	return priv->support_website;
}

const char *
purple_ui_get_client_type(PurpleUi *ui) {
	PurpleUiPrivate *priv = NULL;

	g_return_val_if_fail(PURPLE_IS_UI(ui), NULL);

	priv = purple_ui_get_instance_private(ui);

	return priv->client_type;
}

void
purple_ui_open_uri(PurpleUi *ui, const char *uri, GCancellable *cancellable,
                   GAsyncReadyCallback callback, gpointer data)
{
	PurpleUiClass *klass = NULL;

	g_return_if_fail(PURPLE_IS_UI(ui));
	g_return_if_fail(!purple_strempty(uri));

	klass = PURPLE_UI_GET_CLASS(ui);
	if(klass && klass->open_uri) {
		klass->open_uri(ui, uri, cancellable, callback, data);
	} else {
		g_task_report_new_error(G_OBJECT(ui), callback, data,
		                        purple_ui_open_uri,
		                        PURPLE_UI_ERROR, 0,
		                        "%s does not implement URI handling",
		                        G_OBJECT_TYPE_NAME(ui));
	}
}

gboolean
purple_ui_open_uri_finish(PurpleUi *ui, GAsyncResult *result, GError **error) {
	PurpleUiClass *klass = NULL;

	g_return_val_if_fail(PURPLE_IS_UI(ui), FALSE);
	g_return_val_if_fail(G_IS_ASYNC_RESULT(result), FALSE);

	if(g_async_result_is_tagged(result, purple_ui_open_uri)) {
		return g_task_propagate_boolean(G_TASK(result), error);
	}

	klass = PURPLE_UI_GET_CLASS(ui);
	if(klass != NULL && klass->open_uri_finish != NULL) {
		return klass->open_uri_finish(ui, result, error);
	}

	return FALSE;
}

void
purple_ui_prefs_init(PurpleUi *ui) {
	PurpleUiClass *klass = NULL;

	g_return_if_fail(PURPLE_IS_UI(ui));

	klass = PURPLE_UI_GET_CLASS(ui);
	if(klass != NULL && klass->prefs_init != NULL) {
		klass->prefs_init(ui);
	}
}

gboolean
purple_ui_start(PurpleUi *ui, GError **error) {
	PurpleUiClass *klass = NULL;

	g_return_val_if_fail(PURPLE_IS_UI(ui), FALSE);

	klass = PURPLE_UI_GET_CLASS(ui);
	if(klass != NULL && klass->start != NULL) {
		if(klass->start(ui, error)) {
			g_signal_emit(ui, signals[SIG_STARTED], 0);

			return TRUE;
		}

		return FALSE;
	}

	return FALSE;
}

void
purple_ui_stop(PurpleUi *ui) {
	PurpleUiClass *klass = NULL;

	g_return_if_fail(PURPLE_IS_UI(ui));

	klass = PURPLE_UI_GET_CLASS(ui);
	if(klass != NULL && klass->stop != NULL) {
		klass->stop(ui);
	}
}

gpointer
purple_ui_get_settings_backend(PurpleUi *ui) {
	PurpleUiClass *klass = NULL;

	g_return_val_if_fail(PURPLE_IS_UI(ui), NULL);

	klass = PURPLE_UI_GET_CLASS(ui);
	if(klass != NULL && klass->get_settings_backend != NULL) {
		return klass->get_settings_backend(ui);
	}

	return NULL;
}

PurpleHistoryAdapter *
purple_ui_get_history_adapter(PurpleUi *ui) {
	PurpleUiClass *klass = NULL;

	g_return_val_if_fail(PURPLE_IS_UI(ui), NULL);

	klass = PURPLE_UI_GET_CLASS(ui);
	if(klass != NULL && klass->get_history_adapter != NULL) {
		return klass->get_history_adapter(ui);
	}

	return NULL;
}

PurplePresenceManager *
purple_ui_get_presence_manager(PurpleUi *ui) {
	PurpleUiClass *klass = NULL;

	g_return_val_if_fail(PURPLE_IS_UI(ui), NULL);

	klass = PURPLE_UI_GET_CLASS(ui);
	if(klass != NULL && klass->get_presence_manager != NULL) {
		return klass->get_presence_manager(ui);
	}

	return NULL;
}

mercurial