All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.basex.query.func.fn.FnRemove Maven / Gradle / Ivy

There is a newer version: 11.3
Show newest version
package org.basex.query.func.fn;

import org.basex.query.*;
import org.basex.query.expr.*;
import org.basex.query.func.*;
import org.basex.query.iter.*;
import org.basex.query.value.*;
import org.basex.query.value.item.*;
import org.basex.query.value.seq.*;
import org.basex.query.value.type.*;

/**
 * Function implementation.
 *
 * @author BaseX Team 2005-22, BSD License
 * @author Christian Gruen
 */
public final class FnRemove extends StandardFunc {
  @Override
  public Iter iter(final QueryContext qc) throws QueryException {
    final Iter iter = exprs[0].iter(qc);
    final long pos = toLong(exprs[1], qc), size = iter.size();

    // position out of bounds: return original value
    if(pos <= 0 || size != -1 && pos > size) return iter;

    // check if iterator is value-based
    final Value value = iter.iterValue();
    if(value != null) return value(value, pos, qc).iter();

    return new Iter() {
      long c;

      @Override
      public Item next() throws QueryException {
        return ++c != pos || iter.next() != null ? qc.next(iter) : null;
      }
      @Override
      public Item get(final long i) throws QueryException {
        return iter.get(i + 1 < pos ? i : i + 1);
      }
      @Override
      public long size() {
        return Math.max(-1, size - 1);
      }
    };
  }

  @Override
  public Value value(final QueryContext qc) throws QueryException {
    return value(exprs[0].value(qc), toLong(exprs[1], qc), qc);
  }

  /**
   * Returns the result value.
   * @param value original value
   * @param pos position of the item to remove
   * @param qc query context
   * @return resulting value
   */
  private static Value value(final Value value, final long pos, final QueryContext qc) {
    final long size = value.size();
    // position out of bounds: return original value
    if(pos <= 0 || pos > size) return value;
    // remove first or last item (size > 0)
    if(pos == 1 || pos == size) return value.subsequence(pos == 1 ? 1 : 0, size - 1, qc);
    // remove item at supplied position
    return ((Seq) value).remove(pos - 1, qc);
  }

  @Override
  protected Expr opt(final CompileContext cc) throws QueryException {
    // ignore standard limitation for large values to speed up evaluation of result
    if(allAreValues(false)) return value(cc.qc);

    final Expr expr = exprs[0], pos = exprs[1];
    final SeqType st = expr.seqType();
    if(st.zero()) return expr;

    long sz = -1;
    if(pos instanceof Value) {
      // position is static...
      final long p = toLong(pos, cc.qc);
      // return all items
      final long size = expr.size();
      if(p < 1 || size > 0 && p > size) return expr;
      // skip first item
      if(p == 1) return cc.function(Function.TAIL, info, expr);
      // skip last item
      if(p == size) return cc.function(Function._UTIL_INIT, info, expr);
      // decrement result size
      sz--;
    }

    exprType.assign(st.union(Occ.ZERO), sz);
    data(expr.data());
    return this;
  }

  @Override
  public boolean ddo() {
    return exprs[0].ddo();
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy