/* * Copyright (C) 2015 The Android Open Source Project * * 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. */ #ifndef AAPT_RESOURCE_VALUES_H #define AAPT_RESOURCE_VALUES_H #include #include #include #include #include "Resource.h" #include "ValueTransformer.h" #include "androidfw/IDiagnostics.h" #include "androidfw/ResourceTypes.h" #include "androidfw/StringPiece.h" #include "androidfw/StringPool.h" #include "io/File.h" #include "text/Printer.h" namespace aapt { class ValueVisitor; class ConstValueVisitor; // A resource value. This is an all-encompassing representation // of Item and Map and their subclasses. The way to do // type specific operations is to check the Value's type() and // cast it to the appropriate subclass. This isn't super clean, // but it is the simplest strategy. class Value { public: virtual ~Value() = default; // Whether this value is weak and can be overridden without warning or error. Default is false. bool IsWeak() const { return weak_; } void SetWeak(bool val) { weak_ = val; } // Whether the value is marked as translatable. This does not persist when flattened to binary. // It is only used during compilation phase. void SetTranslatable(bool val) { translatable_ = val; } // Default true. bool IsTranslatable() const { return translatable_; } // Returns the source where this value was defined. const android::Source& GetSource() const { return source_; } void SetSource(const android::Source& source) { source_ = source; } void SetSource(android::Source&& source) { source_ = std::move(source); } // Returns the comment that was associated with this resource. const std::string& GetComment() const { return comment_; } void SetComment(android::StringPiece str) { comment_.assign(str); } void SetComment(std::string&& str) { comment_ = std::move(str); } virtual bool Equals(const Value* value) const = 0; // Calls the appropriate overload of ValueVisitor. virtual void Accept(ValueVisitor* visitor) = 0; // Calls the appropriate overload of ConstValueVisitor. virtual void Accept(ConstValueVisitor* visitor) const = 0; // Transform this Value into another Value using the transformer. std::unique_ptr Transform(ValueTransformer& transformer) const; // Human readable printout of this value. virtual void Print(std::ostream* out) const = 0; // Human readable printout of this value that may omit some information for the sake // of brevity and readability. Default implementation just calls Print(). virtual void PrettyPrint(text::Printer* printer) const; friend std::ostream& operator<<(std::ostream& out, const Value& value); protected: android::Source source_; std::string comment_; bool weak_ = false; bool translatable_ = true; private: virtual Value* TransformValueImpl(ValueTransformer& transformer) const = 0; }; // Inherit from this to get visitor accepting implementations for free. template struct BaseValue : public Value { void Accept(ValueVisitor* visitor) override; void Accept(ConstValueVisitor* visitor) const override; }; // A resource item with a single value. This maps to android::ResTable_entry. struct Item : public Value { // Fills in an android::Res_value structure with this Item's binary representation. // Returns false if an error occurred. virtual bool Flatten(android::Res_value* out_value) const = 0; // Transform this Item into another Item using the transformer. std::unique_ptr Transform(ValueTransformer& transformer) const; private: virtual Item* TransformItemImpl(ValueTransformer& transformer) const = 0; }; // Inherit from this to get visitor accepting implementations for free. template struct BaseItem : public Item { void Accept(ValueVisitor* visitor) override; void Accept(ConstValueVisitor* visitor) const override; }; // A reference to another resource. This maps to android::Res_value::TYPE_REFERENCE. // A reference can be symbolic (with the name set to a valid resource name) or be // numeric (the id is set to a valid resource ID). struct Reference : public TransformableItem> { enum class Type : uint8_t { kResource, kAttribute, }; std::optional name; std::optional id; std::optional type_flags; Reference::Type reference_type; bool private_reference = false; bool is_dynamic = false; bool allow_raw = false; Reference(); explicit Reference(const ResourceNameRef& n, Type type = Type::kResource); explicit Reference(const ResourceId& i, Type type = Type::kResource); Reference(const ResourceNameRef& n, const ResourceId& i); bool Equals(const Value* value) const override; bool Flatten(android::Res_value* out_value) const override; void Print(std::ostream* out) const override; void PrettyPrint(text::Printer* printer) const override; // Prints the reference without a package name if the package name matches the one given. void PrettyPrint(android::StringPiece package, text::Printer* printer) const; }; bool operator<(const Reference&, const Reference&); bool operator==(const Reference&, const Reference&); // An ID resource. Has no real value, just a place holder. struct Id : public TransformableItem> { Id() { weak_ = true; } bool Equals(const Value* value) const override; bool Flatten(android::Res_value* out) const override; void Print(std::ostream* out) const override; }; // A raw, unprocessed string. This may contain quotations, escape sequences, and whitespace. // This shall *NOT* end up in the final resource table. struct RawString : public TransformableItem> { android::StringPool::Ref value; explicit RawString(const android::StringPool::Ref& ref); bool Equals(const Value* value) const override; bool Flatten(android::Res_value* out_value) const override; void Print(std::ostream* out) const override; }; // Identifies a range of characters in a string that are untranslatable. // These should not be pseudolocalized. The start and end indices are measured in bytes. struct UntranslatableSection { // Start offset inclusive. size_t start; // End offset exclusive. size_t end; }; inline bool operator==(const UntranslatableSection& a, const UntranslatableSection& b) { return a.start == b.start && a.end == b.end; } inline bool operator!=(const UntranslatableSection& a, const UntranslatableSection& b) { return a.start != b.start || a.end != b.end; } struct String : public TransformableItem> { android::StringPool::Ref value; // Sections of the string to NOT translate. Mainly used // for pseudolocalization. This data is NOT persisted // in any format. std::vector untranslatable_sections; explicit String(const android::StringPool::Ref& ref); bool Equals(const Value* value) const override; bool Flatten(android::Res_value* out_value) const override; void Print(std::ostream* out) const override; void PrettyPrint(text::Printer* printer) const override; }; struct StyledString : public TransformableItem> { android::StringPool::StyleRef value; // Sections of the string to NOT translate. Mainly used // for pseudolocalization. This data is NOT persisted // in any format. std::vector untranslatable_sections; explicit StyledString(const android::StringPool::StyleRef& ref); bool Equals(const Value* value) const override; bool Flatten(android::Res_value* out_value) const override; void Print(std::ostream* out) const override; }; struct FileReference : public TransformableItem> { android::StringPool::Ref path; // A handle to the file object from which this file can be read. // This field is NOT persisted in any format. It is transient. io::IFile* file = nullptr; // FileType of the file pointed to by `file`. This is used to know how to inflate the file, // or if to inflate at all (just copy). ResourceFile::Type type = ResourceFile::Type::kUnknown; FileReference() = default; explicit FileReference(const android::StringPool::Ref& path); bool Equals(const Value* value) const override; bool Flatten(android::Res_value* out_value) const override; void Print(std::ostream* out) const override; }; // Represents any other android::Res_value. struct BinaryPrimitive : public TransformableItem> { android::Res_value value; BinaryPrimitive() = default; explicit BinaryPrimitive(const android::Res_value& val); BinaryPrimitive(uint8_t dataType, uint32_t data); bool Equals(const Value* value) const override; bool Flatten(android::Res_value* out_value) const override; void Print(std::ostream* out) const override; static const char* DecideFormat(float f); void PrettyPrint(text::Printer* printer) const override; }; struct Attribute : public TransformableValue> { struct Symbol { Reference symbol; uint32_t value; uint8_t type; friend std::ostream& operator<<(std::ostream& out, const Symbol& symbol); }; uint32_t type_mask; int32_t min_int; int32_t max_int; std::vector symbols; explicit Attribute(uint32_t t = 0u); bool Equals(const Value* value) const override; // Returns true if this Attribute's format is compatible with the given Attribute. The basic // rule is that TYPE_REFERENCE can be ignored for both of the Attributes, and TYPE_FLAGS and // TYPE_ENUMS are never compatible. bool IsCompatibleWith(const Attribute& attr) const; std::string MaskString() const; static std::string MaskString(uint32_t type_mask); void Print(std::ostream* out) const override; bool Matches(const Item& item, android::DiagMessage* out_msg = nullptr) const; }; struct Style : public TransformableValue> { struct Entry { Reference key; std::unique_ptr value; friend std::ostream& operator<<(std::ostream& out, const Entry& entry); }; std::optional parent; // If set to true, the parent was auto inferred from the style's name. bool parent_inferred = false; std::vector entries; bool Equals(const Value* value) const override; void Print(std::ostream* out) const override; // Merges `style` into this Style. All identical attributes of `style` take precedence, including // the parent, if there is one. void MergeWith(Style* style, android::StringPool* pool); }; struct Array : public TransformableValue> { std::vector> elements; bool Equals(const Value* value) const override; void Print(std::ostream* out) const override; }; struct Plural : public TransformableValue> { enum { Zero = 0, One, Two, Few, Many, Other, Count }; std::array, Count> values; bool Equals(const Value* value) const override; void Print(std::ostream* out) const override; }; struct Styleable : public TransformableValue> { std::vector entries; bool Equals(const Value* value) const override; void Print(std::ostream* out) const override; void MergeWith(Styleable* styleable); }; struct Macro : public TransformableValue> { std::string raw_value; android::StyleString style_string; std::vector untranslatable_sections; struct Namespace { std::string alias; std::string package_name; bool is_private; bool operator==(const Namespace& right) const { return alias == right.alias && package_name == right.package_name && is_private == right.is_private; } }; std::vector alias_namespaces; bool Equals(const Value* value) const override; void Print(std::ostream* out) const override; }; template typename std::enable_if::value, std::ostream&>::type operator<<( std::ostream& out, const std::unique_ptr& value) { if (value == nullptr) { out << "NULL"; } else { value->Print(&out); } return out; } struct CloningValueTransformer : public ValueTransformer { explicit CloningValueTransformer(android::StringPool* new_pool); std::unique_ptr TransformDerived(const Reference* value) override; std::unique_ptr TransformDerived(const Id* value) override; std::unique_ptr TransformDerived(const RawString* value) override; std::unique_ptr TransformDerived(const String* value) override; std::unique_ptr TransformDerived(const StyledString* value) override; std::unique_ptr TransformDerived(const FileReference* value) override; std::unique_ptr TransformDerived(const BinaryPrimitive* value) override; std::unique_ptr TransformDerived(const Attribute* value) override; std::unique_ptr