// 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_filespec.h" #include #include "build/build_config.h" #include "constants/stream_dict_common.h" #include "core/fpdfapi/parser/cpdf_dictionary.h" #include "core/fpdfapi/parser/cpdf_name.h" #include "core/fpdfapi/parser/cpdf_object.h" #include "core/fpdfapi/parser/cpdf_stream.h" #include "core/fpdfapi/parser/cpdf_string.h" #include "core/fpdfapi/parser/fpdf_parser_decode.h" #include "core/fxcrt/fx_system.h" namespace { #if defined(OS_MACOSX) || defined(OS_WIN) WideString ChangeSlashToPlatform(const wchar_t* str) { WideString result; while (*str) { if (*str == '/') { #if defined(OS_MACOSX) result += L':'; #else result += L'\\'; #endif } else { result += *str; } str++; } return result; } WideString ChangeSlashToPDF(const wchar_t* str) { WideString result; while (*str) { if (*str == '\\' || *str == ':') result += L'/'; else result += *str; str++; } return result; } #endif // defined(OS_MACOSX) || defined(OS_WIN) } // namespace CPDF_FileSpec::CPDF_FileSpec(const CPDF_Object* pObj) : m_pObj(pObj) { ASSERT(m_pObj); } CPDF_FileSpec::CPDF_FileSpec(CPDF_Object* pObj) : m_pObj(pObj), m_pWritableObj(pObj) { ASSERT(m_pObj); } CPDF_FileSpec::~CPDF_FileSpec() {} WideString CPDF_FileSpec::DecodeFileName(const WideString& filepath) { if (filepath.GetLength() <= 1) return WideString(); #if defined(OS_MACOSX) if (filepath.First(sizeof("/Mac") - 1) == WideStringView(L"/Mac")) return ChangeSlashToPlatform(filepath.c_str() + 1); return ChangeSlashToPlatform(filepath.c_str()); #elif defined(OS_WIN) if (filepath[0] != L'/') return ChangeSlashToPlatform(filepath.c_str()); if (filepath[1] == L'/') return ChangeSlashToPlatform(filepath.c_str() + 1); if (filepath[2] == L'/') { WideString result; result += filepath[1]; result += L':'; result += ChangeSlashToPlatform(filepath.c_str() + 2); return result; } WideString result; result += L'\\'; result += ChangeSlashToPlatform(filepath.c_str()); return result; #else return WideString(filepath); #endif } WideString CPDF_FileSpec::GetFileName() const { WideString csFileName; if (const CPDF_Dictionary* pDict = m_pObj->AsDictionary()) { const CPDF_String* pUF = ToString(pDict->GetDirectObjectFor("UF")); if (pUF) csFileName = pUF->GetUnicodeText(); if (csFileName.IsEmpty()) { const CPDF_String* pK = ToString(pDict->GetDirectObjectFor(pdfium::stream::kF)); if (pK) csFileName = WideString::FromDefANSI(pK->GetString().AsStringView()); } if (pDict->GetStringFor("FS") == "URL") return csFileName; if (csFileName.IsEmpty()) { for (const auto* key : {"DOS", "Mac", "Unix"}) { const CPDF_String* pValue = ToString(pDict->GetDirectObjectFor(key)); if (pValue) { csFileName = WideString::FromDefANSI(pValue->GetString().AsStringView()); break; } } } } else if (const CPDF_String* pString = m_pObj->AsString()) { csFileName = WideString::FromDefANSI(pString->GetString().AsStringView()); } return DecodeFileName(csFileName); } const CPDF_Stream* CPDF_FileSpec::GetFileStream() const { const CPDF_Dictionary* pDict = m_pObj->AsDictionary(); if (!pDict) return nullptr; // Get the embedded files dictionary. const CPDF_Dictionary* pFiles = pDict->GetDictFor("EF"); if (!pFiles) return nullptr; // List of keys to check for the file specification string. // Follows the same precedence order as GetFileName(). static constexpr const char* kKeys[] = {"UF", "F", "DOS", "Mac", "Unix"}; size_t end = pDict->GetStringFor("FS") == "URL" ? 2 : FX_ArraySize(kKeys); for (size_t i = 0; i < end; ++i) { ByteString key = kKeys[i]; if (!pDict->GetUnicodeTextFor(key).IsEmpty()) { const CPDF_Stream* pStream = pFiles->GetStreamFor(key); if (pStream) return pStream; } } return nullptr; } CPDF_Stream* CPDF_FileSpec::GetFileStream() { return const_cast( static_cast(this)->GetFileStream()); } const CPDF_Dictionary* CPDF_FileSpec::GetParamsDict() const { const CPDF_Stream* pStream = GetFileStream(); if (!pStream) return nullptr; const CPDF_Dictionary* pDict = pStream->GetDict(); return pDict ? pDict->GetDictFor("Params") : nullptr; } CPDF_Dictionary* CPDF_FileSpec::GetParamsDict() { return const_cast( static_cast(this)->GetParamsDict()); } WideString CPDF_FileSpec::EncodeFileName(const WideString& filepath) { if (filepath.GetLength() <= 1) return WideString(); #if defined(OS_WIN) if (filepath[1] == L':') { WideString result(L'/'); result += filepath[0]; if (filepath[2] != L'\\') result += L'/'; result += ChangeSlashToPDF(filepath.c_str() + 2); return result; } if (filepath[0] == L'\\' && filepath[1] == L'\\') return ChangeSlashToPDF(filepath.c_str() + 1); if (filepath[0] == L'\\') return L'/' + ChangeSlashToPDF(filepath.c_str()); return ChangeSlashToPDF(filepath.c_str()); #elif defined(OS_MACOSX) if (filepath.First(sizeof("Mac") - 1).EqualsASCII("Mac")) return L'/' + ChangeSlashToPDF(filepath.c_str()); return ChangeSlashToPDF(filepath.c_str()); #else return WideString(filepath); #endif } void CPDF_FileSpec::SetFileName(const WideString& wsFileName) { if (!m_pWritableObj) { NOTREACHED(); return; } WideString wsStr = EncodeFileName(wsFileName); if (m_pObj->IsString()) { m_pWritableObj->SetString(wsStr.ToDefANSI()); } else if (CPDF_Dictionary* pDict = m_pWritableObj->AsDictionary()) { pDict->SetNewFor(pdfium::stream::kF, wsStr.ToDefANSI(), false); pDict->SetNewFor("UF", wsStr); } }