mirror of
https://github.com/querydsl/querydsl.git
synced 2026-06-16 21:01:10 +08:00
217 lines
6.7 KiB
Java
217 lines
6.7 KiB
Java
/*
|
|
* Copyright 2011, Mysema Ltd
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
package com.mysema.query.sql;
|
|
|
|
import java.util.ArrayList;
|
|
import java.util.Arrays;
|
|
import java.util.List;
|
|
|
|
import com.google.common.collect.ImmutableList;
|
|
import com.mysema.query.types.Expression;
|
|
import com.mysema.query.types.MutableExpressionBase;
|
|
import com.mysema.query.types.Ops;
|
|
import com.mysema.query.types.OrderSpecifier;
|
|
import com.mysema.query.types.PathImpl;
|
|
import com.mysema.query.types.TemplateFactory;
|
|
import com.mysema.query.types.Visitor;
|
|
import com.mysema.query.types.expr.BooleanExpression;
|
|
import com.mysema.query.types.expr.ComparableExpressionBase;
|
|
import com.mysema.query.types.expr.SimpleExpression;
|
|
import com.mysema.query.types.expr.SimpleOperation;
|
|
import com.mysema.query.types.template.SimpleTemplate;
|
|
|
|
/**
|
|
* @author tiwe
|
|
*/
|
|
public class WindowFunction<A> extends MutableExpressionBase<A> {
|
|
|
|
private static final String ORDER_BY = "order by ";
|
|
|
|
private static final String PARTITION_BY = "partition by ";
|
|
|
|
private static final long serialVersionUID = -4130672293308756779L;
|
|
|
|
private final List<OrderSpecifier<?>> orderBy = new ArrayList<OrderSpecifier<?>>();
|
|
|
|
private final List<Expression<?>> partitionBy = new ArrayList<Expression<?>>();
|
|
|
|
private final Expression<A> target;
|
|
|
|
private volatile SimpleExpression<A> value;
|
|
|
|
private String rowsOrRange;
|
|
|
|
private List<Expression<?>> rowsOrRangeArgs;
|
|
|
|
public WindowFunction(Expression<A> expr) {
|
|
super(expr.getType());
|
|
this.target = expr;
|
|
}
|
|
|
|
public SimpleExpression<A> getValue() {
|
|
if (value == null) {
|
|
int size = 0;
|
|
ImmutableList.Builder<Expression<?>> args = ImmutableList.builder();
|
|
StringBuilder builder = new StringBuilder();
|
|
builder.append("{0} over (");
|
|
args.add(target);
|
|
size++;
|
|
if (!partitionBy.isEmpty()) {
|
|
builder.append(PARTITION_BY);
|
|
boolean first = true;
|
|
for (Expression<?> expr : partitionBy) {
|
|
if (!first) {
|
|
builder.append(", ");
|
|
}
|
|
builder.append("{" + size + "}");
|
|
args.add(expr);
|
|
size++;
|
|
first = false;
|
|
}
|
|
|
|
}
|
|
if (!orderBy.isEmpty()) {
|
|
if (!partitionBy.isEmpty()) {
|
|
builder.append(" ");
|
|
}
|
|
builder.append(ORDER_BY);
|
|
boolean first = true;
|
|
for (OrderSpecifier<?> expr : orderBy) {
|
|
if (!first) {
|
|
builder.append(", ");
|
|
}
|
|
builder.append("{" + size + "}");
|
|
if (!expr.isAscending()) {
|
|
builder.append(" desc");
|
|
}
|
|
args.add(expr.getTarget());
|
|
size++;
|
|
first = false;
|
|
}
|
|
}
|
|
if (rowsOrRange != null) {
|
|
builder.append(rowsOrRange);
|
|
args.addAll(rowsOrRangeArgs);
|
|
size += rowsOrRangeArgs.size();
|
|
}
|
|
builder.append(")");
|
|
value = new SimpleTemplate<A>(
|
|
target.getType(),
|
|
TemplateFactory.DEFAULT.create(builder.toString()),
|
|
args.build());
|
|
}
|
|
return value;
|
|
}
|
|
|
|
@SuppressWarnings("unchecked")
|
|
public SimpleExpression<A> as(Expression<A> alias) {
|
|
return SimpleOperation.create((Class<A>)getType(),Ops.ALIAS, this, alias);
|
|
}
|
|
|
|
public SimpleExpression<A> as(String alias) {
|
|
return SimpleOperation.create((Class<A>)getType(),Ops.ALIAS, this, new PathImpl<A>(getType(), alias));
|
|
}
|
|
|
|
@Override
|
|
public <R,C> R accept(Visitor<R,C> v, C context) {
|
|
return getValue().accept(v, context);
|
|
}
|
|
|
|
@Override
|
|
public boolean equals(Object o) {
|
|
if (o == this) {
|
|
return true;
|
|
} else if (o instanceof WindowFunction) {
|
|
WindowFunction<?> so = (WindowFunction<?>)o;
|
|
return so.target.equals(target)
|
|
&& so.partitionBy.equals(partitionBy)
|
|
&& so.orderBy.equals(orderBy);
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public BooleanExpression eq(Expression<A> expr) {
|
|
return getValue().eq(expr);
|
|
}
|
|
|
|
public BooleanExpression eq(A arg) {
|
|
return getValue().eq(arg);
|
|
}
|
|
|
|
public BooleanExpression ne(Expression<A> expr) {
|
|
return getValue().ne(expr);
|
|
}
|
|
|
|
public BooleanExpression ne(A arg) {
|
|
return getValue().ne(arg);
|
|
}
|
|
|
|
public WindowFunction<A> orderBy(ComparableExpressionBase<?> orderBy) {
|
|
value = null;
|
|
this.orderBy.add(orderBy.asc());
|
|
return this;
|
|
}
|
|
|
|
public WindowFunction<A> orderBy(ComparableExpressionBase<?>... orderBy) {
|
|
value = null;
|
|
for (ComparableExpressionBase<?> e : orderBy) {
|
|
this.orderBy.add(e.asc());
|
|
}
|
|
return this;
|
|
}
|
|
|
|
public WindowFunction<A> orderBy(OrderSpecifier<?> orderBy) {
|
|
value = null;
|
|
this.orderBy.add(orderBy);
|
|
return this;
|
|
}
|
|
|
|
public WindowFunction<A> orderBy(OrderSpecifier<?>... orderBy) {
|
|
value = null;
|
|
this.orderBy.addAll(Arrays.asList(orderBy));
|
|
return this;
|
|
}
|
|
|
|
public WindowFunction<A> partitionBy(Expression<?> partitionBy) {
|
|
value = null;
|
|
this.partitionBy.add(partitionBy);
|
|
return this;
|
|
}
|
|
|
|
public WindowFunction<A> partitionBy(Expression<?>... partitionBy) {
|
|
value = null;
|
|
this.partitionBy.addAll(Arrays.asList(partitionBy));
|
|
return this;
|
|
}
|
|
|
|
WindowFunction<A> withRowsOrRange(String s, List<Expression<?>> args) {
|
|
rowsOrRange = s;
|
|
rowsOrRangeArgs = args;
|
|
return this;
|
|
}
|
|
|
|
public WindowRows<A> rows() {
|
|
value = null;
|
|
int offset = orderBy.size() + partitionBy.size() + 1;
|
|
return new WindowRows<A>(this, " rows", offset);
|
|
}
|
|
|
|
public WindowRows<A> range() {
|
|
value = null;
|
|
int offset = orderBy.size() + partitionBy.size() + 1;
|
|
return new WindowRows<A>(this, " range", offset);
|
|
}
|
|
}
|