DeclarativeEnvironment.h 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687
  1. /*
  2. * Copyright (c) 2020-2021, Andreas Kling <kling@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #pragma once
  7. #include <AK/FlyString.h>
  8. #include <AK/HashMap.h>
  9. #include <LibJS/Runtime/Completion.h>
  10. #include <LibJS/Runtime/Environment.h>
  11. #include <LibJS/Runtime/Value.h>
  12. namespace JS {
  13. class DeclarativeEnvironment : public Environment {
  14. JS_ENVIRONMENT(DeclarativeEnvironment, Environment);
  15. struct Binding {
  16. FlyString name;
  17. Value value;
  18. bool strict { false };
  19. bool mutable_ { false };
  20. bool can_be_deleted { false };
  21. bool initialized { false };
  22. };
  23. public:
  24. static DeclarativeEnvironment* create_for_per_iteration_bindings(Badge<ForStatement>, DeclarativeEnvironment& other, size_t bindings_size);
  25. DeclarativeEnvironment();
  26. explicit DeclarativeEnvironment(Environment* parent_environment);
  27. explicit DeclarativeEnvironment(Environment* parent_environment, Span<Binding const> bindings);
  28. virtual ~DeclarativeEnvironment() override = default;
  29. virtual ThrowCompletionOr<bool> has_binding(FlyString const& name, Optional<size_t>* = nullptr) const override;
  30. virtual ThrowCompletionOr<void> create_mutable_binding(GlobalObject&, FlyString const& name, bool can_be_deleted) override;
  31. virtual ThrowCompletionOr<void> create_immutable_binding(GlobalObject&, FlyString const& name, bool strict) override;
  32. virtual ThrowCompletionOr<void> initialize_binding(GlobalObject&, FlyString const& name, Value) override;
  33. virtual ThrowCompletionOr<void> set_mutable_binding(GlobalObject&, FlyString const& name, Value, bool strict) override;
  34. virtual ThrowCompletionOr<Value> get_binding_value(GlobalObject&, FlyString const& name, bool strict) override;
  35. virtual ThrowCompletionOr<bool> delete_binding(GlobalObject&, FlyString const& name) override;
  36. void initialize_or_set_mutable_binding(Badge<ScopeNode>, GlobalObject& global_object, FlyString const& name, Value value);
  37. ThrowCompletionOr<void> initialize_or_set_mutable_binding(GlobalObject& global_object, FlyString const& name, Value value);
  38. // This is not a method defined in the spec! Do not use this in any LibJS (or other spec related) code.
  39. [[nodiscard]] Vector<FlyString> bindings() const
  40. {
  41. Vector<FlyString> names;
  42. names.ensure_capacity(m_bindings.size());
  43. for (auto const& binding : m_bindings)
  44. names.unchecked_append(binding.name);
  45. return names;
  46. }
  47. ThrowCompletionOr<void> initialize_binding_direct(GlobalObject&, size_t index, Value);
  48. ThrowCompletionOr<Value> get_binding_value_direct(GlobalObject&, size_t index, bool strict);
  49. ThrowCompletionOr<void> set_mutable_binding_direct(GlobalObject&, size_t index, Value, bool strict);
  50. protected:
  51. virtual void visit_edges(Visitor&) override;
  52. private:
  53. virtual bool is_declarative_environment() const override { return true; }
  54. Optional<size_t> find_binding_index(FlyString const& name) const
  55. {
  56. auto it = m_bindings.find_if([&](auto const& binding) {
  57. return binding.name == name;
  58. });
  59. if (it == m_bindings.end())
  60. return {};
  61. return it.index();
  62. }
  63. Vector<Binding> m_bindings;
  64. };
  65. template<>
  66. inline bool Environment::fast_is<DeclarativeEnvironment>() const { return is_declarative_environment(); }
  67. }