/*
* Joinery -- Data frames for Java
* Copyright (c) 2014, 2015 IBM Corp.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
package joinery.impl;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Comparator;
import java.util.List;
import java.util.Map;
import joinery.DataFrame;
import joinery.DataFrame.SortDirection;
public class Sorting {
public static <V> DataFrame<V> sort(
final DataFrame<V> df, final Map<Integer, SortDirection> cols) {
final Comparator<List<V>> comparator = new Comparator<List<V>>() {
@Override
@SuppressWarnings("unchecked")
public int compare(final List<V> r1, final List<V> r2) {
int result = 0;
for (final Map.Entry<Integer, SortDirection> col : cols.entrySet()) {
final int c = col.getKey();
final Comparable<V> v1 = Comparable.class.cast(r1.get(c));
final V v2 = r2.get(c);
result = v1.compareTo(v2);
result *= col.getValue() == SortDirection.DESCENDING ? -1 : 1;
if (result != 0) {
break;
}
}
return result;
}
};
return sort(df, comparator);
}
public static <V> DataFrame<V> sort(
final DataFrame<V> df, final Comparator<List<V>> comparator) {
final DataFrame<V> sorted = new DataFrame<V>(df.columns());
final Comparator<Integer> cmp = new Comparator<Integer>() {
@Override
public int compare(final Integer r1, final Integer r2) {
return comparator.compare(df.row(r1), df.row(r2));
}
};
final Integer[] rows = new Integer[df.length()];
for (int r = 0; r < df.length(); r++) {
rows[r] = r;
}
Arrays.sort(rows, cmp);
final List<Object> labels = new ArrayList<>(df.index());
for (final Integer r : rows) {
final Object label = r < labels.size() ? labels.get(r) : r;
sorted.append(label, df.row(r));
}
return sorted;
}
}