Blame gtksourceview/gtksourceview-utils.c

Packit a7d494
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8; coding: utf-8 -*-
Packit a7d494
 * gtksourceview-utils.c
Packit a7d494
 * This file is part of GtkSourceView
Packit a7d494
 *
Packit a7d494
 * Copyright (C) 2007 - Gustavo Giráldez and Paolo Maggi
Packit a7d494
 *
Packit a7d494
 * This library 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
 * This library 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 Street, 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 "gtksourceview-utils.h"
Packit a7d494
Packit a7d494
#include <errno.h>
Packit a7d494
#include <math.h>
Packit a7d494
Packit a7d494
#define SOURCEVIEW_DIR "gtksourceview-3.0"
Packit a7d494
Packit a7d494
gchar **
Packit a7d494
_gtk_source_view_get_default_dirs (const char *basename,
Packit a7d494
				   gboolean    compat)
Packit a7d494
{
Packit a7d494
	const gchar * const *xdg_dirs;
Packit a7d494
	GPtrArray *dirs;
Packit a7d494
Packit a7d494
	dirs = g_ptr_array_new ();
Packit a7d494
Packit a7d494
	/* user dir */
Packit a7d494
	g_ptr_array_add (dirs, g_build_filename (g_get_user_data_dir (),
Packit a7d494
						 SOURCEVIEW_DIR,
Packit a7d494
						 basename,
Packit a7d494
						 NULL));
Packit a7d494
Packit a7d494
#ifdef G_OS_UNIX
Packit a7d494
	/* Legacy gtsourceview 1 user dir, for backward compatibility */
Packit a7d494
	if (compat)
Packit a7d494
	{
Packit a7d494
		const gchar *home;
Packit a7d494
Packit a7d494
		home = g_get_home_dir ();
Packit a7d494
		if (home != NULL)
Packit a7d494
		{
Packit a7d494
			g_ptr_array_add (dirs,
Packit a7d494
					 g_strdup_printf ("%s/%s",
Packit a7d494
							  home,
Packit a7d494
							  ".gnome2/gtksourceview-1.0/language-specs"));
Packit a7d494
		}
Packit a7d494
	}
Packit a7d494
#endif
Packit a7d494
Packit a7d494
	/* system dir */
Packit a7d494
	for (xdg_dirs = g_get_system_data_dirs (); xdg_dirs && *xdg_dirs; ++xdg_dirs)
Packit a7d494
	{
Packit a7d494
		g_ptr_array_add (dirs, g_build_filename (*xdg_dirs,
Packit a7d494
							 SOURCEVIEW_DIR,
Packit a7d494
							 basename,
Packit a7d494
							 NULL));
Packit a7d494
	}
Packit a7d494
Packit a7d494
	g_ptr_array_add (dirs, NULL);
Packit a7d494
Packit a7d494
	return (gchar**) g_ptr_array_free (dirs, FALSE);
Packit a7d494
}
Packit a7d494
Packit a7d494
static GSList *
Packit a7d494
build_file_listing (const gchar *item,
Packit a7d494
		    GSList      *filenames,
Packit a7d494
		    const gchar *suffix,
Packit a7d494
		    gboolean     only_dirs)
Packit a7d494
{
Packit a7d494
	GDir *dir;
Packit a7d494
	const gchar *name;
Packit a7d494
Packit a7d494
	if (!only_dirs && g_file_test (item, G_FILE_TEST_IS_REGULAR))
Packit a7d494
	{
Packit a7d494
		filenames = g_slist_prepend (filenames, g_strdup(item));
Packit a7d494
		return filenames;
Packit a7d494
Packit a7d494
	}
Packit a7d494
	dir = g_dir_open (item, 0, NULL);
Packit a7d494
Packit a7d494
	if (dir == NULL)
Packit a7d494
	{
Packit a7d494
		return filenames;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	while ((name = g_dir_read_name (dir)) != NULL)
Packit a7d494
	{
Packit a7d494
		gchar *full_path = g_build_filename (item, name, NULL);
Packit a7d494
Packit a7d494
		if (!g_file_test (full_path, G_FILE_TEST_IS_DIR) &&
Packit a7d494
		    g_str_has_suffix (name, suffix))
Packit a7d494
		{
Packit a7d494
			filenames = g_slist_prepend (filenames, full_path);
Packit a7d494
		}
Packit a7d494
		else
Packit a7d494
		{
Packit a7d494
			g_free (full_path);
Packit a7d494
		}
Packit a7d494
	}
Packit a7d494
Packit a7d494
	g_dir_close (dir);
Packit a7d494
Packit a7d494
	return filenames;
Packit a7d494
}
Packit a7d494
Packit a7d494
GSList *
Packit a7d494
_gtk_source_view_get_file_list (gchar       **path,
Packit a7d494
				const gchar  *suffix,
Packit a7d494
				gboolean      only_dirs)
Packit a7d494
{
Packit a7d494
	GSList *files = NULL;
Packit a7d494
Packit a7d494
	for ( ; path && *path; ++path)
Packit a7d494
	{
Packit a7d494
		files = build_file_listing (*path, files, suffix, only_dirs);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	return g_slist_reverse (files);
Packit a7d494
}
Packit a7d494
Packit a7d494
/* Wrapper around strtoull for easier use: tries
Packit a7d494
 * to convert @str to a number and return -1 if it is not.
Packit a7d494
 * Used to check if references in subpattern contexts
Packit a7d494
 * (e.g. \%{1@start} or \%{blah@start}) are named or numbers.
Packit a7d494
 */
Packit a7d494
gint
Packit a7d494
_gtk_source_string_to_int (const gchar *str)
Packit a7d494
{
Packit a7d494
	guint64 number;
Packit a7d494
	gchar *end_str;
Packit a7d494
Packit a7d494
	if (str == NULL || *str == '\0')
Packit a7d494
	{
Packit a7d494
		return -1;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	errno = 0;
Packit a7d494
	number = g_ascii_strtoull (str, &end_str, 10);
Packit a7d494
Packit a7d494
	if (errno != 0 || number > G_MAXINT || *end_str != '\0')
Packit a7d494
	{
Packit a7d494
		return -1;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	return number;
Packit a7d494
}
Packit a7d494
Packit a7d494
#define FONT_FAMILY  "font-family"
Packit a7d494
#define FONT_VARIANT "font-variant"
Packit a7d494
#define FONT_STRETCH "font-stretch"
Packit a7d494
#define FONT_WEIGHT  "font-weight"
Packit a7d494
#define FONT_SIZE    "font-size"
Packit a7d494
Packit a7d494
gchar *
Packit a7d494
_gtk_source_pango_font_description_to_css (const PangoFontDescription *font_desc)
Packit a7d494
{
Packit a7d494
	PangoFontMask mask;
Packit a7d494
	GString *str;
Packit a7d494
Packit a7d494
#define ADD_KEYVAL(key,fmt) \
Packit a7d494
	g_string_append(str,key":"fmt";")
Packit a7d494
#define ADD_KEYVAL_PRINTF(key,fmt,...) \
Packit a7d494
	g_string_append_printf(str,key":"fmt";", __VA_ARGS__)
Packit a7d494
Packit a7d494
	g_return_val_if_fail (font_desc, NULL);
Packit a7d494
Packit a7d494
	str = g_string_new (NULL);
Packit a7d494
Packit a7d494
	mask = pango_font_description_get_set_fields (font_desc);
Packit a7d494
Packit a7d494
	if ((mask & PANGO_FONT_MASK_FAMILY) != 0)
Packit a7d494
	{
Packit a7d494
		const gchar *family;
Packit a7d494
Packit a7d494
		family = pango_font_description_get_family (font_desc);
Packit a7d494
		ADD_KEYVAL_PRINTF (FONT_FAMILY, "\"%s\"", family);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if ((mask & PANGO_FONT_MASK_STYLE) != 0)
Packit a7d494
	{
Packit a7d494
		PangoVariant variant;
Packit a7d494
Packit a7d494
		variant = pango_font_description_get_variant (font_desc);
Packit a7d494
Packit a7d494
		switch (variant)
Packit a7d494
		{
Packit a7d494
			case PANGO_VARIANT_NORMAL:
Packit a7d494
				ADD_KEYVAL (FONT_VARIANT, "normal");
Packit a7d494
				break;
Packit a7d494
Packit a7d494
			case PANGO_VARIANT_SMALL_CAPS:
Packit a7d494
				ADD_KEYVAL (FONT_VARIANT, "small-caps");
Packit a7d494
				break;
Packit a7d494
Packit a7d494
			default:
Packit a7d494
				break;
Packit a7d494
		}
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if ((mask & PANGO_FONT_MASK_WEIGHT))
Packit a7d494
	{
Packit a7d494
		gint weight;
Packit a7d494
Packit a7d494
		weight = pango_font_description_get_weight (font_desc);
Packit a7d494
Packit a7d494
		/*
Packit a7d494
		 * WORKAROUND:
Packit a7d494
		 *
Packit a7d494
		 * font-weight with numbers does not appear to be working as expected
Packit a7d494
		 * right now. So for the common (bold/normal), let's just use the string
Packit a7d494
		 * and let gtk warn for the other values, which shouldn't really be
Packit a7d494
		 * used for this.
Packit a7d494
		 */
Packit a7d494
Packit a7d494
		switch (weight)
Packit a7d494
		{
Packit a7d494
			case PANGO_WEIGHT_SEMILIGHT:
Packit a7d494
			/*
Packit a7d494
			 * 350 is not actually a valid css font-weight, so we will just round
Packit a7d494
			 * up to 400.
Packit a7d494
			 */
Packit a7d494
			case PANGO_WEIGHT_NORMAL:
Packit a7d494
				ADD_KEYVAL (FONT_WEIGHT, "normal");
Packit a7d494
				break;
Packit a7d494
Packit a7d494
			case PANGO_WEIGHT_BOLD:
Packit a7d494
				ADD_KEYVAL (FONT_WEIGHT, "bold");
Packit a7d494
				break;
Packit a7d494
Packit a7d494
			case PANGO_WEIGHT_THIN:
Packit a7d494
			case PANGO_WEIGHT_ULTRALIGHT:
Packit a7d494
			case PANGO_WEIGHT_LIGHT:
Packit a7d494
			case PANGO_WEIGHT_BOOK:
Packit a7d494
			case PANGO_WEIGHT_MEDIUM:
Packit a7d494
			case PANGO_WEIGHT_SEMIBOLD:
Packit a7d494
			case PANGO_WEIGHT_ULTRABOLD:
Packit a7d494
			case PANGO_WEIGHT_HEAVY:
Packit a7d494
			case PANGO_WEIGHT_ULTRAHEAVY:
Packit a7d494
			default:
Packit a7d494
				/* round to nearest hundred */
Packit a7d494
				weight = round (weight / 100.0) * 100;
Packit a7d494
				ADD_KEYVAL_PRINTF ("font-weight", "%d", weight);
Packit a7d494
				break;
Packit a7d494
		}
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if ((mask & PANGO_FONT_MASK_STRETCH))
Packit a7d494
	{
Packit a7d494
		switch (pango_font_description_get_stretch (font_desc))
Packit a7d494
		{
Packit a7d494
			case PANGO_STRETCH_ULTRA_CONDENSED:
Packit a7d494
				ADD_KEYVAL (FONT_STRETCH, "untra-condensed");
Packit a7d494
				break;
Packit a7d494
Packit a7d494
			case PANGO_STRETCH_EXTRA_CONDENSED:
Packit a7d494
				ADD_KEYVAL (FONT_STRETCH, "extra-condensed");
Packit a7d494
				break;
Packit a7d494
Packit a7d494
			case PANGO_STRETCH_CONDENSED:
Packit a7d494
				ADD_KEYVAL (FONT_STRETCH, "condensed");
Packit a7d494
				break;
Packit a7d494
Packit a7d494
			case PANGO_STRETCH_SEMI_CONDENSED:
Packit a7d494
				ADD_KEYVAL (FONT_STRETCH, "semi-condensed");
Packit a7d494
				break;
Packit a7d494
Packit a7d494
			case PANGO_STRETCH_NORMAL:
Packit a7d494
				ADD_KEYVAL (FONT_STRETCH, "normal");
Packit a7d494
				break;
Packit a7d494
Packit a7d494
			case PANGO_STRETCH_SEMI_EXPANDED:
Packit a7d494
				ADD_KEYVAL (FONT_STRETCH, "semi-expanded");
Packit a7d494
				break;
Packit a7d494
Packit a7d494
			case PANGO_STRETCH_EXPANDED:
Packit a7d494
				ADD_KEYVAL (FONT_STRETCH, "expanded");
Packit a7d494
				break;
Packit a7d494
Packit a7d494
			case PANGO_STRETCH_EXTRA_EXPANDED:
Packit a7d494
				ADD_KEYVAL (FONT_STRETCH, "extra-expanded");
Packit a7d494
				break;
Packit a7d494
Packit a7d494
			case PANGO_STRETCH_ULTRA_EXPANDED:
Packit a7d494
				ADD_KEYVAL (FONT_STRETCH, "untra-expanded");
Packit a7d494
				break;
Packit a7d494
Packit a7d494
			default:
Packit a7d494
				break;
Packit a7d494
		}
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if ((mask & PANGO_FONT_MASK_SIZE))
Packit a7d494
	{
Packit a7d494
		gint font_size;
Packit a7d494
Packit a7d494
		font_size = pango_font_description_get_size (font_desc) / PANGO_SCALE;
Packit a7d494
		ADD_KEYVAL_PRINTF ("font-size", "%dpt", font_size);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	return g_string_free (str, FALSE);
Packit a7d494
Packit a7d494
#undef ADD_KEYVAL
Packit a7d494
#undef ADD_KEYVAL_PRINTF
Packit a7d494
}