dfd1ea1782
git-svn-id: https://geany.svn.sourceforge.net/svnroot/geany/trunk@4396 ea778897-0a13-0410-b9d1-a72fbfd435f5
678 lines
21 KiB
C
678 lines
21 KiB
C
/*
|
|
* templates.c - this file is part of Geany, a fast and lightweight IDE
|
|
*
|
|
* Copyright 2005-2009 Enrico Tröger <enrico(dot)troeger(at)uvena(dot)de>
|
|
* Copyright 2006-2009 Nick Treleaven <nick(dot)treleaven(at)btinternet(dot)com>
|
|
*
|
|
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*
|
|
* $Id$
|
|
*/
|
|
|
|
/*
|
|
* Templates to insert into the current document, or filetype templates to create a new
|
|
* document from.
|
|
*/
|
|
|
|
#include <time.h>
|
|
#include <string.h>
|
|
|
|
#include "geany.h"
|
|
|
|
#include "templates.h"
|
|
#include "support.h"
|
|
#include "utils.h"
|
|
#include "document.h"
|
|
#include "filetypes.h"
|
|
#include "ui_utils.h"
|
|
#include "toolbar.h"
|
|
#include "geanymenubuttonaction.h"
|
|
|
|
|
|
GeanyTemplatePrefs template_prefs;
|
|
|
|
static GtkWidget *new_with_template_menu = NULL; /* submenu used for both file menu and toolbar */
|
|
|
|
|
|
/* TODO: implement custom insertion templates, put these into files in data/templates */
|
|
|
|
/* default templates, only for initial tempate file creation on first start of Geany */
|
|
static const gchar templates_gpl_notice[] = "\
|
|
This program is free software; you can redistribute it and/or modify\n\
|
|
it under the terms of the GNU General Public License as published by\n\
|
|
the Free Software Foundation; either version 2 of the License, or\n\
|
|
(at your option) any later version.\n\
|
|
\n\
|
|
This program is distributed in the hope that it will be useful,\n\
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of\n\
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n\
|
|
GNU General Public License for more details.\n\
|
|
\n\
|
|
You should have received a copy of the GNU General Public License\n\
|
|
along with this program; if not, write to the Free Software\n\
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,\n\
|
|
MA 02110-1301, USA.\n\
|
|
";
|
|
|
|
static const gchar templates_bsd_notice[] = "\
|
|
Redistribution and use in source and binary forms, with or without\n\
|
|
modification, are permitted provided that the following conditions are\n\
|
|
met:\n\
|
|
\n\
|
|
* Redistributions of source code must retain the above copyright\n\
|
|
notice, this list of conditions and the following disclaimer.\n\
|
|
* Redistributions in binary form must reproduce the above\n\
|
|
copyright notice, this list of conditions and the following disclaimer\n\
|
|
in the documentation and/or other materials provided with the\n\
|
|
distribution.\n\
|
|
* Neither the name of the {company} nor the names of its\n\
|
|
contributors may be used to endorse or promote products derived from\n\
|
|
this software without specific prior written permission.\n\
|
|
\n\
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS\n\
|
|
\"AS IS\" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT\n\
|
|
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR\n\
|
|
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT\n\
|
|
OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,\n\
|
|
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT\n\
|
|
LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,\n\
|
|
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY\n\
|
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT\n\
|
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE\n\
|
|
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.\n\
|
|
";
|
|
|
|
static const gchar templates_function_description[] = "\
|
|
\n\
|
|
name: {functionname}\n\
|
|
@param\n\
|
|
@return\n\
|
|
";
|
|
|
|
static const gchar templates_multiline[] = "\
|
|
\n\
|
|
\n\
|
|
";
|
|
|
|
static const gchar templates_fileheader[] = "\
|
|
{filename}\n\
|
|
\n\
|
|
Copyright {year} {developer} <{mail}>\n\
|
|
\n\
|
|
{gpl}\
|
|
";
|
|
|
|
static const gchar templates_changelog[] = "\
|
|
{date} {developer} <{mail}>\n\
|
|
\n\
|
|
* \n\n\n";
|
|
|
|
static gchar *templates[GEANY_MAX_TEMPLATES];
|
|
|
|
/* We should probably remove filetype templates support soon - users can use custom
|
|
* file templates instead. */
|
|
static gchar *ft_templates[GEANY_MAX_BUILT_IN_FILETYPES] = {NULL};
|
|
|
|
|
|
/* some simple macros to reduce code size and make the code readable */
|
|
#define TEMPLATES_GET_FILENAME(shortname) \
|
|
g_strconcat(app->configdir, \
|
|
G_DIR_SEPARATOR_S GEANY_TEMPLATES_SUBDIR G_DIR_SEPARATOR_S, shortname, NULL)
|
|
|
|
#define TEMPLATES_READ_FILE(fname, contents_ptr) \
|
|
g_file_get_contents(fname, contents_ptr, NULL, NULL);
|
|
|
|
|
|
static void create_template_file_if_necessary(const gchar *filename, const gchar *content)
|
|
{
|
|
if (! g_file_test(filename, G_FILE_TEST_EXISTS))
|
|
{
|
|
if (file_prefs.default_eol_character != SC_EOL_LF)
|
|
{
|
|
/* Replace the \n characters in the default template text by the proper
|
|
* platform-specific line ending characters. */
|
|
GString *tmp = g_string_new(content);
|
|
const gchar *eol_str = (file_prefs.default_eol_character == SC_EOL_CR) ? "\r" : "\r\n";
|
|
|
|
utils_string_replace_all(tmp, "\n", eol_str);
|
|
utils_write_file(filename, tmp->str);
|
|
g_string_free(tmp, TRUE);
|
|
}
|
|
else
|
|
utils_write_file(filename, content);
|
|
}
|
|
}
|
|
|
|
|
|
/* FIXME the callers should use GStrings instead of char arrays */
|
|
static gchar *replace_all(gchar *text, const gchar *year, const gchar *date, const gchar *datetime)
|
|
{
|
|
GString *str;
|
|
|
|
if (text == NULL)
|
|
return NULL;
|
|
|
|
str = g_string_new(text);
|
|
|
|
g_free(text);
|
|
templates_replace_all(str, year, date, datetime);
|
|
|
|
return g_string_free(str, FALSE);
|
|
}
|
|
|
|
|
|
static void init_general_templates(const gchar *year, const gchar *date, const gchar *datetime)
|
|
{
|
|
gchar *template_filename_fileheader = TEMPLATES_GET_FILENAME("fileheader");
|
|
gchar *template_filename_gpl = TEMPLATES_GET_FILENAME("gpl");
|
|
gchar *template_filename_bsd = TEMPLATES_GET_FILENAME("bsd");
|
|
gchar *template_filename_function = TEMPLATES_GET_FILENAME("function");
|
|
gchar *template_filename_changelog = TEMPLATES_GET_FILENAME("changelog");
|
|
|
|
/* create the template files in the configuration directory, if they don't exist */
|
|
create_template_file_if_necessary(template_filename_fileheader, templates_fileheader);
|
|
create_template_file_if_necessary(template_filename_gpl, templates_gpl_notice);
|
|
create_template_file_if_necessary(template_filename_bsd, templates_bsd_notice);
|
|
create_template_file_if_necessary(template_filename_function, templates_function_description);
|
|
create_template_file_if_necessary(template_filename_changelog, templates_changelog);
|
|
|
|
/* read the contents */
|
|
TEMPLATES_READ_FILE(template_filename_fileheader, &templates[GEANY_TEMPLATE_FILEHEADER]);
|
|
templates[GEANY_TEMPLATE_FILEHEADER] = replace_all(templates[GEANY_TEMPLATE_FILEHEADER], year, date, datetime);
|
|
|
|
TEMPLATES_READ_FILE(template_filename_gpl, &templates[GEANY_TEMPLATE_GPL]);
|
|
templates[GEANY_TEMPLATE_GPL] = replace_all(templates[GEANY_TEMPLATE_GPL], year, date, datetime);
|
|
|
|
TEMPLATES_READ_FILE(template_filename_bsd, &templates[GEANY_TEMPLATE_BSD]);
|
|
templates[GEANY_TEMPLATE_BSD] = replace_all(templates[GEANY_TEMPLATE_BSD], year, date, datetime);
|
|
|
|
TEMPLATES_READ_FILE(template_filename_function, &templates[GEANY_TEMPLATE_FUNCTION]);
|
|
templates[GEANY_TEMPLATE_FUNCTION] = replace_all(templates[GEANY_TEMPLATE_FUNCTION], year, date, datetime);
|
|
|
|
TEMPLATES_READ_FILE(template_filename_changelog, &templates[GEANY_TEMPLATE_CHANGELOG]);
|
|
templates[GEANY_TEMPLATE_CHANGELOG] = replace_all(templates[GEANY_TEMPLATE_CHANGELOG], year, date, datetime);
|
|
|
|
/* free the whole stuff */
|
|
g_free(template_filename_fileheader);
|
|
g_free(template_filename_gpl);
|
|
g_free(template_filename_bsd);
|
|
g_free(template_filename_function);
|
|
g_free(template_filename_changelog);
|
|
}
|
|
|
|
|
|
static void init_ft_templates(const gchar *year, const gchar *date, const gchar *datetime)
|
|
{
|
|
filetype_id ft_id;
|
|
|
|
for (ft_id = 0; ft_id < GEANY_MAX_BUILT_IN_FILETYPES; ft_id++)
|
|
{
|
|
gchar *ext = filetypes_get_conf_extension(ft_id);
|
|
gchar *shortname = g_strconcat("filetype.", ext, NULL);
|
|
gchar *fname = TEMPLATES_GET_FILENAME(shortname);
|
|
|
|
TEMPLATES_READ_FILE(fname, &ft_templates[ft_id]);
|
|
ft_templates[ft_id] = replace_all(ft_templates[ft_id], year, date, datetime);
|
|
|
|
g_free(fname);
|
|
g_free(shortname);
|
|
g_free(ext);
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
on_new_with_filetype_template(GtkMenuItem *menuitem, gpointer user_data)
|
|
{
|
|
GeanyFiletype *ft = user_data;
|
|
gchar *template = templates_get_template_new_file(ft);
|
|
|
|
document_new_file(NULL, ft, template);
|
|
g_free(template);
|
|
}
|
|
|
|
|
|
/* TODO: remove filetype template support after 0.19 */
|
|
static gboolean create_new_filetype_items(void)
|
|
{
|
|
GSList *node;
|
|
gboolean ret = FALSE;
|
|
GtkWidget *menu = NULL;
|
|
|
|
foreach_slist(node, filetypes_by_title)
|
|
{
|
|
GeanyFiletype *ft = node->data;
|
|
GtkWidget *item;
|
|
|
|
if (ft->id >= GEANY_MAX_BUILT_IN_FILETYPES || ft_templates[ft->id] == NULL)
|
|
continue;
|
|
|
|
if (!menu)
|
|
{
|
|
item = gtk_menu_item_new_with_label(_("Old"));
|
|
menu = gtk_menu_new();
|
|
gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), menu);
|
|
gtk_widget_show_all(item);
|
|
gtk_container_add(GTK_CONTAINER(new_with_template_menu), item);
|
|
}
|
|
item = gtk_menu_item_new_with_label(ft->title);
|
|
gtk_widget_show(item);
|
|
gtk_container_add(GTK_CONTAINER(menu), item);
|
|
g_signal_connect(item, "activate", G_CALLBACK(on_new_with_filetype_template), ft);
|
|
ret = TRUE;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
|
|
static gchar *get_template_from_file(const gchar *locale_fname, const gchar *doc_filename,
|
|
GeanyFiletype *ft)
|
|
{
|
|
gchar *content;
|
|
GString *template = NULL;
|
|
|
|
g_file_get_contents(locale_fname, &content, NULL, NULL);
|
|
|
|
if (content != NULL)
|
|
{
|
|
gchar *file_header;
|
|
gchar *year = utils_get_date_time(template_prefs.year_format, NULL);
|
|
gchar *date = utils_get_date_time(template_prefs.date_format, NULL);
|
|
gchar *datetime = utils_get_date_time(template_prefs.datetime_format, NULL);
|
|
|
|
template = g_string_new(content);
|
|
|
|
file_header = templates_get_template_fileheader(FILETYPE_ID(ft), doc_filename);
|
|
templates_replace_all(template, year, date, datetime);
|
|
utils_string_replace_all(template, "{filename}", doc_filename);
|
|
utils_string_replace_all(template, "{fileheader}", file_header);
|
|
|
|
utils_free_pointers(5, year, date, datetime, file_header, content, NULL);
|
|
return g_string_free(template, FALSE);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
static void
|
|
on_new_with_file_template(GtkMenuItem *menuitem, G_GNUC_UNUSED gpointer user_data)
|
|
{
|
|
gchar *fname = ui_menu_item_get_text(menuitem);
|
|
GeanyFiletype *ft;
|
|
gchar *template;
|
|
const gchar *extension = strrchr(fname, '.'); /* easy way to get the file extension */
|
|
gchar *new_filename = g_strconcat(GEANY_STRING_UNTITLED, extension, NULL);
|
|
gchar *path;
|
|
|
|
ft = filetypes_detect_from_extension(fname);
|
|
setptr(fname, utils_get_locale_from_utf8(fname));
|
|
|
|
/* fname is just the basename from the menu item, so prepend the custom files path */
|
|
path = g_build_path(G_DIR_SEPARATOR_S, app->configdir, GEANY_TEMPLATES_SUBDIR,
|
|
"files", fname, NULL);
|
|
template = get_template_from_file(path, new_filename, ft);
|
|
if (!template)
|
|
{
|
|
/* try the system path */
|
|
g_free(path);
|
|
path = g_build_path(G_DIR_SEPARATOR_S, app->datadir, GEANY_TEMPLATES_SUBDIR,
|
|
"files", fname, NULL);
|
|
template = get_template_from_file(path, new_filename, ft);
|
|
}
|
|
if (template)
|
|
document_new_file(new_filename, ft, template);
|
|
else
|
|
{
|
|
setptr(fname, utils_get_utf8_from_locale(fname));
|
|
ui_set_statusbar(TRUE, _("Could not find file '%s'."), fname);
|
|
}
|
|
g_free(template);
|
|
g_free(path);
|
|
g_free(new_filename);
|
|
g_free(fname);
|
|
}
|
|
|
|
|
|
static void add_file_item(const gchar *fname, GtkWidget *menu)
|
|
{
|
|
GtkWidget *tmp_button;
|
|
gchar *label;
|
|
|
|
g_return_if_fail(fname);
|
|
g_return_if_fail(menu);
|
|
|
|
label = utils_get_utf8_from_locale(fname);
|
|
|
|
tmp_button = gtk_menu_item_new_with_label(label);
|
|
gtk_widget_show(tmp_button);
|
|
gtk_container_add(GTK_CONTAINER(menu), tmp_button);
|
|
g_signal_connect(tmp_button, "activate", G_CALLBACK(on_new_with_file_template), NULL);
|
|
|
|
g_free(label);
|
|
}
|
|
|
|
|
|
static gboolean add_custom_template_items(void)
|
|
{
|
|
GSList *list = utils_get_config_files(GEANY_TEMPLATES_SUBDIR G_DIR_SEPARATOR_S "files");
|
|
GSList *node;
|
|
|
|
foreach_slist(node, list)
|
|
{
|
|
gchar *fname = node->data;
|
|
|
|
add_file_item(fname, new_with_template_menu);
|
|
g_free(fname);
|
|
}
|
|
g_slist_free(list);
|
|
return list != NULL;
|
|
}
|
|
|
|
|
|
static void create_file_template_menu(void)
|
|
{
|
|
GtkWidget *sep = NULL;
|
|
|
|
new_with_template_menu = gtk_menu_new();
|
|
|
|
if (add_custom_template_items())
|
|
{
|
|
sep = gtk_separator_menu_item_new();
|
|
gtk_container_add(GTK_CONTAINER(new_with_template_menu), sep);
|
|
}
|
|
if (create_new_filetype_items() && sep)
|
|
{
|
|
gtk_widget_show(sep);
|
|
}
|
|
/* unless the file menu is showing, menu should be in the toolbar widget */
|
|
geany_menu_button_action_set_menu(GEANY_MENU_BUTTON_ACTION(
|
|
toolbar_get_action_by_name("New")), new_with_template_menu);
|
|
}
|
|
|
|
|
|
static void on_file_menu_show(GtkWidget *item)
|
|
{
|
|
geany_menu_button_action_set_menu(
|
|
GEANY_MENU_BUTTON_ACTION(toolbar_get_action_by_name("New")), NULL);
|
|
item = ui_lookup_widget(main_widgets.window, "menu_new_with_template1");
|
|
gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), new_with_template_menu);
|
|
}
|
|
|
|
|
|
static void on_file_menu_hide(GtkWidget *item)
|
|
{
|
|
item = ui_lookup_widget(main_widgets.window, "menu_new_with_template1");
|
|
gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), NULL);
|
|
geany_menu_button_action_set_menu(
|
|
GEANY_MENU_BUTTON_ACTION(toolbar_get_action_by_name("New")), new_with_template_menu);
|
|
}
|
|
|
|
|
|
void templates_init(void)
|
|
{
|
|
gchar *year = utils_get_date_time(template_prefs.year_format, NULL);
|
|
gchar *date = utils_get_date_time(template_prefs.date_format, NULL);
|
|
gchar *datetime = utils_get_date_time(template_prefs.datetime_format, NULL);
|
|
GtkWidget *item;
|
|
|
|
init_general_templates(year, date, datetime);
|
|
init_ft_templates(year, date, datetime);
|
|
|
|
g_free(date);
|
|
g_free(datetime);
|
|
g_free(year);
|
|
|
|
create_file_template_menu();
|
|
/* we hold our own ref for the menu as it has no parent whilst being moved */
|
|
g_object_ref(new_with_template_menu);
|
|
|
|
/* reparent the template menu as needed */
|
|
item = ui_lookup_widget(main_widgets.window, "file1");
|
|
item = gtk_menu_item_get_submenu(GTK_MENU_ITEM(item));
|
|
g_signal_connect(item, "show", G_CALLBACK(on_file_menu_show), NULL);
|
|
g_signal_connect(item, "hide", G_CALLBACK(on_file_menu_hide), NULL);
|
|
}
|
|
|
|
|
|
/* indent is used to make some whitespace between comment char and real start of the line
|
|
* e.g. indent = 8 prints " * here comes the text of the line"
|
|
* indent is meant to be the whole amount of characters before the real line content follows, i.e.
|
|
* 6 characters are filled with whitespace when the comment characters include " *" */
|
|
static gchar *make_comment_block(const gchar *comment_text, gint filetype_idx, guint indent)
|
|
{
|
|
gchar *frame_start; /* to add before comment_text */
|
|
gchar *frame_end; /* to add after comment_text */
|
|
gchar *line_prefix; /* to add before every line in comment_text */
|
|
gchar *result;
|
|
gchar *tmp;
|
|
gchar *prefix;
|
|
gchar **lines;
|
|
guint i, len;
|
|
GeanyFiletype *ft = filetypes_index(filetype_idx);
|
|
|
|
g_return_val_if_fail(ft != NULL, NULL);
|
|
|
|
if (NZV(ft->comment_open))
|
|
{
|
|
if (NZV(ft->comment_close))
|
|
{
|
|
frame_start = g_strconcat(ft->comment_open, "\n", NULL);
|
|
frame_end = g_strconcat(ft->comment_close, "\n", NULL);
|
|
line_prefix = "";
|
|
}
|
|
else
|
|
{
|
|
frame_start = NULL;
|
|
frame_end = NULL;
|
|
line_prefix = ft->comment_open;
|
|
}
|
|
}
|
|
else
|
|
{ /* use C-like multi-line comments as fallback */
|
|
frame_start = g_strdup("/*\n");
|
|
frame_end = g_strdup("*/\n");
|
|
line_prefix = "";
|
|
}
|
|
|
|
/* do some magic to nicely format C-like multi-line comments */
|
|
if (NZV(frame_start) && frame_start[1] == '*')
|
|
{
|
|
/* prefix the string with a space */
|
|
setptr(frame_end, g_strconcat(" ", frame_end, NULL));
|
|
line_prefix = " *";
|
|
}
|
|
|
|
/* construct the real prefix with given amount of whitespace */
|
|
i = (indent > strlen(line_prefix)) ? (indent - strlen(line_prefix)) : strlen(line_prefix);
|
|
tmp = g_strnfill(i, ' ');
|
|
prefix = g_strconcat(line_prefix, tmp, NULL);
|
|
g_free(tmp);
|
|
|
|
/* add line_prefix to every line of comment_text */
|
|
lines = g_strsplit(comment_text, "\n", -1);
|
|
len = g_strv_length(lines) - 1;
|
|
for (i = 0; i < len; i++)
|
|
{
|
|
tmp = lines[i];
|
|
lines[i] = g_strconcat(prefix, tmp, NULL);
|
|
g_free(tmp);
|
|
}
|
|
tmp = g_strjoinv("\n", lines);
|
|
|
|
/* add frame_start and frame_end */
|
|
if (frame_start != NULL)
|
|
result = g_strconcat(frame_start, tmp, frame_end, NULL);
|
|
else
|
|
result = g_strconcat(tmp, frame_end, NULL);
|
|
|
|
utils_free_pointers(4, prefix, tmp, frame_start, frame_end, NULL);
|
|
g_strfreev(lines);
|
|
return result;
|
|
}
|
|
|
|
|
|
gchar *templates_get_template_licence(gint filetype_idx, gint licence_type)
|
|
{
|
|
if (licence_type != GEANY_TEMPLATE_GPL && licence_type != GEANY_TEMPLATE_BSD)
|
|
return NULL;
|
|
|
|
return make_comment_block(templates[licence_type], filetype_idx, 8);
|
|
}
|
|
|
|
|
|
gchar *templates_get_template_fileheader(gint filetype_idx, const gchar *fname)
|
|
{
|
|
gchar *template = g_strdup(templates[GEANY_TEMPLATE_FILEHEADER]);
|
|
gchar *shortname;
|
|
gchar *result;
|
|
gchar *date = utils_get_date_time(template_prefs.datetime_format, NULL);
|
|
filetype_id ft_id = filetype_idx;
|
|
GeanyFiletype *ft = filetypes[ft_id];
|
|
|
|
filetypes_load_config(ft_id, FALSE); /* load any user extension setting */
|
|
|
|
if (fname == NULL)
|
|
{
|
|
if (!ft->extension)
|
|
shortname = g_strdup(GEANY_STRING_UNTITLED);
|
|
else
|
|
shortname = g_strconcat(GEANY_STRING_UNTITLED, ".", ft->extension, NULL);
|
|
}
|
|
else
|
|
shortname = g_path_get_basename(fname);
|
|
|
|
utils_str_replace_all(&template, "{filename}", shortname);
|
|
utils_str_replace_all(&template, "{gpl}", templates[GEANY_TEMPLATE_GPL]);
|
|
utils_str_replace_all(&template, "{bsd}", templates[GEANY_TEMPLATE_BSD]);
|
|
utils_str_replace_all(&template, "{datetime}", date);
|
|
|
|
result = make_comment_block(template, ft_id, 8);
|
|
|
|
g_free(template);
|
|
g_free(shortname);
|
|
g_free(date);
|
|
return result;
|
|
}
|
|
|
|
|
|
static gchar *get_file_template(GeanyFiletype *ft)
|
|
{
|
|
filetype_id ft_id = FILETYPE_ID(ft);
|
|
|
|
g_return_val_if_fail(ft_id < GEANY_MAX_BUILT_IN_FILETYPES, NULL);
|
|
|
|
return g_strdup(ft_templates[ft_id]);
|
|
}
|
|
|
|
|
|
gchar *templates_get_template_new_file(GeanyFiletype *ft)
|
|
{
|
|
gchar *ft_template = NULL;
|
|
gchar *file_header = NULL;
|
|
|
|
g_return_val_if_fail(ft->id < GEANY_MAX_BUILT_IN_FILETYPES, NULL);
|
|
|
|
if (FILETYPE_ID(ft) == GEANY_FILETYPES_NONE)
|
|
return get_file_template(ft);
|
|
|
|
file_header = templates_get_template_fileheader(ft->id, NULL); /* file template only used for new files */
|
|
ft_template = get_file_template(ft);
|
|
utils_str_replace_all(&ft_template, "{fileheader}", file_header);
|
|
g_free(file_header);
|
|
return ft_template;
|
|
}
|
|
|
|
|
|
gchar *templates_get_template_generic(gint template)
|
|
{
|
|
return g_strdup(templates[template]);
|
|
}
|
|
|
|
|
|
gchar *templates_get_template_function(gint filetype_idx, const gchar *func_name)
|
|
{
|
|
gchar *template = g_strdup(templates[GEANY_TEMPLATE_FUNCTION]);
|
|
gchar *date = utils_get_date_time(template_prefs.date_format, NULL);
|
|
gchar *datetime = utils_get_date_time(template_prefs.datetime_format, NULL);
|
|
gchar *result;
|
|
|
|
utils_str_replace_all(&template, "{date}", date);
|
|
utils_str_replace_all(&template, "{datetime}", datetime);
|
|
utils_str_replace_all(&template, "{functionname}", (func_name) ? func_name : "");
|
|
|
|
result = make_comment_block(template, filetype_idx, 3);
|
|
|
|
g_free(template);
|
|
g_free(date);
|
|
g_free(datetime);
|
|
return result;
|
|
}
|
|
|
|
|
|
gchar *templates_get_template_changelog(void)
|
|
{
|
|
gchar *date = utils_get_date_time(template_prefs.datetime_format, NULL);
|
|
gchar *result = g_strdup(templates[GEANY_TEMPLATE_CHANGELOG]);
|
|
|
|
utils_str_replace_all(&result, "{date}", date);
|
|
|
|
g_free(date);
|
|
return result;
|
|
}
|
|
|
|
|
|
void templates_free_templates(void)
|
|
{
|
|
gint i;
|
|
GList *children, *item;
|
|
|
|
for (i = 0; i < GEANY_MAX_TEMPLATES; i++)
|
|
{
|
|
g_free(templates[i]);
|
|
}
|
|
for (i = 0; i < GEANY_MAX_BUILT_IN_FILETYPES; i++)
|
|
{
|
|
g_free(ft_templates[i]);
|
|
}
|
|
/* destroy "New with template" sub menu items (in case we want to reload the templates) */
|
|
children = gtk_container_get_children(GTK_CONTAINER(new_with_template_menu));
|
|
foreach_list(item, children)
|
|
{
|
|
gtk_widget_destroy(GTK_WIDGET(item->data));
|
|
}
|
|
g_list_free(children);
|
|
|
|
/* Shouldn't unrefing destroy children anyway? */
|
|
g_object_unref(new_with_template_menu);
|
|
new_with_template_menu = NULL;
|
|
}
|
|
|
|
|
|
void templates_replace_all(GString *text, const gchar *year, const gchar *date,
|
|
const gchar *datetime)
|
|
{
|
|
utils_string_replace_all(text, "{year}", year);
|
|
utils_string_replace_all(text, "{date}", date);
|
|
utils_string_replace_all(text, "{datetime}", datetime);
|
|
utils_string_replace_all(text, "{version}", template_prefs.version);
|
|
utils_string_replace_all(text, "{initial}", template_prefs.initials);
|
|
utils_string_replace_all(text, "{developer}", template_prefs.developer);
|
|
utils_string_replace_all(text, "{mail}", template_prefs.mail);
|
|
utils_string_replace_all(text, "{company}", template_prefs.company);
|
|
utils_string_replace_all(text, "{untitled}", GEANY_STRING_UNTITLED);
|
|
utils_string_replace_all(text, "{geanyversion}", "Geany " VERSION);
|
|
}
|
|
|