|  | // Copyright 2016 PDFium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com | 
|  |  | 
|  | #include "core/fpdfdoc/cpdf_defaultappearance.h" | 
|  |  | 
|  | #include <algorithm> | 
|  | #include <vector> | 
|  |  | 
|  | #include "core/fpdfapi/parser/cpdf_simple_parser.h" | 
|  | #include "core/fpdfapi/parser/fpdf_parser_utility.h" | 
|  | #include "core/fxge/cfx_color.h" | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | // Find the token and its |nParams| parameters from the start of data, | 
|  | // and move the current position to the start of those parameters. | 
|  | bool FindTagParamFromStart(CPDF_SimpleParser* parser, | 
|  | ByteStringView token, | 
|  | int nParams) { | 
|  | nParams++; | 
|  |  | 
|  | std::vector<uint32_t> pBuf(nParams); | 
|  | int buf_index = 0; | 
|  | int buf_count = 0; | 
|  |  | 
|  | parser->SetCurPos(0); | 
|  | while (1) { | 
|  | pBuf[buf_index++] = parser->GetCurPos(); | 
|  | if (buf_index == nParams) | 
|  | buf_index = 0; | 
|  |  | 
|  | buf_count++; | 
|  | if (buf_count > nParams) | 
|  | buf_count = nParams; | 
|  |  | 
|  | ByteStringView word = parser->GetWord(); | 
|  | if (word.IsEmpty()) | 
|  | return false; | 
|  |  | 
|  | if (word == token) { | 
|  | if (buf_count < nParams) | 
|  | continue; | 
|  |  | 
|  | parser->SetCurPos(pBuf[buf_index]); | 
|  | return true; | 
|  | } | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | Optional<ByteString> CPDF_DefaultAppearance::GetFont(float* fFontSize) { | 
|  | *fFontSize = 0.0f; | 
|  | if (m_csDA.IsEmpty()) | 
|  | return {}; | 
|  |  | 
|  | ByteString csFontNameTag; | 
|  | CPDF_SimpleParser syntax(m_csDA.AsStringView().raw_span()); | 
|  | if (FindTagParamFromStart(&syntax, "Tf", 2)) { | 
|  | csFontNameTag = ByteString(syntax.GetWord()); | 
|  | csFontNameTag.Delete(0, 1); | 
|  | *fFontSize = StringToFloat(syntax.GetWord()); | 
|  | } | 
|  | return {PDF_NameDecode(csFontNameTag.AsStringView())}; | 
|  | } | 
|  |  | 
|  | Optional<CFX_Color::Type> CPDF_DefaultAppearance::GetColor(float fc[4]) { | 
|  | for (int c = 0; c < 4; c++) | 
|  | fc[c] = 0; | 
|  |  | 
|  | if (m_csDA.IsEmpty()) | 
|  | return {}; | 
|  |  | 
|  | CPDF_SimpleParser syntax(m_csDA.AsStringView().raw_span()); | 
|  | if (FindTagParamFromStart(&syntax, "g", 1)) { | 
|  | fc[0] = StringToFloat(syntax.GetWord()); | 
|  | return {CFX_Color::kGray}; | 
|  | } | 
|  | if (FindTagParamFromStart(&syntax, "rg", 3)) { | 
|  | fc[0] = StringToFloat(syntax.GetWord()); | 
|  | fc[1] = StringToFloat(syntax.GetWord()); | 
|  | fc[2] = StringToFloat(syntax.GetWord()); | 
|  | return {CFX_Color::kRGB}; | 
|  | } | 
|  | if (FindTagParamFromStart(&syntax, "k", 4)) { | 
|  | fc[0] = StringToFloat(syntax.GetWord()); | 
|  | fc[1] = StringToFloat(syntax.GetWord()); | 
|  | fc[2] = StringToFloat(syntax.GetWord()); | 
|  | fc[3] = StringToFloat(syntax.GetWord()); | 
|  | return {CFX_Color::kCMYK}; | 
|  | } | 
|  |  | 
|  | return {}; | 
|  | } | 
|  |  | 
|  | std::pair<Optional<CFX_Color::Type>, FX_ARGB> | 
|  | CPDF_DefaultAppearance::GetColor() { | 
|  | float values[4]; | 
|  | Optional<CFX_Color::Type> type = GetColor(values); | 
|  | if (!type) | 
|  | return {type, 0}; | 
|  |  | 
|  | if (*type == CFX_Color::kGray) { | 
|  | int g = static_cast<int>(values[0] * 255 + 0.5f); | 
|  | return {type, ArgbEncode(255, g, g, g)}; | 
|  | } | 
|  | if (*type == CFX_Color::kRGB) { | 
|  | int r = static_cast<int>(values[0] * 255 + 0.5f); | 
|  | int g = static_cast<int>(values[1] * 255 + 0.5f); | 
|  | int b = static_cast<int>(values[2] * 255 + 0.5f); | 
|  | return {type, ArgbEncode(255, r, g, b)}; | 
|  | } | 
|  | if (*type == CFX_Color::kCMYK) { | 
|  | float r = 1.0f - std::min(1.0f, values[0] + values[3]); | 
|  | float g = 1.0f - std::min(1.0f, values[1] + values[3]); | 
|  | float b = 1.0f - std::min(1.0f, values[2] + values[3]); | 
|  | return {type, ArgbEncode(255, static_cast<int>(r * 255 + 0.5f), | 
|  | static_cast<int>(g * 255 + 0.5f), | 
|  | static_cast<int>(b * 255 + 0.5f))}; | 
|  | } | 
|  | NOTREACHED(); | 
|  | return {{}, 0}; | 
|  | } | 
|  |  | 
|  | bool CPDF_DefaultAppearance::FindTagParamFromStartForTesting( | 
|  | CPDF_SimpleParser* parser, | 
|  | ByteStringView token, | 
|  | int nParams) { | 
|  | return FindTagParamFromStart(parser, token, nParams); | 
|  | } |