using System.Diagnostics.CodeAnalysis;
using Jint.Collections;
using Jint.Native;
using Module = Jint.Runtime.Modules.Module;
namespace Jint.Runtime.Environments;
///
/// Represents a module environment record
/// https://tc39.es/ecma262/#sec-module-environment-records
///
internal sealed class ModuleEnvironment : DeclarativeEnvironment
{
private readonly HybridDictionary _importBindings = new();
internal ModuleEnvironment(Engine engine) : base(engine, false)
{
}
///
/// https://tc39.es/ecma262/#sec-module-environment-records-getthisbinding
///
internal override JsValue GetThisBinding()
{
return Undefined;
}
///
/// https://tc39.es/ecma262/#sec-createimportbinding
///
public void CreateImportBinding(string importName, Module module, string name)
{
_importBindings[importName] = new IndirectBinding(module, name);
CreateImmutableBindingAndInitialize(importName, true, Undefined, DisposeHint.Normal);
}
///
/// https://tc39.es/ecma262/#sec-module-environment-records-getbindingvalue-n-s
///
internal override JsValue GetBindingValue(Key name, bool strict)
{
if (_importBindings.TryGetValue(name, out var indirectBinding))
{
return indirectBinding.Module._environment.GetBindingValue(indirectBinding.BindingName, true);
}
return base.GetBindingValue(name, strict);
}
internal override bool TryGetBinding(BindingName name, bool strict, [NotNullWhen(true)] out JsValue? value)
{
if (_importBindings.TryGetValue(name.Key, out var indirectBinding))
{
value = indirectBinding.Module._environment.GetBindingValue(indirectBinding.BindingName, strict: true);
return true;
}
return base.TryGetBinding(name, strict, out value);
}
///
/// https://tc39.es/ecma262/#sec-module-environment-records-hasthisbinding
///
internal override bool HasThisBinding() => true;
private readonly record struct IndirectBinding(Module Module, string BindingName);
}