4652b74203
git-svn-id: https://geany.svn.sourceforge.net/svnroot/geany/trunk@5896 ea778897-0a13-0410-b9d1-a72fbfd435f5
1725 lines
45 KiB
C
1725 lines
45 KiB
C
/*
|
|
* filetypes.c - this file is part of Geany, a fast and lightweight IDE
|
|
*
|
|
* Copyright 2005-2011 Enrico Tröger <enrico(dot)troeger(at)uvena(dot)de>
|
|
* Copyright 2006-2011 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$
|
|
*/
|
|
|
|
/**
|
|
* @file filetypes.h
|
|
* Filetype detection, file extensions and filetype menu items.
|
|
*/
|
|
|
|
/* Note: we use filetype_id for some function arguments, but GeanyFiletype is better; we should
|
|
* only use GeanyFiletype for API functions. */
|
|
|
|
#include <string.h>
|
|
#include <glib/gstdio.h>
|
|
|
|
#include "geany.h"
|
|
#include "filetypes.h"
|
|
#include "filetypesprivate.h"
|
|
#include "highlighting.h"
|
|
#include "support.h"
|
|
#include "templates.h"
|
|
#include "document.h"
|
|
#include "editor.h"
|
|
#include "msgwindow.h"
|
|
#include "utils.h"
|
|
#include "sciwrappers.h"
|
|
#include "ui_utils.h"
|
|
#include "symbols.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
GPtrArray *filetypes_array = NULL; /* Dynamic array of filetype pointers */
|
|
|
|
static GHashTable *filetypes_hash = NULL; /* Hash of filetype pointers based on name keys */
|
|
|
|
/** List of filetype pointers sorted by name, but with @c filetypes_index(GEANY_FILETYPES_NONE)
|
|
* first, as this is usually treated specially.
|
|
* The list does not change (after filetypes have been initialized), so you can use
|
|
* @code g_slist_nth_data(filetypes_by_title, n) @endcode and expect the same result at different times.
|
|
* @see filetypes_get_sorted_by_name(). */
|
|
GSList *filetypes_by_title = NULL;
|
|
|
|
|
|
static void create_radio_menu_item(GtkWidget *menu, GeanyFiletype *ftype);
|
|
|
|
static gchar *filetypes_get_conf_extension(const GeanyFiletype *ft);
|
|
static void read_filetype_config(void);
|
|
|
|
|
|
enum TitleType
|
|
{
|
|
TITLE_SOURCE_FILE,
|
|
TITLE_FILE
|
|
};
|
|
|
|
/* Save adding many translation strings if the filetype name doesn't need translating */
|
|
static void filetype_make_title(GeanyFiletype *ft, enum TitleType type)
|
|
{
|
|
const gchar *fmt = NULL;
|
|
|
|
switch (type)
|
|
{
|
|
default:
|
|
case TITLE_SOURCE_FILE: fmt = _("%s source file"); break;
|
|
case TITLE_FILE: fmt = _("%s file"); break;
|
|
}
|
|
g_assert(!ft->title);
|
|
g_assert(ft->name);
|
|
ft->title = g_strdup_printf(fmt, ft->name);
|
|
}
|
|
|
|
|
|
/* Note: remember to update HACKING if this function is renamed. */
|
|
static void init_builtin_filetypes(void)
|
|
{
|
|
GeanyFiletype *ft;
|
|
|
|
#define NONE /* these macros are only to ease navigation */
|
|
ft = filetypes[GEANY_FILETYPES_NONE];
|
|
/* ft->name must not be translated as it is used for filetype lookup.
|
|
* Use filetypes_get_display_name() instead. */
|
|
ft->name = g_strdup("None");
|
|
ft->title = g_strdup(_("None"));
|
|
ft->group = GEANY_FILETYPE_GROUP_NONE;
|
|
|
|
#define C
|
|
ft = filetypes[GEANY_FILETYPES_C];
|
|
ft->lang = 0;
|
|
ft->name = g_strdup("C");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-csrc");
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define CPP
|
|
ft = filetypes[GEANY_FILETYPES_CPP];
|
|
ft->lang = 1;
|
|
ft->name = g_strdup("C++");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-c++src");
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define CS
|
|
ft = filetypes[GEANY_FILETYPES_CS];
|
|
ft->lang = 25;
|
|
ft->name = g_strdup("C#");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-csharp");
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define VALA
|
|
ft = filetypes[GEANY_FILETYPES_VALA];
|
|
ft->lang = 33;
|
|
ft->name = g_strdup("Vala");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-vala");
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define D
|
|
ft = filetypes[GEANY_FILETYPES_D];
|
|
ft->lang = 17;
|
|
ft->name = g_strdup("D");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-dsrc");
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define JAVA
|
|
ft = filetypes[GEANY_FILETYPES_JAVA];
|
|
ft->lang = 2;
|
|
ft->name = g_strdup("Java");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-java");
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define PAS /* to avoid warnings when building under Windows, the symbol PASCAL is there defined */
|
|
ft = filetypes[GEANY_FILETYPES_PASCAL];
|
|
ft->lang = 4;
|
|
ft->name = g_strdup("Pascal");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-pascal");
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define ASM
|
|
ft = filetypes[GEANY_FILETYPES_ASM];
|
|
ft->lang = 9;
|
|
ft->name = g_strdup("ASM");
|
|
ft->title = g_strdup_printf(_("%s source file"), "Assembler");
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define BASIC
|
|
ft = filetypes[GEANY_FILETYPES_BASIC];
|
|
ft->lang = 26;
|
|
ft->name = g_strdup("FreeBasic");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define FORTRAN
|
|
ft = filetypes[GEANY_FILETYPES_FORTRAN];
|
|
ft->lang = 18;
|
|
ft->name = g_strdup("Fortran");
|
|
ft->title = g_strdup_printf(_("%s source file"), "Fortran (F90)");
|
|
ft->mime_type = g_strdup("text/x-fortran");
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define F77
|
|
ft = filetypes[GEANY_FILETYPES_F77];
|
|
ft->lang = 30;
|
|
ft->name = g_strdup("F77");
|
|
ft->title = g_strdup_printf(_("%s source file"), "Fortran (F77)");
|
|
ft->mime_type = g_strdup("text/x-fortran");
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define GLSL
|
|
ft = filetypes[GEANY_FILETYPES_GLSL];
|
|
ft->lang = 31;
|
|
ft->name = g_strdup("GLSL");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define CAML
|
|
ft = filetypes[GEANY_FILETYPES_CAML];
|
|
ft->name = g_strdup("CAML");
|
|
ft->title = g_strdup_printf(_("%s source file"), "(O)Caml");
|
|
ft->mime_type = g_strdup("text/x-ocaml");
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define PERL
|
|
ft = filetypes[GEANY_FILETYPES_PERL];
|
|
ft->lang = 5;
|
|
ft->name = g_strdup("Perl");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("application/x-perl");
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
|
|
#define PHP
|
|
ft = filetypes[GEANY_FILETYPES_PHP];
|
|
ft->lang = 6;
|
|
ft->name = g_strdup("PHP");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("application/x-php");
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
|
|
#define JAVASCRIPT
|
|
ft = filetypes[GEANY_FILETYPES_JS];
|
|
ft->lang = 23;
|
|
ft->name = g_strdup("Javascript");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("application/javascript");
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
|
|
#define PYTHON
|
|
ft = filetypes[GEANY_FILETYPES_PYTHON];
|
|
ft->lang = 7;
|
|
ft->name = g_strdup("Python");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-python");
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
|
|
#define RUBY
|
|
ft = filetypes[GEANY_FILETYPES_RUBY];
|
|
ft->lang = 14;
|
|
ft->name = g_strdup("Ruby");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("application/x-ruby");
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
|
|
#define TCL
|
|
ft = filetypes[GEANY_FILETYPES_TCL];
|
|
ft->lang = 15;
|
|
ft->name = g_strdup("Tcl");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-tcl");
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
|
|
#define LUA
|
|
ft = filetypes[GEANY_FILETYPES_LUA];
|
|
ft->lang = 22;
|
|
ft->name = g_strdup("Lua");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-lua");
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
|
|
#define FERITE
|
|
ft = filetypes[GEANY_FILETYPES_FERITE];
|
|
ft->lang = 19;
|
|
ft->name = g_strdup("Ferite");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
|
|
#define HASKELL
|
|
ft = filetypes[GEANY_FILETYPES_HASKELL];
|
|
ft->lang = 24;
|
|
ft->name = g_strdup("Haskell");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-haskell");
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define MARKDOWN
|
|
ft = filetypes[GEANY_FILETYPES_MARKDOWN];
|
|
ft->lang = 36;
|
|
ft->name = g_strdup("Markdown");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->group = GEANY_FILETYPE_GROUP_MARKUP;
|
|
|
|
#define TXT2TAGS
|
|
ft = filetypes[GEANY_FILETYPES_TXT2TAGS];
|
|
ft->lang = 37;
|
|
ft->name = g_strdup("Txt2tags");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-txt2tags");
|
|
ft->group = GEANY_FILETYPE_GROUP_MARKUP;
|
|
|
|
#define ABC
|
|
ft = filetypes[GEANY_FILETYPES_ABC];
|
|
ft->lang = 38;
|
|
ft->name = g_strdup("Abc");
|
|
filetype_make_title(ft, TITLE_FILE);
|
|
ft->group = GEANY_FILETYPE_GROUP_MISC;
|
|
|
|
#define SH
|
|
ft = filetypes[GEANY_FILETYPES_SH];
|
|
ft->lang = 16;
|
|
ft->name = g_strdup("Sh");
|
|
ft->title = g_strdup(_("Shell script"));
|
|
ft->mime_type = g_strdup("application/x-shellscript");
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
|
|
#define MAKE
|
|
ft = filetypes[GEANY_FILETYPES_MAKE];
|
|
ft->lang = 3;
|
|
ft->name = g_strdup("Make");
|
|
ft->title = g_strdup(_("Makefile"));
|
|
ft->mime_type = g_strdup("text/x-makefile");
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
|
|
#define XML
|
|
ft = filetypes[GEANY_FILETYPES_XML];
|
|
ft->name = g_strdup("XML");
|
|
ft->title = g_strdup(_("XML document"));
|
|
ft->mime_type = g_strdup("application/xml");
|
|
ft->group = GEANY_FILETYPE_GROUP_MARKUP;
|
|
|
|
#define DOCBOOK
|
|
ft = filetypes[GEANY_FILETYPES_DOCBOOK];
|
|
ft->lang = 12;
|
|
ft->name = g_strdup("Docbook");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("application/docbook+xml");
|
|
ft->group = GEANY_FILETYPE_GROUP_MARKUP;
|
|
|
|
#define HTML
|
|
ft = filetypes[GEANY_FILETYPES_HTML];
|
|
ft->lang = 29;
|
|
ft->name = g_strdup("HTML");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/html");
|
|
ft->group = GEANY_FILETYPE_GROUP_MARKUP;
|
|
|
|
#define CSS
|
|
ft = filetypes[GEANY_FILETYPES_CSS];
|
|
ft->lang = 13;
|
|
ft->name = g_strdup("CSS");
|
|
ft->title = g_strdup(_("Cascading StyleSheet"));
|
|
ft->mime_type = g_strdup("text/css");
|
|
ft->group = GEANY_FILETYPE_GROUP_MARKUP; /* not really markup but fit quite well to HTML */
|
|
|
|
#define SQL
|
|
ft = filetypes[GEANY_FILETYPES_SQL];
|
|
ft->lang = 11;
|
|
ft->name = g_strdup("SQL");
|
|
filetype_make_title(ft, TITLE_FILE);
|
|
ft->mime_type = g_strdup("text/x-sql");
|
|
ft->group = GEANY_FILETYPE_GROUP_MISC;
|
|
|
|
#define COBOL
|
|
ft = filetypes[GEANY_FILETYPES_COBOL];
|
|
ft->lang = 41;
|
|
ft->name = g_strdup("COBOL");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define LATEX
|
|
ft = filetypes[GEANY_FILETYPES_LATEX];
|
|
ft->lang = 8;
|
|
ft->name = g_strdup("LaTeX");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-tex");
|
|
ft->group = GEANY_FILETYPE_GROUP_MARKUP;
|
|
|
|
#define VHDL
|
|
ft = filetypes[GEANY_FILETYPES_VHDL];
|
|
ft->lang = 21;
|
|
ft->name = g_strdup("VHDL");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-vhdl");
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define VERILOG
|
|
ft = filetypes[GEANY_FILETYPES_VERILOG];
|
|
ft->lang = 39;
|
|
ft->name = g_strdup("Verilog");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define DIFF
|
|
ft = filetypes[GEANY_FILETYPES_DIFF];
|
|
ft->lang = 20;
|
|
ft->name = g_strdup("Diff");
|
|
filetype_make_title(ft, TITLE_FILE);
|
|
ft->mime_type = g_strdup("text/x-patch");
|
|
ft->group = GEANY_FILETYPE_GROUP_MISC;
|
|
|
|
#define LISP
|
|
ft = filetypes[GEANY_FILETYPES_LISP];
|
|
ft->name = g_strdup("Lisp");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
|
|
#define ERLANG
|
|
ft = filetypes[GEANY_FILETYPES_ERLANG];
|
|
ft->name = g_strdup("Erlang");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-erlang");
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
|
|
#define CONF
|
|
ft = filetypes[GEANY_FILETYPES_CONF];
|
|
ft->lang = 10;
|
|
ft->name = g_strdup("Conf");
|
|
ft->title = g_strdup(_("Config file"));
|
|
ft->group = GEANY_FILETYPE_GROUP_MISC;
|
|
|
|
#define PO
|
|
ft = filetypes[GEANY_FILETYPES_PO];
|
|
ft->name = g_strdup("Po");
|
|
ft->title = g_strdup(_("Gettext translation file"));
|
|
ft->mime_type = g_strdup("text/x-gettext-translation");
|
|
ft->group = GEANY_FILETYPE_GROUP_MISC;
|
|
|
|
#define HAXE
|
|
ft = filetypes[GEANY_FILETYPES_HAXE];
|
|
ft->lang = 27;
|
|
ft->name = g_strdup("Haxe");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define ACTIONSCRIPT
|
|
ft = filetypes[GEANY_FILETYPES_AS];
|
|
ft->lang = 34;
|
|
ft->name = g_strdup("ActionScript");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("application/ecmascript");
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
|
|
#define R
|
|
ft = filetypes[GEANY_FILETYPES_R];
|
|
ft->lang = 40;
|
|
ft->name = g_strdup("R");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
|
|
#define REST
|
|
ft = filetypes[GEANY_FILETYPES_REST];
|
|
ft->lang = 28;
|
|
ft->name = g_strdup("reStructuredText");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->group = GEANY_FILETYPE_GROUP_MARKUP;
|
|
|
|
#define MATLAB
|
|
ft = filetypes[GEANY_FILETYPES_MATLAB];
|
|
ft->lang = 32;
|
|
ft->name = g_strdup("Matlab/Octave");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-matlab");
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
|
|
#define YAML
|
|
ft = filetypes[GEANY_FILETYPES_YAML];
|
|
ft->name = g_strdup("YAML");
|
|
filetype_make_title(ft, TITLE_FILE);
|
|
ft->group = GEANY_FILETYPE_GROUP_MISC;
|
|
|
|
#define CMAKE
|
|
ft = filetypes[GEANY_FILETYPES_CMAKE];
|
|
ft->name = g_strdup("CMake");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-cmake");
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
|
|
#define NSIS
|
|
ft = filetypes[GEANY_FILETYPES_NSIS];
|
|
ft->lang = 35;
|
|
ft->name = g_strdup("NSIS");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
|
|
#define ADA
|
|
ft = filetypes[GEANY_FILETYPES_ADA];
|
|
ft->name = g_strdup("Ada");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->mime_type = g_strdup("text/x-adasrc");
|
|
ft->group = GEANY_FILETYPE_GROUP_COMPILED;
|
|
|
|
#define FORTH
|
|
ft = filetypes[GEANY_FILETYPES_FORTH];
|
|
ft->name = g_strdup("Forth");
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
ft->group = GEANY_FILETYPE_GROUP_SCRIPT;
|
|
}
|
|
|
|
|
|
/* initialize fields. */
|
|
static GeanyFiletype *filetype_new(void)
|
|
{
|
|
GeanyFiletype *ft = g_new0(GeanyFiletype, 1);
|
|
|
|
ft->group = GEANY_FILETYPE_GROUP_NONE;
|
|
ft->lang = -2; /* assume no tagmanager parser */
|
|
/* pattern must not be null */
|
|
ft->pattern = g_new0(gchar*, 1);
|
|
ft->project_list_entry = -1; /* no entry */
|
|
|
|
ft->priv = g_new0(GeanyFiletypePrivate, 1);
|
|
return ft;
|
|
}
|
|
|
|
|
|
static gint cmp_filetype(gconstpointer pft1, gconstpointer pft2, gpointer data)
|
|
{
|
|
gboolean by_name = GPOINTER_TO_INT(data);
|
|
const GeanyFiletype *ft1 = pft1, *ft2 = pft2;
|
|
|
|
if (G_UNLIKELY(ft1->id == GEANY_FILETYPES_NONE))
|
|
return -1;
|
|
if (G_UNLIKELY(ft2->id == GEANY_FILETYPES_NONE))
|
|
return 1;
|
|
|
|
return by_name ?
|
|
utils_str_casecmp(ft1->name, ft2->name) :
|
|
utils_str_casecmp(ft1->title, ft2->title);
|
|
}
|
|
|
|
|
|
/** Gets a list of filetype pointers sorted by name.
|
|
* The list does not change on subsequent calls.
|
|
* @return The list - do not free.
|
|
* @see filetypes_by_title. */
|
|
const GSList *filetypes_get_sorted_by_name(void)
|
|
{
|
|
static GSList *list = NULL;
|
|
|
|
g_return_val_if_fail(filetypes_by_title, NULL);
|
|
|
|
if (!list)
|
|
{
|
|
list = g_slist_copy(filetypes_by_title);
|
|
list = g_slist_sort_with_data(list, cmp_filetype, GINT_TO_POINTER(TRUE));
|
|
}
|
|
return list;
|
|
}
|
|
|
|
|
|
/* Add a filetype pointer to the lists of available filetypes,
|
|
* and set the filetype::id field. */
|
|
static void filetype_add(GeanyFiletype *ft)
|
|
{
|
|
g_return_if_fail(ft);
|
|
g_return_if_fail(ft->name);
|
|
|
|
ft->id = filetypes_array->len; /* len will be the index for filetype_array */
|
|
g_ptr_array_add(filetypes_array, ft);
|
|
g_hash_table_insert(filetypes_hash, ft->name, ft);
|
|
|
|
/* list will be sorted later */
|
|
filetypes_by_title = g_slist_prepend(filetypes_by_title, ft);
|
|
|
|
if (!ft->mime_type)
|
|
ft->mime_type = g_strdup("text/plain");
|
|
}
|
|
|
|
|
|
static void add_custom_filetype(const gchar *filename)
|
|
{
|
|
gchar *fn = utils_strdupa(strstr(filename, ".") + 1);
|
|
gchar *dot = g_strrstr(fn, ".conf");
|
|
GeanyFiletype *ft;
|
|
|
|
g_return_if_fail(dot);
|
|
|
|
*dot = 0x0;
|
|
|
|
if (g_hash_table_lookup(filetypes_hash, fn))
|
|
return;
|
|
|
|
ft = filetype_new();
|
|
ft->name = g_strdup(fn);
|
|
filetype_make_title(ft, TITLE_FILE);
|
|
ft->priv->custom = TRUE;
|
|
filetype_add(ft);
|
|
geany_debug("Added filetype %s (%d).", ft->name, ft->id);
|
|
}
|
|
|
|
|
|
static void init_custom_filetypes(const gchar *path)
|
|
{
|
|
GDir *dir;
|
|
const gchar *filename;
|
|
|
|
g_return_if_fail(path);
|
|
|
|
dir = g_dir_open(path, 0, NULL);
|
|
if (dir == NULL)
|
|
return;
|
|
|
|
foreach_dir(filename, dir)
|
|
{
|
|
const gchar prefix[] = "filetypes.";
|
|
|
|
if (g_str_has_prefix(filename, prefix) &&
|
|
g_str_has_suffix(filename + strlen(prefix), ".conf"))
|
|
{
|
|
add_custom_filetype(filename);
|
|
}
|
|
}
|
|
g_dir_close(dir);
|
|
}
|
|
|
|
|
|
/* Create the filetypes array and fill it with the known filetypes.
|
|
* Warning: GTK isn't necessarily initialized yet. */
|
|
void filetypes_init_types()
|
|
{
|
|
filetype_id ft_id;
|
|
gchar *f;
|
|
|
|
g_return_if_fail(filetypes_array == NULL);
|
|
g_return_if_fail(filetypes_hash == NULL);
|
|
|
|
filetypes_array = g_ptr_array_sized_new(GEANY_MAX_BUILT_IN_FILETYPES);
|
|
filetypes_hash = g_hash_table_new(g_str_hash, g_str_equal);
|
|
|
|
/* Create built-in filetypes */
|
|
for (ft_id = 0; ft_id < GEANY_MAX_BUILT_IN_FILETYPES; ft_id++)
|
|
{
|
|
filetypes[ft_id] = filetype_new();
|
|
}
|
|
init_builtin_filetypes();
|
|
|
|
/* Add built-in filetypes to the hash now the name fields are set */
|
|
for (ft_id = 0; ft_id < GEANY_MAX_BUILT_IN_FILETYPES; ft_id++)
|
|
{
|
|
filetype_add(filetypes[ft_id]);
|
|
}
|
|
init_custom_filetypes(app->datadir);
|
|
f = utils_build_path(app->configdir, GEANY_FILEDEFS_SUBDIR, NULL);
|
|
init_custom_filetypes(f);
|
|
g_free(f);
|
|
|
|
/* sort last instead of on insertion to prevent exponential time */
|
|
filetypes_by_title = g_slist_sort_with_data(filetypes_by_title,
|
|
cmp_filetype, GINT_TO_POINTER(FALSE));
|
|
|
|
read_filetype_config();
|
|
}
|
|
|
|
|
|
static void on_document_save(G_GNUC_UNUSED GObject *object, GeanyDocument *doc)
|
|
{
|
|
gchar *f;
|
|
|
|
g_return_if_fail(NZV(doc->real_path));
|
|
|
|
f = utils_build_path(app->configdir, "filetype_extensions.conf", NULL);
|
|
if (utils_str_equal(doc->real_path, f))
|
|
filetypes_reload_extensions();
|
|
|
|
g_free(f);
|
|
f = utils_build_path(app->configdir, GEANY_FILEDEFS_SUBDIR, "filetypes.common", NULL);
|
|
if (utils_str_equal(doc->real_path, f))
|
|
{
|
|
guint i;
|
|
|
|
/* Note: we don't reload other filetypes, even though the named styles may have changed.
|
|
* The user can do this manually with 'Tools->Reload Configuration' */
|
|
filetypes_load_config(GEANY_FILETYPES_NONE, TRUE);
|
|
|
|
foreach_document(i)
|
|
document_reload_config(documents[i]);
|
|
}
|
|
g_free(f);
|
|
}
|
|
|
|
|
|
static void setup_config_file_menus(void)
|
|
{
|
|
gchar *f;
|
|
|
|
f = utils_build_path(app->configdir, "filetype_extensions.conf", NULL);
|
|
ui_add_config_file_menu_item(f, NULL, NULL);
|
|
setptr(f, utils_build_path(app->configdir, GEANY_FILEDEFS_SUBDIR, "filetypes.common", NULL));
|
|
ui_add_config_file_menu_item(f, NULL, NULL);
|
|
g_free(f);
|
|
|
|
g_signal_connect(geany_object, "document-save", G_CALLBACK(on_document_save), NULL);
|
|
}
|
|
|
|
|
|
static GtkWidget *group_menus[GEANY_FILETYPE_GROUP_COUNT] = {NULL};
|
|
|
|
static void create_sub_menu(GtkWidget *parent, gsize group_id, const gchar *title)
|
|
{
|
|
GtkWidget *menu, *item;
|
|
|
|
menu = gtk_menu_new();
|
|
item = gtk_menu_item_new_with_mnemonic((title));
|
|
gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), menu);
|
|
gtk_container_add(GTK_CONTAINER(parent), item);
|
|
gtk_widget_show(item);
|
|
group_menus[group_id] = menu;
|
|
}
|
|
|
|
|
|
static void create_set_filetype_menu(void)
|
|
{
|
|
GSList *node;
|
|
GtkWidget *filetype_menu = ui_lookup_widget(main_widgets.window, "set_filetype1_menu");
|
|
|
|
create_sub_menu(filetype_menu, GEANY_FILETYPE_GROUP_COMPILED, _("_Programming Languages"));
|
|
create_sub_menu(filetype_menu, GEANY_FILETYPE_GROUP_SCRIPT, _("_Scripting Languages"));
|
|
create_sub_menu(filetype_menu, GEANY_FILETYPE_GROUP_MARKUP, _("_Markup Languages"));
|
|
create_sub_menu(filetype_menu, GEANY_FILETYPE_GROUP_MISC, _("M_iscellaneous"));
|
|
|
|
/* Append all filetypes to the filetype menu */
|
|
foreach_slist(node, filetypes_by_title)
|
|
{
|
|
GeanyFiletype *ft = node->data;
|
|
|
|
if (ft->group != GEANY_FILETYPE_GROUP_NONE)
|
|
create_radio_menu_item(group_menus[ft->group], ft);
|
|
else
|
|
create_radio_menu_item(filetype_menu, ft);
|
|
}
|
|
}
|
|
|
|
|
|
void filetypes_init()
|
|
{
|
|
GSList *node;
|
|
|
|
filetypes_init_types();
|
|
|
|
/* this has to be here as GTK isn't initialized in filetypes_init_types(). */
|
|
foreach_slist(node, filetypes_by_title)
|
|
{
|
|
GeanyFiletype *ft = node->data;
|
|
ft->icon = ui_get_mime_icon(ft->mime_type, GTK_ICON_SIZE_MENU);
|
|
}
|
|
create_set_filetype_menu();
|
|
setup_config_file_menus();
|
|
}
|
|
|
|
|
|
/* Find a filetype that predicate returns TRUE for, otherwise return NULL. */
|
|
GeanyFiletype *filetypes_find(GCompareFunc predicate, gpointer user_data)
|
|
{
|
|
guint i;
|
|
|
|
for (i = 0; i < filetypes_array->len; i++)
|
|
{
|
|
GeanyFiletype *ft = filetypes[i];
|
|
|
|
if (predicate(ft, user_data))
|
|
return ft;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
static gboolean match_basename(gconstpointer pft, gconstpointer user_data)
|
|
{
|
|
const GeanyFiletype *ft = pft;
|
|
const gchar *base_filename = user_data;
|
|
gint j;
|
|
gboolean ret = FALSE;
|
|
|
|
if (G_UNLIKELY(ft->id == GEANY_FILETYPES_NONE))
|
|
return FALSE;
|
|
|
|
for (j = 0; ft->pattern[j] != NULL; j++)
|
|
{
|
|
GPatternSpec *pattern = g_pattern_spec_new(ft->pattern[j]);
|
|
|
|
if (g_pattern_match_string(pattern, base_filename))
|
|
{
|
|
ret = TRUE;
|
|
g_pattern_spec_free(pattern);
|
|
break;
|
|
}
|
|
g_pattern_spec_free(pattern);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
|
|
static GeanyFiletype *check_builtin_filenames(const gchar *utf8_filename)
|
|
{
|
|
gchar *lfn = NULL;
|
|
gchar *path;
|
|
gboolean found = FALSE;
|
|
|
|
#ifdef G_OS_WIN32
|
|
/* use lower case basename */
|
|
lfn = g_utf8_strdown(utf8_filename, -1);
|
|
#else
|
|
lfn = g_strdup(utf8_filename);
|
|
#endif
|
|
setptr(lfn, utils_get_locale_from_utf8(lfn));
|
|
|
|
path = utils_build_path(app->configdir, GEANY_FILEDEFS_SUBDIR, "filetypes.", NULL);
|
|
if (g_str_has_prefix(lfn, path))
|
|
found = TRUE;
|
|
|
|
setptr(path, utils_build_path(app->datadir, "filetypes.", NULL));
|
|
if (g_str_has_prefix(lfn, path))
|
|
found = TRUE;
|
|
|
|
g_free(path);
|
|
g_free(lfn);
|
|
return found ? filetypes[GEANY_FILETYPES_CONF] : NULL;
|
|
}
|
|
|
|
|
|
/* Detect filetype only based on the filename extension.
|
|
* utf8_filename can include the full path. */
|
|
GeanyFiletype *filetypes_detect_from_extension(const gchar *utf8_filename)
|
|
{
|
|
gchar *base_filename;
|
|
GeanyFiletype *ft;
|
|
|
|
ft = check_builtin_filenames(utf8_filename);
|
|
if (ft)
|
|
return ft;
|
|
|
|
/* to match against the basename of the file (because of Makefile*) */
|
|
base_filename = g_path_get_basename(utf8_filename);
|
|
#ifdef G_OS_WIN32
|
|
/* use lower case basename */
|
|
setptr(base_filename, g_utf8_strdown(base_filename, -1));
|
|
#endif
|
|
|
|
ft = filetypes_find(match_basename, base_filename);
|
|
if (ft == NULL)
|
|
ft = filetypes[GEANY_FILETYPES_NONE];
|
|
|
|
g_free(base_filename);
|
|
return ft;
|
|
}
|
|
|
|
|
|
/* This detects the filetype of the file pointed by 'utf8_filename' and a list of filetype id's,
|
|
* terminated by -1.
|
|
* The detected filetype of the file is checked against every id in the passed list and if
|
|
* there is a match, TRUE is returned. */
|
|
static gboolean shebang_find_and_match_filetype(const gchar *utf8_filename, gint first, ...)
|
|
{
|
|
GeanyFiletype *ft = NULL;
|
|
gint test;
|
|
gboolean result = FALSE;
|
|
va_list args;
|
|
|
|
ft = filetypes_detect_from_extension(utf8_filename);
|
|
if (ft == NULL || ft->id >= filetypes_array->len)
|
|
return FALSE;
|
|
|
|
va_start(args, first);
|
|
test = first;
|
|
while (1)
|
|
{
|
|
if (test == -1)
|
|
break;
|
|
|
|
if (ft->id == (guint) test)
|
|
{
|
|
result = TRUE;
|
|
break;
|
|
}
|
|
test = va_arg(args, gint);
|
|
}
|
|
va_end(args);
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
static GeanyFiletype *find_shebang(const gchar *utf8_filename, const gchar *line)
|
|
{
|
|
GeanyFiletype *ft = NULL;
|
|
|
|
if (strlen(line) > 2 && line[0] == '#' && line[1] == '!')
|
|
{
|
|
gchar *tmp = g_path_get_basename(line + 2);
|
|
gchar *basename_interpreter = tmp;
|
|
|
|
if (strncmp(tmp, "env ", 4) == 0 && strlen(tmp) > 4)
|
|
{ /* skip "env" and read the following interpreter */
|
|
basename_interpreter += 4;
|
|
}
|
|
|
|
if (strncmp(basename_interpreter, "sh", 2) == 0)
|
|
ft = filetypes[GEANY_FILETYPES_SH];
|
|
else if (strncmp(basename_interpreter, "bash", 4) == 0)
|
|
ft = filetypes[GEANY_FILETYPES_SH];
|
|
else if (strncmp(basename_interpreter, "perl", 4) == 0)
|
|
ft = filetypes[GEANY_FILETYPES_PERL];
|
|
else if (strncmp(basename_interpreter, "python", 6) == 0)
|
|
ft = filetypes[GEANY_FILETYPES_PYTHON];
|
|
else if (strncmp(basename_interpreter, "php", 3) == 0)
|
|
ft = filetypes[GEANY_FILETYPES_PHP];
|
|
else if (strncmp(basename_interpreter, "ruby", 4) == 0)
|
|
ft = filetypes[GEANY_FILETYPES_RUBY];
|
|
else if (strncmp(basename_interpreter, "tcl", 3) == 0)
|
|
ft = filetypes[GEANY_FILETYPES_TCL];
|
|
else if (strncmp(basename_interpreter, "make", 4) == 0)
|
|
ft = filetypes[GEANY_FILETYPES_MAKE];
|
|
else if (strncmp(basename_interpreter, "zsh", 3) == 0)
|
|
ft = filetypes[GEANY_FILETYPES_SH];
|
|
else if (strncmp(basename_interpreter, "ksh", 3) == 0)
|
|
ft = filetypes[GEANY_FILETYPES_SH];
|
|
else if (strncmp(basename_interpreter, "csh", 3) == 0)
|
|
ft = filetypes[GEANY_FILETYPES_SH];
|
|
else if (strncmp(basename_interpreter, "ash", 3) == 0)
|
|
ft = filetypes[GEANY_FILETYPES_SH];
|
|
else if (strncmp(basename_interpreter, "dmd", 3) == 0)
|
|
ft = filetypes[GEANY_FILETYPES_D];
|
|
else if (strncmp(basename_interpreter, "wish", 4) == 0)
|
|
ft = filetypes[GEANY_FILETYPES_TCL];
|
|
|
|
g_free(tmp);
|
|
}
|
|
/* detect HTML files */
|
|
if (strncmp(line, "<!DOCTYPE html", 14) == 0 || strncmp(line, "<html", 5) == 0)
|
|
{
|
|
/* PHP, Perl and Python files might also start with <html, so detect them based on filename
|
|
* extension and use the detected filetype, else assume HTML */
|
|
if (! shebang_find_and_match_filetype(utf8_filename,
|
|
GEANY_FILETYPES_PERL, GEANY_FILETYPES_PHP, GEANY_FILETYPES_PYTHON, -1))
|
|
{
|
|
ft = filetypes[GEANY_FILETYPES_HTML];
|
|
}
|
|
}
|
|
/* detect XML files */
|
|
else if (utf8_filename && strncmp(line, "<?xml", 5) == 0)
|
|
{
|
|
/* HTML and DocBook files might also start with <?xml, so detect them based on filename
|
|
* extension and use the detected filetype, else assume XML */
|
|
if (! shebang_find_and_match_filetype(utf8_filename,
|
|
GEANY_FILETYPES_HTML, GEANY_FILETYPES_DOCBOOK,
|
|
/* Perl, Python and PHP only to be safe */
|
|
GEANY_FILETYPES_PERL, GEANY_FILETYPES_PHP, GEANY_FILETYPES_PYTHON, -1))
|
|
{
|
|
ft = filetypes[GEANY_FILETYPES_XML];
|
|
}
|
|
}
|
|
else if (strncmp(line, "<?php", 5) == 0)
|
|
{
|
|
ft = filetypes[GEANY_FILETYPES_PHP];
|
|
}
|
|
|
|
return ft;
|
|
}
|
|
|
|
|
|
/* Detect the filetype checking for a shebang, then filename extension. */
|
|
static GeanyFiletype *filetypes_detect_from_file_internal(const gchar *utf8_filename,
|
|
const gchar *line)
|
|
{
|
|
GeanyFiletype *ft;
|
|
|
|
/* try to find a shebang and if found use it prior to the filename extension
|
|
* also checks for <?xml */
|
|
ft = find_shebang(utf8_filename, line);
|
|
if (ft != NULL)
|
|
return ft;
|
|
|
|
if (utf8_filename == NULL)
|
|
return filetypes[GEANY_FILETYPES_NONE];
|
|
|
|
return filetypes_detect_from_extension(utf8_filename);
|
|
}
|
|
|
|
|
|
/* Detect the filetype for the document, checking for a shebang, then filename extension. */
|
|
GeanyFiletype *filetypes_detect_from_document(GeanyDocument *doc)
|
|
{
|
|
GeanyFiletype *ft;
|
|
gchar *line;
|
|
|
|
if (doc == NULL)
|
|
return filetypes[GEANY_FILETYPES_NONE];
|
|
|
|
line = sci_get_line(doc->editor->sci, 0);
|
|
ft = filetypes_detect_from_file_internal(doc->file_name, line);
|
|
g_free(line);
|
|
return ft;
|
|
}
|
|
|
|
|
|
#ifdef HAVE_PLUGINS
|
|
/* Currently only used by external plugins (e.g. geanyprj). */
|
|
/**
|
|
* Detects filetype based on a shebang line in the file or the filename extension.
|
|
*
|
|
* @param utf8_filename The filename in UTF-8 encoding.
|
|
*
|
|
* @return The detected filetype for @a utf8_filename or @c filetypes[GEANY_FILETYPES_NONE]
|
|
* if it could not be detected.
|
|
**/
|
|
GeanyFiletype *filetypes_detect_from_file(const gchar *utf8_filename)
|
|
{
|
|
gchar line[1024];
|
|
FILE *f;
|
|
gchar *locale_name = utils_get_locale_from_utf8(utf8_filename);
|
|
|
|
f = g_fopen(locale_name, "r");
|
|
g_free(locale_name);
|
|
if (f != NULL)
|
|
{
|
|
if (fgets(line, sizeof(line), f) != NULL)
|
|
{
|
|
fclose(f);
|
|
return filetypes_detect_from_file_internal(utf8_filename, line);
|
|
}
|
|
fclose(f);
|
|
}
|
|
return filetypes_detect_from_extension(utf8_filename);
|
|
}
|
|
#endif
|
|
|
|
|
|
void filetypes_select_radio_item(const GeanyFiletype *ft)
|
|
{
|
|
/* ignore_callback has to be set by the caller */
|
|
g_return_if_fail(ignore_callback);
|
|
|
|
if (ft == NULL)
|
|
ft = filetypes[GEANY_FILETYPES_NONE];
|
|
|
|
gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(ft->priv->menu_item), TRUE);
|
|
}
|
|
|
|
|
|
static void
|
|
on_filetype_change(GtkCheckMenuItem *menuitem, gpointer user_data)
|
|
{
|
|
GeanyDocument *doc = document_get_current();
|
|
if (ignore_callback || doc == NULL || ! gtk_check_menu_item_get_active(menuitem))
|
|
return;
|
|
|
|
document_set_filetype(doc, (GeanyFiletype*)user_data);
|
|
}
|
|
|
|
|
|
static void create_radio_menu_item(GtkWidget *menu, GeanyFiletype *ftype)
|
|
{
|
|
static GSList *group = NULL;
|
|
GtkWidget *tmp;
|
|
|
|
tmp = gtk_radio_menu_item_new_with_label(group, ftype->title);
|
|
group = gtk_radio_menu_item_get_group(GTK_RADIO_MENU_ITEM(tmp));
|
|
ftype->priv->menu_item = tmp;
|
|
gtk_widget_show(tmp);
|
|
gtk_container_add(GTK_CONTAINER(menu), tmp);
|
|
g_signal_connect(tmp, "activate", G_CALLBACK(on_filetype_change), (gpointer) ftype);
|
|
}
|
|
|
|
|
|
#if 0
|
|
/* Remove a filetype pointer from the list of available filetypes. */
|
|
static void filetype_remove(GeanyFiletype *ft)
|
|
{
|
|
g_return_if_fail(ft);
|
|
|
|
g_ptr_array_remove(filetypes_array, ft);
|
|
|
|
if (!g_hash_table_remove(filetypes_hash, ft))
|
|
g_warning("Could not remove filetype %p!", ft);
|
|
}
|
|
#endif
|
|
|
|
|
|
static void set_error_regex(GeanyFiletype *ft, gchar *string)
|
|
{
|
|
setptr(ft->error_regex_string, string);
|
|
|
|
if (ft->priv->error_regex_compiled)
|
|
regfree(&ft->priv->error_regex);
|
|
|
|
ft->priv->error_regex_compiled = FALSE;
|
|
/* regex will be compiled when needed */
|
|
}
|
|
|
|
|
|
static void filetype_free(gpointer data, G_GNUC_UNUSED gpointer user_data)
|
|
{
|
|
GeanyFiletype *ft = data;
|
|
|
|
g_return_if_fail(ft != NULL);
|
|
|
|
g_free(ft->name);
|
|
g_free(ft->title);
|
|
g_free(ft->extension);
|
|
g_free(ft->mime_type);
|
|
g_free(ft->comment_open);
|
|
g_free(ft->comment_close);
|
|
g_free(ft->comment_single);
|
|
g_free(ft->context_action_cmd);
|
|
g_free(ft->filecmds);
|
|
g_free(ft->ftdefcmds);
|
|
g_free(ft->execcmds);
|
|
set_error_regex(ft, NULL);
|
|
if (ft->icon)
|
|
g_object_unref(ft->icon);
|
|
|
|
g_strfreev(ft->pattern);
|
|
g_free(ft->priv);
|
|
g_free(ft);
|
|
}
|
|
|
|
|
|
/* frees the array and all related pointers */
|
|
void filetypes_free_types(void)
|
|
{
|
|
g_return_if_fail(filetypes_array != NULL);
|
|
g_return_if_fail(filetypes_hash != NULL);
|
|
|
|
g_ptr_array_foreach(filetypes_array, filetype_free, NULL);
|
|
g_ptr_array_free(filetypes_array, TRUE);
|
|
g_hash_table_destroy(filetypes_hash);
|
|
}
|
|
|
|
|
|
static void load_settings(gint ft_id, GKeyFile *config, GKeyFile *configh)
|
|
{
|
|
GeanyFiletype *ft = filetypes[ft_id];
|
|
gchar *result;
|
|
|
|
/* default extension */
|
|
result = utils_get_setting(string, configh, config, "settings", "extension", NULL);
|
|
if (result != NULL)
|
|
{
|
|
setptr(filetypes[ft_id]->extension, result);
|
|
}
|
|
|
|
/* read comment notes */
|
|
result = utils_get_setting(string, configh, config, "settings", "comment_open", NULL);
|
|
if (result != NULL)
|
|
{
|
|
setptr(filetypes[ft_id]->comment_open, result);
|
|
}
|
|
|
|
result = utils_get_setting(string, configh, config, "settings", "comment_close", NULL);
|
|
if (result != NULL)
|
|
{
|
|
setptr(filetypes[ft_id]->comment_close, result);
|
|
}
|
|
|
|
result = utils_get_setting(string, configh, config, "settings", "comment_single", NULL);
|
|
if (result != NULL)
|
|
{
|
|
setptr(filetypes[ft_id]->comment_single, result);
|
|
}
|
|
/* import correctly filetypes that use old-style single comments */
|
|
else if (! NZV(filetypes[ft_id]->comment_close))
|
|
{
|
|
setptr(filetypes[ft_id]->comment_single, filetypes[ft_id]->comment_open);
|
|
filetypes[ft_id]->comment_open = NULL;
|
|
}
|
|
|
|
filetypes[ft_id]->comment_use_indent = utils_get_setting(boolean, configh, config,
|
|
"settings", "comment_use_indent", FALSE);
|
|
|
|
/* read context action */
|
|
result = utils_get_setting(string, configh, config, "settings", "context_action_cmd", NULL);
|
|
if (result != NULL)
|
|
{
|
|
setptr(filetypes[ft_id]->context_action_cmd, result);
|
|
}
|
|
|
|
result = utils_get_setting(string, configh, config, "settings", "tag_parser", NULL);
|
|
if (result != NULL)
|
|
{
|
|
ft->lang = tm_source_file_get_named_lang(result);
|
|
if (ft->lang < 0)
|
|
geany_debug("Cannot find tag parser '%s' for custom filetype '%s'.", result, ft->name);
|
|
g_free(result);
|
|
}
|
|
|
|
result = utils_get_setting(string, configh, config, "settings", "lexer_filetype", NULL);
|
|
if (result != NULL)
|
|
{
|
|
ft->lexer_filetype = filetypes_lookup_by_name(result);
|
|
if (!ft->lexer_filetype)
|
|
geany_debug("Cannot find lexer filetype '%s' for custom filetype '%s'.", result, ft->name);
|
|
g_free(result);
|
|
}
|
|
|
|
ft->priv->symbol_list_sort_mode = utils_get_setting(integer, configh, config, "settings",
|
|
"symbol_list_sort_mode", SYMBOLS_SORT_BY_NAME);
|
|
ft->priv->xml_indent_tags = utils_get_setting(boolean, configh, config, "settings",
|
|
"xml_indent_tags", FALSE);
|
|
|
|
/* read build settings */
|
|
build_load_menu(config, GEANY_BCS_FT, (gpointer)ft);
|
|
build_load_menu(configh, GEANY_BCS_HOME_FT, (gpointer)ft);
|
|
}
|
|
|
|
|
|
static void add_keys(GKeyFile *dest, const gchar *group, GKeyFile *src)
|
|
{
|
|
gchar **keys = g_key_file_get_keys(src, group, NULL, NULL);
|
|
gchar **ptr;
|
|
|
|
foreach_strv(ptr, keys)
|
|
{
|
|
gchar *key = *ptr;
|
|
gchar *value = g_key_file_get_value(src, group, key, NULL);
|
|
|
|
g_key_file_set_value(dest, group, key, value);
|
|
g_free(value);
|
|
}
|
|
g_strfreev(keys);
|
|
}
|
|
|
|
|
|
static gchar *filetypes_get_filename(GeanyFiletype *ft, gboolean user)
|
|
{
|
|
gchar *ext = filetypes_get_conf_extension(ft);
|
|
gchar *f;
|
|
|
|
if (user)
|
|
f = utils_make_filename(app->configdir,
|
|
GEANY_FILEDEFS_SUBDIR G_DIR_SEPARATOR_S, "filetypes.", ext, NULL);
|
|
else
|
|
f = utils_make_filename(app->datadir, "filetypes.", ext, NULL);
|
|
|
|
g_free(ext);
|
|
return f;
|
|
}
|
|
|
|
|
|
static void add_group_keys(GKeyFile *kf, const gchar *group, GeanyFiletype *ft)
|
|
{
|
|
gchar *files[2];
|
|
gboolean loaded = FALSE;
|
|
guint i;
|
|
|
|
files[0] = filetypes_get_filename(ft, FALSE);
|
|
files[1] = filetypes_get_filename(ft, TRUE);
|
|
|
|
for (i = 0; i < G_N_ELEMENTS(files); i++)
|
|
{
|
|
GKeyFile *src = g_key_file_new();
|
|
|
|
if (g_key_file_load_from_file(src, files[i], G_KEY_FILE_NONE, NULL))
|
|
{
|
|
add_keys(kf, group, src);
|
|
loaded = TRUE;
|
|
}
|
|
g_key_file_free(src);
|
|
}
|
|
|
|
if (!loaded)
|
|
geany_debug("Could not read config file %s for [%s=%s]!", files[0], group, ft->name);
|
|
|
|
g_free(files[0]);
|
|
g_free(files[1]);
|
|
}
|
|
|
|
|
|
static void copy_ft_groups(GKeyFile *kf)
|
|
{
|
|
gchar **groups = g_key_file_get_groups(kf, NULL);
|
|
gchar **ptr;
|
|
|
|
foreach_strv(ptr, groups)
|
|
{
|
|
gchar *group = *ptr;
|
|
gchar *name = strstr(*ptr, "=");
|
|
GeanyFiletype *ft;
|
|
|
|
if (!name)
|
|
continue;
|
|
|
|
/* terminate group at '=' */
|
|
*name = 0;
|
|
name++;
|
|
if (!name[0])
|
|
continue;
|
|
|
|
ft = filetypes_lookup_by_name(name);
|
|
if (ft)
|
|
add_group_keys(kf, group, ft);
|
|
}
|
|
g_strfreev(groups);
|
|
}
|
|
|
|
|
|
/* simple wrapper function to print file errors in DEBUG mode */
|
|
static void load_system_keyfile(GKeyFile *key_file, const gchar *file, GKeyFileFlags flags,
|
|
GeanyFiletype *ft)
|
|
{
|
|
GError *error = NULL;
|
|
gboolean done = g_key_file_load_from_file(key_file, file, flags, &error);
|
|
|
|
if (error != NULL)
|
|
{
|
|
if (!done && !ft->priv->custom)
|
|
geany_debug("Failed to open %s (%s)", file, error->message);
|
|
|
|
g_error_free(error);
|
|
error = NULL;
|
|
}
|
|
}
|
|
|
|
|
|
/* Load the configuration file for the associated filetype id.
|
|
* This should only be called when the filetype is needed, to save loading
|
|
* 20+ configuration files all at once. */
|
|
void filetypes_load_config(gint ft_id, gboolean reload)
|
|
{
|
|
GKeyFile *config, *config_home;
|
|
GeanyFiletypePrivate *pft;
|
|
GeanyFiletype *ft;
|
|
|
|
g_return_if_fail(ft_id >= 0 && ft_id < (gint) filetypes_array->len);
|
|
|
|
ft = filetypes[ft_id];
|
|
pft = ft->priv;
|
|
|
|
/* when reloading, proceed only if the settings were already loaded */
|
|
if (G_UNLIKELY(reload && ! pft->keyfile_loaded))
|
|
return;
|
|
|
|
/* when not reloading, load the settings only once */
|
|
if (G_LIKELY(! reload && pft->keyfile_loaded))
|
|
return;
|
|
pft->keyfile_loaded = TRUE;
|
|
|
|
config = g_key_file_new();
|
|
config_home = g_key_file_new();
|
|
{
|
|
/* highlighting uses GEANY_FILETYPES_NONE for common settings */
|
|
gchar *f;
|
|
|
|
f = filetypes_get_filename(ft, FALSE);
|
|
load_system_keyfile(config, f, G_KEY_FILE_KEEP_COMMENTS, ft);
|
|
|
|
setptr(f, filetypes_get_filename(ft, TRUE));
|
|
g_key_file_load_from_file(config_home, f, G_KEY_FILE_KEEP_COMMENTS, NULL);
|
|
g_free(f);
|
|
}
|
|
/* Copy keys for any groups with [group=C] from system keyfile */
|
|
copy_ft_groups(config);
|
|
copy_ft_groups(config_home);
|
|
|
|
load_settings(ft_id, config, config_home);
|
|
highlighting_init_styles(ft_id, config, config_home);
|
|
|
|
g_key_file_free(config);
|
|
g_key_file_free(config_home);
|
|
}
|
|
|
|
|
|
static gchar *filetypes_get_conf_extension(const GeanyFiletype *ft)
|
|
{
|
|
gchar *result;
|
|
|
|
if (ft->priv->custom)
|
|
return g_strconcat(ft->name, ".conf", NULL);
|
|
|
|
/* Handle any special extensions different from lowercase filetype->name */
|
|
switch (ft->id)
|
|
{
|
|
case GEANY_FILETYPES_CPP: result = g_strdup("cpp"); break;
|
|
case GEANY_FILETYPES_CS: result = g_strdup("cs"); break;
|
|
case GEANY_FILETYPES_MAKE: result = g_strdup("makefile"); break;
|
|
case GEANY_FILETYPES_NONE: result = g_strdup("common"); break;
|
|
/* name is Matlab/Octave */
|
|
case GEANY_FILETYPES_MATLAB: result = g_strdup("matlab"); break;
|
|
default:
|
|
result = g_ascii_strdown(ft->name, -1);
|
|
break;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
|
|
void filetypes_save_commands(GeanyFiletype *ft)
|
|
{
|
|
GKeyFile *config_home;
|
|
gchar *fname, *data;
|
|
|
|
fname = filetypes_get_filename(ft, TRUE);
|
|
config_home = g_key_file_new();
|
|
g_key_file_load_from_file(config_home, fname, G_KEY_FILE_KEEP_COMMENTS, NULL);
|
|
build_save_menu(config_home, ft, GEANY_BCS_HOME_FT);
|
|
data = g_key_file_to_data(config_home, NULL, NULL);
|
|
utils_write_file(fname, data);
|
|
g_free(data);
|
|
g_key_file_free(config_home);
|
|
g_free(fname);
|
|
}
|
|
|
|
|
|
/* create one file filter which has each file pattern of each filetype */
|
|
GtkFileFilter *filetypes_create_file_filter_all_source(void)
|
|
{
|
|
GtkFileFilter *new_filter;
|
|
guint i, j;
|
|
|
|
new_filter = gtk_file_filter_new();
|
|
gtk_file_filter_set_name(new_filter, _("All Source"));
|
|
|
|
for (i = 0; i < filetypes_array->len; i++)
|
|
{
|
|
if (G_UNLIKELY(i == GEANY_FILETYPES_NONE))
|
|
continue;
|
|
|
|
for (j = 0; filetypes[i]->pattern[j]; j++)
|
|
{
|
|
gtk_file_filter_add_pattern(new_filter, filetypes[i]->pattern[j]);
|
|
}
|
|
}
|
|
return new_filter;
|
|
}
|
|
|
|
|
|
GtkFileFilter *filetypes_create_file_filter(const GeanyFiletype *ft)
|
|
{
|
|
GtkFileFilter *new_filter;
|
|
gint i;
|
|
const gchar *title;
|
|
|
|
g_return_val_if_fail(ft != NULL, NULL);
|
|
|
|
new_filter = gtk_file_filter_new();
|
|
title = ft->id == GEANY_FILETYPES_NONE ? _("All files") : ft->title;
|
|
gtk_file_filter_set_name(new_filter, title);
|
|
|
|
for (i = 0; ft->pattern[i]; i++)
|
|
{
|
|
gtk_file_filter_add_pattern(new_filter, ft->pattern[i]);
|
|
}
|
|
|
|
return new_filter;
|
|
}
|
|
|
|
|
|
/* Indicates whether there is a tag parser for the filetype or not.
|
|
* Only works for custom filetypes if the filetype settings have been loaded. */
|
|
gboolean filetype_has_tags(GeanyFiletype *ft)
|
|
{
|
|
g_return_val_if_fail(ft != NULL, FALSE);
|
|
|
|
return ft->lang >= 0;
|
|
}
|
|
|
|
|
|
/** Finds a filetype pointer from its @a name field.
|
|
* @param name Filetype name.
|
|
* @return The filetype found, or @c NULL.
|
|
*
|
|
* @since 0.15
|
|
**/
|
|
GeanyFiletype *filetypes_lookup_by_name(const gchar *name)
|
|
{
|
|
GeanyFiletype *ft;
|
|
|
|
g_return_val_if_fail(NZV(name), NULL);
|
|
|
|
ft = g_hash_table_lookup(filetypes_hash, name);
|
|
if (G_UNLIKELY(ft == NULL))
|
|
geany_debug("Could not find filetype '%s'.", name);
|
|
return ft;
|
|
}
|
|
|
|
|
|
static gchar *get_regex_match_string(const gchar *message, regmatch_t *pmatch, gint match_idx)
|
|
{
|
|
return g_strndup(&message[pmatch[match_idx].rm_so],
|
|
pmatch[match_idx].rm_eo - pmatch[match_idx].rm_so);
|
|
}
|
|
|
|
|
|
static void compile_regex(GeanyFiletype *ft, regex_t *regex, gchar *regstr)
|
|
{
|
|
gint retval = regcomp(regex, regstr, REG_EXTENDED);
|
|
|
|
ft->priv->error_regex_compiled = (retval == 0); /* prevent recompilation */
|
|
|
|
if (G_UNLIKELY(retval != 0))
|
|
{
|
|
gchar buf[256];
|
|
regerror(retval, regex, buf, sizeof buf);
|
|
ui_set_statusbar(TRUE, _("Bad regex for filetype %s: %s"),
|
|
filetypes_get_display_name(ft), buf);
|
|
}
|
|
/* regex will be freed in set_error_regex(). */
|
|
}
|
|
|
|
|
|
gboolean filetypes_parse_error_message(GeanyFiletype *ft, const gchar *message,
|
|
gchar **filename, gint *line)
|
|
{
|
|
gchar *regstr;
|
|
gchar **tmp;
|
|
GeanyDocument *doc;
|
|
regex_t *regex;
|
|
regmatch_t pmatch[3];
|
|
|
|
if (ft == NULL)
|
|
{
|
|
doc = document_get_current();
|
|
if (doc != NULL)
|
|
ft = doc->file_type;
|
|
}
|
|
tmp = build_get_regex(build_info.grp, ft, NULL);
|
|
if (tmp == NULL)
|
|
return FALSE;
|
|
regstr = *tmp;
|
|
regex = &ft->priv->error_regex;
|
|
|
|
*filename = NULL;
|
|
*line = -1;
|
|
|
|
if (G_UNLIKELY(! NZV(regstr)))
|
|
return FALSE;
|
|
|
|
if (!ft->priv->error_regex_compiled || regstr != ft->priv->last_string)
|
|
{
|
|
compile_regex(ft, regex, regstr);
|
|
ft->priv->last_string = regstr;
|
|
}
|
|
if (!ft->priv->error_regex_compiled) /* regex error */
|
|
return FALSE;
|
|
|
|
if (regexec(regex, message, G_N_ELEMENTS(pmatch), pmatch, 0) != 0)
|
|
return FALSE;
|
|
|
|
if (pmatch[0].rm_so != -1 && pmatch[1].rm_so != -1 && pmatch[2].rm_so != -1)
|
|
{
|
|
gchar *first, *second, *end;
|
|
glong l;
|
|
|
|
first = get_regex_match_string(message, pmatch, 1);
|
|
second = get_regex_match_string(message, pmatch, 2);
|
|
l = strtol(first, &end, 10);
|
|
if (*end == '\0') /* first is purely decimals */
|
|
{
|
|
*line = l;
|
|
g_free(first);
|
|
*filename = second;
|
|
}
|
|
else
|
|
{
|
|
l = strtol(second, &end, 10);
|
|
if (*end == '\0')
|
|
{
|
|
*line = l;
|
|
g_free(second);
|
|
*filename = first;
|
|
}
|
|
else
|
|
{
|
|
g_free(first);
|
|
g_free(second);
|
|
}
|
|
}
|
|
}
|
|
return *filename != NULL;
|
|
}
|
|
|
|
|
|
#ifdef G_OS_WIN32
|
|
static void convert_filetype_extensions_to_lower_case(gchar **patterns, gsize len)
|
|
{
|
|
guint i;
|
|
for (i = 0; i < len; i++)
|
|
{
|
|
setptr(patterns[i], g_ascii_strdown(patterns[i], -1));
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
static void read_extensions(GKeyFile *sysconfig, GKeyFile *userconfig)
|
|
{
|
|
guint i;
|
|
gsize len = 0;
|
|
|
|
/* read the keys */
|
|
for (i = 0; i < filetypes_array->len; i++)
|
|
{
|
|
gboolean userset =
|
|
g_key_file_has_key(userconfig, "Extensions", filetypes[i]->name, NULL);
|
|
gchar **list = g_key_file_get_string_list(
|
|
(userset) ? userconfig : sysconfig, "Extensions", filetypes[i]->name, &len, NULL);
|
|
|
|
g_strfreev(filetypes[i]->pattern);
|
|
/* Note: we allow 'Foo=' to remove all patterns */
|
|
if (!list)
|
|
list = g_new0(gchar*, 1);
|
|
filetypes[i]->pattern = list;
|
|
|
|
#ifdef G_OS_WIN32
|
|
convert_filetype_extensions_to_lower_case(filetypes[i]->pattern, len);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
|
|
static void read_group(GKeyFile *config, const gchar *group_name, gint group_id)
|
|
{
|
|
gchar **names = g_key_file_get_string_list(config, "Groups", group_name, NULL, NULL);
|
|
gchar **name;
|
|
|
|
foreach_strv(name, names)
|
|
{
|
|
GeanyFiletype *ft = filetypes_lookup_by_name(*name);
|
|
|
|
if (ft)
|
|
{
|
|
ft->group = group_id;
|
|
if (ft->priv->custom &&
|
|
(group_id == GEANY_FILETYPE_GROUP_COMPILED || group_id == GEANY_FILETYPE_GROUP_SCRIPT))
|
|
{
|
|
setptr(ft->title, NULL);
|
|
filetype_make_title(ft, TITLE_SOURCE_FILE);
|
|
}
|
|
}
|
|
else
|
|
geany_debug("Filetype '%s' not found for group '%s'!", *name, group_name);
|
|
}
|
|
}
|
|
|
|
|
|
static void read_groups(GKeyFile *config)
|
|
{
|
|
read_group(config, "Programming", GEANY_FILETYPE_GROUP_COMPILED);
|
|
read_group(config, "Script", GEANY_FILETYPE_GROUP_SCRIPT);
|
|
read_group(config, "Markup", GEANY_FILETYPE_GROUP_MARKUP);
|
|
read_group(config, "Misc", GEANY_FILETYPE_GROUP_MISC);
|
|
read_group(config, "None", GEANY_FILETYPE_GROUP_NONE);
|
|
}
|
|
|
|
|
|
static void read_filetype_config(void)
|
|
{
|
|
gchar *sysconfigfile = g_strconcat(app->datadir, G_DIR_SEPARATOR_S,
|
|
"filetype_extensions.conf", NULL);
|
|
gchar *userconfigfile = g_strconcat(app->configdir, G_DIR_SEPARATOR_S,
|
|
"filetype_extensions.conf", NULL);
|
|
GKeyFile *sysconfig = g_key_file_new();
|
|
GKeyFile *userconfig = g_key_file_new();
|
|
|
|
g_key_file_load_from_file(sysconfig, sysconfigfile, G_KEY_FILE_NONE, NULL);
|
|
g_key_file_load_from_file(userconfig, userconfigfile, G_KEY_FILE_NONE, NULL);
|
|
|
|
read_extensions(sysconfig, userconfig);
|
|
read_groups(sysconfig);
|
|
read_groups(userconfig);
|
|
|
|
g_free(sysconfigfile);
|
|
g_free(userconfigfile);
|
|
g_key_file_free(sysconfig);
|
|
g_key_file_free(userconfig);
|
|
}
|
|
|
|
|
|
void filetypes_reload_extensions(void)
|
|
{
|
|
guint i;
|
|
|
|
read_filetype_config();
|
|
|
|
/* Redetect filetype of any documents with none set */
|
|
foreach_document(i)
|
|
{
|
|
GeanyDocument *doc = documents[i];
|
|
if (doc->file_type->id != GEANY_FILETYPES_NONE)
|
|
continue;
|
|
document_set_filetype(doc, filetypes_detect_from_document(doc));
|
|
}
|
|
}
|
|
|
|
|
|
/** Accessor function for @ref GeanyData::filetypes_array items.
|
|
* Example: @code ft = filetypes_index(GEANY_FILETYPES_C); @endcode
|
|
* @param idx @c filetypes_array index.
|
|
* @return The filetype, or @c NULL if @a idx is out of range.
|
|
*
|
|
* @since 0.16
|
|
*/
|
|
GeanyFiletype *filetypes_index(gint idx)
|
|
{
|
|
return (idx >= 0 && idx < (gint) filetypes_array->len) ? filetypes[idx] : NULL;
|
|
}
|
|
|
|
|
|
void filetypes_reload(void)
|
|
{
|
|
guint i;
|
|
GeanyDocument *current_doc;
|
|
|
|
/* reload filetype configs */
|
|
for (i = 0; i < filetypes_array->len; i++)
|
|
{
|
|
/* filetypes_load_config() will skip not loaded filetypes */
|
|
filetypes_load_config(i, TRUE);
|
|
}
|
|
|
|
current_doc = document_get_current();
|
|
if (!current_doc)
|
|
return;
|
|
|
|
/* update document styling */
|
|
foreach_document(i)
|
|
{
|
|
if (current_doc != documents[i])
|
|
document_reload_config(documents[i]);
|
|
}
|
|
/* process the current document at last */
|
|
document_reload_config(current_doc);
|
|
}
|
|
|
|
|
|
/** Gets @c ft->name or a translation for filetype None.
|
|
* @param ft .
|
|
* @return .
|
|
* @since Geany 0.20 */
|
|
const gchar *filetypes_get_display_name(GeanyFiletype *ft)
|
|
{
|
|
return ft->id == GEANY_FILETYPES_NONE ? _("None") : ft->name;
|
|
}
|
|
|