31
31
import org .springframework .data .domain .PageImpl ;
32
32
import org .springframework .data .domain .Pageable ;
33
33
import org .springframework .data .domain .ScrollPosition ;
34
+ import org .springframework .data .domain .Slice ;
35
+ import org .springframework .data .domain .SliceImpl ;
34
36
import org .springframework .data .domain .Sort ;
35
37
import org .springframework .data .domain .Window ;
36
38
import org .springframework .data .jpa .domain .Specification ;
39
+ import org .springframework .data .jpa .repository .JpaSpecificationExecutor .SpecificationFluentQuery ;
37
40
import org .springframework .data .jpa .repository .query .ScrollDelegate ;
38
41
import org .springframework .data .jpa .support .PageableUtils ;
39
42
import org .springframework .data .projection .ProjectionFactory ;
40
43
import org .springframework .data .repository .query .FluentQuery ;
41
44
import org .springframework .data .support .PageableExecutionUtils ;
45
+ import org .springframework .lang .Nullable ;
42
46
import org .springframework .util .Assert ;
43
47
44
48
/**
52
56
* @since 3.0
53
57
*/
54
58
class FetchableFluentQueryBySpecification <S , R > extends FluentQuerySupport <S , R >
55
- implements FluentQuery .FetchableFluentQuery <R > {
59
+ implements FluentQuery .FetchableFluentQuery <R >, SpecificationFluentQuery < R > {
56
60
57
61
private final Specification <S > spec ;
58
62
private final Function <FluentQuerySupport <?, ?>, TypedQuery <S >> finder ;
@@ -85,7 +89,7 @@ private FetchableFluentQueryBySpecification(Specification<S> spec, Class<S> enti
85
89
}
86
90
87
91
@ Override
88
- public FetchableFluentQuery <R > sortBy (Sort sort ) {
92
+ public SpecificationFluentQuery <R > sortBy (Sort sort ) {
89
93
90
94
Assert .notNull (sort , "Sort must not be null" );
91
95
@@ -94,7 +98,7 @@ public FetchableFluentQuery<R> sortBy(Sort sort) {
94
98
}
95
99
96
100
@ Override
97
- public FetchableFluentQuery <R > limit (int limit ) {
101
+ public SpecificationFluentQuery <R > limit (int limit ) {
98
102
99
103
Assert .isTrue (limit >= 0 , "Limit must not be negative" );
100
104
@@ -103,7 +107,7 @@ public FetchableFluentQuery<R> limit(int limit) {
103
107
}
104
108
105
109
@ Override
106
- public <NR > FetchableFluentQuery <NR > as (Class <NR > resultType ) {
110
+ public <NR > SpecificationFluentQuery <NR > as (Class <NR > resultType ) {
107
111
108
112
Assert .notNull (resultType , "Projection target type must not be null" );
109
113
@@ -112,7 +116,7 @@ public <NR> FetchableFluentQuery<NR> as(Class<NR> resultType) {
112
116
}
113
117
114
118
@ Override
115
- public FetchableFluentQuery <R > project (Collection <String > properties ) {
119
+ public SpecificationFluentQuery <R > project (Collection <String > properties ) {
116
120
117
121
return new FetchableFluentQueryBySpecification <>(spec , entityType , resultType , sort , limit , properties , finder ,
118
122
scroll , countOperation , existsOperation , entityManager , projectionFactory );
@@ -155,9 +159,20 @@ public Window<R> scroll(ScrollPosition scrollPosition) {
155
159
return scroll .scroll (this , scrollPosition ).map (getConversionFunction ());
156
160
}
157
161
162
+ @ Override
163
+ public Slice <R > slice (Pageable pageable ) {
164
+ return pageable .isUnpaged () ? new PageImpl <>(all ()) : readSlice (pageable );
165
+ }
166
+
158
167
@ Override
159
168
public Page <R > page (Pageable pageable ) {
160
- return pageable .isUnpaged () ? new PageImpl <>(all ()) : readPage (pageable );
169
+ return pageable .isUnpaged () ? new PageImpl <>(all ()) : readPage (pageable , spec );
170
+ }
171
+
172
+ @ Override
173
+ @ SuppressWarnings ({ "rawtypes" , "unchecked" })
174
+ public Page <R > page (Pageable pageable , Specification <?> countSpec ) {
175
+ return pageable .isUnpaged () ? new PageImpl <>(all ()) : readPage (pageable , (Specification ) countSpec );
161
176
}
162
177
163
178
@ Override
@@ -193,7 +208,27 @@ private TypedQuery<S> createSortedAndProjectedQuery() {
193
208
return query ;
194
209
}
195
210
196
- private Page <R > readPage (Pageable pageable ) {
211
+ private Slice <R > readSlice (Pageable pageable ) {
212
+
213
+ TypedQuery <S > pagedQuery = createSortedAndProjectedQuery ();
214
+
215
+ if (pageable .isPaged ()) {
216
+ pagedQuery .setFirstResult (PageableUtils .getOffsetAsInteger (pageable ));
217
+ pagedQuery .setMaxResults (pageable .getPageSize () + 1 );
218
+ }
219
+
220
+ List <S > resultList = pagedQuery .getResultList ();
221
+ boolean hasNext = resultList .size () > pageable .getPageSize ();
222
+ if (hasNext ) {
223
+ resultList = resultList .subList (0 , pageable .getPageSize ());
224
+ }
225
+
226
+ List <R > slice = convert (resultList );
227
+
228
+ return new SliceImpl <>(slice , pageable , hasNext );
229
+ }
230
+
231
+ private Page <R > readPage (Pageable pageable , @ Nullable Specification <S > countSpec ) {
197
232
198
233
TypedQuery <S > pagedQuery = createSortedAndProjectedQuery ();
199
234
@@ -204,7 +239,7 @@ private Page<R> readPage(Pageable pageable) {
204
239
205
240
List <R > paginatedResults = convert (pagedQuery .getResultList ());
206
241
207
- return PageableExecutionUtils .getPage (paginatedResults , pageable , () -> countOperation .apply (spec ));
242
+ return PageableExecutionUtils .getPage (paginatedResults , pageable , () -> countOperation .apply (countSpec ));
208
243
}
209
244
210
245
private List <R > convert (List <S > resultList ) {
0 commit comments