Intrinsics.h 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108
  1. /*
  2. * Copyright (c) 2022, Andrew Kaster <akaster@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #pragma once
  7. #include <AK/FlyString.h>
  8. #include <AK/Forward.h>
  9. #include <AK/HashMap.h>
  10. #include <LibJS/Forward.h>
  11. #include <LibJS/Heap/Cell.h>
  12. #include <LibJS/Heap/Heap.h>
  13. #include <LibJS/Runtime/VM.h>
  14. #include <LibWeb/Bindings/HostDefined.h>
  15. #define WEB_SET_PROTOTYPE_FOR_INTERFACE_WITH_CUSTOM_NAME(interface_class, interface_name) \
  16. do { \
  17. static auto name = #interface_name##_fly_string; \
  18. set_prototype(&Bindings::ensure_web_prototype<Bindings::interface_class##Prototype>(realm, name)); \
  19. } while (0)
  20. #define WEB_SET_PROTOTYPE_FOR_INTERFACE(interface_name) WEB_SET_PROTOTYPE_FOR_INTERFACE_WITH_CUSTOM_NAME(interface_name, interface_name)
  21. namespace Web::Bindings {
  22. class Intrinsics final : public JS::Cell {
  23. JS_CELL(Intrinsics, JS::Cell);
  24. JS_DECLARE_ALLOCATOR(Intrinsics);
  25. public:
  26. Intrinsics(JS::Realm& realm)
  27. : m_realm(realm)
  28. {
  29. }
  30. template<typename NamespaceType>
  31. JS::Object& ensure_web_namespace(FlyString const& namespace_name)
  32. {
  33. if (auto it = m_namespaces.find(namespace_name); it != m_namespaces.end())
  34. return *it->value;
  35. create_web_namespace<NamespaceType>(*m_realm);
  36. return *m_namespaces.find(namespace_name)->value;
  37. }
  38. template<typename PrototypeType>
  39. JS::Object& ensure_web_prototype(FlyString const& class_name)
  40. {
  41. if (auto it = m_prototypes.find(class_name); it != m_prototypes.end())
  42. return *it->value;
  43. create_web_prototype_and_constructor<PrototypeType>(*m_realm);
  44. return *m_prototypes.find(class_name)->value;
  45. }
  46. template<typename PrototypeType>
  47. JS::NativeFunction& ensure_web_constructor(FlyString const& class_name)
  48. {
  49. if (auto it = m_constructors.find(class_name); it != m_constructors.end())
  50. return *it->value;
  51. create_web_prototype_and_constructor<PrototypeType>(*m_realm);
  52. return *m_constructors.find(class_name)->value;
  53. }
  54. bool is_exposed(StringView name) const;
  55. private:
  56. virtual void visit_edges(JS::Cell::Visitor&) override;
  57. template<typename NamespaceType>
  58. void create_web_namespace(JS::Realm& realm);
  59. template<typename PrototypeType>
  60. void create_web_prototype_and_constructor(JS::Realm& realm);
  61. HashMap<FlyString, JS::NonnullGCPtr<JS::Object>> m_namespaces;
  62. HashMap<FlyString, JS::NonnullGCPtr<JS::Object>> m_prototypes;
  63. HashMap<FlyString, JS::GCPtr<JS::NativeFunction>> m_constructors;
  64. JS::NonnullGCPtr<JS::Realm> m_realm;
  65. };
  66. [[nodiscard]] inline Intrinsics& host_defined_intrinsics(JS::Realm& realm)
  67. {
  68. return *verify_cast<HostDefined>(realm.host_defined())->intrinsics;
  69. }
  70. template<typename T>
  71. [[nodiscard]] JS::Object& ensure_web_namespace(JS::Realm& realm, FlyString const& namespace_name)
  72. {
  73. return host_defined_intrinsics(realm).ensure_web_namespace<T>(namespace_name);
  74. }
  75. template<typename T>
  76. [[nodiscard]] JS::Object& ensure_web_prototype(JS::Realm& realm, FlyString const& class_name)
  77. {
  78. return host_defined_intrinsics(realm).ensure_web_prototype<T>(class_name);
  79. }
  80. template<typename T>
  81. [[nodiscard]] JS::NativeFunction& ensure_web_constructor(JS::Realm& realm, FlyString const& class_name)
  82. {
  83. return host_defined_intrinsics(realm).ensure_web_constructor<T>(class_name);
  84. }
  85. }