diff --git a/README.md b/README.md index 83619fd5..a7e39e33 100644 --- a/README.md +++ b/README.md @@ -8,16 +8,16 @@ DataGear是一款数据可视化分析平台,使用Java语言开发,采用 ## 系统特点 - 可管理数据库驱动 -
管理员可通过驱动程序管理功能添加数据库驱动程序,无需重启,即可支持连接新数据库。 +
可通过驱动程序管理功能添加数据库驱动程序,无需重启,即可支持连接新数据库。 -- 参数化数据集 -
可编写动态SQL语句数据集,为其添加参数,构建可交互式图表。 +- 多种格式的数据集 +
支持SQL、CSV、Excel、HTTP接口、JSON等多种格式的数据集。 -- 多数据源聚合图表 -
一个图表可添加多个不同数据源的数据集,将它们聚合展示于同一图表。 +- 多数据集聚合图表 +
一个图表可添加多个不同格式的数据集,将它们聚合展示。 - 插件式图表类型 -
每一种类型的图表都以图表插件形式提供支持,并内置了大量图表插件,管理员也可上传自定义图表插件,丰富系统图表类型。 +
每一种类型的图表都以图表插件形式提供,并内置了大量图表插件,管理员也可上传自定义图表插件,丰富系统图表类型。 - 可自由编辑的HTML看板模板
看板使用原生的HTML网页作为模板,可自由编辑、绑定、异步加载图表,并支持将任意HTML网页导入为看板。 diff --git a/Roadmap.txt b/Roadmap.txt index a44506df..99c50817 100644 --- a/Roadmap.txt +++ b/Roadmap.txt @@ -1,6 +1,6 @@ 下一主版本: - 扩展数据集功能,支持Excel、HTTP API数据集; + 下一修订版本: @@ -8,6 +8,7 @@ 待定: + 数据分析添加项目概念,用于分组管理数据集、图表、看板; 组合数据集:新建组合数据集(CombineDataSet),选定多个其他数据集,通过SQL语句连接、选取它们,生成新的数据; 全局看板资源管理功能; 图表参数; diff --git a/datagear-analysis/pom.xml b/datagear-analysis/pom.xml index 4f23a66a..c5fc9bf5 100644 --- a/datagear-analysis/pom.xml +++ b/datagear-analysis/pom.xml @@ -7,7 +7,7 @@ org.datagear datagear - 1.11.1 + 1.12.0 datagear-analysis @@ -44,6 +44,31 @@ freemarker ${freemarker.version} + + org.apache.commons + commons-csv + ${commons-csv.version} + + + org.apache.poi + poi + ${poi.version} + + + org.apache.poi + poi-ooxml + ${poi-ooxml.version} + + + org.apache.httpcomponents.client5 + httpclient5 + 5.0.1 + + + com.jayway.jsonpath + json-path + 2.4.0 + diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/DataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/DataSet.java index b5f426ae..b6b6490a 100644 --- a/datagear-analysis/src/main/java/org/datagear/analysis/DataSet.java +++ b/datagear-analysis/src/main/java/org/datagear/analysis/DataSet.java @@ -72,6 +72,9 @@ public interface DataSet extends Identifiable /** * 获取{@linkplain DataSetResult}。 + *

+ * 返回结果中的数据项应已转换为与{@linkplain #getProperties()}的{@linkplain DataSetProperty#getType()}类型一致。 + *

* * @param paramValues * 由{@linkplain #getParams()}所描述的参数值映射表,其关键字是{@linkplain DataSetParam#getName()} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/DataSetProperty.java b/datagear-analysis/src/main/java/org/datagear/analysis/DataSetProperty.java index 9a417611..93a8ab86 100644 --- a/datagear-analysis/src/main/java/org/datagear/analysis/DataSetProperty.java +++ b/datagear-analysis/src/main/java/org/datagear/analysis/DataSetProperty.java @@ -11,14 +11,8 @@ package org.datagear.analysis; import java.io.Serializable; import java.math.BigDecimal; import java.math.BigInteger; -import java.sql.Time; -import java.sql.Timestamp; -import java.util.Date; -import java.util.List; import java.util.Map; -import org.datagear.util.StringUtil; - /** * 数据集属性信息。 *

@@ -67,53 +61,11 @@ public class DataSetProperty extends AbstractDataNameType implements Serializabl } /** - * 连接给定列表的{@linkplain #getLabel()}。 - *

- * 如果{@code dataSetProperties}为{@code null},将返回空字符串。 - *

+ * {@linkplain DataSetProperty#getType()}类型枚举。 * - * @param dataSetProperties - * @param splitter + * @author datagear@163.com + * */ - public static String concatLabels(List dataSetProperties, String splitter) - { - if (dataSetProperties == null) - return ""; - - StringBuilder sb = new StringBuilder(); - - for (int i = 0; i < dataSetProperties.size(); i++) - { - DataSetProperty dataSetProperty = dataSetProperties.get(i); - - String label = dataSetProperty.getLabel(); - if (!StringUtil.isEmpty(label)) - { - if (sb.length() > 0) - sb.append(splitter); - - sb.append(label); - } - } - - return sb.toString(); - } - - /** - * 拆分由{@linkplain #concatLabels(List, String)}连接的字符串。 - * - * @param labelText - * @param splitter - * @return - */ - public static String[] splitLabels(String labelText, String splitter) - { - if (labelText == null) - return new String[0]; - - return StringUtil.split(labelText, splitter, true); - } - public static class DataType { /** 字符串 */ @@ -143,270 +95,6 @@ public class DataSetProperty extends AbstractDataNameType implements Serializabl /** 未知类型 */ public static final String UNKNOWN = "UNKNOWN"; - /** - * 是否是{@linkplain #STRING}。 - * - * @param dataType - * @return - */ - public static boolean isString(String dataType) - { - return STRING.equals(dataType); - } - - /** - * 是否是{@linkplain #BOOLEAN}。 - * - * @param dataType - * @return - */ - public static boolean isBoolean(String dataType) - { - return BOOLEAN.equals(dataType); - } - - /** - * 是否是{@linkplain #NUMBER}。 - * - * @param dataType - * @return - */ - public static boolean isNumber(String dataType) - { - return NUMBER.equals(dataType); - } - - /** - * 是否是{@linkplain #INTEGER}。 - * - * @param dataType - * @return - */ - public static boolean isInteger(String dataType) - { - return INTEGER.equals(dataType); - } - - /** - * 是否是{@linkplain #DECIMAL}。 - * - * @param dataType - * @return - */ - public static boolean isDecimal(String dataType) - { - return DECIMAL.equals(dataType); - } - - /** - * 是否是数值型的。 - * - * @param dataType - * @return - */ - public static boolean isNumberic(String dataType) - { - return (isNumber(dataType) || isInteger(dataType) || isDecimal(dataType)); - } - - /** - * 是否是{@linkplain #DATE}。 - * - * @param dataType - * @return - */ - public static boolean isDate(String dataType) - { - return DATE.equals(dataType); - } - - /** - * 是否是{@linkplain #TIME}。 - * - * @param dataType - * @return - */ - public static boolean isTime(String dataType) - { - return TIME.equals(dataType); - } - - /** - * 是否是{@linkplain #TIMESTAMP}。 - * - * @param dataType - * @return - */ - public static boolean isTimestamp(String dataType) - { - return TIMESTAMP.equals(dataType); - } - - /** - * 是否是{@linkplain #UNKNOWN}。 - * - * @param dataType - * @return - */ - public static boolean isUnknown(String dataType) - { - return UNKNOWN.equals(dataType); - } - - /** - * 将{@linkplain #STRING}类型的值转换为{@linkplain String}值。 - *

- * {@code value}参数必须是{@linkplain String}类型。 - *

- * - * @param value - * @return - */ - public static String castString(Object value) - { - return (String) value; - } - - /** - * 将{@linkplain #BOOLEAN}类型的值转换为{@linkplain Boolean}值。 - *

- * {@code value}参数必须是{@linkplain Boolean}类型。 - *

- * - * @param value - * @return - */ - public static Boolean castBoolean(Object value) - { - return (Boolean) value; - } - - /** - * 将{@linkplain #INTEGER}类型的值转换为{@linkplain Number}值。 - *

- * {@code value}参数必须是{@linkplain Number}或其子类型。 - *

- * - * @param value - * @return - */ - public static Number castInteger(Object value) - { - return (Number) value; - } - - /** - * 将{@linkplain #INTEGER}类型的值转换为{@linkplain BigInteger}值。 - *

- * {@code value}参数必须是{@linkplain Number}或其子类型。 - *

- * - * @param value - * @return - */ - public static BigInteger castBigInteger(Object value) - { - Number number = castInteger(value); - - if (number == null) - return null; - else if (number instanceof BigInteger) - return (BigInteger) number; - else - return BigInteger.valueOf(number.longValue()); - } - - /** - * 将{@linkplain #DECIMAL}类型的值转换为{@linkplain Number}值。 - *

- * {@code value}参数必须是{@linkplain Number}或其子类型。 - *

- * - * @param value - * @return - */ - public static Number castDecimal(Object value) - { - return (Number) value; - } - - /** - * 将{@linkplain #DECIMAL}类型的值转换为{@linkplain BigDecimal}值。 - *

- * {@code value}参数必须是{@linkplain Number}或其子类型。 - *

- * - * @param value - * @return - */ - public static BigDecimal castBigDecimal(Object value) - { - Number number = castInteger(value); - - if (number == null) - return null; - else if (number instanceof BigDecimal) - return (BigDecimal) number; - else - return BigDecimal.valueOf(number.doubleValue()); - } - - /** - * 将{@linkplain #DATE}类型的值转换为{@linkplain Date}值。 - *

- * {@code value}参数必须是{@linkplain Date}或其子类型。 - *

- * - * @param value - * @return - */ - public static Date castDate(Object value) - { - return (Date) value; - } - - /** - * 将{@linkplain #TIME}类型的值转换为{@linkplain Time}值。 - *

- * {@code value}参数必须是{@linkplain Date}或其子类型。 - *

- * - * @param value - * @return - */ - public static Time castTime(Object value) - { - Date date = castDate(value); - - if (date == null) - return null; - else if (date instanceof Time) - return (Time) value; - else - return new Time(date.getTime()); - } - - /** - * 将{@linkplain #TIME}类型的值转换为{@linkplain Timestamp}值。 - *

- * {@code value}参数必须是{@linkplain Date}或其子类型。 - *

- * - * @param value - * @return - */ - public static Timestamp castTimestamp(Object value) - { - Date date = castDate(value); - - if (date == null) - return null; - else if (date instanceof Timestamp) - return (Timestamp) value; - else - return new Timestamp(date.getTime()); - } - /** * 解析对象的数据类型。 * diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractCsvDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractCsvDataSet.java new file mode 100644 index 00000000..5b29780d --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractCsvDataSet.java @@ -0,0 +1,387 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.analysis.support; + +import java.io.Reader; +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import org.apache.commons.csv.CSVFormat; +import org.apache.commons.csv.CSVParser; +import org.apache.commons.csv.CSVRecord; +import org.datagear.analysis.DataSetException; +import org.datagear.analysis.DataSetProperty; +import org.datagear.analysis.DataSetResult; +import org.datagear.analysis.ResolvableDataSet; +import org.datagear.analysis.ResolvedDataSetResult; +import org.datagear.util.IOUtil; + +/** + * 抽象CSV数据集。 + * + * @author datagear@163.com + * + */ +public abstract class AbstractCsvDataSet extends AbstractResolvableDataSet implements ResolvableDataSet +{ + /** 作为名称行的行号 */ + private int nameRow = -1; + + public AbstractCsvDataSet() + { + super(); + } + + public AbstractCsvDataSet(String id, String name) + { + super(id, name); + } + + public AbstractCsvDataSet(String id, String name, List properties) + { + super(id, name, properties); + } + + /** + * 是否有名称行。 + * + * @return + */ + public boolean hasNameRow() + { + return (this.nameRow > 0); + } + + /** + * 获取作为名称行的行号。 + * + * @return + */ + public int getNameRow() + { + return nameRow; + } + + /** + * 设置作为名称行的行号。 + * + * @param nameRow + * 行号,小于{@code 1}则表示无名称行。 + */ + public void setNameRow(int nameRow) + { + this.nameRow = nameRow; + } + + /** + * 解析结果。 + *

+ * 如果{@linkplain #getCsvReader(Map)}返回的{@linkplain TemplateResolvedSource#hasResolvedTemplate()}, + * 此方法将返回{@linkplain TemplateResolvedDataSetResult}。 + *

+ */ + @Override + protected ResolvedDataSetResult resolveResult(Map paramValues, List properties) + throws DataSetException + { + TemplateResolvedSource reader = null; + + try + { + reader = getCsvReader(paramValues); + + ResolvedDataSetResult result = resolveResult(reader.getSource(), properties); + + if (reader.hasResolvedTemplate()) + result = new TemplateResolvedDataSetResult(result.getResult(), result.getProperties(), + reader.getResolvedTemplate()); + + return result; + } + catch (DataSetException e) + { + throw e; + } + catch (Throwable t) + { + throw new DataSetSourceParseException(t); + } + finally + { + if (reader != null) + IOUtil.close(reader.getSource()); + } + } + + /** + * 获取CSV输入流。 + *

+ * 实现方法应该返回实例级不变的输入流。 + *

+ * + * @param paramValues + * @return + * @throws Throwable + */ + protected abstract TemplateResolvedSource getCsvReader(Map paramValues) throws Throwable; + + @SuppressWarnings({ "unchecked", "rawtypes" }) + protected ResolvedDataSetResult resolveResult(Reader csvReader, List properties) throws Throwable + { + boolean resolveProperties = (properties == null || properties.isEmpty()); + + CSVParser csvParser = buildCSVParser(csvReader); + + List propertyNames = null; + List> data = new ArrayList<>(); + + DataSetPropertyValueConverter converter = createDataSetPropertyValueConverter(); + + int rowIdx = 0; + int dataRowIdx = 0; + + for (CSVRecord csvRecord : csvParser) + { + if (isNameRow(rowIdx)) + { + if (resolveProperties) + propertyNames = resolveDataSetPropertyNames(csvRecord, false); + } + else + { + if (resolveProperties && dataRowIdx == 0 && propertyNames == null) + propertyNames = resolveDataSetPropertyNames(csvRecord, true); + + if (resolveProperties) + data.add(resolveCSVRecordValues(csvRecord, null, converter)); + else + data.add(resolveCSVRecordValues(csvRecord, properties, converter)); + + dataRowIdx++; + } + + rowIdx++; + } + + if (resolveProperties) + return resolveResult(propertyNames, ((List) data)); + else + { + DataSetResult result = new DataSetResult(listRowsToMapRows(data, properties)); + return new ResolvedDataSetResult(result, properties); + } + } + + /** + * 由原始的字符串CSV数据解析{@linkplain ResolvedDataSetResult}。 + * + * @param propertyNames + * 允许为{@code null} + * @param data + * 允许为{@code null} + * @return + * @throws Throwable + */ + protected ResolvedDataSetResult resolveResult(List propertyNames, List> data) throws Throwable + { + List properties = new ArrayList<>((propertyNames == null ? 0 : propertyNames.size())); + + if (propertyNames != null) + { + for (String name : propertyNames) + properties.add(new DataSetProperty(name, DataSetProperty.DataType.STRING)); + } + + @SuppressWarnings("unchecked") + List> resultData = Collections.EMPTY_LIST; + + // 根据数据格式,修订可能的数值类型:只有某一列的所有字符串都是数值格式,才认为是数值类型 + if (data != null) + { + int plen = properties.size(); + + // 指定索引的字符串是否都是数值内容 + Map asNumberMap = new HashMap<>(); + + for (List ele : data) + { + for (int i = 0; i < Math.min(plen, ele.size()); i++) + { + Boolean asNumber = asNumberMap.get(i); + + if (Boolean.FALSE.equals(asNumber)) + continue; + + String val = ele.get(i); + asNumberMap.put(i, isNumberString(val)); + } + } + + for (Map.Entry entry : asNumberMap.entrySet()) + { + if (Boolean.TRUE.equals(entry.getValue())) + properties.get(entry.getKey()).setType(DataSetProperty.DataType.NUMBER); + } + + List> newData = new ArrayList<>(data.size()); + + for (List ele : data) + { + int size = Math.min(plen, ele.size()); + + List newEle = new ArrayList<>(size); + + for (int i = 0; i < size; i++) + { + String val = ele.get(i); + + if (Boolean.TRUE.equals(asNumberMap.get(i))) + { + Number num = parseNumberString(val); + newEle.add(num); + } + else + newEle.add(val); + } + + newData.add(newEle); + } + + resultData = listRowsToMapRows(newData, properties); + } + + DataSetResult result = new DataSetResult(resultData); + + return new ResolvedDataSetResult(result, properties); + } + + /** + * 指定的CSV值是否可被当做数值类型。 + * + * @param value + * @return + */ + protected boolean isNumberString(String value) + { + if (value == null || value.isEmpty()) + return false; + + try + { + parseNumberString(value); + + return true; + } + catch (Throwable t) + { + return false; + } + } + + /** + * 解析数值字符串,{@linkplain #isNumberString(String)}应为{@code true}。 + * + * @param s + * @return + * @throws Throwable + */ + protected Number parseNumberString(String s) throws Throwable + { + return Double.parseDouble(s); + } + + /** + * + * @param csvRecord + * @param forceIndex + * 是否强制使用索引号,将返回:{@code ["1"、"2"、....]} + * @return + * @throws Throwable + */ + protected List resolveDataSetPropertyNames(CSVRecord csvRecord, boolean forceIndex) throws Throwable + { + int size = csvRecord.size(); + List list = new ArrayList<>(size); + + for (int i = 0; i < size; i++) + { + if (forceIndex) + list.add(Integer.toString(i + 1)); + else + list.add(csvRecord.get(i)); + } + + return list; + } + + /** + * 解析数据列表。 + *

+ * 如果{@code properties}为{@code null}或者对应元素为{@code null},则返回列表对应元素将是{@linkplain String}类型。 + *

+ * + * @param csvRecord + * @param properties + * 允许为{@code null}、元素为{@code null} + * @param converter + * @return + * @throws Throwable + */ + protected List resolveCSVRecordValues(CSVRecord csvRecord, List properties, + DataSetPropertyValueConverter converter) throws Throwable + { + int size = csvRecord.size(); + List list = new ArrayList<>(size); + + int propertySize = (properties == null ? 0 : properties.size()); + + for (int i = 0; i < size; i++) + { + DataSetProperty property = (i < propertySize ? properties.get(i) : null); + + String rawValue = csvRecord.get(i); + + if (property == null) + list.add(rawValue); + else + { + Object value = convertToPropertyDataType(converter, rawValue, property); + list.add(value); + } + } + + return list; + } + + /** + * 是否名称行 + * + * @param rowIndex + * 行索引(以{@code 0}计数) + * @return + */ + protected boolean isNameRow(int rowIndex) + { + return ((rowIndex + 1) == this.nameRow); + } + + /** + * 构建{@linkplain CSVParser}。 + * + * @param reader + * @return + * @throws Throwable + */ + protected CSVParser buildCSVParser(Reader reader) throws Throwable + { + return CSVFormat.DEFAULT.withIgnoreSurroundingSpaces().parse(reader); + } +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractCsvFileDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractCsvFileDataSet.java new file mode 100644 index 00000000..060ab139 --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractCsvFileDataSet.java @@ -0,0 +1,75 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.analysis.support; + +import java.io.File; +import java.io.Reader; +import java.util.List; +import java.util.Map; + +import org.datagear.analysis.DataSetProperty; +import org.datagear.util.IOUtil; + +/** + * 抽象CSV文件数据集。 + *

+ * 注意:此类不支持Freemarker模板语言。 + *

+ * + * @author datagear@163.com + * + */ +public abstract class AbstractCsvFileDataSet extends AbstractCsvDataSet +{ + /** 文件编码 */ + private String encoding = IOUtil.CHARSET_UTF_8; + + public AbstractCsvFileDataSet() + { + super(); + } + + public AbstractCsvFileDataSet(String id, String name) + { + super(id, name); + } + + public AbstractCsvFileDataSet(String id, String name, List properties) + { + super(id, name, properties); + } + + public String getEncoding() + { + return encoding; + } + + public void setEncoding(String encoding) + { + this.encoding = encoding; + } + + @Override + protected TemplateResolvedSource getCsvReader(Map paramValues) throws Throwable + { + File file = getCsvFile(paramValues); + return new TemplateResolvedSource<>(IOUtil.getReader(file, this.encoding)); + } + + /** + * 获取CSV文件。 + *

+ * 实现方法应该返回实例级不变的文件。 + *

+ * + * @param paramValues + * @return + * @throws Throwable + */ + protected abstract File getCsvFile(Map paramValues) throws Throwable; +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractDataSet.java index a29e6836..a2687ce5 100644 --- a/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractDataSet.java +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractDataSet.java @@ -7,7 +7,9 @@ */ package org.datagear.analysis.support; +import java.util.ArrayList; import java.util.Collections; +import java.util.HashMap; import java.util.List; import java.util.Map; @@ -25,6 +27,9 @@ import org.datagear.analysis.DataSetProperty; */ public abstract class AbstractDataSet extends AbstractIdentifiable implements DataSet { + /** 默认Freemarker模板解析器 */ + public static final DataSetFmkTemplateResolver FMK_TEMPLATE_RESOLVER = new DataSetFmkTemplateResolver(); + private String name; private List properties; @@ -32,9 +37,13 @@ public abstract class AbstractDataSet extends AbstractIdentifiable implements Da @SuppressWarnings("unchecked") private List params = Collections.EMPTY_LIST; + /** 属性数据转换格式 */ + private DataFormat propertyDataFormat = null; + public AbstractDataSet() { super(); + setPropertyDataFormat(new DataFormat()); } public AbstractDataSet(String id, String name, List properties) @@ -42,6 +51,7 @@ public abstract class AbstractDataSet extends AbstractIdentifiable implements Da super(id); this.name = name; this.properties = properties; + setPropertyDataFormat(new DataFormat()); } @Override @@ -94,6 +104,24 @@ public abstract class AbstractDataSet extends AbstractIdentifiable implements Da return getDataNameTypeByName(this.params, name); } + public DataFormat getPropertyDataFormat() + { + return propertyDataFormat; + } + + /** + * 设置属性数据转换格式。 + *

+ * 当{@linkplain DataSetProperty#getType()}不是结果数据的原始类型,而需要进行类型转换时,需要使用数据转换格式进行转换。 + *

+ * + * @param propertyDataFormat + */ + public void setPropertyDataFormat(DataFormat propertyDataFormat) + { + this.propertyDataFormat = propertyDataFormat; + } + @Override public boolean isReady(Map paramValues) { @@ -111,6 +139,76 @@ public abstract class AbstractDataSet extends AbstractIdentifiable implements Da return true; } + /** + * 将源对象转换为指定{@linkplain DataSetProperty.DataType}类型的对象。 + *

+ * 如果{@code property}为{@code null},则什么也不做直接返回。 + *

+ * + * @param converter + * @param source + * @param property + * 允许为{@code null} + * @return + */ + protected Object convertToPropertyDataType(DataSetPropertyValueConverter converter, Object source, + DataSetProperty property) + { + if (property == null) + return source; + + return convertToPropertyDataType(converter, source, property.getType()); + } + + /** + * 将源对象转换为指定{@linkplain DataSetProperty.DataType}类型的对象。 + *

+ * 如果{@code propertyType}为{@code null},则什么也不做直接返回。 + *

+ * + * @param converter + * @param source + * @param propertyType + * 允许为{@code null} + * @return + */ + protected Object convertToPropertyDataType(DataSetPropertyValueConverter converter, Object source, + String propertyType) + { + if (propertyType == null || DataSetProperty.DataType.UNKNOWN.equals(propertyType)) + return source; + + return converter.convert(source, propertyType); + } + + /** + * 创建一个{@linkplain DataSetPropertyValueConverter}实例。 + *

+ * 由于{@linkplain DataSetPropertyValueConverter}不是线程安全的,所以每次使用时要手动创建。 + *

+ * + * @return + */ + protected DataSetPropertyValueConverter createDataSetPropertyValueConverter() + { + DataFormat dataFormat = this.propertyDataFormat; + if (dataFormat == null) + dataFormat = new DataFormat(); + + return new DataSetPropertyValueConverter(dataFormat); + } + + /** + * 解析{@linkplain DataSetProperty.DataType}类型。 + * + * @param value + * @return + */ + protected String resolvePropertyDataType(Object value) + { + return DataSetProperty.DataType.resolveDataType(value); + } + /** * 获取指定名称的{@linkplain DataNameType}对象,没找到则返回{@code null}。 * @@ -133,4 +231,60 @@ public abstract class AbstractDataSet extends AbstractIdentifiable implements Da return null; } + + @SuppressWarnings("unchecked") + protected List> listRowsToMapRows(List> data, List properties) + { + if (data == null) + return Collections.EMPTY_LIST; + + int plen = properties.size(); + + List> maps = new ArrayList<>(data.size()); + + for (List row : data) + { + Map map = new HashMap<>(); + + for (int i = 0; i < Math.min(plen, row.size()); i++) + { + String name = properties.get(i).getName(); + map.put(name, row.get(i)); + } + + maps.add(map); + } + + return maps; + } + + /** + * 只有当{@linkplain #hasParam()}为{@code true}时才将指定文本作为Freemarker模板解析。 + * + * @param text + * @param paramValues + * @return + */ + protected String resolveAsFmkTemplateIfHasParam(String text, Map paramValues) + { + if (!hasParam()) + return text; + + return resolveAsFmkTemplate(text, paramValues); + } + + /** + * 将指定文本作为Freemarker模板解析。 + * + * @param text + * @param paramValues + * @return + */ + protected String resolveAsFmkTemplate(String text, Map paramValues) + { + if (text == null) + return null; + + return FMK_TEMPLATE_RESOLVER.resolve(text, paramValues); + } } diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractExcelDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractExcelDataSet.java new file mode 100644 index 00000000..7fe6ec53 --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractExcelDataSet.java @@ -0,0 +1,684 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.analysis.support; + +import java.io.File; +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; +import java.util.Map; + +import org.apache.poi.hssf.usermodel.HSSFWorkbook; +import org.apache.poi.hssf.util.CellReference; +import org.apache.poi.openxml4j.opc.OPCPackage; +import org.apache.poi.openxml4j.opc.PackageAccess; +import org.apache.poi.poifs.filesystem.POIFSFileSystem; +import org.apache.poi.ss.usermodel.Cell; +import org.apache.poi.ss.usermodel.CellType; +import org.apache.poi.ss.usermodel.DateUtil; +import org.apache.poi.ss.usermodel.Row; +import org.apache.poi.ss.usermodel.Sheet; +import org.apache.poi.xssf.usermodel.XSSFWorkbook; +import org.datagear.analysis.DataSetException; +import org.datagear.analysis.DataSetProperty; +import org.datagear.analysis.DataSetResult; +import org.datagear.analysis.ResolvableDataSet; +import org.datagear.analysis.ResolvedDataSetResult; +import org.datagear.analysis.support.RangeExpResolver.IndexRange; +import org.datagear.analysis.support.RangeExpResolver.Range; +import org.datagear.util.FileUtil; +import org.datagear.util.IOUtil; +import org.datagear.util.StringUtil; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * 抽象Excel数据集。 + *

+ * 此类仅支持从Excel的单个sheet读取数据,具体参考{@linkplain #setSheetIndex(int)}。 + *

+ *

+ * 通过{@linkplain #setDataRowExp(String)}、{@linkplain #setDataColumnExp(String)}来设置读取行、列范围。 + *

+ *

+ * 通过{@linkplain #setNameRow(int)}可设置名称行。 + *

+ *

+ * 注意:此类不支持Freemarker模板语言。 + *

+ * + * @author datagear@163.com + * + */ +public abstract class AbstractExcelDataSet extends AbstractResolvableDataSet implements ResolvableDataSet +{ + protected static final Logger LOGGER = LoggerFactory.getLogger(AbstractExcelDataSet.class); + + public static final String EXTENSION_XLSX = "xlsx"; + + public static final String EXTENSION_XLS = "xls"; + + protected static final RangeExpResolver RANGE_EXP_RESOLVER = RangeExpResolver + .valueOf(RangeExpResolver.RANGE_SPLITTER_CHAR, RangeExpResolver.RANGE_GROUP_SPLITTER_CHAR); + + /** 此数据集所处的sheet索引号(以1计数) */ + private int sheetIndex = 1; + + /** 作为名称行的行号 */ + private int nameRow = -1; + + /** 数据行范围表达式 */ + private String dataRowExp = ""; + + /** 数据列范围表达式 */ + private String dataColumnExp = ""; + + /** 是否强制作为xls文件处理 */ + private boolean forceXls = false; + + private transient List _dataRowRanges = null; + private transient List _dataColumnRanges = null; + + public AbstractExcelDataSet() + { + super(); + } + + public AbstractExcelDataSet(String id, String name) + { + super(id, name); + } + + public AbstractExcelDataSet(String id, String name, List properties) + { + super(id, name, properties); + } + + public int getSheetIndex() + { + return sheetIndex; + } + + /** + * 设置此数据集所处的sheet号。 + * + * @param sheetIndex + * sheet号(以{@code 1}计数) + */ + public void setSheetIndex(int sheetIndex) + { + this.sheetIndex = sheetIndex; + } + + /** + * 是否有名称行。 + * + * @return + */ + public boolean hasNameRow() + { + return (this.nameRow > 0); + } + + /** + * 获取作为名称行的行号。 + * + * @return + */ + public int getNameRow() + { + return nameRow; + } + + /** + * 设置作为名称行的行号。 + * + * @param nameRow + * 行号,小于{@code 1}则表示无名称行。 + */ + public void setNameRow(int nameRow) + { + this.nameRow = nameRow; + } + + public String getDataRowExp() + { + return dataRowExp; + } + + /** + * 设置数据行范围表达式。 + *

+ * 表达式格式示例为: + *

+ *

+ * {@code "6"} :第6行
+ * {@code "3-15"} :第3至15行
+ * {@code "1,4,8-15"}:第1、4、8至15行 + *

+ *

+ * 标题行({@linkplain #getNameRow()})将自动被排除。 + *

+ *

+ * 注意:行号以{@code 1}开始计数。 + *

+ * + * @param dataRowExp + * 表达式,为{@code null}、{@code ""}则不限定 + */ + public void setDataRowExp(String dataRowExp) + { + this.dataRowExp = dataRowExp; + this._dataRowRanges = getRangeExpResolver().resolveIndex(this.dataRowExp); + } + + public String getDataColumnExp() + { + return dataColumnExp; + } + + /** + * 设置数据列范围表达式。 + *

+ * 表达式格式为: + *

+ *

+ * {@code "A"}:第A列
+ * {@code "C-E"}:第C至E列
+ * {@code "A,C,E-H"}:第A、C、E至H列 + *

+ * + * @param dataColumnExp + * 表达式,为{@code null}、{@code ""}则不限定 + */ + public void setDataColumnExp(String dataColumnExp) + { + this.dataColumnExp = dataColumnExp; + this._dataColumnRanges = resolveDataColumnRanges(dataColumnExp); + } + + /** + * 是否强制作为xls文件处理。 + * + * @return + */ + public boolean isForceXls() + { + return forceXls; + } + + /** + * 设置是否强制作为xls文件处理,如果为{@code false},则根据文件扩展名判断。 + * + * @param forceXls + */ + public void setForceXls(boolean forceXls) + { + this.forceXls = forceXls; + } + + @Override + protected ResolvedDataSetResult resolveResult(Map paramValues, List properties) + throws DataSetException + { + File file = getExcelFile(paramValues); + + ResolvedDataSetResult result = null; + + if (isXls(file)) + result = resolveResultForXls(paramValues, file, properties); + else + result = resolveResultForXlsx(paramValues, file, properties); + + return result; + } + + /** + * 解析{@code xls}结果。 + * + * @param paramValues + * @param file + * @param properties + * 允许为{@code null},此时会自动解析 + * @return + * @throws DataSetException + */ + protected ResolvedDataSetResult resolveResultForXls(Map paramValues, File file, + List properties) throws DataSetException + { + POIFSFileSystem poifs = null; + HSSFWorkbook wb = null; + + try + { + poifs = new POIFSFileSystem(file, true); + wb = new HSSFWorkbook(poifs.getRoot(), true); + + Sheet sheet = wb.getSheetAt(getSheetIndex() - 1); + + return resolveResultForSheet(paramValues, sheet, properties); + } + catch (DataSetException e) + { + throw e; + } + catch (Throwable t) + { + throw new DataSetSourceParseException(t); + } + finally + { + IOUtil.close(wb); + IOUtil.close(poifs); + } + } + + /** + * 解析{@code xlsx}结果。 + * + * @param paramValues + * @param file + * @param properties + * 允许为{@code null},此时会自动解析 + * @return + * @throws DataSetException + */ + protected ResolvedDataSetResult resolveResultForXlsx(Map paramValues, File file, + List properties) throws DataSetException + { + OPCPackage pkg = null; + XSSFWorkbook wb = null; + + try + { + pkg = OPCPackage.open(file, PackageAccess.READ); + wb = new XSSFWorkbook(pkg); + + Sheet sheet = wb.getSheetAt(getSheetIndex() - 1); + + return resolveResultForSheet(paramValues, sheet, properties); + } + catch (DataSetException e) + { + throw e; + } + catch (Throwable t) + { + throw new DataSetSourceParseException(t); + } + finally + { + IOUtil.close(wb); + IOUtil.close(pkg); + } + } + + /** + * 解析sheet结果。 + * + * @param paramValues + * @param sheet + * @param properties + * 允许为{@code null},此时会自动解析 + * @return + * @throws DataSetException + */ + protected ResolvedDataSetResult resolveResultForSheet(Map paramValues, Sheet sheet, + List properties) throws DataSetException + { + boolean resolveProperties = (properties == null || properties.isEmpty()); + + if (resolveProperties) + properties = new ArrayList<>(); + + List> data = new ArrayList<>(); + + List propertyNames = null; + + DataSetPropertyValueConverter converter = createDataSetPropertyValueConverter(); + + try + { + int rowIdx = 0; + int dataRowIdx = 0; + + for (Row row : sheet) + { + if (isNameRow(rowIdx)) + { + if (resolveProperties) + propertyNames = resolveDataSetPropertyNames(row, false); + } + else if (isDataRow(rowIdx)) + { + if (resolveProperties && dataRowIdx == 0 && propertyNames == null) + propertyNames = resolveDataSetPropertyNames(row, true); + + // 名称行不一定在数据行之前,此时可能还无法确定属性名,所以暂时采用列表存储 + List rowObj = new ArrayList<>(); + + int colIdx = 0; + int dataColIdx = 0; + + for (Cell cell : row) + { + if (isDataColumn(colIdx)) + { + DataSetProperty property = null; + + if (!resolveProperties) + { + if (dataColIdx >= properties.size()) + throw new DataSetSourceParseException( + "No property defined for column index " + dataColIdx); + + property = properties.get(dataColIdx); + } + + Object value = resolvePropertyValue(cell, property, converter); + + if (resolveProperties) + { + property = resolveDataSetProperty(row, rowIdx, dataRowIdx, cell, colIdx, dataColIdx, + value, properties); + } + + rowObj.add(value); + + dataColIdx++; + } + + colIdx++; + } + + data.add(rowObj); + + dataRowIdx++; + } + + rowIdx++; + } + } + catch (DataSetException e) + { + throw e; + } + catch (Throwable t) + { + throw new DataSetSourceParseException(t); + } + + if (resolveProperties) + inflateDataSetProperties(properties, propertyNames); + + DataSetResult result = new DataSetResult(listRowsToMapRows(data, properties)); + + return new ResolvedDataSetResult(result, properties); + } + + protected void inflateDataSetProperties(List properties, List propertyNames) + { + for (int i = 0; i < properties.size(); i++) + { + DataSetProperty property = properties.get(i); + property.setName(propertyNames.get(i)); + + if (StringUtil.isEmpty(property.getType())) + property.setType(DataSetProperty.DataType.UNKNOWN); + } + } + + /** + * 解析{@linkplain DataSetProperty}并写入{@code properties}。 + * + * @param row + * @param rowIdx + * @param dataRowIdx + * @param cell + * @param colIdx + * @param dataColIdx + * @param cellValue + * @param properties + * @return + */ + protected DataSetProperty resolveDataSetProperty(Row row, int rowIdx, int dataRowIdx, Cell cell, int colIdx, + int dataColIdx, Object cellValue, List properties) + { + DataSetProperty property = null; + + if (dataRowIdx == 0) + { + // 空单元格先不处理数据类型,等待后续有非空单元格再判断 + String dataType = (cellValue == null ? "" : resolvePropertyDataType(cellValue)); + + // 名称行不一定在数据行之前,所以此时可能无法确定属性名 + property = new DataSetProperty("should-be-set-later", dataType); + properties.add(property); + } + else + { + property = properties.get(dataColIdx); + + if (StringUtil.isEmpty(property.getType()) && cellValue != null) + property.setType(resolvePropertyDataType(cellValue)); + } + + return property; + } + + /** + * 解析属性名。 + * + * @param nameRow + * @return + */ + protected List resolveDataSetPropertyNames(Row nameRow, boolean forceColumnString) + { + List propertyNames = new ArrayList<>(); + + int colIdx = 0; + for (Cell cell : nameRow) + { + if (isDataColumn(colIdx)) + { + String name = null; + + if (forceColumnString) + name = CellReference.convertNumToColString(colIdx); + else + { + try + { + name = cell.getStringCellValue(); + } + catch (Throwable t) + { + } + + if (StringUtil.isEmpty(name)) + name = CellReference.convertNumToColString(colIdx); + } + + propertyNames.add(name); + } + + colIdx++; + } + + return propertyNames; + } + + /** + * 解析单元格属性值。 + * + * @param cell + * @param property + * 允许为{@code null} + * @param converter + * @return + * @throws DataSetSourceParseException + * @throws DataSetException + */ + protected Object resolvePropertyValue(Cell cell, DataSetProperty property, DataSetPropertyValueConverter converter) + throws DataSetSourceParseException, DataSetException + { + CellType cellType = cell.getCellTypeEnum(); + + Object cellValue = null; + + try + { + if (CellType.BLANK.equals(cellType)) + { + cellValue = null; + } + else if (CellType.BOOLEAN.equals(cellType)) + { + cellValue = cell.getBooleanCellValue(); + } + else if (CellType.ERROR.equals(cellType)) + { + cellValue = cell.getErrorCellValue(); + } + else if (CellType.FORMULA.equals(cellType)) + { + cellValue = cell.getCellFormula(); + } + else if (CellType.NUMERIC.equals(cellType)) + { + if (DateUtil.isCellDateFormatted(cell)) + cellValue = cell.getDateCellValue(); + else + cellValue = cell.getNumericCellValue(); + } + else if (CellType.STRING.equals(cellType)) + { + cellValue = cell.getStringCellValue(); + } + } + catch (DataSetException e) + { + throw e; + } + catch (Throwable t) + { + throw new DataSetSourceParseException(t); + } + + cellValue = convertToPropertyDataType(converter, cellValue, property); + + return cellValue; + } + + /** + * 是否名称行 + * + * @param rowIndex + * 行索引(以{@code 0}计数) + * @return + */ + protected boolean isNameRow(int rowIndex) + { + return ((rowIndex + 1) == this.nameRow); + } + + /** + * 是否数据行。 + * + * @param rowIndex + * 行索引(以{@code 0}计数) + * @return + */ + protected boolean isDataRow(int rowIndex) + { + if (isNameRow(rowIndex)) + return false; + + if (this._dataRowRanges == null || this._dataRowRanges.isEmpty()) + return true; + + return IndexRange.includes(this._dataRowRanges, rowIndex + 1); + } + + /** + * 是否数据列。 + * + * @param columnIndex + * 列索引(以{@code 0}计数) + * @return + */ + protected boolean isDataColumn(int columnIndex) + { + if (this._dataColumnRanges == null || this._dataColumnRanges.isEmpty()) + return true; + + return IndexRange.includes(this._dataColumnRanges, columnIndex); + } + + @SuppressWarnings("unchecked") + protected List resolveDataColumnRanges(String dataColumnExp) throws DataSetException + { + List ranges = getRangeExpResolver().resolve(dataColumnExp); + + if (ranges == null || ranges.isEmpty()) + return Collections.EMPTY_LIST; + + List indexRanges = new ArrayList<>(ranges.size()); + + for (Range range : ranges) + { + int from = 0; + int to = -1; + + String fromStr = range.trimFrom(); + String toStr = range.trimTo(); + + if (!StringUtil.isEmpty(fromStr)) + from = CellReference.convertColStringToIndex(fromStr); + + if (!StringUtil.isEmpty(toStr)) + to = CellReference.convertColStringToIndex(toStr); + + indexRanges.add(new IndexRange(from, to)); + } + + return indexRanges; + } + + /** + * 给定Excel文件是否是老版本的{@code .xls}文件。 + * + * @param file + * @return + */ + protected boolean isXls(File file) + { + if (this.forceXls) + return true; + + return FileUtil.isExtension(file, EXTENSION_XLS); + } + + protected RangeExpResolver getRangeExpResolver() + { + return RANGE_EXP_RESOLVER; + } + + /** + * 获取Excel文件。 + *

+ * 实现方法应该返回实例级不变的文件。 + *

+ * + * @param paramValues + * @return + * @throws DataSetException + */ + protected abstract File getExcelFile(Map paramValues) throws DataSetException; + +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractFmkTemplateDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractFmkTemplateDataSet.java deleted file mode 100644 index 39ac895e..00000000 --- a/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractFmkTemplateDataSet.java +++ /dev/null @@ -1,43 +0,0 @@ -/* - * Copyright (c) 2018 datagear.tech. All Rights Reserved. - */ - -/** - * - */ -package org.datagear.analysis.support; - -import java.util.List; -import java.util.Map; - -import org.datagear.analysis.DataSet; -import org.datagear.analysis.DataSetProperty; - -/** - * 抽象Freemarker模板{@linkplain DataSet}。 - * - * @author datagear@163.com - * - */ -public abstract class AbstractFmkTemplateDataSet extends AbstractDataSet -{ - public static final DataSetFmkTemplateResolver TEMPLATE_RESOLVER = new DataSetFmkTemplateResolver(); - - public AbstractFmkTemplateDataSet() - { - super(); - } - - public AbstractFmkTemplateDataSet(String id, String name, List properties) - { - super(id, name, properties); - } - - protected String resolveTemplate(String template, Map paramValues) - { - if (template == null) - return null; - - return TEMPLATE_RESOLVER.resolve(template, paramValues); - } -} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractJsonDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractJsonDataSet.java index e0d6ea77..2ce77f2a 100644 --- a/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractJsonDataSet.java +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractJsonDataSet.java @@ -7,7 +7,10 @@ */ package org.datagear.analysis.support; +import java.io.Reader; +import java.util.ArrayList; import java.util.Collections; +import java.util.HashMap; import java.util.List; import java.util.Map; @@ -16,6 +19,18 @@ import org.datagear.analysis.DataSetProperty; import org.datagear.analysis.DataSetResult; import org.datagear.analysis.ResolvableDataSet; import org.datagear.analysis.ResolvedDataSetResult; +import org.datagear.util.IOUtil; +import org.datagear.util.StringUtil; + +import com.fasterxml.jackson.databind.JsonNode; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.fasterxml.jackson.databind.node.ArrayNode; +import com.fasterxml.jackson.databind.node.ObjectNode; +import com.fasterxml.jackson.databind.node.ValueNode; +import com.jayway.jsonpath.Configuration; +import com.jayway.jsonpath.JsonPath; +import com.jayway.jsonpath.spi.json.JacksonJsonProvider; +import com.jayway.jsonpath.spi.mapper.JacksonMappingProvider; /** * 抽象JSON数据集。 @@ -23,19 +38,23 @@ import org.datagear.analysis.ResolvedDataSetResult; * @author datagear@163.com * */ -public abstract class AbstractJsonDataSet extends AbstractFmkTemplateDataSet implements ResolvableDataSet +public abstract class AbstractJsonDataSet extends AbstractResolvableDataSet implements ResolvableDataSet { - public static final JsonDataSetSupport JSON_DATA_SET_SUPPORT = new JsonDataSetSupport(); + /** 使用Jackson的{@code JSONPath}配置 */ + protected static final Configuration JACKSON_JSON_PATH_CONFIGURATION = Configuration.builder() + .jsonProvider(new JacksonJsonProvider()).mappingProvider(new JacksonMappingProvider()).build(); + + /** 数据JSON路径 */ + private String dataJsonPath = ""; public AbstractJsonDataSet() { super(); } - @SuppressWarnings("unchecked") public AbstractJsonDataSet(String id, String name) { - super(id, name, Collections.EMPTY_LIST); + super(id, name); } public AbstractJsonDataSet(String id, String name, List properties) @@ -43,17 +62,350 @@ public abstract class AbstractJsonDataSet extends AbstractFmkTemplateDataSet imp super(id, name, properties); } - protected JsonDataSetSupport getJsonDataSetSupport() + public String getDataJsonPath() { - return JSON_DATA_SET_SUPPORT; + return dataJsonPath; } - @Override - public ResolvedDataSetResult resolve(Map paramValues) throws DataSetException + /** + * 设置数据JSON路径。 + *

+ * 当希望返回的是原始JSON数据的指定JSON路径值时,可以设置此项。 + *

+ *

+ * 例如:"stores[0].books"、"[1].stores"、"$['store']['book'][0]"、 + * "$.store.book[*].author"、"$..book[2]",具体参考{@code JSONPath}相关文档。 + *

+ *

+ * 默认无数据路径,将直接返回原始JSON数据。 + *

+ * + * @param dataJsonPath + */ + public void setDataJsonPath(String dataJsonPath) { - DataSetResult result = getResult(paramValues); - List properties = getJsonDataSetSupport().resolveDataSetProperties(result.getData()); + this.dataJsonPath = dataJsonPath; + } + + /** + * 解析结果。 + *

+ * 如果{@linkplain #getJsonReader(Map)}返回的{@linkplain TemplateResolvedSource#hasResolvedTemplate()}, + * 此方法将返回{@linkplain TemplateResolvedDataSetResult}。 + *

+ */ + @Override + protected ResolvedDataSetResult resolveResult(Map paramValues, List properties) + throws DataSetException + { + TemplateResolvedSource reader = null; + try + { + reader = getJsonReader(paramValues); + + ResolvedDataSetResult result = resolveResult(reader.getSource(), properties); + + if (reader.hasResolvedTemplate()) + result = new TemplateResolvedDataSetResult(result.getResult(), result.getProperties(), + reader.getResolvedTemplate()); + + return result; + } + catch (DataSetException e) + { + throw e; + } + catch (Throwable t) + { + throw new DataSetSourceParseException(t); + } + finally + { + if (reader != null) + IOUtil.close(reader.getSource()); + } + } + + /** + * 获取JSON输入流。 + *

+ * 实现方法应该返回实例级不变的输入流。 + *

+ * + * @param paramValues + * @return + * @throws Throwable + */ + protected abstract TemplateResolvedSource getJsonReader(Map paramValues) throws Throwable; + + /** + * 解析结果。 + * + * @param jsonReader + * JSON输入流 + * @param properties + * 允许为{@code null},此时会自动解析 + * @return + * @throws Throwable + */ + protected ResolvedDataSetResult resolveResult(Reader jsonReader, List properties) throws Throwable + { + boolean resolveProperties = (properties == null || properties.isEmpty()); + + JsonNode jsonNode = getObjectMapperNonStardand().readTree(jsonReader); + + if (!isLegalResultDataJsonNode(jsonNode)) + throw new UnsupportedJsonResultDataException("Result data must be JSON object or array"); + + Object data = null; + + if (jsonNode != null) + data = readDataByJsonPath(jsonNode, getDataJsonPath()); + + if (resolveProperties) + properties = resolveDataSetProperties(data); + + if (!resolveProperties) + data = convertJsonResultData(data, properties, createDataSetPropertyValueConverter()); + + DataSetResult result = new DataSetResult(data); return new ResolvedDataSetResult(result, properties); } + + /** + * 读取指定JSON路径的数据。 + * + * @param jsonNode + * 允许为{@code null} + * @param dataJsonPath + * 允许为{@code null} + * @return + * @throws ReadJsonDataPathException + * @throws Throwable + */ + protected Object readDataByJsonPath(JsonNode jsonNode, String dataJsonPath) + throws ReadJsonDataPathException, Throwable + { + if (jsonNode == null) + return null; + + Object data = getObjectMapperNonStardand().treeToValue(jsonNode, Object.class); + + if (data == null) + return null; + + if (StringUtil.isEmpty(dataJsonPath)) + return data; + + String stdDataJsonPath = dataJsonPath.trim(); + + if (StringUtil.isEmpty(stdDataJsonPath)) + return data; + + // 转换"stores[0].books"、"[1].stores"简化模式为规范的JSONPath + if (!stdDataJsonPath.startsWith("$")) + { + if (stdDataJsonPath.startsWith("[")) + stdDataJsonPath = "$" + stdDataJsonPath; + else + stdDataJsonPath = "$." + stdDataJsonPath; + } + + try + { + return JsonPath.compile(stdDataJsonPath).read(data, JACKSON_JSON_PATH_CONFIGURATION); + } + catch (Throwable t) + { + throw new ReadJsonDataPathException(dataJsonPath, t); + } + } + + /** + * + * @param resultData + * 允许为{@code null} + * @param properties + * @param converter + * @return + * @throws Throwable + */ + protected Object convertJsonResultData(Object resultData, List properties, + DataSetPropertyValueConverter converter) throws Throwable + { + Object re = null; + + // JSON对象 + if (resultData == null) + { + re = null; + } + else if (resultData instanceof Map) + { + Map reMap = new HashMap<>(); + + @SuppressWarnings("unchecked") + Map source = (Map) resultData; + + for (Map.Entry entry : source.entrySet()) + { + String name = entry.getKey(); + Object value = entry.getValue(); + + DataSetProperty property = getDataNameTypeByName(properties, name); + + value = convertToPropertyDataType(converter, value, property); + + reMap.put(name, value); + } + + re = reMap; + } + else if (resultData instanceof List) + { + List list = (List) resultData; + + List reList = new ArrayList<>(list.size()); + + for (Object ele : list) + reList.add(convertJsonResultData(ele, properties, converter)); + + re = reList; + } + else if (resultData instanceof Object[]) + { + Object[] array = (Object[]) resultData; + + Object[] reArray = new Object[array.length]; + + for (int i = 0; i < array.length; i++) + reArray[i] = convertJsonResultData(array[i], properties, converter); + + re = reArray; + } + else + throw new UnsupportedJsonResultDataException("Result data must be object or object array/list"); + + return re; + } + + /** + * 是否是合法的数据集结果数据{@linkplain JsonNode}。 + *

+ * 参考{@linkplain DataSetResult#getData()}说明。 + *

+ * + * @param jsonNode + * 允许为{@code null} + * @return + */ + protected boolean isLegalResultDataJsonNode(JsonNode jsonNode) throws Throwable + { + if (jsonNode == null || jsonNode.isNull()) + return true; + + if (jsonNode instanceof ValueNode) + return false; + + if (jsonNode instanceof ArrayNode) + { + ArrayNode arrayNode = (ArrayNode) jsonNode; + + for (int i = 0; i < arrayNode.size(); i++) + { + JsonNode eleNode = arrayNode.get(i); + + if (eleNode == null || eleNode.isNull()) + continue; + + if (!(eleNode instanceof ObjectNode)) + return false; + } + } + + return true; + } + + /** + * 解析JSON对象的{@linkplain DataSetProperty}。 + * + * @param resultData + * 允许为{@code null},JSON对象、JSON对象数组、JSON对象列表 + * @return + * @throws Throwable + */ + @SuppressWarnings("unchecked") + protected List resolveDataSetProperties(Object resultData) throws Throwable + { + if (resultData == null) + { + return Collections.EMPTY_LIST; + } + else if (resultData instanceof Map) + { + return resolveJsonObjDataSetProperties((Map) resultData); + } + else if (resultData instanceof List) + { + List list = (List) resultData; + + if (list.size() == 0) + return Collections.EMPTY_LIST; + else + return resolveJsonObjDataSetProperties((Map) list.get(0)); + } + else if (resultData instanceof Object[]) + { + Object[] array = (Object[]) resultData; + + if (array.length == 0) + return Collections.EMPTY_LIST; + else + return resolveJsonObjDataSetProperties((Map) array[0]); + } + else + throw new UnsupportedJsonResultDataException("Result data must be object or object array/list"); + } + + /** + * 解析JSON对象的{@linkplain DataSetProperty}。 + * + * @param jsonObj + * @return + * @throws Throwable + */ + protected List resolveJsonObjDataSetProperties(Map jsonObj) throws Throwable + { + List properties = new ArrayList<>(); + + if (jsonObj == null) + { + + } + else + { + for (Map.Entry entry : jsonObj.entrySet()) + { + Object value = entry.getValue(); + String type = DataSetProperty.DataType.resolveDataType(value); + + DataSetProperty property = new DataSetProperty(entry.getKey(), type); + + // JSON数值只有NUMBER类型 + if (DataSetProperty.DataType.INTEGER.equals(property.getType()) + || DataSetProperty.DataType.DECIMAL.equals(property.getType())) + property.setType(DataSetProperty.DataType.NUMBER); + + properties.add(property); + } + } + + return properties; + } + + protected ObjectMapper getObjectMapperNonStardand() + { + return JsonSupport.getObjectMapperNonStardand(); + } } diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractJsonFileDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractJsonFileDataSet.java index 4740ef24..cf849c21 100644 --- a/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractJsonFileDataSet.java +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractJsonFileDataSet.java @@ -8,25 +8,27 @@ package org.datagear.analysis.support; import java.io.File; +import java.io.Reader; import java.util.List; import java.util.Map; import org.datagear.analysis.DataSetException; import org.datagear.analysis.DataSetProperty; -import org.datagear.analysis.DataSetResult; +import org.datagear.util.IOUtil; /** * 抽象JSON文件数据集。 + *

+ * 注意:此类不支持Freemarker模板语言。 + *

* * @author datagear@163.com * */ -public abstract class AbstractJsonFileDataSet extends AbstractJsonSourceDataSet +public abstract class AbstractJsonFileDataSet extends AbstractJsonDataSet { - public static final String DEFAULT_ENCODING = "UTF-8"; - /** 文件编码 */ - private String encoding = DEFAULT_ENCODING; + private String encoding = IOUtil.CHARSET_UTF_8; public AbstractJsonFileDataSet() { @@ -54,15 +56,17 @@ public abstract class AbstractJsonFileDataSet extends AbstractJsonSourceDataSet } @Override - protected DataSetResult getOrginalResult(Map paramValues) throws DataSetException + protected TemplateResolvedSource getJsonReader(Map paramValues) throws Throwable { - File jsonFile = getJsonFile(paramValues); - Object data = getJsonDataSetSupport().resolveResultData(jsonFile, getEncoding()); - return new DataSetResult(data); + File file = getJsonFile(paramValues); + return new TemplateResolvedSource<>(IOUtil.getReader(file, this.encoding)); } /** * 获取JSON文件。 + *

+ * 实现方法应该返回实例级不变的文件。 + *

* * @param paramValues * @return diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractJsonSourceDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractJsonSourceDataSet.java deleted file mode 100644 index 0cfc4198..00000000 --- a/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractJsonSourceDataSet.java +++ /dev/null @@ -1,79 +0,0 @@ -/* - * Copyright (c) 2018 datagear.tech. All Rights Reserved. - */ - -/** - * - */ -package org.datagear.analysis.support; - -import java.util.List; -import java.util.Map; - -import org.datagear.analysis.DataSetException; -import org.datagear.analysis.DataSetProperty; -import org.datagear.analysis.DataSetResult; - -/** - * 抽象JSON源数据集。 - *

- * JSON源数据集的一个特点是:源数据是无法编辑的,因此需要定义{@linkplain DataSetResultTransformer}逻辑。 - *

- * - * @author datagear@163.com - * - */ -public abstract class AbstractJsonSourceDataSet extends AbstractJsonDataSet -{ - private DataSetResultTransformer dataSetResultTransformer; - - public AbstractJsonSourceDataSet() - { - super(); - } - - public AbstractJsonSourceDataSet(String id, String name) - { - super(id, name); - } - - public AbstractJsonSourceDataSet(String id, String name, List properties) - { - super(id, name, properties); - } - - /** - * 获取{@linkplain DataSetResultTransformer}。 - * - * @return 可能为{@code null} - */ - public DataSetResultTransformer getDataSetResultTransformer() - { - return dataSetResultTransformer; - } - - public void setDataSetResultTransformer(DataSetResultTransformer dataSetResultTransformer) - { - this.dataSetResultTransformer = dataSetResultTransformer; - } - - @Override - public DataSetResult getResult(Map paramValues) throws DataSetException - { - DataSetResult result = getOrginalResult(paramValues); - - if (this.dataSetResultTransformer == null) - return result; - - return this.dataSetResultTransformer.transform(result); - } - - /** - * 获取原始的{@linkplain DataSetResult}。 - * - * @param paramValues - * @return - * @throws DataSetException - */ - protected abstract DataSetResult getOrginalResult(Map paramValues) throws DataSetException; -} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractResolvableDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractResolvableDataSet.java new file mode 100644 index 00000000..c1a6ec54 --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/AbstractResolvableDataSet.java @@ -0,0 +1,76 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.analysis.support; + +import java.util.Collections; +import java.util.List; +import java.util.Map; + +import org.datagear.analysis.DataSetException; +import org.datagear.analysis.DataSetProperty; +import org.datagear.analysis.DataSetResult; +import org.datagear.analysis.ResolvableDataSet; +import org.datagear.analysis.ResolvedDataSetResult; + +/** + * 抽象{@linkplain ResolvableDataSet}。 + * + * @author datagear@163.com + * + */ +public abstract class AbstractResolvableDataSet extends AbstractDataSet implements ResolvableDataSet +{ + public AbstractResolvableDataSet() + { + super(); + } + + @SuppressWarnings("unchecked") + public AbstractResolvableDataSet(String id, String name) + { + super(id, name, Collections.EMPTY_LIST); + } + + public AbstractResolvableDataSet(String id, String name, List properties) + { + super(id, name, properties); + } + + @Override + public DataSetResult getResult(Map paramValues) throws DataSetException + { + List properties = getProperties(); + + if (properties == null || properties.isEmpty()) + throw new DataSetException("[getProperties()] must not be empty"); + + ResolvedDataSetResult result = resolveResult(paramValues, properties); + + return result.getResult(); + } + + @Override + public ResolvedDataSetResult resolve(Map paramValues) throws DataSetException + { + return resolveResult(paramValues, null); + } + + /** + * 解析结果。 + * + * @param paramValues + * @param properties + * 允许为{@code null}/空,此时,应自动解析并设置返回结果的{@linkplain ResolvedDataSetResult#setProperties(List)}; + * 如果不为{@code null}/空,直接将{@code properties}作为解析数据依据, 使用它处理结果数据, + * 并设置为返回结果的{@linkplain ResolvedDataSetResult#setProperties(List)} + * @return + * @throws DataSetException + */ + protected abstract ResolvedDataSetResult resolveResult(Map paramValues, List properties) + throws DataSetException; +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/CsvDirectoryFileDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/CsvDirectoryFileDataSet.java new file mode 100644 index 00000000..351c5b0d --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/CsvDirectoryFileDataSet.java @@ -0,0 +1,81 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.analysis.support; + +import java.io.File; +import java.util.List; +import java.util.Map; + +import org.datagear.analysis.DataSet; +import org.datagear.analysis.DataSetProperty; +import org.datagear.util.FileUtil; + +/** + * 目录内CSV文件{@linkplain DataSet}。 + *

+ * 注意:此类不支持Freemarker模板语言。 + *

+ * + * @author datagear@163.com + * + */ +public class CsvDirectoryFileDataSet extends AbstractCsvFileDataSet +{ + /** JSON文件所在的目录 */ + private File directory; + + /** JSON文件名 */ + private String fileName; + + public CsvDirectoryFileDataSet() + { + super(); + } + + public CsvDirectoryFileDataSet(String id, String name, File directory, String fileName) + { + super(id, name); + this.directory = directory; + this.fileName = fileName; + } + + public CsvDirectoryFileDataSet(String id, String name, List properties, File directory, + String fileName) + { + super(id, name, properties); + this.directory = directory; + this.fileName = fileName; + } + + public File getDirectory() + { + return directory; + } + + public void setDirectory(File directory) + { + this.directory = directory; + } + + public String getFileName() + { + return fileName; + } + + public void setFileName(String fileName) + { + this.fileName = fileName; + } + + @Override + protected File getCsvFile(Map paramValues) throws Throwable + { + File jsonFile = FileUtil.getFile(this.directory, this.fileName); + return jsonFile; + } +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/CsvValueDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/CsvValueDataSet.java new file mode 100644 index 00000000..3f8cf9b0 --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/CsvValueDataSet.java @@ -0,0 +1,79 @@ +/* + * Copyright 2018 datagear.tech. All Rights Reserved. + */ + +package org.datagear.analysis.support; + +import java.io.Reader; +import java.util.List; +import java.util.Map; + +import org.datagear.analysis.DataSetException; +import org.datagear.analysis.DataSetProperty; +import org.datagear.util.IOUtil; + +/** + * CSV值数据集。 + *

+ * 此类的{@linkplain #getValue()}支持Freemarker模板语言。 + *

+ * + * @author datagear@163.com + * + */ +public class CsvValueDataSet extends AbstractCsvDataSet +{ + /** CSV字符串 */ + private String value; + + public CsvValueDataSet() + { + super(); + } + + public CsvValueDataSet(String id, String name, String value) + { + super(id, name); + this.value = value; + } + + public CsvValueDataSet(String id, String name, List properties, String value) + { + super(id, name, properties); + this.value = value; + } + + public String getValue() + { + return value; + } + + /** + * 设置CSV字符串值,格式为: + * + *
+	 * name, value
+	 * aaa, 1
+	 * bbb, 2
+	 * 
+ * + * @param value + */ + public void setValue(String value) + { + this.value = value; + } + + @Override + public TemplateResolvedDataSetResult resolve(Map paramValues) throws DataSetException + { + return (TemplateResolvedDataSetResult) resolveResult(paramValues, null); + } + + @Override + protected TemplateResolvedSource getCsvReader(Map paramValues) throws Throwable + { + String csv = resolveAsFmkTemplateIfHasParam(this.value, paramValues); + return new TemplateResolvedSource<>(IOUtil.getReader(csv), csv); + } +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/DataFormat.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/DataFormat.java new file mode 100644 index 00000000..a24b5dc6 --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/DataFormat.java @@ -0,0 +1,144 @@ +/* + * Copyright 2018 datagear.tech. All Rights Reserved. + */ + +package org.datagear.analysis.support; + +import java.io.Serializable; + +/** + * 数据格式。 + * + * @author datagear@163.com + * + */ +public class DataFormat implements Serializable +{ + private static final long serialVersionUID = 1L; + + public static final String DEFAULT_DATE_FORMAT = "yyyy-MM-dd"; + + public static final String DEFAULT_TIME_FORMAT = "hh:mm:ss"; + + public static final String DEFAULT_TIMESTAMP_FORMAT = "yyyy-MM-dd hh:mm:ss"; + + public static final String DEFAULT_NUMBER_FORMAT = "#.##########"; + + /** 日期格式 */ + private String dateFormat = DEFAULT_DATE_FORMAT; + + /** 时间格式 */ + private String timeFormat = DEFAULT_TIME_FORMAT; + + /** 时间戳格式 */ + private String timestampFormat = DEFAULT_TIMESTAMP_FORMAT; + + /** 数值格式 */ + private String numberFormat = DEFAULT_NUMBER_FORMAT; + + public DataFormat() + { + super(); + } + + public String getDateFormat() + { + return dateFormat; + } + + public void setDateFormat(String dateFormat) + { + this.dateFormat = dateFormat; + } + + public String getTimeFormat() + { + return timeFormat; + } + + public void setTimeFormat(String timeFormat) + { + this.timeFormat = timeFormat; + } + + public String getTimestampFormat() + { + return timestampFormat; + } + + public void setTimestampFormat(String timestampFormat) + { + this.timestampFormat = timestampFormat; + } + + public String getNumberFormat() + { + return numberFormat; + } + + public void setNumberFormat(String numberFormat) + { + this.numberFormat = numberFormat; + } + + @Override + public int hashCode() + { + final int prime = 31; + int result = 1; + result = prime * result + ((dateFormat == null) ? 0 : dateFormat.hashCode()); + result = prime * result + ((numberFormat == null) ? 0 : numberFormat.hashCode()); + result = prime * result + ((timeFormat == null) ? 0 : timeFormat.hashCode()); + result = prime * result + ((timestampFormat == null) ? 0 : timestampFormat.hashCode()); + return result; + } + + @Override + public boolean equals(Object obj) + { + if (this == obj) + return true; + if (obj == null) + return false; + if (getClass() != obj.getClass()) + return false; + DataFormat other = (DataFormat) obj; + if (dateFormat == null) + { + if (other.dateFormat != null) + return false; + } + else if (!dateFormat.equals(other.dateFormat)) + return false; + if (numberFormat == null) + { + if (other.numberFormat != null) + return false; + } + else if (!numberFormat.equals(other.numberFormat)) + return false; + if (timeFormat == null) + { + if (other.timeFormat != null) + return false; + } + else if (!timeFormat.equals(other.timeFormat)) + return false; + if (timestampFormat == null) + { + if (other.timestampFormat != null) + return false; + } + else if (!timestampFormat.equals(other.timestampFormat)) + return false; + return true; + } + + @Override + public String toString() + { + return getClass().getSimpleName() + " [dateFormat=" + dateFormat + ", timeFormat=" + timeFormat + + ", timestampFormat=" + + timestampFormat + ", numberFormat=" + numberFormat + "]"; + } +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/DataSetPropertyValueConverter.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/DataSetPropertyValueConverter.java new file mode 100644 index 00000000..a750d498 --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/DataSetPropertyValueConverter.java @@ -0,0 +1,255 @@ +/* + * Copyright 2018 datagear.tech. All Rights Reserved. + */ + +package org.datagear.analysis.support; + +import java.sql.Date; +import java.sql.Time; +import java.sql.Timestamp; +import java.text.DecimalFormat; +import java.text.SimpleDateFormat; + +import org.datagear.analysis.DataSetProperty; +import org.datagear.analysis.DataSetProperty.DataType; + +/** + * {@linkplain DataSetProperty}值转换器。 + *

+ * 它支持将对象转换为{@linkplain DataSetProperty.DataType}类型的值。 + *

+ *

+ * 此类的{@linkplain #convert(java.util.Map, java.util.Collection)}、{@linkplain #convert(Object, String)}不是线程安全的。 + *

+ * + * @author datagear@163.com + * + */ +public class DataSetPropertyValueConverter extends DataValueConverter +{ + private DataFormat dataFormat; + + private SimpleDateFormat _dateFormat = null; + private SimpleDateFormat _timeFormat = null; + private SimpleDateFormat _timestampFormat = null; + private DecimalFormat _numberFormat = null; + + public DataSetPropertyValueConverter() + { + super(); + setDataFormat(new DataFormat()); + } + + public DataSetPropertyValueConverter(DataFormat dataFormat) + { + super(); + setDataFormat(new DataFormat()); + } + + public DataFormat getDataFormat() + { + return dataFormat; + } + + public void setDataFormat(DataFormat dataFormat) + { + this.dataFormat = dataFormat; + + this._dateFormat = new SimpleDateFormat(dataFormat.getDateFormat()); + this._timeFormat = new SimpleDateFormat(dataFormat.getTimeFormat()); + this._timestampFormat = new SimpleDateFormat(dataFormat.getTimestampFormat()); + this._numberFormat = new DecimalFormat(dataFormat.getNumberFormat()); + } + + @Override + protected Object convertValue(Object value, String type) throws DataValueConvertionException + { + if (value == null) + return null; + + if (type == null) + return value; + + try + { + if (value instanceof String) + return convertStringValue((String) value, type); + else if (value instanceof Boolean) + return convertBooleanValue((Boolean) value, type); + else if (value instanceof Number) + return convertNumberValue((Number) value, type); + else if (value instanceof Time) + return convertTimeValue((Time) value, type); + else if (value instanceof Timestamp) + return convertTimestampValue((Timestamp) value, type); + else if (value instanceof java.util.Date) + return convertDateValue((java.util.Date) value, type); + else + { + if (DataType.UNKNOWN.equals(type)) + return value; + else + throw new DataValueConvertionException(value, type); + } + } + catch (DataValueConvertionException e) + { + throw e; + } + catch (Throwable t) + { + throw new DataValueConvertionException(value, type); + } + } + + protected Object convertStringValue(String value, String type) throws Throwable + { + if (DataType.STRING.equals(type) || DataType.UNKNOWN.equals(type)) + return value; + + if (value == null || value.isEmpty()) + return null; + + if (DataType.BOOLEAN.equals(type)) + return "true".equalsIgnoreCase(value) || "1".equals(value); + else if (DataType.NUMBER.equals(type)) + return this._numberFormat.parse(value); + else if (DataType.INTEGER.equals(type)) + return this._numberFormat.parse(value).intValue(); + else if (DataType.DECIMAL.equals(type)) + return this._numberFormat.parse(value).doubleValue(); + else if (DataType.DATE.equals(type)) + { + java.util.Date date = this._dateFormat.parse(value); + return new Date(date.getTime()); + } + else if (DataType.TIME.equals(type)) + { + java.util.Date date = this._timeFormat.parse(value); + return new Time(date.getTime()); + } + else if (DataType.TIMESTAMP.equals(type)) + { + java.util.Date date = this._timestampFormat.parse(value); + return new Timestamp(date.getTime()); + } + else + throw new DataValueConvertionException(value, type); + } + + protected Object convertBooleanValue(Boolean value, String type) throws Throwable + { + if (DataType.BOOLEAN.equals(type) || DataType.UNKNOWN.equals(type)) + return value; + + if (value == null) + return null; + + if (DataType.STRING.equals(type)) + return value.toString(); + else if (DataType.NUMBER.equals(type) || DataType.INTEGER.equals(type) || DataType.DECIMAL.equals(type)) + return (Boolean.TRUE.equals(value) ? 1 : 0); + else + throw new DataValueConvertionException(value, type); + } + + protected Object convertNumberValue(Number value, String type) throws Throwable + { + if (DataType.NUMBER.equals(type) || DataType.UNKNOWN.equals(type)) + return value; + + if (value == null) + return null; + + if (DataType.STRING.equals(type)) + return this._numberFormat.format(value); + else if (DataType.BOOLEAN.equals(type)) + return (value.intValue() > 0); + else if (DataType.INTEGER.equals(type)) + return value.longValue(); + else if (DataType.DECIMAL.equals(type)) + return value.doubleValue(); + else if (DataType.DATE.equals(type)) + return new Date(value.longValue()); + else if (DataType.TIME.equals(type)) + return new Time(value.longValue()); + else if (DataType.TIMESTAMP.equals(type)) + return new Timestamp(value.longValue()); + else + throw new DataValueConvertionException(value, type); + } + + protected Object convertDateValue(java.util.Date value, String type) throws Throwable + { + if (DataType.UNKNOWN.equals(type)) + return value; + + if (value == null) + return null; + + if (DataType.STRING.equals(type)) + return this._dateFormat.format(value); + else if (DataType.NUMBER.equals(type)) + return value.getTime(); + else if (DataType.INTEGER.equals(type)) + return value.getTime(); + else if (DataType.DECIMAL.equals(type)) + return value.getTime(); + else if (DataType.DATE.equals(type)) + return new Date(value.getTime()); + else if (DataType.TIME.equals(type)) + return new Time(value.getTime()); + else if (DataType.TIMESTAMP.equals(type)) + return new Timestamp(value.getTime()); + else + throw new DataValueConvertionException(value, type); + } + + protected Object convertTimeValue(Time value, String type) throws Throwable + { + if (DataType.TIME.equals(type) || DataType.UNKNOWN.equals(type)) + return value; + + if (value == null) + return null; + + if (DataType.STRING.equals(type)) + return this._timeFormat.format(value); + else if (DataType.NUMBER.equals(type)) + return value.getTime(); + else if (DataType.INTEGER.equals(type)) + return value.getTime(); + else if (DataType.DECIMAL.equals(type)) + return value.getTime(); + else if (DataType.DATE.equals(type)) + return new Date(value.getTime()); + else if (DataType.TIMESTAMP.equals(type)) + return new Timestamp(value.getTime()); + else + throw new DataValueConvertionException(value, type); + } + + protected Object convertTimestampValue(Timestamp value, String type) throws Throwable + { + if (DataType.TIMESTAMP.equals(type) || DataType.UNKNOWN.equals(type)) + return value; + + if (value == null) + return null; + + if (DataType.STRING.equals(type)) + return this._timestampFormat.format(value); + else if (DataType.NUMBER.equals(type)) + return value.getTime(); + else if (DataType.INTEGER.equals(type)) + return value.getTime(); + else if (DataType.DECIMAL.equals(type)) + return value.getTime(); + else if (DataType.DATE.equals(type)) + return new Date(value.getTime()); + else if (DataType.TIME.equals(type)) + return new Time(value.getTime()); + else + throw new DataValueConvertionException(value, type); + } +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/DataSetResultTransformException.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/DataSetResultTransformException.java deleted file mode 100644 index b10570a6..00000000 --- a/datagear-analysis/src/main/java/org/datagear/analysis/support/DataSetResultTransformException.java +++ /dev/null @@ -1,41 +0,0 @@ -/* - * Copyright (c) 2018 datagear.tech. All Rights Reserved. - */ - -/** - * - */ -package org.datagear.analysis.support; - -import org.datagear.analysis.DataSetException; - -/** - * {@linkplain DataSetResultTransformer}转换异常。 - * - * @author datagear@163.com - * - */ -public class DataSetResultTransformException extends DataSetException -{ - private static final long serialVersionUID = 1L; - - public DataSetResultTransformException() - { - super(); - } - - public DataSetResultTransformException(String message) - { - super(message); - } - - public DataSetResultTransformException(Throwable cause) - { - super(cause); - } - - public DataSetResultTransformException(String message, Throwable cause) - { - super(message, cause); - } -} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/DataSetResultTransformer.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/DataSetResultTransformer.java deleted file mode 100644 index ad7124b8..00000000 --- a/datagear-analysis/src/main/java/org/datagear/analysis/support/DataSetResultTransformer.java +++ /dev/null @@ -1,33 +0,0 @@ -/* - * Copyright (c) 2018 datagear.tech. All Rights Reserved. - */ - -/** - * - */ -package org.datagear.analysis.support; - -import org.datagear.analysis.DataSet; -import org.datagear.analysis.DataSetResult; - -/** - * {@linkplain DataSetResult}转换器。 - *

- * 某些类型的{@linkplain DataSet}是从不可控制的数据源中读取数据的(比如API调用、JSON文件、CSV文件), - * 此类即为这些场景提供支持,使{@linkplain DataSet}支持对数据源的数据进行转换。 - *

- * - * @author datagear@163.com - * - */ -public interface DataSetResultTransformer -{ - /** - * 转换为新的{@linkplain DataSetResult}。 - * - * @param orginalResult - * @return 已转换的{@linkplain DataSetResult} - * @throws DataSetResultTransformException - */ - DataSetResult transform(DataSetResult orginalResult) throws DataSetResultTransformException; -} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/DataValueConvertionException.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/DataValueConvertionException.java index 95d7f2f2..b6555aa2 100644 --- a/datagear-analysis/src/main/java/org/datagear/analysis/support/DataValueConvertionException.java +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/DataValueConvertionException.java @@ -23,7 +23,7 @@ public class DataValueConvertionException extends RuntimeException public DataValueConvertionException(Object source, String type) { - super(); + super("Convert from [" + source + "] to [" + type + "] is not supported"); this.type = type; this.source = source; } diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/ExcelDirectoryFileDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/ExcelDirectoryFileDataSet.java new file mode 100644 index 00000000..d9b75f7b --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/ExcelDirectoryFileDataSet.java @@ -0,0 +1,87 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.analysis.support; + +import java.io.File; +import java.util.List; +import java.util.Map; + +import org.datagear.analysis.DataSet; +import org.datagear.analysis.DataSetException; +import org.datagear.analysis.DataSetProperty; +import org.datagear.util.FileUtil; + +/** + * 目录内Excel文件{@linkplain DataSet}。 + *

+ * 注意:此类不支持Freemarker模板语言。 + *

+ * + * @author datagear@163.com + * + */ +public class ExcelDirectoryFileDataSet extends AbstractExcelDataSet +{ + /** Excel文件所在的目录 */ + private File directory; + + /** Excel文件名 */ + private String fileName; + + public ExcelDirectoryFileDataSet() + { + super(); + } + + public ExcelDirectoryFileDataSet(String id, String name, File directory, String fileName) + { + super(id, name); + this.directory = directory; + this.fileName = fileName; + } + + /** + * @param id + * @param name + * @param properties + */ + public ExcelDirectoryFileDataSet(String id, String name, List properties, File directory, + String fileName) + { + super(id, name, properties); + this.directory = directory; + this.fileName = fileName; + } + + public File getDirectory() + { + return directory; + } + + public void setDirectory(File directory) + { + this.directory = directory; + } + + public String getFileName() + { + return fileName; + } + + public void setFileName(String fileName) + { + this.fileName = fileName; + } + + @Override + protected File getExcelFile(Map paramValues) throws DataSetException + { + File excelFile = FileUtil.getFile(this.directory, this.fileName); + return excelFile; + } +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/HeaderContentNotNameValueObjArrayJsonException.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/HeaderContentNotNameValueObjArrayJsonException.java new file mode 100644 index 00000000..7cea0844 --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/HeaderContentNotNameValueObjArrayJsonException.java @@ -0,0 +1,39 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.analysis.support; + +/** + * {@linkplain HttpDataSet#getHeaderContent()}不是名/值对象数组JSON异常。 + * + * @author datagear@163.com + * + */ +public class HeaderContentNotNameValueObjArrayJsonException extends NotNameValueObjArrayJsonException +{ + private static final long serialVersionUID = 1L; + + public HeaderContentNotNameValueObjArrayJsonException(String json) + { + super(json); + } + + public HeaderContentNotNameValueObjArrayJsonException(String json, String message) + { + super(json, message); + } + + public HeaderContentNotNameValueObjArrayJsonException(String json, Throwable cause) + { + super(json, cause); + } + + public HeaderContentNotNameValueObjArrayJsonException(String json, String message, Throwable cause) + { + super(json, message, cause); + } +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/HttpDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/HttpDataSet.java new file mode 100644 index 00000000..08ef21db --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/HttpDataSet.java @@ -0,0 +1,622 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.analysis.support; + +import java.io.IOException; +import java.io.Reader; +import java.nio.charset.Charset; +import java.util.ArrayList; +import java.util.Collection; +import java.util.Collections; +import java.util.List; +import java.util.Map; + +import org.apache.hc.client5.http.HttpResponseException; +import org.apache.hc.client5.http.classic.HttpClient; +import org.apache.hc.client5.http.classic.methods.HttpDelete; +import org.apache.hc.client5.http.classic.methods.HttpGet; +import org.apache.hc.client5.http.classic.methods.HttpPatch; +import org.apache.hc.client5.http.classic.methods.HttpPost; +import org.apache.hc.client5.http.classic.methods.HttpPut; +import org.apache.hc.client5.http.entity.UrlEncodedFormEntity; +import org.apache.hc.core5.http.ClassicHttpRequest; +import org.apache.hc.core5.http.ClassicHttpResponse; +import org.apache.hc.core5.http.ContentType; +import org.apache.hc.core5.http.HttpEntity; +import org.apache.hc.core5.http.HttpException; +import org.apache.hc.core5.http.NameValuePair; +import org.apache.hc.core5.http.io.HttpClientResponseHandler; +import org.apache.hc.core5.http.io.entity.StringEntity; +import org.apache.hc.core5.http.message.BasicNameValuePair; +import org.datagear.analysis.DataSetException; +import org.datagear.analysis.DataSetProperty; +import org.datagear.analysis.DataSetResult; +import org.datagear.analysis.ResolvedDataSetResult; +import org.datagear.util.IOUtil; +import org.datagear.util.StringUtil; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.fasterxml.jackson.databind.ObjectMapper; + +/** + * HTTP数据集。 + *

+ * 此类的{@linkplain #getUri()}、{@linkplain #getHeaderContent()}、{@linkplain #getRequestContent()}支持Freemarker模板语言。 + *

+ * + * @author datagear@163.com + * + */ +public class HttpDataSet extends AbstractResolvableDataSet +{ + protected static final Logger LOGGER = LoggerFactory.getLogger(HttpDataSet.class); + + public static final String REQUEST_METHOD_GET = "GET"; + + public static final String REQUEST_METHOD_POST = "POST"; + + public static final String REQUEST_METHOD_PUT = "PUT"; + + public static final String REQUEST_METHOD_PATCH = "PATCH"; + + public static final String REQUEST_METHOD_DELETE = "DELETE"; + + // 这些HTTP方法不适应于此 + // public static final String REQUEST_METHOD_HEAD = "HEAD"; + // public static final String REQUEST_METHOD_OPTIONS = "OPTIONS"; + // public static final String REQUEST_METHOD_TRACE = "TRACE"; + + /** + * 请求内容类型:表单式的参数名/值类型,对应的HTTP请求类型为:application/x-www-form-urlencoded + */ + public static final String REQUEST_CONTENT_TYPE_FORM_URLENCODED = "FORM_URLENCODED"; + + /** + * 请求内容类型:JSON,对应的HTTP请求类型为:application/json + */ + public static final String REQUEST_CONTENT_TYPE_JSON = "JSON"; + + /** + * 响应内容类型:JSON,对应的HTTP响应类型为:application/json + */ + public static final String RESPONSE_CONTENT_TYPE_JSON = "JSON"; + + protected static final List NOT_NAME_VALUE_PAIR_OBJ_ARRAY_JSON = new ArrayList<>(0); + + /** HTTP客户端 */ + private HttpClient httpClient; + + /** HTTP请求地址 */ + private String uri; + + /** 请求头JSON文本 */ + private String headerContent = ""; + + /** 请求方法 */ + private String requestMethod = REQUEST_METHOD_GET; + + /** 请求内容类型 */ + private String requestContentType = REQUEST_CONTENT_TYPE_FORM_URLENCODED; + + /** 请求内容编码 */ + private String requestContentCharset = IOUtil.CHARSET_UTF_8; + + /** 请求内容JSON文本 */ + private String requestContent = ""; + + /** 响应类型 */ + private String responseContentType = RESPONSE_CONTENT_TYPE_JSON; + + /** 响应数据的JSON路径 */ + private String responseDataJsonPath = ""; + + public HttpDataSet() + { + super(); + } + + public HttpDataSet(String id, String name, HttpClient httpClient, String uri) + { + super(id, name); + this.httpClient = httpClient; + this.uri = uri; + } + + public HttpDataSet(String id, String name, List properties, HttpClient httpClient, String uri) + { + super(id, name, properties); + this.httpClient = httpClient; + this.uri = uri; + } + + public HttpClient getHttpClient() + { + return httpClient; + } + + public void setHttpClient(HttpClient httpClient) + { + this.httpClient = httpClient; + } + + public String getUri() + { + return uri; + } + + /** + * 设置请求地址。 + *

+ * 请求地址支持Freemarker模板语言。 + *

+ * + * @param uri + */ + public void setUri(String uri) + { + this.uri = uri; + } + + public String getHeaderContent() + { + return headerContent; + } + + /** + * 设置请求头JSON文本,格式应为: + *

+ * + *

+	 * [
+	 *   {name: "...", value: "..."},
+	 *   {name: "...", value: "..."},
+	 *   ...
+	 * ]
+	 * 
+ * + *

+ *

+ * 请求头JSON文本支持Freemarker模板语言。 + *

+ * + * @param headerContent + */ + public void setHeaderContent(String headerContent) + { + this.headerContent = headerContent; + } + + public String getRequestMethod() + { + return requestMethod; + } + + /** + * 设置HTTP方法,参考{@code REQUEST_METHOD_*}常量。 + * + * @param requestMethod + */ + public void setRequestMethod(String requestMethod) + { + this.requestMethod = requestMethod; + } + + public String getRequestContentType() + { + return requestContentType; + } + + /** + * 设置请求内容类型,允许的值为: + *

+ * {@linkplain #REQUEST_CONTENT_TYPE_FORM_URLENCODED}、{@linkplain #REQUEST_CONTENT_TYPE_JSON}。 + *

+ * + * @param requestContentType + */ + public void setRequestContentType(String requestContentType) + { + this.requestContentType = requestContentType; + } + + public String getRequestContentCharset() + { + return requestContentCharset; + } + + /** + * 设置请求内容编码。 + *

+ * 默认请求内容编码为{@code UTF-8}。 + *

+ * + * @param requestContentCharset + */ + public void setRequestContentCharset(String requestContentCharset) + { + this.requestContentCharset = requestContentCharset; + } + + public String getRequestContent() + { + return requestContent; + } + + /** + * 设置请求内容JSON文本,为{@code null}或{@code ""}表示无请求内容。 + *

+ * 当{@linkplain #getRequestContentType()}为{@linkplain #REQUEST_CONTENT_TYPE_FORM_URLENCODED}时,请求内容JSON文本格式应为: + *

+ * + *
+	 * [
+	 *   {name: "...", value: "..."},
+	 *   {name: "...", value: "..."},
+	 *   ...
+	 * ]
+	 * 
+ *
+ *

+ * 其中,{@code name}表示请求参数名,{@code value}表示请求参数值。 + *

+ *

+ * 当{@linkplain #getRequestContentType()}为{@linkplain #REQUEST_CONTENT_TYPE_JSON}时,请求内容JSON文本没有特殊格式要求。 + *

+ *

+ * 请求内容JSON文本支持Freemarker模板语言。 + *

+ * + * @param requestContent + */ + public void setRequestContent(String requestContent) + { + this.requestContent = requestContent; + } + + public String getResponseContentType() + { + return responseContentType; + } + + /** + * 设置相应类型。 + *

+ * 目前仅支持{@linkplain #RESPONSE_CONTENT_TYPE_JSON},且是默认值。 + *

+ * + * @param responseContentType + */ + public void setResponseContentType(String responseContentType) + { + this.responseContentType = responseContentType; + } + + public String getResponseDataJsonPath() + { + return responseDataJsonPath; + } + + /** + * 设置响应数据的JSON路径。 + *

+ * 当希望返回的是响应原始JSON数据的指定JSON路径值时,可以设置此项。 + *

+ *

+ * 具体格式参考{@linkplain AbstractJsonDataSet#setDataJsonPath(String)}。 + *

+ *

+ * 默认无数据路径,将直接返回响应原始JSON数据。 + *

+ * + * @param responseDataJsonPath + */ + public void setResponseDataJsonPath(String responseDataJsonPath) + { + this.responseDataJsonPath = responseDataJsonPath; + } + + @Override + public TemplateResolvedDataSetResult resolve(Map paramValues) throws DataSetException + { + return resolveResult(paramValues, null); + } + + @Override + protected TemplateResolvedDataSetResult resolveResult(Map paramValues, List properties) + throws DataSetException + { + try + { + String uri = resolveTemplateUri(paramValues); + String headerContent = resolveTemplateHeaderContent(paramValues); + String requestContent = resolveTemplateRequestContent(paramValues); + + ClassicHttpRequest request = createHttpRequest(uri); + + setHttpHeaders(request, headerContent); + setHttpEntity(request, requestContent); + + JsonResponseHandler responseHandler = new JsonResponseHandler(); + responseHandler.setProperties(properties); + responseHandler.setResponseDataJsonPath(getResponseDataJsonPath()); + + ResolvedDataSetResult result = this.httpClient.execute(request, responseHandler); + + String templateResult = "URI:" + System.lineSeparator() + uri // + + System.lineSeparator() + "-----------------------------------------" + System.lineSeparator() // + + "Request headers:" + System.lineSeparator() + headerContent // + + System.lineSeparator() + "-----------------------------------------" + System.lineSeparator() // + + "Request content:" + System.lineSeparator() + requestContent; + + return new TemplateResolvedDataSetResult(result.getResult(), result.getProperties(), templateResult); + } + catch (DataSetException e) + { + throw e; + } + catch (Throwable t) + { + throw new DataSetSourceParseException(t); + } + } + + protected void setHttpHeaders(ClassicHttpRequest request, String headerContent) throws Throwable + { + if (StringUtil.isEmpty(headerContent)) + return; + + List headers = toNameValuePairs(headerContent); + + if (headers == NOT_NAME_VALUE_PAIR_OBJ_ARRAY_JSON) + throw new HeaderContentNotNameValueObjArrayJsonException(headerContent); + + for (NameValuePair header : headers) + request.setHeader(header.getName(), header.getValue()); + } + + protected void setHttpEntity(ClassicHttpRequest request, String requestContent) throws Throwable + { + if (REQUEST_CONTENT_TYPE_FORM_URLENCODED.equals(this.requestContentType)) + { + List params = toNameValuePairs(requestContent); + + if (params == NOT_NAME_VALUE_PAIR_OBJ_ARRAY_JSON) + throw new RequestContentNotNameValueObjArrayJsonException(requestContent); + + request.setEntity(new UrlEncodedFormEntity(params, Charset.forName(this.requestContentCharset))); + } + else if (REQUEST_CONTENT_TYPE_JSON.equals(this.requestContentType)) + { + ContentType contentType = ContentType.create(ContentType.APPLICATION_JSON.getMimeType(), + Charset.forName(this.requestContentCharset)); + StringEntity entity = new StringEntity(requestContent, contentType); + request.setEntity(entity); + } + else + throw new DataSetException("Request content type [" + this.requestContentType + "] is not supported"); + } + + protected String resolveTemplateUri(Map paramValues) throws Throwable + { + return resolveAsFmkTemplateIfHasParam(this.uri, paramValues); + } + + protected String resolveTemplateHeaderContent(Map paramValues) throws Throwable + { + return resolveAsFmkTemplateIfHasParam(this.headerContent, paramValues); + } + + protected String resolveTemplateRequestContent(Map paramValues) throws Throwable + { + return resolveAsFmkTemplateIfHasParam(this.requestContent, paramValues); + } + + protected ClassicHttpRequest createHttpRequest(String uri) throws Throwable + { + if (REQUEST_METHOD_GET.equals(this.requestMethod) || StringUtil.isEmpty(this.requestMethod)) + return new HttpGet(uri); + else if (REQUEST_METHOD_POST.equals(this.requestMethod)) + return new HttpPost(uri); + else if (REQUEST_METHOD_PUT.equals(this.requestMethod)) + return new HttpPut(uri); + else if (REQUEST_METHOD_PATCH.equals(this.requestMethod)) + return new HttpPatch(uri); + else if (REQUEST_METHOD_DELETE.equals(this.requestMethod)) + return new HttpDelete(uri); + // else if (REQUEST_METHOD_HEAD.equals(this.httpMethod)) + // return new HttpHead(uri); + // else if (REQUEST_METHOD_OPTIONS.equals(this.httpMethod)) + // return new HttpOptions(uri); + // else if (REQUEST_METHOD_TRACE.equals(this.httpMethod)) + // return new HttpTrace(uri); + else + throw new DataSetException("HTTP method [" + this.requestMethod + "] is not supported"); + } + + /** + * 将指定JSON字符串转换为名/值列表。 + * + * @param nameValueObjJsonArray + * 允许为{@code null}、{@code ""} + * @return 空列表表示无名/值,返回{@code #NOT_NAME_VALUE_PAIR_OBJ_ARRAY_JSON}表示{@code nameValueObjJsonArray}格式不合法 + * @throws Throwable + */ + @SuppressWarnings("unchecked") + protected List toNameValuePairs(String nameValueObjJsonArray) throws Throwable + { + if (StringUtil.isEmpty(nameValueObjJsonArray)) + return Collections.EMPTY_LIST; + + Object jsonObj = getObjectMapperNonStardand().readValue(nameValueObjJsonArray, Object.class); + + if (jsonObj == null) + return Collections.EMPTY_LIST; + + if (!(jsonObj instanceof Collection)) + return NOT_NAME_VALUE_PAIR_OBJ_ARRAY_JSON; + + Collection collection = (Collection) jsonObj; + + List nameValuePairs = new ArrayList<>(collection.size()); + + for (Object ele : collection) + { + String name = null; + String value = null; + + if (ele instanceof Map) + { + Map eleMap = (Map) ele; + Object nameVal = eleMap.get("name"); + Object valueVal = eleMap.get("value"); + + if (nameVal instanceof String) + { + name = (String) nameVal; + if (valueVal != null) + value = (valueVal instanceof String ? (String) valueVal : valueVal.toString()); + } + } + + if (name == null) + return NOT_NAME_VALUE_PAIR_OBJ_ARRAY_JSON; + + nameValuePairs.add(new BasicNameValuePair(name, value)); + } + + return nameValuePairs; + } + + protected ObjectMapper getObjectMapperNonStardand() + { + return JsonSupport.getObjectMapperNonStardand(); + } + + protected static class JsonResponseHandler implements HttpClientResponseHandler + { + private List properties; + + private String responseDataJsonPath = ""; + + public JsonResponseHandler() + { + super(); + } + + public List getProperties() + { + return properties; + } + + /** + * 设置数据集属性。 + * + * @param properties + * 如果为{@code null}或空,则执行解析 + */ + public void setProperties(List properties) + { + this.properties = properties; + } + + public String getResponseDataJsonPath() + { + return responseDataJsonPath; + } + + public void setResponseDataJsonPath(String responseDataJsonPath) + { + this.responseDataJsonPath = responseDataJsonPath; + } + + @SuppressWarnings("unchecked") + @Override + public ResolvedDataSetResult handleResponse(ClassicHttpResponse response) throws HttpException, IOException + { + int code = response.getCode(); + HttpEntity entity = response.getEntity(); + + if (code < 200 || code >= 300) + throw new HttpResponseException(code, response.getReasonPhrase()); + + Reader reader = null; + + if (entity == null) + reader = IOUtil.getReader(""); + else + { + Charset contentCharset = resolveCharset(entity, ContentType.APPLICATION_JSON.getCharset()); + reader = IOUtil.getReader(entity.getContent(), contentCharset); + } + + if (this.properties == null || this.properties.isEmpty()) + { + HttpResponseJsonDataSet jsonDataSet = new HttpResponseJsonDataSet(reader); + jsonDataSet.setDataJsonPath(this.responseDataJsonPath); + + return jsonDataSet.resolve(Collections.EMPTY_MAP); + } + else + { + HttpResponseJsonDataSet jsonDataSet = new HttpResponseJsonDataSet(this.properties, reader); + jsonDataSet.setDataJsonPath(this.responseDataJsonPath); + + DataSetResult result = jsonDataSet.getResult(Collections.EMPTY_MAP); + return new ResolvedDataSetResult(result, this.properties); + } + } + + protected Charset resolveCharset(HttpEntity entity, Charset defaultCharset) + { + Charset contentCharset = null; + + String contentTypeStr = entity.getContentType(); + + if (!StringUtil.isEmpty(contentTypeStr)) + { + try + { + ContentType contentType = ContentType.parse(contentTypeStr); + contentCharset = contentType.getCharset(); + } + catch (Throwable t) + { + LOGGER.warn("Default charset [" + defaultCharset + "] will be used because parse error", t); + + contentCharset = defaultCharset; + } + } + + return (contentCharset != null ? contentCharset : defaultCharset); + } + } + + protected static class HttpResponseJsonDataSet extends AbstractJsonDataSet + { + private Reader responseJsonReader; + + public HttpResponseJsonDataSet(Reader responseJsonReader) + { + super(HttpResponseJsonDataSet.class.getName(), HttpResponseJsonDataSet.class.getName()); + this.responseJsonReader = responseJsonReader; + } + + public HttpResponseJsonDataSet(List properties, Reader responseJsonReader) + { + super(HttpResponseJsonDataSet.class.getName(), HttpResponseJsonDataSet.class.getName(), properties); + this.responseJsonReader = responseJsonReader; + } + + @Override + protected TemplateResolvedSource getJsonReader(Map paramValues) throws Throwable + { + return new TemplateResolvedSource<>(this.responseJsonReader); + } + } +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/JsonDataSetSupport.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/JsonDataSetSupport.java deleted file mode 100644 index c2e0ae76..00000000 --- a/datagear-analysis/src/main/java/org/datagear/analysis/support/JsonDataSetSupport.java +++ /dev/null @@ -1,274 +0,0 @@ -/* - * Copyright (c) 2018 datagear.tech. All Rights Reserved. - */ - -package org.datagear.analysis.support; - -import java.io.File; -import java.io.IOException; -import java.io.Reader; -import java.io.StringReader; -import java.util.ArrayList; -import java.util.Collections; -import java.util.List; -import java.util.Map; - -import org.datagear.analysis.DataSet; -import org.datagear.analysis.DataSetException; -import org.datagear.analysis.DataSetProperty; -import org.datagear.analysis.DataSetResult; -import org.datagear.util.IOUtil; - -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.node.ArrayNode; -import com.fasterxml.jackson.databind.node.ObjectNode; -import com.fasterxml.jackson.databind.node.ValueNode; - -/** - * JSON {@linkplain DataSet}支持类。 - * - * @author datagear@163.com - * - */ -public class JsonDataSetSupport extends JsonSupport -{ - public JsonDataSetSupport() - { - super(); - } - - /** - * 解析JSON数据。 - * - * @param jsonValue - * @return - * @throws DataSetSourceParseException - * @throws DataSetException - */ - public Object resolveValue(String jsonValue) throws DataSetSourceParseException, DataSetException - { - StringReader reader = new StringReader(jsonValue); - return resolveValue(reader); - } - - /** - * 解析JSON数据。 - * - * @param jsonReader - * @return - * @throws DataSetSourceParseException - * @throws DataSetException - */ - public Object resolveValue(Reader jsonReader) throws DataSetSourceParseException, DataSetException - { - try - { - return parseNonStardand(jsonReader, Object.class); - } - catch (Throwable t) - { - throw new DataSetException(t); - } - } - - /** - * 解析数据集结果数据。 - * - * @param jsonValue - * @return - * @throws DataSetSourceParseException - * @throws UnsupportedJsonResultDataException - * @throws DataSetException - */ - public Object resolveResultData(String jsonValue) - throws DataSetSourceParseException, UnsupportedJsonResultDataException, DataSetException - { - StringReader reader = new StringReader(jsonValue); - return resolveResultData(reader); - } - - /** - * 解析数据集结果数据。 - * - * @param file - * @param encoding - * @return - * @throws DataSetSourceParseException - * @throws UnsupportedJsonResultDataException - * @throws DataSetException - */ - public Object resolveResultData(File file, String encoding) - throws DataSetSourceParseException, UnsupportedJsonResultDataException, DataSetException - { - Reader reader = null; - - try - { - reader = IOUtil.getReader(file, encoding); - return resolveResultData(reader); - } - catch (IOException e) - { - throw new DataSetException(e); - } - finally - { - IOUtil.close(reader); - } - } - - /** - * 解析数据集结果数据。 - * - * @param reader - * @return - * @throws DataSetSourceParseException - * @throws UnsupportedJsonResultDataException - * @throws DataSetException - */ - public Object resolveResultData(Reader reader) - throws DataSetSourceParseException, UnsupportedJsonResultDataException, DataSetException - { - JsonNode jsonNode = null; - - try - { - jsonNode = getObjectMapperNonStardand().readTree(reader); - } - catch (Throwable t) - { - throw new DataSetSourceParseException(t); - } - - if (!isLegalResultDataJsonNode(jsonNode)) - throw new UnsupportedJsonResultDataException("Result data must be object or object array/list"); - - if (jsonNode == null) - return null; - - Object data = null; - - try - { - data = getObjectMapperNonStardand().treeToValue(jsonNode, Object.class); - } - catch (Throwable t) - { - throw new DataSetException(t); - } - - return data; - } - - /** - * 是否是合法的数据集结果数据{@linkplain JsonNode}。 - *

- * 参考{@linkplain DataSetResult#getData()}说明。 - *

- * - * @param jsonNode - * @return - */ - public boolean isLegalResultDataJsonNode(JsonNode jsonNode) - { - if (jsonNode == null || jsonNode.isNull()) - return true; - - if (jsonNode instanceof ValueNode) - return false; - - if (jsonNode instanceof ArrayNode) - { - ArrayNode arrayNode = (ArrayNode) jsonNode; - - for (int i = 0; i < arrayNode.size(); i++) - { - JsonNode eleNode = arrayNode.get(i); - - if (eleNode == null || eleNode.isNull()) - continue; - - if (!(eleNode instanceof ObjectNode)) - return false; - } - } - - return true; - } - - /** - * 解析JSON对象的{@linkplain DataSetProperty}。 - * - * @param resultData - * JSON对象、JSON对象数组、JSON对象列表 - * @return - * @throws UnsupportedJsonResultDataException - */ - @SuppressWarnings("unchecked") - public List resolveDataSetProperties(Object resultData) throws UnsupportedJsonResultDataException - { - if (resultData == null) - { - return Collections.EMPTY_LIST; - } - else if (resultData instanceof Map) - { - return resolveJsonObjDataSetProperties((Map) resultData); - } - else if (resultData instanceof List) - { - List list = (List) resultData; - - if (list.size() == 0) - return Collections.EMPTY_LIST; - else - return resolveJsonObjDataSetProperties((Map) list.get(0)); - } - else if (resultData instanceof Object[]) - { - Object[] array = (Object[]) resultData; - - if (array.length == 0) - return Collections.EMPTY_LIST; - else - return resolveJsonObjDataSetProperties((Map) array[0]); - } - else - throw new UnsupportedJsonResultDataException("Result data must be object or object array/list"); - } - - /** - * 解析JSON对象的{@linkplain DataSetProperty}。 - * - * @param jsonObj - * @return - */ - public List resolveJsonObjDataSetProperties(Map jsonObj) - { - List properties = new ArrayList<>(); - - if (jsonObj == null) - { - - } - else - { - for (Map.Entry entry : jsonObj.entrySet()) - { - Object value = entry.getValue(); - String type = DataSetProperty.DataType.resolveDataType(value); - - DataSetProperty property = new DataSetProperty(entry.getKey(), type); - - // JSON数值只有NUMBER类型 - if (DataSetProperty.DataType.isInteger(property.getType()) - || DataSetProperty.DataType.isDecimal(property.getType())) - property.setType(DataSetProperty.DataType.NUMBER); - - properties.add(property); - } - } - - return properties; - } -} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/JsonDirectoryFileDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/JsonDirectoryFileDataSet.java index f4614ce5..80d14f47 100644 --- a/datagear-analysis/src/main/java/org/datagear/analysis/support/JsonDirectoryFileDataSet.java +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/JsonDirectoryFileDataSet.java @@ -18,6 +18,9 @@ import org.datagear.util.FileUtil; /** * 目录内JSON文件{@linkplain DataSet}。 + *

+ * 注意:此类不支持Freemarker模板语言。 + *

* * @author datagear@163.com * @@ -73,8 +76,7 @@ public class JsonDirectoryFileDataSet extends AbstractJsonFileDataSet @Override protected File getJsonFile(Map paramValues) throws DataSetException { - String fileName = resolveTemplate(this.fileName, paramValues); - File jsonFile = FileUtil.getFile(directory, fileName); + File jsonFile = FileUtil.getFile(this.directory, this.fileName); return jsonFile; } } diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/JsonValueDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/JsonValueDataSet.java index 089d2451..d43eeaed 100644 --- a/datagear-analysis/src/main/java/org/datagear/analysis/support/JsonValueDataSet.java +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/JsonValueDataSet.java @@ -7,12 +7,13 @@ */ package org.datagear.analysis.support; +import java.io.Reader; import java.util.List; import java.util.Map; import org.datagear.analysis.DataSetException; import org.datagear.analysis.DataSetProperty; -import org.datagear.analysis.DataSetResult; +import org.datagear.util.IOUtil; /** * JSON字符串值数据集。 @@ -55,25 +56,15 @@ public class JsonValueDataSet extends AbstractJsonDataSet } @Override - public DataSetResult getResult(Map paramValues) throws DataSetException + public TemplateResolvedDataSetResult resolve(Map paramValues) throws DataSetException { - String json = resolveTemplate(this.value, paramValues); - - Object data = getJsonDataSetSupport().resolveResultData(json); - - return new DataSetResult(data); + return (TemplateResolvedDataSetResult) resolveResult(paramValues, null); } @Override - public TemplateResolvedDataSetResult resolve(Map paramValues) throws DataSetException + protected TemplateResolvedSource getJsonReader(Map paramValues) throws Throwable { - String json = resolveTemplate(this.value, paramValues); - - Object data = getJsonDataSetSupport().resolveResultData(json); - DataSetResult result = new DataSetResult(data); - - List properties = getJsonDataSetSupport().resolveDataSetProperties(result.getData()); - - return new TemplateResolvedDataSetResult(result, properties, json); + String json = resolveAsFmkTemplateIfHasParam(this.value, paramValues); + return new TemplateResolvedSource<>(IOUtil.getReader(json), json); } } diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/NotNameValueObjArrayJsonException.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/NotNameValueObjArrayJsonException.java new file mode 100644 index 00000000..7b3298d8 --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/NotNameValueObjArrayJsonException.java @@ -0,0 +1,57 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.analysis.support; + +import org.datagear.analysis.DataSetException; + +/** + * JSON字符串不是名/值数组格式异常。 + * + * @author datagear@163.com + * + */ +public class NotNameValueObjArrayJsonException extends DataSetException +{ + private static final long serialVersionUID = 1L; + + private String json; + + public NotNameValueObjArrayJsonException(String json) + { + super("The json must be name/value object array"); + this.json = json; + } + + public NotNameValueObjArrayJsonException(String json, String message) + { + super(message); + this.json = json; + } + + public NotNameValueObjArrayJsonException(String json, Throwable cause) + { + super(cause); + this.json = json; + } + + public NotNameValueObjArrayJsonException(String json, String message, Throwable cause) + { + super(message, cause); + this.json = json; + } + + public String getJson() + { + return json; + } + + protected void setJson(String json) + { + this.json = json; + } +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/RangeExpResolver.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/RangeExpResolver.java new file mode 100644 index 00000000..2d2ed442 --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/RangeExpResolver.java @@ -0,0 +1,465 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.analysis.support; + +import java.io.Serializable; +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; + +import org.datagear.util.StringUtil; + +/** + * 范围解析器。 + *

+ * 此类用于解析诸如{@code "1, 2-3, 6-, -15}之类的范围表达式。 + *

+ * + * @author datagear@163.com + * + */ +public class RangeExpResolver +{ + public static final char RANGE_SPLITTER_CHAR = '-'; + + public static final String RANGE_SPLITTER_STRING = "-"; + + public static final char RANGE_GROUP_SPLITTER_CHAR = ','; + + public static final String RANGE_GROUP_SPLITTER_STRING = ","; + + private char rangeSplitter = RANGE_SPLITTER_CHAR; + + private char rangeGroupSplitter = RANGE_GROUP_SPLITTER_CHAR; + + public RangeExpResolver() + { + super(); + } + + public char getRangeSplitter() + { + return rangeSplitter; + } + + public void setRangeSplitter(char rangeSplitter) + { + this.rangeSplitter = rangeSplitter; + } + + public char getRangeGroupSplitter() + { + return rangeGroupSplitter; + } + + public void setRangeGroupSplitter(char rangeGroupSplitter) + { + this.rangeGroupSplitter = rangeGroupSplitter; + } + + @SuppressWarnings("unchecked") + public List resolveIndex(String exp) throws NumberFormatException + { + List ranges = resolve(exp); + + if (ranges.isEmpty()) + return Collections.EMPTY_LIST; + + List indexRanges = new ArrayList<>(ranges.size()); + + for (Range range : ranges) + indexRanges.add(new IndexRange(range)); + + return indexRanges; + } + + /** + * 解析范围表达式组。 + *

+ * 例如:{@code "1, 2-5, 8-, -15"} + *

+ * + * @param exp + * @return 如果{@code exp}为{@code null}、或{@code ""},将返回空列表 + */ + @SuppressWarnings("unchecked") + public List resolve(String exp) + { + if (exp != null) + exp = exp.trim(); + + if (exp == null || exp.isEmpty()) + return Collections.EMPTY_LIST; + + List ranges = new ArrayList<>(); + + String[] ss = StringUtil.split(exp, this.rangeGroupSplitter + "", false); + + for (String s : ss) + { + Range range = resolveSingle(s); + + if (range != null) + ranges.add(range); + } + + return ranges; + } + + public IndexRange resolveSingleIndex(String exp) throws NumberFormatException + { + Range range = resolveSingle(exp); + + if (range == null) + return null; + + return new IndexRange(range); + } + + /** + * 解析单个范围表达式。 + *

+ * 例如:{@code "1"}、{@code "4-5"}、{@code "8-"}、{@code "-15"} + *

+ * + * @param exp + * @return 如果{@code exp}为{@code null}、或{@code ""},将返回{@code null} + */ + public Range resolveSingle(String exp) + { + if (exp != null) + exp = exp.trim(); + + if (exp == null || exp.isEmpty()) + return null; + + String from = ""; + String to = ""; + + int idx = exp.indexOf(this.rangeSplitter); + int len = exp.length(); + + // 单个值:"1" + if (idx < 0) + { + from = exp; + to = from; + } + // 都未指定:"-" + else if (idx == 0 && len == 1) + { + from = ""; + to = ""; + } + // 仅指定截至值:"-4" + else if (idx == 0) + { + from = ""; + to = exp.substring(1); + } + // 仅指定起始值:"4-" + else if (idx == len - 1) + { + from = exp.substring(0, len - 1); + to = ""; + } + // 都指定:"3-5" + else + { + from = exp.substring(0, idx); + to = exp.substring(idx + 1); + } + + return new Range(from, to); + } + + public static RangeExpResolver valueOf(char rangeSplitter, char rangeGroupSplitter) + { + RangeExpResolver resolver = new RangeExpResolver(); + resolver.setRangeSplitter(rangeSplitter); + resolver.setRangeGroupSplitter(rangeGroupSplitter); + + return resolver; + } + + /** + * 范围。 + * + * @author datagear@163.com + * + */ + public static class Range implements Serializable + { + private static final long serialVersionUID = 1L; + + /** 起始 */ + private String from = ""; + + /** 截至 */ + private String to = ""; + + public Range() + { + super(); + } + + public Range(String from) + { + super(); + this.from = from; + } + + public Range(String from, String to) + { + super(); + this.from = from; + this.to = to; + } + + public boolean hasFrom() + { + return (this.from != null && !this.from.isEmpty()); + } + + public String trimFrom() + { + if (this.from == null) + return ""; + + return this.from.trim(); + } + + public String getFrom() + { + return from; + } + + public void setFrom(String from) + { + this.from = from; + } + + public boolean hasTo() + { + return (this.to != null && !this.to.isEmpty()); + } + + public String trimTo() + { + if (this.to == null) + return ""; + + return this.to.trim(); + } + + public String getTo() + { + return to; + } + + public void setTo(String to) + { + this.to = to; + } + + @Override + public int hashCode() + { + final int prime = 31; + int result = 1; + result = prime * result + ((from == null) ? 0 : from.hashCode()); + result = prime * result + ((to == null) ? 0 : to.hashCode()); + return result; + } + + @Override + public boolean equals(Object obj) + { + if (this == obj) + return true; + if (obj == null) + return false; + if (getClass() != obj.getClass()) + return false; + Range other = (Range) obj; + if (from == null) + { + if (other.from != null) + return false; + } + else if (!from.equals(other.from)) + return false; + if (to == null) + { + if (other.to != null) + return false; + } + else if (!to.equals(other.to)) + return false; + return true; + } + + @Override + public String toString() + { + return getClass().getSimpleName() + " [from=" + from + ", to=" + to + "]"; + } + } + + /** + * 索引范围。 + *

+ * 索引指大于或等于{@code 0}的整数值。 + *

+ * + * @author datagear@163.com + * + */ + protected static class IndexRange implements Serializable + { + private static final long serialVersionUID = 1L; + + /** 起始索引 */ + private int from = 0; + + /** 截至索引(包含) */ + private int to = -1; + + public IndexRange() + { + super(); + this.from = 0; + this.to = -1; + } + + public IndexRange(int from) + { + super(); + this.from = from; + this.to = -1; + } + + public IndexRange(int from, int to) + { + super(); + this.from = from; + this.to = to; + } + + public IndexRange(Range range) throws NumberFormatException + { + super(); + + int from = 0; + int to = -1; + + String fromStr = range.trimFrom(); + String toStr = range.trimTo(); + + if (!StringUtil.isEmpty(fromStr)) + from = Integer.parseInt(fromStr); + + if (!StringUtil.isEmpty(toStr)) + to = Integer.parseInt(toStr); + + this.from = from; + this.to = to; + } + + public int getFrom() + { + return from; + } + + /** + * 设置起始索引。 + * + * @param from + * 起始索引,小于{@code 0}表示不限定 + */ + public void setFrom(int from) + { + this.from = from; + } + + public int getTo() + { + return to; + } + + /** + * 设置截至索引(包含)。 + * + * @param to + * 截至索引,小于{@code 0}表示不限定 + */ + public void setTo(int to) + { + this.to = to; + } + + /** + * 是否包含给定索引数值。 + * + * @param index + * @return + */ + public boolean includes(int index) + { + if (this.from > -1 && index < this.from) + return false; + + if (this.to > -1 && index > this.to) + return false; + + return true; + } + + @Override + public int hashCode() + { + final int prime = 31; + int result = 1; + result = prime * result + from; + result = prime * result + to; + return result; + } + + @Override + public boolean equals(Object obj) + { + if (this == obj) + return true; + if (obj == null) + return false; + if (getClass() != obj.getClass()) + return false; + IndexRange other = (IndexRange) obj; + if (from != other.from) + return false; + if (to != other.to) + return false; + return true; + } + + @Override + public String toString() + { + return getClass().getSimpleName() + " [from=" + from + ", to=" + to + "]"; + } + + public static boolean includes(List indexRanges, int index) + { + for (int i = 0; i < indexRanges.size(); i++) + if (indexRanges.get(i).includes(index)) + return true; + + return false; + } + } +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/ReadJsonDataPathException.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/ReadJsonDataPathException.java new file mode 100644 index 00000000..741b7b62 --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/ReadJsonDataPathException.java @@ -0,0 +1,55 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.analysis.support; + +/** + * 读取指定JSON路径的数据异常。 + * + * @author datagear@163.com + * + */ +public class ReadJsonDataPathException extends DataSetSourceParseException +{ + private static final long serialVersionUID = 1L; + + private String dataPath; + + public ReadJsonDataPathException(String dataPath) + { + super(); + this.dataPath = dataPath; + } + + public ReadJsonDataPathException(String dataPath, String message) + { + super(message); + this.dataPath = dataPath; + } + + public ReadJsonDataPathException(String dataPath, Throwable cause) + { + super(cause); + this.dataPath = dataPath; + } + + public ReadJsonDataPathException(String dataPath, String message, Throwable cause) + { + super(message, cause); + this.dataPath = dataPath; + } + + public String getDataPath() + { + return dataPath; + } + + protected void setDataPath(String dataPath) + { + this.dataPath = dataPath; + } +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/RequestContentNotNameValueObjArrayJsonException.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/RequestContentNotNameValueObjArrayJsonException.java new file mode 100644 index 00000000..e5813096 --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/RequestContentNotNameValueObjArrayJsonException.java @@ -0,0 +1,39 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.analysis.support; + +/** + * {@linkplain HttpDataSet#getRequestContent()}不是名/值对象数组JSON异常。 + * + * @author datagear@163.com + * + */ +public class RequestContentNotNameValueObjArrayJsonException extends NotNameValueObjArrayJsonException +{ + private static final long serialVersionUID = 1L; + + public RequestContentNotNameValueObjArrayJsonException(String json) + { + super(json); + } + + public RequestContentNotNameValueObjArrayJsonException(String json, String message) + { + super(json, message); + } + + public RequestContentNotNameValueObjArrayJsonException(String json, Throwable cause) + { + super(json, cause); + } + + public RequestContentNotNameValueObjArrayJsonException(String json, String message, Throwable cause) + { + super(json, message, cause); + } +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/SimpleCsvFileDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/SimpleCsvFileDataSet.java new file mode 100644 index 00000000..9b5ef2f9 --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/SimpleCsvFileDataSet.java @@ -0,0 +1,59 @@ +/* + * Copyright 2018 datagear.tech. All Rights Reserved. + */ + +package org.datagear.analysis.support; + +import java.io.File; +import java.util.List; +import java.util.Map; + +import org.datagear.analysis.DataSetProperty; + +/** + * 简单CSV文件数据集。 + *

+ * 注意:此类不支持Freemarker模板语言。 + *

+ * + * @author datagear@163.com + * + */ +public class SimpleCsvFileDataSet extends AbstractCsvFileDataSet +{ + /** CSV文件 */ + private File file; + + public SimpleCsvFileDataSet() + { + super(); + } + + public SimpleCsvFileDataSet(String id, String name, File file) + { + super(id, name); + this.file = file; + } + + public SimpleCsvFileDataSet(String id, String name, List properties, File file) + { + super(id, name, properties); + this.file = file; + } + + public File getFile() + { + return file; + } + + public void setFile(File file) + { + this.file = file; + } + + @Override + protected File getCsvFile(Map paramValues) throws Throwable + { + return this.file; + } +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/SimpleExcelDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/SimpleExcelDataSet.java new file mode 100644 index 00000000..51660a5b --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/SimpleExcelDataSet.java @@ -0,0 +1,53 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.analysis.support; + +import java.io.File; +import java.util.List; +import java.util.Map; + +import org.datagear.analysis.DataSetException; +import org.datagear.analysis.DataSetProperty; + +/** + * 简单Excel数据集。 + *

+ * 注意:此类不支持Freemarker模板语言。 + *

+ * + * @author datagear@163.com + * + */ +public class SimpleExcelDataSet extends AbstractExcelDataSet +{ + /** Excel文件 */ + private File file; + + public SimpleExcelDataSet() + { + super(); + } + + public SimpleExcelDataSet(String id, String name, File file) + { + super(id, name); + this.file = file; + } + + public SimpleExcelDataSet(String id, String name, List properties, File file) + { + super(id, name, properties); + this.file = file; + } + + @Override + protected File getExcelFile(Map paramValues) throws DataSetException + { + return this.file; + } +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/SimpleJsonFileDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/SimpleJsonFileDataSet.java index 2770127d..73643a26 100644 --- a/datagear-analysis/src/main/java/org/datagear/analysis/support/SimpleJsonFileDataSet.java +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/SimpleJsonFileDataSet.java @@ -13,6 +13,9 @@ import org.datagear.analysis.DataSetProperty; /** * 简单JSON文件数据集。 + *

+ * 注意:此类不支持Freemarker模板语言。 + *

* * @author datagear@163.com * diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/SqlDataSet.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/SqlDataSet.java index 812d1f4e..45231616 100644 --- a/datagear-analysis/src/main/java/org/datagear/analysis/support/SqlDataSet.java +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/SqlDataSet.java @@ -9,20 +9,27 @@ package org.datagear.analysis.support; import java.sql.Connection; import java.sql.ResultSet; +import java.sql.ResultSetMetaData; import java.sql.SQLException; -import java.util.Collections; +import java.sql.Types; +import java.util.ArrayList; +import java.util.HashMap; import java.util.List; import java.util.Map; import org.datagear.analysis.DataSet; import org.datagear.analysis.DataSetException; import org.datagear.analysis.DataSetProperty; +import org.datagear.analysis.DataSetProperty.DataType; import org.datagear.analysis.DataSetResult; import org.datagear.analysis.ResolvableDataSet; import org.datagear.analysis.ResolvedDataSetResult; +import org.datagear.util.JDBCCompatiblity; +import org.datagear.util.JdbcSupport; import org.datagear.util.JdbcUtil; import org.datagear.util.QueryResultSet; import org.datagear.util.Sql; +import org.datagear.util.SqlType; import org.datagear.util.resource.ConnectionFactory; /** @@ -34,9 +41,9 @@ import org.datagear.util.resource.ConnectionFactory; * @author datagear@163.com * */ -public class SqlDataSet extends AbstractFmkTemplateDataSet implements ResolvableDataSet +public class SqlDataSet extends AbstractResolvableDataSet implements ResolvableDataSet { - protected static final SqlDataSetSupport SQL_DATA_SET_SUPPORT = new SqlDataSetSupport(); + protected static final JdbcSupport JDBC_SUPPORT = new JdbcSupport(); private ConnectionFactory connectionFactory; @@ -47,10 +54,9 @@ public class SqlDataSet extends AbstractFmkTemplateDataSet implements Resolvable super(); } - @SuppressWarnings("unchecked") public SqlDataSet(String id, String name, ConnectionFactory connectionFactory, String sql) { - super(id, name, Collections.EMPTY_LIST); + super(id, name); this.connectionFactory = connectionFactory; this.sql = sql; } @@ -84,35 +90,16 @@ public class SqlDataSet extends AbstractFmkTemplateDataSet implements Resolvable } @Override - public DataSetResult getResult(Map paramValues) throws DataSetException + public TemplateResolvedDataSetResult resolve(Map paramValues) throws DataSetException { - List properties = getProperties(); - - if (properties == null || properties.isEmpty()) - throw new DataSetException("[getProperties()] must not be empty"); - - ResolvedDataSetResult result = getResolvedDataSetResult(paramValues, properties); - return result.getResult(); + return resolveResult(paramValues, null); } @Override - public TemplateResolvedDataSetResult resolve(Map paramValues) throws DataSetException + protected TemplateResolvedDataSetResult resolveResult(Map paramValues, List properties) + throws DataSetException { - return getResolvedDataSetResult(paramValues, null); - } - - /** - * - * @param paramValues - * @param properties - * 允许为{@code null},此时会自动解析 - * @return - * @throws DataSetException - */ - protected TemplateResolvedDataSetResult getResolvedDataSetResult(Map paramValues, - List properties) throws DataSetException - { - String sql = resolveTemplate(getSql(), paramValues); + String sql = resolveAsFmkTemplateIfHasParam(getSql(), paramValues); Connection cn = null; @@ -120,32 +107,42 @@ public class SqlDataSet extends AbstractFmkTemplateDataSet implements Resolvable { cn = getConnectionFactory().get(); } - catch (Exception e) + catch (Throwable t) { JdbcUtil.closeConnection(cn); - throw new SqlDataSetConnectionException(e); + throw new SqlDataSetConnectionException(t); } Sql sqlObj = Sql.valueOf(sql); + JdbcSupport jdbcSupport = getJdbcSupport(); + QueryResultSet qrs = null; try { - qrs = getSqlDataSetSupport().executeQuery(cn, sqlObj, ResultSet.TYPE_FORWARD_ONLY); + qrs = jdbcSupport.executeQuery(cn, sqlObj, ResultSet.TYPE_FORWARD_ONLY); + } + catch (Throwable t) + { + throw new SqlDataSetSqlExecutionException(sql, t); + } + + try + { ResultSet rs = qrs.getResultSet(); - if (properties == null || properties.isEmpty()) - properties = getSqlDataSetSupport().resolveDataSetProperties(cn, rs, null); + ResolvedDataSetResult result = resolveResult(cn, rs, properties); - List> data = getSqlDataSetSupport().resolveResultData(cn, rs, properties); - DataSetResult result = new DataSetResult(data); - - return new TemplateResolvedDataSetResult(result, properties, sql); + return new TemplateResolvedDataSetResult(result.getResult(), result.getProperties(), sql); } - catch (SQLException e) + catch (DataSetException e) { - throw new SqlDataSetSqlExecutionException(sql, e); + throw e; + } + catch (Throwable t) + { + throw new DataSetException(t); } finally { @@ -167,46 +164,163 @@ public class SqlDataSet extends AbstractFmkTemplateDataSet implements Resolvable } /** + * 解析结果。 * * @param cn - * @param sql + * @param rs * @param properties * 允许为{@code null},此时会自动解析 * @return - * @throws DataSetException + * @throws Throwable */ - protected ResolvedDataSetResult getResolvedDataSetResult(Connection cn, String sql, - List properties) throws DataSetException + protected ResolvedDataSetResult resolveResult(Connection cn, ResultSet rs, List properties) + throws Throwable { - Sql sqlObj = Sql.valueOf(sql); + boolean resolveProperties = (properties == null || properties.isEmpty()); - QueryResultSet qrs = null; + List> datas = new ArrayList<>(); - try + JdbcSupport jdbcSupport = getJdbcSupport(); + DataSetPropertyValueConverter converter = createDataSetPropertyValueConverter(); + + ResultSetMetaData rsMeta = rs.getMetaData(); + String[] colNames = jdbcSupport.getColumnNames(rsMeta); + SqlType[] sqlTypes = jdbcSupport.getColumnSqlTypes(rsMeta); + + if (resolveProperties) { - qrs = getSqlDataSetSupport().executeQuery(cn, sqlObj, ResultSet.TYPE_FORWARD_ONLY); - ResultSet rs = qrs.getResultSet(); - - if (properties == null || properties.isEmpty()) - properties = getSqlDataSetSupport().resolveDataSetProperties(cn, rs, null); - - List> data = getSqlDataSetSupport().resolveResultData(cn, rs, properties); - DataSetResult result = new DataSetResult(data); - - return new ResolvedDataSetResult(result, properties); + properties = new ArrayList<>(colNames.length); + for (int i = 0; i < colNames.length; i++) + properties.add(new DataSetProperty(colNames[i], toPropertyDataType(sqlTypes[i], colNames[i]))); } - catch (SQLException e) + + int maxColumnSize = Math.min(colNames.length, properties.size()); + + int rowIdx = 0; + + while (rs.next()) { - throw new SqlDataSetSqlExecutionException(sql, e); - } - finally - { - QueryResultSet.close(qrs); + Map row = new HashMap<>(); + + for (int i = 0; i < maxColumnSize; i++) + { + DataSetProperty property = properties.get(i); + + Object value = jdbcSupport.getColumnValue(cn, rs, colNames[i], sqlTypes[i].getType()); + + if (resolveProperties && rowIdx == 0) + { + @JDBCCompatiblity("某些驱动程序可能存在一种情况,列类型会被toPropertyDataType()解析为DataType.UNKNOWN,但是实际值是允许的," + + "比如PostgreSQL-42.2.5驱动对于[SELECT 'aaa' as NAME]语句,结果的SQL类型是Types.OTHER,但实际值是允许的字符串") + boolean resolveTypeByValue = (DataType.UNKNOWN.equals(property.getType())); + + if (resolveTypeByValue) + property.setType(resolvePropertyDataType(value)); + } + + value = convertToPropertyDataType(converter, value, property); + + row.put(property.getName(), value); + } + + datas.add(row); + + rowIdx++; } + + DataSetResult result = new DataSetResult(datas); + + return new ResolvedDataSetResult(result, properties); } - protected SqlDataSetSupport getSqlDataSetSupport() + /** + * 由SQL类型转换为{@linkplain DataSetProperty#getType()}。 + * + * @param sqlType + * @param columnName + * 允许为{@code null},列名称 + * @return + * @throws SQLException + * @throws SqlDataSetUnsupportedSqlTypeException + */ + protected String toPropertyDataType(SqlType sqlType, String columnName) + throws SQLException, SqlDataSetUnsupportedSqlTypeException { - return SQL_DATA_SET_SUPPORT; + String dataType = null; + + int type = sqlType.getType(); + + switch (type) + { + // 确定不支持的类型 + case Types.BINARY: + case Types.BLOB: + case Types.LONGVARBINARY: + case Types.VARBINARY: + throw new SqlDataSetUnsupportedSqlTypeException(sqlType, columnName); + + case Types.CHAR: + case Types.NCHAR: + case Types.NVARCHAR: + case Types.VARCHAR: + { + dataType = DataType.STRING; + break; + } + + case Types.BOOLEAN: + { + dataType = DataType.BOOLEAN; + break; + } + + case Types.BIGINT: + case Types.BIT: + case Types.INTEGER: + case Types.SMALLINT: + case Types.TINYINT: + { + dataType = DataType.INTEGER; + break; + } + + case Types.DECIMAL: + case Types.DOUBLE: + case Types.FLOAT: + case Types.NUMERIC: + case Types.REAL: + { + dataType = DataType.DECIMAL; + break; + } + + case Types.DATE: + { + dataType = DataType.DATE; + break; + } + + case Types.TIME: + { + dataType = DataType.TIME; + break; + } + + case Types.TIMESTAMP: + { + dataType = DataType.TIMESTAMP; + break; + } + + default: + dataType = DataType.UNKNOWN; + } + + return dataType; + } + + protected JdbcSupport getJdbcSupport() + { + return JDBC_SUPPORT; } } diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/SqlDataSetSupport.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/SqlDataSetSupport.java deleted file mode 100644 index 456e731e..00000000 --- a/datagear-analysis/src/main/java/org/datagear/analysis/support/SqlDataSetSupport.java +++ /dev/null @@ -1,433 +0,0 @@ -/* - * Copyright (c) 2018 datagear.tech. All Rights Reserved. - */ - -/** - * - */ -package org.datagear.analysis.support; - -import java.sql.Connection; -import java.sql.ResultSet; -import java.sql.ResultSetMetaData; -import java.sql.SQLException; -import java.sql.Types; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import org.datagear.analysis.DataSetException; -import org.datagear.analysis.DataSetProperty; -import org.datagear.analysis.DataSetProperty.DataType; -import org.datagear.util.JdbcSupport; -import org.datagear.util.SqlType; - -/** - * SQL数据集支持类。 - * - * @author datagear@163.com - * - */ -public class SqlDataSetSupport extends JdbcSupport -{ - public SqlDataSetSupport() - { - super(); - } - - /** - * 解析结果数据。 - * - * @param cn - * @param rs - * @param properties - * @return - * @throws SQLException - */ - public List> resolveResultData(Connection cn, ResultSet rs, List properties) - throws SQLException - { - List> datas = new ArrayList<>(); - - ResultSetMetaData rsMeta = rs.getMetaData(); - int[] rsColumns = resolveResultsetColumns(properties, rsMeta); - - while (rs.next()) - { - Map row = new HashMap<>(); - - for (int i = 0; i < rsColumns.length; i++) - { - DataSetProperty property = properties.get(i); - int rsColumn = rsColumns[i]; - - Object value = resolvePropertyDataValue(cn, rs, rsColumn, getColumnSqlType(rsMeta, rsColumn), - property.getType()); - - row.put(property.getName(), value); - } - - datas.add(row); - } - - return datas; - } - - /** - * 解析结果集中对应{@linkplain DataSetProperty}的索引数组。 - * - * @param properties - * @param rsMeta - * @return - * @throws SQLException - * @throws DataSetException - */ - public int[] resolveResultsetColumns(List properties, ResultSetMetaData rsMeta) - throws SQLException, DataSetException - { - int[] columns = new int[properties.size()]; - - int rsColumnCount = rsMeta.getColumnCount(); - - for (int i = 0; i < columns.length; i++) - { - String pname = properties.get(i).getName(); - - int myIndex = -1; - - for (int j = 1; j <= rsColumnCount; j++) - { - if (pname.equalsIgnoreCase(getColumnName(rsMeta, j))) - { - myIndex = j; - break; - } - } - - if (myIndex <= 0) - throw new DataSetException( - "Column named '" + pname + "' not found in the " + ResultSet.class.getSimpleName()); - - columns[i] = myIndex; - } - - return columns; - } - - /** - * 解析数据值。 - * - * @param cn - * @param rs - * @param column - * @param sqlType - * @param dataType - * @return - * @throws SQLException - */ - public Object resolvePropertyDataValue(Connection cn, ResultSet rs, int column, SqlType sqlType, String dataType) - throws SQLException - { - Object value = null; - - int type = sqlType.getType(); - - if (DataType.isString(dataType)) - { - switch (type) - { - case Types.CHAR: - case Types.NCHAR: - case Types.NVARCHAR: - case Types.VARCHAR: - { - value = rs.getString(column); - break; - } - - default: - throw new SqlDataSetUnsupportedSqlTypeException(sqlType); - } - } - else if (DataType.isBoolean(dataType)) - { - switch (type) - { - case Types.BIT: - case Types.BIGINT: - case Types.INTEGER: - case Types.SMALLINT: - case Types.TINYINT: - { - value = (rs.getInt(column) > 0); - break; - } - - case Types.BOOLEAN: - { - value = rs.getBoolean(column); - break; - } - - default: - throw new SqlDataSetUnsupportedSqlTypeException(sqlType); - } - } - else if (DataType.isInteger(dataType)) - { - switch (type) - { - case Types.BIGINT: - { - value = rs.getLong(column); - break; - } - - case Types.BIT: - case Types.INTEGER: - case Types.SMALLINT: - case Types.TINYINT: - { - value = rs.getInt(column); - break; - } - - case Types.DECIMAL: - case Types.NUMERIC: - { - value = rs.getBigDecimal(column).toBigInteger(); - break; - } - - case Types.DOUBLE: - { - value = new Double(rs.getDouble(column)).longValue(); - break; - } - - case Types.FLOAT: - case Types.REAL: - { - value = new Float(rs.getFloat(column)).longValue(); - break; - } - - default: - throw new SqlDataSetUnsupportedSqlTypeException(sqlType); - } - } - else if (DataType.isDecimal(dataType)) - { - switch (type) - { - case Types.BIGINT: - { - value = rs.getLong(column); - break; - } - - case Types.BIT: - case Types.INTEGER: - case Types.SMALLINT: - case Types.TINYINT: - { - value = rs.getInt(column); - break; - } - - case Types.DECIMAL: - case Types.NUMERIC: - { - value = rs.getBigDecimal(column); - break; - } - - case Types.DOUBLE: - { - value = rs.getDouble(column); - break; - } - - case Types.FLOAT: - case Types.REAL: - { - value = rs.getFloat(column); - break; - } - - default: - throw new SqlDataSetUnsupportedSqlTypeException(sqlType); - } - } - else if (DataType.isDate(dataType)) - { - switch (type) - { - case Types.DATE: - { - value = rs.getDate(column); - break; - } - - default: - throw new SqlDataSetUnsupportedSqlTypeException(sqlType); - } - } - else if (DataType.isTime(dataType)) - { - switch (type) - { - case Types.TIME: - { - value = rs.getTime(column); - break; - } - - default: - throw new SqlDataSetUnsupportedSqlTypeException(sqlType); - } - } - else if (DataType.isTimestamp(dataType)) - { - switch (type) - { - case Types.TIMESTAMP: - { - value = rs.getTimestamp(column); - break; - } - - default: - throw new SqlDataSetUnsupportedSqlTypeException(sqlType); - } - } - else - throw new UnsupportedOperationException(); - - if (rs.wasNull()) - value = null; - - return value; - } - - /** - * 解析{@linkplain DataSetProperty}列表。 - * - * @param cn - * @param rs - * @param labels - * {@linkplain DataSetProperty#getLabel()}数组,允许为{@code null}或任意长度的数组 - * @return - * @throws SQLException - */ - public List resolveDataSetProperties(Connection cn, ResultSet rs, String[] labels) - throws SQLException - { - ResultSetMetaData metaData = rs.getMetaData(); - int columnCount = metaData.getColumnCount(); - - List properties = new ArrayList<>(columnCount); - - for (int i = 1; i <= columnCount; i++) - { - String columnName = getColumnName(metaData, i); - SqlType sqlType = getColumnSqlType(metaData, i); - - String dataType = toPropertyDataType(sqlType, columnName); - - DataSetProperty property = createDataSetProperty(); - property.setName(columnName); - property.setType(dataType); - - if (labels != null && labels.length > i - 1) - property.setLabel(labels[i - 1]); - - properties.add(property); - } - - return properties; - } - - /** - * 由SQL类型转换为{@linkplain DataSetProperty#getType()}。 - * - * @param sqlType - * @param columnName - * 允许为{@code null},列名称 - * @return - * @throws SQLException - */ - public String toPropertyDataType(SqlType sqlType, String columnName) throws SQLException - { - String dataType = null; - - int type = sqlType.getType(); - - switch (type) - { - case Types.CHAR: - case Types.NCHAR: - case Types.NVARCHAR: - case Types.VARCHAR: - { - dataType = DataType.STRING; - break; - } - - case Types.BOOLEAN: - { - dataType = DataType.BOOLEAN; - break; - } - - case Types.BIGINT: - case Types.BIT: - case Types.INTEGER: - case Types.SMALLINT: - case Types.TINYINT: - { - dataType = DataType.INTEGER; - break; - } - - case Types.DECIMAL: - case Types.DOUBLE: - case Types.FLOAT: - case Types.NUMERIC: - case Types.REAL: - { - dataType = DataType.DECIMAL; - break; - } - - case Types.DATE: - { - dataType = DataType.DATE; - break; - } - - case Types.TIME: - { - dataType = DataType.TIME; - break; - } - - case Types.TIMESTAMP: - { - dataType = DataType.TIMESTAMP; - break; - } - - default: - throw new SqlDataSetUnsupportedSqlTypeException(sqlType, columnName); - } - - return dataType; - } - - protected DataSetProperty createDataSetProperty() - { - return new DataSetProperty(); - } -} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/TemplateResolvedSource.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/TemplateResolvedSource.java new file mode 100644 index 00000000..53589f53 --- /dev/null +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/TemplateResolvedSource.java @@ -0,0 +1,62 @@ +/* + * Copyright 2018 datagear.tech. All Rights Reserved. + */ + +package org.datagear.analysis.support; + +/** + * 模板已解析的源。 + * + * @author datagear@163.com + * + */ +public class TemplateResolvedSource +{ + private T source; + + /** 已解析的模板内容 */ + private String resolvedTemplate = null; + + public TemplateResolvedSource() + { + super(); + } + + public TemplateResolvedSource(T source) + { + super(); + this.source = source; + } + + public TemplateResolvedSource(T source, String resolvedTemplate) + { + super(); + this.source = source; + this.resolvedTemplate = resolvedTemplate; + } + + public T getSource() + { + return source; + } + + public void setSource(T source) + { + this.source = source; + } + + public boolean hasResolvedTemplate() + { + return (this.resolvedTemplate != null && !this.resolvedTemplate.isEmpty()); + } + + public String getResolvedTemplate() + { + return resolvedTemplate; + } + + public void setResolvedTemplate(String resolvedTemplate) + { + this.resolvedTemplate = resolvedTemplate; + } +} diff --git a/datagear-analysis/src/main/java/org/datagear/analysis/support/html/HtmlChartPluginLoader.java b/datagear-analysis/src/main/java/org/datagear/analysis/support/html/HtmlChartPluginLoader.java index a3297310..2fac4c2b 100644 --- a/datagear-analysis/src/main/java/org/datagear/analysis/support/html/HtmlChartPluginLoader.java +++ b/datagear-analysis/src/main/java/org/datagear/analysis/support/html/HtmlChartPluginLoader.java @@ -81,7 +81,7 @@ public class HtmlChartPluginLoader private JsonChartPluginPropertiesResolver jsonChartPluginPropertiesResolver = new JsonChartPluginPropertiesResolver(); /** 文件编码 */ - private String encoding = "UTF-8"; + private String encoding = IOUtil.CHARSET_UTF_8; public HtmlChartPluginLoader() { diff --git a/datagear-analysis/src/test/java/org/datagear/analysis/support/AbstractJsonDataSetTest.java b/datagear-analysis/src/test/java/org/datagear/analysis/support/AbstractJsonDataSetTest.java new file mode 100644 index 00000000..b7ee6ba2 --- /dev/null +++ b/datagear-analysis/src/test/java/org/datagear/analysis/support/AbstractJsonDataSetTest.java @@ -0,0 +1,78 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.analysis.support; + +import static org.junit.Assert.assertEquals; + +import java.util.Collections; +import java.util.List; +import java.util.Map; + +import org.datagear.analysis.DataSetProperty; +import org.junit.Test; + +/** + * {@linkplain AbstractJsonDataSet}单元测试用例。 + * + * @author datagear@163.com + * + */ +public class AbstractJsonDataSetTest +{ + @Test + public void resolveTest_dataJsonPath() + { + String jsonString = "{ path0: { path1: [ { path2: [ { name:'aaa', value: 11, size: 12 } ] } ] } }"; + + JsonValueDataSet dataSet = new JsonValueDataSet(JsonValueDataSet.class.getSimpleName(), + JsonValueDataSet.class.getSimpleName(), jsonString); + + dataSet.setDataJsonPath("path0.path1[0].path2"); + + TemplateResolvedDataSetResult result = dataSet.resolve(Collections.emptyMap()); + List properties = result.getProperties(); + @SuppressWarnings("unchecked") + List> data = (List>) result.getResult().getData(); + + assertEquals(jsonString, result.getTemplateResult()); + + { + assertEquals(3, properties.size()); + + { + DataSetProperty property = properties.get(0); + assertEquals("name", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + + { + DataSetProperty property = properties.get(1); + assertEquals("value", property.getName()); + assertEquals(DataSetProperty.DataType.NUMBER, property.getType()); + } + + { + DataSetProperty property = properties.get(2); + assertEquals("size", property.getName()); + assertEquals(DataSetProperty.DataType.NUMBER, property.getType()); + } + } + + { + assertEquals(1, data.size()); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(11, ((Number) row.get("value")).intValue()); + assertEquals(12, ((Number) row.get("size")).intValue()); + } + } + } +} diff --git a/datagear-analysis/src/test/java/org/datagear/analysis/support/CsvDirectoryFileDataSetTest.java b/datagear-analysis/src/test/java/org/datagear/analysis/support/CsvDirectoryFileDataSetTest.java new file mode 100644 index 00000000..e0623551 --- /dev/null +++ b/datagear-analysis/src/test/java/org/datagear/analysis/support/CsvDirectoryFileDataSetTest.java @@ -0,0 +1,210 @@ +/* + * Copyright 2018 datagear.tech. All Rights Reserved. + */ + +package org.datagear.analysis.support; + +import static org.junit.Assert.assertEquals; + +import java.io.File; +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; +import java.util.Map; + +import org.datagear.analysis.DataSetProperty; +import org.datagear.analysis.DataSetResult; +import org.datagear.analysis.ResolvedDataSetResult; +import org.junit.Test; + +/** + * {@linkplain CsvDirectoryFileDataSet}单元测试类。 + * + * @author datagear@163.com + * + */ +public class CsvDirectoryFileDataSetTest +{ + private static final File DIRECTORY = new File("src/test/resources/org/datagear/analysis/support/"); + + @Test + public void getResultTest() + { + List properties = new ArrayList<>(); + properties.add(new DataSetProperty("name", DataSetProperty.DataType.STRING)); + properties.add(new DataSetProperty("value", DataSetProperty.DataType.NUMBER)); + properties.add(new DataSetProperty("尺寸", DataSetProperty.DataType.NUMBER)); + + CsvDirectoryFileDataSet dataSet = new CsvDirectoryFileDataSet("a", "a", properties, DIRECTORY, + "CsvDirectoryFileDataSetTest-0.csv"); + dataSet.setNameRow(1); + + @SuppressWarnings("unchecked") + DataSetResult result = dataSet.getResult(Collections.EMPTY_MAP); + @SuppressWarnings("unchecked") + List> data = (List>) result.getData(); + + { + assertEquals(3, data.size()); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(11, ((Number) row.get("value")).intValue()); + assertEquals(12, ((Number) row.get("尺寸")).intValue()); + } + + { + Map row = data.get(1); + + assertEquals("bbb", row.get("name")); + assertEquals(21, ((Number) row.get("value")).intValue()); + assertEquals(22, ((Number) row.get("尺寸")).intValue()); + } + + { + Map row = data.get(2); + + assertEquals("ccc", row.get("name")); + assertEquals(31, ((Number) row.get("value")).intValue()); + assertEquals(32, ((Number) row.get("尺寸")).intValue()); + } + } + } + + @Test + public void resolveTest() + { + CsvDirectoryFileDataSet dataSet = new CsvDirectoryFileDataSet("a", "a", DIRECTORY, + "CsvDirectoryFileDataSetTest-0.csv"); + dataSet.setNameRow(1); + + @SuppressWarnings("unchecked") + ResolvedDataSetResult result = dataSet.resolve(Collections.EMPTY_MAP); + List properties = result.getProperties(); + @SuppressWarnings("unchecked") + List> data = (List>) result.getResult().getData(); + + { + assertEquals(3, properties.size()); + + { + DataSetProperty property = properties.get(0); + assertEquals("name", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + + { + DataSetProperty property = properties.get(1); + assertEquals("value", property.getName()); + assertEquals(DataSetProperty.DataType.NUMBER, property.getType()); + } + + { + DataSetProperty property = properties.get(2); + assertEquals("尺寸", property.getName()); + assertEquals(DataSetProperty.DataType.NUMBER, property.getType()); + } + } + + { + assertEquals(3, data.size()); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(11, ((Number) row.get("value")).intValue()); + assertEquals(12, ((Number) row.get("尺寸")).intValue()); + } + + { + Map row = data.get(1); + + assertEquals("bbb", row.get("name")); + assertEquals(21, ((Number) row.get("value")).intValue()); + assertEquals(22, ((Number) row.get("尺寸")).intValue()); + } + + { + Map row = data.get(2); + + assertEquals("ccc", row.get("name")); + assertEquals(31, ((Number) row.get("value")).intValue()); + assertEquals(32, ((Number) row.get("尺寸")).intValue()); + } + } + } + + @Test + public void resolveTest_noNameRow() + { + CsvDirectoryFileDataSet dataSet = new CsvDirectoryFileDataSet("a", "a", DIRECTORY, + "CsvDirectoryFileDataSetTest-0.csv"); + + @SuppressWarnings("unchecked") + ResolvedDataSetResult result = dataSet.resolve(Collections.EMPTY_MAP); + List properties = result.getProperties(); + @SuppressWarnings("unchecked") + List> data = (List>) result.getResult().getData(); + + { + assertEquals(3, properties.size()); + + { + DataSetProperty property = properties.get(0); + assertEquals("1", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + + { + DataSetProperty property = properties.get(1); + assertEquals("2", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + + { + DataSetProperty property = properties.get(2); + assertEquals("3", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + } + + { + assertEquals(4, data.size()); + + { + Map row = data.get(0); + + assertEquals("name", row.get("1")); + assertEquals("value", row.get("2")); + assertEquals("尺寸", row.get("3")); + } + + { + Map row = data.get(1); + + assertEquals("aaa", row.get("1")); + assertEquals("11", row.get("2")); + assertEquals("12", row.get("3")); + } + + { + Map row = data.get(2); + + assertEquals("bbb", row.get("1")); + assertEquals("21", row.get("2")); + assertEquals("22", row.get("3")); + } + + { + Map row = data.get(3); + + assertEquals("ccc", row.get("1")); + assertEquals("31", row.get("2")); + assertEquals("32", row.get("3")); + } + } + } +} diff --git a/datagear-analysis/src/test/java/org/datagear/analysis/support/CsvValueDataSetTest.java b/datagear-analysis/src/test/java/org/datagear/analysis/support/CsvValueDataSetTest.java new file mode 100644 index 00000000..80b4ec34 --- /dev/null +++ b/datagear-analysis/src/test/java/org/datagear/analysis/support/CsvValueDataSetTest.java @@ -0,0 +1,154 @@ +/* + * Copyright 2018 datagear.tech. All Rights Reserved. + */ + +package org.datagear.analysis.support; + +import static org.junit.Assert.assertEquals; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import org.datagear.analysis.DataSetParam; +import org.datagear.analysis.DataSetProperty; +import org.datagear.analysis.DataSetResult; +import org.junit.Test; + +/** + * {@linkplain CsvValueDataSet}单元测试类。 + * + * @author datagear@163.com + * + */ +public class CsvValueDataSetTest +{ + @Test + public void getResultTest_hasParam() + { + List properties = new ArrayList<>(); + properties.add(new DataSetProperty("name", DataSetProperty.DataType.STRING)); + properties.add(new DataSetProperty("value", DataSetProperty.DataType.NUMBER)); + properties.add(new DataSetProperty("size", DataSetProperty.DataType.NUMBER)); + + List params = new ArrayList(); + params.add(new DataSetParam("size", DataSetParam.DataType.NUMBER, true)); + + CsvValueDataSet dataSet = new CsvValueDataSet("a", "a", properties, + "name, value, size \n aaa, 11, ${size}"); + dataSet.setParams(params); + dataSet.setNameRow(1); + + Map paramValues = new HashMap<>(); + paramValues.put("size", 12); + + DataSetResult result = dataSet.getResult(paramValues); + @SuppressWarnings("unchecked") + List> data = (List>) result.getData(); + + { + assertEquals(1, data.size()); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(11, ((Number) row.get("value")).intValue()); + assertEquals(12, ((Number) row.get("size")).intValue()); + } + } + } + + @Test + public void getResultTest_hasParam_convertPropertyValue() + { + List properties = new ArrayList<>(); + properties.add(new DataSetProperty("name", DataSetProperty.DataType.STRING)); + properties.add(new DataSetProperty("value", DataSetProperty.DataType.NUMBER)); + properties.add(new DataSetProperty("size", DataSetProperty.DataType.STRING)); + + List params = new ArrayList(); + params.add(new DataSetParam("size", DataSetParam.DataType.NUMBER, true)); + + CsvValueDataSet dataSet = new CsvValueDataSet("a", "a", properties, + "name, value, size \n aaa, 11, ${size}"); + dataSet.setParams(params); + dataSet.setNameRow(1); + + Map paramValues = new HashMap<>(); + paramValues.put("size", 12); + + DataSetResult result = dataSet.getResult(paramValues); + @SuppressWarnings("unchecked") + List> data = (List>) result.getData(); + + { + assertEquals(1, data.size()); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(11, ((Number) row.get("value")).intValue()); + assertEquals("12", row.get("size")); + } + } + } + + @Test + public void resolveTest_hasParam() + { + List params = new ArrayList(); + params.add(new DataSetParam("size", DataSetParam.DataType.NUMBER, true)); + + CsvValueDataSet dataSet = new CsvValueDataSet("a", "a", + "name, value, size \n aaa, 11, ${size}"); + dataSet.setParams(params); + dataSet.setNameRow(1); + + Map paramValues = new HashMap<>(); + paramValues.put("size", 12); + + TemplateResolvedDataSetResult result = dataSet.resolve(paramValues); + List properties = result.getProperties(); + @SuppressWarnings("unchecked") + List> data = (List>) result.getResult().getData(); + + assertEquals("name, value, size \n aaa, 11, 12", result.getTemplateResult()); + + { + assertEquals(3, properties.size()); + + { + DataSetProperty property = properties.get(0); + assertEquals("name", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + + { + DataSetProperty property = properties.get(1); + assertEquals("value", property.getName()); + assertEquals(DataSetProperty.DataType.NUMBER, property.getType()); + } + + { + DataSetProperty property = properties.get(2); + assertEquals("size", property.getName()); + assertEquals(DataSetProperty.DataType.NUMBER, property.getType()); + } + } + + { + assertEquals(1, data.size()); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(11, ((Number) row.get("value")).intValue()); + assertEquals(12, ((Number) row.get("size")).intValue()); + } + } + } +} diff --git a/datagear-analysis/src/test/java/org/datagear/analysis/support/DBTestSupport.java b/datagear-analysis/src/test/java/org/datagear/analysis/support/DBTestSupport.java deleted file mode 100644 index a61f8c1e..00000000 --- a/datagear-analysis/src/test/java/org/datagear/analysis/support/DBTestSupport.java +++ /dev/null @@ -1,74 +0,0 @@ -/* - * Copyright (c) 2018 datagear.org. All Rights Reserved. - */ - -package org.datagear.analysis.support; - -import java.io.File; -import java.io.FileReader; -import java.io.Reader; -import java.sql.Connection; -import java.sql.DriverManager; -import java.sql.SQLException; -import java.util.Properties; - -/** - * 数据库测试支持类。 - * - * @author datagear@163.com - * - */ -public abstract class DBTestSupport -{ - private static final Properties JDBC_PROPERTIES = new Properties(); - - static - { - File jdbcConfigFile = new File("test/config/jdbc.properties"); - if (!jdbcConfigFile.exists()) - jdbcConfigFile = new File("../test/config/jdbc.properties"); - - try - { - Reader reader = new FileReader(jdbcConfigFile); - JDBC_PROPERTIES.load(reader); - reader.close(); - } - catch (Exception e) - { - if (e instanceof RuntimeException) - throw (RuntimeException) e; - else - throw new RuntimeException(e); - } - } - - protected Connection getConnection() throws SQLException - { - return DriverManager.getConnection(JDBC_PROPERTIES.getProperty("jdbc.url"), - JDBC_PROPERTIES.getProperty("jdbc.user"), JDBC_PROPERTIES.getProperty("jdbc.password")); - } - - protected Connection getConnection(Properties properties) throws Exception - { - properties.setProperty("user", JDBC_PROPERTIES.getProperty("jdbc.user")); - properties.setProperty("password", JDBC_PROPERTIES.getProperty("jdbc.password")); - - return DriverManager.getConnection(JDBC_PROPERTIES.getProperty("jdbc.url"), properties); - } - - protected void println() - { - System.out.println(); - } - - protected void println(Object o) - { - System.out.println((o == null ? "null" : o.toString())); - } - - protected void print(Object o) - { - System.out.print((o == null ? "null" : o.toString())); - } -} diff --git a/datagear-analysis/src/test/java/org/datagear/analysis/support/ExcelDirectoryFileDataSetTest.java b/datagear-analysis/src/test/java/org/datagear/analysis/support/ExcelDirectoryFileDataSetTest.java new file mode 100644 index 00000000..fc331f7a --- /dev/null +++ b/datagear-analysis/src/test/java/org/datagear/analysis/support/ExcelDirectoryFileDataSetTest.java @@ -0,0 +1,355 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.analysis.support; + +import static org.junit.Assert.assertEquals; + +import java.io.File; +import java.text.SimpleDateFormat; +import java.util.ArrayList; +import java.util.Collections; +import java.util.Date; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import org.datagear.analysis.DataSetProperty; +import org.datagear.analysis.DataSetResult; +import org.datagear.analysis.ResolvedDataSetResult; +import org.junit.Test; + +/** + * {@linkplain ExcelDirectoryFileDataSet}单元测试用例。 + * + * @author datagear@163.com + * + */ +public class ExcelDirectoryFileDataSetTest +{ + private static final File DIRECTORY = new File("src/test/resources/org/datagear/analysis/support/"); + + @Test + public void getResultTest_xlsx() + { + List properties = new ArrayList<>(); + properties.add(new DataSetProperty("name", DataSetProperty.DataType.STRING)); + properties.add(new DataSetProperty("value", DataSetProperty.DataType.NUMBER)); + properties.add(new DataSetProperty("size", DataSetProperty.DataType.NUMBER)); + properties.add(new DataSetProperty("date", DataSetProperty.DataType.DATE)); + + ExcelDirectoryFileDataSet dataSet = new ExcelDirectoryFileDataSet("a", "a", properties, DIRECTORY, + "ExcelDirectoryFileDataSetTest-0.xlsx"); + dataSet.setNameRow(1); + + @SuppressWarnings("unchecked") + DataSetResult result = dataSet.getResult(Collections.EMPTY_MAP); + @SuppressWarnings("unchecked") + List> data = (List>) result.getData(); + + { + assertEquals(3, data.size()); + + SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd"); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(15, ((Number) row.get("value")).intValue()); + assertEquals(16, ((Number) row.get("size")).intValue()); + assertEquals("2020-08-01", dateFormat.format((Date) row.get("date"))); + } + + { + Map row = data.get(1); + + assertEquals("bbb", row.get("name")); + assertEquals(25, ((Number) row.get("value")).intValue()); + assertEquals(26, ((Number) row.get("size")).intValue()); + assertEquals("2020-08-02", dateFormat.format((Date) row.get("date"))); + } + + { + Map row = data.get(2); + + assertEquals("ccc", row.get("name")); + assertEquals(35, ((Number) row.get("value")).intValue()); + assertEquals(36, ((Number) row.get("size")).intValue()); + assertEquals("2020-08-03", dateFormat.format((Date) row.get("date"))); + } + } + } + + @Test + public void getResultTest_xlsx_convertPropertyValue() + { + List properties = new ArrayList<>(); + properties.add(new DataSetProperty("name", DataSetProperty.DataType.STRING)); + properties.add(new DataSetProperty("value", DataSetProperty.DataType.NUMBER)); + properties.add(new DataSetProperty("size", DataSetProperty.DataType.STRING)); + properties.add(new DataSetProperty("date", DataSetProperty.DataType.STRING)); + + ExcelDirectoryFileDataSet dataSet = new ExcelDirectoryFileDataSet("a", "a", properties, DIRECTORY, + "ExcelDirectoryFileDataSetTest-0.xlsx"); + dataSet.setNameRow(1); + + @SuppressWarnings("unchecked") + DataSetResult result = dataSet.getResult(Collections.EMPTY_MAP); + @SuppressWarnings("unchecked") + List> data = (List>) result.getData(); + + { + assertEquals(3, data.size()); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(15, ((Number) row.get("value")).intValue()); + assertEquals("16", row.get("size")); + assertEquals("2020-08-01", row.get("date")); + } + + { + Map row = data.get(1); + + assertEquals("bbb", row.get("name")); + assertEquals(25, ((Number) row.get("value")).intValue()); + assertEquals("26", row.get("size")); + assertEquals("2020-08-02", row.get("date")); + } + + { + Map row = data.get(2); + + assertEquals("ccc", row.get("name")); + assertEquals(35, ((Number) row.get("value")).intValue()); + assertEquals("36", row.get("size")); + assertEquals("2020-08-03", row.get("date")); + } + } + } + + @Test + public void resolveTest_xlsx() + { + ExcelDirectoryFileDataSet dataSet = new ExcelDirectoryFileDataSet("a", "a", DIRECTORY, + "ExcelDirectoryFileDataSetTest-0.xlsx"); + dataSet.setNameRow(1); + + ResolvedDataSetResult resolvedResult = dataSet.resolve(new HashMap<>()); + + @SuppressWarnings("unchecked") + List> data = (List>) resolvedResult.getResult().getData(); + List properties = resolvedResult.getProperties(); + + { + assertEquals(4, properties.size()); + + { + DataSetProperty property = properties.get(0); + assertEquals("name", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + + { + DataSetProperty property = properties.get(1); + assertEquals("value", property.getName()); + assertEquals(DataSetProperty.DataType.DECIMAL, property.getType()); + } + + { + DataSetProperty property = properties.get(2); + assertEquals("size", property.getName()); + assertEquals(DataSetProperty.DataType.DECIMAL, property.getType()); + } + + { + DataSetProperty property = properties.get(3); + assertEquals("date", property.getName()); + assertEquals(DataSetProperty.DataType.DATE, property.getType()); + } + } + + { + assertEquals(3, data.size()); + + SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd"); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(15, ((Number) row.get("value")).intValue()); + assertEquals(16, ((Number) row.get("size")).intValue()); + assertEquals("2020-08-01", dateFormat.format((Date) row.get("date"))); + } + + { + Map row = data.get(1); + + assertEquals("bbb", row.get("name")); + assertEquals(25, ((Number) row.get("value")).intValue()); + assertEquals(26, ((Number) row.get("size")).intValue()); + assertEquals("2020-08-02", dateFormat.format((Date) row.get("date"))); + } + + { + Map row = data.get(2); + + assertEquals("ccc", row.get("name")); + assertEquals(35, ((Number) row.get("value")).intValue()); + assertEquals(36, ((Number) row.get("size")).intValue()); + assertEquals("2020-08-03", dateFormat.format((Date) row.get("date"))); + } + } + } + + @Test + public void resolveTest_xls() + { + ExcelDirectoryFileDataSet dataSet = new ExcelDirectoryFileDataSet("a", "a", DIRECTORY, + "ExcelDirectoryFileDataSetTest-1.xls"); + dataSet.setNameRow(1); + + ResolvedDataSetResult resolvedResult = dataSet.resolve(new HashMap<>()); + + @SuppressWarnings("unchecked") + List> data = (List>) resolvedResult.getResult().getData(); + List properties = resolvedResult.getProperties(); + + { + assertEquals(4, properties.size()); + + { + DataSetProperty property = properties.get(0); + assertEquals("name", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + + { + DataSetProperty property = properties.get(1); + assertEquals("value", property.getName()); + assertEquals(DataSetProperty.DataType.DECIMAL, property.getType()); + } + + { + DataSetProperty property = properties.get(2); + assertEquals("size", property.getName()); + assertEquals(DataSetProperty.DataType.DECIMAL, property.getType()); + } + + { + DataSetProperty property = properties.get(3); + assertEquals("date", property.getName()); + assertEquals(DataSetProperty.DataType.DATE, property.getType()); + } + } + + { + assertEquals(3, data.size()); + + SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd"); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(15, ((Number) row.get("value")).intValue()); + assertEquals(16, ((Number) row.get("size")).intValue()); + assertEquals("2020-08-01", dateFormat.format((Date) row.get("date"))); + } + + { + Map row = data.get(1); + + assertEquals("bbb", row.get("name")); + assertEquals(25, ((Number) row.get("value")).intValue()); + assertEquals(26, ((Number) row.get("size")).intValue()); + assertEquals("2020-08-02", dateFormat.format((Date) row.get("date"))); + } + + { + Map row = data.get(2); + + assertEquals("ccc", row.get("name")); + assertEquals(35, ((Number) row.get("value")).intValue()); + assertEquals(36, ((Number) row.get("size")).intValue()); + assertEquals("2020-08-03", dateFormat.format((Date) row.get("date"))); + } + } + } + + @Test + public void resolveTest_dataRowColumnExp() + { + ExcelDirectoryFileDataSet dataSet = new ExcelDirectoryFileDataSet("a", "a", DIRECTORY, + "ExcelDirectoryFileDataSetTest-0.xlsx"); + dataSet.setNameRow(1); + dataSet.setDataRowExp("2,3-"); + dataSet.setDataColumnExp("A,C-"); + + ResolvedDataSetResult resolvedResult = dataSet.resolve(new HashMap<>()); + + @SuppressWarnings("unchecked") + List> data = (List>) resolvedResult.getResult().getData(); + List properties = resolvedResult.getProperties(); + + { + assertEquals(3, properties.size()); + + { + DataSetProperty property = properties.get(0); + assertEquals("name", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + + { + DataSetProperty property = properties.get(1); + assertEquals("size", property.getName()); + assertEquals(DataSetProperty.DataType.DECIMAL, property.getType()); + } + + { + DataSetProperty property = properties.get(2); + assertEquals("date", property.getName()); + assertEquals(DataSetProperty.DataType.DATE, property.getType()); + } + } + + { + assertEquals(3, data.size()); + + SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd"); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(16, ((Number) row.get("size")).intValue()); + assertEquals("2020-08-01", dateFormat.format((Date) row.get("date"))); + } + + { + Map row = data.get(1); + + assertEquals("bbb", row.get("name")); + assertEquals(26, ((Number) row.get("size")).intValue()); + assertEquals("2020-08-02", dateFormat.format((Date) row.get("date"))); + } + + { + Map row = data.get(2); + + assertEquals("ccc", row.get("name")); + assertEquals(36, ((Number) row.get("size")).intValue()); + assertEquals("2020-08-03", dateFormat.format((Date) row.get("date"))); + } + } + } +} diff --git a/datagear-analysis/src/test/java/org/datagear/analysis/support/HttpDataSetTest.java b/datagear-analysis/src/test/java/org/datagear/analysis/support/HttpDataSetTest.java new file mode 100644 index 00000000..d9d427ed --- /dev/null +++ b/datagear-analysis/src/test/java/org/datagear/analysis/support/HttpDataSetTest.java @@ -0,0 +1,496 @@ +/* + * Copyright 2018 datagear.tech. All Rights Reserved. + */ + +package org.datagear.analysis.support; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertTrue; + +import java.io.IOException; +import java.io.InputStreamReader; +import java.io.Reader; +import java.net.URLDecoder; +import java.util.Arrays; +import java.util.Collections; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import org.apache.hc.client5.http.impl.classic.CloseableHttpClient; +import org.apache.hc.client5.http.impl.classic.HttpClients; +import org.apache.hc.core5.http.ClassicHttpRequest; +import org.apache.hc.core5.http.ClassicHttpResponse; +import org.apache.hc.core5.http.ContentType; +import org.apache.hc.core5.http.Header; +import org.apache.hc.core5.http.HttpEntity; +import org.apache.hc.core5.http.HttpException; +import org.apache.hc.core5.http.impl.bootstrap.HttpServer; +import org.apache.hc.core5.http.impl.bootstrap.ServerBootstrap; +import org.apache.hc.core5.http.io.HttpRequestHandler; +import org.apache.hc.core5.http.io.entity.StringEntity; +import org.apache.hc.core5.http.protocol.HttpContext; +import org.datagear.analysis.DataSetParam; +import org.datagear.analysis.DataSetProperty; +import org.datagear.util.IOUtil; +import org.junit.AfterClass; +import org.junit.BeforeClass; +import org.junit.Test; + +/** + * {@linkplain HttpDataSet}单元测试用例。 + * + * @author datagear@163.com + * + */ +public class HttpDataSetTest +{ + protected static final int PORT = 50402; + + protected static final String SERVER = "http://localhost:" + PORT; + + protected static final String PARAM_NAME_0 = "param0"; + + protected static final String PARAM_NAME_1 = "param1"; + + protected static HttpServer server; + + protected static CloseableHttpClient httpClient; + + @BeforeClass + public static void initTestHttpServer() throws Throwable + { + server = ServerBootstrap.bootstrap().setListenerPort(PORT) + // + .register("/testSimple", new HttpRequestHandler() + { + @Override + public void handle(ClassicHttpRequest request, ClassicHttpResponse response, HttpContext context) + throws HttpException, IOException + { + StringEntity responseEntity = new StringEntity( + "[{name: 'aaa', value: 11}, {name: '名称b', value: 22}]", ContentType.APPLICATION_JSON); + response.setEntity(responseEntity); + } + }) + // + .register("/testParam", new HttpRequestHandler() + { + @Override + public void handle(ClassicHttpRequest request, ClassicHttpResponse response, HttpContext context) + throws HttpException, IOException + { + Map params = parseRequestParams(request); + String p0 = params.get(PARAM_NAME_0); + String p1 = params.get(PARAM_NAME_1); + + StringEntity responseEntity = new StringEntity("[{name: '" + PARAM_NAME_0 + "', value: '" + p0 + + "'}, {name: '" + PARAM_NAME_1 + "', value: '" + p1 + "'}]", + ContentType.APPLICATION_JSON); + response.setEntity(responseEntity); + } + }) + // + .register("/testJson", new HttpRequestHandler() + { + @Override + public void handle(ClassicHttpRequest request, ClassicHttpResponse response, HttpContext context) + throws HttpException, IOException + { + String reqJson = getRequestStringContent(request); + + StringEntity responseEntity = new StringEntity(reqJson, ContentType.APPLICATION_JSON); + response.setEntity(responseEntity); + } + }) + // + .register("/testHeader", new HttpRequestHandler() + { + @Override + public void handle(ClassicHttpRequest request, ClassicHttpResponse response, HttpContext context) + throws HttpException, IOException + { + Header h0 = request.getHeader(PARAM_NAME_0); + Header h1 = request.getHeader(PARAM_NAME_1); + + StringEntity responseEntity = new StringEntity( + "[{name: '" + PARAM_NAME_0 + "', value: '" + h0.getValue() + "'}, {name: '" + + PARAM_NAME_1 + "', value: '" + h1.getValue() + "'}]", + ContentType.APPLICATION_JSON); + response.setEntity(responseEntity); + } + }) + // + .register("/testResponseJsonPath", new HttpRequestHandler() + { + @Override + public void handle(ClassicHttpRequest request, ClassicHttpResponse response, HttpContext context) + throws HttpException, IOException + { + StringEntity responseEntity = new StringEntity( + "{ path0: { path1: [ { path2: [{name: 'aaa', value: 11}, {name: '名称b', value: 22}] } ] } }", + ContentType.APPLICATION_JSON); + response.setEntity(responseEntity); + } + }) + // + .create(); + + server.start(); + + httpClient = HttpClients.createDefault(); + } + + @AfterClass + public static void closeHttpServer() throws Throwable + { + server.close(); + httpClient.close(); + } + + @Test + public void resolveTest_onlyUri() throws Throwable + { + List params = Arrays.asList(new DataSetParam("param", DataSetParam.DataType.NUMBER, true)); + + HttpDataSet dataSet = new HttpDataSet(HttpDataSet.class.getName(), HttpDataSet.class.getName(), httpClient, + SERVER + "/testSimple?param=${param}"); + + dataSet.setParams(params); + + Map paramValues = new HashMap<>(); + paramValues.put("param", "pv"); + + TemplateResolvedDataSetResult result = dataSet.resolve(paramValues); + List properties = result.getProperties(); + @SuppressWarnings("unchecked") + List> data = (List>) result.getResult().getData(); + String templateResult = result.getTemplateResult(); + + { + assertEquals(2, properties.size()); + + assertTrue(templateResult.contains("param=pv")); + + { + DataSetProperty property = properties.get(0); + assertEquals("name", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + + { + DataSetProperty property = properties.get(1); + assertEquals("value", property.getName()); + assertEquals(DataSetProperty.DataType.NUMBER, property.getType()); + } + } + + { + assertEquals(2, data.size()); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(11, ((Number) row.get("value")).intValue()); + } + + { + Map row = data.get(1); + + assertEquals("名称b", row.get("name")); + assertEquals(22, ((Number) row.get("value")).intValue()); + } + } + } + + @Test + public void resolveTest_setRequestMethod_GET() throws Throwable + { + HttpDataSet dataSet = new HttpDataSet(HttpDataSet.class.getName(), HttpDataSet.class.getName(), httpClient, + SERVER + "/testSimple"); + + dataSet.setRequestMethod(HttpDataSet.REQUEST_METHOD_GET); + + TemplateResolvedDataSetResult result = dataSet.resolve(Collections.emptyMap()); + List properties = result.getProperties(); + @SuppressWarnings("unchecked") + List> data = (List>) result.getResult().getData(); + + assertEquals(2, properties.size()); + assertEquals(2, data.size()); + } + + @Test + public void resolveTest_setRequestMethod_POST() throws Throwable + { + HttpDataSet dataSet = new HttpDataSet(HttpDataSet.class.getName(), HttpDataSet.class.getName(), httpClient, + SERVER + "/testSimple"); + + dataSet.setRequestMethod(HttpDataSet.REQUEST_METHOD_POST); + + TemplateResolvedDataSetResult result = dataSet.resolve(Collections.emptyMap()); + List properties = result.getProperties(); + @SuppressWarnings("unchecked") + List> data = (List>) result.getResult().getData(); + + assertEquals(2, properties.size()); + assertEquals(2, data.size()); + } + + @Test + public void resolveTest_REQUEST_CONTENT_TYPE_FORM_URLENCODED() throws Throwable + { + String pv0 = "p0"; + String pv1 = "参数值1"; + + List params = Arrays.asList(new DataSetParam("param", DataSetParam.DataType.NUMBER, true)); + + HttpDataSet dataSet = new HttpDataSet(HttpDataSet.class.getName(), HttpDataSet.class.getName(), httpClient, + SERVER + "/testParam"); + dataSet.setRequestContent("[ { name: '" + PARAM_NAME_0 + "', value: '" + pv0 + "' }, { name: '" + PARAM_NAME_1 + + "', value: '${param}' } ]"); + dataSet.setParams(params); + + Map paramValues = new HashMap<>(); + paramValues.put("param", pv1); + + TemplateResolvedDataSetResult result = dataSet.resolve(paramValues); + List properties = result.getProperties(); + @SuppressWarnings("unchecked") + List> data = (List>) result.getResult().getData(); + String templateResult = result.getTemplateResult(); + + { + assertEquals(2, properties.size()); + + assertTrue(templateResult.contains("value: '" + pv1 + "'")); + + { + DataSetProperty property = properties.get(0); + assertEquals("name", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + + { + DataSetProperty property = properties.get(1); + assertEquals("value", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + } + + { + assertEquals(2, data.size()); + + { + Map row = data.get(0); + + assertEquals(PARAM_NAME_0, row.get("name")); + assertEquals(pv0, row.get("value")); + } + + { + Map row = data.get(1); + + assertEquals(PARAM_NAME_1, row.get("name")); + assertEquals(pv1, row.get("value")); + } + } + } + + @Test + public void resolveTest_REQUEST_CONTENT_TYPE_JSON() throws Throwable + { + String pv0 = "p0"; + String pv1 = "参数值1"; + + List params = Arrays.asList(new DataSetParam("param", DataSetParam.DataType.NUMBER, true)); + + HttpDataSet dataSet = new HttpDataSet(HttpDataSet.class.getName(), HttpDataSet.class.getName(), httpClient, + SERVER + "/testJson"); + dataSet.setRequestContentType(HttpDataSet.REQUEST_CONTENT_TYPE_JSON); + dataSet.setRequestContent("[ { name: '" + PARAM_NAME_0 + "', value: '" + pv0 + "' }, { name: '" + PARAM_NAME_1 + + "', value: '${param}' } ]"); + dataSet.setParams(params); + + Map paramValues = new HashMap<>(); + paramValues.put("param", pv1); + + TemplateResolvedDataSetResult result = dataSet.resolve(paramValues); + List properties = result.getProperties(); + @SuppressWarnings("unchecked") + List> data = (List>) result.getResult().getData(); + String templateResult = result.getTemplateResult(); + + { + assertEquals(2, properties.size()); + + assertTrue(templateResult.contains("value: '" + pv1 + "'")); + + { + DataSetProperty property = properties.get(0); + assertEquals("name", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + + { + DataSetProperty property = properties.get(1); + assertEquals("value", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + } + + { + assertEquals(2, data.size()); + + { + Map row = data.get(0); + + assertEquals(PARAM_NAME_0, row.get("name")); + assertEquals(pv0, row.get("value")); + } + + { + Map row = data.get(1); + + assertEquals(PARAM_NAME_1, row.get("name")); + assertEquals(pv1, row.get("value")); + } + } + } + + @Test + public void resolveTest_setHeaderContent() throws Throwable + { + String pv0 = "p0"; + String pv1 = "p1"; + + List params = Arrays.asList(new DataSetParam("param", DataSetParam.DataType.NUMBER, true)); + + HttpDataSet dataSet = new HttpDataSet(HttpDataSet.class.getName(), HttpDataSet.class.getName(), httpClient, + SERVER + "/testHeader"); + dataSet.setHeaderContent("[ { name: '" + PARAM_NAME_0 + "', value: '" + pv0 + "' }, { name: '" + PARAM_NAME_1 + + "', value: '${param}' } ]"); + dataSet.setParams(params); + + Map paramValues = new HashMap<>(); + paramValues.put("param", pv1); + + TemplateResolvedDataSetResult result = dataSet.resolve(paramValues); + List properties = result.getProperties(); + @SuppressWarnings("unchecked") + List> data = (List>) result.getResult().getData(); + String templateResult = result.getTemplateResult(); + + { + assertEquals(2, properties.size()); + + assertTrue(templateResult.contains("value: '" + pv1 + "'")); + + { + DataSetProperty property = properties.get(0); + assertEquals("name", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + + { + DataSetProperty property = properties.get(1); + assertEquals("value", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + } + + { + assertEquals(2, data.size()); + + { + Map row = data.get(0); + + assertEquals(PARAM_NAME_0, row.get("name")); + assertEquals(pv0, row.get("value")); + } + + { + Map row = data.get(1); + + assertEquals(PARAM_NAME_1, row.get("name")); + assertEquals(pv1, row.get("value")); + } + } + } + + @Test + public void resolveTest_setResponseDataJsonPath() throws Throwable + { + HttpDataSet dataSet = new HttpDataSet(HttpDataSet.class.getName(), HttpDataSet.class.getName(), httpClient, + SERVER + "/testResponseJsonPath"); + dataSet.setResponseDataJsonPath("path0.path1[0].path2"); + + TemplateResolvedDataSetResult result = dataSet.resolve(Collections.emptyMap()); + List properties = result.getProperties(); + @SuppressWarnings("unchecked") + List> data = (List>) result.getResult().getData(); + + { + assertEquals(2, properties.size()); + + { + DataSetProperty property = properties.get(0); + assertEquals("name", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + + { + DataSetProperty property = properties.get(1); + assertEquals("value", property.getName()); + assertEquals(DataSetProperty.DataType.NUMBER, property.getType()); + } + } + + { + assertEquals(2, data.size()); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(11, ((Number) row.get("value")).intValue()); + } + + { + Map row = data.get(1); + + assertEquals("名称b", row.get("name")); + assertEquals(22, ((Number) row.get("value")).intValue()); + } + } + } + + protected static Map parseRequestParams(ClassicHttpRequest request) throws IOException + { + Map map = new HashMap<>(); + + String content = getRequestStringContent(request); + + String[] strss = content.split("&"); + for (String strs : strss) + { + String[] pv = strs.split("="); + + map.put(pv[0], URLDecoder.decode(pv[1], IOUtil.CHARSET_UTF_8)); + } + + return map; + } + + protected static String getRequestStringContent(ClassicHttpRequest request) throws IOException + { + HttpEntity entity = request.getEntity(); + String contentTypeStr = entity.getContentType(); + ContentType contentType = ContentType.parse(contentTypeStr); + Reader reader = new InputStreamReader(entity.getContent(), contentType.getCharset()); + String content = IOUtil.readString(reader, false); + + return content; + } +} diff --git a/datagear-analysis/src/test/java/org/datagear/analysis/support/JsonDataSetSupportTest.java b/datagear-analysis/src/test/java/org/datagear/analysis/support/JsonDataSetSupportTest.java deleted file mode 100644 index db13d9c8..00000000 --- a/datagear-analysis/src/test/java/org/datagear/analysis/support/JsonDataSetSupportTest.java +++ /dev/null @@ -1,38 +0,0 @@ -/* - * Copyright (c) 2018 datagear.tech. All Rights Reserved. - */ - -/** - * - */ -package org.datagear.analysis.support; - -import java.util.Map; - -import org.junit.Assert; -import org.junit.Test; - -/** - * {@linkplain JsonDataSetSupport}单元测试类。 - * - * @author datagear@163.com - * - */ -public class JsonDataSetSupportTest -{ - private JsonDataSetSupport jsonDataSetSupport = new JsonDataSetSupport(); - - @Test - public void resolveResultDataTest_String() - { - Object data = jsonDataSetSupport.resolveResultData("{name:'a', value: 3}"); - - Assert.assertTrue(data instanceof Map); - - @SuppressWarnings("unchecked") - Map map = (Map) data; - - Assert.assertEquals("a", map.get("name")); - Assert.assertEquals(3, ((Number) map.get("value")).intValue()); - } -} diff --git a/datagear-analysis/src/test/java/org/datagear/analysis/support/JsonDirectoryFileDataSetTest.java b/datagear-analysis/src/test/java/org/datagear/analysis/support/JsonDirectoryFileDataSetTest.java new file mode 100644 index 00000000..db5cfa9c --- /dev/null +++ b/datagear-analysis/src/test/java/org/datagear/analysis/support/JsonDirectoryFileDataSetTest.java @@ -0,0 +1,203 @@ +/* + * Copyright 2018 datagear.tech. All Rights Reserved. + */ + +package org.datagear.analysis.support; + +import static org.junit.Assert.assertEquals; + +import java.io.File; +import java.text.SimpleDateFormat; +import java.util.ArrayList; +import java.util.Collections; +import java.util.Date; +import java.util.List; +import java.util.Map; + +import org.datagear.analysis.DataSetProperty; +import org.datagear.analysis.DataSetResult; +import org.datagear.analysis.ResolvedDataSetResult; +import org.junit.Test; + +/** + * {@linkplain JsonDirectoryFileDataSet}单元测试类。 + * + * @author datagear@163.com + * + */ +public class JsonDirectoryFileDataSetTest +{ + private static final File DIRECTORY = new File("src/test/resources/org/datagear/analysis/support/"); + + @Test + public void getResultTest() + { + List properties = new ArrayList<>(); + properties.add(new DataSetProperty("name", DataSetProperty.DataType.STRING)); + properties.add(new DataSetProperty("value", DataSetProperty.DataType.NUMBER)); + properties.add(new DataSetProperty("尺寸", DataSetProperty.DataType.NUMBER)); + properties.add(new DataSetProperty("date", DataSetProperty.DataType.STRING)); + + JsonDirectoryFileDataSet dataSet = new JsonDirectoryFileDataSet("a", "a", properties, DIRECTORY, + "JsonDirectoryFileDataSetTest-0.json"); + + @SuppressWarnings("unchecked") + DataSetResult result = dataSet.getResult(Collections.EMPTY_MAP); + @SuppressWarnings("unchecked") + List> data = (List>) result.getData(); + + { + assertEquals(3, data.size()); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(11, ((Number) row.get("value")).intValue()); + assertEquals(12, ((Number) row.get("尺寸")).intValue()); + assertEquals("2020-08-01", row.get("date")); + } + + { + Map row = data.get(1); + + assertEquals("bbb", row.get("name")); + assertEquals(21, ((Number) row.get("value")).intValue()); + assertEquals(22, ((Number) row.get("尺寸")).intValue()); + assertEquals("2020-08-02", row.get("date")); + } + + { + Map row = data.get(2); + + assertEquals("ccc", row.get("name")); + assertEquals(31, ((Number) row.get("value")).intValue()); + assertEquals(32, ((Number) row.get("尺寸")).intValue()); + assertEquals("2020-08-03", row.get("date")); + } + } + } + + @Test + public void getResultTest_convertPropertyValue() + { + List properties = new ArrayList<>(); + properties.add(new DataSetProperty("name", DataSetProperty.DataType.STRING)); + properties.add(new DataSetProperty("value", DataSetProperty.DataType.NUMBER)); + properties.add(new DataSetProperty("尺寸", DataSetProperty.DataType.NUMBER)); + properties.add(new DataSetProperty("date", DataSetProperty.DataType.DATE)); + + JsonDirectoryFileDataSet dataSet = new JsonDirectoryFileDataSet("a", "a", properties, DIRECTORY, + "JsonDirectoryFileDataSetTest-0.json"); + + @SuppressWarnings("unchecked") + DataSetResult result = dataSet.getResult(Collections.EMPTY_MAP); + @SuppressWarnings("unchecked") + List> data = (List>) result.getData(); + + { + assertEquals(3, data.size()); + + SimpleDateFormat dateFormat = new SimpleDateFormat(DataFormat.DEFAULT_DATE_FORMAT); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(11, ((Number) row.get("value")).intValue()); + assertEquals(12, ((Number) row.get("尺寸")).intValue()); + assertEquals("2020-08-01", dateFormat.format(((Date) row.get("date")))); + } + + { + Map row = data.get(1); + + assertEquals("bbb", row.get("name")); + assertEquals(21, ((Number) row.get("value")).intValue()); + assertEquals(22, ((Number) row.get("尺寸")).intValue()); + assertEquals("2020-08-02", dateFormat.format(((Date) row.get("date")))); + } + + { + Map row = data.get(2); + + assertEquals("ccc", row.get("name")); + assertEquals(31, ((Number) row.get("value")).intValue()); + assertEquals(32, ((Number) row.get("尺寸")).intValue()); + assertEquals("2020-08-03", dateFormat.format(((Date) row.get("date")))); + } + } + } + + @Test + public void resolveTest() + { + JsonDirectoryFileDataSet dataSet = new JsonDirectoryFileDataSet("a", "a", DIRECTORY, + "JsonDirectoryFileDataSetTest-0.json"); + + @SuppressWarnings("unchecked") + ResolvedDataSetResult result = dataSet.resolve(Collections.EMPTY_MAP); + List properties = result.getProperties(); + @SuppressWarnings("unchecked") + List> data = (List>) result.getResult().getData(); + + { + assertEquals(4, properties.size()); + + { + DataSetProperty property = properties.get(0); + assertEquals("name", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + + { + DataSetProperty property = properties.get(1); + assertEquals("value", property.getName()); + assertEquals(DataSetProperty.DataType.NUMBER, property.getType()); + } + + { + DataSetProperty property = properties.get(2); + assertEquals("尺寸", property.getName()); + assertEquals(DataSetProperty.DataType.NUMBER, property.getType()); + } + + { + DataSetProperty property = properties.get(3); + assertEquals("date", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + } + + { + assertEquals(3, data.size()); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(11, ((Number) row.get("value")).intValue()); + assertEquals(12, ((Number) row.get("尺寸")).intValue()); + assertEquals("2020-08-01", row.get("date")); + } + + { + Map row = data.get(1); + + assertEquals("bbb", row.get("name")); + assertEquals(21, ((Number) row.get("value")).intValue()); + assertEquals(22, ((Number) row.get("尺寸")).intValue()); + assertEquals("2020-08-02", row.get("date")); + } + + { + Map row = data.get(2); + + assertEquals("ccc", row.get("name")); + assertEquals(31, ((Number) row.get("value")).intValue()); + assertEquals(32, ((Number) row.get("尺寸")).intValue()); + assertEquals("2020-08-03", row.get("date")); + } + } + } +} diff --git a/datagear-analysis/src/test/java/org/datagear/analysis/support/JsonValueDataSetTest.java b/datagear-analysis/src/test/java/org/datagear/analysis/support/JsonValueDataSetTest.java new file mode 100644 index 00000000..f11af80e --- /dev/null +++ b/datagear-analysis/src/test/java/org/datagear/analysis/support/JsonValueDataSetTest.java @@ -0,0 +1,151 @@ +/* + * Copyright 2018 datagear.tech. All Rights Reserved. + */ + +package org.datagear.analysis.support; + +import static org.junit.Assert.assertEquals; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import org.datagear.analysis.DataSetParam; +import org.datagear.analysis.DataSetProperty; +import org.datagear.analysis.DataSetResult; +import org.junit.Test; + +/** + * {@linkplain JsonValueDataSet}单元测试类。 + * + * @author datagear@163.com + * + */ +public class JsonValueDataSetTest +{ + @Test + public void getResultTest_hasParam() + { + List properties = new ArrayList<>(); + properties.add(new DataSetProperty("name", DataSetProperty.DataType.STRING)); + properties.add(new DataSetProperty("value", DataSetProperty.DataType.NUMBER)); + properties.add(new DataSetProperty("size", DataSetProperty.DataType.NUMBER)); + + List params = new ArrayList<>(); + params.add(new DataSetParam("size", DataSetParam.DataType.NUMBER, true)); + + JsonValueDataSet dataSet = new JsonValueDataSet(JsonValueDataSet.class.getSimpleName(), + JsonValueDataSet.class.getSimpleName(), properties, "[ { name:'aaa', value: 11, size: ${size} } ]"); + dataSet.setParams(params); + + Map paramValues = new HashMap<>(); + paramValues.put("size", 12); + + DataSetResult result = dataSet.getResult(paramValues); + @SuppressWarnings("unchecked") + List> data = (List>) result.getData(); + + { + assertEquals(1, data.size()); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(11, ((Number) row.get("value")).intValue()); + assertEquals(12, ((Number) row.get("size")).intValue()); + } + } + } + + @Test + public void getResultTest_hasParam_convertPropertyValue() + { + List properties = new ArrayList<>(); + properties.add(new DataSetProperty("name", DataSetProperty.DataType.STRING)); + properties.add(new DataSetProperty("value", DataSetProperty.DataType.NUMBER)); + properties.add(new DataSetProperty("size", DataSetProperty.DataType.STRING)); + + List params = new ArrayList<>(); + params.add(new DataSetParam("size", DataSetParam.DataType.NUMBER, true)); + + JsonValueDataSet dataSet = new JsonValueDataSet(JsonValueDataSet.class.getSimpleName(), + JsonValueDataSet.class.getSimpleName(), properties, "[ { name:'aaa', value: 11, size: ${size} } ]"); + dataSet.setParams(params); + + Map paramValues = new HashMap<>(); + paramValues.put("size", 12); + + DataSetResult result = dataSet.getResult(paramValues); + @SuppressWarnings("unchecked") + List> data = (List>) result.getData(); + + { + assertEquals(1, data.size()); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(11, ((Number) row.get("value")).intValue()); + assertEquals("12", row.get("size")); + } + } + } + + @Test + public void resolveTest_hasParam() + { + List params = new ArrayList<>(); + params.add(new DataSetParam("size", DataSetParam.DataType.NUMBER, true)); + + JsonValueDataSet dataSet = new JsonValueDataSet(JsonValueDataSet.class.getSimpleName(), + JsonValueDataSet.class.getSimpleName(), "[ { name:'aaa', value: 11, size: ${size} } ]"); + dataSet.setParams(params); + + Map paramValues = new HashMap<>(); + paramValues.put("size", 12); + + TemplateResolvedDataSetResult result = dataSet.resolve(paramValues); + List properties = result.getProperties(); + @SuppressWarnings("unchecked") + List> data = (List>) result.getResult().getData(); + + assertEquals("[ { name:'aaa', value: 11, size: 12 } ]", result.getTemplateResult()); + + { + assertEquals(3, properties.size()); + + { + DataSetProperty property = properties.get(0); + assertEquals("name", property.getName()); + assertEquals(DataSetProperty.DataType.STRING, property.getType()); + } + + { + DataSetProperty property = properties.get(1); + assertEquals("value", property.getName()); + assertEquals(DataSetProperty.DataType.NUMBER, property.getType()); + } + + { + DataSetProperty property = properties.get(2); + assertEquals("size", property.getName()); + assertEquals(DataSetProperty.DataType.NUMBER, property.getType()); + } + } + + { + assertEquals(1, data.size()); + + { + Map row = data.get(0); + + assertEquals("aaa", row.get("name")); + assertEquals(11, ((Number) row.get("value")).intValue()); + assertEquals(12, ((Number) row.get("size")).intValue()); + } + } + } +} diff --git a/datagear-analysis/src/test/java/org/datagear/analysis/support/SqlDataSetTest.java b/datagear-analysis/src/test/java/org/datagear/analysis/support/SqlDataSetTest.java index 262f0eb2..05ee358d 100644 --- a/datagear-analysis/src/test/java/org/datagear/analysis/support/SqlDataSetTest.java +++ b/datagear-analysis/src/test/java/org/datagear/analysis/support/SqlDataSetTest.java @@ -16,10 +16,10 @@ import java.util.Map; import org.datagear.analysis.DataSetParam; import org.datagear.analysis.DataSetProperty; -import org.datagear.analysis.DataSetProperty.DataType; import org.datagear.analysis.DataSetResult; import org.datagear.util.JdbcUtil; import org.datagear.util.resource.SimpleConnectionFactory; +import org.datagear.util.test.DBTestSupport; import org.junit.Assert; import org.junit.Test; @@ -61,10 +61,12 @@ public class SqlDataSetTest extends DBTestSupport String sql = "SELECT ID, NAME FROM T_ACCOUNT <#if id??>WHERE ID = ${id} AND NAME != '${name}'"; - List dataSetProperties = Arrays.asList(new DataSetProperty("ID", DataType.INTEGER), - new DataSetProperty("NAME", DataType.STRING)); - List dataSetParams = Arrays.asList(new DataSetParam("id", DataType.STRING, true), - new DataSetParam("name", DataType.STRING, true)); + List dataSetProperties = Arrays.asList( + new DataSetProperty("ID", DataSetProperty.DataType.INTEGER), + new DataSetProperty("NAME", DataSetProperty.DataType.STRING)); + + List dataSetParams = Arrays.asList(new DataSetParam("id", DataSetParam.DataType.STRING, true), + new DataSetParam("name", DataSetParam.DataType.STRING, true)); SqlDataSet sqlDataSet = new SqlDataSet("1", "1", dataSetProperties, connectionFactory, sql); sqlDataSet.setParams(dataSetParams); diff --git a/datagear-analysis/src/test/resources/org/datagear/analysis/support/CsvDirectoryFileDataSetTest-0.csv b/datagear-analysis/src/test/resources/org/datagear/analysis/support/CsvDirectoryFileDataSetTest-0.csv new file mode 100644 index 00000000..09481a79 --- /dev/null +++ b/datagear-analysis/src/test/resources/org/datagear/analysis/support/CsvDirectoryFileDataSetTest-0.csv @@ -0,0 +1,4 @@ +name, value, 尺寸 +aaa, 11, 12 +bbb, 21, 22 +ccc, 31, 32 \ No newline at end of file diff --git a/datagear-analysis/src/test/resources/org/datagear/analysis/support/ExcelDirectoryFileDataSetTest-0.xlsx b/datagear-analysis/src/test/resources/org/datagear/analysis/support/ExcelDirectoryFileDataSetTest-0.xlsx new file mode 100644 index 00000000..5fa938a1 Binary files /dev/null and b/datagear-analysis/src/test/resources/org/datagear/analysis/support/ExcelDirectoryFileDataSetTest-0.xlsx differ diff --git a/datagear-analysis/src/test/resources/org/datagear/analysis/support/ExcelDirectoryFileDataSetTest-1.xls b/datagear-analysis/src/test/resources/org/datagear/analysis/support/ExcelDirectoryFileDataSetTest-1.xls new file mode 100644 index 00000000..a6d3974f Binary files /dev/null and b/datagear-analysis/src/test/resources/org/datagear/analysis/support/ExcelDirectoryFileDataSetTest-1.xls differ diff --git a/datagear-analysis/src/test/resources/org/datagear/analysis/support/JsonDirectoryFileDataSetTest-0.json b/datagear-analysis/src/test/resources/org/datagear/analysis/support/JsonDirectoryFileDataSetTest-0.json new file mode 100644 index 00000000..664c8866 --- /dev/null +++ b/datagear-analysis/src/test/resources/org/datagear/analysis/support/JsonDirectoryFileDataSetTest-0.json @@ -0,0 +1,20 @@ +[ + { + name: "aaa", + value: 11, + "尺寸": 12, + date: "2020-08-01" + }, + { + name: "bbb", + value: 21, + "尺寸": 22, + date: "2020-08-02" + }, + { + name: "ccc", + value: 31, + "尺寸": 32, + date: "2020-08-03" + } +] \ No newline at end of file diff --git a/datagear-connection/pom.xml b/datagear-connection/pom.xml index fdf7f331..1f4c4db4 100644 --- a/datagear-connection/pom.xml +++ b/datagear-connection/pom.xml @@ -6,7 +6,7 @@ org.datagear datagear - 1.11.1 + 1.12.0 datagear-connection diff --git a/datagear-dataexchange/pom.xml b/datagear-dataexchange/pom.xml index d9003775..b5c937a5 100644 --- a/datagear-dataexchange/pom.xml +++ b/datagear-dataexchange/pom.xml @@ -7,7 +7,7 @@ org.datagear datagear - 1.11.1 + 1.12.0 datagear-dataexchange @@ -27,17 +27,17 @@ org.apache.commons commons-csv - 1.4 + ${commons-csv.version} org.apache.poi poi - 3.17 + ${poi.version} org.apache.poi poi-ooxml - 3.17 + ${poi-ooxml.version} org.glassfish diff --git a/datagear-management/pom.xml b/datagear-management/pom.xml index 83586394..809309d3 100644 --- a/datagear-management/pom.xml +++ b/datagear-management/pom.xml @@ -6,7 +6,7 @@ org.datagear datagear - 1.11.1 + 1.12.0 datagear-management diff --git a/datagear-management/src/main/java/org/datagear/management/domain/CsvFileDataSetEntity.java b/datagear-management/src/main/java/org/datagear/management/domain/CsvFileDataSetEntity.java new file mode 100644 index 00000000..ab983c24 --- /dev/null +++ b/datagear-management/src/main/java/org/datagear/management/domain/CsvFileDataSetEntity.java @@ -0,0 +1,114 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.management.domain; + +import java.io.File; +import java.util.Date; +import java.util.List; + +import org.datagear.analysis.DataSetProperty; +import org.datagear.analysis.support.CsvDirectoryFileDataSet; + +/** + * {@linkplain CsvDirectoryFileDataSet}实体。 + * + * @author datagear@163.com + * + */ +public class CsvFileDataSetEntity extends CsvDirectoryFileDataSet implements DirectoryFileDataSetEntity +{ + private static final long serialVersionUID = 1L; + + /** 展示名 */ + private String displayName; + + /** 创建用户 */ + private User createUser; + + /** 创建时间 */ + private Date createTime; + + /** 权限 */ + private int dataPermission = PERMISSION_NOT_LOADED; + + public CsvFileDataSetEntity() + { + super(); + this.createTime = new Date(); + } + + public CsvFileDataSetEntity(String id, String name, List properties, File directory, + String fileName, String displayName, User createUser) + { + super(id, name, properties, directory, fileName); + this.displayName = displayName; + this.createTime = new Date(); + this.createUser = createUser; + } + + @Override + public String getDisplayName() + { + return displayName; + } + + @Override + public void setDisplayName(String displayName) + { + this.displayName = displayName; + } + + @Override + public String getDataSetType() + { + return DataSetEntity.DATA_SET_TYPE_CsvFile; + } + + @Override + public void setDataSetType(String dataSetType) + { + // XXX 什么也不做,不采用抛出异常的方式,便于统一底层SQL查询语句 + // throw new UnsupportedOperationException(); + } + + @Override + public User getCreateUser() + { + return createUser; + } + + @Override + public void setCreateUser(User createUser) + { + this.createUser = createUser; + } + + @Override + public Date getCreateTime() + { + return createTime; + } + + @Override + public void setCreateTime(Date createTime) + { + this.createTime = createTime; + } + + @Override + public int getDataPermission() + { + return dataPermission; + } + + @Override + public void setDataPermission(int dataPermission) + { + this.dataPermission = dataPermission; + } +} diff --git a/datagear-management/src/main/java/org/datagear/management/domain/CsvValueDataSetEntity.java b/datagear-management/src/main/java/org/datagear/management/domain/CsvValueDataSetEntity.java new file mode 100644 index 00000000..8a69eaec --- /dev/null +++ b/datagear-management/src/main/java/org/datagear/management/domain/CsvValueDataSetEntity.java @@ -0,0 +1,97 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.management.domain; + +import java.util.Date; +import java.util.List; + +import org.datagear.analysis.DataSetProperty; +import org.datagear.analysis.support.CsvValueDataSet; + +/** + * {@linkplain CsvValueDataSet}实体。 + * + * @author datagear@163.com + * + */ +public class CsvValueDataSetEntity extends CsvValueDataSet implements DataSetEntity +{ + private static final long serialVersionUID = 1L; + + /** 创建用户 */ + private User createUser; + + /** 创建时间 */ + private Date createTime; + + /** 权限 */ + private int dataPermission = PERMISSION_NOT_LOADED; + + public CsvValueDataSetEntity() + { + super(); + this.createTime = new Date(); + } + + public CsvValueDataSetEntity(String id, String name, List properties, String value, + User createUser) + { + super(id, name, properties, value); + this.createTime = new Date(); + this.createUser = createUser; + } + + @Override + public String getDataSetType() + { + return DataSetEntity.DATA_SET_TYPE_CsvValue; + } + + @Override + public void setDataSetType(String dataSetType) + { + // XXX 什么也不做,不采用抛出异常的方式,便于统一底层SQL查询语句 + // throw new UnsupportedOperationException(); + } + + @Override + public User getCreateUser() + { + return createUser; + } + + @Override + public void setCreateUser(User createUser) + { + this.createUser = createUser; + } + + @Override + public Date getCreateTime() + { + return createTime; + } + + @Override + public void setCreateTime(Date createTime) + { + this.createTime = createTime; + } + + @Override + public int getDataPermission() + { + return dataPermission; + } + + @Override + public void setDataPermission(int dataPermission) + { + this.dataPermission = dataPermission; + } +} diff --git a/datagear-management/src/main/java/org/datagear/management/domain/DataSetEntity.java b/datagear-management/src/main/java/org/datagear/management/domain/DataSetEntity.java index 1e20595b..b3e52bff 100644 --- a/datagear-management/src/main/java/org/datagear/management/domain/DataSetEntity.java +++ b/datagear-management/src/main/java/org/datagear/management/domain/DataSetEntity.java @@ -31,6 +31,18 @@ public interface DataSetEntity extends DataSet, CreateUserEntity, DataPe /** 数据集类型:JSON文件 */ String DATA_SET_TYPE_JsonFile = "JsonFile"; + /** 数据集类型:Excel */ + String DATA_SET_TYPE_Excel = "Excel"; + + /** 数据集类型:CSV值 */ + String DATA_SET_TYPE_CsvValue = "CsvValue"; + + /** 数据集类型:CSV文件 */ + String DATA_SET_TYPE_CsvFile = "CsvFile"; + + /** 数据集类型:HTTP接口 */ + String DATA_SET_TYPE_Http = "Http"; + /** * 设置名称。 * diff --git a/datagear-management/src/main/java/org/datagear/management/domain/DirectoryFileDataSetEntity.java b/datagear-management/src/main/java/org/datagear/management/domain/DirectoryFileDataSetEntity.java new file mode 100644 index 00000000..069e2cb3 --- /dev/null +++ b/datagear-management/src/main/java/org/datagear/management/domain/DirectoryFileDataSetEntity.java @@ -0,0 +1,58 @@ +/* + * Copyright 2018 datagear.tech. All Rights Reserved. + */ + +package org.datagear.management.domain; + +import java.io.File; + +/** + * 目录内文件数据集实体。 + * + * @author datagear@163.com + * + */ +public interface DirectoryFileDataSetEntity extends DataSetEntity +{ + /** + * 获取目录。 + * + * @return + */ + File getDirectory(); + + /** + * 设置目录。 + * + * @param directory + */ + void setDirectory(File directory); + + /** + * 获取文件名。 + * + * @return + */ + String getFileName(); + + /** + * 设置文件名。 + * + * @param fileName + */ + void setFileName(String fileName); + + /** + * 获取文件展示名。 + * + * @return + */ + String getDisplayName(); + + /** + * 设置文件展示名。 + * + * @param displayName + */ + void setDisplayName(String displayName); +} diff --git a/datagear-management/src/main/java/org/datagear/management/domain/ExcelDataSetEntity.java b/datagear-management/src/main/java/org/datagear/management/domain/ExcelDataSetEntity.java new file mode 100644 index 00000000..ed8a57c7 --- /dev/null +++ b/datagear-management/src/main/java/org/datagear/management/domain/ExcelDataSetEntity.java @@ -0,0 +1,114 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.management.domain; + +import java.io.File; +import java.util.Date; +import java.util.List; + +import org.datagear.analysis.DataSetProperty; +import org.datagear.analysis.support.ExcelDirectoryFileDataSet; + +/** + * {@linkplain ExcelDirectoryFileDataSet}实体。 + * + * @author datagear@163.com + * + */ +public class ExcelDataSetEntity extends ExcelDirectoryFileDataSet implements DirectoryFileDataSetEntity +{ + private static final long serialVersionUID = 1L; + + /** 展示名 */ + private String displayName; + + /** 创建用户 */ + private User createUser; + + /** 创建时间 */ + private Date createTime; + + /** 权限 */ + private int dataPermission = PERMISSION_NOT_LOADED; + + public ExcelDataSetEntity() + { + super(); + this.createTime = new Date(); + } + + public ExcelDataSetEntity(String id, String name, List properties, File directory, String fileName, + String displayName, User createUser) + { + super(id, name, properties, directory, fileName); + this.displayName = displayName; + this.createTime = new Date(); + this.createUser = createUser; + } + + @Override + public String getDisplayName() + { + return displayName; + } + + @Override + public void setDisplayName(String displayName) + { + this.displayName = displayName; + } + + @Override + public String getDataSetType() + { + return DataSetEntity.DATA_SET_TYPE_Excel; + } + + @Override + public void setDataSetType(String dataSetType) + { + // XXX 什么也不做,不采用抛出异常的方式,便于统一底层SQL查询语句 + // throw new UnsupportedOperationException(); + } + + @Override + public User getCreateUser() + { + return createUser; + } + + @Override + public void setCreateUser(User createUser) + { + this.createUser = createUser; + } + + @Override + public Date getCreateTime() + { + return createTime; + } + + @Override + public void setCreateTime(Date createTime) + { + this.createTime = createTime; + } + + @Override + public int getDataPermission() + { + return dataPermission; + } + + @Override + public void setDataPermission(int dataPermission) + { + this.dataPermission = dataPermission; + } +} diff --git a/datagear-management/src/main/java/org/datagear/management/domain/HttpDataSetEntity.java b/datagear-management/src/main/java/org/datagear/management/domain/HttpDataSetEntity.java new file mode 100644 index 00000000..d4be2c53 --- /dev/null +++ b/datagear-management/src/main/java/org/datagear/management/domain/HttpDataSetEntity.java @@ -0,0 +1,105 @@ +/* + * Copyright (c) 2018 datagear.tech. All Rights Reserved. + */ + +/** + * + */ +package org.datagear.management.domain; + +import java.util.Date; +import java.util.List; + +import org.apache.hc.client5.http.classic.HttpClient; +import org.datagear.analysis.DataSetProperty; +import org.datagear.analysis.support.HttpDataSet; + +/** + * {@linkplain HttpDataSet}实体。 + * + * @author datagear@163.com + * + */ +public class HttpDataSetEntity extends HttpDataSet implements DataSetEntity +{ + private static final long serialVersionUID = 1L; + + /** 创建用户 */ + private User createUser; + + /** 创建时间 */ + private Date createTime; + + /** 权限 */ + private int dataPermission = PERMISSION_NOT_LOADED; + + public HttpDataSetEntity() + { + super(); + this.createTime = new Date(); + } + + public HttpDataSetEntity(String id, String name, HttpClient httpClient, String uri, User createUser) + { + super(id, name, httpClient, uri); + this.createTime = new Date(); + this.createUser = createUser; + } + + public HttpDataSetEntity(String id, String name, List properties, HttpClient httpClient, + String uri, User createUser) + { + super(id, name, properties, httpClient, uri); + this.createTime = new Date(); + this.createUser = createUser; + } + + @Override + public String getDataSetType() + { + return DataSetEntity.DATA_SET_TYPE_Http; + } + + @Override + public void setDataSetType(String dataSetType) + { + // XXX 什么也不做,不采用抛出异常的方式,便于统一底层SQL查询语句 + // throw new UnsupportedOperationException(); + } + + @Override + public User getCreateUser() + { + return createUser; + } + + @Override + public void setCreateUser(User createUser) + { + this.createUser = createUser; + } + + @Override + public Date getCreateTime() + { + return createTime; + } + + @Override + public void setCreateTime(Date createTime) + { + this.createTime = createTime; + } + + @Override + public int getDataPermission() + { + return dataPermission; + } + + @Override + public void setDataPermission(int dataPermission) + { + this.dataPermission = dataPermission; + } +} diff --git a/datagear-management/src/main/java/org/datagear/management/domain/JsonFileDataSetEntity.java b/datagear-management/src/main/java/org/datagear/management/domain/JsonFileDataSetEntity.java index b272edd0..1f426316 100644 --- a/datagear-management/src/main/java/org/datagear/management/domain/JsonFileDataSetEntity.java +++ b/datagear-management/src/main/java/org/datagear/management/domain/JsonFileDataSetEntity.java @@ -20,7 +20,7 @@ import org.datagear.analysis.support.JsonDirectoryFileDataSet; * @author datagear@163.com * */ -public class JsonFileDataSetEntity extends JsonDirectoryFileDataSet implements DataSetEntity +public class JsonFileDataSetEntity extends JsonDirectoryFileDataSet implements DirectoryFileDataSetEntity { private static final long serialVersionUID = 1L; @@ -51,11 +51,13 @@ public class JsonFileDataSetEntity extends JsonDirectoryFileDataSet implements D this.createUser = createUser; } + @Override public String getDisplayName() { return displayName; } + @Override public void setDisplayName(String displayName) { this.displayName = displayName; diff --git a/datagear-management/src/main/java/org/datagear/management/service/DataSetEntityService.java b/datagear-management/src/main/java/org/datagear/management/service/DataSetEntityService.java index 87625424..dac4f907 100644 --- a/datagear-management/src/main/java/org/datagear/management/service/DataSetEntityService.java +++ b/datagear-management/src/main/java/org/datagear/management/service/DataSetEntityService.java @@ -9,6 +9,7 @@ package org.datagear.management.service; import java.io.File; +import org.apache.hc.client5.http.classic.HttpClient; import org.datagear.analysis.DataSet; import org.datagear.management.domain.DataSetEntity; @@ -36,4 +37,11 @@ public interface DataSetEntityService * @return */ File getDataSetDirectory(String dataSetId); + + /** + * 获取{@linkplain HttpClient}。 + * + * @return + */ + HttpClient getHttpClient(); } diff --git a/datagear-management/src/main/java/org/datagear/management/service/impl/DataSetEntityServiceImpl.java b/datagear-management/src/main/java/org/datagear/management/service/impl/DataSetEntityServiceImpl.java index d4418886..48c67f6c 100644 --- a/datagear-management/src/main/java/org/datagear/management/service/impl/DataSetEntityServiceImpl.java +++ b/datagear-management/src/main/java/org/datagear/management/service/impl/DataSetEntityServiceImpl.java @@ -12,12 +12,17 @@ import java.util.ArrayList; import java.util.List; import java.util.Map; +import org.apache.hc.client5.http.classic.HttpClient; import org.apache.ibatis.session.SqlSessionFactory; import org.datagear.analysis.DataSet; import org.datagear.analysis.DataSetParam; import org.datagear.analysis.DataSetProperty; import org.datagear.connection.ConnectionSource; +import org.datagear.management.domain.CsvFileDataSetEntity; +import org.datagear.management.domain.CsvValueDataSetEntity; import org.datagear.management.domain.DataSetEntity; +import org.datagear.management.domain.ExcelDataSetEntity; +import org.datagear.management.domain.HttpDataSetEntity; import org.datagear.management.domain.JsonFileDataSetEntity; import org.datagear.management.domain.JsonValueDataSetEntity; import org.datagear.management.domain.SchemaConnectionFactory; @@ -51,29 +56,35 @@ public class DataSetEntityServiceImpl extends AbstractMybatisDataPermissionEntit /** 数据集文件存储根目录 */ private File dataSetRootDirectory; + private HttpClient httpClient; + public DataSetEntityServiceImpl() { super(); } public DataSetEntityServiceImpl(SqlSessionFactory sqlSessionFactory, ConnectionSource connectionSource, - SchemaService schemaService, AuthorizationService authorizationService, File dataSetRootDirectory) + SchemaService schemaService, AuthorizationService authorizationService, File dataSetRootDirectory, + HttpClient httpClient) { super(sqlSessionFactory); this.connectionSource = connectionSource; this.schemaService = schemaService; this.authorizationService = authorizationService; setDataSetRootDirectory(dataSetRootDirectory); + this.httpClient = httpClient; } public DataSetEntityServiceImpl(SqlSessionTemplate sqlSessionTemplate, ConnectionSource connectionSource, - SchemaService schemaService, AuthorizationService authorizationService, File dataSetRootDirectory) + SchemaService schemaService, AuthorizationService authorizationService, File dataSetRootDirectory, + HttpClient httpClient) { super(sqlSessionTemplate); this.connectionSource = connectionSource; this.schemaService = schemaService; this.authorizationService = authorizationService; setDataSetRootDirectory(dataSetRootDirectory); + this.httpClient = httpClient; } public ConnectionSource getConnectionSource() @@ -116,6 +127,17 @@ public class DataSetEntityServiceImpl extends AbstractMybatisDataPermissionEntit this.dataSetRootDirectory = dataSetRootDirectory; } + @Override + public HttpClient getHttpClient() + { + return httpClient; + } + + public void setHttpClient(HttpClient httpClient) + { + this.httpClient = httpClient; + } + @Override public File getDataSetDirectory(String dataSetId) { @@ -167,6 +189,14 @@ public class DataSetEntityServiceImpl extends AbstractMybatisDataPermissionEntit success = addJsonValueDataSetEntity((JsonValueDataSetEntity) entity); else if (entity instanceof JsonFileDataSetEntity) success = addJsonFileDataSetEntity((JsonFileDataSetEntity) entity); + else if (entity instanceof ExcelDataSetEntity) + success = addExcelDataSetEntity((ExcelDataSetEntity) entity); + else if (entity instanceof CsvValueDataSetEntity) + success = addCsvValueDataSetEntity((CsvValueDataSetEntity) entity); + else if (entity instanceof CsvFileDataSetEntity) + success = addCsvFileDataSetEntity((CsvFileDataSetEntity) entity); + else if (entity instanceof HttpDataSetEntity) + success = addHttpDataSetEntity((HttpDataSetEntity) entity); } if (success) @@ -199,6 +229,38 @@ public class DataSetEntityServiceImpl extends AbstractMybatisDataPermissionEntit return (updateMybatis("insertJsonFileDataSetEntity", params) > 0); } + protected boolean addExcelDataSetEntity(ExcelDataSetEntity entity) + { + Map params = buildParamMapWithIdentifierQuoteParameter(); + params.put("entity", entity); + + return (updateMybatis("insertExcelDataSetEntity", params) > 0); + } + + protected boolean addCsvValueDataSetEntity(CsvValueDataSetEntity entity) + { + Map params = buildParamMapWithIdentifierQuoteParameter(); + params.put("entity", entity); + + return (updateMybatis("insertCsvValueDataSetEntity", params) > 0); + } + + protected boolean addCsvFileDataSetEntity(CsvFileDataSetEntity entity) + { + Map params = buildParamMapWithIdentifierQuoteParameter(); + params.put("entity", entity); + + return (updateMybatis("insertCsvFileDataSetEntity", params) > 0); + } + + protected boolean addHttpDataSetEntity(HttpDataSetEntity entity) + { + Map params = buildParamMapWithIdentifierQuoteParameter(); + params.put("entity", entity); + + return (updateMybatis("insertHttpDataSetEntity", params) > 0); + } + @Override protected boolean update(DataSetEntity entity, Map params) { @@ -215,6 +277,14 @@ public class DataSetEntityServiceImpl extends AbstractMybatisDataPermissionEntit success = updateJsonValueDataSetEntity((JsonValueDataSetEntity) entity); else if (entity instanceof JsonFileDataSetEntity) success = updateJsonFileDataSetEntity((JsonFileDataSetEntity) entity); + else if (entity instanceof ExcelDataSetEntity) + success = updateExcelDataSetEntity((ExcelDataSetEntity) entity); + else if (entity instanceof CsvValueDataSetEntity) + success = updateCsvValueDataSetEntity((CsvValueDataSetEntity) entity); + else if (entity instanceof CsvFileDataSetEntity) + success = updateCsvFileDataSetEntity((CsvFileDataSetEntity) entity); + else if (entity instanceof HttpDataSetEntity) + success = updateHttpDataSetEntity((HttpDataSetEntity) entity); } if (success) @@ -247,6 +317,38 @@ public class DataSetEntityServiceImpl extends AbstractMybatisDataPermissionEntit return (updateMybatis("updateJsonFileDataSetEntity", params) > 0); } + protected boolean updateExcelDataSetEntity(ExcelDataSetEntity entity) + { + Map params = buildParamMapWithIdentifierQuoteParameter(); + params.put("entity", entity); + + return (updateMybatis("updateExcelDataSetEntity", params) > 0); + } + + protected boolean updateCsvValueDataSetEntity(CsvValueDataSetEntity entity) + { + Map params = buildParamMapWithIdentifierQuoteParameter(); + params.put("entity", entity); + + return (updateMybatis("updateCsvValueDataSetEntity", params) > 0); + } + + protected boolean updateCsvFileDataSetEntity(CsvFileDataSetEntity entity) + { + Map params = buildParamMapWithIdentifierQuoteParameter(); + params.put("entity", entity); + + return (updateMybatis("updateCsvFileDataSetEntity", params) > 0); + } + + protected boolean updateHttpDataSetEntity(HttpDataSetEntity entity) + { + Map params = buildParamMapWithIdentifierQuoteParameter(); + params.put("entity", entity); + + return (updateMybatis("updateHttpDataSetEntity", params) > 0); + } + @Override public String getResourceType() { @@ -291,6 +393,14 @@ public class DataSetEntityServiceImpl extends AbstractMybatisDataPermissionEntit obj = getJsonValueDataSetEntityById(obj.getId()); else if (DataSetEntity.DATA_SET_TYPE_JsonFile.equals(obj.getDataSetType())) obj = getJsonFileDataSetEntityById(obj.getId()); + else if (DataSetEntity.DATA_SET_TYPE_Excel.equals(obj.getDataSetType())) + obj = getExcelDataSetEntityById(obj.getId()); + else if (DataSetEntity.DATA_SET_TYPE_CsvValue.equals(obj.getDataSetType())) + obj = getCsvValueDataSetEntityById(obj.getId()); + else if (DataSetEntity.DATA_SET_TYPE_CsvFile.equals(obj.getDataSetType())) + obj = getCsvFileDataSetEntityById(obj.getId()); + else if (DataSetEntity.DATA_SET_TYPE_Http.equals(obj.getDataSetType())) + obj = getHttpDataSetEntityById(obj.getId()); if (obj == null) return null; @@ -342,6 +452,55 @@ public class DataSetEntityServiceImpl extends AbstractMybatisDataPermissionEntit return entity; } + protected ExcelDataSetEntity getExcelDataSetEntityById(String id) + { + Map params = buildParamMapWithIdentifierQuoteParameter(); + params.put("id", id); + + ExcelDataSetEntity entity = selectOneMybatis("getExcelDataSetEntityById", params); + + if (entity != null) + entity.setDirectory(getDataSetDirectory(id)); + + return entity; + } + + protected CsvValueDataSetEntity getCsvValueDataSetEntityById(String id) + { + Map params = buildParamMapWithIdentifierQuoteParameter(); + params.put("id", id); + + CsvValueDataSetEntity entity = selectOneMybatis("getCsvValueDataSetEntityById", params); + + return entity; + } + + protected CsvFileDataSetEntity getCsvFileDataSetEntityById(String id) + { + Map params = buildParamMapWithIdentifierQuoteParameter(); + params.put("id", id); + + CsvFileDataSetEntity entity = selectOneMybatis("getCsvFileDataSetEntityById", params); + + if (entity != null) + entity.setDirectory(getDataSetDirectory(id)); + + return entity; + } + + protected HttpDataSetEntity getHttpDataSetEntityById(String id) + { + Map params = buildParamMapWithIdentifierQuoteParameter(); + params.put("id", id); + + HttpDataSetEntity entity = selectOneMybatis("getHttpDataSetEntityById", params); + + if (entity != null) + entity.setHttpClient(this.httpClient); + + return entity; + } + @Override protected void addDataPermissionParameters(Map params, User user) { diff --git a/datagear-management/src/main/resources/org/datagear/management/ddl/datagear.sql b/datagear-management/src/main/resources/org/datagear/management/ddl/datagear.sql index 502fffba..cb44ff61 100644 --- a/datagear-management/src/main/resources/org/datagear/management/ddl/datagear.sql +++ b/datagear-management/src/main/resources/org/datagear/management/ddl/datagear.sql @@ -506,3 +506,76 @@ ALTER TABLE DATAGEAR_DATA_SET_JSON_FILE ADD FOREIGN KEY (DS_ID) REFERENCES DATAG --version[1.11.1], DO NOT EDIT THIS LINE! ----------------------------------------- + +----------------------------------------- +--version[1.12.0], DO NOT EDIT THIS LINE! +----------------------------------------- + +--2020-08-28 +--JSON文件数据集添加编码字段 +ALTER TABLE DATAGEAR_DATA_SET_JSON_FILE ADD COLUMN DS_FILE_ENCODING VARCHAR(50); + +--2020-08-28 +--Excel文件数据集 +CREATE TABLE DATAGEAR_DATA_SET_EXCEL +( + DS_ID VARCHAR(50) NOT NULL, + DS_FILE_NAME VARCHAR(100) NOT NULL, + DS_DISPLAY_NAME VARCHAR(100) NOT NULL, + DS_SHEET_INDEX INTEGER, + DS_NAME_ROW INTEGER, + DS_DATA_ROW_EXP VARCHAR(100), + DS_DATA_COLUMN_EXP VARCHAR(100), + DS_FORCE_XLS VARCHAR(10), + PRIMARY KEY (DS_ID) +); + +ALTER TABLE DATAGEAR_DATA_SET_EXCEL ADD FOREIGN KEY (DS_ID) REFERENCES DATAGEAR_DATA_SET (DS_ID) ON DELETE CASCADE; + +--2020-08-31 +--CSV值数据集 +CREATE TABLE DATAGEAR_DATA_SET_CSV_VALUE +( + DS_ID VARCHAR(50) NOT NULL, + DS_VALUE VARCHAR(10000) NOT NULL, + DS_NAME_ROW INTEGER, + PRIMARY KEY (DS_ID) +); + +ALTER TABLE DATAGEAR_DATA_SET_CSV_VALUE ADD FOREIGN KEY (DS_ID) REFERENCES DATAGEAR_DATA_SET (DS_ID) ON DELETE CASCADE; + +--2020-08-31 +--CSV文件数据集 +CREATE TABLE DATAGEAR_DATA_SET_CSV_FILE +( + DS_ID VARCHAR(50) NOT NULL, + DS_FILE_NAME VARCHAR(100) NOT NULL, + DS_DISPLAY_NAME VARCHAR(100) NOT NULL, + DS_FILE_ENCODING VARCHAR(50), + DS_NAME_ROW INTEGER, + PRIMARY KEY (DS_ID) +); + +ALTER TABLE DATAGEAR_DATA_SET_CSV_FILE ADD FOREIGN KEY (DS_ID) REFERENCES DATAGEAR_DATA_SET (DS_ID) ON DELETE CASCADE; + +--2020-09-03 +--JSON文件数据集添加数据JSON路径字段 +ALTER TABLE DATAGEAR_DATA_SET_JSON_FILE ADD COLUMN DS_DATA_JSON_PATH VARCHAR(200); + +--2020-09-05 +--HTTP数据集 +CREATE TABLE DATAGEAR_DATA_SET_HTTP +( + DS_ID VARCHAR(50) NOT NULL, + DS_URI VARCHAR(1000) NOT NULL, + DS_HEADER_CONTENT VARCHAR(5000), + DS_RQT_METHOD VARCHAR(50), + DS_RQT_CONTENT_TYPE VARCHAR(100), + DS_RQT_CONTENT_CHARSET VARCHAR(100), + DS_RQT_CONTENT varchar(10000), + DS_RPS_CONTENT_TYPE VARCHAR(100), + DS_RPS_DATA_JSON_PATH VARCHAR(200), + PRIMARY KEY (DS_ID) +); + +ALTER TABLE DATAGEAR_DATA_SET_HTTP ADD FOREIGN KEY (DS_ID) REFERENCES DATAGEAR_DATA_SET (DS_ID) ON DELETE CASCADE; diff --git a/datagear-management/src/main/resources/org/datagear/management/mapper/DataSetEntityMapper.xml b/datagear-management/src/main/resources/org/datagear/management/mapper/DataSetEntityMapper.xml index fa6ffd7e..8acde195 100644 --- a/datagear-management/src/main/resources/org/datagear/management/mapper/DataSetEntityMapper.xml +++ b/datagear-management/src/main/resources/org/datagear/management/mapper/DataSetEntityMapper.xml @@ -39,11 +39,59 @@ INSERT INTO DATAGEAR_DATA_SET_JSON_FILE ( - DS_ID, DS_FILE_NAME, DS_DISPLAY_NAME + DS_ID, DS_FILE_NAME, DS_FILE_ENCODING, DS_DISPLAY_NAME, DS_DATA_JSON_PATH ) VALUES ( - #{entity.id}, #{entity.fileName}, #{entity.displayName} + #{entity.id}, #{entity.fileName}, #{entity.encoding}, #{entity.displayName}, #{entity.dataJsonPath} + ) + + + + INSERT INTO DATAGEAR_DATA_SET_EXCEL + ( + DS_ID, DS_FILE_NAME, DS_DISPLAY_NAME, DS_SHEET_INDEX, DS_NAME_ROW, + DS_DATA_ROW_EXP, DS_DATA_COLUMN_EXP, DS_FORCE_XLS + ) + VALUES + ( + #{entity.id}, #{entity.fileName}, #{entity.displayName}, #{entity.sheetIndex}, #{entity.nameRow}, + #{entity.dataRowExp}, #{entity.dataColumnExp}, #{entity.forceXls} + ) + + + + INSERT INTO DATAGEAR_DATA_SET_CSV_VALUE + ( + DS_ID, DS_VALUE, DS_NAME_ROW + ) + VALUES + ( + #{entity.id}, #{entity.value}, #{entity.nameRow} + ) + + + + INSERT INTO DATAGEAR_DATA_SET_CSV_FILE + ( + DS_ID, DS_FILE_NAME, DS_FILE_ENCODING, DS_DISPLAY_NAME, DS_NAME_ROW + ) + VALUES + ( + #{entity.id}, #{entity.fileName}, #{entity.encoding}, #{entity.displayName}, #{entity.nameRow} + ) + + + + INSERT INTO DATAGEAR_DATA_SET_HTTP + ( + DS_ID, DS_URI, DS_HEADER_CONTENT, DS_RQT_METHOD, DS_RQT_CONTENT_TYPE, + DS_RQT_CONTENT_CHARSET, DS_RQT_CONTENT, DS_RPS_CONTENT_TYPE, DS_RPS_DATA_JSON_PATH + ) + VALUES + ( + #{entity.id}, #{entity.uri}, #{entity.headerContent}, #{entity.requestMethod}, #{entity.requestContentType}, + #{entity.requestContentCharset}, #{entity.requestContent}, #{entity.responseContentType}, #{entity.responseDataJsonPath} ) @@ -103,7 +151,54 @@ UPDATE DATAGEAR_DATA_SET_JSON_FILE SET DS_FILE_NAME = #{entity.fileName}, - DS_DISPLAY_NAME = #{entity.displayName} + DS_FILE_ENCODING = #{entity.encoding}, + DS_DISPLAY_NAME = #{entity.displayName}, + DS_DATA_JSON_PATH = #{entity.dataJsonPath} + WHERE + DS_ID = #{entity.id} + + + + UPDATE DATAGEAR_DATA_SET_EXCEL SET + DS_FILE_NAME = #{entity.fileName}, + DS_DISPLAY_NAME = #{entity.displayName}, + DS_SHEET_INDEX = #{entity.sheetIndex}, + DS_NAME_ROW = #{entity.nameRow}, + DS_DATA_ROW_EXP = #{entity.dataRowExp}, + DS_DATA_COLUMN_EXP = #{entity.dataColumnExp}, + DS_FORCE_XLS = #{entity.forceXls} + WHERE + DS_ID = #{entity.id} + + + + UPDATE DATAGEAR_DATA_SET_CSV_VALUE SET + DS_VALUE = #{entity.value}, + DS_NAME_ROW = #{entity.nameRow} + WHERE + DS_ID = #{entity.id} + + + + UPDATE DATAGEAR_DATA_SET_CSV_FILE SET + DS_FILE_NAME = #{entity.fileName}, + DS_FILE_ENCODING = #{entity.encoding}, + DS_DISPLAY_NAME = #{entity.displayName}, + DS_NAME_ROW = #{entity.nameRow} + WHERE + DS_ID = #{entity.id} + + + + UPDATE DATAGEAR_DATA_SET_HTTP SET + DS_URI = #{entity.uri}, + DS_HEADER_CONTENT = #{entity.headerContent}, + DS_RQT_METHOD = #{entity.requestMethod}, + DS_RQT_CONTENT_TYPE = #{entity.requestContentType}, + DS_RQT_CONTENT_CHARSET = #{entity.requestContentCharset}, + DS_RQT_CONTENT = #{entity.requestContent}, + DS_RPS_CONTENT_TYPE = #{entity.responseContentType}, + DS_RPS_DATA_JSON_PATH = #{entity.responseDataJsonPath} WHERE DS_ID = #{entity.id} @@ -169,7 +264,9 @@ SELECT T1.*, T2.DS_FILE_NAME AS ${_iq_}fileName${_iq_}, - T2.DS_DISPLAY_NAME AS ${_iq_}displayName${_iq_} + T2.DS_FILE_ENCODING AS ${_iq_}encoding${_iq_}, + T2.DS_DISPLAY_NAME AS ${_iq_}displayName${_iq_}, + T2.DS_DATA_JSON_PATH AS ${_iq_}dataJsonPath${_iq_} FROM (SELECT * FROM () T0 WHERE T0.${_iq_}id${_iq_} = #{id}) T1 INNER JOIN @@ -178,6 +275,72 @@ T1.${_iq_}id${_iq_} = T2.DS_ID + + + + + + + + ").attr("name", dataSetParam.name) .attr("value", (value || "")).appendTo($parent); + if(chartForm.disableDateAwareInputAutocomplete) + $input.attr("autocomplete", "off"); + if((dataSetParam.required+"") == "true") $input.attr("dg-validation-required", "true"); @@ -435,6 +441,9 @@ var $input = $("").attr("name", dataSetParam.name) .attr("value", (value || "")).appendTo($parent); + if(chartForm.disableDateAwareInputAutocomplete) + $input.attr("autocomplete", "off"); + if((dataSetParam.required+"") == "true") $input.attr("dg-validation-required", "true"); @@ -472,6 +481,9 @@ var $input = $("").attr("name", dataSetParam.name) .attr("value", (value || "")).appendTo($parent); + if(chartForm.disableDateAwareInputAutocomplete) + $input.attr("autocomplete", "off"); + if((dataSetParam.required+"") == "true") $input.attr("dg-validation-required", "true"); @@ -626,6 +638,9 @@ chartForm.datetimepicker = function($input, datetimepickerOptions, chartTheme) { + //在这里检查并重写,避免依赖加载顺序 + global.overwriteDateFormatter_parseDate(); + if(chartForm._datetimepickerSetLocale !== true) { $.datetimepicker.setLocale('zh'); @@ -652,6 +667,45 @@ }, datetimepickerOptions); + //初始化为年份选择器 + var isOnlySelectYear = ("Y" == datetimepickerOptions.format || "y" == datetimepickerOptions.format); + if(isOnlySelectYear) + { + //显示确定按钮,用于直接选中默认年份 + datetimepickerOptions.showApplyButton = true; + datetimepickerOptions.onGenerate = function(currentValue,$input) + { + var yearPickerInited = $(this).attr("yearPickerInited"); + if(!yearPickerInited) + { + $(this).attr("yearPickerInited", "yes"); + + $(".xdsoft_prev", this).hide(); + $(".xdsoft_today_button", this).hide(); + $(".xdsoft_month", this).hide(); + $(".xdsoft_next", this).hide(); + $(".xdsoft_calendar", this).hide(); + + $(".xdsoft_save_selected", this).removeClass("blue-gradient-button") + .addClass("xdsoft_save_selected_year ui-button ui-corner-all").html(chartForm.labels.confirm); + } + }; + datetimepickerOptions.onShow = function(currentValue,$input) + { + if(!$input.val()) + { + //这样可以直接点击【确定】按钮选择默认年份 + this.setOptions({value: currentValue}); + $input.val(""); + } + }; + datetimepickerOptions.onChangeYear = function(currentValue,$input) + { + this.setOptions({value: currentValue}); + $(".xdsoft_save_selected", this).click(); + }; + } + $input.datetimepicker(datetimepickerOptions); }; @@ -738,6 +792,14 @@ +" box-shadow: none;" +" -webkit-box-shadow: none;" +"} " + +parentSelector + " .xdsoft_datetimepicker .xdsoft_save_selected.xdsoft_save_selected_year{" + +" color: "+color+";" + +" background: "+bgColor+";" + +" border: 1px solid "+borderColor+" !important;" + +"} " + +parentSelector + " .xdsoft_datetimepicker .xdsoft_save_selected.xdsoft_save_selected_year:hover{" + +" background: "+hoverColor+";" + +"} " ; global.chartFactory.createStyleSheet(styleId, cssText); @@ -1280,4 +1342,139 @@ return ($panel.length == 0 || $panel.is(":hidden")); }; }) +(this); + +(function(global) +{ + global.overwriteDateFormatter_parseDate = function() + { + if(global._overwriteDateFormatter_parseDate == true) + return; + + global._overwriteDateFormatter_parseDate = true; + + //重写datetimepicker的DateFormatter.prototype.parseDate函数, + //解决当options配置为仅选择年份({format:'Y'})时,选择完毕后输入框blur后,年份会被重置的的BUG + DateFormatter.prototype.parseDate = function(e, r) { + var n, a, u, i, s, o, c, f, l, h, d = this, g = !1, m = !1, p = d.dateSettings, y = { + date: null, + year: null, + month: null, + day: null, + hour: 0, + min: 0, + sec: 0 + }; + if (!e) + return null; + if (e instanceof Date) + return e; + if ("U" === r) + return u = parseInt(e), + u ? new Date(1e3 * u) : e; + switch (typeof e) { + case "number": + return new Date(e); + case "string": + break; + default: + return null + } + if (n = r.match(d.validParts), + !n || 0 === n.length) + throw new Error("Invalid date format definition."); + for (a = e.replace(d.separators, "\x00").split("\x00"), + u = 0; u < a.length; u++) + switch (i = a[u], + s = parseInt(i), + n[u]) { + case "y": + case "Y": + if (!s) + return null; + l = i.length, + y.year = 2 === l ? parseInt((70 > s ? "20" : "19") + i) : s, + g = !0; + break; + case "m": + case "n": + case "M": + case "F": + if (isNaN(s)) { + if (o = d.getMonth(i), + !(o > 0)) + return null; + y.month = o + } else { + if (!(s >= 1 && 12 >= s)) + return null; + y.month = s + } + g = !0; + break; + case "d": + case "j": + if (!(s >= 1 && 31 >= s)) + return null; + y.day = s, + g = !0; + break; + case "g": + case "h": + if (c = n.indexOf("a") > -1 ? n.indexOf("a") : n.indexOf("A") > -1 ? n.indexOf("A") : -1, + h = a[c], + c > -1) + f = t(h, p.meridiem[0]) ? 0 : t(h, p.meridiem[1]) ? 12 : -1, + s >= 1 && 12 >= s && f > -1 ? y.hour = s + f - 1 : s >= 0 && 23 >= s && (y.hour = s); + else { + if (!(s >= 0 && 23 >= s)) + return null; + y.hour = s + } + m = !0; + break; + case "G": + case "H": + if (!(s >= 0 && 23 >= s)) + return null; + y.hour = s, + m = !0; + break; + case "i": + if (!(s >= 0 && 59 >= s)) + return null; + y.min = s, + m = !0; + break; + case "s": + if (!(s >= 0 && 59 >= s)) + return null; + y.sec = s, + m = !0 + } + + //----添加的内容 + //如果选择了年份,检查并设置月份、日的初值,使下面的: + //if (g === !0 && y.year && y.month && y.day) + //逻辑可以执行到 + if (g === !0 && y.year) + { + if(!y.month) + y.month = 1; + if(!y.day) + y.day = 1; + } + //----添加的内容 + + if (g === !0 && y.year && y.month && y.day) + y.date = new Date(y.year,y.month - 1,y.day,y.hour,y.min,y.sec,0); + else { + if (m !== !0) + return null; + y.date = new Date(0,0,0,y.hour,y.min,y.sec,0) + } + return y.date + }; + } +}) (this); \ No newline at end of file diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/static/script/datagear-chartSupport.js b/datagear-web/src/main/resources/org/datagear/web/webapp/static/script/datagear-chartSupport.js index 17dcdd33..bdd09c82 100644 --- a/datagear-web/src/main/resources/org/datagear/web/webapp/static/script/datagear-chartSupport.js +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/static/script/datagear-chartSupport.js @@ -133,6 +133,20 @@ return re; }; + /** + * 为数组追加单个元素、数组 + */ + chartSupport.appendElement = function(array, eles) + { + if($.isArray(eles)) + { + for(var i=0; i dataIndex) - return { chartDataSetIndex: ele.chartDataSetIndex, resultDataIndex: dataIndex - ele.dataIndexStart + ele.resultDataIndexStart }; + { + var originalDataIndex= + { + chartDataSetIndex: ele.chartDataSetIndex + }; + + var indexInfo = ele.resultDataIndexInfo; + + if(indexInfo.type == "ordinal") + { + originalDataIndex.resultDataIndex = dataIndex - ele.dataIndexStart + indexInfo.start; + } + else if(indexInfo.type == "identical") + { + if(indexInfo.value != null) + originalDataIndex.resultDataIndex = indexInfo.value; + else + originalDataIndex.resultDataIndex = { start: indexInfo.start , end: indexInfo.end }; + } + else + throw new Error("Unknown type ["+indexInfo.type+"]"); + + return originalDataIndex; + } } return undefined; }; /** - * 清除图表数据范围的原始信息。 + * 清除图表所有数据范围对应的的原始数据索引对象。 * * @param chart */ - chartSupport.clearChartOriginalDataInfoRange = function(chart) + chartSupport.clearChartOriginalDataIndexForRange = function(chart) { - chart.extValue("chartOriginalDataInfoRange", null); + chart.extValue("chartOriginalDataIndexForRange", null); }; - chartSupport.getChartOriginalDataInfoForRangeEcharts = function(chart, echartsEventParams) - { - if(!echartsEventParams) - return undefined; - - return this.getChartOriginalDataInfoForRange(chart, echartsEventParams.seriesIndex, echartsEventParams.dataIndex); - } - chartSupport.bindChartEventHandlerDelegationEcharts = function(chart, eventType, chartEventHanlder, chartEventDataSetter) { var echartsEventHandler = function(params) @@ -419,36 +533,6 @@ } }; - chartSupport.setChartEventOriginalDataForEchartsRange = function(chart, chartEvent, echartsEventParams) - { - var originDataInfo = this.getChartOriginalDataInfoForRangeEcharts(chart, echartsEventParams); - this.setChartEventOriginalDataByInfo(chart, chartEvent, originDataInfo); - }; - - chartSupport.setChartEventOriginalDataForChartData = function(chart, chartEvent, chartData) - { - var originDataInfo = chartSupport.getChartOriginalDataInfo(chartData); - this.setChartEventOriginalDataByInfo(chart, chartEvent, originDataInfo); - }; - - chartSupport.setChartEventOriginalDataByInfo = function(chart, chartEvent, originDataInfo) - { - if(!originDataInfo) - { - chart.eventOriginalData(chartEvent, null); - chart.eventOriginalChartDataSetIndex(chartEvent, null); - chart.eventOriginalResultDataIndex(chartEvent, null); - return; - } - - var resultDatas = chart.resultDatas(chart.resultAt(chart.getUpdateResults(), originDataInfo.chartDataSetIndex)); - var originalData = resultDatas[originDataInfo.resultDataIndex]; - - chart.eventOriginalData(chartEvent, originalData); - chart.eventOriginalChartDataSetIndex(chartEvent, originDataInfo.chartDataSetIndex); - chart.eventOriginalResultDataIndex(chartEvent, originDataInfo.resultDataIndex); - }; - //折线图 chartSupport.lineRender = function(chart, nameSign, valueSign, options) @@ -507,7 +591,7 @@ var isCategory = (initOptions.xAxis.type == "category"); - chartSupport.clearChartOriginalDataInfoRange(chart); + chartSupport.clearChartOriginalDataIndexForRange(chart); var legendData = []; var xAxisData = []; @@ -540,7 +624,7 @@ legendData.push(legendName); series.push(mySeries); - chartSupport.setChartOriginalDataInfoRange(chart, series.length-1, 0, data.length, i); + chartSupport.setChartOriginalDataIndexForRange(chart, series.length-1, 0, data.length, i); //类目轴需要设置data,不然图表刷新数据有变化时,类目轴坐标不能自动更新 if(isCategory) @@ -668,7 +752,7 @@ var isCategory = ((horizontal ? initOptions.yAxis.type : initOptions.xAxis.type) == "category"); - chartSupport.clearChartOriginalDataInfoRange(chart); + chartSupport.clearChartOriginalDataIndexForRange(chart); var legendData = []; var axisData = []; @@ -700,7 +784,7 @@ legendData.push(legendName); series.push(mySeries); - chartSupport.setChartOriginalDataInfoRange(chart, series.length-1, 0, data.length, i); + chartSupport.setChartOriginalDataIndexForRange(chart, series.length-1, 0, data.length, i); //类目轴需要设置data,不然图表刷新数据有变化时,类目轴坐标不能自动更新 if(isCategory) @@ -810,7 +894,7 @@ var initOptions= chartSupport.initOptions(chart); var chartDataSets = chart.chartDataSetsNonNull(); - chartSupport.clearChartOriginalDataInfoRange(chart); + chartSupport.clearChartOriginalDataIndexForRange(chart); var legendData = []; var seriesName = ""; @@ -832,7 +916,7 @@ seriesName = dataSetName; seriesData = seriesData.concat(nvv); - chartSupport.setChartOriginalDataInfoRange(chart, 0, seriesData.length - nvv.length, seriesData.length, i); + chartSupport.setChartOriginalDataIndexForRange(chart, 0, seriesData.length - nvv.length, seriesData.length, i); } var series = [ chartSupport.optionsSeries(initOptions, 0, {name: seriesName, data: seriesData}) ]; @@ -913,7 +997,7 @@ var chartDataSet = chart.chartDataSetFirst(); var result = chart.resultFirst(results); - chartSupport.clearChartOriginalDataInfoRange(chart); + chartSupport.clearChartOriginalDataIndexForRange(chart); var seriesName = chart.chartDataSetName(chartDataSet); @@ -927,7 +1011,7 @@ var seriesData = [ { value: value, name: chart.dataSetPropertyLabel(vp), min: min, max: max } ]; - chartSupport.setChartOriginalDataInfoRange(chart, 0, 0, seriesData.length, 0); + chartSupport.setChartOriginalDataIndexForRange(chart, 0, 0, seriesData.length, 0); var options = { series : [ { name: seriesName, min: min, max: max, data: seriesData } ]}; @@ -1030,7 +1114,7 @@ var isCategory = (initOptions.xAxis.type == "category"); - chartSupport.clearChartOriginalDataInfoRange(chart); + chartSupport.clearChartOriginalDataIndexForRange(chart); var legendData = []; var xAxisData = []; @@ -1070,7 +1154,7 @@ legendData.push(legendName); series.push(mySeries); - chartSupport.setChartOriginalDataInfoRange(chart, series.length-1, 0, data.length, i); + chartSupport.setChartOriginalDataIndexForRange(chart, series.length-1, 0, data.length, i); //类目轴需要设置data,不然图表刷新数据有变化时,类目轴坐标不能自动更新 if(isCategory) @@ -1197,7 +1281,7 @@ var initOptions= chartSupport.initOptions(chart); var chartDataSets = chart.chartDataSetsNonNull(); - chartSupport.clearChartOriginalDataInfoRange(chart); + chartSupport.clearChartOriginalDataIndexForRange(chart); var legendData = []; var series = []; @@ -1231,7 +1315,7 @@ legendData.push(dataSetName); series.push(mySeries); - chartSupport.setChartOriginalDataInfoRange(chart, series.length-1, 0, data.length, i); + chartSupport.setChartOriginalDataIndexForRange(chart, series.length-1, 0, data.length, i); } if(min != null && max != null && max <= min) @@ -1387,11 +1471,11 @@ var initOptions= chartSupport.initOptions(chart); var chartDataSets = chart.chartDataSetsNonNull(); - chartSupport.clearChartOriginalDataInfoRange(chart); + chartSupport.clearChartOriginalDataIndexForRange(chart); var legendData = []; var indicatorData = []; - var series = []; + var seriesData = []; for(var i=0; i 0 ? dnpv[0] : []); - var dmp = chart.dataSetPropertiesOfSign(chartDataSet, signNameMap.max); - var dmpv = chart.resultRowArrays(result, dmp, 0, 1); - dmpv = (dmpv.length > 0 ? dmpv[0] : []); - - var indicatorLen = Math.min(dnp.length, dmp.length); - - for(var j=0; j 0 ? npv[0] : []); + + var mp = chart.dataSetPropertiesOfSign(chartDataSet, signNameMap.max); + var mpv = chart.resultRowArrays(result, mp, 0, 1); + mpv = (mpv.length > 0 ? mpv[0] : []); + + var indicatorLen = Math.min(np.length, mp.length); + + for(var j=0; j 0 ? updateOptions.series[0].map : undefined); + + var presetMap = chart.extValue("presetMap"); if(!map) { var currentMap = chart.map(); - var presetMap = chart.extValue("presetMap"); //通过chart.map(...)设置了新的地图 if(currentMap && currentMap != presetMap) @@ -1694,6 +1842,21 @@ } } + //如果更新了地图,则要重置缩放比例和中心位置,避免出现某些地图无法显示的情况 + if(map && map != presetMap) + { + if(isGeo) + { + updateOptions.geo.center = null; + updateOptions.geo.zoom = 1;//此项非必须 + } + else + { + updateOptions.series[0].center = null; + updateOptions.series[0].zoom = 1;//此项非必须 + } + } + //没有更新地图、或者更新的地图已注册 if(!map || chart.echartsMapRegistered(map)) { @@ -1763,7 +1926,7 @@ var initOptions= chartSupport.initOptions(chart); var chartDataSets = chart.chartDataSetsNonNull(); - chartSupport.clearChartOriginalDataInfoRange(chart); + chartSupport.clearChartOriginalDataIndexForRange(chart); var min = undefined; var max = undefined; @@ -1797,7 +1960,7 @@ seriesData = seriesData.concat(nvv); - chartSupport.setChartOriginalDataInfoRange(chart, 0, seriesData.length - nvv.length, seriesData.length, i); + chartSupport.setChartOriginalDataIndexForRange(chart, 0, seriesData.length - nvv.length, seriesData.length, i); if(nvv && nvv.length) { @@ -1910,7 +2073,7 @@ var initOptions= chartSupport.initOptions(chart); var chartDataSets = chart.chartDataSetsNonNull(); - chartSupport.clearChartOriginalDataInfoRange(chart); + chartSupport.clearChartOriginalDataIndexForRange(chart); var legendData = []; var series = []; @@ -1958,7 +2121,7 @@ legendData[i] = dataSetName; series[i] = chartSupport.optionsSeries(initOptions, i, { name: dataSetName, data: data }); - chartSupport.setChartOriginalDataInfoRange(chart, series.length-1, 0, data.length, i); + chartSupport.setChartOriginalDataIndexForRange(chart, series.length-1, 0, data.length, i); } if(min != null && max != null && max <= min) @@ -2080,7 +2243,7 @@ var initOptions= chartSupport.initOptions(chart); var chartDataSets = chart.chartDataSetsNonNull(); - chartSupport.clearChartOriginalDataInfoRange(chart); + chartSupport.clearChartOriginalDataIndexForRange(chart); var legendData = []; var seriesName = ""; @@ -2183,7 +2346,7 @@ //新插入 if(sidx == seriesData.length - 1 && seriesData[seriesData.length - 1] === sd) { - chartSupport.setChartOriginalDataInfo(sd, i, j); + chartSupport.chartDataOriginalDataIndex(sd, i, j); } var tidx = chartSupport.appendDistinct(seriesData, td, (tip ? "id" : "name")); @@ -2191,7 +2354,7 @@ //新插入 if(tidx == seriesData.length - 1 && seriesData[seriesData.length - 1] === td) { - chartSupport.setChartOriginalDataInfo(td, i, j); + chartSupport.chartDataOriginalDataIndex(td, i, j); } //如果使用id值表示关系,对于数值型id,echarts会误当做数据索引,所以这里直接使用数据索引 @@ -2199,7 +2362,7 @@ link.source = sidx; link.target = tidx; - chartSupport.setChartOriginalDataInfo(link, i, j); + chartSupport.chartDataOriginalDataIndex(link, i, j); seriesLinks.push(link); } @@ -2364,7 +2527,7 @@ var initOptions= chartSupport.initOptions(chart); var chartDataSets = chart.chartDataSetsNonNull(); - chartSupport.clearChartOriginalDataInfoRange(chart); + chartSupport.clearChartOriginalDataIndexForRange(chart); var legendData = []; var series = []; @@ -2383,7 +2546,8 @@ chart.dataSetPropertyOfSign(chartDataSet, signNameMap.max) ]); - chartSupport.setChartOriginalDataInfo(data, i); + for(var j=0; js?"20":"19")+i):s,g=!0;break;case"m":case"n":case"M":case"F":if(isNaN(s)){if(o=d.getMonth(i),!(o>0))return null;y.month=o}else{if(!(s>=1&&12>=s))return null;y.month=s}g=!0;break;case"d":case"j":if(!(s>=1&&31>=s))return null;y.day=s,g=!0;break;case"g":case"h":if(c=n.indexOf("a")>-1?n.indexOf("a"):n.indexOf("A")>-1?n.indexOf("A"):-1,h=a[c],c>-1)f=t(h,p.meridiem[0])?0:t(h,p.meridiem[1])?12:-1,s>=1&&12>=s&&f>-1?y.hour=s+f-1:s>=0&&23>=s&&(y.hour=s);else{if(!(s>=0&&23>=s))return null;y.hour=s}m=!0;break;case"G":case"H":if(!(s>=0&&23>=s))return null;y.hour=s,m=!0;break;case"i":if(!(s>=0&&59>=s))return null;y.min=s,m=!0;break;case"s":if(!(s>=0&&59>=s))return null;y.sec=s,m=!0}if(g===!0&&y.year&&y.month&&y.day)y.date=new Date(y.year,y.month-1,y.day,y.hour,y.min,y.sec,0);else{if(m!==!0)return null;y.date=new Date(0,0,0,y.hour,y.min,y.sec,0)}return y.date},guessDate:function(t,e){if("string"!=typeof t)return t;var r,n,a,u,i,s,o=this,c=t.replace(o.separators,"\x00").split("\x00"),f=/^[djmn]/g,l=e.match(o.validParts),h=new Date,d=0;if(!f.test(l[0]))return t;for(a=0;ar?r:4,n=parseInt(4>r?n.toString().substr(0,4-r)+i:i.substr(0,4)),!n)return null;h.setFullYear(n);break;case 3:h.setHours(s);break;case 4:h.setMinutes(s);break;case 5:h.setSeconds(s)}u=i.substr(d),u.length>0&&c.splice(a+1,0,u)}return h},parseFormat:function(t,r){var n,a=this,s=a.dateSettings,o=/\\?(.?)/gi,c=function(t,e){return n[t]?n[t]():e};return n={d:function(){return e(n.j(),2)},D:function(){return s.daysShort[n.w()]},j:function(){return r.getDate()},l:function(){return s.days[n.w()]},N:function(){return n.w()||7},w:function(){return r.getDay()},z:function(){var t=new Date(n.Y(),n.n()-1,n.j()),e=new Date(n.Y(),0,1);return Math.round((t-e)/u)},W:function(){var t=new Date(n.Y(),n.n()-1,n.j()-n.N()+3),r=new Date(t.getFullYear(),0,4);return e(1+Math.round((t-r)/u/7),2)},F:function(){return s.months[r.getMonth()]},m:function(){return e(n.n(),2)},M:function(){return s.monthsShort[r.getMonth()]},n:function(){return r.getMonth()+1},t:function(){return new Date(n.Y(),n.n(),0).getDate()},L:function(){var t=n.Y();return t%4===0&&t%100!==0||t%400===0?1:0},o:function(){var t=n.n(),e=n.W(),r=n.Y();return r+(12===t&&9>e?1:1===t&&e>9?-1:0)},Y:function(){return r.getFullYear()},y:function(){return n.Y().toString().slice(-2)},a:function(){return n.A().toLowerCase()},A:function(){var t=n.G()<12?0:1;return s.meridiem[t]},B:function(){var t=r.getUTCHours()*i,n=60*r.getUTCMinutes(),a=r.getUTCSeconds();return e(Math.floor((t+n+a+i)/86.4)%1e3,3)},g:function(){return n.G()%12||12},G:function(){return r.getHours()},h:function(){return e(n.g(),2)},H:function(){return e(n.G(),2)},i:function(){return e(r.getMinutes(),2)},s:function(){return e(r.getSeconds(),2)},u:function(){return e(1e3*r.getMilliseconds(),6)},e:function(){var t=/\((.*)\)/.exec(String(r))[1];return t||"Coordinated Universal Time"},I:function(){var t=new Date(n.Y(),0),e=Date.UTC(n.Y(),0),r=new Date(n.Y(),6),a=Date.UTC(n.Y(),6);return t-e!==r-a?1:0},O:function(){var t=r.getTimezoneOffset(),n=Math.abs(t);return(t>0?"-":"+")+e(100*Math.floor(n/60)+n%60,4)},P:function(){var t=n.O();return t.substr(0,3)+":"+t.substr(3,2)},T:function(){var t=(String(r).match(a.tzParts)||[""]).pop().replace(a.tzClip,"");return t||"UTC"},Z:function(){return 60*-r.getTimezoneOffset()},c:function(){return"Y-m-d\\TH:i:sP".replace(o,c)},r:function(){return"D, d M Y H:i:s O".replace(o,c)},U:function(){return r.getTime()/1e3||0}},c(t,t)},formatDate:function(t,e){var r,n,a,u,i,s=this,o="",c="\\";if("string"==typeof t&&(t=s.parseDate(t,e),!t))return null;if(t instanceof Date){for(a=e.length,r=0;a>r;r++)i=e.charAt(r),"S"!==i&&i!==c&&(r>0&&e.charAt(r-1)===c?o+=i:(u=s.parseFormat(i,t),r!==a-1&&s.intParts.test(i)&&"S"===e.charAt(r+1)&&(n=parseInt(u)||0,u+=s.dateSettings.ordinal(n)),o+=u));return o}return""}}}();/** + * @preserve jQuery DateTimePicker + * @homepage http://xdsoft.net/jqplugins/datetimepicker/ + * @author Chupurnov Valeriy () + */ + +/** + * @param {jQuery} $ + */ +var datetimepickerFactory = function ($) { + 'use strict'; + + var default_options = { + i18n: { + ar: { // Arabic + months: [ + "كانون الثاني", "شباط", "آذار", "نيسان", "مايو", "حزيران", "تموز", "آب", "أيلول", "تشرين الأول", "تشرين الثاني", "كانون الأول" + ], + dayOfWeekShort: [ + "ن", "ث", "ع", "خ", "ج", "س", "ح" + ], + dayOfWeek: ["الأحد", "الاثنين", "الثلاثاء", "الأربعاء", "الخميس", "الجمعة", "السبت", "الأحد"] + }, + ro: { // Romanian + months: [ + "Ianuarie", "Februarie", "Martie", "Aprilie", "Mai", "Iunie", "Iulie", "August", "Septembrie", "Octombrie", "Noiembrie", "Decembrie" + ], + dayOfWeekShort: [ + "Du", "Lu", "Ma", "Mi", "Jo", "Vi", "Sâ" + ], + dayOfWeek: ["Duminică", "Luni", "Marţi", "Miercuri", "Joi", "Vineri", "Sâmbătă"] + }, + id: { // Indonesian + months: [ + "Januari", "Februari", "Maret", "April", "Mei", "Juni", "Juli", "Agustus", "September", "Oktober", "November", "Desember" + ], + dayOfWeekShort: [ + "Min", "Sen", "Sel", "Rab", "Kam", "Jum", "Sab" + ], + dayOfWeek: ["Minggu", "Senin", "Selasa", "Rabu", "Kamis", "Jumat", "Sabtu"] + }, + is: { // Icelandic + months: [ + "Janúar", "Febrúar", "Mars", "Apríl", "Maí", "Júní", "Júlí", "Ágúst", "September", "Október", "Nóvember", "Desember" + ], + dayOfWeekShort: [ + "Sun", "Mán", "Þrið", "Mið", "Fim", "Fös", "Lau" + ], + dayOfWeek: ["Sunnudagur", "Mánudagur", "Þriðjudagur", "Miðvikudagur", "Fimmtudagur", "Föstudagur", "Laugardagur"] + }, + bg: { // Bulgarian + months: [ + "Януари", "Февруари", "Март", "Април", "Май", "Юни", "Юли", "Август", "Септември", "Октомври", "Ноември", "Декември" + ], + dayOfWeekShort: [ + "Нд", "Пн", "Вт", "Ср", "Чт", "Пт", "Сб" + ], + dayOfWeek: ["Неделя", "Понеделник", "Вторник", "Сряда", "Четвъртък", "Петък", "Събота"] + }, + fa: { // Persian/Farsi + months: [ + 'فروردین', 'اردیبهشت', 'خرداد', 'تیر', 'مرداد', 'شهریور', 'مهر', 'آبان', 'آذر', 'دی', 'بهمن', 'اسفند' + ], + dayOfWeekShort: [ + 'یکشنبه', 'دوشنبه', 'سه شنبه', 'چهارشنبه', 'پنجشنبه', 'جمعه', 'شنبه' + ], + dayOfWeek: ["یک‌شنبه", "دوشنبه", "سه‌شنبه", "چهارشنبه", "پنج‌شنبه", "جمعه", "شنبه", "یک‌شنبه"] + }, + ru: { // Russian + months: [ + 'Январь', 'Февраль', 'Март', 'Апрель', 'Май', 'Июнь', 'Июль', 'Август', 'Сентябрь', 'Октябрь', 'Ноябрь', 'Декабрь' + ], + dayOfWeekShort: [ + "Вс", "Пн", "Вт", "Ср", "Чт", "Пт", "Сб" + ], + dayOfWeek: ["Воскресенье", "Понедельник", "Вторник", "Среда", "Четверг", "Пятница", "Суббота"] + }, + uk: { // Ukrainian + months: [ + 'Січень', 'Лютий', 'Березень', 'Квітень', 'Травень', 'Червень', 'Липень', 'Серпень', 'Вересень', 'Жовтень', 'Листопад', 'Грудень' + ], + dayOfWeekShort: [ + "Ндл", "Пнд", "Втр", "Срд", "Чтв", "Птн", "Сбт" + ], + dayOfWeek: ["Неділя", "Понеділок", "Вівторок", "Середа", "Четвер", "П'ятниця", "Субота"] + }, + en: { // English + months: [ + "January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December" + ], + dayOfWeekShort: [ + "Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat" + ], + dayOfWeek: ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"] + }, + el: { // Ελληνικά + months: [ + "Ιανουάριος", "Φεβρουάριος", "Μάρτιος", "Απρίλιος", "Μάιος", "Ιούνιος", "Ιούλιος", "Αύγουστος", "Σεπτέμβριος", "Οκτώβριος", "Νοέμβριος", "Δεκέμβριος" + ], + dayOfWeekShort: [ + "Κυρ", "Δευ", "Τρι", "Τετ", "Πεμ", "Παρ", "Σαβ" + ], + dayOfWeek: ["Κυριακή", "Δευτέρα", "Τρίτη", "Τετάρτη", "Πέμπτη", "Παρασκευή", "Σάββατο"] + }, + de: { // German + months: [ + 'Januar', 'Februar', 'März', 'April', 'Mai', 'Juni', 'Juli', 'August', 'September', 'Oktober', 'November', 'Dezember' + ], + dayOfWeekShort: [ + "So", "Mo", "Di", "Mi", "Do", "Fr", "Sa" + ], + dayOfWeek: ["Sonntag", "Montag", "Dienstag", "Mittwoch", "Donnerstag", "Freitag", "Samstag"] + }, + nl: { // Dutch + months: [ + "januari", "februari", "maart", "april", "mei", "juni", "juli", "augustus", "september", "oktober", "november", "december" + ], + dayOfWeekShort: [ + "zo", "ma", "di", "wo", "do", "vr", "za" + ], + dayOfWeek: ["zondag", "maandag", "dinsdag", "woensdag", "donderdag", "vrijdag", "zaterdag"] + }, + tr: { // Turkish + months: [ + "Ocak", "Şubat", "Mart", "Nisan", "Mayıs", "Haziran", "Temmuz", "Ağustos", "Eylül", "Ekim", "Kasım", "Aralık" + ], + dayOfWeekShort: [ + "Paz", "Pts", "Sal", "Çar", "Per", "Cum", "Cts" + ], + dayOfWeek: ["Pazar", "Pazartesi", "Salı", "Çarşamba", "Perşembe", "Cuma", "Cumartesi"] + }, + fr: { //French + months: [ + "Janvier", "Février", "Mars", "Avril", "Mai", "Juin", "Juillet", "Août", "Septembre", "Octobre", "Novembre", "Décembre" + ], + dayOfWeekShort: [ + "Dim", "Lun", "Mar", "Mer", "Jeu", "Ven", "Sam" + ], + dayOfWeek: ["dimanche", "lundi", "mardi", "mercredi", "jeudi", "vendredi", "samedi"] + }, + es: { // Spanish + months: [ + "Enero", "Febrero", "Marzo", "Abril", "Mayo", "Junio", "Julio", "Agosto", "Septiembre", "Octubre", "Noviembre", "Diciembre" + ], + dayOfWeekShort: [ + "Dom", "Lun", "Mar", "Mié", "Jue", "Vie", "Sáb" + ], + dayOfWeek: ["Domingo", "Lunes", "Martes", "Miércoles", "Jueves", "Viernes", "Sábado"] + }, + th: { // Thai + months: [ + 'มกราคม', 'กุมภาพันธ์', 'มีนาคม', 'เมษายน', 'พฤษภาคม', 'มิถุนายน', 'กรกฎาคม', 'สิงหาคม', 'กันยายน', 'ตุลาคม', 'พฤศจิกายน', 'ธันวาคม' + ], + dayOfWeekShort: [ + 'อา.', 'จ.', 'อ.', 'พ.', 'พฤ.', 'ศ.', 'ส.' + ], + dayOfWeek: ["อาทิตย์", "จันทร์", "อังคาร", "พุธ", "พฤหัส", "ศุกร์", "เสาร์", "อาทิตย์"] + }, + pl: { // Polish + months: [ + "styczeń", "luty", "marzec", "kwiecień", "maj", "czerwiec", "lipiec", "sierpień", "wrzesień", "październik", "listopad", "grudzień" + ], + dayOfWeekShort: [ + "nd", "pn", "wt", "śr", "cz", "pt", "sb" + ], + dayOfWeek: ["niedziela", "poniedziałek", "wtorek", "środa", "czwartek", "piątek", "sobota"] + }, + pt: { // Portuguese + months: [ + "Janeiro", "Fevereiro", "Março", "Abril", "Maio", "Junho", "Julho", "Agosto", "Setembro", "Outubro", "Novembro", "Dezembro" + ], + dayOfWeekShort: [ + "Dom", "Seg", "Ter", "Qua", "Qui", "Sex", "Sab" + ], + dayOfWeek: ["Domingo", "Segunda", "Terça", "Quarta", "Quinta", "Sexta", "Sábado"] + }, + ch: { // Simplified Chinese + months: [ + "一月", "二月", "三月", "四月", "五月", "六月", "七月", "八月", "九月", "十月", "十一月", "十二月" + ], + dayOfWeekShort: [ + "日", "一", "二", "三", "四", "五", "六" + ] + }, + se: { // Swedish + months: [ + "Januari", "Februari", "Mars", "April", "Maj", "Juni", "Juli", "Augusti", "September", "Oktober", "November", "December" + ], + dayOfWeekShort: [ + "Sön", "Mån", "Tis", "Ons", "Tor", "Fre", "Lör" + ] + }, + km: { // Khmer (ភាសាខ្មែរ) + months: [ + "មករា​", "កុម្ភៈ", "មិនា​", "មេសា​", "ឧសភា​", "មិថុនា​", "កក្កដា​", "សីហា​", "កញ្ញា​", "តុលា​", "វិច្ឆិកា", "ធ្នូ​" + ], + dayOfWeekShort: ["អាទិ​", "ច័ន្ទ​", "អង្គារ​", "ពុធ​", "ព្រហ​​", "សុក្រ​", "សៅរ៍"], + dayOfWeek: ["អាទិត្យ​", "ច័ន្ទ​", "អង្គារ​", "ពុធ​", "ព្រហស្បតិ៍​", "សុក្រ​", "សៅរ៍"] + }, + kr: { // Korean + months: [ + "1월", "2월", "3월", "4월", "5월", "6월", "7월", "8월", "9월", "10월", "11월", "12월" + ], + dayOfWeekShort: [ + "일", "월", "화", "수", "목", "금", "토" + ], + dayOfWeek: ["일요일", "월요일", "화요일", "수요일", "목요일", "금요일", "토요일"] + }, + it: { // Italian + months: [ + "Gennaio", "Febbraio", "Marzo", "Aprile", "Maggio", "Giugno", "Luglio", "Agosto", "Settembre", "Ottobre", "Novembre", "Dicembre" + ], + dayOfWeekShort: [ + "Dom", "Lun", "Mar", "Mer", "Gio", "Ven", "Sab" + ], + dayOfWeek: ["Domenica", "Lunedì", "Martedì", "Mercoledì", "Giovedì", "Venerdì", "Sabato"] + }, + da: { // Dansk + months: [ + "Januar", "Februar", "Marts", "April", "Maj", "Juni", "Juli", "August", "September", "Oktober", "November", "December" + ], + dayOfWeekShort: [ + "Søn", "Man", "Tir", "Ons", "Tor", "Fre", "Lør" + ], + dayOfWeek: ["søndag", "mandag", "tirsdag", "onsdag", "torsdag", "fredag", "lørdag"] + }, + no: { // Norwegian + months: [ + "Januar", "Februar", "Mars", "April", "Mai", "Juni", "Juli", "August", "September", "Oktober", "November", "Desember" + ], + dayOfWeekShort: [ + "Søn", "Man", "Tir", "Ons", "Tor", "Fre", "Lør" + ], + dayOfWeek: ['Søndag', 'Mandag', 'Tirsdag', 'Onsdag', 'Torsdag', 'Fredag', 'Lørdag'] + }, + ja: { // Japanese + months: [ + "1月", "2月", "3月", "4月", "5月", "6月", "7月", "8月", "9月", "10月", "11月", "12月" + ], + dayOfWeekShort: [ + "日", "月", "火", "水", "木", "金", "土" + ], + dayOfWeek: ["日曜", "月曜", "火曜", "水曜", "木曜", "金曜", "土曜"] + }, + vi: { // Vietnamese + months: [ + "Tháng 1", "Tháng 2", "Tháng 3", "Tháng 4", "Tháng 5", "Tháng 6", "Tháng 7", "Tháng 8", "Tháng 9", "Tháng 10", "Tháng 11", "Tháng 12" + ], + dayOfWeekShort: [ + "CN", "T2", "T3", "T4", "T5", "T6", "T7" + ], + dayOfWeek: ["Chủ nhật", "Thứ hai", "Thứ ba", "Thứ tư", "Thứ năm", "Thứ sáu", "Thứ bảy"] + }, + sl: { // Slovenščina + months: [ + "Januar", "Februar", "Marec", "April", "Maj", "Junij", "Julij", "Avgust", "September", "Oktober", "November", "December" + ], + dayOfWeekShort: [ + "Ned", "Pon", "Tor", "Sre", "Čet", "Pet", "Sob" + ], + dayOfWeek: ["Nedelja", "Ponedeljek", "Torek", "Sreda", "Četrtek", "Petek", "Sobota"] + }, + cs: { // Čeština + months: [ + "Leden", "Únor", "Březen", "Duben", "Květen", "Červen", "Červenec", "Srpen", "Září", "Říjen", "Listopad", "Prosinec" + ], + dayOfWeekShort: [ + "Ne", "Po", "Út", "St", "Čt", "Pá", "So" + ] + }, + hu: { // Hungarian + months: [ + "Január", "Február", "Március", "Április", "Május", "Június", "Július", "Augusztus", "Szeptember", "Október", "November", "December" + ], + dayOfWeekShort: [ + "Va", "Hé", "Ke", "Sze", "Cs", "Pé", "Szo" + ], + dayOfWeek: ["vasárnap", "hétfő", "kedd", "szerda", "csütörtök", "péntek", "szombat"] + }, + az: { //Azerbaijanian (Azeri) + months: [ + "Yanvar", "Fevral", "Mart", "Aprel", "May", "Iyun", "Iyul", "Avqust", "Sentyabr", "Oktyabr", "Noyabr", "Dekabr" + ], + dayOfWeekShort: [ + "B", "Be", "Ça", "Ç", "Ca", "C", "Ş" + ], + dayOfWeek: ["Bazar", "Bazar ertəsi", "Çərşənbə axşamı", "Çərşənbə", "Cümə axşamı", "Cümə", "Şənbə"] + }, + bs: { //Bosanski + months: [ + "Januar", "Februar", "Mart", "April", "Maj", "Jun", "Jul", "Avgust", "Septembar", "Oktobar", "Novembar", "Decembar" + ], + dayOfWeekShort: [ + "Ned", "Pon", "Uto", "Sri", "Čet", "Pet", "Sub" + ], + dayOfWeek: ["Nedjelja","Ponedjeljak", "Utorak", "Srijeda", "Četvrtak", "Petak", "Subota"] + }, + ca: { //Català + months: [ + "Gener", "Febrer", "Març", "Abril", "Maig", "Juny", "Juliol", "Agost", "Setembre", "Octubre", "Novembre", "Desembre" + ], + dayOfWeekShort: [ + "Dg", "Dl", "Dt", "Dc", "Dj", "Dv", "Ds" + ], + dayOfWeek: ["Diumenge", "Dilluns", "Dimarts", "Dimecres", "Dijous", "Divendres", "Dissabte"] + }, + 'en-GB': { //English (British) + months: [ + "January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December" + ], + dayOfWeekShort: [ + "Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat" + ], + dayOfWeek: ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"] + }, + et: { //"Eesti" + months: [ + "Jaanuar", "Veebruar", "Märts", "Aprill", "Mai", "Juuni", "Juuli", "August", "September", "Oktoober", "November", "Detsember" + ], + dayOfWeekShort: [ + "P", "E", "T", "K", "N", "R", "L" + ], + dayOfWeek: ["Pühapäev", "Esmaspäev", "Teisipäev", "Kolmapäev", "Neljapäev", "Reede", "Laupäev"] + }, + eu: { //Euskara + months: [ + "Urtarrila", "Otsaila", "Martxoa", "Apirila", "Maiatza", "Ekaina", "Uztaila", "Abuztua", "Iraila", "Urria", "Azaroa", "Abendua" + ], + dayOfWeekShort: [ + "Ig.", "Al.", "Ar.", "Az.", "Og.", "Or.", "La." + ], + dayOfWeek: ['Igandea', 'Astelehena', 'Asteartea', 'Asteazkena', 'Osteguna', 'Ostirala', 'Larunbata'] + }, + fi: { //Finnish (Suomi) + months: [ + "Tammikuu", "Helmikuu", "Maaliskuu", "Huhtikuu", "Toukokuu", "Kesäkuu", "Heinäkuu", "Elokuu", "Syyskuu", "Lokakuu", "Marraskuu", "Joulukuu" + ], + dayOfWeekShort: [ + "Su", "Ma", "Ti", "Ke", "To", "Pe", "La" + ], + dayOfWeek: ["sunnuntai", "maanantai", "tiistai", "keskiviikko", "torstai", "perjantai", "lauantai"] + }, + gl: { //Galego + months: [ + "Xan", "Feb", "Maz", "Abr", "Mai", "Xun", "Xul", "Ago", "Set", "Out", "Nov", "Dec" + ], + dayOfWeekShort: [ + "Dom", "Lun", "Mar", "Mer", "Xov", "Ven", "Sab" + ], + dayOfWeek: ["Domingo", "Luns", "Martes", "Mércores", "Xoves", "Venres", "Sábado"] + }, + hr: { //Hrvatski + months: [ + "Siječanj", "Veljača", "Ožujak", "Travanj", "Svibanj", "Lipanj", "Srpanj", "Kolovoz", "Rujan", "Listopad", "Studeni", "Prosinac" + ], + dayOfWeekShort: [ + "Ned", "Pon", "Uto", "Sri", "Čet", "Pet", "Sub" + ], + dayOfWeek: ["Nedjelja", "Ponedjeljak", "Utorak", "Srijeda", "Četvrtak", "Petak", "Subota"] + }, + ko: { //Korean (한국어) + months: [ + "1월", "2월", "3월", "4월", "5월", "6월", "7월", "8월", "9월", "10월", "11월", "12월" + ], + dayOfWeekShort: [ + "일", "월", "화", "수", "목", "금", "토" + ], + dayOfWeek: ["일요일", "월요일", "화요일", "수요일", "목요일", "금요일", "토요일"] + }, + lt: { //Lithuanian (lietuvių) + months: [ + "Sausio", "Vasario", "Kovo", "Balandžio", "Gegužės", "Birželio", "Liepos", "Rugpjūčio", "Rugsėjo", "Spalio", "Lapkričio", "Gruodžio" + ], + dayOfWeekShort: [ + "Sek", "Pir", "Ant", "Tre", "Ket", "Pen", "Šeš" + ], + dayOfWeek: ["Sekmadienis", "Pirmadienis", "Antradienis", "Trečiadienis", "Ketvirtadienis", "Penktadienis", "Šeštadienis"] + }, + lv: { //Latvian (Latviešu) + months: [ + "Janvāris", "Februāris", "Marts", "Aprīlis ", "Maijs", "Jūnijs", "Jūlijs", "Augusts", "Septembris", "Oktobris", "Novembris", "Decembris" + ], + dayOfWeekShort: [ + "Sv", "Pr", "Ot", "Tr", "Ct", "Pk", "St" + ], + dayOfWeek: ["Svētdiena", "Pirmdiena", "Otrdiena", "Trešdiena", "Ceturtdiena", "Piektdiena", "Sestdiena"] + }, + mk: { //Macedonian (Македонски) + months: [ + "јануари", "февруари", "март", "април", "мај", "јуни", "јули", "август", "септември", "октомври", "ноември", "декември" + ], + dayOfWeekShort: [ + "нед", "пон", "вто", "сре", "чет", "пет", "саб" + ], + dayOfWeek: ["Недела", "Понеделник", "Вторник", "Среда", "Четврток", "Петок", "Сабота"] + }, + mn: { //Mongolian (Монгол) + months: [ + "1-р сар", "2-р сар", "3-р сар", "4-р сар", "5-р сар", "6-р сар", "7-р сар", "8-р сар", "9-р сар", "10-р сар", "11-р сар", "12-р сар" + ], + dayOfWeekShort: [ + "Дав", "Мяг", "Лха", "Пүр", "Бсн", "Бям", "Ням" + ], + dayOfWeek: ["Даваа", "Мягмар", "Лхагва", "Пүрэв", "Баасан", "Бямба", "Ням"] + }, + 'pt-BR': { //Português(Brasil) + months: [ + "Janeiro", "Fevereiro", "Março", "Abril", "Maio", "Junho", "Julho", "Agosto", "Setembro", "Outubro", "Novembro", "Dezembro" + ], + dayOfWeekShort: [ + "Dom", "Seg", "Ter", "Qua", "Qui", "Sex", "Sáb" + ], + dayOfWeek: ["Domingo", "Segunda", "Terça", "Quarta", "Quinta", "Sexta", "Sábado"] + }, + sk: { //Slovenčina + months: [ + "Január", "Február", "Marec", "Apríl", "Máj", "Jún", "Júl", "August", "September", "Október", "November", "December" + ], + dayOfWeekShort: [ + "Ne", "Po", "Ut", "St", "Št", "Pi", "So" + ], + dayOfWeek: ["Nedeľa", "Pondelok", "Utorok", "Streda", "Štvrtok", "Piatok", "Sobota"] + }, + sq: { //Albanian (Shqip) + months: [ + "Janar", "Shkurt", "Mars", "Prill", "Maj", "Qershor", "Korrik", "Gusht", "Shtator", "Tetor", "Nëntor", "Dhjetor" + ], + dayOfWeekShort: [ + "Die", "Hën", "Mar", "Mër", "Enj", "Pre", "Shtu" + ], + dayOfWeek: ["E Diel", "E Hënë", "E Martē", "E Mërkurë", "E Enjte", "E Premte", "E Shtunë"] + }, + 'sr-YU': { //Serbian (Srpski) + months: [ + "Januar", "Februar", "Mart", "April", "Maj", "Jun", "Jul", "Avgust", "Septembar", "Oktobar", "Novembar", "Decembar" + ], + dayOfWeekShort: [ + "Ned", "Pon", "Uto", "Sre", "čet", "Pet", "Sub" + ], + dayOfWeek: ["Nedelja","Ponedeljak", "Utorak", "Sreda", "Četvrtak", "Petak", "Subota"] + }, + sr: { //Serbian Cyrillic (Српски) + months: [ + "јануар", "фебруар", "март", "април", "мај", "јун", "јул", "август", "септембар", "октобар", "новембар", "децембар" + ], + dayOfWeekShort: [ + "нед", "пон", "уто", "сре", "чет", "пет", "суб" + ], + dayOfWeek: ["Недеља","Понедељак", "Уторак", "Среда", "Четвртак", "Петак", "Субота"] + }, + sv: { //Svenska + months: [ + "Januari", "Februari", "Mars", "April", "Maj", "Juni", "Juli", "Augusti", "September", "Oktober", "November", "December" + ], + dayOfWeekShort: [ + "Sön", "Mån", "Tis", "Ons", "Tor", "Fre", "Lör" + ], + dayOfWeek: ["Söndag", "Måndag", "Tisdag", "Onsdag", "Torsdag", "Fredag", "Lördag"] + }, + 'zh-TW': { //Traditional Chinese (繁體中文) + months: [ + "一月", "二月", "三月", "四月", "五月", "六月", "七月", "八月", "九月", "十月", "十一月", "十二月" + ], + dayOfWeekShort: [ + "日", "一", "二", "三", "四", "五", "六" + ], + dayOfWeek: ["星期日", "星期一", "星期二", "星期三", "星期四", "星期五", "星期六"] + }, + zh: { //Simplified Chinese (简体中文) + months: [ + "一月", "二月", "三月", "四月", "五月", "六月", "七月", "八月", "九月", "十月", "十一月", "十二月" + ], + dayOfWeekShort: [ + "日", "一", "二", "三", "四", "五", "六" + ], + dayOfWeek: ["星期日", "星期一", "星期二", "星期三", "星期四", "星期五", "星期六"] + }, + ug:{ // Uyghur(ئۇيغۇرچە) + months: [ + "1-ئاي","2-ئاي","3-ئاي","4-ئاي","5-ئاي","6-ئاي","7-ئاي","8-ئاي","9-ئاي","10-ئاي","11-ئاي","12-ئاي" + ], + dayOfWeek: [ + "يەكشەنبە", "دۈشەنبە","سەيشەنبە","چارشەنبە","پەيشەنبە","جۈمە","شەنبە" + ] + }, + he: { //Hebrew (עברית) + months: [ + 'ינואר', 'פברואר', 'מרץ', 'אפריל', 'מאי', 'יוני', 'יולי', 'אוגוסט', 'ספטמבר', 'אוקטובר', 'נובמבר', 'דצמבר' + ], + dayOfWeekShort: [ + 'א\'', 'ב\'', 'ג\'', 'ד\'', 'ה\'', 'ו\'', 'שבת' + ], + dayOfWeek: ["ראשון", "שני", "שלישי", "רביעי", "חמישי", "שישי", "שבת", "ראשון"] + }, + hy: { // Armenian + months: [ + "Հունվար", "Փետրվար", "Մարտ", "Ապրիլ", "Մայիս", "Հունիս", "Հուլիս", "Օգոստոս", "Սեպտեմբեր", "Հոկտեմբեր", "Նոյեմբեր", "Դեկտեմբեր" + ], + dayOfWeekShort: [ + "Կի", "Երկ", "Երք", "Չոր", "Հնգ", "Ուրբ", "Շբթ" + ], + dayOfWeek: ["Կիրակի", "Երկուշաբթի", "Երեքշաբթի", "Չորեքշաբթի", "Հինգշաբթի", "Ուրբաթ", "Շաբաթ"] + }, + kg: { // Kyrgyz + months: [ + 'Үчтүн айы', 'Бирдин айы', 'Жалган Куран', 'Чын Куран', 'Бугу', 'Кулжа', 'Теке', 'Баш Оона', 'Аяк Оона', 'Тогуздун айы', 'Жетинин айы', 'Бештин айы' + ], + dayOfWeekShort: [ + "Жек", "Дүй", "Шей", "Шар", "Бей", "Жум", "Ише" + ], + dayOfWeek: [ + "Жекшемб", "Дүйшөмб", "Шейшемб", "Шаршемб", "Бейшемби", "Жума", "Ишенб" + ] + }, + rm: { // Romansh + months: [ + "Schaner", "Favrer", "Mars", "Avrigl", "Matg", "Zercladur", "Fanadur", "Avust", "Settember", "October", "November", "December" + ], + dayOfWeekShort: [ + "Du", "Gli", "Ma", "Me", "Gie", "Ve", "So" + ], + dayOfWeek: [ + "Dumengia", "Glindesdi", "Mardi", "Mesemna", "Gievgia", "Venderdi", "Sonda" + ] + }, + ka: { // Georgian + months: [ + 'იანვარი', 'თებერვალი', 'მარტი', 'აპრილი', 'მაისი', 'ივნისი', 'ივლისი', 'აგვისტო', 'სექტემბერი', 'ოქტომბერი', 'ნოემბერი', 'დეკემბერი' + ], + dayOfWeekShort: [ + "კვ", "ორშ", "სამშ", "ოთხ", "ხუთ", "პარ", "შაბ" + ], + dayOfWeek: ["კვირა", "ორშაბათი", "სამშაბათი", "ოთხშაბათი", "ხუთშაბათი", "პარასკევი", "შაბათი"] + } + }, + + ownerDocument: document, + contentWindow: window, + + value: '', + rtl: false, + + format: 'Y/m/d H:i', + formatTime: 'H:i', + formatDate: 'Y/m/d', + + startDate: false, // new Date(), '1986/12/08', '-1970/01/05','-1970/01/05', + step: 60, + monthChangeSpinner: true, + + closeOnDateSelect: false, + closeOnTimeSelect: true, + closeOnWithoutClick: true, + closeOnInputClick: true, + openOnFocus: true, + + timepicker: true, + datepicker: true, + weeks: false, + + defaultTime: false, // use formatTime format (ex. '10:00' for formatTime: 'H:i') + defaultDate: false, // use formatDate format (ex new Date() or '1986/12/08' or '-1970/01/05' or '-1970/01/05') + + minDate: false, + maxDate: false, + minTime: false, + maxTime: false, + minDateTime: false, + maxDateTime: false, + + allowTimes: [], + opened: false, + initTime: true, + inline: false, + theme: '', + touchMovedThreshold: 5, + + onSelectDate: function () {}, + onSelectTime: function () {}, + onChangeMonth: function () {}, + onGetWeekOfYear: function () {}, + onChangeYear: function () {}, + onChangeDateTime: function () {}, + onShow: function () {}, + onClose: function () {}, + onGenerate: function () {}, + + withoutCopyright: true, + inverseButton: false, + hours12: false, + next: 'xdsoft_next', + prev : 'xdsoft_prev', + dayOfWeekStart: 0, + parentID: 'body', + timeHeightInTimePicker: 25, + timepickerScrollbar: true, + todayButton: true, + prevButton: true, + nextButton: true, + defaultSelect: true, + + scrollMonth: true, + scrollTime: true, + scrollInput: true, + + lazyInit: false, + mask: false, + validateOnBlur: true, + allowBlank: true, + yearStart: 1950, + yearEnd: 2050, + monthStart: 0, + monthEnd: 11, + style: '', + id: '', + fixed: false, + roundTime: 'round', // ceil, floor + className: '', + weekends: [], + highlightedDates: [], + highlightedPeriods: [], + allowDates : [], + allowDateRe : null, + disabledDates : [], + disabledWeekDays: [], + yearOffset: 0, + beforeShowDay: null, + + enterLikeTab: true, + showApplyButton: false + }; + + var dateHelper = null, + defaultDateHelper = null, + globalLocaleDefault = 'en', + globalLocale = 'en'; + + var dateFormatterOptionsDefault = { + meridiem: ['AM', 'PM'] + }; + + var initDateFormatter = function(){ + var locale = default_options.i18n[globalLocale], + opts = { + days: locale.dayOfWeek, + daysShort: locale.dayOfWeekShort, + months: locale.months, + monthsShort: $.map(locale.months, function(n){ return n.substring(0, 3) }) + }; + + if (typeof DateFormatter === 'function') { + dateHelper = defaultDateHelper = new DateFormatter({ + dateSettings: $.extend({}, dateFormatterOptionsDefault, opts) + }); + } + }; + + var dateFormatters = { + moment: { + default_options:{ + format: 'YYYY/MM/DD HH:mm', + formatDate: 'YYYY/MM/DD', + formatTime: 'HH:mm', + }, + formatter: { + parseDate: function (date, format) { + if(isFormatStandard(format)){ + return defaultDateHelper.parseDate(date, format); + } + var d = moment(date, format); + return d.isValid() ? d.toDate() : false; + }, + + formatDate: function (date, format) { + if(isFormatStandard(format)){ + return defaultDateHelper.formatDate(date, format); + } + return moment(date).format(format); + }, + + formatMask: function(format){ + return format + .replace(/Y{4}/g, '9999') + .replace(/Y{2}/g, '99') + .replace(/M{2}/g, '19') + .replace(/D{2}/g, '39') + .replace(/H{2}/g, '29') + .replace(/m{2}/g, '59') + .replace(/s{2}/g, '59'); + }, + } + } + } + + // for locale settings + $.datetimepicker = { + setLocale: function(locale){ + var newLocale = default_options.i18n[locale] ? locale : globalLocaleDefault; + if (globalLocale !== newLocale) { + globalLocale = newLocale; + // reinit date formatter + initDateFormatter(); + } + }, + + setDateFormatter: function(dateFormatter) { + if(typeof dateFormatter === 'string' && dateFormatters.hasOwnProperty(dateFormatter)){ + var df = dateFormatters[dateFormatter]; + $.extend(default_options, df.default_options); + dateHelper = df.formatter; + } + else { + dateHelper = dateFormatter; + } + }, + }; + + var standardFormats = { + RFC_2822: 'D, d M Y H:i:s O', + ATOM: 'Y-m-d\TH:i:sP', + ISO_8601: 'Y-m-d\TH:i:sO', + RFC_822: 'D, d M y H:i:s O', + RFC_850: 'l, d-M-y H:i:s T', + RFC_1036: 'D, d M y H:i:s O', + RFC_1123: 'D, d M Y H:i:s O', + RSS: 'D, d M Y H:i:s O', + W3C: 'Y-m-d\TH:i:sP' + } + + var isFormatStandard = function(format){ + return Object.values(standardFormats).indexOf(format) === -1 ? false : true; + } + + $.extend($.datetimepicker, standardFormats); + + // first init date formatter + initDateFormatter(); + + // fix for ie8 + if (!window.getComputedStyle) { + window.getComputedStyle = function (el) { + this.el = el; + this.getPropertyValue = function (prop) { + var re = /(-([a-z]))/g; + if (prop === 'float') { + prop = 'styleFloat'; + } + if (re.test(prop)) { + prop = prop.replace(re, function (a, b, c) { + return c.toUpperCase(); + }); + } + return el.currentStyle[prop] || null; + }; + return this; + }; + } + if (!Array.prototype.indexOf) { + Array.prototype.indexOf = function (obj, start) { + var i, j; + for (i = (start || 0), j = this.length; i < j; i += 1) { + if (this[i] === obj) { return i; } + } + return -1; + }; + } + + Date.prototype.countDaysInMonth = function () { + return new Date(this.getFullYear(), this.getMonth() + 1, 0).getDate(); + }; + + $.fn.xdsoftScroller = function (options, percent) { + return this.each(function () { + var timeboxparent = $(this), + pointerEventToXY = function (e) { + var out = {x: 0, y: 0}, + touch; + if (e.type === 'touchstart' || e.type === 'touchmove' || e.type === 'touchend' || e.type === 'touchcancel') { + touch = e.originalEvent.touches[0] || e.originalEvent.changedTouches[0]; + out.x = touch.clientX; + out.y = touch.clientY; + } else if (e.type === 'mousedown' || e.type === 'mouseup' || e.type === 'mousemove' || e.type === 'mouseover' || e.type === 'mouseout' || e.type === 'mouseenter' || e.type === 'mouseleave') { + out.x = e.clientX; + out.y = e.clientY; + } + return out; + }, + timebox, + parentHeight, + height, + scrollbar, + scroller, + maximumOffset = 100, + start = false, + startY = 0, + startTop = 0, + h1 = 0, + touchStart = false, + startTopScroll = 0, + calcOffset = function () {}; + + if (percent === 'hide') { + timeboxparent.find('.xdsoft_scrollbar').hide(); + return; + } + + if (!$(this).hasClass('xdsoft_scroller_box')) { + timebox = timeboxparent.children().eq(0); + parentHeight = timeboxparent[0].clientHeight; + height = timebox[0].offsetHeight; + scrollbar = $('
'); + scroller = $('
'); + scrollbar.append(scroller); + + timeboxparent.addClass('xdsoft_scroller_box').append(scrollbar); + calcOffset = function calcOffset(event) { + var offset = pointerEventToXY(event).y - startY + startTopScroll; + if (offset < 0) { + offset = 0; + } + if (offset + scroller[0].offsetHeight > h1) { + offset = h1 - scroller[0].offsetHeight; + } + timeboxparent.trigger('scroll_element.xdsoft_scroller', [maximumOffset ? offset / maximumOffset : 0]); + }; + + scroller + .on('touchstart.xdsoft_scroller mousedown.xdsoft_scroller', function (event) { + if (!parentHeight) { + timeboxparent.trigger('resize_scroll.xdsoft_scroller', [percent]); + } + + startY = pointerEventToXY(event).y; + startTopScroll = parseInt(scroller.css('margin-top'), 10); + h1 = scrollbar[0].offsetHeight; + + if (event.type === 'mousedown' || event.type === 'touchstart') { + if (options.ownerDocument) { + $(options.ownerDocument.body).addClass('xdsoft_noselect'); + } + $([options.ownerDocument.body, options.contentWindow]).on('touchend mouseup.xdsoft_scroller', function arguments_callee() { + $([options.ownerDocument.body, options.contentWindow]).off('touchend mouseup.xdsoft_scroller', arguments_callee) + .off('mousemove.xdsoft_scroller', calcOffset) + .removeClass('xdsoft_noselect'); + }); + $(options.ownerDocument.body).on('mousemove.xdsoft_scroller', calcOffset); + } else { + touchStart = true; + event.stopPropagation(); + event.preventDefault(); + } + }) + .on('touchmove', function (event) { + if (touchStart) { + event.preventDefault(); + calcOffset(event); + } + }) + .on('touchend touchcancel', function () { + touchStart = false; + startTopScroll = 0; + }); + + timeboxparent + .on('scroll_element.xdsoft_scroller', function (event, percentage) { + if (!parentHeight) { + timeboxparent.trigger('resize_scroll.xdsoft_scroller', [percentage, true]); + } + percentage = percentage > 1 ? 1 : (percentage < 0 || isNaN(percentage)) ? 0 : percentage; + + scroller.css('margin-top', maximumOffset * percentage); + + setTimeout(function () { + timebox.css('marginTop', -parseInt((timebox[0].offsetHeight - parentHeight) * percentage, 10)); + }, 10); + }) + .on('resize_scroll.xdsoft_scroller', function (event, percentage, noTriggerScroll) { + var percent, sh; + parentHeight = timeboxparent[0].clientHeight; + height = timebox[0].offsetHeight; + percent = parentHeight / height; + sh = percent * scrollbar[0].offsetHeight; + if (percent > 1) { + scroller.hide(); + } else { + scroller.show(); + scroller.css('height', parseInt(sh > 10 ? sh : 10, 10)); + maximumOffset = scrollbar[0].offsetHeight - scroller[0].offsetHeight; + if (noTriggerScroll !== true) { + timeboxparent.trigger('scroll_element.xdsoft_scroller', [percentage || Math.abs(parseInt(timebox.css('marginTop'), 10)) / (height - parentHeight)]); + } + } + }); + + timeboxparent.on('mousewheel', function (event) { + var top = Math.abs(parseInt(timebox.css('marginTop'), 10)); + + top = top - (event.deltaY * 20); + if (top < 0) { + top = 0; + } + + timeboxparent.trigger('scroll_element.xdsoft_scroller', [top / (height - parentHeight)]); + event.stopPropagation(); + return false; + }); + + timeboxparent.on('touchstart', function (event) { + start = pointerEventToXY(event); + startTop = Math.abs(parseInt(timebox.css('marginTop'), 10)); + }); + + timeboxparent.on('touchmove', function (event) { + if (start) { + event.preventDefault(); + var coord = pointerEventToXY(event); + timeboxparent.trigger('scroll_element.xdsoft_scroller', [(startTop - (coord.y - start.y)) / (height - parentHeight)]); + } + }); + + timeboxparent.on('touchend touchcancel', function () { + start = false; + startTop = 0; + }); + } + timeboxparent.trigger('resize_scroll.xdsoft_scroller', [percent]); + }); + }; + + $.fn.datetimepicker = function (opt, opt2) { + var result = this, + KEY0 = 48, + KEY9 = 57, + _KEY0 = 96, + _KEY9 = 105, + CTRLKEY = 17, + DEL = 46, + ENTER = 13, + ESC = 27, + BACKSPACE = 8, + ARROWLEFT = 37, + ARROWUP = 38, + ARROWRIGHT = 39, + ARROWDOWN = 40, + TAB = 9, + F5 = 116, + AKEY = 65, + CKEY = 67, + VKEY = 86, + ZKEY = 90, + YKEY = 89, + ctrlDown = false, + options = ($.isPlainObject(opt) || !opt) ? $.extend(true, {}, default_options, opt) : $.extend(true, {}, default_options), + + lazyInitTimer = 0, + createDateTimePicker, + destroyDateTimePicker, + + lazyInit = function (input) { + input + .on('open.xdsoft focusin.xdsoft mousedown.xdsoft touchstart', function initOnActionCallback() { + if (input.is(':disabled') || input.data('xdsoft_datetimepicker')) { + return; + } + clearTimeout(lazyInitTimer); + lazyInitTimer = setTimeout(function () { + + if (!input.data('xdsoft_datetimepicker')) { + createDateTimePicker(input); + } + input + .off('open.xdsoft focusin.xdsoft mousedown.xdsoft touchstart', initOnActionCallback) + .trigger('open.xdsoft'); + }, 100); + }); + }; + + createDateTimePicker = function (input) { + var datetimepicker = $('
'), + xdsoft_copyright = $(''), + datepicker = $('
'), + month_picker = $('
' + + '
' + + '
' + + '
'), + calendar = $('
'), + timepicker = $('
'), + timeboxparent = timepicker.find('.xdsoft_time_box').eq(0), + timebox = $('
'), + applyButton = $(''), + + monthselect = $('
'), + yearselect = $('
'), + triggerAfterOpen = false, + XDSoft_datetime, + + xchangeTimer, + timerclick, + current_time_index, + setPos, + timer = 0, + _xdsoft_datetime, + forEachAncestorOf; + + if (options.id) { + datetimepicker.attr('id', options.id); + } + if (options.style) { + datetimepicker.attr('style', options.style); + } + if (options.weeks) { + datetimepicker.addClass('xdsoft_showweeks'); + } + if (options.rtl) { + datetimepicker.addClass('xdsoft_rtl'); + } + + datetimepicker.addClass('xdsoft_' + options.theme); + datetimepicker.addClass(options.className); + + month_picker + .find('.xdsoft_month span') + .after(monthselect); + month_picker + .find('.xdsoft_year span') + .after(yearselect); + + month_picker + .find('.xdsoft_month,.xdsoft_year') + .on('touchstart mousedown.xdsoft', function (event) { + var select = $(this).find('.xdsoft_select').eq(0), + val = 0, + top = 0, + visible = select.is(':visible'), + items, + i; + + month_picker + .find('.xdsoft_select') + .hide(); + if (_xdsoft_datetime.currentTime) { + val = _xdsoft_datetime.currentTime[$(this).hasClass('xdsoft_month') ? 'getMonth' : 'getFullYear'](); + } + + select[visible ? 'hide' : 'show'](); + for (items = select.find('div.xdsoft_option'), i = 0; i < items.length; i += 1) { + if (items.eq(i).data('value') === val) { + break; + } else { + top += items[0].offsetHeight; + } + } + + select.xdsoftScroller(options, top / (select.children()[0].offsetHeight - (select[0].clientHeight))); + event.stopPropagation(); + return false; + }); + + var handleTouchMoved = function (event) { + var evt = event.originalEvent; + var touchPosition = evt.touches ? evt.touches[0] : evt; + this.touchStartPosition = this.touchStartPosition || touchPosition; + var xMovement = Math.abs(this.touchStartPosition.clientX - touchPosition.clientX); + var yMovement = Math.abs(this.touchStartPosition.clientY - touchPosition.clientY); + var distance = Math.sqrt(xMovement * xMovement + yMovement * yMovement); + if(distance > options.touchMovedThreshold) { + this.touchMoved = true; + } + } + + month_picker + .find('.xdsoft_select') + .xdsoftScroller(options) + .on('touchstart mousedown.xdsoft', function (event) { + var evt = event.originalEvent; + this.touchMoved = false; + this.touchStartPosition = evt.touches ? evt.touches[0] : evt; + event.stopPropagation(); + event.preventDefault(); + }) + .on('touchmove', '.xdsoft_option', handleTouchMoved) + .on('touchend mousedown.xdsoft', '.xdsoft_option', function () { + if (!this.touchMoved) { + if (_xdsoft_datetime.currentTime === undefined || _xdsoft_datetime.currentTime === null) { + _xdsoft_datetime.currentTime = _xdsoft_datetime.now(); + } + + var year = _xdsoft_datetime.currentTime.getFullYear(); + if (_xdsoft_datetime && _xdsoft_datetime.currentTime) { + _xdsoft_datetime.currentTime[$(this).parent().parent().hasClass('xdsoft_monthselect') ? 'setMonth' : 'setFullYear']($(this).data('value')); + } + + $(this).parent().parent().hide(); + + datetimepicker.trigger('xchange.xdsoft'); + if (options.onChangeMonth && $.isFunction(options.onChangeMonth)) { + options.onChangeMonth.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input')); + } + + if (year !== _xdsoft_datetime.currentTime.getFullYear() && $.isFunction(options.onChangeYear)) { + options.onChangeYear.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input')); + } + } + }); + + datetimepicker.getValue = function () { + return _xdsoft_datetime.getCurrentTime(); + }; + + datetimepicker.setOptions = function (_options) { + var highlightedDates = {}; + + options = $.extend(true, {}, options, _options); + + if (_options.allowTimes && $.isArray(_options.allowTimes) && _options.allowTimes.length) { + options.allowTimes = $.extend(true, [], _options.allowTimes); + } + + if (_options.weekends && $.isArray(_options.weekends) && _options.weekends.length) { + options.weekends = $.extend(true, [], _options.weekends); + } + + if (_options.allowDates && $.isArray(_options.allowDates) && _options.allowDates.length) { + options.allowDates = $.extend(true, [], _options.allowDates); + } + + if (_options.allowDateRe && Object.prototype.toString.call(_options.allowDateRe)==="[object String]") { + options.allowDateRe = new RegExp(_options.allowDateRe); + } + + if (_options.highlightedDates && $.isArray(_options.highlightedDates) && _options.highlightedDates.length) { + $.each(_options.highlightedDates, function (index, value) { + var splitData = $.map(value.split(','), $.trim), + exDesc, + hDate = new HighlightedDate(dateHelper.parseDate(splitData[0], options.formatDate), splitData[1], splitData[2]), // date, desc, style + keyDate = dateHelper.formatDate(hDate.date, options.formatDate); + if (highlightedDates[keyDate] !== undefined) { + exDesc = highlightedDates[keyDate].desc; + if (exDesc && exDesc.length && hDate.desc && hDate.desc.length) { + highlightedDates[keyDate].desc = exDesc + "\n" + hDate.desc; + } + } else { + highlightedDates[keyDate] = hDate; + } + }); + + options.highlightedDates = $.extend(true, [], highlightedDates); + } + + if (_options.highlightedPeriods && $.isArray(_options.highlightedPeriods) && _options.highlightedPeriods.length) { + highlightedDates = $.extend(true, [], options.highlightedDates); + $.each(_options.highlightedPeriods, function (index, value) { + var dateTest, // start date + dateEnd, + desc, + hDate, + keyDate, + exDesc, + style; + if ($.isArray(value)) { + dateTest = value[0]; + dateEnd = value[1]; + desc = value[2]; + style = value[3]; + } + else { + var splitData = $.map(value.split(','), $.trim); + dateTest = dateHelper.parseDate(splitData[0], options.formatDate); + dateEnd = dateHelper.parseDate(splitData[1], options.formatDate); + desc = splitData[2]; + style = splitData[3]; + } + + while (dateTest <= dateEnd) { + hDate = new HighlightedDate(dateTest, desc, style); + keyDate = dateHelper.formatDate(dateTest, options.formatDate); + dateTest.setDate(dateTest.getDate() + 1); + if (highlightedDates[keyDate] !== undefined) { + exDesc = highlightedDates[keyDate].desc; + if (exDesc && exDesc.length && hDate.desc && hDate.desc.length) { + highlightedDates[keyDate].desc = exDesc + "\n" + hDate.desc; + } + } else { + highlightedDates[keyDate] = hDate; + } + } + }); + + options.highlightedDates = $.extend(true, [], highlightedDates); + } + + if (_options.disabledDates && $.isArray(_options.disabledDates) && _options.disabledDates.length) { + options.disabledDates = $.extend(true, [], _options.disabledDates); + } + + if (_options.disabledWeekDays && $.isArray(_options.disabledWeekDays) && _options.disabledWeekDays.length) { + options.disabledWeekDays = $.extend(true, [], _options.disabledWeekDays); + } + + if ((options.open || options.opened) && (!options.inline)) { + input.trigger('open.xdsoft'); + } + + if (options.inline) { + triggerAfterOpen = true; + datetimepicker.addClass('xdsoft_inline'); + input.after(datetimepicker).hide(); + } + + if (options.inverseButton) { + options.next = 'xdsoft_prev'; + options.prev = 'xdsoft_next'; + } + + if (options.datepicker) { + datepicker.addClass('active'); + } else { + datepicker.removeClass('active'); + } + + if (options.timepicker) { + timepicker.addClass('active'); + } else { + timepicker.removeClass('active'); + } + + if (options.value) { + _xdsoft_datetime.setCurrentTime(options.value); + if (input && input.val) { + input.val(_xdsoft_datetime.str); + } + } + + if (isNaN(options.dayOfWeekStart)) { + options.dayOfWeekStart = 0; + } else { + options.dayOfWeekStart = parseInt(options.dayOfWeekStart, 10) % 7; + } + + if (!options.timepickerScrollbar) { + timeboxparent.xdsoftScroller(options, 'hide'); + } + + if (options.minDate && /^[\+\-](.*)$/.test(options.minDate)) { + options.minDate = dateHelper.formatDate(_xdsoft_datetime.strToDateTime(options.minDate), options.formatDate); + } + + if (options.maxDate && /^[\+\-](.*)$/.test(options.maxDate)) { + options.maxDate = dateHelper.formatDate(_xdsoft_datetime.strToDateTime(options.maxDate), options.formatDate); + } + + if (options.minDateTime && /^\+(.*)$/.test(options.minDateTime)) { + options.minDateTime = _xdsoft_datetime.strToDateTime(options.minDateTime).dateFormat(options.formatDate); + } + + if (options.maxDateTime && /^\+(.*)$/.test(options.maxDateTime)) { + options.maxDateTime = _xdsoft_datetime.strToDateTime(options.maxDateTime).dateFormat(options.formatDate); + } + + applyButton.toggle(options.showApplyButton); + + month_picker + .find('.xdsoft_today_button') + .css('visibility', !options.todayButton ? 'hidden' : 'visible'); + + month_picker + .find('.' + options.prev) + .css('visibility', !options.prevButton ? 'hidden' : 'visible'); + + month_picker + .find('.' + options.next) + .css('visibility', !options.nextButton ? 'hidden' : 'visible'); + + setMask(options); + + if (options.validateOnBlur) { + input + .off('blur.xdsoft') + .on('blur.xdsoft', function () { + if (options.allowBlank && (!$.trim($(this).val()).length || + (typeof options.mask === "string" && $.trim($(this).val()) === options.mask.replace(/[0-9]/g, '_')))) { + $(this).val(null); + datetimepicker.data('xdsoft_datetime').empty(); + } else { + var d = dateHelper.parseDate($(this).val(), options.format); + if (d) { // parseDate() may skip some invalid parts like date or time, so make it clear for user: show parsed date/time + $(this).val(dateHelper.formatDate(d, options.format)); + } else { + var splittedHours = +([$(this).val()[0], $(this).val()[1]].join('')), + splittedMinutes = +([$(this).val()[2], $(this).val()[3]].join('')); + + // parse the numbers as 0312 => 03:12 + if (!options.datepicker && options.timepicker && splittedHours >= 0 && splittedHours < 24 && splittedMinutes >= 0 && splittedMinutes < 60) { + $(this).val([splittedHours, splittedMinutes].map(function (item) { + return item > 9 ? item : '0' + item; + }).join(':')); + } else { + $(this).val(dateHelper.formatDate(_xdsoft_datetime.now(), options.format)); + } + } + datetimepicker.data('xdsoft_datetime').setCurrentTime($(this).val()); + } + + datetimepicker.trigger('changedatetime.xdsoft'); + datetimepicker.trigger('close.xdsoft'); + }); + } + options.dayOfWeekStartPrev = (options.dayOfWeekStart === 0) ? 6 : options.dayOfWeekStart - 1; + + datetimepicker + .trigger('xchange.xdsoft') + .trigger('afterOpen.xdsoft'); + }; + + datetimepicker + .data('options', options) + .on('touchstart mousedown.xdsoft', function (event) { + event.stopPropagation(); + event.preventDefault(); + yearselect.hide(); + monthselect.hide(); + return false; + }); + + //scroll_element = timepicker.find('.xdsoft_time_box'); + timeboxparent.append(timebox); + timeboxparent.xdsoftScroller(options); + + datetimepicker.on('afterOpen.xdsoft', function () { + timeboxparent.xdsoftScroller(options); + }); + + datetimepicker + .append(datepicker) + .append(timepicker); + + if (options.withoutCopyright !== true) { + datetimepicker + .append(xdsoft_copyright); + } + + datepicker + .append(month_picker) + .append(calendar) + .append(applyButton); + + $(options.parentID) + .append(datetimepicker); + + XDSoft_datetime = function () { + var _this = this; + _this.now = function (norecursion) { + var d = new Date(), + date, + time; + + if (!norecursion && options.defaultDate) { + date = _this.strToDateTime(options.defaultDate); + d.setFullYear(date.getFullYear()); + d.setMonth(date.getMonth()); + d.setDate(date.getDate()); + } + + d.setFullYear(d.getFullYear()); + + if (!norecursion && options.defaultTime) { + time = _this.strtotime(options.defaultTime); + d.setHours(time.getHours()); + d.setMinutes(time.getMinutes()); + d.setSeconds(time.getSeconds()); + d.setMilliseconds(time.getMilliseconds()); + } + return d; + }; + + _this.isValidDate = function (d) { + if (Object.prototype.toString.call(d) !== "[object Date]") { + return false; + } + return !isNaN(d.getTime()); + }; + + _this.setCurrentTime = function (dTime, requireValidDate) { + if (typeof dTime === 'string') { + _this.currentTime = _this.strToDateTime(dTime); + } + else if (_this.isValidDate(dTime)) { + _this.currentTime = dTime; + } + else if (!dTime && !requireValidDate && options.allowBlank && !options.inline) { + _this.currentTime = null; + } + else { + _this.currentTime = _this.now(); + } + + datetimepicker.trigger('xchange.xdsoft'); + }; + + _this.empty = function () { + _this.currentTime = null; + }; + + _this.getCurrentTime = function () { + return _this.currentTime; + }; + + _this.nextMonth = function () { + + if (_this.currentTime === undefined || _this.currentTime === null) { + _this.currentTime = _this.now(); + } + + var month = _this.currentTime.getMonth() + 1, + year; + if (month === 12) { + _this.currentTime.setFullYear(_this.currentTime.getFullYear() + 1); + month = 0; + } + + year = _this.currentTime.getFullYear(); + + _this.currentTime.setDate( + Math.min( + new Date(_this.currentTime.getFullYear(), month + 1, 0).getDate(), + _this.currentTime.getDate() + ) + ); + _this.currentTime.setMonth(month); + + if (options.onChangeMonth && $.isFunction(options.onChangeMonth)) { + options.onChangeMonth.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input')); + } + + if (year !== _this.currentTime.getFullYear() && $.isFunction(options.onChangeYear)) { + options.onChangeYear.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input')); + } + + datetimepicker.trigger('xchange.xdsoft'); + return month; + }; + + _this.prevMonth = function () { + + if (_this.currentTime === undefined || _this.currentTime === null) { + _this.currentTime = _this.now(); + } + + var month = _this.currentTime.getMonth() - 1; + if (month === -1) { + _this.currentTime.setFullYear(_this.currentTime.getFullYear() - 1); + month = 11; + } + _this.currentTime.setDate( + Math.min( + new Date(_this.currentTime.getFullYear(), month + 1, 0).getDate(), + _this.currentTime.getDate() + ) + ); + _this.currentTime.setMonth(month); + if (options.onChangeMonth && $.isFunction(options.onChangeMonth)) { + options.onChangeMonth.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input')); + } + datetimepicker.trigger('xchange.xdsoft'); + return month; + }; + + _this.getWeekOfYear = function (datetime) { + if (options.onGetWeekOfYear && $.isFunction(options.onGetWeekOfYear)) { + var week = options.onGetWeekOfYear.call(datetimepicker, datetime); + if (typeof week !== 'undefined') { + return week; + } + } + var onejan = new Date(datetime.getFullYear(), 0, 1); + + //First week of the year is th one with the first Thursday according to ISO8601 + if (onejan.getDay() !== 4) { + onejan.setMonth(0, 1 + ((4 - onejan.getDay()+ 7) % 7)); + } + + return Math.ceil((((datetime - onejan) / 86400000) + onejan.getDay() + 1) / 7); + }; + + _this.strToDateTime = function (sDateTime) { + var tmpDate = [], timeOffset, currentTime; + + if (sDateTime && sDateTime instanceof Date && _this.isValidDate(sDateTime)) { + return sDateTime; + } + + tmpDate = /^([+-]{1})(.*)$/.exec(sDateTime); + + if (tmpDate) { + tmpDate[2] = dateHelper.parseDate(tmpDate[2], options.formatDate); + } + + if (tmpDate && tmpDate[2]) { + timeOffset = tmpDate[2].getTime() - (tmpDate[2].getTimezoneOffset()) * 60000; + currentTime = new Date((_this.now(true)).getTime() + parseInt(tmpDate[1] + '1', 10) * timeOffset); + } else { + currentTime = sDateTime ? dateHelper.parseDate(sDateTime, options.format) : _this.now(); + } + + if (!_this.isValidDate(currentTime)) { + currentTime = _this.now(); + } + + return currentTime; + }; + + _this.strToDate = function (sDate) { + if (sDate && sDate instanceof Date && _this.isValidDate(sDate)) { + return sDate; + } + + var currentTime = sDate ? dateHelper.parseDate(sDate, options.formatDate) : _this.now(true); + if (!_this.isValidDate(currentTime)) { + currentTime = _this.now(true); + } + return currentTime; + }; + + _this.strtotime = function (sTime) { + if (sTime && sTime instanceof Date && _this.isValidDate(sTime)) { + return sTime; + } + var currentTime = sTime ? dateHelper.parseDate(sTime, options.formatTime) : _this.now(true); + if (!_this.isValidDate(currentTime)) { + currentTime = _this.now(true); + } + return currentTime; + }; + + _this.str = function () { + var format = options.format; + if (options.yearOffset) { + format = format.replace('Y', _this.currentTime.getFullYear() + options.yearOffset); + format = format.replace('y', String(_this.currentTime.getFullYear() + options.yearOffset).substring(2, 4)); + } + return dateHelper.formatDate(_this.currentTime, format); + }; + _this.currentTime = this.now(); + }; + + _xdsoft_datetime = new XDSoft_datetime(); + + applyButton.on('touchend click', function (e) {//pathbrite + e.preventDefault(); + datetimepicker.data('changed', true); + _xdsoft_datetime.setCurrentTime(getCurrentValue()); + input.val(_xdsoft_datetime.str()); + datetimepicker.trigger('close.xdsoft'); + }); + month_picker + .find('.xdsoft_today_button') + .on('touchend mousedown.xdsoft', function () { + datetimepicker.data('changed', true); + _xdsoft_datetime.setCurrentTime(0, true); + datetimepicker.trigger('afterOpen.xdsoft'); + }).on('dblclick.xdsoft', function () { + var currentDate = _xdsoft_datetime.getCurrentTime(), minDate, maxDate; + currentDate = new Date(currentDate.getFullYear(), currentDate.getMonth(), currentDate.getDate()); + minDate = _xdsoft_datetime.strToDate(options.minDate); + minDate = new Date(minDate.getFullYear(), minDate.getMonth(), minDate.getDate()); + if (currentDate < minDate) { + return; + } + maxDate = _xdsoft_datetime.strToDate(options.maxDate); + maxDate = new Date(maxDate.getFullYear(), maxDate.getMonth(), maxDate.getDate()); + if (currentDate > maxDate) { + return; + } + input.val(_xdsoft_datetime.str()); + input.trigger('change'); + datetimepicker.trigger('close.xdsoft'); + }); + month_picker + .find('.xdsoft_prev,.xdsoft_next') + .on('touchend mousedown.xdsoft', function () { + var $this = $(this), + timer = 0, + stop = false; + + (function arguments_callee1(v) { + if ($this.hasClass(options.next)) { + _xdsoft_datetime.nextMonth(); + } else if ($this.hasClass(options.prev)) { + _xdsoft_datetime.prevMonth(); + } + if (options.monthChangeSpinner) { + if (!stop) { + timer = setTimeout(arguments_callee1, v || 100); + } + } + }(500)); + + $([options.ownerDocument.body, options.contentWindow]).on('touchend mouseup.xdsoft', function arguments_callee2() { + clearTimeout(timer); + stop = true; + $([options.ownerDocument.body, options.contentWindow]).off('touchend mouseup.xdsoft', arguments_callee2); + }); + }); + + timepicker + .find('.xdsoft_prev,.xdsoft_next') + .on('touchend mousedown.xdsoft', function () { + var $this = $(this), + timer = 0, + stop = false, + period = 110; + (function arguments_callee4(v) { + var pheight = timeboxparent[0].clientHeight, + height = timebox[0].offsetHeight, + top = Math.abs(parseInt(timebox.css('marginTop'), 10)); + if ($this.hasClass(options.next) && (height - pheight) - options.timeHeightInTimePicker >= top) { + timebox.css('marginTop', '-' + (top + options.timeHeightInTimePicker) + 'px'); + } else if ($this.hasClass(options.prev) && top - options.timeHeightInTimePicker >= 0) { + timebox.css('marginTop', '-' + (top - options.timeHeightInTimePicker) + 'px'); + } + /** + * Fixed bug: + * When using css3 transition, it will cause a bug that you cannot scroll the timepicker list. + * The reason is that the transition-duration time, if you set it to 0, all things fine, otherwise, this + * would cause a bug when you use jquery.css method. + * Let's say: * { transition: all .5s ease; } + * jquery timebox.css('marginTop') will return the original value which is before you clicking the next/prev button, + * meanwhile the timebox[0].style.marginTop will return the right value which is after you clicking the + * next/prev button. + * + * What we should do: + * Replace timebox.css('marginTop') with timebox[0].style.marginTop. + */ + timeboxparent.trigger('scroll_element.xdsoft_scroller', [Math.abs(parseInt(timebox[0].style.marginTop, 10) / (height - pheight))]); + period = (period > 10) ? 10 : period - 10; + if (!stop) { + timer = setTimeout(arguments_callee4, v || period); + } + }(500)); + $([options.ownerDocument.body, options.contentWindow]).on('touchend mouseup.xdsoft', function arguments_callee5() { + clearTimeout(timer); + stop = true; + $([options.ownerDocument.body, options.contentWindow]) + .off('touchend mouseup.xdsoft', arguments_callee5); + }); + }); + + xchangeTimer = 0; + // base handler - generating a calendar and timepicker + datetimepicker + .on('xchange.xdsoft', function (event) { + clearTimeout(xchangeTimer); + xchangeTimer = setTimeout(function () { + + if (_xdsoft_datetime.currentTime === undefined || _xdsoft_datetime.currentTime === null) { + _xdsoft_datetime.currentTime = _xdsoft_datetime.now(); + } + + var table = '', + start = new Date(_xdsoft_datetime.currentTime.getFullYear(), _xdsoft_datetime.currentTime.getMonth(), 1, 12, 0, 0), + i = 0, + j, + today = _xdsoft_datetime.now(), + maxDate = false, + minDate = false, + minDateTime = false, + maxDateTime = false, + hDate, + day, + d, + y, + m, + w, + classes = [], + customDateSettings, + newRow = true, + time = '', + h, + line_time, + description; + + while (start.getDay() !== options.dayOfWeekStart) { + start.setDate(start.getDate() - 1); + } + + table += ''; + + if (options.weeks) { + table += ''; + } + + for (j = 0; j < 7; j += 1) { + table += ''; + } + + table += ''; + table += ''; + + if (options.maxDate !== false) { + maxDate = _xdsoft_datetime.strToDate(options.maxDate); + maxDate = new Date(maxDate.getFullYear(), maxDate.getMonth(), maxDate.getDate(), 23, 59, 59, 999); + } + + if (options.minDate !== false) { + minDate = _xdsoft_datetime.strToDate(options.minDate); + minDate = new Date(minDate.getFullYear(), minDate.getMonth(), minDate.getDate()); + } + + if (options.minDateTime !== false) { + minDateTime = _xdsoft_datetime.strToDate(options.minDateTime); + minDateTime = new Date(minDateTime.getFullYear(), minDateTime.getMonth(), minDateTime.getDate(), minDateTime.getHours(), minDateTime.getMinutes(), minDateTime.getSeconds()); + } + + if (options.maxDateTime !== false) { + maxDateTime = _xdsoft_datetime.strToDate(options.maxDateTime); + maxDateTime = new Date(maxDateTime.getFullYear(), maxDateTime.getMonth(), maxDateTime.getDate(), maxDateTime.getHours(), maxDateTime.getMinutes(), maxDateTime.getSeconds()); + } + + var maxDateTimeDay; + if (maxDateTime !== false) { + maxDateTimeDay = ((maxDateTime.getFullYear() * 12) + maxDateTime.getMonth()) * 31 + maxDateTime.getDate(); + } + + while (i < _xdsoft_datetime.currentTime.countDaysInMonth() || start.getDay() !== options.dayOfWeekStart || _xdsoft_datetime.currentTime.getMonth() === start.getMonth()) { + classes = []; + i += 1; + + day = start.getDay(); + d = start.getDate(); + y = start.getFullYear(); + m = start.getMonth(); + w = _xdsoft_datetime.getWeekOfYear(start); + description = ''; + + classes.push('xdsoft_date'); + + if (options.beforeShowDay && $.isFunction(options.beforeShowDay.call)) { + customDateSettings = options.beforeShowDay.call(datetimepicker, start); + } else { + customDateSettings = null; + } + + if(options.allowDateRe && Object.prototype.toString.call(options.allowDateRe) === "[object RegExp]"){ + if(!options.allowDateRe.test(dateHelper.formatDate(start, options.formatDate))){ + classes.push('xdsoft_disabled'); + } + } + + if(options.allowDates && options.allowDates.length>0){ + if(options.allowDates.indexOf(dateHelper.formatDate(start, options.formatDate)) === -1){ + classes.push('xdsoft_disabled'); + } + } + + var currentDay = ((start.getFullYear() * 12) + start.getMonth()) * 31 + start.getDate(); + if ((maxDate !== false && start > maxDate) || (minDateTime !== false && start < minDateTime) || (minDate !== false && start < minDate) || (maxDateTime !== false && currentDay > maxDateTimeDay) || (customDateSettings && customDateSettings[0] === false)) { + classes.push('xdsoft_disabled'); + } + + if (options.disabledDates.indexOf(dateHelper.formatDate(start, options.formatDate)) !== -1) { + classes.push('xdsoft_disabled'); + } + + if (options.disabledWeekDays.indexOf(day) !== -1) { + classes.push('xdsoft_disabled'); + } + + if (input.is('[disabled]')) { + classes.push('xdsoft_disabled'); + } + + if (customDateSettings && customDateSettings[1] !== "") { + classes.push(customDateSettings[1]); + } + + if (_xdsoft_datetime.currentTime.getMonth() !== m) { + classes.push('xdsoft_other_month'); + } + + if ((options.defaultSelect || datetimepicker.data('changed')) && dateHelper.formatDate(_xdsoft_datetime.currentTime, options.formatDate) === dateHelper.formatDate(start, options.formatDate)) { + classes.push('xdsoft_current'); + } + + if (dateHelper.formatDate(today, options.formatDate) === dateHelper.formatDate(start, options.formatDate)) { + classes.push('xdsoft_today'); + } + + if (start.getDay() === 0 || start.getDay() === 6 || options.weekends.indexOf(dateHelper.formatDate(start, options.formatDate)) !== -1) { + classes.push('xdsoft_weekend'); + } + + if (options.highlightedDates[dateHelper.formatDate(start, options.formatDate)] !== undefined) { + hDate = options.highlightedDates[dateHelper.formatDate(start, options.formatDate)]; + classes.push(hDate.style === undefined ? 'xdsoft_highlighted_default' : hDate.style); + description = hDate.desc === undefined ? '' : hDate.desc; + } + + if (options.beforeShowDay && $.isFunction(options.beforeShowDay)) { + classes.push(options.beforeShowDay(start)); + } + + if (newRow) { + table += ''; + newRow = false; + if (options.weeks) { + table += ''; + } + } + + table += ''; + + if (start.getDay() === options.dayOfWeekStartPrev) { + table += ''; + newRow = true; + } + + start.setDate(d + 1); + } + table += '
' + options.i18n[globalLocale].dayOfWeekShort[(j + options.dayOfWeekStart) % 7] + '
' + w + '' + + '
' + d + '
' + + '
'; + + calendar.html(table); + + month_picker.find('.xdsoft_label span').eq(0).text(options.i18n[globalLocale].months[_xdsoft_datetime.currentTime.getMonth()]); + month_picker.find('.xdsoft_label span').eq(1).text(_xdsoft_datetime.currentTime.getFullYear() + options.yearOffset); + + // generate timebox + time = ''; + h = ''; + m = ''; + + var minTimeMinutesOfDay = 0; + if (options.minTime !== false) { + var t = _xdsoft_datetime.strtotime(options.minTime); + minTimeMinutesOfDay = 60 * t.getHours() + t.getMinutes(); + } + var maxTimeMinutesOfDay = 24 * 60; + if (options.maxTime !== false) { + var t = _xdsoft_datetime.strtotime(options.maxTime); + maxTimeMinutesOfDay = 60 * t.getHours() + t.getMinutes(); + } + + if (options.minDateTime !== false) { + var t = _xdsoft_datetime.strToDateTime(options.minDateTime); + var currentDayIsMinDateTimeDay = dateHelper.formatDate(_xdsoft_datetime.currentTime, options.formatDate) === dateHelper.formatDate(t, options.formatDate); + if (currentDayIsMinDateTimeDay) { + var m = 60 * t.getHours() + t.getMinutes(); + if (m > minTimeMinutesOfDay) minTimeMinutesOfDay = m; + } + } + + if (options.maxDateTime !== false) { + var t = _xdsoft_datetime.strToDateTime(options.maxDateTime); + var currentDayIsMaxDateTimeDay = dateHelper.formatDate(_xdsoft_datetime.currentTime, options.formatDate) === dateHelper.formatDate(t, options.formatDate); + if (currentDayIsMaxDateTimeDay) { + var m = 60 * t.getHours() + t.getMinutes(); + if (m < maxTimeMinutesOfDay) maxTimeMinutesOfDay = m; + } + } + + line_time = function line_time(h, m) { + var now = _xdsoft_datetime.now(), current_time, + isALlowTimesInit = options.allowTimes && $.isArray(options.allowTimes) && options.allowTimes.length; + now.setHours(h); + h = parseInt(now.getHours(), 10); + now.setMinutes(m); + m = parseInt(now.getMinutes(), 10); + classes = []; + var currentMinutesOfDay = 60 * h + m; + if (input.is('[disabled]') || (currentMinutesOfDay >= maxTimeMinutesOfDay) || (currentMinutesOfDay < minTimeMinutesOfDay)) { + classes.push('xdsoft_disabled'); + } + + current_time = new Date(_xdsoft_datetime.currentTime); + current_time.setHours(parseInt(_xdsoft_datetime.currentTime.getHours(), 10)); + + if (!isALlowTimesInit) { + current_time.setMinutes(Math[options.roundTime](_xdsoft_datetime.currentTime.getMinutes() / options.step) * options.step); + } + + if ((options.initTime || options.defaultSelect || datetimepicker.data('changed')) && current_time.getHours() === parseInt(h, 10) && ((!isALlowTimesInit && options.step > 59) || current_time.getMinutes() === parseInt(m, 10))) { + if (options.defaultSelect || datetimepicker.data('changed')) { + classes.push('xdsoft_current'); + } else if (options.initTime) { + classes.push('xdsoft_init_time'); + } + } + if (parseInt(today.getHours(), 10) === parseInt(h, 10) && parseInt(today.getMinutes(), 10) === parseInt(m, 10)) { + classes.push('xdsoft_today'); + } + time += '
' + dateHelper.formatDate(now, options.formatTime) + '
'; + }; + + if (!options.allowTimes || !$.isArray(options.allowTimes) || !options.allowTimes.length) { + for (i = 0, j = 0; i < (options.hours12 ? 12 : 24); i += 1) { + for (j = 0; j < 60; j += options.step) { + var currentMinutesOfDay = i * 60 + j; + if (currentMinutesOfDay < minTimeMinutesOfDay) continue; + if (currentMinutesOfDay >= maxTimeMinutesOfDay) continue; + h = (i < 10 ? '0' : '') + i; + m = (j < 10 ? '0' : '') + j; + line_time(h, m); + } + } + } else { + for (i = 0; i < options.allowTimes.length; i += 1) { + h = _xdsoft_datetime.strtotime(options.allowTimes[i]).getHours(); + m = _xdsoft_datetime.strtotime(options.allowTimes[i]).getMinutes(); + line_time(h, m); + } + } + + timebox.html(time); + + opt = ''; + + for (i = parseInt(options.yearStart, 10); i <= parseInt(options.yearEnd, 10); i += 1) { + opt += '
' + (i + options.yearOffset) + '
'; + } + yearselect.children().eq(0) + .html(opt); + + for (i = parseInt(options.monthStart, 10), opt = ''; i <= parseInt(options.monthEnd, 10); i += 1) { + opt += '
' + options.i18n[globalLocale].months[i] + '
'; + } + monthselect.children().eq(0).html(opt); + $(datetimepicker) + .trigger('generate.xdsoft'); + }, 10); + event.stopPropagation(); + }) + .on('afterOpen.xdsoft', function () { + if (options.timepicker) { + var classType, pheight, height, top; + if (timebox.find('.xdsoft_current').length) { + classType = '.xdsoft_current'; + } else if (timebox.find('.xdsoft_init_time').length) { + classType = '.xdsoft_init_time'; + } + if (classType) { + pheight = timeboxparent[0].clientHeight; + height = timebox[0].offsetHeight; + top = timebox.find(classType).index() * options.timeHeightInTimePicker + 1; + if ((height - pheight) < top) { + top = height - pheight; + } + timeboxparent.trigger('scroll_element.xdsoft_scroller', [parseInt(top, 10) / (height - pheight)]); + } else { + timeboxparent.trigger('scroll_element.xdsoft_scroller', [0]); + } + } + }); + + timerclick = 0; + calendar + .on('touchend click.xdsoft', 'td', function (xdevent) { + xdevent.stopPropagation(); // Prevents closing of Pop-ups, Modals and Flyouts in Bootstrap + timerclick += 1; + var $this = $(this), + currentTime = _xdsoft_datetime.currentTime; + + if (currentTime === undefined || currentTime === null) { + _xdsoft_datetime.currentTime = _xdsoft_datetime.now(); + currentTime = _xdsoft_datetime.currentTime; + } + + if ($this.hasClass('xdsoft_disabled')) { + return false; + } + + currentTime.setDate(1); + currentTime.setFullYear($this.data('year')); + currentTime.setMonth($this.data('month')); + currentTime.setDate($this.data('date')); + + datetimepicker.trigger('select.xdsoft', [currentTime]); + + input.val(_xdsoft_datetime.str()); + + if (options.onSelectDate && $.isFunction(options.onSelectDate)) { + options.onSelectDate.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input'), xdevent); + } + + datetimepicker.data('changed', true); + datetimepicker.trigger('xchange.xdsoft'); + datetimepicker.trigger('changedatetime.xdsoft'); + if ((timerclick > 1 || (options.closeOnDateSelect === true || (options.closeOnDateSelect === false && !options.timepicker))) && !options.inline) { + datetimepicker.trigger('close.xdsoft'); + } + setTimeout(function () { + timerclick = 0; + }, 200); + }); + + timebox + .on('touchstart', 'div', function (xdevent) { + this.touchMoved = false; + }) + .on('touchmove', 'div', handleTouchMoved) + .on('touchend click.xdsoft', 'div', function (xdevent) { + if (!this.touchMoved) { + xdevent.stopPropagation(); + var $this = $(this), + currentTime = _xdsoft_datetime.currentTime; + + if (currentTime === undefined || currentTime === null) { + _xdsoft_datetime.currentTime = _xdsoft_datetime.now(); + currentTime = _xdsoft_datetime.currentTime; + } + + if ($this.hasClass('xdsoft_disabled')) { + return false; + } + currentTime.setHours($this.data('hour')); + currentTime.setMinutes($this.data('minute')); + datetimepicker.trigger('select.xdsoft', [currentTime]); + + datetimepicker.data('input').val(_xdsoft_datetime.str()); + + if (options.onSelectTime && $.isFunction(options.onSelectTime)) { + options.onSelectTime.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input'), xdevent); + } + datetimepicker.data('changed', true); + datetimepicker.trigger('xchange.xdsoft'); + datetimepicker.trigger('changedatetime.xdsoft'); + if (options.inline !== true && options.closeOnTimeSelect === true) { + datetimepicker.trigger('close.xdsoft'); + } + } + }); + + datepicker + .on('mousewheel.xdsoft', function (event) { + if (!options.scrollMonth) { + return true; + } + if (event.deltaY < 0) { + _xdsoft_datetime.nextMonth(); + } else { + _xdsoft_datetime.prevMonth(); + } + return false; + }); + + input + .on('mousewheel.xdsoft', function (event) { + if (!options.scrollInput) { + return true; + } + if (!options.datepicker && options.timepicker) { + current_time_index = timebox.find('.xdsoft_current').length ? timebox.find('.xdsoft_current').eq(0).index() : 0; + if (current_time_index + event.deltaY >= 0 && current_time_index + event.deltaY < timebox.children().length) { + current_time_index += event.deltaY; + } + if (timebox.children().eq(current_time_index).length) { + timebox.children().eq(current_time_index).trigger('mousedown'); + } + return false; + } + if (options.datepicker && !options.timepicker) { + datepicker.trigger(event, [event.deltaY, event.deltaX, event.deltaY]); + if (input.val) { + input.val(_xdsoft_datetime.str()); + } + datetimepicker.trigger('changedatetime.xdsoft'); + return false; + } + }); + + datetimepicker + .on('changedatetime.xdsoft', function (event) { + if (options.onChangeDateTime && $.isFunction(options.onChangeDateTime)) { + var $input = datetimepicker.data('input'); + options.onChangeDateTime.call(datetimepicker, _xdsoft_datetime.currentTime, $input, event); + delete options.value; + $input.trigger('change'); + } + }) + .on('generate.xdsoft', function () { + if (options.onGenerate && $.isFunction(options.onGenerate)) { + options.onGenerate.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input')); + } + if (triggerAfterOpen) { + datetimepicker.trigger('afterOpen.xdsoft'); + triggerAfterOpen = false; + } + }) + .on('click.xdsoft', function (xdevent) { + xdevent.stopPropagation(); + }); + + current_time_index = 0; + + /** + * Runs the callback for each of the specified node's ancestors. + * + * Return FALSE from the callback to stop ascending. + * + * @param {DOMNode} node + * @param {Function} callback + * @returns {undefined} + */ + forEachAncestorOf = function (node, callback) { + do { + node = node.parentNode; + + if (!node || callback(node) === false) { + break; + } + } while (node.nodeName !== 'HTML'); + }; + + /** + * Sets the position of the picker. + * + * @returns {undefined} + */ + setPos = function () { + var dateInputOffset, + dateInputElem, + verticalPosition, + left, + position, + datetimepickerElem, + dateInputHasFixedAncestor, + $dateInput, + windowWidth, + verticalAnchorEdge, + datetimepickerCss, + windowHeight, + windowScrollTop; + + $dateInput = datetimepicker.data('input'); + dateInputOffset = $dateInput.offset(); + dateInputElem = $dateInput[0]; + + verticalAnchorEdge = 'top'; + verticalPosition = (dateInputOffset.top + dateInputElem.offsetHeight) - 1; + left = dateInputOffset.left; + position = "absolute"; + + windowWidth = $(options.contentWindow).width(); + windowHeight = $(options.contentWindow).height(); + windowScrollTop = $(options.contentWindow).scrollTop(); + + if ((options.ownerDocument.documentElement.clientWidth - dateInputOffset.left) < datepicker.parent().outerWidth(true)) { + var diff = datepicker.parent().outerWidth(true) - dateInputElem.offsetWidth; + left = left - diff; + } + + if ($dateInput.parent().css('direction') === 'rtl') { + left -= (datetimepicker.outerWidth() - $dateInput.outerWidth()); + } + + if (options.fixed) { + verticalPosition -= windowScrollTop; + left -= $(options.contentWindow).scrollLeft(); + position = "fixed"; + } else { + dateInputHasFixedAncestor = false; + + forEachAncestorOf(dateInputElem, function (ancestorNode) { + if (ancestorNode === null) { + return false; + } + + if (options.contentWindow.getComputedStyle(ancestorNode).getPropertyValue('position') === 'fixed') { + dateInputHasFixedAncestor = true; + return false; + } + }); + + if (dateInputHasFixedAncestor) { + position = 'fixed'; + + //If the picker won't fit entirely within the viewport then display it above the date input. + if (verticalPosition + datetimepicker.outerHeight() > windowHeight + windowScrollTop) { + verticalAnchorEdge = 'bottom'; + verticalPosition = (windowHeight + windowScrollTop) - dateInputOffset.top; + } else { + verticalPosition -= windowScrollTop; + } + } else { + if (verticalPosition + datetimepicker[0].offsetHeight > windowHeight + windowScrollTop) { + verticalPosition = dateInputOffset.top - datetimepicker[0].offsetHeight + 1; + } + } + + if (verticalPosition < 0) { + verticalPosition = 0; + } + + if (left + dateInputElem.offsetWidth > windowWidth) { + left = windowWidth - dateInputElem.offsetWidth; + } + } + + datetimepickerElem = datetimepicker[0]; + + forEachAncestorOf(datetimepickerElem, function (ancestorNode) { + var ancestorNodePosition; + + ancestorNodePosition = options.contentWindow.getComputedStyle(ancestorNode).getPropertyValue('position'); + + if (ancestorNodePosition === 'relative' && windowWidth >= ancestorNode.offsetWidth) { + left = left - ((windowWidth - ancestorNode.offsetWidth) / 2); + return false; + } + }); + + datetimepickerCss = { + position: position, + left: left, + top: '', //Initialize to prevent previous values interfering with new ones. + bottom: '' //Initialize to prevent previous values interfering with new ones. + }; + + datetimepickerCss[verticalAnchorEdge] = verticalPosition; + + datetimepicker.css(datetimepickerCss); + }; + + datetimepicker + .on('open.xdsoft', function (event) { + var onShow = true; + if (options.onShow && $.isFunction(options.onShow)) { + onShow = options.onShow.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input'), event); + } + if (onShow !== false) { + datetimepicker.show(); + setPos(); + $(options.contentWindow) + .off('resize.xdsoft', setPos) + .on('resize.xdsoft', setPos); + + if (options.closeOnWithoutClick) { + $([options.ownerDocument.body, options.contentWindow]).on('touchstart mousedown.xdsoft', function arguments_callee6() { + datetimepicker.trigger('close.xdsoft'); + $([options.ownerDocument.body, options.contentWindow]).off('touchstart mousedown.xdsoft', arguments_callee6); + }); + } + } + }) + .on('close.xdsoft', function (event) { + var onClose = true; + month_picker + .find('.xdsoft_month,.xdsoft_year') + .find('.xdsoft_select') + .hide(); + if (options.onClose && $.isFunction(options.onClose)) { + onClose = options.onClose.call(datetimepicker, _xdsoft_datetime.currentTime, datetimepicker.data('input'), event); + } + if (onClose !== false && !options.opened && !options.inline) { + datetimepicker.hide(); + } + event.stopPropagation(); + }) + .on('toggle.xdsoft', function () { + if (datetimepicker.is(':visible')) { + datetimepicker.trigger('close.xdsoft'); + } else { + datetimepicker.trigger('open.xdsoft'); + } + }) + .data('input', input); + + timer = 0; + + datetimepicker.data('xdsoft_datetime', _xdsoft_datetime); + datetimepicker.setOptions(options); + + function getCurrentValue() { + var ct = false, time; + + if (options.startDate) { + ct = _xdsoft_datetime.strToDate(options.startDate); + } else { + ct = options.value || ((input && input.val && input.val()) ? input.val() : ''); + if (ct) { + ct = _xdsoft_datetime.strToDateTime(ct); + if (options.yearOffset) { + ct = new Date(ct.getFullYear() - options.yearOffset, ct.getMonth(), ct.getDate(), ct.getHours(), ct.getMinutes(), ct.getSeconds(), ct.getMilliseconds()); + } + } else if (options.defaultDate) { + ct = _xdsoft_datetime.strToDateTime(options.defaultDate); + if (options.defaultTime) { + time = _xdsoft_datetime.strtotime(options.defaultTime); + ct.setHours(time.getHours()); + ct.setMinutes(time.getMinutes()); + } + } + } + + if (ct && _xdsoft_datetime.isValidDate(ct)) { + datetimepicker.data('changed', true); + } else { + ct = ''; + } + + return ct || 0; + } + + function setMask(options) { + + var isValidValue = function (mask, value) { + var reg = mask + .replace(/([\[\]\/\{\}\(\)\-\.\+]{1})/g, '\\$1') + .replace(/_/g, '{digit+}') + .replace(/([0-9]{1})/g, '{digit$1}') + .replace(/\{digit([0-9]{1})\}/g, '[0-$1_]{1}') + .replace(/\{digit[\+]\}/g, '[0-9_]{1}'); + return (new RegExp(reg)).test(value); + }, + getCaretPos = function (input) { + try { + if (options.ownerDocument.selection && options.ownerDocument.selection.createRange) { + var range = options.ownerDocument.selection.createRange(); + return range.getBookmark().charCodeAt(2) - 2; + } + if (input.setSelectionRange) { + return input.selectionStart; + } + } catch (e) { + return 0; + } + }, + setCaretPos = function (node, pos) { + node = (typeof node === "string" || node instanceof String) ? options.ownerDocument.getElementById(node) : node; + if (!node) { + return false; + } + if (node.createTextRange) { + var textRange = node.createTextRange(); + textRange.collapse(true); + textRange.moveEnd('character', pos); + textRange.moveStart('character', pos); + textRange.select(); + return true; + } + if (node.setSelectionRange) { + node.setSelectionRange(pos, pos); + return true; + } + return false; + }; + + if(options.mask) { + input.off('keydown.xdsoft'); + } + + if (options.mask === true) { + if (dateHelper.formatMask) { + options.mask = dateHelper.formatMask(options.format) + } else { + options.mask = options.format + .replace(/Y/g, '9999') + .replace(/F/g, '9999') + .replace(/m/g, '19') + .replace(/d/g, '39') + .replace(/H/g, '29') + .replace(/i/g, '59') + .replace(/s/g, '59'); + } + } + + if ($.type(options.mask) === 'string') { + if (!isValidValue(options.mask, input.val())) { + input.val(options.mask.replace(/[0-9]/g, '_')); + setCaretPos(input[0], 0); + } + + input.on('paste.xdsoft', function (event) { + // couple options here + // 1. return false - tell them they can't paste + // 2. insert over current characters - minimal validation + // 3. full fledged parsing and validation + // let's go option 2 for now + + // fires multiple times for some reason + + // https://stackoverflow.com/a/30496488/1366033 + var clipboardData = event.clipboardData || event.originalEvent.clipboardData || window.clipboardData, + pastedData = clipboardData.getData('text'), + val = this.value, + pos = this.selectionStart + + var valueBeforeCursor = val.substr(0, pos); + var valueAfterPaste = val.substr(pos + pastedData.length); + + val = valueBeforeCursor + pastedData + valueAfterPaste; + pos += pastedData.length; + + if (isValidValue(options.mask, val)) { + this.value = val; + setCaretPos(this, pos); + } else if ($.trim(val) === '') { + this.value = options.mask.replace(/[0-9]/g, '_'); + } else { + input.trigger('error_input.xdsoft'); + } + + event.preventDefault(); + return false; + }); + + input.on('keydown.xdsoft', function (event) { + var val = this.value, + key = event.which, + pos = this.selectionStart, + selEnd = this.selectionEnd, + hasSel = pos !== selEnd, + digit; + + // only alow these characters + if (((key >= KEY0 && key <= KEY9) || + (key >= _KEY0 && key <= _KEY9)) || + (key === BACKSPACE || key === DEL)) { + + // get char to insert which is new character or placeholder ('_') + digit = (key === BACKSPACE || key === DEL) ? '_' : + String.fromCharCode((_KEY0 <= key && key <= _KEY9) ? key - KEY0 : key); + + // we're deleting something, we're not at the start, and have normal cursor, move back one + // if we have a selection length, cursor actually sits behind deletable char, not in front + if (key === BACKSPACE && pos && !hasSel) { + pos -= 1; + } + + // don't stop on a separator, continue whatever direction you were going + // value char - keep incrementing position while on separator char and we still have room + // del char - keep decrementing position while on separator char and we still have room + while (true) { + var maskValueAtCurPos = options.mask.substr(pos, 1); + var posShorterThanMaskLength = pos < options.mask.length; + var posGreaterThanZero = pos > 0; + var notNumberOrPlaceholder = /[^0-9_]/; + var curPosOnSep = notNumberOrPlaceholder.test(maskValueAtCurPos); + var continueMovingPosition = curPosOnSep && posShorterThanMaskLength && posGreaterThanZero + + // if we hit a real char, stay where we are + if (!continueMovingPosition) break; + + // hitting backspace in a selection, you can possibly go back any further - go forward + pos += (key === BACKSPACE && !hasSel) ? -1 : 1; + + } + + + if (hasSel) { + // pos might have moved so re-calc length + var selLength = selEnd - pos + + // if we have a selection length we will wipe out entire selection and replace with default template for that range + var defaultBlank = options.mask.replace(/[0-9]/g, '_'); + var defaultBlankSelectionReplacement = defaultBlank.substr(pos, selLength); + var selReplacementRemainder = defaultBlankSelectionReplacement.substr(1) // might be empty + + var valueBeforeSel = val.substr(0, pos); + var insertChars = digit + selReplacementRemainder; + var charsAfterSelection = val.substr(pos + selLength); + + val = valueBeforeSel + insertChars + charsAfterSelection + + } else { + var valueBeforeCursor = val.substr(0, pos); + var insertChar = digit; + var valueAfterNextChar = val.substr(pos + 1); + + val = valueBeforeCursor + insertChar + valueAfterNextChar + } + + if ($.trim(val) === '') { + // if empty, set to default + val = defaultBlank + } else { + // if at the last character don't need to do anything + if (pos === options.mask.length) { + event.preventDefault(); + return false; + } + } + + // resume cursor location + pos += (key === BACKSPACE) ? 0 : 1; + // don't stop on a separator, continue whatever direction you were going + while (/[^0-9_]/.test(options.mask.substr(pos, 1)) && pos < options.mask.length && pos > 0) { + pos += (key === BACKSPACE) ? 0 : 1; + } + + if (isValidValue(options.mask, val)) { + this.value = val; + setCaretPos(this, pos); + } else if ($.trim(val) === '') { + this.value = options.mask.replace(/[0-9]/g, '_'); + } else { + input.trigger('error_input.xdsoft'); + } + } else { + if (([AKEY, CKEY, VKEY, ZKEY, YKEY].indexOf(key) !== -1 && ctrlDown) || [ESC, ARROWUP, ARROWDOWN, ARROWLEFT, ARROWRIGHT, F5, CTRLKEY, TAB, ENTER].indexOf(key) !== -1) { + return true; + } + } + + event.preventDefault(); + return false; + }); + } + } + + _xdsoft_datetime.setCurrentTime(getCurrentValue()); + + input + .data('xdsoft_datetimepicker', datetimepicker) + .on('open.xdsoft focusin.xdsoft mousedown.xdsoft touchstart', function () { + if (input.is(':disabled') || (input.data('xdsoft_datetimepicker').is(':visible') && options.closeOnInputClick)) { + return; + } + if (!options.openOnFocus) { + return; + } + clearTimeout(timer); + timer = setTimeout(function () { + if (input.is(':disabled')) { + return; + } + + triggerAfterOpen = true; + _xdsoft_datetime.setCurrentTime(getCurrentValue(), true); + if(options.mask) { + setMask(options); + } + datetimepicker.trigger('open.xdsoft'); + }, 100); + }) + .on('keydown.xdsoft', function (event) { + var elementSelector, + key = event.which; + if ([ENTER].indexOf(key) !== -1 && options.enterLikeTab) { + elementSelector = $("input:visible,textarea:visible,button:visible,a:visible"); + datetimepicker.trigger('close.xdsoft'); + elementSelector.eq(elementSelector.index(this) + 1).focus(); + return false; + } + if ([TAB].indexOf(key) !== -1) { + datetimepicker.trigger('close.xdsoft'); + return true; + } + }) + .on('blur.xdsoft', function () { + datetimepicker.trigger('close.xdsoft'); + }); + }; + destroyDateTimePicker = function (input) { + var datetimepicker = input.data('xdsoft_datetimepicker'); + if (datetimepicker) { + datetimepicker.data('xdsoft_datetime', null); + datetimepicker.remove(); + input + .data('xdsoft_datetimepicker', null) + .off('.xdsoft'); + $(options.contentWindow).off('resize.xdsoft'); + $([options.contentWindow, options.ownerDocument.body]).off('mousedown.xdsoft touchstart'); + if (input.unmousewheel) { + input.unmousewheel(); + } + } + }; + $(options.ownerDocument) + .off('keydown.xdsoftctrl keyup.xdsoftctrl') + .on('keydown.xdsoftctrl', function (e) { + if (e.keyCode === CTRLKEY) { + ctrlDown = true; + } + }) + .on('keyup.xdsoftctrl', function (e) { + if (e.keyCode === CTRLKEY) { + ctrlDown = false; + } + }); + + this.each(function () { + var datetimepicker = $(this).data('xdsoft_datetimepicker'), $input; + if (datetimepicker) { + if ($.type(opt) === 'string') { + switch (opt) { + case 'show': + $(this).select().focus(); + datetimepicker.trigger('open.xdsoft'); + break; + case 'hide': + datetimepicker.trigger('close.xdsoft'); + break; + case 'toggle': + datetimepicker.trigger('toggle.xdsoft'); + break; + case 'destroy': + destroyDateTimePicker($(this)); + break; + case 'reset': + this.value = this.defaultValue; + if (!this.value || !datetimepicker.data('xdsoft_datetime').isValidDate(dateHelper.parseDate(this.value, options.format))) { + datetimepicker.data('changed', false); + } + datetimepicker.data('xdsoft_datetime').setCurrentTime(this.value); + break; + case 'validate': + $input = datetimepicker.data('input'); + $input.trigger('blur.xdsoft'); + break; + default: + if (datetimepicker[opt] && $.isFunction(datetimepicker[opt])) { + result = datetimepicker[opt](opt2); + } + } + } else { + datetimepicker + .setOptions(opt); + } + return 0; + } + if ($.type(opt) !== 'string') { + if (!options.lazyInit || options.open || options.inline) { + createDateTimePicker($(this)); + } else { + lazyInit($(this)); + } + } + }); + + return result; + }; + + $.fn.datetimepicker.defaults = default_options; + + function HighlightedDate(date, desc, style) { + "use strict"; + this.date = date; + this.desc = desc; + this.style = style; + } +}; +;(function (factory) { + if ( typeof define === 'function' && define.amd ) { + // AMD. Register as an anonymous module. + define(['jquery', 'jquery-mousewheel'], factory); + } else if (typeof exports === 'object') { + // Node/CommonJS style for Browserify + module.exports = factory(require('jquery'));; + } else { + // Browser globals + factory(jQuery); + } +}(datetimepickerFactory)); + + +/*! + * jQuery Mousewheel 3.1.13 + * + * Copyright jQuery Foundation and other contributors + * Released under the MIT license + * http://jquery.org/license + */ + +(function (factory) { + if ( typeof define === 'function' && define.amd ) { + // AMD. Register as an anonymous module. + define(['jquery'], factory); + } else if (typeof exports === 'object') { + // Node/CommonJS style for Browserify + module.exports = factory; + } else { + // Browser globals + factory(jQuery); + } +}(function ($) { + + var toFix = ['wheel', 'mousewheel', 'DOMMouseScroll', 'MozMousePixelScroll'], + toBind = ( 'onwheel' in document || document.documentMode >= 9 ) ? + ['wheel'] : ['mousewheel', 'DomMouseScroll', 'MozMousePixelScroll'], + slice = Array.prototype.slice, + nullLowestDeltaTimeout, lowestDelta; + + if ( $.event.fixHooks ) { + for ( var i = toFix.length; i; ) { + $.event.fixHooks[ toFix[--i] ] = $.event.mouseHooks; + } + } + + var special = $.event.special.mousewheel = { + version: '3.1.12', + + setup: function() { + if ( this.addEventListener ) { + for ( var i = toBind.length; i; ) { + this.addEventListener( toBind[--i], handler, false ); + } + } else { + this.onmousewheel = handler; + } + // Store the line height and page height for this particular element + $.data(this, 'mousewheel-line-height', special.getLineHeight(this)); + $.data(this, 'mousewheel-page-height', special.getPageHeight(this)); + }, + + teardown: function() { + if ( this.removeEventListener ) { + for ( var i = toBind.length; i; ) { + this.removeEventListener( toBind[--i], handler, false ); + } + } else { + this.onmousewheel = null; + } + // Clean up the data we added to the element + $.removeData(this, 'mousewheel-line-height'); + $.removeData(this, 'mousewheel-page-height'); + }, + + getLineHeight: function(elem) { + var $elem = $(elem), + $parent = $elem['offsetParent' in $.fn ? 'offsetParent' : 'parent'](); + if (!$parent.length) { + $parent = $('body'); + } + return parseInt($parent.css('fontSize'), 10) || parseInt($elem.css('fontSize'), 10) || 16; + }, + + getPageHeight: function(elem) { + return $(elem).height(); + }, + + settings: { + adjustOldDeltas: true, // see shouldAdjustOldDeltas() below + normalizeOffset: true // calls getBoundingClientRect for each event + } + }; + + $.fn.extend({ + mousewheel: function(fn) { + return fn ? this.bind('mousewheel', fn) : this.trigger('mousewheel'); + }, + + unmousewheel: function(fn) { + return this.unbind('mousewheel', fn); + } + }); + + + function handler(event) { + var orgEvent = event || window.event, + args = slice.call(arguments, 1), + delta = 0, + deltaX = 0, + deltaY = 0, + absDelta = 0, + offsetX = 0, + offsetY = 0; + event = $.event.fix(orgEvent); + event.type = 'mousewheel'; + + // Old school scrollwheel delta + if ( 'detail' in orgEvent ) { deltaY = orgEvent.detail * -1; } + if ( 'wheelDelta' in orgEvent ) { deltaY = orgEvent.wheelDelta; } + if ( 'wheelDeltaY' in orgEvent ) { deltaY = orgEvent.wheelDeltaY; } + if ( 'wheelDeltaX' in orgEvent ) { deltaX = orgEvent.wheelDeltaX * -1; } + + // Firefox < 17 horizontal scrolling related to DOMMouseScroll event + if ( 'axis' in orgEvent && orgEvent.axis === orgEvent.HORIZONTAL_AXIS ) { + deltaX = deltaY * -1; + deltaY = 0; + } + + // Set delta to be deltaY or deltaX if deltaY is 0 for backwards compatabilitiy + delta = deltaY === 0 ? deltaX : deltaY; + + // New school wheel delta (wheel event) + if ( 'deltaY' in orgEvent ) { + deltaY = orgEvent.deltaY * -1; + delta = deltaY; + } + if ( 'deltaX' in orgEvent ) { + deltaX = orgEvent.deltaX; + if ( deltaY === 0 ) { delta = deltaX * -1; } + } + + // No change actually happened, no reason to go any further + if ( deltaY === 0 && deltaX === 0 ) { return; } + + // Need to convert lines and pages to pixels if we aren't already in pixels + // There are three delta modes: + // * deltaMode 0 is by pixels, nothing to do + // * deltaMode 1 is by lines + // * deltaMode 2 is by pages + if ( orgEvent.deltaMode === 1 ) { + var lineHeight = $.data(this, 'mousewheel-line-height'); + delta *= lineHeight; + deltaY *= lineHeight; + deltaX *= lineHeight; + } else if ( orgEvent.deltaMode === 2 ) { + var pageHeight = $.data(this, 'mousewheel-page-height'); + delta *= pageHeight; + deltaY *= pageHeight; + deltaX *= pageHeight; + } + + // Store lowest absolute delta to normalize the delta values + absDelta = Math.max( Math.abs(deltaY), Math.abs(deltaX) ); + + if ( !lowestDelta || absDelta < lowestDelta ) { + lowestDelta = absDelta; + + // Adjust older deltas if necessary + if ( shouldAdjustOldDeltas(orgEvent, absDelta) ) { + lowestDelta /= 40; + } + } + + // Adjust older deltas if necessary + if ( shouldAdjustOldDeltas(orgEvent, absDelta) ) { + // Divide all the things by 40! + delta /= 40; + deltaX /= 40; + deltaY /= 40; + } + + // Get a whole, normalized value for the deltas + delta = Math[ delta >= 1 ? 'floor' : 'ceil' ](delta / lowestDelta); + deltaX = Math[ deltaX >= 1 ? 'floor' : 'ceil' ](deltaX / lowestDelta); + deltaY = Math[ deltaY >= 1 ? 'floor' : 'ceil' ](deltaY / lowestDelta); + + // Normalise offsetX and offsetY properties + if ( special.settings.normalizeOffset && this.getBoundingClientRect ) { + var boundingRect = this.getBoundingClientRect(); + offsetX = event.clientX - boundingRect.left; + offsetY = event.clientY - boundingRect.top; + } + + // Add information to the event object + event.deltaX = deltaX; + event.deltaY = deltaY; + event.deltaFactor = lowestDelta; + event.offsetX = offsetX; + event.offsetY = offsetY; + // Go ahead and set deltaMode to 0 since we converted to pixels + // Although this is a little odd since we overwrite the deltaX/Y + // properties with normalized deltas. + event.deltaMode = 0; + + // Add event and delta to the front of the arguments + args.unshift(event, delta, deltaX, deltaY); + + // Clearout lowestDelta after sometime to better + // handle multiple device types that give different + // a different lowestDelta + // Ex: trackpad = 3 and mouse wheel = 120 + if (nullLowestDeltaTimeout) { clearTimeout(nullLowestDeltaTimeout); } + nullLowestDeltaTimeout = setTimeout(nullLowestDelta, 200); + + return ($.event.dispatch || $.event.handle).apply(this, args); + } + + function nullLowestDelta() { + lowestDelta = null; + } + + function shouldAdjustOldDeltas(orgEvent, absDelta) { + // If this is an older event and the delta is divisable by 120, + // then we are assuming that the browser is treating this as an + // older mouse wheel event and that we should divide the deltas + // by 40 to try and get a more usable deltaFactor. + // Side note, this actually impacts the reported scroll distance + // in older browsers and can cause scrolling to be slower than native. + // Turn this off by setting $.event.special.mousewheel.settings.adjustOldDeltas to false. + return special.settings.adjustOldDeltas && orgEvent.type === 'mousewheel' && absDelta % 120 === 0; + } + +})); diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/static/theme/dark/common.css b/datagear-web/src/main/resources/org/datagear/web/webapp/static/theme/dark/common.css index 55ce933b..3d6baca7 100644 --- a/datagear-web/src/main/resources/org/datagear/web/webapp/static/theme/dark/common.css +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/static/theme/dark/common.css @@ -246,4 +246,15 @@ table.dataTable.display tbody > tr > .selected:hover { background-color: #f99117; box-shadow: none; -webkit-box-shadow: none; +} +/**年份选择器确定按钮*/ +.xdsoft_datetimepicker .xdsoft_save_selected.xdsoft_save_selected_year{ + color: #eeeeee !important; + border: 1px solid #666666 !important; + background: #555555 !important; +} +.xdsoft_datetimepicker .xdsoft_save_selected.xdsoft_save_selected_year:hover{ + color: #ffffff !important; + border: 1px solid #59b4d4 !important; + background: #0078a3 !important; } \ No newline at end of file diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/static/theme/green/common.css b/datagear-web/src/main/resources/org/datagear/web/webapp/static/theme/green/common.css index 957ab1ca..c8fd52fb 100644 --- a/datagear-web/src/main/resources/org/datagear/web/webapp/static/theme/green/common.css +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/static/theme/green/common.css @@ -246,4 +246,15 @@ table.dataTable.display tbody > tr > .selected:hover { background: #59c908; box-shadow: none; -webkit-box-shadow: none; +} +/**年份选择器确定按钮*/ +.xdsoft_datetimepicker .xdsoft_save_selected.xdsoft_save_selected_year{ + color: #ffffff !important; + border: 1px solid #45930b !important; + background: #3c8009 !important; +} +.xdsoft_datetimepicker .xdsoft_save_selected.xdsoft_save_selected_year:hover{ + color: #ffffff !important; + border: 1px solid #6cc510 !important; + background: #459e05 !important; } \ No newline at end of file diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/static/theme/light/common.css b/datagear-web/src/main/resources/org/datagear/web/webapp/static/theme/light/common.css index fc2436b9..8d402fcb 100644 --- a/datagear-web/src/main/resources/org/datagear/web/webapp/static/theme/light/common.css +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/static/theme/light/common.css @@ -246,4 +246,15 @@ table.dataTable.display tbody > tr > .selected:hover { background-color: #007FFF; box-shadow: none; -webkit-box-shadow: none; +} +/**年份选择器确定按钮*/ +.xdsoft_datetimepicker .xdsoft_save_selected.xdsoft_save_selected_year{ + color: #454545 !important; + border: 1px solid #c5c5c5 !important; + background: #f6f6f6 !important; +} +.xdsoft_datetimepicker .xdsoft_save_selected.xdsoft_save_selected_year:hover{ + color: #2b2b2b !important; + border: 1px solid #cccccc !important; + background: #ededed !important; } \ No newline at end of file diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/chart/chart_form.ftl b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/chart/chart_form.ftl index 35e8be15..ad5c4161 100644 --- a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/chart/chart_form.ftl +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/chart/chart_form.ftl @@ -41,7 +41,9 @@ readonly 是否只读操作,允许为null
- +
@@ -64,7 +66,9 @@ readonly 是否只读操作,允许为null
- +
diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dashboard/dashboard_form.ftl b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dashboard/dashboard_form.ftl index 2c39116f..1af856df 100644 --- a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dashboard/dashboard_form.ftl +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dashboard/dashboard_form.ftl @@ -132,7 +132,7 @@ readonly 是否只读操作,允许为null { return "${contextPath}/analysis/dashboard/" + action; }; - + po.getLastTagText = function(text) { if(!text) diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_CsvFile.ftl b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_CsvFile.ftl new file mode 100644 index 00000000..b8992a23 --- /dev/null +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_CsvFile.ftl @@ -0,0 +1,238 @@ +<#include "../../include/import_global.ftl"> +<#include "../../include/html_doctype.ftl"> +<#-- +titleMessageKey 标题标签I18N关键字,不允许null +formAction 表单提交action,允许为null +readonly 是否只读操作,允许为null +--> +<#assign formAction=(formAction!'#')> +<#assign readonly=(readonly!false)> +<#assign isAdd=(formAction == 'saveAdd')> + + +<#include "../../include/html_head.ftl"> +<#include "../../include/html_title_app_name.ftl"> + <@spring.message code='${titleMessageKey}' /> - <@spring.message code='dataSet.dataSetType.CsvFile' /> + + + +
+
+
+
+ <#include "include/dataSet_form_html_name.ftl"> +
+
+
+ +
+
+ + + + + <#if formAction != 'saveAddForCsvFile'> + <@spring.message code='download' /> + + + <#if !readonly> +
+
<@spring.message code='upload' />
+
+
+ +
+
+
+
+ +
+
+ + + + + + + +   + +
+
+
+
+ +
+
+ +
+
+ <#include "include/dataSet_form_html_wow.ftl" > +
+
+
+ <#if !readonly> + +    + + <#else> +
 
+ +
+
+ <#include "include/dataSet_form_html_preview_pvp.ftl" > +
+<#include "../../include/page_js_obj.ftl" > +<#include "../../include/page_obj_form.ftl"> +<#include "include/dataSet_form_js.ftl"> +<#include "include/dataSet_form_js_nameRow.ftl"> + + + \ No newline at end of file diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_CsvValue.ftl b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_CsvValue.ftl new file mode 100644 index 00000000..c6f0625d --- /dev/null +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_CsvValue.ftl @@ -0,0 +1,206 @@ +<#include "../../include/import_global.ftl"> +<#include "../../include/html_doctype.ftl"> +<#-- +titleMessageKey 标题标签I18N关键字,不允许null +formAction 表单提交action,允许为null +readonly 是否只读操作,允许为null +--> +<#assign formAction=(formAction!'#')> +<#assign readonly=(readonly!false)> +<#assign isAdd=(formAction == 'saveAdd')> + + +<#include "../../include/html_head.ftl"> +<#include "../../include/html_title_app_name.ftl"> + <@spring.message code='${titleMessageKey}' /> - <@spring.message code='dataSet.dataSetType.CsvValue' /> + + + +
+
+
+
+ <#include "include/dataSet_form_html_name.ftl"> +
+
+
+ +
+
+ +
+
+
+
+
+ <#include "include/dataSet_form_html_wow.ftl" > +
+
+
+ +
+
+ + + + + + + +   + +
+
+
+
+ <#if !readonly> + +    + + <#else> +
 
+ +
+
+ <#include "include/dataSet_form_html_preview_pvp.ftl" > +
+<#include "../../include/page_js_obj.ftl" > +<#include "../../include/page_obj_form.ftl"> +<#include "include/dataSet_form_js.ftl"> +<#include "include/dataSet_form_js_nameRow.ftl"> + + + \ No newline at end of file diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_Excel.ftl b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_Excel.ftl new file mode 100644 index 00000000..fddc1ca3 --- /dev/null +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_Excel.ftl @@ -0,0 +1,308 @@ +<#include "../../include/import_global.ftl"> +<#include "../../include/html_doctype.ftl"> +<#-- +titleMessageKey 标题标签I18N关键字,不允许null +formAction 表单提交action,允许为null +readonly 是否只读操作,允许为null +--> +<#assign formAction=(formAction!'#')> +<#assign readonly=(readonly!false)> +<#assign isAdd=(formAction == 'saveAdd')> + + +<#include "../../include/html_head.ftl"> +<#include "../../include/html_title_app_name.ftl"> + <@spring.message code='${titleMessageKey}' /> - <@spring.message code='dataSet.dataSetType.Excel' /> + + + +
+
+
+
+ <#include "include/dataSet_form_html_name.ftl"> +
+
+
+ +
+
+ + + + + <#if formAction != 'saveAddForExcel'> + <@spring.message code='download' /> + + + <#if !readonly> +
+
<@spring.message code='upload' />
+
+
+ +
+
+
+
+ +
+
+ +
+
+
+
+ +
+
+ + + + + + + +   + +
+
+
+
+ +
+
+ +
+
+
+
+ +
+
+ +
+
+
+
+ +
+
+
+ + + + +
+
+
+ <#include "include/dataSet_form_html_wow.ftl" > +
+
+
+ <#if !readonly> + +    + + <#else> +
 
+ +
+
+ <#include "include/dataSet_form_html_preview_pvp.ftl" > +
+<#include "../../include/page_js_obj.ftl" > +<#include "../../include/page_obj_form.ftl"> +<#include "include/dataSet_form_js.ftl"> +<#include "include/dataSet_form_js_nameRow.ftl"> + + + \ No newline at end of file diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_Http.ftl b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_Http.ftl new file mode 100644 index 00000000..df5b00c9 --- /dev/null +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_Http.ftl @@ -0,0 +1,311 @@ +<#include "../../include/import_global.ftl"> +<#include "../../include/html_doctype.ftl"> +<#-- +titleMessageKey 标题标签I18N关键字,不允许null +formAction 表单提交action,允许为null +readonly 是否只读操作,允许为null +--> +<#assign formAction=(formAction!'#')> +<#assign readonly=(readonly!false)> +<#assign isAdd=(formAction == 'saveAdd')> +<#assign HttpDataSet=statics['org.datagear.analysis.support.HttpDataSet']> + + +<#include "../../include/html_head.ftl"> +<#include "../../include/html_title_app_name.ftl"> + <@spring.message code='${titleMessageKey}' /> - <@spring.message code='dataSet.dataSetType.Http' /> + + + +
+
+
+
+ <#include "include/dataSet_form_html_name.ftl"> +
+
+ +
+
+ + +
+ +
+ +
+
+
+
+
+
+ +
+
+ + +
+
+ +
+
+ +
+
+ +
+
+ +
+
+ +
+
+
+
+
+
+
+ +
+
+ + + +
+
+
+
+ +
+
+ +
+
+ <#include "include/dataSet_form_html_wow.ftl" > +
+
+
+ <#if !readonly> + +    + + <#else> +
 
+ +
+
+ <#include "include/dataSet_form_html_preview_pvp.ftl" > +
+<#include "../../include/page_js_obj.ftl" > +<#include "../../include/page_obj_form.ftl"> +<#include "include/dataSet_form_js.ftl"> + + + \ No newline at end of file diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_JsonFile.ftl b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_JsonFile.ftl index aa3ac2e1..fb645a36 100644 --- a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_JsonFile.ftl +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_JsonFile.ftl @@ -21,28 +21,20 @@ readonly 是否只读操作,允许为null
<#include "include/dataSet_form_html_name.ftl"> -
-
- -
-
- -
-
-
-
- -
-
- - -
+
+
+
+ +
+
+ + - + + <#if formAction != 'saveAddForJsonFile'> + <@spring.message code='download' /> + + <#if !readonly>
<@spring.message code='upload' />
@@ -50,8 +42,30 @@ readonly 是否只读操作,允许为null
- <#include "include/dataSet_form_html_wow.ftl" >
+
+
+ +
+
+ +
+
+
+
+ +
+
+ +
+
+ <#include "include/dataSet_form_html_wow.ftl" >
@@ -78,66 +92,57 @@ readonly 是否只读操作,允许为null $.initButtons(po.element()); po.element("select[name='encoding']").selectmenu({ appendTo : po.element(), classes : { "ui-selectmenu-menu" : "encoding-selectmenu-menu" } }); po.initWorkspaceHeight(); + po.initWorkspaceTabs(true); + po.initDataSetPropertiesTable(po.dataSetProperties); + po.initDataSetParamsTable(po.dataSetParams); + po.initPreviewParamValuePanel(); - po.fileNameEditorValue = function(value) + po.updatePreviewOptionsData = function() { - var $editor = po.element("#${pageId}-workspaceEditor"); + var dataSet = po.previewOptions.data.dataSet; - if(value === undefined) - return $editor.val(); - else - $editor.val(value); + dataSet.fileName = po.element("input[name='fileName']").val(); + dataSet.dataJsonPath = po.element("input[name='dataJsonPath']").val(); + dataSet.encoding = po.element("select[name='encoding']").val(); + + po.previewOptions.data.originalFileName = po.element("#${pageId}-originalFileName").val(); }; + <#if formAction != 'saveAddForJsonFile'> + //编辑、查看操作应初始化为已完成预览的状态 + po.updatePreviewOptionsData(); + po.previewSuccess(true); + + po.isPreviewValueModified = function() { - return po.isFileNameModified(); - }; - - po.isFileNameModified = function(inputValue, editorValue) - { - if(inputValue == undefined) - inputValue = po.element("input[name='fileName']").val(); - if(editorValue == undefined) - editorValue = po.fileNameEditorValue(); + var fileName = po.element("input[name='fileName']").val(); + var dataJsonPath = po.element("input[name='dataJsonPath']").val(); + var encoding = po.element("select[name='encoding']").val(); - return po.isModifiedIgnoreBlank(inputValue, editorValue); + var pd = po.previewOptions.data.dataSet; + + return (pd.fileName != fileName) || (pd.dataJsonPath != dataJsonPath) || (pd.encoding != encoding); }; - po.initWorkspaceTabs(); - - po.initDataSetPropertiesTable(po.dataSetProperties); - - po.initDataSetParamsTable(po.dataSetParams); - - po.initPreviewParamValuePanel(); - po.previewOptions.url = po.url("previewJsonFile"); po.previewOptions.beforePreview = function() { - var fileName = po.fileNameEditorValue(); + po.updatePreviewOptionsData(); - if(!fileName) + if(!this.data.dataSet.fileName) return false; - - this.data.dataSet.encoding = po.element("select[name='encoding']").val(); - this.data.dataSet.fileName = fileName; - this.data.originalFileName = po.element("#${pageId}-originalFileName").val(); }; po.previewOptions.beforeRefresh = function() { - if(!this.data.dataSet || !this.data.dataSet.fileName) + if(!this.data.dataSet.fileName) return false; }; - po.previewOptions.success = function(previewResponse) - { - po.element("input[name='fileName']").val(this.data.dataSet.fileName); - }; po.initPreviewOperations(); po.fileUploadInfo = function(){ return this.element(".upload-file-info"); }; - + po.element(".fileinput-button").fileupload( { url : po.url("uploadFile"), @@ -145,8 +150,8 @@ readonly 是否只读操作,允许为null success : function(uploadResult, textStatus, jqXHR) { $.fileuploadsuccessHandlerForUploadInfo(po.fileUploadInfo(), false); + po.element("input[name='fileName']").val(uploadResult.fileName); po.element("input[name='displayName']").val(uploadResult.displayName); - po.element("#${pageId}-workspaceEditor").val(uploadResult.fileName); } }) .bind('fileuploadadd', function (e, data) @@ -161,7 +166,7 @@ readonly 是否只读操作,允许为null $.validator.addMethod("dataSetJsonFilePreviewRequired", function(value, element) { - return !po.isFileNameModified(); + return !po.isPreviewValueModified() && po.previewSuccess(); }); po.form().validate( diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_JsonValue.ftl b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_JsonValue.ftl index afe78bd9..2f68140d 100644 --- a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_JsonValue.ftl +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_JsonValue.ftl @@ -21,17 +21,21 @@ readonly 是否只读操作,允许为null
<#include "include/dataSet_form_html_name.ftl"> -
-
- -
-
- -
-
+
+
+
+ +
+
+ +
+
+
- <#include "include/dataSet_form_html_wow.ftl" >
+ <#include "include/dataSet_form_html_wow.ftl" >
@@ -58,21 +62,6 @@ readonly 是否只读操作,允许为null $.initButtons(po.element()); po.initWorkspaceHeight(); - po.isPreviewValueModified = function() - { - return po.isValueModified(); - }; - - po.isValueModified = function(textareaValue, editorValue) - { - if(textareaValue == undefined) - textareaValue = po.element("textarea[name='value']").val(); - if(editorValue == undefined) - editorValue = po.jsonEditor.getValue(); - - return po.isModifiedIgnoreBlank(textareaValue, editorValue); - }; - var languageTools = ace.require("ace/ext/language_tools"); var JsonMode = ace.require("ace/mode/json").Mode; po.jsonEditor = ace.edit("${pageId}-workspaceEditor"); @@ -80,40 +69,53 @@ readonly 是否只读操作,允许为null po.jsonEditor.setShowPrintMargin(false); po.initWorkspaceEditor(po.jsonEditor, po.element("textarea[name='value']").val()); - po.initWorkspaceTabs(); - po.getAddPropertyName = function() { var selectionRange = po.jsonEditor.getSelectionRange(); return (po.jsonEditor.session.getTextRange(selectionRange) || ""); }; po.initDataSetPropertiesTable(po.dataSetProperties); - po.initDataSetParamsTable(po.dataSetParams); - po.initPreviewParamValuePanel(); + + po.updatePreviewOptionsData = function() + { + var value = po.jsonEditor.getValue(); + + var dataSet = po.previewOptions.data.dataSet; + + dataSet.value = value; + }; + + <#if formAction != 'saveAddForJsonValue'> + //编辑、查看操作应初始化为已完成预览的状态 + po.updatePreviewOptionsData(); + po.previewSuccess(true); + + + po.isPreviewValueModified = function() + { + var value = po.jsonEditor.getValue(); + + var pd = po.previewOptions.data.dataSet; + + return (pd.value != value); + }; po.previewOptions.url = po.url("previewJsonValue"); po.previewOptions.beforePreview = function() { - var value = po.jsonEditor.getValue(); + po.updatePreviewOptionsData(); - if(!value) + if(!this.data.dataSet.value) return false; - - this.data.dataSet.value = value; }; po.previewOptions.beforeRefresh = function() { - if(!this.data.dataSet || !this.data.dataSet.value) + if(!this.data.dataSet.value) return false; }; - po.previewOptions.success = function(previewResponse) - { - po.element("textarea[name='value']").val(this.data.dataSet.value); - po.jsonEditor.focus(); - }; po.initPreviewOperations(); @@ -125,7 +127,7 @@ readonly 是否只读操作,允许为null $.validator.addMethod("dataSetJsonValuePreviewRequired", function(value, element) { - return !po.isValueModified(value); + return !po.isPreviewValueModified() && po.previewSuccess(); }); po.form().validate( @@ -151,6 +153,7 @@ readonly 是否只读操作,允许为null var formData = $.formToJson(form); formData["properties"] = po.getFormDataSetProperties(); formData["params"] = po.getFormDataSetParams(); + formData["value"] = po.jsonEditor.getValue(); $.postJson("${contextPath}/analysis/dataSet/${formAction}", formData, function(response) diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_SQL.ftl b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_SQL.ftl index b0effc4f..e27ce0bd 100644 --- a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_SQL.ftl +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_form_SQL.ftl @@ -33,17 +33,21 @@ readonly 是否只读操作,允许为null
-
-
- -
-
- -
-
+
+
+
+ +
+
+ +
+
+
- <#include "include/dataSet_form_html_wow.ftl" >
+ <#include "include/dataSet_form_html_wow.ftl" >
@@ -78,21 +82,6 @@ readonly 是否只读操作,允许为null po.getDataSetSchemaId = function(){ return po.element("input[name='schemaConnectionFactory.schema.id']").val(); }; - po.isPreviewValueModified = function() - { - return po.isSqlModified(); - }; - - po.isSqlModified = function(textareaValue, editorValue) - { - if(textareaValue == undefined) - textareaValue = po.element("textarea[name='sql']").val(); - if(editorValue == undefined) - editorValue = po.sqlEditor.getValue(); - - return po.isModifiedIgnoreBlank(textareaValue, editorValue); - }; - po.element(".select-schema-button").click(function() { var options = @@ -117,42 +106,56 @@ readonly 是否只读操作,允许为null po.initSqlEditor(); po.initWorkspaceEditor(po.sqlEditor, po.element("textarea[name='sql']").val()); - po.initWorkspaceTabs(); - po.getAddPropertyName = function() { var selectionRange = po.sqlEditor.getSelectionRange(); return (po.sqlEditor.session.getTextRange(selectionRange) || ""); }; po.initDataSetPropertiesTable(po.dataSetProperties); - po.initDataSetParamsTable(po.dataSetParams); - po.initPreviewParamValuePanel(); - - po.previewOptions.url = po.url("previewSql"); - po.previewOptions.beforePreview = function() + + po.updatePreviewOptionsData = function() { var schemaId = po.getDataSetSchemaId(); var sql = po.sqlEditor.getValue(); - if(!schemaId || !sql) - return false; + var dataSet = po.previewOptions.data.dataSet; - this.data.dataSet.sql = sql; - this.data.schemaId = schemaId; + dataSet.sql = sql; + po.previewOptions.data.schemaId = schemaId; + }; + + <#if formAction != 'saveAddForSql'> + //编辑、查看操作应初始化为已完成预览的状态 + po.updatePreviewOptionsData(); + po.previewSuccess(true); + + + po.isPreviewValueModified = function() + { + var schemaId = po.getDataSetSchemaId(); + var sql = po.sqlEditor.getValue(); + + var pd = po.previewOptions.data.dataSet; + + return (pd.sql != sql) || (po.previewOptions.data.schemaId != schemaId); + }; + + po.previewOptions.url = po.url("previewSql"); + po.previewOptions.beforePreview = function() + { + po.updatePreviewOptionsData(); + + if(!this.data.dataSet.sql || !this.data.schemaId) + return false; }; po.previewOptions.beforeRefresh = function() { - if(!this.data.dataSet || !this.data.dataSet.sql || !this.data.schemaId) + if(!this.data.dataSet.sql || !this.data.schemaId) return false; }; - po.previewOptions.success = function(previewResponse) - { - po.element("textarea[name='sql']").val(this.data.dataSet.sql); - po.sqlEditor.focus(); - }; po.initPreviewOperations(); @@ -207,7 +210,7 @@ readonly 是否只读操作,允许为null $.validator.addMethod("dataSetSqlPreviewRequired", function(value, element) { - return !po.isSqlModified(value); + return !po.isPreviewValueModified() && po.previewSuccess(); }); po.form().validate( @@ -235,6 +238,7 @@ readonly 是否只读操作,允许为null var formData = $.formToJson(form); formData["properties"] = po.getFormDataSetProperties(); formData["params"] = po.getFormDataSetParams(); + formData["sql"] = po.sqlEditor.getValue(); $.postJson("${contextPath}/analysis/dataSet/${formAction}", formData, function(response) diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_grid.ftl b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_grid.ftl index 42848dcf..10a1f9bf 100644 --- a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_grid.ftl +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/dataSet_grid.ftl @@ -39,6 +39,10 @@ boolean readonly 是否只读操作,默认为false
  • <@spring.message code='dataSet.dataSetType.SQL' />
  • +
  • <@spring.message code='dataSet.dataSetType.CsvValue' />
  • +
  • <@spring.message code='dataSet.dataSetType.CsvFile' />
  • +
  • <@spring.message code='dataSet.dataSetType.Excel' />
  • +
  • <@spring.message code='dataSet.dataSetType.Http' />
  • <@spring.message code='dataSet.dataSetType.JsonValue' />
  • <@spring.message code='dataSet.dataSetType.JsonFile' />
@@ -189,10 +193,18 @@ boolean readonly 是否只读操作,默认为false { if("${DataSetEntity.DATA_SET_TYPE_SQL}" == data) return "<@spring.message code='dataSet.dataSetType.SQL' />"; + else if("${DataSetEntity.DATA_SET_TYPE_Excel}" == data) + return "<@spring.message code='dataSet.dataSetType.Excel' />"; + else if("${DataSetEntity.DATA_SET_TYPE_CsvValue}" == data) + return "<@spring.message code='dataSet.dataSetType.CsvValue' />"; + else if("${DataSetEntity.DATA_SET_TYPE_CsvFile}" == data) + return "<@spring.message code='dataSet.dataSetType.CsvFile' />"; else if("${DataSetEntity.DATA_SET_TYPE_JsonValue}" == data) return "<@spring.message code='dataSet.dataSetType.JsonValue' />"; else if("${DataSetEntity.DATA_SET_TYPE_JsonFile}" == data) return "<@spring.message code='dataSet.dataSetType.JsonFile' />"; + else if("${DataSetEntity.DATA_SET_TYPE_Http}" == data) + return "<@spring.message code='dataSet.dataSetType.Http' />"; else return ""; }; diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/include/dataSet_form_html_wow.ftl b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/include/dataSet_form_html_wow.ftl index 7359990e..6f4886a8 100644 --- a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/include/dataSet_form_html_wow.ftl +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/include/dataSet_form_html_wow.ftl @@ -1,7 +1,7 @@ <#-- 数据集表单页:预览、参数、属性操作区 --> -
+
-
+
+ +
+
+ +
+
+
diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/include/dataSet_form_js.ftl b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/include/dataSet_form_js.ftl index 953a7824..a2d4d4df 100644 --- a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/include/dataSet_form_js.ftl +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/include/dataSet_form_js.ftl @@ -28,6 +28,9 @@ po.previewOptions.url = "..."; po.isModifiedIgnoreBlank = function(sourceVal, targetVal) { + sourceVal = (sourceVal || ""); + targetVal = (targetVal || ""); + sourceVal = sourceVal.replace(/\s/g, ''); targetVal = targetVal.replace(/\s/g, ''); @@ -51,12 +54,16 @@ po.previewOptions.url = "..."; po.calWorkspaceOperationTableHeight = function() { - return po.element(".preview-result-table-wrapper").height() - 30; + var tableTitleHeight = 30; + return po.element(".preview-result-table-wrapper").height() - tableTitleHeight; }; - po.initWorkspaceHeight = function() + po.initWorkspaceHeight = function(calFormContentHeight) { + calFormContentHeight = (calFormContentHeight == true ? true : false); + var height = $(window).height(); + //减去上下留白 height = height - height/10; //减去对话框标题高度 @@ -64,19 +71,25 @@ po.previewOptions.url = "..."; height = height - 41; //减去其他表单元素高度 height = height - po.element(".form-head").outerHeight(true); - po.element(".form-content > .form-item:not(.form-item-workspace)").each(function() + height = height - po.element(".form-foot").outerHeight(true); + + var formContentHeight = height - 41; + + po.element(".form-content > .form-item").each(function() { height = height - $(this).outerHeight(true); }); - height = height - po.element(".form-foot").outerHeight(true); //减去杂项高度 height = height - 41 - 10; - var errorInfoHeight = 25; + var errorInfoHeight = 41 + 10; - po.element(".form-item-workspace .form-item-value").height(height); + po.element(".workspace").css("min-height", height+"px"); po.element(".workspace-editor-wrapper").height(height - errorInfoHeight); po.element(".workspace-operation-wrapper").height(height - errorInfoHeight); + + if(calFormContentHeight) + po.element(".form-content").css("max-height", formContentHeight+"px"); }; po.initWorkspaceEditor = function(editor, initValue) @@ -103,8 +116,10 @@ po.previewOptions.url = "..."; }; - po.initWorkspaceTabs = function() + po.initWorkspaceTabs = function(disableParams) { + disableParams = (disableParams == true ? true : false); + po.element(".workspace-operation-wrapper").tabs( { activate: function(event, ui) @@ -129,8 +144,14 @@ po.previewOptions.url = "..."; } } }); + + if(disableParams) + { + var paramsIndex = $(".workspace-operation-nav .operation-params", po.element(".workspace-operation-wrapper")).index(); + po.element(".workspace-operation-wrapper").tabs("disable", paramsIndex); + } }; - + //获取用于添加数据集属性的名 po.getAddPropertyName = function() { @@ -572,6 +593,15 @@ po.previewOptions.url = "..."; success: function(previewResponse){} }; + //获取、设置上一次预览是否成功 + po.previewSuccess = function(success) + { + if(success === undefined) + return po._previewSuccess == true; + else + po._previewSuccess = success; + }; + po.destroyPreviewResultTable = function() { var table = po.previewResultTableElement(); @@ -581,11 +611,13 @@ po.previewOptions.url = "..."; table.empty(); } }; - + po.initPreviewOperations = function() { po.element(".preview-result-table-wrapper .preview-button").click(function(event) { + var previewValueModified = po.isPreviewValueModified(); + if(po.previewOptions.beforePreview() == false) return; @@ -606,7 +638,7 @@ po.previewOptions.url = "..."; po.previewOptions.data.dataSet.params = po.getFormDataSetParams(); po.previewOptions.data.paramValues = chartFactory.chartForm.getDataSetParamValueObj(this); - po.executePreview(); + po.executePreview(previewValueModified); } }); } @@ -617,7 +649,7 @@ po.previewOptions.url = "..."; po.previewOptions.data.dataSet.params = []; po.previewOptions.data.paramValues = {}; - po.executePreview(); + po.executePreview(previewValueModified); } }); @@ -626,11 +658,36 @@ po.previewOptions.url = "..."; if(po.previewOptions.beforeRefresh() == false) return; - po.executePreview(); + po.executePreview(false); + }); + + po.element(".show-resolved-source-button").click(function() + { + var $panel = po.element(".result-resolved-source-panel"); + + if($panel.is(":hidden")) + { + $panel.show(); + $panel.position({ my: "right bottom", at: "right top-5", of: this }); + } + else + $panel.hide(); + }); + + $(po.element()).on("click", function(event) + { + var $target = $(event.target); + + var $panel = po.element(".result-resolved-source-panel"); + if(!$panel.is(":hidden")) + { + if($target.closest(".result-resolved-source").length == 0) + $panel.hide(); + } }); }; - po.executePreview = function() + po.executePreview = function(previewValueModified) { if(po.previewOptions.beforeRequest() == false) return; @@ -650,9 +707,11 @@ po.previewOptions.url = "..."; data : po.previewOptions.data, success : function(previewResponse) { + po.previewSuccess(true); + //如果工作区内容已变更才更新属性,防止上次保存后的属性被刷新 //属性表单内容为空也更新,比如用户删除了所有属性时 - if(po.isPreviewValueModified() || !po.hasFormDataSetProperty()) + if(previewValueModified || !po.hasFormDataSetProperty()) po.updateFormDataSetProperties(previewResponse.properties); var tableData = (previewResponse.result.data || []); @@ -712,6 +771,10 @@ po.previewOptions.url = "..."; po.previewOptions.success(previewResponse); }, + error: function() + { + po.previewSuccess(false); + }, complete: function() { $buttons.each(function() @@ -755,7 +818,7 @@ po.previewOptions.url = "..."; var name = dataSetProperties[colIndex].name; if(setValue === undefined) - return row[name]; + return chartFactory.escapeHtml(row[name]); else row[name] = setValue; }, diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/include/dataSet_form_js_nameRow.ftl b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/include/dataSet_form_js_nameRow.ftl new file mode 100644 index 00000000..439167c1 --- /dev/null +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/analysis/dataSet/include/dataSet_form_js_nameRow.ftl @@ -0,0 +1,70 @@ +<#-- +数据集表单页:标题行操作JS片段 + +依赖: +page_js_obj.ftl + +变量: +--> + \ No newline at end of file diff --git a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/include/html_head.ftl b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/include/html_head.ftl index 60d05e27..278f4fd9 100644 --- a/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/include/html_head.ftl +++ b/datagear-web/src/main/resources/org/datagear/web/webapp/view/freemarker/include/html_head.ftl @@ -1,6 +1,6 @@ - + <#if !isAjaxRequest> <#assign _hh_Version=statics['org.datagear.util.Global'].VERSION>