/*
***************************************************************************************
* Copyright (C) 2006 EsperTech, Inc. All rights reserved. *
* http://www.espertech.com/esper *
* http://www.espertech.com *
* ---------------------------------------------------------------------------------- *
* The software in this package is published under the terms of the GPL license *
* a copy of which has been included with this distribution in the license.txt file. *
***************************************************************************************
*/
package com.espertech.esper.epl.index.quadtree;
import com.espertech.esper.client.EventBean;
import com.espertech.esper.epl.expression.core.ExprEvaluatorContext;
import com.espertech.esper.epl.join.table.EventTableOrganization;
import com.espertech.esper.spatial.quadtree.core.BoundingBox;
import com.espertech.esper.spatial.quadtree.mxcif.MXCIFQuadTree;
import com.espertech.esper.spatial.quadtree.mxcifrowindex.MXCIFQuadTreeRowIndexAdd;
import com.espertech.esper.spatial.quadtree.mxcifrowindex.MXCIFQuadTreeRowIndexQuery;
import com.espertech.esper.spatial.quadtree.mxcifrowindex.MXCIFQuadTreeRowIndexRemove;
import java.util.Collection;
import java.util.Iterator;
import static com.espertech.esper.epl.index.quadtree.AdvancedIndexQuadTreeConstants.*;
import static com.espertech.esper.epl.index.service.AdvancedIndexEvaluationHelper.evalDoubleColumn;
import static com.espertech.esper.epl.index.service.AdvancedIndexEvaluationHelper.invalidColumnValue;
public class EventTableQuadTreeMXCIFImpl implements EventTableQuadTree {
private final EventTableOrganization organization;
private final EventBean[] eventsPerStream = new EventBean[1];
private final AdvancedIndexConfigStatementMXCIFQuadtree config;
private final MXCIFQuadTree<Object> quadTree;
public EventTableQuadTreeMXCIFImpl(EventTableOrganization organization, AdvancedIndexConfigStatementMXCIFQuadtree config, MXCIFQuadTree<Object> quadTree) {
this.organization = organization;
this.config = config;
this.quadTree = quadTree;
}
public Collection<EventBean> queryRange(double x, double y, double width, double height) {
return (Collection<EventBean>) (Collection) MXCIFQuadTreeRowIndexQuery.queryRange(quadTree, x, y, width, height);
}
public void addRemove(EventBean[] newData, EventBean[] oldData, ExprEvaluatorContext exprEvaluatorContext) {
remove(oldData, exprEvaluatorContext);
add(newData, exprEvaluatorContext);
}
public void add(EventBean[] events, ExprEvaluatorContext exprEvaluatorContext) {
for (EventBean added : events) {
add(added, exprEvaluatorContext);
}
}
public void remove(EventBean[] events, ExprEvaluatorContext exprEvaluatorContext) {
for (EventBean removed : events) {
remove(removed, exprEvaluatorContext);
}
}
public void add(EventBean event, ExprEvaluatorContext exprEvaluatorContext) {
eventsPerStream[0] = event;
double x = evalDoubleColumn(config.getxEval(), organization.getIndexName(), COL_X, eventsPerStream, true, exprEvaluatorContext);
double y = evalDoubleColumn(config.getyEval(), organization.getIndexName(), COL_Y, eventsPerStream, true, exprEvaluatorContext);
double width = evalDoubleColumn(config.getWidthEval(), organization.getIndexName(), COL_WIDTH, eventsPerStream, true, exprEvaluatorContext);
double height = evalDoubleColumn(config.getHeightEval(), organization.getIndexName(), COL_HEIGHT, eventsPerStream, true, exprEvaluatorContext);
boolean added = MXCIFQuadTreeRowIndexAdd.add(x, y, width, height, event, quadTree, organization.isUnique(), organization.getIndexName());
if (!added) {
throw invalidColumnValue(organization.getIndexName(), "(x,y,width,height)", "(" + x + "," + y + "," + width + "," + height + ")", "a value intersecting index bounding box (range-end-inclusive) " + quadTree.getRoot().getBb());
}
}
public void remove(EventBean event, ExprEvaluatorContext exprEvaluatorContext) {
eventsPerStream[0] = event;
double x = evalDoubleColumn(config.getxEval(), organization.getIndexName(), COL_X, eventsPerStream, false, exprEvaluatorContext);
double y = evalDoubleColumn(config.getyEval(), organization.getIndexName(), COL_Y, eventsPerStream, false, exprEvaluatorContext);
double width = evalDoubleColumn(config.getWidthEval(), organization.getIndexName(), COL_WIDTH, eventsPerStream, false, exprEvaluatorContext);
double height = evalDoubleColumn(config.getHeightEval(), organization.getIndexName(), COL_HEIGHT, eventsPerStream, false, exprEvaluatorContext);
MXCIFQuadTreeRowIndexRemove.remove(x, y, width, height, event, quadTree);
}
public Iterator<EventBean> iterator() {
BoundingBox bb = quadTree.getRoot().getBb();
Collection<EventBean> events = queryRange(bb.getMinX(), bb.getMinY(), bb.getMaxX() - bb.getMinX(), bb.getMaxY() - bb.getMinY());
return events.iterator();
}
public boolean isEmpty() {
return false; // assumed non-empty
}
public void clear() {
quadTree.clear();
}
public void destroy() {
}
public String toQueryPlan() {
return this.getClass().toString();
}
public Class getProviderClass() {
return this.getClass();
}
public Integer getNumberOfEvents() {
return null;
}
public int getNumKeys() {
return -1;
}
public Object getIndex() {
return quadTree;
}
public EventTableOrganization getOrganization() {
return organization;
}
}