mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-31 22:12:44 +00:00 
			
		
		
		
	 44221756ab
			
		
	
	
		44221756ab
		
	
	
	
	
		
			
			"Records" in the spec are basically C++ classes, so let's drop this mouthful of a suffix.
		
			
				
	
	
		
			119 lines
		
	
	
	
		
			3.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			119 lines
		
	
	
	
		
			3.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2020-2021, Andreas Kling <kling@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <AK/String.h>
 | |
| #include <LibJS/Runtime/Environment.h>
 | |
| #include <LibJS/Runtime/PropertyName.h>
 | |
| #include <LibJS/Runtime/Value.h>
 | |
| 
 | |
| namespace JS {
 | |
| 
 | |
| class Reference {
 | |
| public:
 | |
|     enum class BaseType : u8 {
 | |
|         Unresolvable,
 | |
|         Value,
 | |
|         Environment,
 | |
|     };
 | |
| 
 | |
|     Reference() { }
 | |
|     Reference(BaseType type, PropertyName const& name, bool strict)
 | |
|         : m_base_type(type)
 | |
|         , m_name(name)
 | |
|         , m_strict(strict)
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     Reference(Value base, PropertyName const& name, Value this_value, bool strict = false)
 | |
|         : m_base_type(BaseType::Value)
 | |
|         , m_base_value(base)
 | |
|         , m_name(name)
 | |
|         , m_this_value(this_value)
 | |
|         , m_strict(strict)
 | |
|     {
 | |
|         if (base.is_nullish()) {
 | |
|             m_base_type = BaseType::Unresolvable;
 | |
|             m_base_value = {};
 | |
|             m_this_value = {};
 | |
|             m_name = {};
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     Reference(Environment& base, FlyString const& referenced_name, bool strict = false)
 | |
|         : m_base_type(BaseType::Environment)
 | |
|         , m_base_environment(&base)
 | |
|         , m_name(referenced_name)
 | |
|         , m_strict(strict)
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     Value base() const
 | |
|     {
 | |
|         VERIFY(m_base_type == BaseType::Value);
 | |
|         return m_base_value;
 | |
|     }
 | |
| 
 | |
|     Environment& base_environment() const
 | |
|     {
 | |
|         VERIFY(m_base_type == BaseType::Environment);
 | |
|         return *m_base_environment;
 | |
|     }
 | |
| 
 | |
|     PropertyName const& name() const { return m_name; }
 | |
|     bool is_strict() const { return m_strict; }
 | |
| 
 | |
|     // 6.2.4.2 IsUnresolvableReference ( V ), https://tc39.es/ecma262/#sec-isunresolvablereference
 | |
|     bool is_unresolvable() const { return m_base_type == BaseType::Unresolvable; }
 | |
| 
 | |
|     // 6.2.4.1 IsPropertyReference ( V ), https://tc39.es/ecma262/#sec-ispropertyreference
 | |
|     bool is_property_reference() const
 | |
|     {
 | |
|         if (is_unresolvable())
 | |
|             return false;
 | |
|         if (m_base_type == BaseType::Environment)
 | |
|             return false;
 | |
|         if (m_base_value.is_boolean() || m_base_value.is_string() || m_base_value.is_symbol() || m_base_value.is_bigint() || m_base_value.is_number() || m_base_value.is_object())
 | |
|             return true;
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     // 6.2.4.7 GetThisValue ( V ), https://tc39.es/ecma262/#sec-getthisvalue
 | |
|     Value get_this_value() const
 | |
|     {
 | |
|         VERIFY(is_property_reference());
 | |
|         if (is_super_reference())
 | |
|             return m_this_value;
 | |
|         return m_base_value;
 | |
|     }
 | |
| 
 | |
|     // 6.2.4.3 IsSuperReference ( V ), https://tc39.es/ecma262/#sec-issuperreference
 | |
|     bool is_super_reference() const
 | |
|     {
 | |
|         return !m_this_value.is_empty();
 | |
|     }
 | |
| 
 | |
|     void put_value(GlobalObject&, Value);
 | |
|     Value get_value(GlobalObject&, bool throw_if_undefined = true);
 | |
|     bool delete_(GlobalObject&);
 | |
| 
 | |
|     String to_string() const;
 | |
| 
 | |
| private:
 | |
|     void throw_reference_error(GlobalObject&);
 | |
| 
 | |
|     BaseType m_base_type { BaseType::Unresolvable };
 | |
|     union {
 | |
|         Value m_base_value {};
 | |
|         Environment* m_base_environment;
 | |
|     };
 | |
|     PropertyName m_name;
 | |
|     Value m_this_value;
 | |
|     bool m_strict { false };
 | |
| };
 | |
| 
 | |
| }
 |