// Tangible_Is.js
(function () {
  /* global adventurejs A */
  /**
   * @ajspath adventurejs.Atom.StateManager.Asset_Is.Tangible_Is
   * @augments adventurejs.Asset_Is
   * @class adventurejs.Tangible_Is
   * @ajsnavheading StateClasses
   * @param {String} game_name Name of top level game instance that is scoped to window.
   * @param {String} name Instance name.
   * @summary A container for state variables.
   * @classdesc
   * <strong>Tangible_Is.js</strong> is a state management class
   * used to handle a variety of properties for
   * {@link adventurejs.Tangible|Tangible Assets}.
   **/
  class Tangible_Is extends adventurejs.Asset_Is {
    constructor(name = "is", game_name, context_id) {
      // Call the constructor of the super class
      super(name, game_name, context_id);
      this.class = "Tangible_Is";
      /**
       * Used to track whether player has seen an asset.
       * @var {Boolean} adventurejs.Asset#is!known
       * @default false
       */
      this._seen = false;
      //this._known = false;
      /**
       * Set whether this asset is plugged, like a drain.
       * @var {Boolean} adventurejs.Asset#is!plugged
       * @default false
       */
      this._plugged = false;
      /**
       * Set whether this asset is plugged in, like an appliance.
       * @var {Boolean} adventurejs.Asset#is!pluggedIn
       * @default false
       */
      this._pluggedIn = false;
      /**
       * Set whether this asset is hidden.
       * @var {Getter|Boolean} adventurejs.Tangible#is!hidden
       * @default false
       * @todo add logic to test conditions?
       */
      this.hidden = false;
      /**
       * Set whether this asset is supported. Used in conjuction with
       * can.swing_on_if_holding_and_supported to determine whether player can swing
       * on the asset in its current state. Meant for assets like two-ended ropes
       * which must be tied to a supporting structure.
       * @var {Boolean} adventurejs.Tangible#is!supported
       * @default false
       */
      this.supported = false;
      /**
       * Set whether this asset is hollow. Used to determine whether player can look in asset.
       * @var {Boolean} adventurejs.Tangible#is!hollow
       * @default false
       */
      this.hollow = false; // determines whether you can look in
      /**
       * Set whether this asset is listed as an inventory item.
       * If parent is a room, this means the asset won't be listed along
       * with other inventory in the room. This is useful for fixed
       * items that are mentioned in the room description, such as furniture.
       * For an example of a parent that is not a room, consider a desk drawer
       * that is attached to a desk and mentioned in the desk's description.
       * The drawer should not be listed along with inventory items on the desk.
       * @var {Boolean} adventurejs.Tangible#is!listed
       * @default true
       */
      this.listed = true;
      /**
       * Set whether this asset is distant. Originally meant to help define scenery assets,
       * but mostly superseded by other scenery handling and no logic has been written for it.
       * @var {Boolean} adventurejs.Tangible#is!distant
       * @default false
       */
      this.distant = false;
      /**
       * Set whether this asset is watertight.
       * No logic has been written for this.
       * @var {Boolean} adventurejs.Tangible#is!watertight
       * @default false
       * @todo Write logic for this. Is this the same as airtight?
       */
      this.watertight = false;
      /**
       * Set whether this asset is being worn by player or other characters.
       * @var {Boolean} adventurejs.Tangible#is!worn
       * @default false
       */
      this.worn = false;
      /**
       * Set whether this asset is screwed.
       * @var {Boolean|Int} adventurejs.Tangible#is!screwed
       * @default false
       */
      this.screwed = false;
      /**
       * Set whether this asset is zipped.
       * @var {Boolean} adventurejs.Tangible#is!zipped
       * @default false
       */
      this.zipped = false;
      /**
       * Set whether this asset is buttoned.
       * @var {Boolean} adventurejs.Tangible#is!buttoned
       * @default false
       */
      this.buttoned = false;
      /**
       * Set whether this asset is fixed in its place, such as drawers in a desk.
       * (This may be redundant to can_be_taken.)
       * @var {Boolean} adventurejs.Tangible#is!fixed
       * @default false
       */
      this.fixed = false;
      /**
       * Set whether this asset is locked.
       * @var {Boolean} adventurejs.Tangible#is!locked
       * @default false
       */
      this._locked = false;
      /**
       * Set whether this asset is closed.
       * @var {Boolean} adventurejs.Tangible#is!closed
       * @default false
       */
      this._closed = false;
      /**
       * Set whether this asset is sealed.
       * @var {Boolean} adventurejs.Tangible#is!sealed
       * @default false
       */
      this._sealed = false;
      /**
       * Set whether this asset can be looked through,
       * like a window or a telescope.
       * @var {Boolean} adventurejs.Tangible#is!viewport
       * @default false
       */
      this.viewport = false;
      /**
       * Set whether player can nest inside this asset when this asset is in an
       * aspect of another asset (as opposed to being a direct child of a room).
       * For example, a carousel horse might be on a carousel platform.
       * @var {Boolean} adventurejs.Tangible#is!deep_nest
       * @default false
       */
      this.deep_nest = false;
      /**
       *
       * @var {Boolean} adventurejs.Tangible#is!false_nest
       * @default false
       */
      this.false_nest = false;
      /**
       * Set whether asset is a light source.
       * @var {Boolean} adventurejs.Tangible#is!light_source
       * @default false
       */
      this.light_source = false;
      /**
       * Set whether asset is a heat source.
       * @var {Boolean} adventurejs.Tangible#is!heat_source
       * @default false
       */
      this.heat_source = false;
      /**
       * Set whether asset is a cold source.
       * @var {Boolean} adventurejs.Tangible#is!cold_source
       * @default false
       */
      this.cold_source = false;
      /**
       * Set whether asset is on.
       * @var {Boolean} adventurejs.Tangible#is!on
       * @default false
       */
      this.on = false;
      /**
       * Set whether asset is connected by verbs to other assets.
       * @var {Boolean} adventurejs.Tangible#is!connected_by
       * @default {}
       */
      this.connected_by = {};
      /**
       * Set whether asset is on, such as a switch.
       * @var {Boolean} adventurejs.Tangible#is!on
       * @default false
       */
      this.on = false;
      return this;
    }
    get off() {
      return !this.on;
    }
    get emitting_light() {
      return this.light_source && this.on;
    }
    /**
     * We store an asset's seen state per character so that
     * each character can have distinct knowledge. However,
     * the seen property needs to be accessed during game
     * construction, before the player character is available.
     * So, during construction, we save state to an object
     * property, and during play, we save state to the active
     * character.
     */
    get seen() {
      if (this.game.game_state < this.game.game_states.PLAYING) {
        return this._seen;
      } else {
        return this.game.getInput().getSubject().has_seen[this.context_id]
          ? true
          : false;
      }
    }
    set seen(value) {
      const subject = this.game.getSubject();
      let self = this.context;
      let linked_asset;
      this._seen = value;
      if (self && self.linked_asset) {
        linked_asset = this.game.getAsset(self.linked_asset);
      }
      if (linked_asset) {
        linked_asset.is._seen = value;
      }
      // see contents
      if (self && self.aspects) {
        for (let item in self.aspects) {
          let aspect = self.aspects[item];
          if (
            aspect.name === "in" &&
            self.isDOV("open") &&
            this.closed &&
            self.appearance.opacity >= 1
          )
            continue;
          if (aspect.see_contents_with_parent) {
            let contents = aspect.contents;
            for (let i = 0; i < contents.length; i++) {
              let nested_asset = self.game.getAsset(contents[i]);
              nested_asset.is._seen = value;
            }
          }
        }
      }
    }
    get plugged() {
      return this._plugged;
    }
    set plugged(value) {
      this._plugged = value;
    }
    get pluggedIn() {
      return this._pluggedIn;
    }
    set pluggedIn(value) {
      this._pluggedIn = value;
    }
    get sealed() {
      return this._sealed;
    }
    set sealed(value) {
      this._sealed = value;
    }
    get closed() {
      return this._closed;
    }
    set closed(value) {
      this._closed = value;
    }
    get locked() {
      return this._locked;
    }
    set locked(value) {
      this._locked = value;
    }
    /**
     * Get whether this asset has a vessel that is a reservoir.
     * @var {Getter} adventurejs.Tangible#is!reservoir
     * @returns {Boolean}
     */
    get reservoir() {
      return this.context.aspects.in?.vessel?.is?.reservoir;
    }
    /**
     * Get whether this asset is present in the current room.
     * @var {Getter} adventurejs.Tangible#is!present
     * @returns {Boolean}
     */
    get present() {
      return this.game.parser.selectPresent(this.context.id).length
        ? true
        : false;
    }
    /**
     * Get whether this asset is carried by player.
     * @var {Getter} adventurejs.Tangible#is!carried
     * @returns {Boolean}
     */
    get carried() {
      return this.context.isWithin(this.game.getPlayer());
    }
    /**
     * Get whether this asset is in player's hands.
     * @var {Getter} adventurejs.Tangible#is!inhands
     * @returns {Boolean}
     */
    get inhands() {
      return this.game.parser.selectInHands(this.id).length ? true : false;
    }
    /**
     * Get whether this asset is reachable by player.
     * @var {Getter} adventurejs.Tangible#is!reachable
     * @returns {Boolean}
     */
    get reachable() {
      return this.game.parser.selectReachable(this.id).length ? true : false;
    }
    /**
     * Get whether this asset is visible by player.
     * @var {Getter} adventurejs.Tangible#is!visible
     * @returns {Boolean}
     */
    get visible() {
      return this.game.parser.selectVisible(this.id).length ? true : false;
    }
    /**
     * Get whether this asset is takeable by player.
     * @var {Getter} adventurejs.Tangible#is!takeable
     * @returns {Boolean}
     */
    get takeable() {
      return this.isDOV("take");
    }
  }
  adventurejs.Tangible_Is = Tangible_Is;
})();