diff --git a/apidocs/c/profapi.h b/apidocs/c/profapi.h index fd872dd1..a1578739 100644 --- a/apidocs/c/profapi.h +++ b/apidocs/c/profapi.h @@ -23,16 +23,16 @@ typedef void(*WINDOW_CB)(PROF_WIN_TAG win, char *line); /** Highlights the console window in the status bar. */ void prof_cons_alert(void); -/** +/** Show a message in the console window. @param message the message to print @return 1 on success, 0 on failure */ int prof_cons_show(const char * const message); -/** +/** Show a message in the console, using the specified theme. -Themes can be must be specified in ~/.local/share/profanity/plugin_themes +Themes are specified in ~/.local/share/profanity/plugin_themes @param group the group name in the themes file @param item the item name within the group @param def default colour if the theme cannot be found @@ -206,7 +206,7 @@ int prof_win_show(PROF_WIN_TAG win, char *message); /** Show a message in the plugin window, using the specified theme. -Themes must be specified in ~/.local/share/profanity/plugin_themes +Themes are specified in ~/.local/share/profanity/plugin_themes @param tag The {@link PROF_WIN_TAG} of the window to display the message @param group the group name in the themes file @param key the item name within the group @@ -414,3 +414,45 @@ Reset the message prefix character for specified room. @return 1 on success, 0 on failure */ int prof_room_unset_message_char(char *roomjid); + +/** +Show a message in a chat window. +@param barejid Jabber ID of the recipient +@param message the message to print +@return 1 on success, 0 on failure +*/ +int prof_chat_show(char *barejid, char *message); + +/** +Show a message in a chat window, using the specified theme, and prefix character +Themes are specified in ~/.local/share/profanity/plugin_themes +@param barejid Jabber ID of the recipient +@param group the group name in the themes file or NULL +@param item the item name within the group or NULL +@param def default colour if the theme cannot be found +@param ch The character to prefix the message, or NULL for default behaviour +@param message the message to print +@return 1 on success, 0 on failure +*/ +int prof_chat_show_themed(char *barejid, char *group, char *item, char *def, char *ch, char *message); + +/** +Show a message in a chat room window. +@param barejid Jabber ID of the room +@param message the message to print +@return 1 on success, 0 on failure +*/ +int prof_room_show(char *roomjid, char *message); + +/** +Show a message in a chat room window, using the specified theme, and prefix character +Themes are specified in ~/.local/share/profanity/plugin_themes +@param barejid Jabber ID of the room +@param group the group name in the themes file or NULL +@param item the item name within the group or NULL +@param def default colour if the theme cannot be found +@param ch The character to prefix the message, or NULL for default behaviour +@param message the message to print +@return 1 on success, 0 on failure +*/ +int prof_room_show_themed(char *roomjid, char *group, char *item, char *def, char *ch, char *message); diff --git a/apidocs/c/profhooks.h b/apidocs/c/profhooks.h index d0beed28..cb14e794 100644 --- a/apidocs/c/profhooks.h +++ b/apidocs/c/profhooks.h @@ -61,7 +61,7 @@ void prof_post_chat_message_display(const char * const barejid, const char *cons Called before a chat message is sent @param barejid Jabber ID of the message recipient @param message the message to be sent -@return the new message to send, or NULL to preserve the original message +@return the modified or original message to send, or NULL to cancel sending of the message */ char* prof_pre_chat_message_send(const char * const barejid, const char *message); @@ -93,7 +93,7 @@ void prof_post_room_message_display(const char * const barejid, const char * con Called before a chat room message is sent @param barejid Jabber ID of the room @param message the message to be sent -@return the new message to send, or NULL to preserve the original message +@return the modified or original message to send, or NULL to cancel sending of the message */ char* prof_pre_room_message_send(const char * const barejid, const char *message); @@ -135,7 +135,7 @@ Called before a private chat room message is sent @param barejid Jabber ID of the room @param nick nickname of message recipient @param message the message to be sent -@return the new message to send, or NULL to preserve the original message +@return the modified or original message to send, or NULL to cancel sending of the message */ char* prof_pre_priv_message_send(const char * const barejid, const char * const nick, const char *message); diff --git a/apidocs/python/src/plugin.py b/apidocs/python/src/plugin.py index 7e22d5e5..6e200bad 100644 --- a/apidocs/python/src/plugin.py +++ b/apidocs/python/src/plugin.py @@ -106,7 +106,7 @@ def prof_pre_chat_message_send(barejid, message): :param message: the message to be sent :type barejid: str or unicode :type message: str or unicode - :return: the new message to send, or ``None`` to preserve the original message + :return: the modified or original message to send, or ``None`` to cancel sending of the message :rtype: str or unicode """ pass @@ -158,7 +158,7 @@ def prof_pre_room_message_send(barejid, message): :param message: the message to be sent :type barejid: str or unicode :type message: str or unicode - :return: the new message to send, or ``None`` to preserve the original message + :return: the modified or original message to send, or ``None`` to cancel sending of the message :rtype: str or unicode """ pass @@ -227,7 +227,7 @@ def prof_pre_priv_message_send(barejid, nick, message): :type barejid: str or unicode :type nick: str or unicode :type message: str or unicode - :return: the new message to send, or ``None`` to preserve the original message + :return: the modified or original message to send, or ``None`` to cancel sending of the message :rtype: str or unicode """ pass diff --git a/apidocs/python/src/prof.py b/apidocs/python/src/prof.py index 35981473..6c80e2f2 100644 --- a/apidocs/python/src/prof.py +++ b/apidocs/python/src/prof.py @@ -31,7 +31,7 @@ def cons_show(message): def cons_show_themed(group, key, default, message): """Show a message in the console, using the specified theme.\n - Themes can be must be specified in ``~/.local/share/profanity/plugin_themes`` + Themes are specified in ``~/.local/share/profanity/plugin_themes`` :param group: the group name in the themes file :param key: the item name within the group @@ -376,7 +376,7 @@ def win_show(tag, message): def win_show_themed(tag, group, key, default, message): """Show a message in the plugin window, using the specified theme.\n - Themes must be specified in ``~/.local/share/profanity/plugin_themes`` + Themes are specified in ``~/.local/share/profanity/plugin_themes`` :param tag: The tag of the window to display the message :type tag: str or unicode @@ -797,3 +797,89 @@ def room_unset_message_char(roomjid): prof.room_unset_message_char("ohnoes@conference.chat.org") """ pass + + +def chat_show(barejid, message): + """Show a message in a chat window. + + :param barejid: Jabber ID of the recipient + :param message: the message to print + :type barejid: str or unicode + :type message: str or unicode + :return: ``True`` if the message was printed, ``False`` otherwise + :rtype: boolean + + Example: + :: + prof.chat_show("bob@server.org", "From a plugin in the chat window for bob") + """ + pass + + +def chat_show_themed(barejid, group, key, default, ch, message): + """Show a message a chat window, using the specified theme and prefix character.\n + Themes are specified in ``~/.local/share/profanity/plugin_themes`` + + :param barejid: Jabber ID of the recipient + :param group: the group name in the themes file or ``None`` + :param key: the item name within the group or ``None`` + :param default: default colour if the theme cannot be found or ``None`` + :param ch: The prefix character to show, or ``None`` for default behaviour + :param message: the message to print + :type barejid: str or unicode + :type group: str, unicode or None + :type key: str, unicode or None + :type default: str, unicode or None + :type ch: str or unicode + :type message: str or unicode + :return: ``True`` if the message was printed, ``False`` otherwise + :rtype: boolean + + Example: + :: + prof.chat_show_themed("bob@server.org", "myplugin", "text", None, "!", "Plugin themed message") + """ + pass + + +def room_show(roomjid, message): + """Show a message in a chat room window. + + :param roomjid: Jabber ID of the room + :param message: the message to print + :type roomjid: str or unicode + :type message: str or unicode + :return: ``True`` if the message was printed, ``False`` otherwise + :rtype: boolean + + Example: + :: + prof.room_show("chat@conference.chat.org", "From a plugin in the chat room window") + """ + pass + + +def room_show_themed(roomjid, group, key, default, ch, message): + """Show a message a chat room window, using the specified theme and prefix character.\n + Themes are specified in ``~/.local/share/profanity/plugin_themes`` + + :param roomjid: Jabber ID of the room + :param group: the group name in the themes file or ``None`` + :param key: the item name within the group or ``None`` + :param default: default colour if the theme cannot be found or ``None`` + :param ch: The prefix character to show, or ``None`` for default behaviour + :param message: the message to print + :type roomjid: str or unicode + :type group: str, unicode or None + :type key: str, unicode or None + :type default: str, unicode or None + :type ch: str or unicode + :type message: str or unicode + :return: ``True`` if the message was printed, ``False`` otherwise + :rtype: boolean + + Example: + :: + prof.room_show_themed("chat@conference.chat.org", "myplugin", "text", None, "!", "Plugin themed message") + """ + pass diff --git a/src/event/client_events.c b/src/event/client_events.c index 3ef193b1..bdc25931 100644 --- a/src/event/client_events.c +++ b/src/event/client_events.c @@ -136,6 +136,9 @@ cl_ev_send_msg(ProfChatWin *chatwin, const char *const msg, const char *const oo } char *plugin_msg = plugins_pre_chat_message_send(chatwin->barejid, msg); + if (plugin_msg == NULL) { + return; + } // OTR suported, PGP supported #ifdef HAVE_LIBOTR @@ -218,6 +221,9 @@ void cl_ev_send_muc_msg(ProfMucWin *mucwin, const char *const msg, const char *const oob_url) { char *plugin_msg = plugins_pre_room_message_send(mucwin->roomjid, msg); + if (plugin_msg == NULL) { + return; + } message_send_groupchat(mucwin->roomjid, plugin_msg, oob_url); diff --git a/src/plugins/api.c b/src/plugins/api.c index a1cbd35e..9aa41485 100644 --- a/src/plugins/api.c +++ b/src/plugins/api.c @@ -483,6 +483,7 @@ void api_disco_add_feature(char *plugin_name, char *feature) { if (feature == NULL) { + log_warning("%s", "api_disco_add_feature failed, feature is NULL"); return; } @@ -501,11 +502,13 @@ void api_encryption_reset(const char *const barejid) { if (barejid == NULL) { + log_warning("%s", "api_encryption_reset failed, barejid is NULL"); return; } ProfChatWin *chatwin = wins_get_chat(barejid); if (chatwin == NULL) { + log_warning("%s", "api_encryption_reset failed, could not find chat window for %s", barejid); return; } @@ -528,15 +531,18 @@ int api_chat_set_titlebar_enctext(const char *const barejid, const char *const enctext) { if (enctext == NULL) { + log_warning("%s", "api_chat_set_titlebar_enctext failed, enctext is NULL"); return 0; } if (barejid == NULL) { + log_warning("%s", "api_chat_set_titlebar_enctext failed, barejid is NULL"); return 0; } ProfChatWin *chatwin = wins_get_chat(barejid); if (chatwin == NULL) { + log_warning("%s", "api_chat_set_titlebar_enctext failed, could not find chat window for %s", barejid); return 0; } @@ -549,11 +555,13 @@ int api_chat_unset_titlebar_enctext(const char *const barejid) { if (barejid == NULL) { + log_warning("%s", "api_chat_unset_titlebar_enctext failed, barejid is NULL"); return 0; } ProfChatWin *chatwin = wins_get_chat(barejid); if (chatwin == NULL) { + log_warning("%s", "api_chat_unset_titlebar_enctext failed, could not find chat window for %s", barejid); return 0; } @@ -566,19 +574,23 @@ int api_chat_set_incoming_char(const char *const barejid, const char *const ch) { if (ch == NULL) { + log_warning("%s", "api_chat_set_incoming_char failed, ch is NULL"); return 0; } if (strlen(ch) != 1) { + log_warning("%s", "api_chat_set_incoming_char failed, ch must be a string of length 1"); return 0; } if (barejid == NULL) { + log_warning("%s", "api_chat_set_incoming_char failed, barejid is NULL"); return 0; } ProfChatWin *chatwin = wins_get_chat(barejid); if (chatwin == NULL) { + log_warning("%s", "api_chat_set_incoming_char failed, could not find chat window for %s", barejid); return 0; } @@ -591,11 +603,13 @@ int api_chat_unset_incoming_char(const char *const barejid) { if (barejid == NULL) { + log_warning("%s", "api_chat_unset_incoming_char failed, barejid is NULL"); return 0; } ProfChatWin *chatwin = wins_get_chat(barejid); if (chatwin == NULL) { + log_warning("%s", "api_chat_unset_incoming_char failed, could not find chat window for %s", barejid); return 0; } @@ -608,19 +622,23 @@ int api_chat_set_outgoing_char(const char *const barejid, const char *const ch) { if (ch == NULL) { + log_warning("%s", "api_chat_set_outgoing_char failed, ch is NULL"); return 0; } if (strlen(ch) != 1) { + log_warning("%s", "api_chat_set_outgoing_char failed, ch must be a string of length 1"); return 0; } if (barejid == NULL) { + log_warning("%s", "api_chat_set_outgoing_char failed, barejid is NULL"); return 0; } ProfChatWin *chatwin = wins_get_chat(barejid); if (chatwin == NULL) { + log_warning("%s", "api_chat_set_outgoing_char failed, could not find chat window for %s", barejid); return 0; } @@ -633,11 +651,13 @@ int api_chat_unset_outgoing_char(const char *const barejid) { if (barejid == NULL) { + log_warning("%s", "api_chat_unset_outgoing_char failed, barejid is NULL"); return 0; } ProfChatWin *chatwin = wins_get_chat(barejid); if (chatwin == NULL) { + log_warning("%s", "api_chat_unset_outgoing_char failed, could not find chat window for %s", barejid); return 0; } @@ -650,15 +670,18 @@ int api_room_set_titlebar_enctext(const char *const roomjid, const char *const enctext) { if (enctext == NULL) { + log_warning("%s", "api_room_set_titlebar_enctext failed, enctext is NULL"); return 0; } if (roomjid == NULL) { + log_warning("%s", "api_room_set_titlebar_enctext failed, roomjid is NULL"); return 0; } ProfMucWin *mucwin = wins_get_muc(roomjid); if (mucwin == NULL) { + log_warning("%s", "api_room_set_titlebar_enctext failed, coudl not find room window for %s", roomjid); return 0; } @@ -671,11 +694,13 @@ int api_room_unset_titlebar_enctext(const char *const roomjid) { if (roomjid == NULL) { + log_warning("%s", "api_room_unset_titlebar_enctext failed, roomjid is NULL"); return 0; } ProfMucWin *mucwin = wins_get_muc(roomjid); if (mucwin == NULL) { + log_warning("%s", "api_room_unset_titlebar_enctext failed, coudl not find room window for %s", roomjid); return 0; } @@ -688,19 +713,23 @@ int api_room_set_message_char(const char *const roomjid, const char *const ch) { if (ch == NULL) { + log_warning("%s", "api_room_set_message_char failed, ch is NULL"); return 0; } if (strlen(ch) != 1) { + log_warning("%s", "api_room_set_message_char failed, ch must be a string of length 1"); return 0; } if (roomjid == NULL) { + log_warning("%s", "api_room_set_message_char failed, roomjid is NULL"); return 0; } ProfMucWin *mucwin = wins_get_muc(roomjid); if (mucwin == NULL) { + log_warning("%s", "api_room_set_message_char failed, could not find room window for %s", roomjid); return 0; } @@ -713,11 +742,13 @@ int api_room_unset_message_char(const char *const roomjid) { if (roomjid == NULL) { + log_warning("%s", "api_room_unset_message_char failed, roomjid is NULL"); return 0; } ProfMucWin *mucwin = wins_get_muc(roomjid); if (mucwin == NULL) { + log_warning("%s", "api_room_unset_message_char failed, could not find room window for %s", roomjid); return 0; } @@ -725,3 +756,133 @@ api_room_unset_message_char(const char *const roomjid) return 1; } + +int +api_chat_show(const char *const barejid, const char *message) +{ + if (message == NULL) { + log_warning("%s", "api_chat_show failed, message is NULL"); + return 0; + } + + if (barejid == NULL) { + log_warning("%s", "api_chat_show failed, barejid is NULL"); + return 0; + } + + ProfChatWin *chatwin = wins_get_chat(barejid); + if (chatwin == NULL) { + log_warning("%s", "api_chat_show failed, could not find chat window for %s", barejid); + return 0; + } + + char *parsed = str_replace(message, "\r\n", "\n"); + win_println((ProfWin*)chatwin, THEME_TEXT, '-', "%s", parsed); + free(parsed); + + return 1; +} + +int +api_chat_show_themed(const char *const barejid, const char *const group, const char *const key, const char *const def, + const char *const ch, const char *const message) +{ + if (message == NULL) { + log_warning("%s", "api_chat_show_themed failed, message is NULL"); + return 0; + } + + if (barejid == NULL) { + log_warning("%s", "api_chat_show_themed failed, barejid is NULL"); + return 0; + } + + char show_ch = '-'; + if (ch) { + if (strlen(ch) != 1) { + log_warning("%s", "api_chat_show_themed failed, ch must be a string of length 1"); + return 0; + } else { + show_ch = ch[0]; + } + } + + ProfChatWin *chatwin = wins_get_chat(barejid); + if (chatwin == NULL) { + log_warning("%s", "api_chat_show_themed failed, could not find chat window for %s", barejid); + return 0; + } + + char *parsed = str_replace(message, "\r\n", "\n"); + theme_item_t themeitem = plugin_themes_get(group, key, def); + + win_println((ProfWin*)chatwin, themeitem, show_ch, "%s", parsed); + free(parsed); + + return 1; +} + +int +api_room_show(const char *const roomjid, const char *message) +{ + if (message == NULL) { + log_warning("%s", "api_room_show failed, message is NULL"); + return 0; + } + + if (roomjid == NULL) { + log_warning("%s", "api_room_show failed, roomjid is NULL"); + return 0; + } + + ProfMucWin *mucwin = wins_get_muc(roomjid); + if (mucwin == NULL) { + log_warning("%s", "api_room_show failed, could not find room window for %s", roomjid); + return 0; + } + + char *parsed = str_replace(message, "\r\n", "\n"); + win_println((ProfWin*)mucwin, THEME_TEXT, '-', "%s", parsed); + free(parsed); + + return 1; +} + +int +api_room_show_themed(const char *const roomjid, const char *const group, const char *const key, const char *const def, + const char *const ch, const char *const message) +{ + if (message == NULL) { + log_warning("%s", "api_room_show_themed failed, message is NULL"); + return 0; + } + + if (roomjid == NULL) { + log_warning("%s", "api_room_show_themed failed, roomjid is NULL"); + return 0; + } + + char show_ch = '-'; + if (ch) { + if (strlen(ch) != 1) { + log_warning("%s", "api_room_show_themed failed, ch must be a string of length 1"); + return 0; + } else { + show_ch = ch[0]; + } + } + + ProfMucWin *mucwin = wins_get_muc(roomjid); + if (mucwin == NULL) { + log_warning("%s", "api_room_show_themed failed, could not find room window for %s", roomjid); + return 0; + } + + char *parsed = str_replace(message, "\r\n", "\n"); + theme_item_t themeitem = plugin_themes_get(group, key, def); + + win_println((ProfWin*)mucwin, themeitem, show_ch, "%s", parsed); + free(parsed); + + return 1; +} diff --git a/src/plugins/api.h b/src/plugins/api.h index 75c159a3..9e5c7c53 100644 --- a/src/plugins/api.h +++ b/src/plugins/api.h @@ -109,4 +109,12 @@ int api_room_unset_titlebar_enctext(const char *const roomjid); int api_room_set_message_char(const char *const roomjid, const char *const ch); int api_room_unset_message_char(const char *const roomjid); +int api_chat_show(const char *const barejid, const char *const message); +int api_chat_show_themed(const char *const barejid, const char *const group, const char *const key, const char *const def, + const char *const ch, const char *const message); + +int api_room_show(const char *const roomjid, const char *message); +int api_room_show_themed(const char *const roomjid, const char *const group, const char *const key, const char *const def, + const char *const ch, const char *const message); + #endif diff --git a/src/plugins/c_api.c b/src/plugins/c_api.c index 5cd23732..dd63d004 100644 --- a/src/plugins/c_api.c +++ b/src/plugins/c_api.c @@ -407,6 +407,32 @@ c_api_room_unset_message_char(const char *roomjid) return api_room_unset_message_char(roomjid); } +static int +c_api_chat_show(const char *const barejid, const char *const message) +{ + return api_chat_show(barejid, message); +} + +static int +c_api_chat_show_themed(const char *const barejid, const char *const group, const char *const item, const char *const def, + const char *const ch, const char *const message) +{ + return api_chat_show_themed(barejid, group, item, def, ch, message); +} + +static int +c_api_room_show(const char *const roomjid, const char *const message) +{ + return api_room_show(roomjid, message); +} + +static int +c_api_room_show_themed(const char *const roomjid, const char *const group, const char *const item, const char *const def, + const char *const ch, const char *const message) +{ + return api_room_show_themed(roomjid, group, item, def, ch, message); +} + void c_command_callback(PluginCommand *command, gchar **args) { @@ -485,6 +511,10 @@ c_api_init(void) prof_room_unset_titlebar_enctext = c_api_room_unset_titlebar_enctext; prof_room_set_message_char = c_api_room_set_message_char; prof_room_unset_message_char = c_api_room_unset_message_char; + prof_chat_show = c_api_chat_show; + prof_chat_show_themed = c_api_chat_show_themed; + prof_room_show = c_api_room_show; + prof_room_show_themed = c_api_room_show_themed; } static char * diff --git a/src/plugins/c_plugins.c b/src/plugins/c_plugins.c index e6ec2b7d..675457e5 100644 --- a/src/plugins/c_plugins.c +++ b/src/plugins/c_plugins.c @@ -81,6 +81,7 @@ c_plugin_create(const char *const filename) plugin->lang = LANG_C; plugin->module = handle; plugin->init_func = c_init_hook; + plugin->contains_hook = c_contains_hook; plugin->on_start_func = c_on_start_hook; plugin->on_shutdown_func = c_on_shutdown_hook; plugin->on_unload_func = c_on_unload_hook; @@ -136,6 +137,16 @@ c_init_hook(ProfPlugin *plugin, const char *const version, const char *const sta func(version, status, account_name, fulljid); } +gboolean +c_contains_hook(ProfPlugin *plugin, const char *const hook) +{ + if (dlsym(plugin->module, hook)) { + return TRUE; + } else { + return FALSE; + } +} + void c_on_start_hook(ProfPlugin *plugin) { diff --git a/src/plugins/c_plugins.h b/src/plugins/c_plugins.h index c802d5a5..42c65f38 100644 --- a/src/plugins/c_plugins.h +++ b/src/plugins/c_plugins.h @@ -45,6 +45,9 @@ void c_shutdown(void); void c_init_hook(ProfPlugin *plugin, const char *const version, const char *const status, const char *const account_name, const char *const fulljid); + +gboolean c_contains_hook(ProfPlugin *plugin, const char *const hook); + void c_on_start_hook(ProfPlugin *plugin); void c_on_shutdown_hook(ProfPlugin *plugin); void c_on_unload_hook(ProfPlugin *plugin); diff --git a/src/plugins/plugins.c b/src/plugins/plugins.c index 054cb8e1..f190a8d0 100644 --- a/src/plugins/plugins.c +++ b/src/plugins/plugins.c @@ -413,11 +413,18 @@ plugins_pre_chat_message_send(const char * const barejid, const char *message) GList *curr = values; while (curr) { ProfPlugin *plugin = curr->data; - new_message = plugin->pre_chat_message_send(plugin, barejid, curr_message); - if (new_message) { - free(curr_message); - curr_message = strdup(new_message); - free(new_message); + if (plugin->contains_hook(plugin, "prof_pre_chat_message_send")) { + new_message = plugin->pre_chat_message_send(plugin, barejid, curr_message); + if (new_message) { + free(curr_message); + curr_message = strdup(new_message); + free(new_message); + } else { + free(curr_message); + g_list_free(values); + + return NULL; + } } curr = g_list_next(curr); } @@ -485,11 +492,18 @@ plugins_pre_room_message_send(const char * const barejid, const char *message) GList *curr = values; while (curr) { ProfPlugin *plugin = curr->data; - new_message = plugin->pre_room_message_send(plugin, barejid, curr_message); - if (new_message) { - free(curr_message); - curr_message = strdup(new_message); - free(new_message); + if (plugin->contains_hook(plugin, "prof_pre_room_message_send")) { + new_message = plugin->pre_room_message_send(plugin, barejid, curr_message); + if (new_message) { + free(curr_message); + curr_message = strdup(new_message); + free(new_message); + } else { + free(curr_message); + g_list_free(values); + + return NULL; + } } curr = g_list_next(curr); } @@ -587,11 +601,19 @@ plugins_pre_priv_message_send(const char * const fulljid, const char * const mes GList *curr = values; while (curr) { ProfPlugin *plugin = curr->data; - new_message = plugin->pre_priv_message_send(plugin, jidp->barejid, jidp->resourcepart, curr_message); - if (new_message) { - free(curr_message); - curr_message = strdup(new_message); - free(new_message); + if (plugin->contains_hook(plugin, "prof_pre_priv_message_send")) { + new_message = plugin->pre_priv_message_send(plugin, jidp->barejid, jidp->resourcepart, curr_message); + if (new_message) { + free(curr_message); + curr_message = strdup(new_message); + free(new_message); + } else { + free(curr_message); + g_list_free(values); + jid_destroy(jidp); + + return NULL; + } } curr = g_list_next(curr); } diff --git a/src/plugins/plugins.h b/src/plugins/plugins.h index b7fdf297..ce97237d 100644 --- a/src/plugins/plugins.h +++ b/src/plugins/plugins.h @@ -49,6 +49,8 @@ typedef struct prof_plugin_t { void (*init_func)(struct prof_plugin_t* plugin, const char * const version, const char * const status, const char *const account_name, const char *const fulljid); + gboolean (*contains_hook)(struct prof_plugin_t* plugin, const char *const hook); + void (*on_start_func)(struct prof_plugin_t* plugin); void (*on_shutdown_func)(struct prof_plugin_t* plugin); void (*on_unload_func)(struct prof_plugin_t* plugin); diff --git a/src/plugins/profapi.c b/src/plugins/profapi.c index e9d3cff6..6d580162 100644 --- a/src/plugins/profapi.c +++ b/src/plugins/profapi.c @@ -105,3 +105,11 @@ int (*prof_room_set_titlebar_enctext)(const char *roomjid, const char *enctext) int (*prof_room_unset_titlebar_enctext)(const char *roomjid) = NULL; int (*prof_room_set_message_char)(const char *roomjid, const char *ch) = NULL; int (*prof_room_unset_message_char)(const char *roomjid) = NULL; + +int (*prof_chat_show)(const char *const barejid, const char *const message) = NULL; +int (*prof_chat_show_themed)(const char *const barejid, const char *const group, const char *const item, const char *const def, + const char *const ch, const char *const message) = NULL; + +int (*prof_room_show)(const char *const roomjid, const char *const message) = NULL; +int (*prof_room_show_themed)(const char *const roomjid, const char *const group, const char *const item, const char *const def, + const char *const ch, const char *const message) = NULL; diff --git a/src/plugins/profapi.h b/src/plugins/profapi.h index d17152fc..a684821e 100644 --- a/src/plugins/profapi.h +++ b/src/plugins/profapi.h @@ -118,4 +118,12 @@ int (*prof_room_unset_titlebar_enctext)(const char *roomjid); int (*prof_room_set_message_char)(const char *roomjid, const char *ch); int (*prof_room_unset_message_char)(const char *roomjid); +int (*prof_chat_show)(const char *const barejid, const char *const message); +int (*prof_chat_show_themed)(const char *const barejid, const char *const group, const char *const item, const char *const def, + const char *const ch, const char *const message); + +int (*prof_room_show)(const char *const roomjid, const char *const message); +int (*prof_room_show_themed)(const char *const roomjid, const char *const group, const char *const item, const char *const def, + const char *const ch, const char *const message); + #endif diff --git a/src/plugins/python_api.c b/src/plugins/python_api.c index 4e5bdfb5..4ee13039 100644 --- a/src/plugins/python_api.c +++ b/src/plugins/python_api.c @@ -1284,6 +1284,130 @@ python_api_room_unset_message_char(PyObject *self, PyObject *args) } } +static PyObject* +python_api_chat_show(PyObject *self, PyObject *args) +{ + PyObject *barejid = NULL; + PyObject *message = NULL; + if (!PyArg_ParseTuple(args, "OO", &barejid, &message)) { + Py_RETURN_NONE; + } + + char *barejid_str = python_str_or_unicode_to_string(barejid); + char *message_str = python_str_or_unicode_to_string(message); + + allow_python_threads(); + int res = api_chat_show(barejid_str, message_str); + free(barejid_str); + free(message_str); + disable_python_threads(); + + if (res) { + return Py_BuildValue("O", Py_True); + } else { + return Py_BuildValue("O", Py_False); + } +} + +static PyObject* +python_api_chat_show_themed(PyObject *self, PyObject *args) +{ + PyObject *barejid = NULL; + PyObject *group = NULL; + PyObject *key = NULL; + PyObject *def = NULL; + PyObject *ch = NULL; + PyObject *message = NULL; + if (!PyArg_ParseTuple(args, "OOOOOO", &barejid, &group, &key, &def, &ch, &message)) { + Py_RETURN_NONE; + } + + char *barejid_str = python_str_or_unicode_to_string(barejid); + char *group_str = python_str_or_unicode_to_string(group); + char *key_str = python_str_or_unicode_to_string(key); + char *def_str = python_str_or_unicode_to_string(def); + char *ch_str = python_str_or_unicode_to_string(ch); + char *message_str = python_str_or_unicode_to_string(message); + + allow_python_threads(); + int res = api_chat_show_themed(barejid_str, group_str, key_str, def_str, ch_str, message_str); + free(barejid_str); + free(group_str); + free(key_str); + free(def_str); + free(ch_str); + free(message_str); + disable_python_threads(); + + if (res) { + return Py_BuildValue("O", Py_True); + } else { + return Py_BuildValue("O", Py_False); + } +} + +static PyObject* +python_api_room_show(PyObject *self, PyObject *args) +{ + PyObject *roomjid = NULL; + PyObject *message = NULL; + if (!PyArg_ParseTuple(args, "OO", &roomjid, &message)) { + Py_RETURN_NONE; + } + + char *roomjid_str = python_str_or_unicode_to_string(roomjid); + char *message_str = python_str_or_unicode_to_string(message); + + allow_python_threads(); + int res = api_room_show(roomjid_str, message_str); + free(roomjid_str); + free(message_str); + disable_python_threads(); + + if (res) { + return Py_BuildValue("O", Py_True); + } else { + return Py_BuildValue("O", Py_False); + } +} + +static PyObject* +python_api_room_show_themed(PyObject *self, PyObject *args) +{ + PyObject *roomjid = NULL; + PyObject *group = NULL; + PyObject *key = NULL; + PyObject *def = NULL; + PyObject *ch = NULL; + PyObject *message = NULL; + if (!PyArg_ParseTuple(args, "OOOOOO", &roomjid, &group, &key, &def, &ch, &message)) { + Py_RETURN_NONE; + } + + char *roomjid_str = python_str_or_unicode_to_string(roomjid); + char *group_str = python_str_or_unicode_to_string(group); + char *key_str = python_str_or_unicode_to_string(key); + char *def_str = python_str_or_unicode_to_string(def); + char *ch_str = python_str_or_unicode_to_string(ch); + char *message_str = python_str_or_unicode_to_string(message); + + allow_python_threads(); + int res = api_room_show_themed(roomjid_str, group_str, key_str, def_str, ch_str, message_str); + free(roomjid_str); + free(group_str); + free(key_str); + free(def_str); + free(ch_str); + free(message_str); + disable_python_threads(); + + if (res) { + return Py_BuildValue("O", Py_True); + } else { + return Py_BuildValue("O", Py_False); + } +} + void python_command_callback(PluginCommand *command, gchar **args) { @@ -1403,6 +1527,10 @@ static PyMethodDef apiMethods[] = { { "room_unset_titlebar_enctext", python_api_room_unset_titlebar_enctext, METH_VARARGS, "Reset the encryption status in the title bar for the specified room" }, { "room_set_message_char", python_api_room_set_message_char, METH_VARARGS, "Set the message prefix character for specified room" }, { "room_unset_message_char", python_api_room_unset_message_char, METH_VARARGS, "Reset the message prefix character for specified room" }, + { "chat_show", python_api_chat_show, METH_VARARGS, "Print a line in a chat window" }, + { "chat_show_themed", python_api_chat_show_themed, METH_VARARGS, "Print a themed line in a chat window" }, + { "room_show", python_api_room_show, METH_VARARGS, "Print a line in a chat room window" }, + { "room_show_themed", python_api_room_show_themed, METH_VARARGS, "Print a themed line in a chat room window" }, { NULL, NULL, 0, NULL } }; diff --git a/src/plugins/python_plugins.c b/src/plugins/python_plugins.c index 66efec54..897f7b54 100644 --- a/src/plugins/python_plugins.c +++ b/src/plugins/python_plugins.c @@ -125,6 +125,7 @@ python_plugin_create(const char *const filename) plugin->lang = LANG_PYTHON; plugin->module = p_module; plugin->init_func = python_init_hook; + plugin->contains_hook = python_contains_hook; plugin->on_start_func = python_on_start_hook; plugin->on_shutdown_func = python_on_shutdown_hook; plugin->on_unload_func = python_on_unload_hook; @@ -184,6 +185,22 @@ python_init_hook(ProfPlugin *plugin, const char *const version, const char *cons allow_python_threads(); } +gboolean +python_contains_hook(ProfPlugin *plugin, const char *const hook) +{ + disable_python_threads(); + gboolean res = FALSE; + + PyObject *p_module = plugin->module; + if (PyObject_HasAttrString(p_module, hook)) { + res = TRUE; + } + + allow_python_threads(); + + return res; +} + void python_on_start_hook(ProfPlugin *plugin) { diff --git a/src/plugins/python_plugins.h b/src/plugins/python_plugins.h index 0969b700..24b04b8c 100644 --- a/src/plugins/python_plugins.h +++ b/src/plugins/python_plugins.h @@ -47,6 +47,9 @@ const char* python_get_version(void); void python_init_hook(ProfPlugin *plugin, const char *const version, const char *const status, const char *const account_name, const char *const fulljid); + +gboolean python_contains_hook(ProfPlugin *plugin, const char *const hook); + void python_on_start_hook(ProfPlugin *plugin); void python_on_shutdown_hook(ProfPlugin *plugin); void python_on_unload_hook(ProfPlugin *plugin);