| /* |
| * 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 |