mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-04 01:31:13 +00:00
css: Use an array when parsing selectors
There's no need for us to have a hard size-limit here; with GdkArray we don't have to give up on stack-preallocation either.
This commit is contained in:
parent
635492b016
commit
ffc77ca926
@ -43,6 +43,12 @@
|
|||||||
#include "gdk/gdkprofilerprivate.h"
|
#include "gdk/gdkprofilerprivate.h"
|
||||||
#include <cairo-gobject.h>
|
#include <cairo-gobject.h>
|
||||||
|
|
||||||
|
#define GDK_ARRAY_NAME gtk_css_selectors
|
||||||
|
#define GDK_ARRAY_TYPE_NAME GtkCssSelectors
|
||||||
|
#define GDK_ARRAY_ELEMENT_TYPE GtkCssSelector *
|
||||||
|
#define GDK_ARRAY_PREALLOC 64
|
||||||
|
#include "gdk/gdkarrayimpl.c"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* SECTION:gtkcssprovider
|
* SECTION:gtkcssprovider
|
||||||
* @Short_description: CSS-like styling for widgets
|
* @Short_description: CSS-like styling for widgets
|
||||||
@ -551,8 +557,7 @@ gtk_css_provider_new (void)
|
|||||||
|
|
||||||
static void
|
static void
|
||||||
css_provider_commit (GtkCssProvider *css_provider,
|
css_provider_commit (GtkCssProvider *css_provider,
|
||||||
GtkCssSelector **selectors,
|
GtkCssSelectors *selectors,
|
||||||
guint n_selectors,
|
|
||||||
GtkCssRuleset *ruleset)
|
GtkCssRuleset *ruleset)
|
||||||
{
|
{
|
||||||
GtkCssProviderPrivate *priv = gtk_css_provider_get_instance_private (css_provider);
|
GtkCssProviderPrivate *priv = gtk_css_provider_get_instance_private (css_provider);
|
||||||
@ -560,19 +565,19 @@ css_provider_commit (GtkCssProvider *css_provider,
|
|||||||
|
|
||||||
if (ruleset->styles == NULL)
|
if (ruleset->styles == NULL)
|
||||||
{
|
{
|
||||||
for (i = 0; i < n_selectors; i++)
|
for (i = 0; i < gtk_css_selectors_get_size (selectors); i++)
|
||||||
_gtk_css_selector_free (selectors[i]);
|
_gtk_css_selector_free (gtk_css_selectors_get (selectors, i));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (i = 0; i < n_selectors; i++)
|
for (i = 0; i < gtk_css_selectors_get_size (selectors); i++)
|
||||||
{
|
{
|
||||||
GtkCssRuleset *new;
|
GtkCssRuleset *new;
|
||||||
|
|
||||||
g_array_set_size (priv->rulesets, priv->rulesets->len + 1);
|
g_array_set_size (priv->rulesets, priv->rulesets->len + 1);
|
||||||
|
|
||||||
new = &g_array_index (priv->rulesets, GtkCssRuleset, priv->rulesets->len - 1);
|
new = &g_array_index (priv->rulesets, GtkCssRuleset, priv->rulesets->len - 1);
|
||||||
gtk_css_ruleset_init_copy (new, ruleset, selectors[i]);
|
gtk_css_ruleset_init_copy (new, ruleset, gtk_css_selectors_get (selectors, i));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -758,38 +763,24 @@ parse_at_keyword (GtkCssScanner *scanner)
|
|||||||
gtk_css_parser_end_block (scanner->parser);
|
gtk_css_parser_end_block (scanner->parser);
|
||||||
}
|
}
|
||||||
|
|
||||||
static guint
|
static void
|
||||||
parse_selector_list (GtkCssScanner *scanner,
|
parse_selector_list (GtkCssScanner *scanner,
|
||||||
GtkCssSelector *out_selectors[MAX_SELECTOR_LIST_LENGTH])
|
GtkCssSelectors *selectors)
|
||||||
{
|
{
|
||||||
guint n_selectors = 0;
|
|
||||||
|
|
||||||
do {
|
do {
|
||||||
GtkCssSelector *select = _gtk_css_selector_parse (scanner->parser);
|
GtkCssSelector *select = _gtk_css_selector_parse (scanner->parser);
|
||||||
|
|
||||||
if (select == NULL)
|
if (select == NULL)
|
||||||
{
|
{
|
||||||
for (int i = 0; i < n_selectors; i++)
|
for (int i = 0; i < gtk_css_selectors_get_size (selectors); i++)
|
||||||
g_clear_pointer (&out_selectors[i], _gtk_css_selector_free);
|
_gtk_css_selector_free (gtk_css_selectors_get (selectors, i));
|
||||||
return 0;
|
gtk_css_selectors_clear (selectors);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
out_selectors[n_selectors] = select;
|
gtk_css_selectors_append (selectors, select);
|
||||||
n_selectors++;
|
|
||||||
|
|
||||||
if (G_UNLIKELY (n_selectors > MAX_SELECTOR_LIST_LENGTH))
|
|
||||||
{
|
|
||||||
gtk_css_parser_error_syntax (scanner->parser,
|
|
||||||
"Only %u selectors per ruleset allowed",
|
|
||||||
MAX_SELECTOR_LIST_LENGTH);
|
|
||||||
for (int i = 0; i < MAX_SELECTOR_LIST_LENGTH; i++)
|
|
||||||
g_clear_pointer (&out_selectors[i], _gtk_css_selector_free);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
while (gtk_css_parser_try_token (scanner->parser, GTK_CSS_TOKEN_COMMA));
|
while (gtk_css_parser_try_token (scanner->parser, GTK_CSS_TOKEN_COMMA));
|
||||||
|
|
||||||
return n_selectors;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -899,27 +890,28 @@ parse_declarations (GtkCssScanner *scanner,
|
|||||||
static void
|
static void
|
||||||
parse_ruleset (GtkCssScanner *scanner)
|
parse_ruleset (GtkCssScanner *scanner)
|
||||||
{
|
{
|
||||||
GtkCssSelector *selectors[MAX_SELECTOR_LIST_LENGTH];
|
GtkCssSelectors selectors;
|
||||||
guint n_selectors;
|
|
||||||
GtkCssRuleset ruleset = { 0, };
|
GtkCssRuleset ruleset = { 0, };
|
||||||
|
|
||||||
n_selectors = parse_selector_list (scanner, selectors);
|
gtk_css_selectors_init (&selectors);
|
||||||
if (n_selectors == 0)
|
|
||||||
|
parse_selector_list (scanner, &selectors);
|
||||||
|
if (gtk_css_selectors_get_size (&selectors) == 0)
|
||||||
{
|
{
|
||||||
gtk_css_parser_skip_until (scanner->parser, GTK_CSS_TOKEN_OPEN_CURLY);
|
gtk_css_parser_skip_until (scanner->parser, GTK_CSS_TOKEN_OPEN_CURLY);
|
||||||
gtk_css_parser_skip (scanner->parser);
|
gtk_css_parser_skip (scanner->parser);
|
||||||
return;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!gtk_css_parser_has_token (scanner->parser, GTK_CSS_TOKEN_OPEN_CURLY))
|
if (!gtk_css_parser_has_token (scanner->parser, GTK_CSS_TOKEN_OPEN_CURLY))
|
||||||
{
|
{
|
||||||
guint i;
|
guint i;
|
||||||
gtk_css_parser_error_syntax (scanner->parser, "Expected '{' after selectors");
|
gtk_css_parser_error_syntax (scanner->parser, "Expected '{' after selectors");
|
||||||
for (i = 0; i < n_selectors; i++)
|
for (i = 0; i < gtk_css_selectors_get_size (&selectors); i++)
|
||||||
_gtk_css_selector_free (selectors[i]);
|
_gtk_css_selector_free (gtk_css_selectors_get (&selectors, i));
|
||||||
gtk_css_parser_skip_until (scanner->parser, GTK_CSS_TOKEN_OPEN_CURLY);
|
gtk_css_parser_skip_until (scanner->parser, GTK_CSS_TOKEN_OPEN_CURLY);
|
||||||
gtk_css_parser_skip (scanner->parser);
|
gtk_css_parser_skip (scanner->parser);
|
||||||
return;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
gtk_css_parser_start_block (scanner->parser);
|
gtk_css_parser_start_block (scanner->parser);
|
||||||
@ -928,8 +920,11 @@ parse_ruleset (GtkCssScanner *scanner)
|
|||||||
|
|
||||||
gtk_css_parser_end_block (scanner->parser);
|
gtk_css_parser_end_block (scanner->parser);
|
||||||
|
|
||||||
css_provider_commit (scanner->provider, selectors, n_selectors, &ruleset);
|
css_provider_commit (scanner->provider, &selectors, &ruleset);
|
||||||
gtk_css_ruleset_clear (&ruleset);
|
gtk_css_ruleset_clear (&ruleset);
|
||||||
|
|
||||||
|
out:
|
||||||
|
gtk_css_selectors_clear (&selectors);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
Loading…
Reference in New Issue
Block a user