mirror of
https://github.com/rkd77/elinks.git
synced 2025-01-03 14:57:44 -05:00
DOM select: Completely rewrite the parser for nth arguments
This commit is contained in:
parent
c2e30c8eea
commit
5777941d06
@ -156,119 +156,124 @@ parse_dom_select_attribute(struct dom_select_node *sel, struct scanner *scanner)
|
|||||||
* 1n+0 / n+0 / n
|
* 1n+0 / n+0 / n
|
||||||
* 0n+0
|
* 0n+0
|
||||||
*/
|
*/
|
||||||
static enum dom_exception_code
|
|
||||||
parse_dom_select_nth_numeric(struct dom_select_nth_match *nth,
|
static size_t
|
||||||
struct scanner_token *arg)
|
get_scanner_token_number(struct scanner_token *token)
|
||||||
{
|
{
|
||||||
size_t sign = 1;
|
size_t number = 0;
|
||||||
size_t number;
|
|
||||||
|
|
||||||
/* Parse negated value: -0n+1 */
|
while (token->length > 0 && isdigit(token->string[0])) {
|
||||||
if (arg->string[0] == '-') {
|
|
||||||
arg->string++, arg->length--;
|
|
||||||
sign = -1;
|
|
||||||
if (arg->length == 0)
|
|
||||||
return DOM_ERR_SYNTAX;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Parse -n or n */
|
|
||||||
if (arg->string[0] == 'n') {
|
|
||||||
nth->step = sign;
|
|
||||||
arg->string++, arg->length--;
|
|
||||||
|
|
||||||
} else if (isdigit(arg->string[0])) {
|
|
||||||
number = 0;
|
|
||||||
do {
|
|
||||||
size_t old_number = number;
|
|
||||||
|
|
||||||
number *= 10;
|
|
||||||
if (old_number > number)
|
|
||||||
return DOM_ERR_NOT_SUPPORTED;
|
|
||||||
|
|
||||||
number += arg->string[0] - '0';
|
|
||||||
arg->string++, arg->length--;
|
|
||||||
|
|
||||||
} while (arg->length > 0 && isdigit(arg->string[0]));
|
|
||||||
|
|
||||||
if (arg->length > 0 && arg->string[0] == 'n') {
|
|
||||||
nth->step = number * sign;
|
|
||||||
arg->string++, arg->length--;
|
|
||||||
|
|
||||||
} else {
|
|
||||||
nth->step = 0;
|
|
||||||
nth->index = number * sign;
|
|
||||||
return DOM_ERR_NONE;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Parse the +... part of n+2 */
|
|
||||||
|
|
||||||
if (arg->length <= 1 || arg->string[0] != '+')
|
|
||||||
return DOM_ERR_NONE;
|
|
||||||
|
|
||||||
arg->string++, arg->length--;
|
|
||||||
|
|
||||||
/* Accept 2n+ */
|
|
||||||
if (!isdigit(arg->string[0]))
|
|
||||||
return DOM_ERR_NONE;
|
|
||||||
|
|
||||||
number = 0;
|
|
||||||
do {
|
|
||||||
size_t old_number = number;
|
size_t old_number = number;
|
||||||
|
|
||||||
number *= 10;
|
number *= 10;
|
||||||
|
|
||||||
|
/* -E2BIG */
|
||||||
if (old_number > number)
|
if (old_number > number)
|
||||||
return DOM_ERR_NOT_SUPPORTED;
|
return -1;
|
||||||
|
|
||||||
number += arg->string[0] - '0';
|
number += token->string[0] - '0';
|
||||||
arg->string++, arg->length--;
|
token->string++, token->length--;
|
||||||
|
}
|
||||||
|
|
||||||
} while (arg->length > 0 && isdigit(arg->string[0]));
|
return number;
|
||||||
|
|
||||||
nth->index = number * sign;
|
|
||||||
|
|
||||||
return DOM_ERR_NONE;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static enum dom_exception_code
|
static enum dom_exception_code
|
||||||
parse_dom_select_nth_arg(struct dom_select_nth_match *nth, struct scanner *scanner)
|
parse_dom_select_nth_arg(struct dom_select_nth_match *nth, struct scanner *scanner)
|
||||||
{
|
{
|
||||||
struct scanner_token *token = get_next_scanner_token(scanner);
|
struct scanner_token *token = get_next_scanner_token(scanner);
|
||||||
struct scanner_token arg;
|
int sign = 1;
|
||||||
|
int number = -1;
|
||||||
|
|
||||||
if (!token || token->type != '(')
|
if (!token || token->type != '(')
|
||||||
return DOM_ERR_SYNTAX;
|
return DOM_ERR_SYNTAX;
|
||||||
|
|
||||||
token = get_next_scanner_token(scanner);
|
token = get_next_scanner_token(scanner);
|
||||||
if (!token || token->type == ')')
|
if (!token)
|
||||||
return DOM_ERR_SYNTAX;
|
return DOM_ERR_SYNTAX;
|
||||||
|
|
||||||
copy_struct(&arg, token);
|
switch (token->type) {
|
||||||
|
case CSS_TOKEN_IDENT:
|
||||||
|
if (scanner_token_contains(token, "even")) {
|
||||||
|
nth->step = 2;
|
||||||
|
nth->index = 0;
|
||||||
|
|
||||||
do {
|
} else if (scanner_token_contains(token, "odd")) {
|
||||||
/* Combine all the arg material to one token. */
|
nth->step = 2;
|
||||||
arg.length = (token->string + token->length) - arg.string;
|
nth->index = 1;
|
||||||
|
|
||||||
|
} else {
|
||||||
|
/* Check for 'n' ident below. */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (skip_css_tokens(scanner, ')'))
|
||||||
|
return DOM_ERR_NONE;
|
||||||
|
|
||||||
|
return DOM_ERR_SYNTAX;
|
||||||
|
|
||||||
|
case '-':
|
||||||
|
sign = -1;
|
||||||
|
|
||||||
token = get_next_scanner_token(scanner);
|
token = get_next_scanner_token(scanner);
|
||||||
if (!token) return DOM_ERR_SYNTAX;
|
if (!token) return DOM_ERR_SYNTAX;
|
||||||
|
|
||||||
} while (token->type != ')');
|
if (token->type != CSS_TOKEN_IDENT)
|
||||||
|
break;
|
||||||
|
|
||||||
if (scanner_token_contains(&arg, "even")) {
|
if (token->type != CSS_TOKEN_NUMBER)
|
||||||
nth->step = 2;
|
return DOM_ERR_SYNTAX;
|
||||||
nth->index = 0;
|
/* Fall-through */
|
||||||
|
|
||||||
} else if (scanner_token_contains(&arg, "odd")) {
|
case CSS_TOKEN_NUMBER:
|
||||||
nth->step = 2;
|
number = get_scanner_token_number(token);
|
||||||
nth->index = 1;
|
if (number < 0)
|
||||||
|
return DOM_ERR_INVALID_STATE;
|
||||||
|
|
||||||
} else if (arg.length > 0) {
|
token = get_next_scanner_token(scanner);
|
||||||
return parse_dom_select_nth_numeric(nth, &arg);
|
if (!token) return DOM_ERR_SYNTAX;
|
||||||
|
break;
|
||||||
|
|
||||||
return DOM_ERR_NONE;
|
default:
|
||||||
|
return DOM_ERR_SYNTAX;
|
||||||
}
|
}
|
||||||
|
|
||||||
return DOM_ERR_NONE;
|
/* The rest can contain n+ part */
|
||||||
|
switch (token->type) {
|
||||||
|
case CSS_TOKEN_IDENT:
|
||||||
|
if (!scanner_token_contains(token, "n"))
|
||||||
|
return DOM_ERR_SYNTAX;
|
||||||
|
|
||||||
|
nth->step = sign * number;
|
||||||
|
|
||||||
|
token = get_next_scanner_token(scanner);
|
||||||
|
if (!token) return DOM_ERR_SYNTAX;
|
||||||
|
|
||||||
|
if (token->type != '+')
|
||||||
|
break;
|
||||||
|
|
||||||
|
token = get_next_scanner_token(scanner);
|
||||||
|
if (!token) return DOM_ERR_SYNTAX;
|
||||||
|
|
||||||
|
if (token->type != CSS_TOKEN_NUMBER)
|
||||||
|
break;
|
||||||
|
|
||||||
|
number = get_scanner_token_number(token);
|
||||||
|
if (number < 0)
|
||||||
|
return DOM_ERR_INVALID_STATE;
|
||||||
|
|
||||||
|
nth->index = sign * number;
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
nth->step = 0;
|
||||||
|
nth->index = sign * number;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (skip_css_tokens(scanner, ')'))
|
||||||
|
return DOM_ERR_NONE;
|
||||||
|
|
||||||
|
return DOM_ERR_SYNTAX;
|
||||||
}
|
}
|
||||||
|
|
||||||
static enum dom_exception_code
|
static enum dom_exception_code
|
||||||
|
Loading…
Reference in New Issue
Block a user