/* * Copyright (c) 2020, Matthew Olsson * Copyright (c) 2024, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include namespace JS { ThrowCompletionOr> regexp_create(VM&, Value pattern, Value flags); ThrowCompletionOr> regexp_alloc(VM&, FunctionObject& new_target); Result, ByteString> regex_flags_from_string(StringView flags); struct ParseRegexPatternError { ByteString error; }; ErrorOr parse_regex_pattern(StringView pattern, bool unicode, bool unicode_sets); ThrowCompletionOr parse_regex_pattern(VM& vm, StringView pattern, bool unicode, bool unicode_sets); class RegExpObject : public Object { JS_OBJECT(RegExpObject, Object); GC_DECLARE_ALLOCATOR(RegExpObject); public: // JS regexps are all 'global' by default as per our definition, but the "global" flag enables "stateful". // FIXME: Enable 'BrowserExtended' only if in a browser context. static constexpr regex::RegexOptions default_flags { (regex::ECMAScriptFlags)regex::AllFlags::SingleMatch | (regex::ECMAScriptFlags)regex::AllFlags::Global | (regex::ECMAScriptFlags)regex::AllFlags::SkipTrimEmptyMatches | regex::ECMAScriptFlags::BrowserExtended }; enum class Flags { HasIndices = 1 << 0, Global = 1 << 1, IgnoreCase = 1 << 2, Multiline = 1 << 3, DotAll = 1 << 4, UnicodeSets = 1 << 5, Unicode = 1 << 6, Sticky = 1 << 7, }; static GC::Ref create(Realm&); static GC::Ref create(Realm&, Regex regex, ByteString pattern, ByteString flags); ThrowCompletionOr> regexp_initialize(VM&, Value pattern, Value flags); ByteString escape_regexp_pattern() const; virtual void initialize(Realm&) override; virtual ~RegExpObject() override = default; ByteString const& pattern() const { return m_pattern; } ByteString const& flags() const { return m_flags; } Flags flag_bits() const { return m_flag_bits; } Regex const& regex() { return *m_regex; } Regex const& regex() const { return *m_regex; } Realm& realm() { return *m_realm; } Realm const& realm() const { return *m_realm; } bool legacy_features_enabled() const { return m_legacy_features_enabled; } void set_legacy_features_enabled(bool legacy_features_enabled) { m_legacy_features_enabled = legacy_features_enabled; } void set_realm(Realm& realm) { m_realm = &realm; } private: RegExpObject(Object& prototype); RegExpObject(Regex regex, ByteString pattern, ByteString flags, Object& prototype); virtual void visit_edges(Visitor&) override; ByteString m_pattern; ByteString m_flags; Flags m_flag_bits { 0 }; bool m_legacy_features_enabled { false }; // [[LegacyFeaturesEnabled]] // Note: This is initialized in RegExpAlloc, but will be non-null afterwards GC::Ptr m_realm; // [[Realm]] Optional> m_regex; }; AK_ENUM_BITWISE_OPERATORS(RegExpObject::Flags); }