419 lines
13 KiB
C
419 lines
13 KiB
C
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4; coding: utf-8 -*-
|
|
*
|
|
* mooeditlangmgr.c
|
|
*
|
|
* Copyright (C) 2004-2005 by Yevgen Muntyan <muntyan@math.tamu.edu>
|
|
*
|
|
* 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.
|
|
*
|
|
* See COPYING file that comes with this distribution.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include "mooedit/mooeditlangmgr.h"
|
|
#include "mooutils/moocompat.h"
|
|
#include <string.h>
|
|
#include "xdgmime/xdgmime.h"
|
|
|
|
|
|
#define LANG_FILE_SUFFIX ".lang"
|
|
#define LANG_FILE_SUFFIX_LEN 5
|
|
|
|
|
|
struct _MooEditLangMgrPrivate {
|
|
GSList *lang_dirs;
|
|
GSList *langs;
|
|
MooEditLang *default_lang;
|
|
};
|
|
|
|
|
|
static void moo_edit_lang_mgr_class_init (MooEditLangMgrClass *klass);
|
|
static void moo_edit_lang_mgr_init (MooEditLangMgr *mgr);
|
|
static void moo_edit_lang_mgr_finalize (GObject *object);
|
|
|
|
static gboolean moo_edit_lang_mgr_read_lang_file(MooEditLangMgr *mgr,
|
|
const char *filename);
|
|
|
|
|
|
/* MOO_TYPE_EDIT_LANG_MGR */
|
|
G_DEFINE_TYPE (MooEditLangMgr, moo_edit_lang_mgr, G_TYPE_OBJECT)
|
|
|
|
|
|
static void moo_edit_lang_mgr_class_init (MooEditLangMgrClass *klass)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
gobject_class->finalize = moo_edit_lang_mgr_finalize;
|
|
}
|
|
|
|
|
|
static void moo_edit_lang_mgr_init (MooEditLangMgr *mgr)
|
|
{
|
|
mgr->priv = g_new0 (MooEditLangMgrPrivate, 1);
|
|
}
|
|
|
|
|
|
static void moo_edit_lang_mgr_finalize (GObject *object)
|
|
{
|
|
MooEditLangMgr *mgr = MOO_EDIT_LANG_MGR (object);
|
|
|
|
g_slist_foreach (mgr->priv->lang_dirs, (GFunc) g_free, NULL);
|
|
g_slist_foreach (mgr->priv->langs, (GFunc) g_object_unref, NULL);
|
|
g_slist_free (mgr->priv->lang_dirs);
|
|
g_slist_free (mgr->priv->langs);
|
|
g_free (mgr->priv);
|
|
|
|
G_OBJECT_CLASS (moo_edit_lang_mgr_parent_class)->finalize (object);
|
|
}
|
|
|
|
|
|
const GSList *moo_edit_lang_mgr_get_available_languages (MooEditLangMgr *mgr)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_EDIT_LANG_MGR (mgr), NULL);
|
|
return mgr->priv->langs;
|
|
}
|
|
|
|
|
|
void moo_edit_lang_mgr_add_language (MooEditLangMgr *mgr,
|
|
MooEditLang *lang)
|
|
{
|
|
MooEditLang *old = NULL;
|
|
const char *id;
|
|
|
|
g_return_if_fail (MOO_IS_EDIT_LANG_MGR (mgr) && MOO_IS_EDIT_LANG (lang));
|
|
g_return_if_fail (!g_slist_find (mgr->priv->langs, lang));
|
|
|
|
id = moo_edit_lang_get_id (lang);
|
|
g_assert (id != NULL);
|
|
old = moo_edit_lang_mgr_get_language_by_id (mgr, id);
|
|
|
|
if (old)
|
|
{
|
|
#ifdef DEBUG_LANGS
|
|
g_message ("%s: loading another instance of language '%s'", G_STRLOC, id);
|
|
#endif
|
|
g_slist_remove (mgr->priv->langs, old);
|
|
g_object_unref (G_OBJECT (old));
|
|
}
|
|
|
|
mgr->priv->langs = g_slist_prepend (mgr->priv->langs, lang);
|
|
g_object_ref (G_OBJECT (lang));
|
|
|
|
if (old && old == mgr->priv->default_lang)
|
|
mgr->priv->default_lang = lang;
|
|
}
|
|
|
|
|
|
MooEditLang *moo_edit_lang_mgr_get_language_by_id (MooEditLangMgr *mgr,
|
|
const char *lang_id)
|
|
{
|
|
GSList *l;
|
|
|
|
g_return_val_if_fail (MOO_IS_EDIT_LANG_MGR (mgr) && lang_id != NULL, NULL);
|
|
|
|
for (l = mgr->priv->langs; l != NULL; l = l->next)
|
|
if (!strcmp (moo_edit_lang_get_id (MOO_EDIT_LANG(l->data)), lang_id))
|
|
return MOO_EDIT_LANG (l->data);
|
|
return NULL;
|
|
}
|
|
|
|
|
|
MooEditLang *moo_edit_lang_mgr_get_language_for_file (MooEditLangMgr *mgr,
|
|
const char *filename)
|
|
{
|
|
MooEditLang *lang = NULL;
|
|
char *utf8_filename;
|
|
GError *err = NULL;
|
|
|
|
g_return_val_if_fail (MOO_IS_EDIT_LANG_MGR (mgr) && filename != NULL, NULL);
|
|
|
|
utf8_filename = g_filename_to_utf8 (filename, -1, NULL, NULL, &err);
|
|
if (!utf8_filename) {
|
|
g_critical ("%s: could not convert filename to UTF8", G_STRLOC);
|
|
if (err) {
|
|
g_critical ("%s: %s", G_STRLOC, err->message);
|
|
g_error_free (err);
|
|
}
|
|
}
|
|
|
|
if (utf8_filename) {
|
|
GSList *l;
|
|
gboolean found = FALSE;
|
|
|
|
for (l = mgr->priv->langs; l != NULL && !found; l = l->next) {
|
|
GSList *globs, *g;
|
|
lang = MOO_EDIT_LANG (l->data);
|
|
globs = moo_edit_lang_get_extensions (lang);
|
|
for (g = globs; g != NULL && !found; g = g->next) {
|
|
if (g_pattern_match_simple ((char*) g->data, utf8_filename))
|
|
found = TRUE;
|
|
}
|
|
g_slist_foreach (globs, (GFunc) g_free, NULL);
|
|
g_slist_free (globs);
|
|
}
|
|
|
|
if (!found)
|
|
lang = NULL;
|
|
|
|
#ifdef DEBUG_LANGS
|
|
if (found)
|
|
g_message ("%s: found lang '%s' for file '%s' by extension",
|
|
G_STRLOC, moo_edit_lang_get_id (lang), filename);
|
|
#endif
|
|
}
|
|
|
|
#ifdef MOO_USE_XDGMIME
|
|
/* TODO: xdgmime wants utf8-encoded filename here. is it a problem? */
|
|
if (!lang)
|
|
{
|
|
const char *mime_type = xdg_mime_get_mime_type_for_file (filename);
|
|
if (!xdg_mime_mime_type_equal (mime_type, XDG_MIME_TYPE_UNKNOWN))
|
|
lang = moo_edit_lang_mgr_get_language_for_mime_type (mgr, mime_type);
|
|
#ifdef DEBUG_LANGS
|
|
if (lang)
|
|
g_message ("%s: found lang '%s' for file '%s' by mime type",
|
|
G_STRLOC, moo_edit_lang_get_id (lang), filename);
|
|
#endif
|
|
}
|
|
#endif /* MOO_USE_XDGMIME */
|
|
|
|
/* check if it's backup file */
|
|
if (!lang && utf8_filename)
|
|
{
|
|
char *base = NULL;
|
|
int len = strlen (utf8_filename);
|
|
guint i;
|
|
|
|
static const char *bak_globs[] = {"*~", "*.bak", "*.in"};
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (bak_globs); ++i)
|
|
{
|
|
int ext_len = strlen (bak_globs[i]) - 1;
|
|
if (len > ext_len && g_pattern_match_simple (bak_globs[i], utf8_filename))
|
|
{
|
|
base = g_strndup (utf8_filename, len - ext_len);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (base)
|
|
{
|
|
lang = moo_edit_lang_mgr_get_language_for_filename (mgr, base);
|
|
g_free (base);
|
|
}
|
|
}
|
|
|
|
#ifdef DEBUG_LANGS
|
|
if (!lang)
|
|
g_message ("%s: could not find lang for file '%s'", G_STRLOC, filename);
|
|
#endif
|
|
|
|
return lang;
|
|
}
|
|
|
|
|
|
MooEditLang *moo_edit_lang_mgr_get_language_for_mime_type (MooEditLangMgr *mgr,
|
|
const char *mime_type)
|
|
{
|
|
GSList *l;
|
|
MooEditLang *lang = NULL;
|
|
gboolean found = FALSE;
|
|
|
|
g_return_val_if_fail (MOO_IS_EDIT_LANG_MGR (mgr) && mime_type != NULL, NULL);
|
|
|
|
for (l = mgr->priv->langs; l != NULL && !found; l = l->next) {
|
|
GSList *mime_types;
|
|
|
|
lang = MOO_EDIT_LANG (l->data);
|
|
mime_types = moo_edit_lang_get_mime_types (lang);
|
|
if (g_slist_find_custom (mime_types, mime_type, (GCompareFunc) strcmp))
|
|
found = TRUE;
|
|
g_slist_foreach (mime_types, (GFunc) g_free, NULL);
|
|
g_slist_free (mime_types);
|
|
}
|
|
|
|
if (found)
|
|
return lang;
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
|
|
MooEditLang *moo_edit_lang_mgr_get_language_for_filename (MooEditLangMgr *mgr,
|
|
const char *filename)
|
|
{
|
|
MooEditLang *lang = NULL;
|
|
char *utf8_filename;
|
|
GError *err = NULL;
|
|
|
|
g_return_val_if_fail (MOO_IS_EDIT_LANG_MGR (mgr) && filename != NULL, NULL);
|
|
|
|
utf8_filename = g_filename_to_utf8 (filename, -1, NULL, NULL, &err);
|
|
if (!utf8_filename) {
|
|
g_critical ("%s: could not convert filename to UTF8", G_STRLOC);
|
|
if (err) {
|
|
g_critical ("%s: %s", G_STRLOC, err->message);
|
|
g_error_free (err);
|
|
}
|
|
}
|
|
|
|
if (utf8_filename) {
|
|
GSList *l;
|
|
gboolean found = FALSE;
|
|
|
|
for (l = mgr->priv->langs; l != NULL && !found; l = l->next) {
|
|
GSList *globs, *g;
|
|
lang = MOO_EDIT_LANG (l->data);
|
|
globs = moo_edit_lang_get_extensions (lang);
|
|
for (g = globs; g != NULL && !found; g = g->next) {
|
|
if (g_pattern_match_simple ((char*) g->data, utf8_filename))
|
|
found = TRUE;
|
|
}
|
|
g_slist_foreach (globs, (GFunc) g_free, NULL);
|
|
g_slist_free (globs);
|
|
}
|
|
|
|
if (!found)
|
|
lang = NULL;
|
|
|
|
#ifdef DEBUG_LANGS
|
|
if (found)
|
|
g_message ("%s: found lang '%s' for file '%s' by extension",
|
|
G_STRLOC, moo_edit_lang_get_id (lang), filename);
|
|
#endif
|
|
}
|
|
|
|
#ifdef MOO_USE_XDGMIME
|
|
/* TODO: xdgmime wants utf8-encoded filename here. is it a problem? */
|
|
if (!lang)
|
|
{
|
|
const char *mime_type = xdg_mime_get_mime_type_from_file_name (filename);
|
|
if (!xdg_mime_mime_type_equal (mime_type, XDG_MIME_TYPE_UNKNOWN))
|
|
lang = moo_edit_lang_mgr_get_language_for_mime_type (mgr, mime_type);
|
|
#ifdef DEBUG_LANGS
|
|
if (lang)
|
|
g_message ("%s: found lang '%s' for file '%s' by mime type",
|
|
G_STRLOC, moo_edit_lang_get_id (lang), filename);
|
|
#endif
|
|
}
|
|
#endif /* MOO_USE_XDGMIME */
|
|
|
|
if (!lang)
|
|
g_message ("%s: could not find lang for file '%s'", G_STRLOC, filename);
|
|
|
|
return lang;
|
|
}
|
|
|
|
|
|
const GSList *moo_edit_lang_mgr_get_lang_files_dirs (MooEditLangMgr *mgr)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_EDIT_LANG_MGR (mgr), NULL);
|
|
return mgr->priv->lang_dirs;
|
|
}
|
|
|
|
|
|
void moo_edit_lang_mgr_add_lang_files_dir (MooEditLangMgr *mgr,
|
|
const char *dirname)
|
|
{
|
|
GError *err = NULL;
|
|
GDir *dir;
|
|
const char *entry;
|
|
|
|
#ifndef MOO_USE_XML
|
|
g_warning ("%s: xml support disabled, can't load lang files", G_STRLOC);
|
|
return;
|
|
#endif /* MOO_USE_XML */
|
|
|
|
g_return_if_fail (MOO_IS_EDIT_LANG_MGR (mgr) && dirname != NULL);
|
|
g_return_if_fail (!g_slist_find_custom (mgr->priv->lang_dirs, dirname,
|
|
(GCompareFunc)strcmp));
|
|
|
|
dir = g_dir_open (dirname, 0, &err);
|
|
if (!dir) {
|
|
g_critical ("%s: could not open directory '%s'", G_STRLOC, dirname);
|
|
if (err) {
|
|
g_critical ("%s: %s", G_STRLOC, err->message);
|
|
g_error_free (err);
|
|
}
|
|
return;
|
|
}
|
|
|
|
for (entry = g_dir_read_name (dir); entry != NULL; entry = g_dir_read_name (dir))
|
|
{
|
|
char *file = g_build_filename (dirname, entry, NULL);
|
|
|
|
if (file && g_file_test (file, G_FILE_TEST_IS_REGULAR | G_FILE_TEST_IS_SYMLINK))
|
|
{
|
|
guint len = strlen (file);
|
|
if (len >= LANG_FILE_SUFFIX_LEN &&
|
|
!strcmp (LANG_FILE_SUFFIX, file + (len - LANG_FILE_SUFFIX_LEN)))
|
|
moo_edit_lang_mgr_read_lang_file (mgr, file);
|
|
}
|
|
|
|
g_free (file);
|
|
}
|
|
|
|
g_dir_close (dir);
|
|
g_slist_prepend (mgr->priv->lang_dirs, g_strdup (dirname));
|
|
}
|
|
|
|
|
|
static gboolean moo_edit_lang_mgr_read_lang_file(MooEditLangMgr *mgr,
|
|
const char *filename)
|
|
{
|
|
MooEditLang *lang = moo_edit_lang_new_from_file (filename);
|
|
g_return_val_if_fail (lang != NULL, FALSE);
|
|
moo_edit_lang_mgr_add_language (mgr, lang);
|
|
g_object_unref (G_OBJECT (lang));
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
MooEditLangMgr *moo_edit_lang_mgr_new (void)
|
|
{
|
|
return MOO_EDIT_LANG_MGR (g_object_new (MOO_TYPE_EDIT_LANG_MGR, NULL));
|
|
}
|
|
|
|
|
|
MooEditLang *moo_edit_lang_mgr_get_default_language (MooEditLangMgr *mgr)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_EDIT_LANG_MGR (mgr), NULL);
|
|
return mgr->priv->default_lang;
|
|
}
|
|
|
|
|
|
void moo_edit_lang_mgr_set_default_language (MooEditLangMgr *mgr,
|
|
MooEditLang *lang)
|
|
{
|
|
g_return_if_fail (MOO_IS_EDIT_LANG_MGR (mgr));
|
|
g_return_if_fail (lang == NULL ||
|
|
g_slist_find (mgr->priv->langs, lang) != NULL);
|
|
mgr->priv->default_lang = lang;
|
|
}
|
|
|
|
|
|
GSList *moo_edit_lang_mgr_get_sections (MooEditLangMgr *mgr)
|
|
{
|
|
GSList *list = NULL;
|
|
GSList *l;
|
|
|
|
g_return_val_if_fail (MOO_IS_EDIT_LANG_MGR (mgr), NULL);
|
|
|
|
for (l = mgr->priv->langs; l != NULL; l = l->next)
|
|
{
|
|
const char *section =
|
|
moo_edit_lang_get_section (MOO_EDIT_LANG (l->data));
|
|
if (!section)
|
|
section = "Others";
|
|
if (!g_slist_find_custom (list, section, (GCompareFunc) strcmp))
|
|
list = g_slist_prepend (list, g_strdup (section));
|
|
}
|
|
|
|
return list;
|
|
}
|