mirror of
https://github.com/netsurf-browser/netsurf
synced 2024-12-28 06:49:41 +03:00
ca96353d9f
svn://svn.netsurf-browser.org/branches/paulblokus/textinput ........ r7769 | paulblokus | 2009-06-11 22:26:16 +0100 (Thu, 11 Jun 2009) | 4 lines replace global history window with an empty window for future tests add the necessary files first lines ported ........ r7771 | paulblokus | 2009-06-11 23:51:46 +0100 (Thu, 11 Jun 2009) | 1 line more functions ........ r7772 | paulblokus | 2009-06-12 02:07:36 +0100 (Fri, 12 Jun 2009) | 1 line redraw working ........ r7777 | paulblokus | 2009-06-12 11:35:45 +0100 (Fri, 12 Jun 2009) | 3 lines plotter fix make use of the provided clipping rectangle ........ r7781 | paulblokus | 2009-06-12 16:26:51 +0100 (Fri, 12 Jun 2009) | 3 lines callbacks for taxtarea to request a [caret]redraw basic caret handling drawing ........ r7782 | paulblokus | 2009-06-12 22:36:50 +0100 (Fri, 12 Jun 2009) | 1 line single character insertion ........ r7783 | paulblokus | 2009-06-12 22:41:37 +0100 (Fri, 12 Jun 2009) | 1 line single character insertion ........ r7784 | paulblokus | 2009-06-12 23:55:40 +0100 (Fri, 12 Jun 2009) | 3 lines fixed caret clipping arrows, delete and backspace ........ r7812 | paulblokus | 2009-06-16 14:55:41 +0100 (Tue, 16 Jun 2009) | 1 line remove bug causing NS hang on \n in textarea ........ r7816 | paulblokus | 2009-06-16 16:29:48 +0100 (Tue, 16 Jun 2009) | 1 line Enter, Home, End keys ........ r7817 | paulblokus | 2009-06-16 16:56:16 +0100 (Tue, 16 Jun 2009) | 1 line Ctrl + Home/End ........ r7818 | paulblokus | 2009-06-16 17:16:51 +0100 (Tue, 16 Jun 2009) | 1 line redraw caret only on caret moves ........ r7821 | paulblokus | 2009-06-16 20:18:30 +0100 (Tue, 16 Jun 2009) | 1 line line end/start delete ........ r7822 | paulblokus | 2009-06-16 23:43:42 +0100 (Tue, 16 Jun 2009) | 1 line selection drawing + select all ........ r7823 | paulblokus | 2009-06-17 02:31:07 +0100 (Wed, 17 Jun 2009) | 3 lines auto scrolling on caret moves clear selection ........ r7845 | paulblokus | 2009-06-18 17:35:03 +0100 (Thu, 18 Jun 2009) | 1 line page up/down ........ r7846 | paulblokus | 2009-06-18 17:38:45 +0100 (Thu, 18 Jun 2009) | 1 line remove unnecessary fix ........ r7847 | paulblokus | 2009-06-18 18:00:16 +0100 (Thu, 18 Jun 2009) | 1 line clipping fixes ........ r7849 | paulblokus | 2009-06-18 18:21:02 +0100 (Thu, 18 Jun 2009) | 1 line scroll fix ........ r7850 | paulblokus | 2009-06-18 18:45:13 +0100 (Thu, 18 Jun 2009) | 1 line simplified redraw request logic ........ r7855 | paulblokus | 2009-06-18 19:56:24 +0100 (Thu, 18 Jun 2009) | 1 line front end passing mouse events ........ r7858 | paulblokus | 2009-06-18 22:18:39 +0100 (Thu, 18 Jun 2009) | 3 lines drag selection bug fixes ........ r7860 | paulblokus | 2009-06-18 23:32:39 +0100 (Thu, 18 Jun 2009) | 3 lines take selection into account on keypress of different types a few bugs fixed ........ r7876 | paulblokus | 2009-06-19 13:43:07 +0100 (Fri, 19 Jun 2009) | 3 lines pango nsfont_split fix a few textarea fixes ........ r7879 | paulblokus | 2009-06-19 17:33:10 +0100 (Fri, 19 Jun 2009) | 4 lines newline handling seems to work this way clear selection on mouse click more bug fixes ........ r7880 | paulblokus | 2009-06-19 18:16:27 +0100 (Fri, 19 Jun 2009) | 3 lines no caret option selection follows drag ........ r7883 | paulblokus | 2009-06-19 19:08:44 +0100 (Fri, 19 Jun 2009) | 3 lines o width selection bug fix caret at correct side of drag selection ........ r7918 | paulblokus | 2009-06-22 21:01:28 +0100 (Mon, 22 Jun 2009) | 3 lines fix caret positioning at line end CR removal in input methods ........ r7919 | paulblokus | 2009-06-22 21:34:39 +0100 (Mon, 22 Jun 2009) | 1 line fix crash on 0 length text ........ r7926 | paulblokus | 2009-06-23 09:53:56 +0100 (Tue, 23 Jun 2009) | 3 lines change LF into spaces for single line widget text normalisation at one place ........ r7931 | paulblokus | 2009-06-23 10:51:25 +0100 (Tue, 23 Jun 2009) | 1 line cleanup ........ r7933 | paulblokus | 2009-06-23 11:17:22 +0100 (Tue, 23 Jun 2009) | 1 line fix selection draw ........ r7935 | paulblokus | 2009-06-23 11:41:30 +0100 (Tue, 23 Jun 2009) | 1 line guard readonly ........ r7942 | paulblokus | 2009-06-24 08:19:39 +0100 (Wed, 24 Jun 2009) | 1 line applied changes suggested by jmb ........ r7943 | paulblokus | 2009-06-24 09:04:49 +0100 (Wed, 24 Jun 2009) | 1 line little fixes ........ r7945 | paulblokus | 2009-06-24 12:50:14 +0100 (Wed, 24 Jun 2009) | 1 line correct line length and wrapping ........ r7947 | paulblokus | 2009-06-24 14:32:36 +0100 (Wed, 24 Jun 2009) | 3 lines fixed page up/down broken in last commit changed logic for caret positioning on soft breaks ........ r7949 | paulblokus | 2009-06-24 16:31:42 +0100 (Wed, 24 Jun 2009) | 1 line remove temporary/test code ........ r7975 | paulblokus | 2009-06-25 16:00:46 +0100 (Thu, 25 Jun 2009) | 1 line changes suggested by jmb ........ r7976 | paulblokus | 2009-06-25 16:33:23 +0100 (Thu, 25 Jun 2009) | 1 line added ro_ prefix to RISC OS textarea code ........ svn path=/trunk/netsurf/; revision=8060
372 lines
9.9 KiB
C
372 lines
9.9 KiB
C
/*
|
|
* Copyright 2005 James Bursa <bursa@users.sourceforge.net>
|
|
*
|
|
* This file is part of NetSurf, http://www.netsurf-browser.org/
|
|
*
|
|
* NetSurf 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; version 2 of the License.
|
|
*
|
|
* NetSurf 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, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
/** \file
|
|
* Font handling (GTK implementation).
|
|
*
|
|
* Pango is used handle and render fonts.
|
|
*/
|
|
|
|
|
|
#include <assert.h>
|
|
#include <stdio.h>
|
|
#include <gtk/gtk.h>
|
|
#include "css/css.h"
|
|
#include "gtk/font_pango.h"
|
|
#include "gtk/gtk_plotters.h"
|
|
#include "render/font.h"
|
|
#include "utils/utils.h"
|
|
#include "utils/log.h"
|
|
#include "desktop/options.h"
|
|
|
|
/* Until we can consider the descenders etc, we need to not render using cairo */
|
|
#undef CAIRO_VERSION
|
|
|
|
static bool nsfont_width(const struct css_style *style,
|
|
const char *string, size_t length,
|
|
int *width);
|
|
|
|
static bool nsfont_position_in_string(const struct css_style *style,
|
|
const char *string, size_t length,
|
|
int x, size_t *char_offset, int *actual_x);
|
|
|
|
static bool nsfont_split(const struct css_style *style,
|
|
const char *string, size_t length,
|
|
int x, size_t *char_offset, int *actual_x);
|
|
|
|
const struct font_functions nsfont = {
|
|
nsfont_width,
|
|
nsfont_position_in_string,
|
|
nsfont_split
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
* Measure the width of a string.
|
|
*
|
|
* \param style css_style for this text, with style->font_size.size ==
|
|
* CSS_FONT_SIZE_LENGTH
|
|
* \param string UTF-8 string to measure
|
|
* \param length length of string
|
|
* \param width updated to width of string[0..length)
|
|
* \return true on success, false on error and error reported
|
|
*/
|
|
|
|
bool nsfont_width(const struct css_style *style,
|
|
const char *string, size_t length,
|
|
int *width)
|
|
{
|
|
PangoFontDescription *desc;
|
|
PangoContext *context;
|
|
PangoLayout *layout;
|
|
|
|
if (length == 0) {
|
|
*width = 0;
|
|
return true;
|
|
}
|
|
|
|
desc = nsfont_style_to_description(style);
|
|
context = gdk_pango_context_get();
|
|
layout = pango_layout_new(context);
|
|
pango_layout_set_font_description(layout, desc);
|
|
pango_layout_set_text(layout, string, length);
|
|
|
|
pango_layout_get_pixel_size(layout, width, 0);
|
|
|
|
g_object_unref(layout);
|
|
g_object_unref(context);
|
|
pango_font_description_free(desc);
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
/**
|
|
* Find the position in a string where an x coordinate falls.
|
|
*
|
|
* \param style css_style for this text, with style->font_size.size ==
|
|
* CSS_FONT_SIZE_LENGTH
|
|
* \param string UTF-8 string to measure
|
|
* \param length length of string
|
|
* \param x x coordinate to search for
|
|
* \param char_offset updated to offset in string of actual_x, [0..length]
|
|
* \param actual_x updated to x coordinate of character closest to x
|
|
* \return true on success, false on error and error reported
|
|
*/
|
|
|
|
bool nsfont_position_in_string(const struct css_style *style,
|
|
const char *string, size_t length,
|
|
int x, size_t *char_offset, int *actual_x)
|
|
{
|
|
int index;
|
|
PangoFontDescription *desc;
|
|
PangoContext *context;
|
|
PangoLayout *layout;
|
|
PangoRectangle pos;
|
|
|
|
desc = nsfont_style_to_description(style);
|
|
context = gdk_pango_context_get();
|
|
layout = pango_layout_new(context);
|
|
pango_layout_set_font_description(layout, desc);
|
|
pango_layout_set_text(layout, string, length);
|
|
|
|
pango_layout_xy_to_index(layout, x * PANGO_SCALE, 0, &index, 0);
|
|
if (pango_layout_xy_to_index(layout, x * PANGO_SCALE,
|
|
0, &index, 0) == 0)
|
|
index = length;
|
|
|
|
pango_layout_index_to_pos(layout, index, &pos);
|
|
|
|
g_object_unref(layout);
|
|
g_object_unref(context);
|
|
pango_font_description_free(desc);
|
|
|
|
*char_offset = index;
|
|
*actual_x = PANGO_PIXELS(pos.x);
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
/**
|
|
* Find where to split a string to make it fit a width.
|
|
*
|
|
* \param style css_style for this text, with style->font_size.size ==
|
|
* CSS_FONT_SIZE_LENGTH
|
|
* \param string UTF-8 string to measure
|
|
* \param length length of string
|
|
* \param x width available
|
|
* \param char_offset updated to offset in string of actual_x, [0..length]
|
|
* \param actual_x updated to x coordinate of character closest to x
|
|
* \return true on success, false on error and error reported
|
|
*
|
|
* On exit, [char_offset == 0 ||
|
|
* string[char_offset] == ' ' ||
|
|
* char_offset == length]
|
|
*/
|
|
|
|
bool nsfont_split(const struct css_style *style,
|
|
const char *string, size_t length,
|
|
int x, size_t *char_offset, int *actual_x)
|
|
{
|
|
int index = length;
|
|
PangoFontDescription *desc;
|
|
PangoContext *context;
|
|
PangoLayout *layout;
|
|
PangoLayoutLine *line;
|
|
PangoLayoutIter *iter;
|
|
PangoRectangle rect;
|
|
|
|
desc = nsfont_style_to_description(style);
|
|
context = gdk_pango_context_get();
|
|
layout = pango_layout_new(context);
|
|
pango_layout_set_font_description(layout, desc);
|
|
pango_layout_set_text(layout, string, length);
|
|
|
|
pango_layout_set_width(layout, x * PANGO_SCALE);
|
|
pango_layout_set_wrap(layout, PANGO_WRAP_WORD);
|
|
pango_layout_set_single_paragraph_mode(layout, true);
|
|
line = pango_layout_get_line(layout, 1);
|
|
if (line)
|
|
index = line->start_index - 1;
|
|
|
|
iter = pango_layout_get_iter(layout);
|
|
pango_layout_iter_get_line_extents(iter, NULL, &rect);
|
|
pango_layout_iter_free(iter);
|
|
|
|
g_object_unref(layout);
|
|
g_object_unref(context);
|
|
pango_font_description_free(desc);
|
|
|
|
*char_offset = index;
|
|
*actual_x = PANGO_PIXELS(rect.width);
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
/**
|
|
* Render a string.
|
|
*
|
|
* \param style css_style for this text, with style->font_size.size ==
|
|
* CSS_FONT_SIZE_LENGTH
|
|
* \param string UTF-8 string to measure
|
|
* \param length length of string
|
|
* \param x x coordinate
|
|
* \param y y coordinate
|
|
* \param c colour for text
|
|
* \return true on success, false on error and error reported
|
|
*/
|
|
|
|
bool nsfont_paint(const struct css_style *style,
|
|
const char *string, size_t length,
|
|
int x, int y, colour c)
|
|
{
|
|
PangoFontDescription *desc;
|
|
PangoLayout *layout;
|
|
PangoLayoutLine *line;
|
|
gint size;
|
|
#ifdef CAIRO_VERSION
|
|
int width, height;
|
|
#else
|
|
PangoContext *context;
|
|
GdkColor colour = { 0,
|
|
((c & 0xff) << 8) | (c & 0xff),
|
|
(c & 0xff00) | (c & 0xff00 >> 8),
|
|
((c & 0xff0000) >> 8) | (c & 0xff0000 >> 16) };
|
|
#endif
|
|
|
|
if (length == 0)
|
|
return true;
|
|
|
|
desc = nsfont_style_to_description(style);
|
|
size = (gint)((double)pango_font_description_get_size(desc) * nsgtk_plot_get_scale());
|
|
if (pango_font_description_get_size_is_absolute(desc))
|
|
pango_font_description_set_absolute_size(desc, size);
|
|
else
|
|
pango_font_description_set_size(desc, size);
|
|
|
|
#ifdef CAIRO_VERSION
|
|
layout = pango_cairo_create_layout(current_cr);
|
|
#else
|
|
context = gdk_pango_context_get();
|
|
layout = pango_layout_new(context);
|
|
#endif
|
|
|
|
pango_layout_set_font_description(layout, desc);
|
|
pango_layout_set_text(layout, string, length);
|
|
line = pango_layout_get_line(layout, 0);
|
|
|
|
#ifdef CAIRO_VERSION
|
|
cairo_move_to(current_cr, x, y);
|
|
nsgtk_set_colour(c);
|
|
pango_cairo_show_layout_line(current_cr, layout, line);
|
|
#else
|
|
gdk_draw_layout_line_with_colors(current_drawable, current_gc,
|
|
x, y, line, &colour, 0);
|
|
|
|
g_object_unref(context);
|
|
#endif
|
|
g_object_unref(layout);
|
|
pango_font_description_free(desc);
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
/**
|
|
* Convert a css_style to a PangoFontDescription.
|
|
*
|
|
* \param style css_style for this text, with style->font_size.size ==
|
|
* CSS_FONT_SIZE_LENGTH
|
|
* \return a new Pango font description
|
|
*/
|
|
|
|
PangoFontDescription *nsfont_style_to_description(
|
|
const struct css_style *style)
|
|
{
|
|
unsigned int size;
|
|
PangoFontDescription *desc;
|
|
PangoWeight weight = PANGO_WEIGHT_NORMAL;
|
|
PangoStyle styl = PANGO_STYLE_NORMAL;
|
|
|
|
assert(style->font_size.size == CSS_FONT_SIZE_LENGTH);
|
|
|
|
switch (style->font_family) {
|
|
case CSS_FONT_FAMILY_SERIF:
|
|
desc = pango_font_description_from_string(option_font_serif);
|
|
break;
|
|
case CSS_FONT_FAMILY_MONOSPACE:
|
|
desc = pango_font_description_from_string(option_font_mono);
|
|
break;
|
|
case CSS_FONT_FAMILY_CURSIVE:
|
|
desc = pango_font_description_from_string(option_font_cursive);
|
|
break;
|
|
case CSS_FONT_FAMILY_FANTASY:
|
|
desc = pango_font_description_from_string(option_font_fantasy);
|
|
break;
|
|
case CSS_FONT_FAMILY_SANS_SERIF:
|
|
default:
|
|
desc = pango_font_description_from_string(option_font_sans);
|
|
break;
|
|
}
|
|
|
|
|
|
if (style->font_size.value.length.unit == CSS_UNIT_PX)
|
|
size = style->font_size.value.length.value;
|
|
else
|
|
size = css_len2pt(&style->font_size.value.length, style);
|
|
|
|
if (size < (unsigned)abs(option_font_min_size / 10))
|
|
size = option_font_min_size / 10;
|
|
|
|
size *= PANGO_SCALE;
|
|
|
|
switch (style->font_style) {
|
|
case CSS_FONT_STYLE_ITALIC:
|
|
styl = PANGO_STYLE_ITALIC;
|
|
break;
|
|
case CSS_FONT_STYLE_OBLIQUE:
|
|
styl = PANGO_STYLE_OBLIQUE;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
pango_font_description_set_style(desc, styl);
|
|
|
|
switch (style->font_weight) {
|
|
case CSS_FONT_WEIGHT_NORMAL:
|
|
weight = PANGO_WEIGHT_NORMAL; break;
|
|
case CSS_FONT_WEIGHT_BOLD:
|
|
weight = PANGO_WEIGHT_BOLD; break;
|
|
case CSS_FONT_WEIGHT_100: weight = 100; break;
|
|
case CSS_FONT_WEIGHT_200: weight = 200; break;
|
|
case CSS_FONT_WEIGHT_300: weight = 300; break;
|
|
case CSS_FONT_WEIGHT_400: weight = 400; break;
|
|
case CSS_FONT_WEIGHT_500: weight = 500; break;
|
|
case CSS_FONT_WEIGHT_600: weight = 600; break;
|
|
case CSS_FONT_WEIGHT_700: weight = 700; break;
|
|
case CSS_FONT_WEIGHT_800: weight = 800; break;
|
|
case CSS_FONT_WEIGHT_900: weight = 900; break;
|
|
default: break;
|
|
}
|
|
|
|
pango_font_description_set_weight(desc, weight);
|
|
|
|
if (style->font_size.value.length.unit == CSS_UNIT_PX)
|
|
pango_font_description_set_absolute_size(desc, size);
|
|
else
|
|
pango_font_description_set_size(desc, size);
|
|
|
|
switch (style->font_variant) {
|
|
case CSS_FONT_VARIANT_SMALL_CAPS:
|
|
pango_font_description_set_variant(desc, PANGO_VARIANT_SMALL_CAPS);
|
|
break;
|
|
case CSS_FONT_VARIANT_NORMAL:
|
|
default:
|
|
pango_font_description_set_variant(desc, PANGO_VARIANT_NORMAL);
|
|
}
|
|
|
|
return desc;
|
|
}
|
|
|