?? cssstyleselector.cpp
字號:
/**
* This file is part of the CSS implementation for KDE.
*
* Copyright (C) 1999 Lars Knoll (knoll@kde.org)
* Copyright (C) 2004 Apple Computer, Inc.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library 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
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public License
* along with this library; see the file COPYING.LIB. If not, write to
* the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*/
#include "css/cssstyleselector.h"
#include "rendering/render_style.h"
#include "css/css_stylesheetimpl.h"
#include "css/css_ruleimpl.h"
#include "css/css_valueimpl.h"
#include "css/csshelper.h"
#include "rendering/render_object.h"
#include "html/html_documentimpl.h"
#include "html/html_elementimpl.h"
#include "xml/dom_elementimpl.h"
#include "dom/css_rule.h"
#include "dom/css_value.h"
#include "khtml_factory.h"
#include "khtmlpart_p.h"
using namespace khtml;
using namespace DOM;
#include "css/cssproperties.h"
#include "css/cssvalues.h"
#include "misc/khtmllayout.h"
#include "khtml_settings.h"
#include "misc/htmlhashes.h"
#include "misc/helper.h"
#include "misc/loader.h"
#include "rendering/font.h"
#include "khtmlview.h"
#include "khtml_part.h"
#include <kstandarddirs.h>
#include <kcharsets.h>
#include <kglobal.h>
#include <qfile.h>
#include <qfontdatabase.h>
#include <qfontinfo.h>
#include <qvaluelist.h>
#include <qstring.h>
#include <kdebug.h>
#include <kurl.h>
#include <qdatetime.h>
#include <assert.h>
#include <qpaintdevicemetrics.h>
#include <qintcache.h>
#include <stdlib.h>
// #define STYLE_SHARING_STATS 1
#define HANDLE_INHERIT(prop, Prop) \
if (isInherit) \
{\
style->set##Prop(parentStyle->prop());\
return;\
}
#define HANDLE_INHERIT_AND_INITIAL(prop, Prop) \
HANDLE_INHERIT(prop, Prop) \
else if (isInitial) \
{\
style->set##Prop(RenderStyle::initial##Prop());\
return;\
}
#define HANDLE_INHERIT_AND_INITIAL_WITH_VALUE(prop, Prop, Value) \
HANDLE_INHERIT(prop, Prop) \
else if (isInitial) \
{\
style->set##Prop(RenderStyle::initial##Value());\
return;\
}
#define HANDLE_BACKGROUND_INHERIT_AND_INITIAL(prop, Prop) \
if (isInherit) { \
BackgroundLayer* currChild = style->accessBackgroundLayers(); \
BackgroundLayer* prevChild = 0; \
const BackgroundLayer* currParent = parentStyle->backgroundLayers(); \
while (currParent && currParent->is##Prop##Set()) { \
if (!currChild) { \
/* Need to make a new layer.*/ \
currChild = new BackgroundLayer(); \
prevChild->setNext(currChild); \
} \
currChild->set##Prop(currParent->prop()); \
prevChild = currChild; \
currChild = prevChild->next(); \
currParent = currParent->next(); \
} \
\
while (currChild) { \
/* Reset any remaining layers to not have the property set. */ \
currChild->clear##Prop(); \
currChild = currChild->next(); \
} \
return; \
} \
if (isInitial) { \
BackgroundLayer* currChild = style->accessBackgroundLayers(); \
currChild->set##Prop(RenderStyle::initial##Prop()); \
for (currChild = currChild->next(); currChild; currChild = currChild->next()) \
currChild->clear##Prop(); \
return; \
}
#define HANDLE_BACKGROUND_VALUE(prop, Prop, value) { \
HANDLE_BACKGROUND_INHERIT_AND_INITIAL(prop, Prop) \
if (!value->isPrimitiveValue() && !value->isValueList()) \
return; \
BackgroundLayer* currChild = style->accessBackgroundLayers(); \
BackgroundLayer* prevChild = 0; \
if (value->isPrimitiveValue()) { \
map##Prop(currChild, value); \
currChild = currChild->next(); \
} \
else { \
/* Walk each value and put it into a layer, creating new layers as needed. */ \
CSSValueListImpl* valueList = static_cast<CSSValueListImpl*>(value); \
for (unsigned int i = 0; i < valueList->length(); i++) { \
if (!currChild) { \
/* Need to make a new layer to hold this value */ \
currChild = new BackgroundLayer(); \
prevChild->setNext(currChild); \
} \
map##Prop(currChild, valueList->item(i)); \
prevChild = currChild; \
currChild = currChild->next(); \
} \
} \
while (currChild) { \
/* Reset all remaining layers to not have the property set. */ \
currChild->clear##Prop(); \
currChild = currChild->next(); \
} }
#define HANDLE_INHERIT_COND(propID, prop, Prop) \
if (id == propID) \
{\
style->set##Prop(parentStyle->prop());\
return;\
}
#define HANDLE_INITIAL_COND(propID, Prop) \
if (id == propID) \
{\
style->set##Prop(RenderStyle::initial##Prop());\
return;\
}
#define HANDLE_INITIAL_COND_WITH_VALUE(propID, Prop, Value) \
if (id == propID) \
{\
style->set##Prop(RenderStyle::initial##Value());\
return;\
}
namespace khtml {
CSSRuleSet *CSSStyleSelector::defaultStyle = 0;
CSSRuleSet *CSSStyleSelector::defaultQuirksStyle = 0;
CSSRuleSet *CSSStyleSelector::defaultPrintStyle = 0;
CSSStyleSheetImpl *CSSStyleSelector::defaultSheet = 0;
RenderStyle* CSSStyleSelector::styleNotYetAvailable = 0;
CSSStyleSheetImpl *CSSStyleSelector::quirksSheet = 0;
static CSSStyleSelector::Encodedurl *encodedurl = 0;
static PseudoState pseudoState;
CSSStyleSelector::CSSStyleSelector( DocumentImpl* doc, QString userStyleSheet, StyleSheetListImpl *styleSheets,
bool _strictParsing )
{
init();
view = doc->view();
strictParsing = _strictParsing;
settings = view ? view->part()->settings() : 0;
if(!defaultStyle) loadDefaultStyle(settings);
m_medium = view ? view->mediaType() : QString("all");
m_userStyle = 0;
m_userSheet = 0;
paintDeviceMetrics = doc->paintDeviceMetrics();
// FIXME: This sucks! The user sheet is reparsed every time!
if (!userStyleSheet.isEmpty()) {
m_userSheet = new DOM::CSSStyleSheetImpl(doc);
m_userSheet->parseString(DOMString(userStyleSheet), strictParsing);
m_userStyle = new CSSRuleSet();
m_userStyle->addRulesFromSheet( m_userSheet, m_medium );
}
// add stylesheets from document
m_authorStyle = new CSSRuleSet();
QPtrListIterator<StyleSheetImpl> it(styleSheets->styleSheets);
for (; it.current(); ++it)
if (it.current()->isCSSStyleSheet())
m_authorStyle->addRulesFromSheet(static_cast<CSSStyleSheetImpl*>(it.current()), m_medium);
//kdDebug( 6080 ) << "number of style sheets in document " << authorStyleSheets.count() << endl;
//kdDebug( 6080 ) << "CSSStyleSelector: author style has " << authorStyle->count() << " elements"<< endl;
}
CSSStyleSelector::CSSStyleSelector( CSSStyleSheetImpl *sheet )
{
init();
if(!defaultStyle) loadDefaultStyle();
KHTMLView *view = sheet->doc()->view();
m_medium = view ? view->mediaType() : QString("all");
m_authorStyle = new CSSRuleSet();
m_authorStyle->addRulesFromSheet( sheet, m_medium );
}
void CSSStyleSelector::init()
{
element = 0;
settings = 0;
paintDeviceMetrics = 0;
m_matchedRuleCount = m_matchedDeclCount = m_tmpRuleCount = 0;
}
void CSSStyleSelector::setEncodedURL(const KURL& url)
{
KURL u = url;
u.setQuery( QString::null );
u.setRef( QString::null );
encodedurl.file = u.url();
int pos = encodedurl.file.findRev('/');
encodedurl.path = encodedurl.file;
if ( pos > 0 ) {
encodedurl.path.truncate( pos );
encodedurl.path += '/';
}
u.setPath( QString::null );
encodedurl.host = u.url();
//kdDebug() << "CSSStyleSelector::CSSStyleSelector encoded url " << encodedurl.path << endl;
}
CSSStyleSelector::~CSSStyleSelector()
{
delete m_authorStyle;
delete m_userStyle;
delete m_userSheet;
}
void CSSStyleSelector::loadDefaultStyle(const KHTMLSettings *s)
{
OOM_CRITICAL_PATH_BEGIN("CSSStyleSelector::loadDefaultStyle", 0)
if(defaultStyle) return;
{
QFile f(locate( "data", "khtml/css/html4.css" ) );
f.open(IO_ReadOnly);
QCString file( f.size()+1 );
int readbytes = f.readBlock( file.data(), f.size() );
f.close();
if ( readbytes >= 0 )
file[readbytes] = '\0';
QString style = QString::fromLatin1( file.data() );
if(s)
style += s->settingsToCSS();
DOMString str(style);
defaultSheet = new DOM::CSSStyleSheetImpl((DOM::CSSStyleSheetImpl * ) 0);
defaultSheet->parseString( str );
// Collect only strict-mode rules.
defaultStyle = new CSSRuleSet();
defaultStyle->addRulesFromSheet( defaultSheet, "screen" );
defaultPrintStyle = new CSSRuleSet();
defaultPrintStyle->addRulesFromSheet( defaultSheet, "print" );
}
{
QFile f(locate( "data", "khtml/css/quirks.css" ) );
f.open(IO_ReadOnly);
QCString file( f.size()+1 );
int readbytes = f.readBlock( file.data(), f.size() );
f.close();
if ( readbytes >= 0 )
file[readbytes] = '\0';
QString style = QString::fromLatin1( file.data() );
DOMString str(style);
quirksSheet = new DOM::CSSStyleSheetImpl((DOM::CSSStyleSheetImpl * ) 0);
quirksSheet->parseString( str );
// Collect only quirks-mode rules.
defaultQuirksStyle = new CSSRuleSet();
defaultQuirksStyle->addRulesFromSheet( quirksSheet, "screen" );
}
//kdDebug() << "CSSStyleSelector: default style has " << defaultStyle->count() << " elements"<< endl;
OOM_CRITICAL_PATH_END
}
void CSSStyleSelector::addMatchedRule(CSSRuleData* rule)
{
if (m_matchedRules.size() <= m_matchedRuleCount)
m_matchedRules.resize(2*m_matchedRules.size()+1);
m_matchedRules[m_matchedRuleCount++] = rule;
}
void CSSStyleSelector::addMatchedDeclaration(CSSMutableStyleDeclarationImpl* decl)
{
if (m_matchedDecls.size() <= m_matchedDeclCount)
m_matchedDecls.resize(2*m_matchedDecls.size()+1);
m_matchedDecls[m_matchedDeclCount++] = decl;
}
void CSSStyleSelector::matchRules(CSSRuleSet* rules, int& firstRuleIndex, int& lastRuleIndex)
{
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -