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

cn.nukkit.utils.HumanStringComparator Maven / Gradle / Ivy

package cn.nukkit.utils;

import cn.nukkit.api.PowerNukkitOnly;
import cn.nukkit.api.Since;
import lombok.NoArgsConstructor;

import org.jetbrains.annotations.NotNull;

import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;

@PowerNukkitOnly
@Since("1.4.0.0-PN")
@NoArgsConstructor(onConstructor = @__(@PowerNukkitOnly))
public class HumanStringComparator implements Comparator {
    private static final HumanStringComparator INSTANCE = new HumanStringComparator();
    private static final int LEFT = -1;
    private static final int RIGHT = 1;
    private static final int EQUALS = 0;
    private static final String SYMBOLS = "[:.;,/\\]{}|=";

    @Override
    public int compare(String o1, String o2) {
        if (o1.equals(o2)) {
            return EQUALS;
        }

        List l1 = splitSymbols(combineNegativeSign(split(o1)));
        List l2 = splitSymbols(combineNegativeSign(split(o2)));

        return compare(l1, l2);
    }

    private List splitSymbols(List list) {
        boolean changed = false;
        List result = list;
        int size = list.size();
        for (int i = size - 1; i >= 0; i--) {
            String str = result.get(i);
            int length = str.length();
            int lastPart = length;
            for (int j = length - 1; j >= 0; j--) {
                char c = str.charAt(j);
                if (SYMBOLS.indexOf(c) != -1) {
                    if (!changed) {
                        result = list instanceof ArrayList ? list : new ArrayList<>(list);
                        changed = true;
                    }
                    int indexToAddLast;
                    if (j > 0) {
                        result.set(i, str.substring(0, j));
                        result.add(i + 1, Character.toString(c));
                        indexToAddLast = i + 2;
                    } else {
                        result.set(i, Character.toString(c));
                        indexToAddLast = i + 1;
                    }
                    if (j + 2 <= length) {
                        result.add(indexToAddLast, str.substring(j + 1, lastPart));
                    }
                    lastPart = j;
                }
            }
        }
        return result;
    }

    private int compare(List l1, List l2) {
        int len1 = l1.size();
        int len2 = l2.size();
        int minLen = Math.min(len1, len2);
        for (int i = 0; i < minLen; i++) {
            String str1 = l1.get(i);
            String str2 = l2.get(i);
            int strLen1 = str1.length();
            int strLen2 = str2.length();
            assert strLen1 > 0;
            assert strLen2 > 0;
            boolean isNum1 = Character.isDigit(str1.charAt(strLen1 - 1));
            boolean isNum2 = Character.isDigit(str2.charAt(strLen2 - 1));
            if (isNum1) {
                if (isNum2) {
                    int i1 = Integer.parseInt(str1);
                    int i2 = Integer.parseInt(str2);
                    int result = Integer.compare(i1, i2);
                    if (result != EQUALS) {
                        return result;
                    }
                    // Number with higher 0 padding goes before
                    result = Integer.compare(strLen1, strLen2);
                    if (result != EQUALS) {
                        return result;
                    }
                } else {
                    return RIGHT;
                }
            } else if (isNum2) {
                return LEFT;
            } else {
                if (strLen1 == strLen2) {
                    int result = str1.compareTo(str2);
                    if (result != EQUALS) {
                        return result;
                    }
                } else {
                    int minStrLen = Math.min(strLen1, strLen2);
                    String commonPart1 = str1.substring(0, minStrLen);
                    String commonPart2 = str2.substring(0, minStrLen);
                    int result = commonPart1.compareTo(commonPart2);
                    if (result != EQUALS) {
                        return result;
                    }

                    // Detect omitted number
                    if (strLen1 < strLen2) {
                        if (detectOmittedNumber(l1, len1, i, str2, strLen2, minStrLen, commonPart1)) {
                            return RIGHT;
                        }
                    } else if (detectOmittedNumber(l2, len2, i, str1, strLen1, minStrLen, commonPart2)) {
                        return LEFT;
                    }

                    return Integer.compare(strLen1, strLen2);
                }
            }
        }

        return Integer.compare(len1, len2);
    }

    private boolean detectOmittedNumber(List l1, int len1, int i, String str2, int strLen2, int minStrLen, String commonPart1) {
        String combined;
        String comparingWith;
        String next1 = len1 > i + 1 ? l1.get(i + 1) : null;
        int nextLen1 = next1 == null ? 0 : next1.length();
        boolean isDigit1 = next1 != null && Character.isDigit(next1.charAt(nextLen1 - 1));
        String afterNext1 = isDigit1 && len1 > i + 2 ? l1.get(i + 2) : null;
        int afterNextLen1 = afterNext1 != null ? afterNext1.length() : 0;
        if (afterNextLen1 > 0) {
            combined = commonPart1 + afterNext1.substring(0, Math.min(afterNextLen1, strLen2 - minStrLen));
            comparingWith = str2;
            return combined.equals(comparingWith);
        }
        return false;
    }

    private List combineNegativeSign(List list) {
        int size = list.size();
        if (size < 2) {
            return list;
        }

        for (int i = size - 1; i > 0; i--) {
            String str1 = list.get(i);
            int strLen1 = str1.length();
            if (strLen1 > 0 && Character.isDigit(str1.charAt(strLen1 - 1))) {
                String str2 = list.get(i - 1);
                int strLen2 = str2.length();
                if (strLen2 > 0 && str2.charAt(strLen2 - 1) == '-') {
                    list.set(i, "-" + str1);
                    if (strLen2 == 1) {
                        list.remove(i - 1);
                        i -= 2;
                    } else {
                        list.set(i - 1, str2.substring(0, strLen2 - 1));
                        i--;
                    }
                }
            }
        }

        return list;
    }

    @NotNull
    private List split(String str) {
        int length = str.length();
        if (length == 0) {
            return Collections.emptyList();
        } else if (length == 1) {
            return Collections.singletonList(str);
        }

        List list = null;
        boolean wasDigit = false;
        int start = -1;
        for (int i = 0; i < length; i++) {
            if (Character.isDigit(str.charAt(i))) {
                if (!wasDigit && start == -1) {
                    start = i;
                    wasDigit = true;
                } else if (!wasDigit) {
                    if (list == null) {
                        list = new ArrayList<>(2);
                    }
                    list.add(str.substring(start, i));
                    start = i;
                    wasDigit = true;
                }
            } else {
                if (wasDigit) {
                    if (list == null) {
                        list = new ArrayList<>(2);
                    }
                    list.add(str.substring(start, i));
                    start = i;
                    wasDigit = false;
                } else if (start == -1) {
                    start = i;
                }
            }
        }

        String substring = str.substring(start, length);
        if (list == null) {
            list = Collections.singletonList(substring);
        } else {
            list.add(substring);
        }

        return list;
    }

    @PowerNukkitOnly
    @Since("1.4.0.0-PN")
    public static HumanStringComparator getInstance() {
        return INSTANCE;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy