/**
 * @className: IntervalUtil
 * @program: property-admin-old
 * @description: 范围比较器
 * @author: wangliangzhi
 * @create: 2022-01-13 20:08
 */
@Slf4j
public class IntervalUtil {


    /**
     * 判断data_value是否在interval区间范围内
     *
     * @param dataValue         数值类型的
     * @param minValue,maxValue 正常的数学区间,包括无穷大等,如:(1,3)、>5%、(-∞,6]、(125%,135%)U(70%,80%)
     * @return true:表示data_value在区间interval范围内,false:表示data_value不在区间interval范围内
     * @author: kangyl17909
     * @date: 2018年7月3日
     */

    public static boolean isRange(String dataValue, String minValue, String maxValue) {
        //将区间和data_value转化为可计算的表达式
        return isInTheInterval(dataValue, "[" + minValue + "," + maxValue + "]");
    }

    private static boolean isInTheInterval(String dataValue, String interval) {
        //将区间和data_value转化为可计算的表达式
        String formula = getFormulaByAllInterval(dataValue, interval);
        ScriptEngine jse = new ScriptEngineManager().getEngineByName("JavaScript");
        try {
            //计算表达式
            return (Boolean) jse.eval(formula);
        } catch (Exception t) {
            return false;
        }
    }

    /**
     * 将所有阀值区间转化为公式:如
     * [75,80) =》 dataValue < 80 && dataValue >= 75
     * (125%,135%)U(70%,80%) =》 (dataValue < 1.35 && dataValue > 1.25) || (dataValue < 0.8 && dataValue > 0.7)
     *
     * @param dataValue
     * @param interval  形式如:(125%,135%)U(70%,80%)
     */
    private static String getFormulaByAllInterval(String dataValue, String interval) {
        StringBuilder buff = new StringBuilder();
        //如:(125%,135%)U (70%,80%)
        for (String limit : interval.split("U")) {
            buff.append("(").append(getFormulaByInterval(dataValue, limit)).append(")").append("||");
        }
        String allLimitInvel = buff.toString();
        int index = allLimitInvel.lastIndexOf("||");
        allLimitInvel = allLimitInvel.substring(0, index);
        return allLimitInvel;
    }

    /**
     * 将整个阀值区间转化为公式:如
     * 145) =》 dataValue < 145
     * [75,80) =》 dataValue < 80 && dataValue >= 75
     *
     * @param dataValue
     * @param interval  形式如:145)、[75,80)
     */
    private static String getFormulaByInterval(String dataValue, String interval) {
        StringBuffer buff = new StringBuffer();
        for (String halfInterval : interval.split(",")) {//如:[75,80)、≥80
            buff.append(getFormulaByHalfInterval(halfInterval, dataValue)).append(" && ");
        }
        String limitInvel = buff.toString();
        int index = limitInvel.lastIndexOf(" && ");
        limitInvel = limitInvel.substring(0, index);
        return limitInvel;
    }

    /**
     * 将半个阀值区间转化为公式:如
     * 145) =》 dataValue < 145
     * ≥80% =》 dataValue >= 0.8
     * [130 =》 dataValue >= 130
     * <80% =》 dataValue < 0.8
     *
     * @param halfInterval 形式如:145)、≥80%、[130、<80%
     * @param dataValue
     * @return dataValue < 145
     */
    private static String getFormulaByHalfInterval(String halfInterval, String dataValue) {
        halfInterval = halfInterval.trim();
        //包含无穷大则不需要公式
        if (halfInterval.contains("∞")) {
            return "1 == 1";
        }
        StringBuffer formula = new StringBuffer();
        String data = "";
        String opera = "";
        //表示判断方向(如>)在前面 如:≥80%
        if (halfInterval.matches("^([<>≤≥\\[\\(]{1}(-?\\d+.?\\d*\\%?))$")) {
            opera = halfInterval.substring(0, 1);
            data = halfInterval.substring(1);
        } else {//[130、145)
            opera = halfInterval.substring(halfInterval.length() - 1);
            data = halfInterval.substring(0, halfInterval.length() - 1);
        }
        double value = dealPercent(data);
        formula.append(dataValue).append(" ").append(opera).append(" ").append(value);
        String a = formula.toString();
        //转化特定字符
        return a.replace("[", ">=").replace("(", ">").replace("]", "<=").replace(")", "<").replace("≤", "<=").replace("≥", ">=");
    }

    /**
     * 去除百分号,转为小数
     *
     * @param str 可能含百分号的数字
     * @return
     */
    private static double dealPercent(String str) {
        double d = 0.0;
        if (str.contains("%")) {
            str = str.substring(0, str.length() - 1);
            d = Double.parseDouble(str) / 100;
        } else {
            d = Double.parseDouble(str);
        }
        return d;
    }

    public static void main(String[] args) {
        System.out.println(isInTheInterval("2", "[1.5,2]"));
    }
}