blob: de6958ae43119ccb7583ac988f6e4a730c022c8a [file] [log] [blame]
/**
* Copyright 2010 Google Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
// Author: jmarantz@google.com (Joshua Marantz)
#include "public/add_head_filter.h"
#include "net/instaweb/htmlparse/public/html_parse.h"
#include "net/instaweb/htmlparse/public/html_element.h"
namespace net_instaweb {
AddHeadFilter::AddHeadFilter(HtmlParse* html_parse, bool combine_multiple_heads)
: html_parse_(html_parse),
combine_multiple_heads_(combine_multiple_heads),
found_head_(false),
s_head_(html_parse->Intern("head")),
s_body_(html_parse->Intern("body")),
head_element_(NULL) {
}
void AddHeadFilter::StartDocument() {
found_head_ = false;
head_element_ = NULL;
}
void AddHeadFilter::StartElement(HtmlElement* element) {
if (!found_head_) {
if (element->tag() == s_body_) {
head_element_ = html_parse_->NewElement(
element->parent(), s_head_);
html_parse_->InsertElementBeforeElement(element, head_element_);
found_head_ = true;
} else if (element->tag() == s_head_) {
found_head_ = true;
head_element_ = element;
}
}
}
void AddHeadFilter::EndElement(HtmlElement* element) {
// Detect elements that are children of a subsequent head. Move them
// into the first head if possible.
if (combine_multiple_heads_ && found_head_ && (head_element_ != NULL) &&
html_parse_->IsRewritable(head_element_)) {
if ((element->tag() == s_head_) && (element != head_element_)) {
// There should be no elements left in the subsequent head, so
// remove it.
bool moved = html_parse_->MoveCurrentInto(head_element_);
CHECK(moved) << "failed to move new head under old head";
bool deleted = html_parse_->DeleteSavingChildren(element);
CHECK(deleted) << "failed to delete extra head";
}
}
}
void AddHeadFilter::Flush() {
head_element_ = NULL;
}
void AddHeadFilter::EndDocument() {
if (!found_head_) {
// In order to insert a <head> in a docucument that lacks one, we
// must first find the body. If we get through the whole doc without
// finding a <head> or a <body> then this filter will have failed to
// add a head.
html_parse_->ErrorHere("Reached end of document without finding <body>");
}
}
} // namespace net_instaweb