2006-05-10 13:37:48 +00:00
|
|
|
/***************************************************************************
|
|
|
|
* css.c
|
|
|
|
* Character-based parser for Css definitions
|
|
|
|
* Author - Iago Rubio <iagorubio(at)users.sourceforge.net>
|
2008-10-12 17:52:22 +00:00
|
|
|
* - Bronisław Białek <after89(at)gmail.com>
|
2006-05-10 13:37:48 +00:00
|
|
|
**************************************************************************/
|
|
|
|
#include "general.h"
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
|
|
|
|
#include "parse.h"
|
|
|
|
#include "read.h"
|
|
|
|
|
|
|
|
|
|
|
|
typedef enum eCssKinds {
|
2008-10-12 17:52:22 +00:00
|
|
|
K_NONE = -1, K_SELECTOR, K_ID, K_CLASS
|
2006-05-10 13:37:48 +00:00
|
|
|
} cssKind;
|
|
|
|
|
|
|
|
static kindOption CssKinds [] = {
|
2008-10-12 17:52:22 +00:00
|
|
|
{ TRUE, 's', "struct", "selectors" },
|
|
|
|
{ TRUE, 'v', "variable", "identities" },
|
|
|
|
{ TRUE, 'c', "class", "classes" }
|
2006-05-10 13:37:48 +00:00
|
|
|
};
|
|
|
|
|
2008-02-27 13:17:29 +00:00
|
|
|
typedef enum _CssParserState { /* state of parsing */
|
|
|
|
P_STATE_NONE, /* default state */
|
|
|
|
P_STATE_IN_COMMENT, /* into a comment, only multi line in CSS */
|
|
|
|
P_STATE_IN_SINGLE_STRING, /* into a single quoted string */
|
|
|
|
P_STATE_IN_DOUBLE_STRING, /* into a double quoted string */
|
|
|
|
P_STATE_IN_DEFINITION, /* on the body of the style definition, nothing for us */
|
|
|
|
P_STATE_IN_MEDIA, /* on a @media declaration, can be multi-line */
|
|
|
|
P_STATE_IN_IMPORT, /* on a @import declaration, can be multi-line */
|
|
|
|
P_STATE_IN_NAMESPACE, /* on a @namespace declaration */
|
|
|
|
P_STATE_IN_PAGE, /* on a @page declaration */
|
|
|
|
P_STATE_IN_FONTFACE, /* on a @font-face declaration */
|
|
|
|
P_STATE_AT_END /* end of parsing */
|
2006-05-10 13:37:48 +00:00
|
|
|
} CssParserState;
|
|
|
|
|
|
|
|
static void makeCssSimpleTag( vString *name, cssKind kind, boolean delete )
|
|
|
|
{
|
|
|
|
vStringTerminate (name);
|
|
|
|
makeSimpleTag (name, CssKinds, kind);
|
|
|
|
vStringClear (name);
|
|
|
|
if( delete )
|
|
|
|
vStringDelete (name);
|
|
|
|
}
|
|
|
|
|
|
|
|
static boolean isCssDeclarationAllowedChar( const unsigned char *cp )
|
|
|
|
{
|
|
|
|
return isalnum ((int) *cp) ||
|
|
|
|
isspace ((int) *cp) ||
|
2008-02-27 13:17:29 +00:00
|
|
|
*cp == '_' || /* allowed char */
|
|
|
|
*cp == '-' || /* allowed char */
|
|
|
|
*cp == '+' || /* allow all sibling in a single tag */
|
|
|
|
*cp == '>' || /* allow all child in a single tag */
|
|
|
|
*cp == '{' || /* allow the start of the declaration */
|
|
|
|
*cp == '.' || /* allow classes and selectors */
|
|
|
|
*cp == ',' || /* allow multiple declarations */
|
|
|
|
*cp == ':' || /* allow pseudo classes */
|
|
|
|
*cp == '*' || /* allow globs as P + * */
|
|
|
|
*cp == '#'; /* allow ids */
|
2006-05-10 13:37:48 +00:00
|
|
|
}
|
|
|
|
|
2008-10-12 17:52:22 +00:00
|
|
|
static CssParserState parseCssDeclaration( const unsigned char **position, cssKind kind, const char *aname)
|
2006-05-10 13:37:48 +00:00
|
|
|
{
|
|
|
|
const unsigned char *cp = *position;
|
2008-10-12 17:52:22 +00:00
|
|
|
vString *name = vStringNew ();
|
|
|
|
vStringCopyS(name, aname);
|
2006-05-10 13:37:48 +00:00
|
|
|
|
2008-02-27 13:17:29 +00:00
|
|
|
/* pick to the end of line including children and sibling
|
|
|
|
* if declaration is multiline go for the next line */
|
2006-05-10 13:37:48 +00:00
|
|
|
while ( isCssDeclarationAllowedChar(cp) ||
|
2008-02-27 13:17:29 +00:00
|
|
|
*cp == '\0' ) /* track the end of line into the loop */
|
2006-05-10 13:37:48 +00:00
|
|
|
{
|
2009-03-22 22:49:29 +00:00
|
|
|
if( *cp == ',' )
|
2006-05-10 13:37:48 +00:00
|
|
|
{
|
|
|
|
makeCssSimpleTag(name, kind, TRUE);
|
2008-10-12 17:52:22 +00:00
|
|
|
*position = cp;
|
2006-05-10 13:37:48 +00:00
|
|
|
return P_STATE_NONE;
|
|
|
|
}
|
2009-03-22 22:49:29 +00:00
|
|
|
else if( *cp == '{' || *cp == '\0' )
|
|
|
|
{ /* assume that line end is the same as a starting definition (i.e. the { is on the next line */
|
2006-05-10 13:37:48 +00:00
|
|
|
makeCssSimpleTag(name, kind, TRUE);
|
2008-10-12 17:52:22 +00:00
|
|
|
*position = cp;
|
2006-05-10 13:37:48 +00:00
|
|
|
return P_STATE_IN_DEFINITION;
|
|
|
|
}
|
|
|
|
|
|
|
|
vStringPut (name, (int) *cp);
|
|
|
|
++cp;
|
|
|
|
}
|
|
|
|
|
|
|
|
makeCssSimpleTag(name, kind, TRUE);
|
|
|
|
*position = cp;
|
|
|
|
|
|
|
|
return P_STATE_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static CssParserState parseCssLine( const unsigned char *line, CssParserState state )
|
|
|
|
{
|
|
|
|
vString *aux;
|
2008-10-12 17:52:22 +00:00
|
|
|
vString *stack = vStringNew ();
|
2006-05-10 13:37:48 +00:00
|
|
|
|
2008-02-27 13:17:29 +00:00
|
|
|
while( *line != '\0' ) /* fileReadLine returns NULL terminated strings */
|
2006-05-10 13:37:48 +00:00
|
|
|
{
|
2008-10-12 17:52:22 +00:00
|
|
|
vStringClear (stack);
|
2008-10-13 15:26:41 +00:00
|
|
|
while (state == P_STATE_NONE &&
|
|
|
|
(isspace ((int) *line) || isalnum ((int) *line) || ( *line == '*' && *(line-1) != '/' )))
|
2008-10-12 17:52:22 +00:00
|
|
|
{
|
|
|
|
if ((stack->length > 0 && isspace((int) *line)) || isalnum ((int) *line) || *line == '*') {
|
|
|
|
vStringPut(stack, (int) *line);
|
|
|
|
}
|
|
|
|
|
2006-05-10 13:37:48 +00:00
|
|
|
++line;
|
2008-10-12 17:52:22 +00:00
|
|
|
}
|
|
|
|
vStringTerminate (stack);
|
|
|
|
|
2006-05-10 13:37:48 +00:00
|
|
|
switch( state )
|
|
|
|
{
|
|
|
|
case P_STATE_NONE:
|
2008-10-12 17:52:22 +00:00
|
|
|
if( *line == '.' ) /* a class */
|
|
|
|
state = parseCssDeclaration( &line, K_CLASS, vStringValue(stack) );
|
2008-02-27 13:17:29 +00:00
|
|
|
else if( *line == '#' ) /* an id */
|
2008-10-12 17:52:22 +00:00
|
|
|
state = parseCssDeclaration( &line, K_ID, vStringValue(stack) );
|
2008-02-27 13:17:29 +00:00
|
|
|
else if( *line == '@' ) /* at-rules, we'll ignore them */
|
2006-05-10 13:37:48 +00:00
|
|
|
{
|
|
|
|
++line;
|
|
|
|
aux = vStringNew();
|
|
|
|
while( !isspace((int) *line) )
|
|
|
|
{
|
|
|
|
vStringPut (aux, (int) *line);
|
|
|
|
++line;
|
|
|
|
}
|
|
|
|
vStringTerminate (aux);
|
|
|
|
if( strcmp( aux->buffer, "media" ) == 0 )
|
|
|
|
state = P_STATE_IN_MEDIA;
|
|
|
|
else if ( strcmp( aux->buffer, "import" ) == 0 )
|
|
|
|
state = P_STATE_IN_IMPORT;
|
|
|
|
else if ( strcmp( aux->buffer, "namespace" ) == 0 )
|
|
|
|
state = P_STATE_IN_NAMESPACE;
|
|
|
|
else if ( strcmp( aux->buffer, "page" ) == 0 )
|
|
|
|
state = P_STATE_IN_PAGE;
|
|
|
|
else if ( strcmp( aux->buffer, "font-face" ) == 0 )
|
|
|
|
state = P_STATE_IN_FONTFACE;
|
|
|
|
vStringDelete (aux);
|
|
|
|
}
|
2008-02-27 13:17:29 +00:00
|
|
|
else if( *line == '*' && *(line-1) == '/' ) /* multi-line comment */
|
2006-05-10 13:37:48 +00:00
|
|
|
state = P_STATE_IN_COMMENT;
|
2008-10-12 17:52:22 +00:00
|
|
|
else if ( stack->length > 0 )
|
|
|
|
state = parseCssDeclaration( &line, K_SELECTOR, vStringValue(stack) );
|
|
|
|
|
2006-05-10 13:37:48 +00:00
|
|
|
break;
|
|
|
|
case P_STATE_IN_COMMENT:
|
|
|
|
if( *line == '/' && *(line-1) == '*')
|
|
|
|
state = P_STATE_NONE;
|
|
|
|
break;
|
|
|
|
case P_STATE_IN_SINGLE_STRING:
|
|
|
|
if( *line == '\'' && *(line-1) != '\\' )
|
2008-02-27 13:17:29 +00:00
|
|
|
state = P_STATE_IN_DEFINITION; /* PAGE, FONTFACE and DEFINITION are treated the same way */
|
2006-05-10 13:37:48 +00:00
|
|
|
break;
|
|
|
|
case P_STATE_IN_DOUBLE_STRING:
|
|
|
|
if( *line=='"' && *(line-1) != '\\' )
|
2008-02-27 13:17:29 +00:00
|
|
|
state = P_STATE_IN_DEFINITION; /* PAGE, FONTFACE and DEFINITION are treated the same way */
|
2006-05-10 13:37:48 +00:00
|
|
|
break;
|
|
|
|
case P_STATE_IN_MEDIA:
|
2008-02-27 13:17:29 +00:00
|
|
|
/* skip to start of media body or line end */
|
2006-05-10 13:37:48 +00:00
|
|
|
while( *line != '{' )
|
|
|
|
{
|
|
|
|
if( *line == '\0' )
|
|
|
|
break;
|
|
|
|
++line;
|
|
|
|
}
|
|
|
|
if( *line == '{' )
|
|
|
|
state = P_STATE_NONE;
|
|
|
|
break;
|
|
|
|
case P_STATE_IN_IMPORT:
|
|
|
|
case P_STATE_IN_NAMESPACE:
|
2008-02-27 13:17:29 +00:00
|
|
|
/* skip to end of declaration or line end */
|
2006-05-10 13:37:48 +00:00
|
|
|
while( *line != ';' )
|
|
|
|
{
|
|
|
|
if( *line == '\0' )
|
|
|
|
break;
|
|
|
|
++line;
|
|
|
|
}
|
|
|
|
if( *line == ';' )
|
|
|
|
state = P_STATE_NONE;
|
|
|
|
break;
|
|
|
|
case P_STATE_IN_PAGE:
|
|
|
|
case P_STATE_IN_FONTFACE:
|
|
|
|
case P_STATE_IN_DEFINITION:
|
2009-03-22 22:49:29 +00:00
|
|
|
if( *line == '\0' )
|
|
|
|
line = fileReadLine ();
|
2006-05-10 13:37:48 +00:00
|
|
|
if( *line == '}' )
|
|
|
|
state = P_STATE_NONE;
|
|
|
|
else if( *line == '\'' )
|
|
|
|
state = P_STATE_IN_SINGLE_STRING;
|
|
|
|
else if( *line == '"' )
|
|
|
|
state = P_STATE_IN_DOUBLE_STRING;
|
|
|
|
break;
|
|
|
|
case P_STATE_AT_END:
|
|
|
|
return state;
|
|
|
|
break;
|
|
|
|
}
|
2006-05-30 20:50:50 +00:00
|
|
|
if (line == NULL) return P_STATE_AT_END;
|
2006-05-10 13:37:48 +00:00
|
|
|
line++;
|
|
|
|
}
|
2008-10-12 17:52:22 +00:00
|
|
|
vStringDelete (stack);
|
|
|
|
|
2006-05-10 13:37:48 +00:00
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void findCssTags (void)
|
|
|
|
{
|
|
|
|
const unsigned char *line;
|
|
|
|
CssParserState state = P_STATE_NONE;
|
|
|
|
|
|
|
|
while ( (line = fileReadLine ()) != NULL )
|
|
|
|
{
|
|
|
|
state = parseCssLine( line, state );
|
|
|
|
if( state==P_STATE_AT_END ) return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* parser definition */
|
|
|
|
extern parserDefinition* CssParser (void)
|
|
|
|
{
|
|
|
|
static const char *const extensions [] = { "css", NULL };
|
|
|
|
parserDefinition* def = parserNew ("CSS");
|
|
|
|
def->kinds = CssKinds;
|
|
|
|
def->kindCount = KIND_COUNT (CssKinds);
|
|
|
|
def->extensions = extensions;
|
|
|
|
def->parser = findCssTags;
|
|
|
|
return def;
|
|
|
|
}
|