forked from AuroraMiddleware/gtk
Make dimension values and calc values just number values
This commit is contained in:
parent
df08387e35
commit
b25f93e24c
@ -21,300 +21,6 @@
|
||||
|
||||
#include <string.h>
|
||||
|
||||
struct _GtkCssValue {
|
||||
GTK_CSS_VALUE_BASE
|
||||
gsize n_terms;
|
||||
GtkCssValue * terms[1];
|
||||
};
|
||||
|
||||
static gsize
|
||||
gtk_css_value_calc_get_size (gsize n_terms)
|
||||
{
|
||||
g_assert (n_terms > 0);
|
||||
|
||||
return sizeof (GtkCssValue) + sizeof (GtkCssValue *) * (n_terms - 1);
|
||||
}
|
||||
static void
|
||||
gtk_css_value_calc_free (GtkCssValue *value)
|
||||
{
|
||||
gsize i;
|
||||
|
||||
for (i = 0; i < value->n_terms; i++)
|
||||
{
|
||||
_gtk_css_value_unref (value->terms[i]);
|
||||
}
|
||||
|
||||
g_slice_free1 (gtk_css_value_calc_get_size (value->n_terms), value);
|
||||
}
|
||||
|
||||
static GtkCssValue *gtk_css_calc_value_new (gsize n_terms);
|
||||
|
||||
static GtkCssValue *
|
||||
gtk_css_value_new_from_array (GtkCssValue **values,
|
||||
guint n_values)
|
||||
{
|
||||
GtkCssValue *result;
|
||||
|
||||
if (n_values > 1)
|
||||
{
|
||||
result = gtk_css_calc_value_new (n_values);
|
||||
memcpy (result->terms, values, n_values * sizeof (GtkCssValue *));
|
||||
}
|
||||
else
|
||||
{
|
||||
result = values[0];
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
static void
|
||||
gtk_css_calc_array_add (GPtrArray *array, GtkCssValue *value)
|
||||
{
|
||||
gsize i;
|
||||
gint calc_term_order;
|
||||
|
||||
calc_term_order = gtk_css_number_value_get_calc_term_order (value);
|
||||
|
||||
for (i = 0; i < array->len; i++)
|
||||
{
|
||||
GtkCssValue *sum = gtk_css_number_value_try_add (g_ptr_array_index (array, i), value);
|
||||
|
||||
if (sum)
|
||||
{
|
||||
g_ptr_array_index (array, i) = sum;
|
||||
_gtk_css_value_unref (value);
|
||||
return;
|
||||
}
|
||||
else if (gtk_css_number_value_get_calc_term_order (g_ptr_array_index (array, i)) > calc_term_order)
|
||||
{
|
||||
g_ptr_array_insert (array, i, value);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
g_ptr_array_add (array, value);
|
||||
}
|
||||
|
||||
static GtkCssValue *
|
||||
gtk_css_value_calc_compute (GtkCssValue *value,
|
||||
guint property_id,
|
||||
GtkStyleProvider *provider,
|
||||
GtkCssStyle *style,
|
||||
GtkCssStyle *parent_style)
|
||||
{
|
||||
GtkCssValue *result;
|
||||
gboolean changed = FALSE;
|
||||
gsize i;
|
||||
GtkCssValue **new_values;
|
||||
|
||||
new_values = g_alloca (sizeof (GtkCssValue *) * value->n_terms);
|
||||
|
||||
for (i = 0; i < value->n_terms; i++)
|
||||
{
|
||||
GtkCssValue *computed = _gtk_css_value_compute (value->terms[i], property_id, provider, style, parent_style);
|
||||
changed |= computed != value->terms[i];
|
||||
new_values[i] = computed;
|
||||
}
|
||||
|
||||
if (changed)
|
||||
{
|
||||
result = gtk_css_value_new_from_array (new_values, value->n_terms);
|
||||
}
|
||||
else
|
||||
{
|
||||
for (i = 0; i < value->n_terms; i++)
|
||||
gtk_css_value_unref (new_values[i]);
|
||||
|
||||
result = _gtk_css_value_ref (value);
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
|
||||
static gboolean
|
||||
gtk_css_value_calc_equal (const GtkCssValue *value1,
|
||||
const GtkCssValue *value2)
|
||||
{
|
||||
gsize i;
|
||||
|
||||
if (value1->n_terms != value2->n_terms)
|
||||
return FALSE;
|
||||
|
||||
for (i = 0; i < value1->n_terms; i++)
|
||||
{
|
||||
if (!_gtk_css_value_equal (value1->terms[i], value2->terms[i]))
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static void
|
||||
gtk_css_value_calc_print (const GtkCssValue *value,
|
||||
GString *string)
|
||||
{
|
||||
gsize i;
|
||||
|
||||
g_string_append (string, "calc(");
|
||||
_gtk_css_value_print (value->terms[0], string);
|
||||
|
||||
for (i = 1; i < value->n_terms; i++)
|
||||
{
|
||||
g_string_append (string, " + ");
|
||||
_gtk_css_value_print (value->terms[i], string);
|
||||
}
|
||||
g_string_append (string, ")");
|
||||
}
|
||||
|
||||
static double
|
||||
gtk_css_value_calc_get (const GtkCssValue *value,
|
||||
double one_hundred_percent)
|
||||
{
|
||||
double result = 0.0;
|
||||
gsize i;
|
||||
|
||||
for (i = 0; i < value->n_terms; i++)
|
||||
{
|
||||
result += _gtk_css_number_value_get (value->terms[i], one_hundred_percent);
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
static GtkCssDimension
|
||||
gtk_css_value_calc_get_dimension (const GtkCssValue *value)
|
||||
{
|
||||
GtkCssDimension dimension = GTK_CSS_DIMENSION_PERCENTAGE;
|
||||
gsize i;
|
||||
|
||||
for (i = 0; i < value->n_terms && dimension == GTK_CSS_DIMENSION_PERCENTAGE; i++)
|
||||
{
|
||||
dimension = gtk_css_number_value_get_dimension (value->terms[i]);
|
||||
}
|
||||
|
||||
return dimension;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
gtk_css_value_calc_has_percent (const GtkCssValue *value)
|
||||
{
|
||||
gsize i;
|
||||
|
||||
for (i = 0; i < value->n_terms; i++)
|
||||
{
|
||||
if (gtk_css_number_value_has_percent (value->terms[i]))
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
static GtkCssValue *
|
||||
gtk_css_value_calc_multiply (GtkCssValue *value,
|
||||
double factor)
|
||||
{
|
||||
GtkCssValue *result;
|
||||
gsize i;
|
||||
|
||||
result = gtk_css_calc_value_new (value->n_terms);
|
||||
|
||||
for (i = 0; i < value->n_terms; i++)
|
||||
{
|
||||
result->terms[i] = gtk_css_number_value_multiply (value->terms[i], factor);
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
static GtkCssValue *
|
||||
gtk_css_value_calc_try_add (GtkCssValue *value1,
|
||||
GtkCssValue *value2)
|
||||
{
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static gint
|
||||
gtk_css_value_calc_get_calc_term_order (const GtkCssValue *value)
|
||||
{
|
||||
/* This should never be needed because calc() can't contain calc(),
|
||||
* but eh...
|
||||
*/
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const GtkCssNumberValueClass GTK_CSS_VALUE_CALC = {
|
||||
{
|
||||
"GtkCssCalcValue",
|
||||
gtk_css_value_calc_free,
|
||||
gtk_css_value_calc_compute,
|
||||
gtk_css_value_calc_equal,
|
||||
gtk_css_number_value_transition,
|
||||
NULL,
|
||||
NULL,
|
||||
gtk_css_value_calc_print
|
||||
},
|
||||
gtk_css_value_calc_get,
|
||||
gtk_css_value_calc_get_dimension,
|
||||
gtk_css_value_calc_has_percent,
|
||||
gtk_css_value_calc_multiply,
|
||||
gtk_css_value_calc_try_add,
|
||||
gtk_css_value_calc_get_calc_term_order
|
||||
};
|
||||
|
||||
static GtkCssValue *
|
||||
gtk_css_calc_value_new (gsize n_terms)
|
||||
{
|
||||
GtkCssValue *result;
|
||||
|
||||
result = _gtk_css_value_alloc (>K_CSS_VALUE_CALC.value_class,
|
||||
gtk_css_value_calc_get_size (n_terms));
|
||||
result->n_terms = n_terms;
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
GtkCssValue *
|
||||
gtk_css_calc_value_new_sum (GtkCssValue *value1,
|
||||
GtkCssValue *value2)
|
||||
{
|
||||
GPtrArray *array;
|
||||
GtkCssValue *result;
|
||||
gsize i;
|
||||
|
||||
array = g_ptr_array_new ();
|
||||
|
||||
if (value1->class == >K_CSS_VALUE_CALC.value_class)
|
||||
{
|
||||
for (i = 0; i < value1->n_terms; i++)
|
||||
{
|
||||
gtk_css_calc_array_add (array, _gtk_css_value_ref (value1->terms[i]));
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
gtk_css_calc_array_add (array, _gtk_css_value_ref (value1));
|
||||
}
|
||||
|
||||
if (value2->class == >K_CSS_VALUE_CALC.value_class)
|
||||
{
|
||||
for (i = 0; i < value2->n_terms; i++)
|
||||
{
|
||||
gtk_css_calc_array_add (array, _gtk_css_value_ref (value2->terms[i]));
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
gtk_css_calc_array_add (array, _gtk_css_value_ref (value2));
|
||||
}
|
||||
|
||||
result = gtk_css_value_new_from_array ((GtkCssValue **)array->pdata, array->len);
|
||||
g_ptr_array_free (array, TRUE);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
GtkCssValue * gtk_css_calc_value_parse_sum (GtkCssParser *parser,
|
||||
GtkCssNumberParseFlags flags);
|
||||
|
||||
|
@ -22,9 +22,6 @@
|
||||
|
||||
G_BEGIN_DECLS
|
||||
|
||||
GtkCssValue * gtk_css_calc_value_new_sum (GtkCssValue *value1,
|
||||
GtkCssValue *value2);
|
||||
|
||||
GtkCssValue * gtk_css_calc_value_parse (GtkCssParser *parser,
|
||||
GtkCssNumberParseFlags flags);
|
||||
|
||||
|
@ -19,370 +19,6 @@
|
||||
|
||||
#include "gtkcssdimensionvalueprivate.h"
|
||||
|
||||
#include "gtkcssenumvalueprivate.h"
|
||||
#include "gtkstylepropertyprivate.h"
|
||||
|
||||
#include "fallback-c89.c"
|
||||
|
||||
struct _GtkCssValue {
|
||||
GTK_CSS_VALUE_BASE
|
||||
GtkCssUnit unit;
|
||||
double value;
|
||||
};
|
||||
|
||||
static void
|
||||
gtk_css_value_dimension_free (GtkCssValue *value)
|
||||
{
|
||||
g_slice_free (GtkCssValue, value);
|
||||
}
|
||||
|
||||
static double
|
||||
get_base_font_size_px (guint property_id,
|
||||
GtkStyleProvider *provider,
|
||||
GtkCssStyle *style,
|
||||
GtkCssStyle *parent_style)
|
||||
{
|
||||
if (property_id == GTK_CSS_PROPERTY_FONT_SIZE)
|
||||
{
|
||||
if (parent_style)
|
||||
return _gtk_css_number_value_get (parent_style->core->font_size, 100);
|
||||
else
|
||||
return gtk_css_font_size_get_default_px (provider, style);
|
||||
}
|
||||
|
||||
return _gtk_css_number_value_get (style->core->font_size, 100);
|
||||
}
|
||||
|
||||
static double
|
||||
get_dpi (GtkCssStyle *style)
|
||||
{
|
||||
return _gtk_css_number_value_get (style->core->dpi, 96);
|
||||
}
|
||||
|
||||
static GtkCssValue *
|
||||
gtk_css_value_dimension_compute (GtkCssValue *number,
|
||||
guint property_id,
|
||||
GtkStyleProvider *provider,
|
||||
GtkCssStyle *style,
|
||||
GtkCssStyle *parent_style)
|
||||
{
|
||||
switch (number->unit)
|
||||
{
|
||||
case GTK_CSS_PERCENT:
|
||||
/* percentages for font sizes are computed, other percentages aren't */
|
||||
if (property_id == GTK_CSS_PROPERTY_FONT_SIZE)
|
||||
return gtk_css_dimension_value_new (number->value / 100.0 *
|
||||
get_base_font_size_px (property_id, provider, style, parent_style),
|
||||
GTK_CSS_PX);
|
||||
G_GNUC_FALLTHROUGH;
|
||||
case GTK_CSS_NUMBER:
|
||||
case GTK_CSS_PX:
|
||||
case GTK_CSS_DEG:
|
||||
case GTK_CSS_S:
|
||||
return _gtk_css_value_ref (number);
|
||||
case GTK_CSS_PT:
|
||||
return gtk_css_dimension_value_new (number->value * get_dpi (style) / 72.0,
|
||||
GTK_CSS_PX);
|
||||
case GTK_CSS_PC:
|
||||
return gtk_css_dimension_value_new (number->value * get_dpi (style) / 72.0 * 12.0,
|
||||
GTK_CSS_PX);
|
||||
case GTK_CSS_IN:
|
||||
return gtk_css_dimension_value_new (number->value * get_dpi (style),
|
||||
GTK_CSS_PX);
|
||||
case GTK_CSS_CM:
|
||||
return gtk_css_dimension_value_new (number->value * get_dpi (style) * 0.39370078740157477,
|
||||
GTK_CSS_PX);
|
||||
case GTK_CSS_MM:
|
||||
return gtk_css_dimension_value_new (number->value * get_dpi (style) * 0.039370078740157477,
|
||||
GTK_CSS_PX);
|
||||
case GTK_CSS_EM:
|
||||
return gtk_css_dimension_value_new (number->value *
|
||||
get_base_font_size_px (property_id, provider, style, parent_style),
|
||||
GTK_CSS_PX);
|
||||
case GTK_CSS_EX:
|
||||
/* for now we pretend ex is half of em */
|
||||
return gtk_css_dimension_value_new (number->value * 0.5 *
|
||||
get_base_font_size_px (property_id, provider, style, parent_style),
|
||||
GTK_CSS_PX);
|
||||
case GTK_CSS_REM:
|
||||
return gtk_css_dimension_value_new (number->value *
|
||||
gtk_css_font_size_get_default_px (provider, style),
|
||||
GTK_CSS_PX);
|
||||
case GTK_CSS_RAD:
|
||||
return gtk_css_dimension_value_new (number->value * 360.0 / (2 * G_PI),
|
||||
GTK_CSS_DEG);
|
||||
case GTK_CSS_GRAD:
|
||||
return gtk_css_dimension_value_new (number->value * 360.0 / 400.0,
|
||||
GTK_CSS_DEG);
|
||||
case GTK_CSS_TURN:
|
||||
return gtk_css_dimension_value_new (number->value * 360.0,
|
||||
GTK_CSS_DEG);
|
||||
case GTK_CSS_MS:
|
||||
return gtk_css_dimension_value_new (number->value / 1000.0,
|
||||
GTK_CSS_S);
|
||||
default:
|
||||
g_assert_not_reached();
|
||||
}
|
||||
}
|
||||
|
||||
static gboolean
|
||||
gtk_css_value_dimension_equal (const GtkCssValue *number1,
|
||||
const GtkCssValue *number2)
|
||||
{
|
||||
return number1->unit == number2->unit &&
|
||||
number1->value == number2->value;
|
||||
}
|
||||
|
||||
static void
|
||||
gtk_css_value_dimension_print (const GtkCssValue *number,
|
||||
GString *string)
|
||||
{
|
||||
char buf[G_ASCII_DTOSTR_BUF_SIZE];
|
||||
|
||||
const char *names[] = {
|
||||
/* [GTK_CSS_NUMBER] = */ "",
|
||||
/* [GTK_CSS_PERCENT] = */ "%",
|
||||
/* [GTK_CSS_PX] = */ "px",
|
||||
/* [GTK_CSS_PT] = */ "pt",
|
||||
/* [GTK_CSS_EM] = */ "em",
|
||||
/* [GTK_CSS_EX] = */ "ex",
|
||||
/* [GTK_CSS_REM] = */ "rem",
|
||||
/* [GTK_CSS_PC] = */ "pc",
|
||||
/* [GTK_CSS_IN] = */ "in",
|
||||
/* [GTK_CSS_CM] = */ "cm",
|
||||
/* [GTK_CSS_MM] = */ "mm",
|
||||
/* [GTK_CSS_RAD] = */ "rad",
|
||||
/* [GTK_CSS_DEG] = */ "deg",
|
||||
/* [GTK_CSS_GRAD] = */ "grad",
|
||||
/* [GTK_CSS_TURN] = */ "turn",
|
||||
/* [GTK_CSS_S] = */ "s",
|
||||
/* [GTK_CSS_MS] = */ "ms",
|
||||
};
|
||||
|
||||
if (isinf (number->value))
|
||||
g_string_append (string, "infinite");
|
||||
else
|
||||
{
|
||||
g_ascii_dtostr (buf, sizeof (buf), number->value);
|
||||
g_string_append (string, buf);
|
||||
if (number->value != 0.0)
|
||||
g_string_append (string, names[number->unit]);
|
||||
}
|
||||
}
|
||||
|
||||
static double
|
||||
gtk_css_value_dimension_get (const GtkCssValue *value,
|
||||
double one_hundred_percent)
|
||||
{
|
||||
if (value->unit == GTK_CSS_PERCENT)
|
||||
return value->value * one_hundred_percent / 100;
|
||||
else
|
||||
return value->value;
|
||||
}
|
||||
|
||||
static GtkCssDimension
|
||||
gtk_css_value_dimension_get_dimension (const GtkCssValue *value)
|
||||
{
|
||||
return gtk_css_unit_get_dimension (value->unit);
|
||||
}
|
||||
|
||||
static gboolean
|
||||
gtk_css_value_dimension_has_percent (const GtkCssValue *value)
|
||||
{
|
||||
return gtk_css_unit_get_dimension (value->unit) == GTK_CSS_DIMENSION_PERCENTAGE;
|
||||
}
|
||||
|
||||
static GtkCssValue *
|
||||
gtk_css_value_dimension_multiply (GtkCssValue *value,
|
||||
double factor)
|
||||
{
|
||||
return gtk_css_dimension_value_new (value->value * factor, value->unit);
|
||||
}
|
||||
|
||||
static GtkCssValue *
|
||||
gtk_css_value_dimension_try_add (GtkCssValue *value1,
|
||||
GtkCssValue *value2)
|
||||
{
|
||||
if (value1->unit != value2->unit)
|
||||
return NULL;
|
||||
|
||||
if (value1->value == 0)
|
||||
return _gtk_css_value_ref (value2);
|
||||
|
||||
if (value2->value == 0)
|
||||
return _gtk_css_value_ref (value1);
|
||||
|
||||
return gtk_css_dimension_value_new (value1->value + value2->value, value1->unit);
|
||||
}
|
||||
|
||||
static gint
|
||||
gtk_css_value_dimension_get_calc_term_order (const GtkCssValue *value)
|
||||
{
|
||||
/* note: the order is alphabetic */
|
||||
guint order_per_unit[] = {
|
||||
/* [GTK_CSS_NUMBER] = */ 0,
|
||||
/* [GTK_CSS_PERCENT] = */ 16,
|
||||
/* [GTK_CSS_PX] = */ 11,
|
||||
/* [GTK_CSS_PT] = */ 10,
|
||||
/* [GTK_CSS_EM] = */ 3,
|
||||
/* [GTK_CSS_EX] = */ 4,
|
||||
/* [GTK_CSS_REM] = */ 13,
|
||||
/* [GTK_CSS_PC] = */ 9,
|
||||
/* [GTK_CSS_IN] = */ 6,
|
||||
/* [GTK_CSS_CM] = */ 1,
|
||||
/* [GTK_CSS_MM] = */ 7,
|
||||
/* [GTK_CSS_RAD] = */ 12,
|
||||
/* [GTK_CSS_DEG] = */ 2,
|
||||
/* [GTK_CSS_GRAD] = */ 5,
|
||||
/* [GTK_CSS_TURN] = */ 15,
|
||||
/* [GTK_CSS_S] = */ 14,
|
||||
/* [GTK_CSS_MS] = */ 8
|
||||
};
|
||||
|
||||
return 1000 + order_per_unit[value->unit];
|
||||
}
|
||||
|
||||
static GtkCssValue *
|
||||
gtk_css_value_dimension_transition (GtkCssValue *start,
|
||||
GtkCssValue *end,
|
||||
guint property_id,
|
||||
double progress)
|
||||
{
|
||||
if (start->unit == end->unit)
|
||||
return gtk_css_dimension_value_new (start->value + (end->value - start->value) * progress, start->unit);
|
||||
|
||||
return gtk_css_number_value_transition (start, end, property_id, progress);
|
||||
}
|
||||
|
||||
static const GtkCssNumberValueClass GTK_CSS_VALUE_DIMENSION = {
|
||||
{
|
||||
"GtkCssDimensionValue",
|
||||
gtk_css_value_dimension_free,
|
||||
gtk_css_value_dimension_compute,
|
||||
gtk_css_value_dimension_equal,
|
||||
gtk_css_value_dimension_transition,
|
||||
NULL,
|
||||
NULL,
|
||||
gtk_css_value_dimension_print
|
||||
},
|
||||
gtk_css_value_dimension_get,
|
||||
gtk_css_value_dimension_get_dimension,
|
||||
gtk_css_value_dimension_has_percent,
|
||||
gtk_css_value_dimension_multiply,
|
||||
gtk_css_value_dimension_try_add,
|
||||
gtk_css_value_dimension_get_calc_term_order
|
||||
};
|
||||
|
||||
GtkCssValue *
|
||||
gtk_css_dimension_value_new (double value,
|
||||
GtkCssUnit unit)
|
||||
{
|
||||
static GtkCssValue number_singletons[] = {
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_NUMBER, 0 },
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_NUMBER, 1 },
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_NUMBER, 96 }, /* DPI default */
|
||||
};
|
||||
static GtkCssValue px_singletons[] = {
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_PX, 0 },
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_PX, 1 },
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_PX, 2 },
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_PX, 3 },
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_PX, 4 },
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_PX, 8 },
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_PX, 16 }, /* Icon size default */
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_PX, 32 },
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_PX, 64 },
|
||||
};
|
||||
static GtkCssValue percent_singletons[] = {
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_PERCENT, 0 },
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, FALSE, GTK_CSS_PERCENT, 50 },
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, FALSE, GTK_CSS_PERCENT, 100 },
|
||||
};
|
||||
static GtkCssValue second_singletons[] = {
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_S, 0 },
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_S, 1 },
|
||||
};
|
||||
static GtkCssValue deg_singletons[] = {
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_DEG, 0 },
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_DEG, 90 },
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_DEG, 180 },
|
||||
{ >K_CSS_VALUE_DIMENSION.value_class, 1, TRUE, GTK_CSS_DEG, 270 },
|
||||
};
|
||||
GtkCssValue *result;
|
||||
|
||||
switch ((guint)unit)
|
||||
{
|
||||
case GTK_CSS_NUMBER:
|
||||
if (value == 0 || value == 1)
|
||||
return _gtk_css_value_ref (&number_singletons[(int) value]);
|
||||
|
||||
if (value == 96)
|
||||
return _gtk_css_value_ref (&number_singletons[2]);
|
||||
|
||||
break;
|
||||
|
||||
case GTK_CSS_PX:
|
||||
if (value == 0 ||
|
||||
value == 1 ||
|
||||
value == 2 ||
|
||||
value == 3 ||
|
||||
value == 4)
|
||||
return _gtk_css_value_ref (&px_singletons[(int) value]);
|
||||
if (value == 8)
|
||||
return _gtk_css_value_ref (&px_singletons[5]);
|
||||
if (value == 16)
|
||||
return _gtk_css_value_ref (&px_singletons[6]);
|
||||
if (value == 32)
|
||||
return _gtk_css_value_ref (&px_singletons[7]);
|
||||
if (value == 64)
|
||||
return _gtk_css_value_ref (&px_singletons[8]);
|
||||
|
||||
break;
|
||||
|
||||
case GTK_CSS_PERCENT:
|
||||
if (value == 0)
|
||||
return _gtk_css_value_ref (&percent_singletons[0]);
|
||||
if (value == 50)
|
||||
return _gtk_css_value_ref (&percent_singletons[1]);
|
||||
if (value == 100)
|
||||
return _gtk_css_value_ref (&percent_singletons[2]);
|
||||
|
||||
break;
|
||||
|
||||
case GTK_CSS_S:
|
||||
if (value == 0 || value == 1)
|
||||
return _gtk_css_value_ref (&second_singletons[(int)value]);
|
||||
|
||||
break;
|
||||
|
||||
case GTK_CSS_DEG:
|
||||
if (value == 0)
|
||||
return _gtk_css_value_ref (°_singletons[0]);
|
||||
if (value == 90)
|
||||
return _gtk_css_value_ref (°_singletons[1]);
|
||||
if (value == 180)
|
||||
return _gtk_css_value_ref (°_singletons[2]);
|
||||
if (value == 270)
|
||||
return _gtk_css_value_ref (°_singletons[3]);
|
||||
|
||||
break;
|
||||
|
||||
default:
|
||||
;
|
||||
}
|
||||
|
||||
result = _gtk_css_value_new (GtkCssValue, >K_CSS_VALUE_DIMENSION.value_class);
|
||||
result->unit = unit;
|
||||
result->value = value;
|
||||
result->is_computed = value == 0 ||
|
||||
unit == GTK_CSS_NUMBER ||
|
||||
unit == GTK_CSS_PX ||
|
||||
unit == GTK_CSS_DEG ||
|
||||
unit == GTK_CSS_S;
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
GtkCssValue *
|
||||
gtk_css_dimension_value_parse (GtkCssParser *parser,
|
||||
GtkCssNumberParseFlags flags)
|
||||
@ -495,15 +131,3 @@ gtk_css_dimension_value_parse (GtkCssParser *parser,
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
gboolean
|
||||
gtk_css_dimension_value_is_zero (const GtkCssValue *value)
|
||||
{
|
||||
if (!value)
|
||||
return TRUE;
|
||||
|
||||
if (value->class != >K_CSS_VALUE_DIMENSION.value_class)
|
||||
return FALSE;
|
||||
|
||||
return value->value == 0;
|
||||
}
|
||||
|
@ -24,14 +24,8 @@
|
||||
|
||||
G_BEGIN_DECLS
|
||||
|
||||
GtkCssValue * gtk_css_dimension_value_new (double value,
|
||||
GtkCssUnit unit);
|
||||
|
||||
GtkCssValue * gtk_css_dimension_value_parse (GtkCssParser *parser,
|
||||
GtkCssNumberParseFlags flags);
|
||||
|
||||
gboolean gtk_css_dimension_value_is_zero (const GtkCssValue *value) G_GNUC_PURE;
|
||||
|
||||
G_END_DECLS
|
||||
|
||||
#endif /* __GTK_CSS_DIMENSION_VALUE_PRIVATE_H__ */
|
||||
|
@ -294,7 +294,7 @@ parse_progress (GtkCssParser *parser,
|
||||
|
||||
if (*progress > 1.0)
|
||||
{
|
||||
gtk_css_parser_error_value (parser, "Percentages over 100%% are not allowed");
|
||||
gtk_css_parser_error_value (parser, "Percentages over 100%% are not allowed. Given value: %f", *progress);
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
|
@ -20,39 +20,633 @@
|
||||
#include "gtkcssnumbervalueprivate.h"
|
||||
|
||||
#include "gtkcsscalcvalueprivate.h"
|
||||
#include "gtkcssenumvalueprivate.h"
|
||||
#include "gtkcssdimensionvalueprivate.h"
|
||||
#include "gtkcssstyleprivate.h"
|
||||
#include "gtkprivate.h"
|
||||
|
||||
static GtkCssValue * gtk_css_calc_value_new (guint n_terms);
|
||||
static GtkCssValue * gtk_css_calc_value_new_sum (GtkCssValue *a,
|
||||
GtkCssValue *b);
|
||||
|
||||
enum {
|
||||
TYPE_CALC = 0,
|
||||
TYPE_DIMENSION = 1,
|
||||
};
|
||||
|
||||
struct _GtkCssValue {
|
||||
GTK_CSS_VALUE_BASE
|
||||
guint type : 1; /* Calc or dimension */
|
||||
union {
|
||||
struct {
|
||||
GtkCssUnit unit;
|
||||
double value;
|
||||
} dimension;
|
||||
struct {
|
||||
guint n_terms;
|
||||
GtkCssValue *terms[1];
|
||||
} calc;
|
||||
};
|
||||
};
|
||||
|
||||
|
||||
static GtkCssValue *
|
||||
gtk_css_calc_value_new_from_array (GtkCssValue **values,
|
||||
guint n_values)
|
||||
{
|
||||
GtkCssValue *result;
|
||||
|
||||
if (n_values > 1)
|
||||
{
|
||||
result = gtk_css_calc_value_new (n_values);
|
||||
memcpy (result->calc.terms, values, n_values * sizeof (GtkCssValue *));
|
||||
}
|
||||
else
|
||||
{
|
||||
result = values[0];
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
|
||||
static void
|
||||
gtk_css_value_number_free (GtkCssValue *value)
|
||||
{
|
||||
g_slice_free (GtkCssValue, value);
|
||||
}
|
||||
|
||||
static double
|
||||
get_dpi (GtkCssStyle *style)
|
||||
{
|
||||
return _gtk_css_number_value_get (style->core->dpi, 96);
|
||||
}
|
||||
|
||||
static double
|
||||
get_base_font_size_px (guint property_id,
|
||||
GtkStyleProvider *provider,
|
||||
GtkCssStyle *style,
|
||||
GtkCssStyle *parent_style)
|
||||
{
|
||||
if (property_id == GTK_CSS_PROPERTY_FONT_SIZE)
|
||||
{
|
||||
if (parent_style)
|
||||
return _gtk_css_number_value_get (parent_style->core->font_size, 100);
|
||||
else
|
||||
return gtk_css_font_size_get_default_px (provider, style);
|
||||
}
|
||||
|
||||
return _gtk_css_number_value_get (style->core->font_size, 100);
|
||||
}
|
||||
|
||||
static GtkCssValue *
|
||||
gtk_css_value_number_compute (GtkCssValue *number,
|
||||
guint property_id,
|
||||
GtkStyleProvider *provider,
|
||||
GtkCssStyle *style,
|
||||
GtkCssStyle *parent_style)
|
||||
{
|
||||
double value;
|
||||
|
||||
if (G_UNLIKELY (number->type == TYPE_CALC))
|
||||
{
|
||||
const guint n_terms = number->calc.n_terms;
|
||||
GtkCssValue *result;
|
||||
gboolean changed = FALSE;
|
||||
gsize i;
|
||||
GtkCssValue **new_values;
|
||||
|
||||
new_values = g_alloca (sizeof (GtkCssValue *) * n_terms);
|
||||
|
||||
for (i = 0; i < n_terms; i++)
|
||||
{
|
||||
GtkCssValue *computed = _gtk_css_value_compute (number->calc.terms[i],
|
||||
property_id, provider, style,
|
||||
parent_style);
|
||||
changed |= computed != number->calc.terms[i];
|
||||
new_values[i] = computed;
|
||||
}
|
||||
|
||||
if (changed)
|
||||
{
|
||||
result = gtk_css_calc_value_new_from_array (new_values, n_terms);
|
||||
}
|
||||
else
|
||||
{
|
||||
for (i = 0; i < n_terms; i++)
|
||||
gtk_css_value_unref (new_values[i]);
|
||||
|
||||
result = _gtk_css_value_ref (number);
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
g_assert (number->type == TYPE_DIMENSION);
|
||||
|
||||
value = number->dimension.value;
|
||||
switch (number->dimension.unit)
|
||||
{
|
||||
case GTK_CSS_PERCENT:
|
||||
/* percentages for font sizes are computed, other percentages aren't */
|
||||
if (property_id == GTK_CSS_PROPERTY_FONT_SIZE)
|
||||
return gtk_css_dimension_value_new (value / 100.0 *
|
||||
get_base_font_size_px (property_id, provider, style, parent_style),
|
||||
GTK_CSS_PX);
|
||||
G_GNUC_FALLTHROUGH;
|
||||
case GTK_CSS_NUMBER:
|
||||
case GTK_CSS_PX:
|
||||
case GTK_CSS_DEG:
|
||||
case GTK_CSS_S:
|
||||
return _gtk_css_value_ref (number);
|
||||
case GTK_CSS_PT:
|
||||
return gtk_css_dimension_value_new (value * get_dpi (style) / 72.0,
|
||||
GTK_CSS_PX);
|
||||
case GTK_CSS_PC:
|
||||
return gtk_css_dimension_value_new (value * get_dpi (style) / 72.0 * 12.0,
|
||||
GTK_CSS_PX);
|
||||
case GTK_CSS_IN:
|
||||
return gtk_css_dimension_value_new (value * get_dpi (style),
|
||||
GTK_CSS_PX);
|
||||
case GTK_CSS_CM:
|
||||
return gtk_css_dimension_value_new (value * get_dpi (style) * 0.39370078740157477,
|
||||
GTK_CSS_PX);
|
||||
case GTK_CSS_MM:
|
||||
return gtk_css_dimension_value_new (value * get_dpi (style) * 0.039370078740157477,
|
||||
GTK_CSS_PX);
|
||||
case GTK_CSS_EM:
|
||||
return gtk_css_dimension_value_new (value *
|
||||
get_base_font_size_px (property_id, provider, style, parent_style),
|
||||
GTK_CSS_PX);
|
||||
case GTK_CSS_EX:
|
||||
/* for now we pretend ex is half of em */
|
||||
return gtk_css_dimension_value_new (value * 0.5 *
|
||||
get_base_font_size_px (property_id, provider, style, parent_style),
|
||||
GTK_CSS_PX);
|
||||
case GTK_CSS_REM:
|
||||
return gtk_css_dimension_value_new (value *
|
||||
gtk_css_font_size_get_default_px (provider, style),
|
||||
GTK_CSS_PX);
|
||||
case GTK_CSS_RAD:
|
||||
return gtk_css_dimension_value_new (value * 360.0 / (2 * G_PI),
|
||||
GTK_CSS_DEG);
|
||||
case GTK_CSS_GRAD:
|
||||
return gtk_css_dimension_value_new (value * 360.0 / 400.0,
|
||||
GTK_CSS_DEG);
|
||||
case GTK_CSS_TURN:
|
||||
return gtk_css_dimension_value_new (value * 360.0,
|
||||
GTK_CSS_DEG);
|
||||
case GTK_CSS_MS:
|
||||
return gtk_css_dimension_value_new (value / 1000.0,
|
||||
GTK_CSS_S);
|
||||
default:
|
||||
g_assert_not_reached();
|
||||
}
|
||||
}
|
||||
|
||||
static gboolean
|
||||
gtk_css_value_number_equal (const GtkCssValue *val1,
|
||||
const GtkCssValue *val2)
|
||||
{
|
||||
guint i;
|
||||
|
||||
if (val1->type != val2->type)
|
||||
return FALSE;
|
||||
|
||||
if (G_LIKELY (val1->type == TYPE_DIMENSION))
|
||||
{
|
||||
return val1->dimension.unit == val2->dimension.unit &&
|
||||
val1->dimension.value == val2->dimension.value;
|
||||
}
|
||||
|
||||
g_assert (val1->type == TYPE_CALC);
|
||||
|
||||
if (val1->calc.n_terms != val2->calc.n_terms)
|
||||
return FALSE;
|
||||
|
||||
for (i = 0; i < val1->calc.n_terms; i++)
|
||||
{
|
||||
if (!_gtk_css_value_equal (val1->calc.terms[i], val2->calc.terms[i]))
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static void
|
||||
gtk_css_value_number_print (const GtkCssValue *value,
|
||||
GString *string)
|
||||
{
|
||||
guint i;
|
||||
|
||||
if (G_LIKELY (value->type == TYPE_DIMENSION))
|
||||
{
|
||||
const char *names[] = {
|
||||
/* [GTK_CSS_NUMBER] = */ "",
|
||||
/* [GTK_CSS_PERCENT] = */ "%",
|
||||
/* [GTK_CSS_PX] = */ "px",
|
||||
/* [GTK_CSS_PT] = */ "pt",
|
||||
/* [GTK_CSS_EM] = */ "em",
|
||||
/* [GTK_CSS_EX] = */ "ex",
|
||||
/* [GTK_CSS_REM] = */ "rem",
|
||||
/* [GTK_CSS_PC] = */ "pc",
|
||||
/* [GTK_CSS_IN] = */ "in",
|
||||
/* [GTK_CSS_CM] = */ "cm",
|
||||
/* [GTK_CSS_MM] = */ "mm",
|
||||
/* [GTK_CSS_RAD] = */ "rad",
|
||||
/* [GTK_CSS_DEG] = */ "deg",
|
||||
/* [GTK_CSS_GRAD] = */ "grad",
|
||||
/* [GTK_CSS_TURN] = */ "turn",
|
||||
/* [GTK_CSS_S] = */ "s",
|
||||
/* [GTK_CSS_MS] = */ "ms",
|
||||
};
|
||||
char buf[G_ASCII_DTOSTR_BUF_SIZE];
|
||||
|
||||
if (isinf (value->dimension.value))
|
||||
g_string_append (string, "infinite");
|
||||
else
|
||||
{
|
||||
g_ascii_dtostr (buf, sizeof (buf), value->dimension.value);
|
||||
g_string_append (string, buf);
|
||||
if (value->dimension.value != 0.0)
|
||||
g_string_append (string, names[value->dimension.unit]);
|
||||
}
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
g_assert (value->type == TYPE_CALC);
|
||||
|
||||
g_string_append (string, "calc(");
|
||||
_gtk_css_value_print (value->calc.terms[0], string);
|
||||
|
||||
for (i = 1; i < value->calc.n_terms; i++)
|
||||
{
|
||||
g_string_append (string, " + ");
|
||||
_gtk_css_value_print (value->calc.terms[i], string);
|
||||
}
|
||||
g_string_append (string, ")");
|
||||
}
|
||||
|
||||
static GtkCssValue *
|
||||
gtk_css_value_number_transition (GtkCssValue *start,
|
||||
GtkCssValue *end,
|
||||
guint property_id,
|
||||
double progress)
|
||||
{
|
||||
GtkCssValue *result, *mul_start, *mul_end;
|
||||
|
||||
if (start == end)
|
||||
return _gtk_css_value_ref (start);
|
||||
|
||||
if (G_LIKELY (start->type == TYPE_DIMENSION && end->type == TYPE_DIMENSION))
|
||||
{
|
||||
if (start->dimension.unit == end->dimension.unit)
|
||||
{
|
||||
const double start_value = start->dimension.value;
|
||||
const double end_value = end->dimension.value;
|
||||
return gtk_css_dimension_value_new (start_value + (end_value - start_value) * progress,
|
||||
start->dimension.unit);
|
||||
}
|
||||
}
|
||||
|
||||
mul_start = gtk_css_number_value_multiply (start, 1 - progress);
|
||||
mul_end = gtk_css_number_value_multiply (end, progress);
|
||||
|
||||
result = gtk_css_number_value_add (mul_start, mul_end);
|
||||
|
||||
_gtk_css_value_unref (mul_start);
|
||||
_gtk_css_value_unref (mul_end);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
static const GtkCssValueClass GTK_CSS_VALUE_NUMBER = {
|
||||
"GtkCssNumberValue",
|
||||
gtk_css_value_number_free,
|
||||
gtk_css_value_number_compute,
|
||||
gtk_css_value_number_equal,
|
||||
gtk_css_value_number_transition,
|
||||
NULL,
|
||||
NULL,
|
||||
gtk_css_value_number_print
|
||||
};
|
||||
|
||||
static gsize
|
||||
gtk_css_value_calc_get_size (gsize n_terms)
|
||||
{
|
||||
g_assert (n_terms > 0);
|
||||
|
||||
return sizeof (GtkCssValue) + sizeof (GtkCssValue *) * (n_terms - 1);
|
||||
}
|
||||
|
||||
static GtkCssValue *
|
||||
gtk_css_calc_value_new (guint n_terms)
|
||||
{
|
||||
GtkCssValue *result;
|
||||
|
||||
result = _gtk_css_value_alloc (>K_CSS_VALUE_NUMBER,
|
||||
gtk_css_value_calc_get_size (n_terms));
|
||||
result->type = TYPE_CALC;
|
||||
result->calc.n_terms = n_terms;
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
GtkCssValue *
|
||||
gtk_css_dimension_value_new (double value,
|
||||
GtkCssUnit unit)
|
||||
{
|
||||
static GtkCssValue number_singletons[] = {
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_NUMBER, 0 }} },
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_NUMBER, 1 }} },
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_NUMBER, 96 }} }, /* DPI default */
|
||||
};
|
||||
static GtkCssValue px_singletons[] = {
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_PX, 0 }} },
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_PX, 1 }} },
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_PX, 2 }} },
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_PX, 3 }} },
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_PX, 4 }} },
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_PX, 8 }} },
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_PX, 16 }} }, /* Icon size default */
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_PX, 32 }} },
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_PX, 64 }} },
|
||||
};
|
||||
static GtkCssValue percent_singletons[] = {
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_PERCENT, 0 }} },
|
||||
{ >K_CSS_VALUE_NUMBER, 1, FALSE, TYPE_DIMENSION, {{ GTK_CSS_PERCENT, 50 }} },
|
||||
{ >K_CSS_VALUE_NUMBER, 1, FALSE, TYPE_DIMENSION, {{ GTK_CSS_PERCENT, 100 }} },
|
||||
};
|
||||
static GtkCssValue second_singletons[] = {
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_S, 0 }} },
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_S, 1 }} },
|
||||
};
|
||||
static GtkCssValue deg_singletons[] = {
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_DEG, 0 }} },
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_DEG, 90 }} },
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_DEG, 180 }} },
|
||||
{ >K_CSS_VALUE_NUMBER, 1, TRUE, TYPE_DIMENSION, {{ GTK_CSS_DEG, 270 }} },
|
||||
};
|
||||
GtkCssValue *result;
|
||||
|
||||
switch ((guint)unit)
|
||||
{
|
||||
case GTK_CSS_NUMBER:
|
||||
if (value == 0 || value == 1)
|
||||
return _gtk_css_value_ref (&number_singletons[(int) value]);
|
||||
|
||||
if (value == 96)
|
||||
return _gtk_css_value_ref (&number_singletons[2]);
|
||||
|
||||
break;
|
||||
|
||||
case GTK_CSS_PX:
|
||||
if (value == 0 ||
|
||||
value == 1 ||
|
||||
value == 2 ||
|
||||
value == 3 ||
|
||||
value == 4)
|
||||
return _gtk_css_value_ref (&px_singletons[(int) value]);
|
||||
if (value == 8)
|
||||
return _gtk_css_value_ref (&px_singletons[5]);
|
||||
if (value == 16)
|
||||
return _gtk_css_value_ref (&px_singletons[6]);
|
||||
if (value == 32)
|
||||
return _gtk_css_value_ref (&px_singletons[7]);
|
||||
if (value == 64)
|
||||
return _gtk_css_value_ref (&px_singletons[8]);
|
||||
|
||||
break;
|
||||
|
||||
case GTK_CSS_PERCENT:
|
||||
if (value == 0)
|
||||
return _gtk_css_value_ref (&percent_singletons[0]);
|
||||
if (value == 50)
|
||||
return _gtk_css_value_ref (&percent_singletons[1]);
|
||||
if (value == 100)
|
||||
return _gtk_css_value_ref (&percent_singletons[2]);
|
||||
|
||||
break;
|
||||
|
||||
case GTK_CSS_S:
|
||||
if (value == 0 || value == 1)
|
||||
return _gtk_css_value_ref (&second_singletons[(int)value]);
|
||||
|
||||
break;
|
||||
|
||||
case GTK_CSS_DEG:
|
||||
if (value == 0)
|
||||
return _gtk_css_value_ref (°_singletons[0]);
|
||||
if (value == 90)
|
||||
return _gtk_css_value_ref (°_singletons[1]);
|
||||
if (value == 180)
|
||||
return _gtk_css_value_ref (°_singletons[2]);
|
||||
if (value == 270)
|
||||
return _gtk_css_value_ref (°_singletons[3]);
|
||||
|
||||
break;
|
||||
|
||||
default:
|
||||
;
|
||||
}
|
||||
|
||||
result = _gtk_css_value_new (GtkCssValue, >K_CSS_VALUE_NUMBER);
|
||||
result->type = TYPE_DIMENSION;
|
||||
result->dimension.unit = unit;
|
||||
result->dimension.value = value;
|
||||
result->is_computed = value == 0 ||
|
||||
unit == GTK_CSS_NUMBER ||
|
||||
unit == GTK_CSS_PX ||
|
||||
unit == GTK_CSS_DEG ||
|
||||
unit == GTK_CSS_S;
|
||||
if (value == 5040)
|
||||
g_critical ("%s: %f", __FUNCTION__, value);
|
||||
return result;
|
||||
}
|
||||
|
||||
/*
|
||||
* gtk_css_number_value_get_calc_term_order:
|
||||
* @value: Value to compute order for
|
||||
*
|
||||
* Determines the position of @value when printed as part of a calc()
|
||||
* expression. Values with lower numbers are printed first. Note that
|
||||
* these numbers are arbitrary, so when adding new types of values to
|
||||
* print, feel free to change them in implementations so that they
|
||||
* match.
|
||||
*
|
||||
* Returns: Magic value determining placement when printing calc()
|
||||
* expression.
|
||||
*/
|
||||
static int
|
||||
gtk_css_number_value_get_calc_term_order (const GtkCssValue *value)
|
||||
{
|
||||
if (G_LIKELY (value->type == TYPE_DIMENSION))
|
||||
{
|
||||
/* note: the order is alphabetic */
|
||||
static const guint order_per_unit[] = {
|
||||
/* [GTK_CSS_NUMBER] = */ 0,
|
||||
/* [GTK_CSS_PERCENT] = */ 16,
|
||||
/* [GTK_CSS_PX] = */ 11,
|
||||
/* [GTK_CSS_PT] = */ 10,
|
||||
/* [GTK_CSS_EM] = */ 3,
|
||||
/* [GTK_CSS_EX] = */ 4,
|
||||
/* [GTK_CSS_REM] = */ 13,
|
||||
/* [GTK_CSS_PC] = */ 9,
|
||||
/* [GTK_CSS_IN] = */ 6,
|
||||
/* [GTK_CSS_CM] = */ 1,
|
||||
/* [GTK_CSS_MM] = */ 7,
|
||||
/* [GTK_CSS_RAD] = */ 12,
|
||||
/* [GTK_CSS_DEG] = */ 2,
|
||||
/* [GTK_CSS_GRAD] = */ 5,
|
||||
/* [GTK_CSS_TURN] = */ 15,
|
||||
/* [GTK_CSS_S] = */ 14,
|
||||
/* [GTK_CSS_MS] = */ 8
|
||||
};
|
||||
return 1000 + order_per_unit[value->dimension.unit];
|
||||
}
|
||||
|
||||
g_assert (value->type == TYPE_CALC);
|
||||
|
||||
/* This should never be needed because calc() can't contain calc(),
|
||||
* but eh...
|
||||
*/
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void
|
||||
gtk_css_calc_array_add (GPtrArray *array, GtkCssValue *value)
|
||||
{
|
||||
gsize i;
|
||||
gint calc_term_order;
|
||||
|
||||
calc_term_order = gtk_css_number_value_get_calc_term_order (value);
|
||||
|
||||
for (i = 0; i < array->len; i++)
|
||||
{
|
||||
GtkCssValue *sum = gtk_css_number_value_try_add (g_ptr_array_index (array, i), value);
|
||||
|
||||
if (sum)
|
||||
{
|
||||
g_ptr_array_index (array, i) = sum;
|
||||
_gtk_css_value_unref (value);
|
||||
return;
|
||||
}
|
||||
else if (gtk_css_number_value_get_calc_term_order (g_ptr_array_index (array, i)) > calc_term_order)
|
||||
{
|
||||
g_ptr_array_insert (array, i, value);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
g_ptr_array_add (array, value);
|
||||
}
|
||||
|
||||
static GtkCssValue *
|
||||
gtk_css_calc_value_new_sum (GtkCssValue *value1,
|
||||
GtkCssValue *value2)
|
||||
{
|
||||
GPtrArray *array;
|
||||
GtkCssValue *result;
|
||||
gsize i;
|
||||
|
||||
array = g_ptr_array_new ();
|
||||
|
||||
if (value1->class == >K_CSS_VALUE_NUMBER &&
|
||||
value1->type == TYPE_CALC)
|
||||
{
|
||||
for (i = 0; i < value1->calc.n_terms; i++)
|
||||
{
|
||||
gtk_css_calc_array_add (array, _gtk_css_value_ref (value1->calc.terms[i]));
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
gtk_css_calc_array_add (array, _gtk_css_value_ref (value1));
|
||||
}
|
||||
|
||||
if (value2->class == >K_CSS_VALUE_NUMBER &&
|
||||
value2->type == TYPE_CALC)
|
||||
{
|
||||
for (i = 0; i < value2->calc.n_terms; i++)
|
||||
{
|
||||
gtk_css_calc_array_add (array, _gtk_css_value_ref (value2->calc.terms[i]));
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
gtk_css_calc_array_add (array, _gtk_css_value_ref (value2));
|
||||
}
|
||||
|
||||
result = gtk_css_calc_value_new_from_array ((GtkCssValue **)array->pdata, array->len);
|
||||
g_ptr_array_free (array, TRUE);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
GtkCssDimension
|
||||
gtk_css_number_value_get_dimension (const GtkCssValue *value)
|
||||
{
|
||||
GtkCssNumberValueClass *number_value_class = (GtkCssNumberValueClass *) value->class;
|
||||
GtkCssDimension dimension = GTK_CSS_DIMENSION_PERCENTAGE;
|
||||
guint i;
|
||||
|
||||
return number_value_class->get_dimension (value);
|
||||
if (G_LIKELY (value->type == TYPE_DIMENSION))
|
||||
return gtk_css_unit_get_dimension (value->dimension.unit);
|
||||
|
||||
g_assert (value->type == TYPE_CALC);
|
||||
|
||||
for (i = 0; i < value->calc.n_terms && dimension == GTK_CSS_DIMENSION_PERCENTAGE; i++)
|
||||
{
|
||||
dimension = gtk_css_number_value_get_dimension (value->calc.terms[i]);
|
||||
if (dimension != GTK_CSS_DIMENSION_PERCENTAGE)
|
||||
break;
|
||||
}
|
||||
|
||||
return dimension;
|
||||
}
|
||||
|
||||
gboolean
|
||||
gtk_css_number_value_has_percent (const GtkCssValue *value)
|
||||
{
|
||||
GtkCssNumberValueClass *number_value_class = (GtkCssNumberValueClass *) value->class;
|
||||
guint i;
|
||||
|
||||
return number_value_class->has_percent (value);
|
||||
if (G_LIKELY (value->type == TYPE_DIMENSION))
|
||||
{
|
||||
return gtk_css_unit_get_dimension (value->dimension.unit) == GTK_CSS_DIMENSION_PERCENTAGE;
|
||||
}
|
||||
|
||||
for (i = 0; i < value->calc.n_terms; i++)
|
||||
{
|
||||
if (gtk_css_number_value_has_percent (value->calc.terms[i]))
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
GtkCssValue *
|
||||
gtk_css_number_value_multiply (GtkCssValue *value,
|
||||
double factor)
|
||||
{
|
||||
GtkCssNumberValueClass *number_value_class = (GtkCssNumberValueClass *) value->class;
|
||||
GtkCssValue *result;
|
||||
guint i;
|
||||
|
||||
if (factor == 1)
|
||||
return _gtk_css_value_ref (value);
|
||||
|
||||
return number_value_class->multiply (value, factor);
|
||||
if (G_LIKELY (value->type == TYPE_DIMENSION))
|
||||
{
|
||||
return gtk_css_dimension_value_new (value->dimension.value * factor,
|
||||
value->dimension.unit);
|
||||
}
|
||||
|
||||
g_assert (value->type == TYPE_CALC);
|
||||
|
||||
result = gtk_css_calc_value_new (value->calc.n_terms);
|
||||
for (i = 0; i < value->calc.n_terms; i++)
|
||||
result->calc.terms[i] = gtk_css_number_value_multiply (value->calc.terms[i], factor);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
GtkCssValue *
|
||||
@ -72,35 +666,27 @@ GtkCssValue *
|
||||
gtk_css_number_value_try_add (GtkCssValue *value1,
|
||||
GtkCssValue *value2)
|
||||
{
|
||||
GtkCssNumberValueClass *number_value_class;
|
||||
|
||||
if (value1->class != value2->class)
|
||||
if (G_UNLIKELY (value1->type != value2->type))
|
||||
return NULL;
|
||||
|
||||
number_value_class = (GtkCssNumberValueClass *) value1->class;
|
||||
if (G_LIKELY (value1->type == TYPE_DIMENSION))
|
||||
{
|
||||
if (value1->dimension.unit != value2->dimension.unit)
|
||||
return NULL;
|
||||
|
||||
return number_value_class->try_add (value1, value2);
|
||||
}
|
||||
if (value1->dimension.value == 0)
|
||||
return _gtk_css_value_ref (value2);
|
||||
|
||||
/*
|
||||
* gtk_css_number_value_get_calc_term_order:
|
||||
* @value: Value to compute order for
|
||||
*
|
||||
* Determines the position of @value when printed as part of a calc()
|
||||
* expression. Values with lower numbers are printed first. Note that
|
||||
* these numbers are arbitrary, so when adding new types of values to
|
||||
* print, feel free to change them in implementations so that they
|
||||
* match.
|
||||
*
|
||||
* Returns: Magic value determining placement when printing calc()
|
||||
* expression.
|
||||
*/
|
||||
gint
|
||||
gtk_css_number_value_get_calc_term_order (const GtkCssValue *value)
|
||||
{
|
||||
GtkCssNumberValueClass *number_value_class = (GtkCssNumberValueClass *) value->class;
|
||||
if (value2->dimension.value == 0)
|
||||
return _gtk_css_value_ref (value1);
|
||||
|
||||
return number_value_class->get_calc_term_order (value);
|
||||
return gtk_css_dimension_value_new (value1->dimension.value + value2->dimension.value,
|
||||
value1->dimension.unit);
|
||||
}
|
||||
|
||||
g_assert (value1->type == TYPE_CALC);
|
||||
/* Not possible for calc() values */
|
||||
return NULL;
|
||||
}
|
||||
|
||||
GtkCssValue *
|
||||
@ -110,28 +696,6 @@ _gtk_css_number_value_new (double value,
|
||||
return gtk_css_dimension_value_new (value, unit);
|
||||
}
|
||||
|
||||
GtkCssValue *
|
||||
gtk_css_number_value_transition (GtkCssValue *start,
|
||||
GtkCssValue *end,
|
||||
guint property_id,
|
||||
double progress)
|
||||
{
|
||||
GtkCssValue *result, *mul_start, *mul_end;
|
||||
|
||||
if (start == end)
|
||||
return _gtk_css_value_ref (start);
|
||||
|
||||
mul_start = gtk_css_number_value_multiply (start, 1 - progress);
|
||||
mul_end = gtk_css_number_value_multiply (end, progress);
|
||||
|
||||
result = gtk_css_number_value_add (mul_start, mul_end);
|
||||
|
||||
_gtk_css_value_unref (mul_start);
|
||||
_gtk_css_value_unref (mul_end);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
gboolean
|
||||
gtk_css_number_value_can_parse (GtkCssParser *parser)
|
||||
{
|
||||
@ -157,15 +721,40 @@ _gtk_css_number_value_parse (GtkCssParser *parser,
|
||||
}
|
||||
|
||||
double
|
||||
_gtk_css_number_value_get (const GtkCssValue *number,
|
||||
_gtk_css_number_value_get (const GtkCssValue *value,
|
||||
double one_hundred_percent)
|
||||
{
|
||||
GtkCssNumberValueClass *number_value_class;
|
||||
double result;
|
||||
guint i;
|
||||
|
||||
gtk_internal_return_val_if_fail (number != NULL, 0.0);
|
||||
if (G_LIKELY (value->type == TYPE_DIMENSION))
|
||||
{
|
||||
if (value->dimension.unit == GTK_CSS_PERCENT)
|
||||
return value->dimension.value * one_hundred_percent / 100;
|
||||
else
|
||||
return value->dimension.value;
|
||||
}
|
||||
|
||||
number_value_class = (GtkCssNumberValueClass *) number->class;
|
||||
g_assert (value->type == TYPE_CALC);
|
||||
|
||||
return number_value_class->get (number, one_hundred_percent);
|
||||
result = 0.0;
|
||||
for (i = 0; i < value->calc.n_terms; i++)
|
||||
result += _gtk_css_number_value_get (value->calc.terms[i], one_hundred_percent);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
gboolean
|
||||
gtk_css_dimension_value_is_zero (const GtkCssValue *value)
|
||||
{
|
||||
if (!value)
|
||||
return TRUE;
|
||||
|
||||
if (value->class != >K_CSS_VALUE_NUMBER)
|
||||
return FALSE;
|
||||
|
||||
if (value->type != TYPE_DIMENSION)
|
||||
return FALSE;
|
||||
|
||||
return value->dimension.value == 0;
|
||||
}
|
||||
|
@ -35,28 +35,11 @@ typedef enum /*< skip >*/ {
|
||||
GTK_CSS_PARSE_TIME = (1 << 5)
|
||||
} GtkCssNumberParseFlags;
|
||||
|
||||
typedef struct _GtkCssNumberValueClass GtkCssNumberValueClass;
|
||||
|
||||
struct _GtkCssNumberValueClass {
|
||||
GtkCssValueClass value_class;
|
||||
|
||||
double (* get) (const GtkCssValue *value,
|
||||
double one_hundred_percent);
|
||||
GtkCssDimension (* get_dimension) (const GtkCssValue *value);
|
||||
gboolean (* has_percent) (const GtkCssValue *value);
|
||||
GtkCssValue * (* multiply) (GtkCssValue *value,
|
||||
double factor);
|
||||
GtkCssValue * (* try_add) (GtkCssValue *value1,
|
||||
GtkCssValue *value2);
|
||||
gint (* get_calc_term_order) (const GtkCssValue *value);
|
||||
};
|
||||
GtkCssValue * gtk_css_dimension_value_new (double value,
|
||||
GtkCssUnit unit);
|
||||
|
||||
GtkCssValue * _gtk_css_number_value_new (double value,
|
||||
GtkCssUnit unit);
|
||||
GtkCssValue * gtk_css_number_value_transition (GtkCssValue *start,
|
||||
GtkCssValue *end,
|
||||
guint property_id,
|
||||
double progress);
|
||||
gboolean gtk_css_number_value_can_parse (GtkCssParser *parser);
|
||||
GtkCssValue * _gtk_css_number_value_parse (GtkCssParser *parser,
|
||||
GtkCssNumberParseFlags flags);
|
||||
@ -69,11 +52,10 @@ GtkCssValue * gtk_css_number_value_add (GtkCssValue *val
|
||||
GtkCssValue *value2);
|
||||
GtkCssValue * gtk_css_number_value_try_add (GtkCssValue *value1,
|
||||
GtkCssValue *value2);
|
||||
gint gtk_css_number_value_get_calc_term_order (const GtkCssValue *value);
|
||||
|
||||
double _gtk_css_number_value_get (const GtkCssValue *number,
|
||||
double one_hundred_percent) G_GNUC_PURE;
|
||||
|
||||
gboolean gtk_css_dimension_value_is_zero (const GtkCssValue *value) G_GNUC_PURE;
|
||||
|
||||
G_END_DECLS
|
||||
|
||||
|
@ -263,11 +263,6 @@ _gtk_css_value_transition (GtkCssValue *start,
|
||||
gtk_internal_return_val_if_fail (start != NULL, FALSE);
|
||||
gtk_internal_return_val_if_fail (end != NULL, FALSE);
|
||||
|
||||
/* We compare functions here instead of classes so that number
|
||||
* values can all transition to each other */
|
||||
if (start->class->transition != end->class->transition)
|
||||
return NULL;
|
||||
|
||||
if (progress == 0)
|
||||
return _gtk_css_value_ref (start);
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user