blob: 1215f5060c465823ea73d1895fc73f7bb3c38e55 [file] [log] [blame]
/*
* Copyright 2011 Google Inc. All Rights Reserved.
*
* 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.
*/
#include "sfntly/table.h"
#include "sfntly/font.h"
#include "sfntly/tag.h"
#include "sfntly/cmap_table.h"
#include "sfntly/font_header_table.h"
#include "sfntly/glyph_table.h"
#include "sfntly/horizontal_header_table.h"
#include "sfntly/horizontal_metrics_table.h"
#include "sfntly/loca_table.h"
#include "sfntly/maximum_profile_table.h"
#include "sfntly/name_table.h"
#include "sfntly/os2_table.h"
namespace sfntly {
/******************************************************************************
* Table class
******************************************************************************/
Table::Table(Header* header, ReadableFontData* data) : FontDataTable(data) {
header_ = header;
}
Table::~Table() {}
int64_t Table::calculatedChecksum() {
return data_->checksum();
}
Table::Header* Table::header() {
return header_;
}
int32_t Table::headerTag() {
return header_->tag();
}
int32_t Table::headerOffset() {
return header_->offset();
}
int32_t Table::headerLength() {
return header_->length();
}
int64_t Table::headerChecksum() {
return header_->checksum();
}
WritableFontData* Table::getNewData(int32_t size) {
return font_->getNewData(size);
}
void Table::setFont(Font* font) {
font_ = font;
}
/******************************************************************************
* Table::Header class
******************************************************************************/
Table::Header::Header(int32_t tag)
: tag_(tag), checksum_(0), checksum_valid_(false), offset_(0), length_(0),
offset_valid_(false) {
}
Table::Header::Header(int32_t tag, int32_t length)
: tag_(tag), checksum_(0), checksum_valid_(false), offset_(0),
length_(length), offset_valid_(false) {
}
Table::Header::Header(int32_t tag, int64_t checksum, int32_t offset,
int32_t length)
: tag_(tag), checksum_(checksum), checksum_valid_(true), offset_(offset),
length_(length), offset_valid_(true) {
}
Table::Header::~Header() {}
bool TableHeaderComparator::operator() (const TableHeaderPtr lhs,
const TableHeaderPtr rhs) {
return lhs->offset_ > rhs->offset_;
}
int32_t Table::Header::tag() { return tag_; }
int32_t Table::Header::offset() { return offset_; }
int32_t Table::Header::length() { return length_; }
bool Table::Header::offsetValid() { return offset_valid_; }
int64_t Table::Header::checksum() { return checksum_; }
bool Table::Header::checksumValid() { return checksum_valid_; }
/******************************************************************************
* Table::Builder class
******************************************************************************/
Table::Builder::Builder(FontDataTableBuilderContainer* font_builder,
Header* header, WritableFontData* data) :
FontDataTable::Builder(font_builder, data) {
header_ = header;
}
Table::Builder::Builder(FontDataTableBuilderContainer* font_builder,
Header* header, ReadableFontData* data) :
FontDataTable::Builder(font_builder, data) {
header_ = header;
}
Table::Builder::Builder(FontDataTableBuilderContainer* font_builder,
Header* header) :
FontDataTable::Builder(font_builder) {
header_ = header;
}
Table::Builder::~Builder() {
header_.release();
}
Table::Header* Table::Builder::header() {
return header_;
}
void Table::Builder::notifyPostTableBuild(FontDataTable* table) {
if (modelChanged() || dataChanged()) {
Table* derived_table = down_cast<Table*>(table);
header_ = new Header(header()->tag(),
derived_table->readFontData()->length());
}
}
WritableFontData* Table::Builder::getNewData(int32_t size) {
UNREFERENCED_PARAMETER(size);
return internalWriteData();
}
CALLER_ATTACH Table::Builder* Table::Builder::getBuilder(
FontDataTableBuilderContainer* font_builder, Header* header,
WritableFontData* table_data) {
int32_t tag = header->tag();
TableBuilderPtr builder;
Table::Builder* builder_raw = NULL;
// Note: Tables are commented out when they are not used/ported.
// TODO(arthurhsu): IMPLEMENT: finish tables that are not ported.
/*if (tag == Tag::cmap) {
builder_raw = static_cast<Table::Builder*>(
new CMapTable::Builder(font_builder, header, table_data));
} else*/ if (tag == Tag::head) {
builder_raw = static_cast<Table::Builder*>(
new FontHeaderTable::Builder(font_builder, header, table_data));
} else if (tag == Tag::hhea) {
builder_raw = static_cast<Table::Builder*>(
new HorizontalHeaderTable::Builder(font_builder, header, table_data));
} else if (tag == Tag::hmtx) {
builder_raw = static_cast<Table::Builder*>(
new HorizontalMetricsTable::Builder(font_builder, header, table_data));
} else if (tag == Tag::maxp) {
builder_raw = static_cast<Table::Builder*>(
new MaximumProfileTable::Builder(font_builder, header, table_data));
}/* else if (tag == Tag::name) {
builder_raw = static_cast<Table::Builder*>(
new NameTable::Builder(font_builder, header, table_data));
}*/ else if (tag == Tag::OS_2) {
builder_raw = static_cast<Table::Builder*>(
new OS2Table::Builder(font_builder, header, table_data));
}/* else if (tag == Tag::PostScript) {
builder_raw = static_cast<Table::Builder*>(
new PostScriptTable::Builder(font_builder, header, table_data));
} else if (tag == Tag::cvt) {
builder_raw = static_cast<Table::Builder*>(
new ControlValueTable::Builder(font_builder, header, table_data));
}*/ else if (tag == Tag::glyf) {
builder_raw = static_cast<Table::Builder*>(
new GlyphTable::Builder(font_builder, header, table_data));
} else if (tag == Tag::loca) {
builder_raw = static_cast<Table::Builder*>(
new LocaTable::Builder(font_builder, header, table_data));
}/* else if (tag == Tag::prep) {
builder_raw = static_cast<Table::Builder*>(
new ControlProgramTable::Builder(font_builder, header, table_data));
}*/ else if (tag == Tag::bhed) {
builder_raw = static_cast<Table::Builder*>(
new FontHeaderTable::Builder(font_builder, header, table_data));
} else {
builder_raw = static_cast<Table::Builder*>(
new Table::GenericTableBuilder(font_builder, header, table_data));
}
builder = builder_raw;
return builder.detach();
}
/******************************************************************************
* Table::TableBasedTableBuilder class
******************************************************************************/
Table::TableBasedTableBuilder::TableBasedTableBuilder(
FontDataTableBuilderContainer* font_builder, Header* header,
WritableFontData* data) : Builder(font_builder, header, data) {}
Table::TableBasedTableBuilder::TableBasedTableBuilder(
FontDataTableBuilderContainer* font_builder, Header* header,
ReadableFontData* data) : Builder(font_builder, header, data) {}
Table::TableBasedTableBuilder::TableBasedTableBuilder(
FontDataTableBuilderContainer* font_builder, Header* header) :
Builder(font_builder, header) {}
Table::TableBasedTableBuilder::~TableBasedTableBuilder() {}
Table* Table::TableBasedTableBuilder::table() {
if (table_ == NULL) {
table_.attach(down_cast<Table*>(subBuildTable(internalReadData())));
}
return table_;
}
void Table::TableBasedTableBuilder::subDataSet() {
table_ = NULL;
}
int32_t Table::TableBasedTableBuilder::subDataSizeToSerialize() {
return 0;
}
bool Table::TableBasedTableBuilder::subReadyToSerialize() {
return false;
}
int32_t Table::TableBasedTableBuilder::subSerialize(WritableFontData* data) {
UNREFERENCED_PARAMETER(data);
return 0;
}
/******************************************************************************
* Table::GenericTableBuilder class
******************************************************************************/
Table::GenericTableBuilder::GenericTableBuilder(
FontDataTableBuilderContainer* font_builder, Header* header,
WritableFontData* data) :
TableBasedTableBuilder(font_builder, header, data) {}
CALLER_ATTACH FontDataTable* Table::GenericTableBuilder::subBuildTable(
ReadableFontData* data) {
// Note: In C++ port, we use GenericTable, the ref-counted version of Table
UNREFERENCED_PARAMETER(data);
FontDataTablePtr table = new GenericTable(this->header(), internalReadData());
return table.detach();
}
/******************************************************************************
* Table::ArrayElementTableBuilder class
******************************************************************************/
Table::ArrayElementTableBuilder::ArrayElementTableBuilder(
FontDataTableBuilderContainer* font_builder, Header* header,
WritableFontData* data) : Builder(font_builder, header, data) {}
Table::ArrayElementTableBuilder::ArrayElementTableBuilder(
FontDataTableBuilderContainer* font_builder, Header* header,
ReadableFontData* data) : Builder(font_builder, header, data) {}
Table::ArrayElementTableBuilder::~ArrayElementTableBuilder() {}
} // namespace sfntly