org.jetbrains.kotlin.js.backend.ast.JsScope Maven / Gradle / Ivy
// Copyright (c) 2011, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
package org.jetbrains.kotlin.js.backend.ast;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import org.jetbrains.kotlin.js.util.Maps;
import java.util.*;
/**
* A scope is a factory for creating and allocating
* {@link JsName}s. A JavaScript AST is
* built in terms of abstract name objects without worrying about obfuscation,
* keyword/identifier blacklisting, and so on.
*
*
*
* Scopes are associated with
* {@link JsFunction}s, but the two are
* not equivalent. Functions have scopes, but a scope does not
* necessarily have an associated Function. Examples of this include the
* {@link JsRootScope} and synthetic
* scopes that might be created by a client.
*
*
*
* Scopes can have parents to provide constraints when allocating actual
* identifiers for names. Specifically, names in child scopes are chosen such
* that they do not conflict with names in their parent scopes. The ultimate
* parent is usually the global scope (see
* {@link JsProgram#getRootScope()}),
* but parentless scopes are useful for managing names that are always accessed
* with a qualifier and could therefore never be confused with the global scope
* hierarchy.
*/
public abstract class JsScope {
@NotNull
private final String description;
private Map names = Collections.emptyMap();
private final JsScope parent;
public JsScope(JsScope parent, @NotNull String description) {
this.description = description;
this.parent = parent;
}
protected JsScope(@NotNull String description) {
this.description = description;
parent = null;
}
/**
* Gets a name object associated with the specified identifier in this scope,
* creating it if necessary.
* If the JsName does not exist yet, a new JsName is created. The identifier,
* short name, and original name of the newly created JsName are equal to
* the given identifier.
*
* @param identifier An identifier that is unique within this scope.
*/
@NotNull
public JsName declareName(@NotNull String identifier) {
JsName name = findOwnName(identifier);
return name != null ? name : doCreateName(identifier);
}
@NotNull
public static JsName declareTemporaryName(@NotNull String suggestedName) {
assert !suggestedName.isEmpty();
return new JsName(suggestedName, true);
}
/**
* Attempts to find the name object for the specified ident, searching in this
* scope, and if not found, in the parent scopes.
*
* @return null
if the identifier has no associated name
*/
@Nullable
public final JsName findName(@NotNull String ident) {
JsName name = findOwnName(ident);
if (name == null && parent != null) {
return parent.findName(ident);
}
return name;
}
public boolean hasOwnName(@NotNull String name) {
return names.containsKey(name);
}
private boolean hasName(@NotNull String name) {
return hasOwnName(name) || (parent != null && parent.hasName(name));
}
/**
* Returns the parent scope of this scope, or null
if this is the
* root scope.
*/
public final JsScope getParent() {
return parent;
}
@Override
public final String toString() {
if (parent != null) {
return description + "->" + parent;
}
else {
return description;
}
}
public void copyOwnNames(JsScope other) {
if (!other.names.isEmpty()) {
names = new HashMap<>(names);
names.putAll(other.names);
}
}
@NotNull
public String getDescription() {
return description;
}
@NotNull
protected JsName doCreateName(@NotNull String ident) {
JsName name = new JsName(ident, false);
names = Maps.put(names, ident, name);
return name;
}
/**
* Attempts to find the name object for the specified ident, searching in this
* scope only.
*
* @return null
if the identifier has no associated name
*/
protected JsName findOwnName(@NotNull String ident) {
return names.get(ident);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy