CachedCSSStyleSheet.cpp [plain text]
#include "config.h"
#include "CachedCSSStyleSheet.h"
#include "CSSStyleSheet.h"
#include "CachedResourceClientWalker.h"
#include "CachedStyleSheetClient.h"
#include "HTTPHeaderNames.h"
#include "HTTPParsers.h"
#include "MemoryCache.h"
#include "SharedBuffer.h"
#include "StyleSheetContents.h"
#include "TextResourceDecoder.h"
#include <wtf/CurrentTime.h>
#include <wtf/Vector.h>
namespace WebCore {
CachedCSSStyleSheet::CachedCSSStyleSheet(const ResourceRequest& resourceRequest, const String& charset, SessionID sessionID)
: CachedResource(resourceRequest, CSSStyleSheet, sessionID)
, m_decoder(TextResourceDecoder::create("text/css", charset))
{
setAccept("text/css,*/*;q=0.1");
}
CachedCSSStyleSheet::~CachedCSSStyleSheet()
{
if (m_parsedStyleSheetCache)
m_parsedStyleSheetCache->removedFromMemoryCache();
}
void CachedCSSStyleSheet::didAddClient(CachedResourceClient* c)
{
ASSERT(c->resourceClientType() == CachedStyleSheetClient::expectedType());
CachedResource::didAddClient(c);
if (!isLoading())
static_cast<CachedStyleSheetClient*>(c)->setCSSStyleSheet(m_resourceRequest.url(), m_response.url(), m_decoder->encoding().name(), this);
}
void CachedCSSStyleSheet::setEncoding(const String& chs)
{
m_decoder->setEncoding(chs, TextResourceDecoder::EncodingFromHTTPHeader);
}
String CachedCSSStyleSheet::encoding() const
{
return m_decoder->encoding().name();
}
const String CachedCSSStyleSheet::sheetText(MIMETypeCheck mimeTypeCheck, bool* hasValidMIMEType) const
{
if (!m_data || m_data->isEmpty() || !canUseSheet(mimeTypeCheck, hasValidMIMEType))
return String();
if (!m_decodedSheetText.isNull())
return m_decodedSheetText;
return m_decoder->decodeAndFlush(m_data->data(), m_data->size());
}
void CachedCSSStyleSheet::finishLoading(SharedBuffer* data)
{
m_data = data;
setEncodedSize(data ? data->size() : 0);
if (data)
m_decodedSheetText = m_decoder->decodeAndFlush(data->data(), data->size());
setLoading(false);
checkNotify();
m_decodedSheetText = String();
}
void CachedCSSStyleSheet::checkNotify()
{
if (isLoading())
return;
CachedResourceClientWalker<CachedStyleSheetClient> w(m_clients);
while (CachedStyleSheetClient* c = w.next())
c->setCSSStyleSheet(m_resourceRequest.url(), m_response.url(), m_decoder->encoding().name(), this);
}
bool CachedCSSStyleSheet::canUseSheet(MIMETypeCheck mimeTypeCheck, bool* hasValidMIMEType) const
{
if (errorOccurred())
return false;
if (mimeTypeCheck == MIMETypeCheck::Lax)
return true;
String mimeType = extractMIMETypeFromMediaType(response().httpHeaderField(HTTPHeaderName::ContentType));
bool typeOK = mimeType.isEmpty() || equalIgnoringCase(mimeType, "text/css") || equalIgnoringCase(mimeType, "application/x-unknown-content-type");
if (hasValidMIMEType)
*hasValidMIMEType = typeOK;
return typeOK;
}
void CachedCSSStyleSheet::destroyDecodedData()
{
if (!m_parsedStyleSheetCache)
return;
m_parsedStyleSheetCache->removedFromMemoryCache();
m_parsedStyleSheetCache = nullptr;
setDecodedSize(0);
}
PassRefPtr<StyleSheetContents> CachedCSSStyleSheet::restoreParsedStyleSheet(const CSSParserContext& context, CachePolicy cachePolicy)
{
if (!m_parsedStyleSheetCache)
return nullptr;
if (!m_parsedStyleSheetCache->subresourcesAllowReuse(cachePolicy)) {
m_parsedStyleSheetCache->removedFromMemoryCache();
m_parsedStyleSheetCache = nullptr;
return nullptr;
}
ASSERT(m_parsedStyleSheetCache->isCacheable());
ASSERT(m_parsedStyleSheetCache->isInMemoryCache());
if (m_parsedStyleSheetCache->parserContext() != context)
return nullptr;
didAccessDecodedData(monotonicallyIncreasingTime());
return m_parsedStyleSheetCache;
}
void CachedCSSStyleSheet::saveParsedStyleSheet(Ref<StyleSheetContents>&& sheet)
{
ASSERT(sheet.get().isCacheable());
if (m_parsedStyleSheetCache)
m_parsedStyleSheetCache->removedFromMemoryCache();
m_parsedStyleSheetCache = WTF::move(sheet);
m_parsedStyleSheetCache->addedToMemoryCache();
setDecodedSize(m_parsedStyleSheetCache->estimatedSizeInBytes());
}
}