Blame gtksourceview/gtksourcelanguagemanager.c

Packit a7d494
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8; coding: utf-8 -*- */
Packit a7d494
/* gtksourcelanguagemanager.c
Packit a7d494
 * This file is part of GtkSourceView
Packit a7d494
 *
Packit a7d494
 * Copyright (C) 2003-2007 - Paolo Maggi <paolo.maggi@polito.it>
Packit a7d494
 *
Packit a7d494
 * GtkSourceView is free software; you can redistribute it and/or
Packit a7d494
 * modify it under the terms of the GNU Lesser General Public
Packit a7d494
 * License as published by the Free Software Foundation; either
Packit a7d494
 * version 2.1 of the License, or (at your option) any later version.
Packit a7d494
 *
Packit a7d494
 * GtkSourceView is distributed in the hope that it will be useful,
Packit a7d494
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit a7d494
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit a7d494
 * Lesser General Public License for more details.
Packit a7d494
 *
Packit a7d494
 * You should have received a copy of the GNU Lesser General Public
Packit a7d494
 * License along with this library; if not, write to the Free Software
Packit a7d494
 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
Packit a7d494
 */
Packit a7d494
Packit a7d494
#ifdef HAVE_CONFIG_H
Packit a7d494
#include <config.h>
Packit a7d494
#endif
Packit a7d494
Packit a7d494
#include "gtksourcelanguagemanager.h"
Packit a7d494
Packit a7d494
#include <string.h>
Packit a7d494
#include <gio/gio.h>
Packit a7d494
Packit a7d494
#include "gtksourcelanguage.h"
Packit a7d494
#include "gtksourcelanguage-private.h"
Packit a7d494
#include "gtksourceview-utils.h"
Packit a7d494
#include "gtksourceview-i18n.h"
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * SECTION:languagemanager
Packit a7d494
 * @Short_description: Provides access to GtkSourceLanguages
Packit a7d494
 * @Title: GtkSourceLanguageManager
Packit a7d494
 * @See_also: #GtkSourceLanguage
Packit a7d494
 *
Packit a7d494
 * #GtkSourceLanguageManager is an object which processes language description
Packit a7d494
 * files and creates and stores #GtkSourceLanguage objects, and provides API to
Packit a7d494
 * access them.
Packit a7d494
 * Use gtk_source_language_manager_get_default() to retrieve the default
Packit a7d494
 * instance of #GtkSourceLanguageManager, and
Packit a7d494
 * gtk_source_language_manager_guess_language() to get a #GtkSourceLanguage for
Packit a7d494
 * given file name and content type.
Packit a7d494
 */
Packit a7d494
Packit a7d494
#define RNG_SCHEMA_FILE		"language2.rng"
Packit a7d494
#define LANGUAGE_DIR		"language-specs"
Packit a7d494
#define LANG_FILE_SUFFIX	".lang"
Packit a7d494
Packit a7d494
enum {
Packit a7d494
	PROP_0,
Packit a7d494
	PROP_SEARCH_PATH,
Packit a7d494
	PROP_LANGUAGE_IDS
Packit a7d494
};
Packit a7d494
Packit a7d494
struct _GtkSourceLanguageManagerPrivate
Packit a7d494
{
Packit a7d494
	GHashTable	*language_ids;
Packit a7d494
Packit a7d494
	gchar	       **lang_dirs;
Packit a7d494
	gchar		*rng_file;
Packit a7d494
Packit a7d494
	gchar          **ids; /* Cache the IDs of the available languages */
Packit a7d494
};
Packit a7d494
Packit a7d494
static GtkSourceLanguageManager *default_instance;
Packit a7d494
Packit a7d494
G_DEFINE_TYPE_WITH_PRIVATE (GtkSourceLanguageManager, gtk_source_language_manager, G_TYPE_OBJECT)
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_language_manager_set_property (GObject 	*object,
Packit a7d494
					  guint 	 prop_id,
Packit a7d494
					  const GValue *value,
Packit a7d494
					  GParamSpec	*pspec)
Packit a7d494
{
Packit a7d494
	GtkSourceLanguageManager *lm;
Packit a7d494
Packit a7d494
	lm = GTK_SOURCE_LANGUAGE_MANAGER (object);
Packit a7d494
Packit a7d494
	switch (prop_id)
Packit a7d494
	{
Packit a7d494
		case PROP_SEARCH_PATH:
Packit a7d494
			gtk_source_language_manager_set_search_path (lm, g_value_get_boxed (value));
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		default:
Packit a7d494
			G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit a7d494
			break;
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_language_manager_get_property (GObject 	*object,
Packit a7d494
					  guint 	 prop_id,
Packit a7d494
					  GValue 	*value,
Packit a7d494
					  GParamSpec	*pspec)
Packit a7d494
{
Packit a7d494
	GtkSourceLanguageManager *lm;
Packit a7d494
Packit a7d494
	lm = GTK_SOURCE_LANGUAGE_MANAGER (object);
Packit a7d494
Packit a7d494
	switch (prop_id)
Packit a7d494
	{
Packit a7d494
		case PROP_SEARCH_PATH:
Packit a7d494
			g_value_set_boxed (value, gtk_source_language_manager_get_search_path (lm));
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_LANGUAGE_IDS:
Packit a7d494
			g_value_set_boxed (value, gtk_source_language_manager_get_language_ids (lm));
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		default:
Packit a7d494
			G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit a7d494
			break;
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_language_manager_finalize (GObject *object)
Packit a7d494
{
Packit a7d494
	GtkSourceLanguageManager *lm;
Packit a7d494
Packit a7d494
	lm = GTK_SOURCE_LANGUAGE_MANAGER (object);
Packit a7d494
Packit a7d494
	if (lm->priv->language_ids)
Packit a7d494
		g_hash_table_destroy (lm->priv->language_ids);
Packit a7d494
Packit a7d494
	g_strfreev (lm->priv->ids);
Packit a7d494
Packit a7d494
	g_strfreev (lm->priv->lang_dirs);
Packit a7d494
	g_free (lm->priv->rng_file);
Packit a7d494
Packit a7d494
	G_OBJECT_CLASS (gtk_source_language_manager_parent_class)->finalize (object);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_language_manager_class_init (GtkSourceLanguageManagerClass *klass)
Packit a7d494
{
Packit a7d494
	GObjectClass *object_class = G_OBJECT_CLASS (klass);
Packit a7d494
Packit a7d494
	object_class->finalize	= gtk_source_language_manager_finalize;
Packit a7d494
Packit a7d494
	object_class->set_property = gtk_source_language_manager_set_property;
Packit a7d494
	object_class->get_property = gtk_source_language_manager_get_property;
Packit a7d494
Packit a7d494
	g_object_class_install_property (object_class,
Packit a7d494
					 PROP_SEARCH_PATH,
Packit a7d494
					 g_param_spec_boxed ("search-path",
Packit a7d494
						 	     "Language specification directories",
Packit a7d494
							     "List of directories where the "
Packit a7d494
							     "language specification files (.lang) "
Packit a7d494
							     "are located",
Packit a7d494
							     G_TYPE_STRV,
Packit a7d494
							     G_PARAM_READWRITE |
Packit a7d494
							     G_PARAM_STATIC_STRINGS));
Packit a7d494
Packit a7d494
	g_object_class_install_property (object_class,
Packit a7d494
					 PROP_LANGUAGE_IDS,
Packit a7d494
					 g_param_spec_boxed ("language-ids",
Packit a7d494
						 	     "Language ids",
Packit a7d494
							     "List of the ids of the available languages",
Packit a7d494
							     G_TYPE_STRV,
Packit a7d494
							     G_PARAM_READABLE |
Packit a7d494
							     G_PARAM_STATIC_STRINGS));
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_language_manager_init (GtkSourceLanguageManager *lm)
Packit a7d494
{
Packit a7d494
	lm->priv = gtk_source_language_manager_get_instance_private (lm);
Packit a7d494
	lm->priv->language_ids = NULL;
Packit a7d494
	lm->priv->ids = NULL;
Packit a7d494
	lm->priv->lang_dirs = NULL;
Packit a7d494
	lm->priv->rng_file = NULL;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_language_manager_new:
Packit a7d494
 *
Packit a7d494
 * Creates a new language manager. If you do not need more than one language
Packit a7d494
 * manager or a private language manager instance then use
Packit a7d494
 * gtk_source_language_manager_get_default() instead.
Packit a7d494
 *
Packit a7d494
 * Returns: a new #GtkSourceLanguageManager.
Packit a7d494
 */
Packit a7d494
GtkSourceLanguageManager *
Packit a7d494
gtk_source_language_manager_new (void)
Packit a7d494
{
Packit a7d494
	return g_object_new (GTK_SOURCE_TYPE_LANGUAGE_MANAGER, NULL);
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_language_manager_get_default:
Packit a7d494
 *
Packit a7d494
 * Returns the default #GtkSourceLanguageManager instance.
Packit a7d494
 *
Packit a7d494
 * Returns: (transfer none): a #GtkSourceLanguageManager.
Packit a7d494
 * Return value is owned by GtkSourceView library and must not be unref'ed.
Packit a7d494
 */
Packit a7d494
GtkSourceLanguageManager *
Packit a7d494
gtk_source_language_manager_get_default (void)
Packit a7d494
{
Packit a7d494
	if (default_instance == NULL)
Packit a7d494
	{
Packit a7d494
		default_instance = gtk_source_language_manager_new ();
Packit a7d494
		g_object_add_weak_pointer (G_OBJECT (default_instance),
Packit a7d494
					   (gpointer) &default_instance);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	return default_instance;
Packit a7d494
}
Packit a7d494
Packit a7d494
GtkSourceLanguageManager *
Packit a7d494
_gtk_source_language_manager_peek_default (void)
Packit a7d494
{
Packit a7d494
	return default_instance;
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
notify_search_path (GtkSourceLanguageManager *mgr)
Packit a7d494
{
Packit a7d494
	g_object_notify (G_OBJECT (mgr), "search-path");
Packit a7d494
	g_object_notify (G_OBJECT (mgr), "language-ids");
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_language_manager_set_search_path:
Packit a7d494
 * @lm: a #GtkSourceLanguageManager.
Packit a7d494
 * @dirs: (nullable) (array zero-terminated=1):
Packit a7d494
 * a %NULL-terminated array of strings or %NULL.
Packit a7d494
 *
Packit a7d494
 * Sets the list of directories where the @lm looks for
Packit a7d494
 * language files.
Packit a7d494
 * If @dirs is %NULL, the search path is reset to default.
Packit a7d494
 *
Packit a7d494
 * <note>
Packit a7d494
 *   <para>
Packit a7d494
 *     At the moment this function can be called only before the
Packit a7d494
 *     language files are loaded for the first time. In practice
Packit a7d494
 *     to set a custom search path for a #GtkSourceLanguageManager,
Packit a7d494
 *     you have to call this function right after creating it.
Packit a7d494
 *   </para>
Packit a7d494
 * </note>
Packit a7d494
 */
Packit a7d494
void
Packit a7d494
gtk_source_language_manager_set_search_path (GtkSourceLanguageManager *lm,
Packit a7d494
					     gchar                   **dirs)
Packit a7d494
{
Packit a7d494
	gchar **tmp;
Packit a7d494
Packit a7d494
	g_return_if_fail (GTK_SOURCE_IS_LANGUAGE_MANAGER (lm));
Packit a7d494
Packit a7d494
	/* Search path cannot be changed in the list of available languages
Packit a7d494
	 * as been already computed */
Packit a7d494
	g_return_if_fail (lm->priv->ids == NULL);
Packit a7d494
Packit a7d494
	tmp = lm->priv->lang_dirs;
Packit a7d494
Packit a7d494
	if (dirs == NULL)
Packit a7d494
		lm->priv->lang_dirs = _gtk_source_view_get_default_dirs (LANGUAGE_DIR, TRUE);
Packit a7d494
	else
Packit a7d494
		lm->priv->lang_dirs = g_strdupv (dirs);
Packit a7d494
Packit a7d494
	g_strfreev (tmp);
Packit a7d494
Packit a7d494
	notify_search_path (lm);
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_language_manager_get_search_path:
Packit a7d494
 * @lm: a #GtkSourceLanguageManager.
Packit a7d494
 *
Packit a7d494
 * Gets the list directories where @lm looks for language files.
Packit a7d494
 *
Packit a7d494
 * Returns: (array zero-terminated=1) (transfer none): %NULL-terminated array
Packit a7d494
 * containg a list of language files directories.
Packit a7d494
 * The array is owned by @lm and must not be modified.
Packit a7d494
 */
Packit a7d494
const gchar * const *
Packit a7d494
gtk_source_language_manager_get_search_path (GtkSourceLanguageManager *lm)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_LANGUAGE_MANAGER (lm), NULL);
Packit a7d494
Packit a7d494
	if (lm->priv->lang_dirs == NULL)
Packit a7d494
		lm->priv->lang_dirs = _gtk_source_view_get_default_dirs (LANGUAGE_DIR, TRUE);
Packit a7d494
Packit a7d494
	return (const gchar * const *)lm->priv->lang_dirs;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * _gtk_source_language_manager_get_rng_file:
Packit a7d494
 * @lm: a #GtkSourceLanguageManager.
Packit a7d494
 *
Packit a7d494
 * Returns location of the RNG schema file for lang files version 2.
Packit a7d494
 *
Packit a7d494
 * Returns: path to RNG file. It belongs to %lm and must not be freed or modified.
Packit a7d494
 */
Packit a7d494
const char *
Packit a7d494
_gtk_source_language_manager_get_rng_file (GtkSourceLanguageManager *lm)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_LANGUAGE_MANAGER (lm), NULL);
Packit a7d494
Packit a7d494
	if (lm->priv->rng_file == NULL)
Packit a7d494
	{
Packit a7d494
		const gchar * const *dirs;
Packit a7d494
Packit a7d494
		for (dirs = gtk_source_language_manager_get_search_path (lm);
Packit a7d494
		     dirs != NULL && *dirs != NULL;
Packit a7d494
		     ++dirs)
Packit a7d494
		{
Packit a7d494
			gchar *file;
Packit a7d494
Packit a7d494
			file = g_build_filename (*dirs, RNG_SCHEMA_FILE, NULL);
Packit a7d494
			if (g_file_test (file, G_FILE_TEST_EXISTS))
Packit a7d494
			{
Packit a7d494
				lm->priv->rng_file = file;
Packit a7d494
				break;
Packit a7d494
			}
Packit a7d494
Packit a7d494
			g_free (file);
Packit a7d494
		}
Packit a7d494
	}
Packit a7d494
Packit a7d494
	return lm->priv->rng_file;
Packit a7d494
}
Packit a7d494
Packit a7d494
static gint
Packit a7d494
language_compare (const gchar **id1, const gchar **id2, GHashTable *language_ids)
Packit a7d494
{
Packit a7d494
	GtkSourceLanguage *lang1, *lang2;
Packit a7d494
	const gchar *name1, *name2;
Packit a7d494
Packit a7d494
	lang1 = g_hash_table_lookup (language_ids, *id1);
Packit a7d494
	lang2 = g_hash_table_lookup (language_ids, *id2);
Packit a7d494
Packit a7d494
	name1 = gtk_source_language_get_name (lang1);
Packit a7d494
	name2 = gtk_source_language_get_name (lang2);
Packit a7d494
Packit a7d494
	return g_utf8_collate (name1, name2);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
ensure_languages (GtkSourceLanguageManager *lm)
Packit a7d494
{
Packit a7d494
	GSList *filenames, *l;
Packit a7d494
	GPtrArray *ids_array = NULL;
Packit a7d494
Packit a7d494
	if (lm->priv->language_ids != NULL)
Packit a7d494
		return;
Packit a7d494
Packit a7d494
	lm->priv->language_ids = g_hash_table_new_full (g_str_hash, g_str_equal,
Packit a7d494
							g_free, g_object_unref);
Packit a7d494
Packit a7d494
	filenames = _gtk_source_view_get_file_list ((gchar **)gtk_source_language_manager_get_search_path (lm),
Packit a7d494
						    LANG_FILE_SUFFIX,
Packit a7d494
						    TRUE);
Packit a7d494
Packit a7d494
	for (l = filenames; l != NULL; l = l->next)
Packit a7d494
	{
Packit a7d494
		GtkSourceLanguage *lang;
Packit a7d494
		gchar *filename;
Packit a7d494
Packit a7d494
		filename = l->data;
Packit a7d494
Packit a7d494
		lang = _gtk_source_language_new_from_file (filename, lm);
Packit a7d494
Packit a7d494
		if (lang == NULL)
Packit a7d494
		{
Packit a7d494
			g_warning ("Error reading language specification file '%s'", filename);
Packit a7d494
			continue;
Packit a7d494
		}
Packit a7d494
Packit a7d494
		if (g_hash_table_lookup (lm->priv->language_ids, lang->priv->id) == NULL)
Packit a7d494
		{
Packit a7d494
			g_hash_table_insert (lm->priv->language_ids,
Packit a7d494
					     g_strdup (lang->priv->id),
Packit a7d494
					     lang);
Packit a7d494
Packit a7d494
			if (ids_array == NULL)
Packit a7d494
				ids_array = g_ptr_array_new ();
Packit a7d494
Packit a7d494
			g_ptr_array_add (ids_array, g_strdup (lang->priv->id));
Packit a7d494
		}
Packit a7d494
		else
Packit a7d494
		{
Packit a7d494
			g_object_unref (lang);
Packit a7d494
		}
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if (ids_array != NULL)
Packit a7d494
	{
Packit a7d494
		/* Sort the array alphabetically so that it
Packit a7d494
		 * is ready to use in a list of a GUI */
Packit a7d494
		g_ptr_array_sort_with_data (ids_array,
Packit a7d494
		                            (GCompareDataFunc)language_compare,
Packit a7d494
		                            lm->priv->language_ids);
Packit a7d494
Packit a7d494
		/* Ensure the array is NULL terminated */
Packit a7d494
		g_ptr_array_add (ids_array, NULL);
Packit a7d494
Packit a7d494
		lm->priv->ids = (gchar **)g_ptr_array_free (ids_array, FALSE);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	g_slist_free_full (filenames, g_free);
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_language_manager_get_language_ids:
Packit a7d494
 * @lm: a #GtkSourceLanguageManager.
Packit a7d494
 *
Packit a7d494
 * Returns the ids of the available languages.
Packit a7d494
 *
Packit a7d494
 * Returns: (nullable) (array zero-terminated=1) (transfer none):
Packit a7d494
 * a %NULL-terminated array of strings containing the ids of the available
Packit a7d494
 * languages or %NULL if no language is available.
Packit a7d494
 * The array is sorted alphabetically according to the language name.
Packit a7d494
 * The array is owned by @lm and must not be modified.
Packit a7d494
 */
Packit a7d494
const gchar * const *
Packit a7d494
gtk_source_language_manager_get_language_ids (GtkSourceLanguageManager *lm)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_LANGUAGE_MANAGER (lm), NULL);
Packit a7d494
Packit a7d494
	ensure_languages (lm);
Packit a7d494
Packit a7d494
	return (const gchar * const *)lm->priv->ids;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_language_manager_get_language:
Packit a7d494
 * @lm: a #GtkSourceLanguageManager.
Packit a7d494
 * @id: a language id.
Packit a7d494
 *
Packit a7d494
 * Gets the #GtkSourceLanguage identified by the given @id in the language
Packit a7d494
 * manager.
Packit a7d494
 *
Packit a7d494
 * Returns: (nullable) (transfer none): a #GtkSourceLanguage, or %NULL
Packit a7d494
 * if there is no language identified by the given @id. Return value is
Packit a7d494
 * owned by @lm and should not be freed.
Packit a7d494
 */
Packit a7d494
GtkSourceLanguage *
Packit a7d494
gtk_source_language_manager_get_language (GtkSourceLanguageManager *lm,
Packit a7d494
					  const gchar              *id)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_LANGUAGE_MANAGER (lm), NULL);
Packit a7d494
	g_return_val_if_fail (id != NULL, NULL);
Packit a7d494
Packit a7d494
	ensure_languages (lm);
Packit a7d494
Packit a7d494
	return g_hash_table_lookup (lm->priv->language_ids, id);
Packit a7d494
}
Packit a7d494
Packit a7d494
static GSList *
Packit a7d494
pick_langs_for_filename (GtkSourceLanguageManager *lm,
Packit a7d494
			 const gchar              *filename)
Packit a7d494
{
Packit a7d494
	char *filename_utf8;
Packit a7d494
	const gchar* const * p;
Packit a7d494
	GSList *langs = NULL;
Packit a7d494
Packit a7d494
	/* Use g_filename_display_name() instead of g_filename_to_utf8() because
Packit a7d494
	 * g_filename_display_name() doesn't fail and replaces non-convertible
Packit a7d494
	 * characters to unicode substitution symbol. */
Packit a7d494
	filename_utf8 = g_filename_display_name (filename);
Packit a7d494
Packit a7d494
	for (p = gtk_source_language_manager_get_language_ids (lm);
Packit a7d494
	     p != NULL && *p != NULL;
Packit a7d494
	     p++)
Packit a7d494
	{
Packit a7d494
		GtkSourceLanguage *lang;
Packit a7d494
		gchar **globs, **gptr;
Packit a7d494
Packit a7d494
		lang = gtk_source_language_manager_get_language (lm, *p);
Packit a7d494
		globs = gtk_source_language_get_globs (lang);
Packit a7d494
Packit a7d494
		for (gptr = globs; gptr != NULL && *gptr != NULL; gptr++)
Packit a7d494
		{
Packit a7d494
			/* FIXME g_pattern_match is wrong: there are no '[...]'
Packit a7d494
			 * character ranges and '*' and '?' can not be escaped
Packit a7d494
			 * to include them literally in a pattern.  */
Packit a7d494
			if (g_pattern_match_simple (*gptr, filename_utf8))
Packit a7d494
			{
Packit a7d494
				langs = g_slist_prepend (langs, lang);
Packit a7d494
			}
Packit a7d494
		}
Packit a7d494
Packit a7d494
		g_strfreev (globs);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	g_free (filename_utf8);
Packit a7d494
	return langs;
Packit a7d494
}
Packit a7d494
Packit a7d494
static GtkSourceLanguage *
Packit a7d494
pick_lang_for_mime_type_pass (GtkSourceLanguageManager *lm,
Packit a7d494
			      const char               *mime_type,
Packit a7d494
			      gboolean                  exact_match)
Packit a7d494
{
Packit a7d494
	const gchar* const * id_ptr;
Packit a7d494
Packit a7d494
	for (id_ptr = gtk_source_language_manager_get_language_ids (lm);
Packit a7d494
	     id_ptr != NULL && *id_ptr != NULL;
Packit a7d494
	     id_ptr++)
Packit a7d494
	{
Packit a7d494
		GtkSourceLanguage *lang;
Packit a7d494
		gchar **mime_types, **mptr;
Packit a7d494
Packit a7d494
		lang = gtk_source_language_manager_get_language (lm, *id_ptr);
Packit a7d494
		mime_types = gtk_source_language_get_mime_types (lang);
Packit a7d494
Packit a7d494
		for (mptr = mime_types; mptr != NULL && *mptr != NULL; mptr++)
Packit a7d494
		{
Packit a7d494
			gboolean matches;
Packit a7d494
Packit a7d494
			if (exact_match)
Packit a7d494
				matches = strcmp (mime_type, *mptr) == 0;
Packit a7d494
			else
Packit a7d494
				matches = g_content_type_is_a (mime_type, *mptr);
Packit a7d494
Packit a7d494
			if (matches)
Packit a7d494
			{
Packit a7d494
				g_strfreev (mime_types);
Packit a7d494
				return lang;
Packit a7d494
			}
Packit a7d494
		}
Packit a7d494
Packit a7d494
		g_strfreev (mime_types);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	return NULL;
Packit a7d494
}
Packit a7d494
Packit a7d494
static GtkSourceLanguage *
Packit a7d494
pick_lang_for_mime_type_real (GtkSourceLanguageManager *lm,
Packit a7d494
			      const char               *mime_type)
Packit a7d494
{
Packit a7d494
	GtkSourceLanguage *lang;
Packit a7d494
	lang = pick_lang_for_mime_type_pass (lm, mime_type, TRUE);
Packit a7d494
	if (!lang)
Packit a7d494
		lang = pick_lang_for_mime_type_pass (lm, mime_type, FALSE);
Packit a7d494
	return lang;
Packit a7d494
}
Packit a7d494
Packit a7d494
#ifdef G_OS_WIN32
Packit a7d494
static void
Packit a7d494
grok_win32_content_type (const gchar  *content_type,
Packit a7d494
			 gchar       **alt_filename,
Packit a7d494
			 gchar       **mime_type)
Packit a7d494
{
Packit a7d494
	*alt_filename = NULL;
Packit a7d494
	*mime_type = NULL;
Packit a7d494
Packit a7d494
	/* If it contains slash, then it's probably a mime type.
Packit a7d494
	 * Otherwise treat is an extension. */
Packit a7d494
	if (strchr (content_type, '/') != NULL)
Packit a7d494
		*mime_type = g_strdup (content_type);
Packit a7d494
	else
Packit a7d494
		*alt_filename = g_strjoin ("filename", content_type, NULL);
Packit a7d494
}
Packit a7d494
#endif
Packit a7d494
Packit a7d494
static GtkSourceLanguage *
Packit a7d494
pick_lang_for_mime_type (GtkSourceLanguageManager *lm,
Packit a7d494
			 const gchar              *content_type)
Packit a7d494
{
Packit a7d494
	GtkSourceLanguage *lang = NULL;
Packit a7d494
Packit a7d494
#ifndef G_OS_WIN32
Packit a7d494
	/* On Unix "content type" is mime type */
Packit a7d494
	lang = pick_lang_for_mime_type_real (lm, content_type);
Packit a7d494
#else
Packit a7d494
	/* On Windows "content type" is an extension, but user may pass a mime type too */
Packit a7d494
	gchar *mime_type;
Packit a7d494
	gchar *alt_filename;
Packit a7d494
Packit a7d494
	grok_win32_content_type (content_type, &alt_filename, &mime_type);
Packit a7d494
Packit a7d494
	if (alt_filename != NULL)
Packit a7d494
	{
Packit a7d494
		GSList *langs;
Packit a7d494
Packit a7d494
		langs = pick_langs_for_filename (lm, alt_filename);
Packit a7d494
Packit a7d494
		if (langs != NULL)
Packit a7d494
			lang = GTK_SOURCE_LANGUAGE (langs->data);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if (lang == NULL && mime_type != NULL)
Packit a7d494
		lang = pick_lang_for_mime_type_real (lm, mime_type);
Packit a7d494
Packit a7d494
	g_free (mime_type);
Packit a7d494
	g_free (alt_filename);
Packit a7d494
#endif
Packit a7d494
	return lang;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_language_manager_guess_language:
Packit a7d494
 * @lm: a #GtkSourceLanguageManager.
Packit a7d494
 * @filename: (nullable): a filename in Glib filename encoding, or %NULL.
Packit a7d494
 * @content_type: (nullable): a content type (as in GIO API), or %NULL.
Packit a7d494
 *
Packit a7d494
 * Picks a #GtkSourceLanguage for given file name and content type,
Packit a7d494
 * according to the information in lang files. Either @filename or
Packit a7d494
 * @content_type may be %NULL. This function can be used as follows:
Packit a7d494
 *
Packit a7d494
 * <informalexample><programlisting>
Packit a7d494
 *   GtkSourceLanguage *lang;
Packit a7d494
 *   lang = gtk_source_language_manager_guess_language (filename, NULL);
Packit a7d494
 *   gtk_source_buffer_set_language (buffer, lang);
Packit a7d494
 * </programlisting></informalexample>
Packit a7d494
 *
Packit a7d494
 * or
Packit a7d494
 *
Packit a7d494
 * <informalexample><programlisting>
Packit a7d494
 *   GtkSourceLanguage *lang = NULL;
Packit a7d494
 *   gboolean result_uncertain;
Packit a7d494
 *   gchar *content_type;
Packit a7d494
 *
Packit a7d494
 *   content_type = g_content_type_guess (filename, NULL, 0, &result_uncertain);
Packit a7d494
 *   if (result_uncertain)
Packit a7d494
 *     {
Packit a7d494
 *       g_free (content_type);
Packit a7d494
 *       content_type = NULL;
Packit a7d494
 *     }
Packit a7d494
 *
Packit a7d494
 *   lang = gtk_source_language_manager_guess_language (manager, filename, content_type);
Packit a7d494
 *   gtk_source_buffer_set_language (buffer, lang);
Packit a7d494
 *
Packit a7d494
 *   g_free (content_type);
Packit a7d494
 * </programlisting></informalexample>
Packit a7d494
 *
Packit a7d494
 * etc. Use gtk_source_language_get_mime_types() and gtk_source_language_get_globs()
Packit a7d494
 * if you need full control over file -> language mapping.
Packit a7d494
 *
Packit a7d494
 * Returns: (nullable) (transfer none): a #GtkSourceLanguage, or %NULL if there
Packit a7d494
 * is no suitable language for given @filename and/or @content_type. Return
Packit a7d494
 * value is owned by @lm and should not be freed.
Packit a7d494
 *
Packit a7d494
 * Since: 2.4
Packit a7d494
 */
Packit a7d494
GtkSourceLanguage *
Packit a7d494
gtk_source_language_manager_guess_language (GtkSourceLanguageManager *lm,
Packit a7d494
					    const gchar		     *filename,
Packit a7d494
					    const gchar		     *content_type)
Packit a7d494
{
Packit a7d494
	GtkSourceLanguage *lang = NULL;
Packit a7d494
	GSList *langs = NULL;
Packit a7d494
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_LANGUAGE_MANAGER (lm), NULL);
Packit a7d494
	g_return_val_if_fail ((filename != NULL && *filename != '\0') ||
Packit a7d494
	                      (content_type != NULL && *content_type != '\0'), NULL);
Packit a7d494
Packit a7d494
	ensure_languages (lm);
Packit a7d494
Packit a7d494
	/* Glob take precedence over mime match. Mime match is used in the
Packit a7d494
	   following cases:
Packit a7d494
	  - to pick among the list of glob matches
Packit a7d494
	  - to refine a glob match (e.g. glob is xml and mime is an xml dialect)
Packit a7d494
	  - no glob matches
Packit a7d494
	*/
Packit a7d494
Packit a7d494
	if (filename != NULL && *filename != '\0')
Packit a7d494
		langs = pick_langs_for_filename (lm, filename);
Packit a7d494
Packit a7d494
	if (langs != NULL)
Packit a7d494
	{
Packit a7d494
		/* Use mime to pick among glob matches */
Packit a7d494
		if (content_type != NULL)
Packit a7d494
		{
Packit a7d494
			GSList *l;
Packit a7d494
Packit a7d494
			for (l = langs; l != NULL; l = g_slist_next (l))
Packit a7d494
			{
Packit a7d494
				gchar **mime_types, **gptr;
Packit a7d494
Packit a7d494
				lang = GTK_SOURCE_LANGUAGE (l->data);
Packit a7d494
				mime_types = gtk_source_language_get_mime_types (lang);
Packit a7d494
Packit a7d494
				for (gptr = mime_types; gptr != NULL && *gptr != NULL; gptr++)
Packit a7d494
				{
Packit a7d494
					gchar *content;
Packit a7d494
Packit a7d494
					content = g_content_type_from_mime_type (*gptr);
Packit a7d494
Packit a7d494
					if (content != NULL && g_content_type_is_a (content_type, content))
Packit a7d494
					{
Packit a7d494
						if (!g_content_type_equals (content_type, content))
Packit a7d494
						{
Packit a7d494
							GtkSourceLanguage *mimelang;
Packit a7d494
Packit a7d494
							mimelang = pick_lang_for_mime_type (lm, content_type);
Packit a7d494
Packit a7d494
							if (mimelang != NULL)
Packit a7d494
								lang = mimelang;
Packit a7d494
						}
Packit a7d494
Packit a7d494
						g_strfreev (mime_types);
Packit a7d494
						g_slist_free (langs);
Packit a7d494
						g_free (content);
Packit a7d494
Packit a7d494
						return lang;
Packit a7d494
					}
Packit a7d494
					g_free (content);
Packit a7d494
				}
Packit a7d494
Packit a7d494
				g_strfreev (mime_types);
Packit a7d494
			}
Packit a7d494
		}
Packit a7d494
		lang = GTK_SOURCE_LANGUAGE (langs->data);
Packit a7d494
Packit a7d494
		g_slist_free (langs);
Packit a7d494
	}
Packit a7d494
	/* No glob match */
Packit a7d494
	else if (langs == NULL && content_type != NULL)
Packit a7d494
	{
Packit a7d494
		lang = pick_lang_for_mime_type (lm, content_type);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	return lang;
Packit a7d494
}