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

com.orangesignal.csv.bean.CsvEntityTemplate Maven / Gradle / Ivy

Go to download

OrangeSignal CSV is a very flexible csv (comma-separated values) read and write library for Java.

There is a newer version: 2.2.1
Show newest version
/*
 * Copyright 2013 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.orangesignal.csv.bean;

import java.lang.reflect.Field;
import java.text.DecimalFormat;
import java.text.DecimalFormatSymbols;
import java.text.Format;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Currency;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.SortedMap;
import java.util.TimeZone;
import java.util.TreeMap;

import com.orangesignal.csv.annotation.CsvColumn;
import com.orangesignal.csv.annotation.CsvColumns;
import com.orangesignal.csv.annotation.CsvEntity;
import com.orangesignal.csv.filters.CsvNamedValueFilter;

/**
 * 区切り文字形式データ注釈要素 {@link CsvEntity} で注釈付けされた Java プログラム要素の操作を簡素化するヘルパークラスを提供します。
 * 
 * @author Koji Sugisawa
 * @since 1.4.0
 * @see CsvEntity
 * @see CsvColumn
 * @see CsvColumns
 */
public class CsvEntityTemplate extends AbstractCsvBeanTemplate> implements CsvEntityOperation> {

	/**
	 * 区切り文字形式データフィルタを保持します。
	 */
	private CsvNamedValueFilter filter;

	// ------------------------------------------------------------------------
	// 利便性のための静的メソッド

	/**
	 * 新しい {@link CsvEntityTemplate} のインスタンスを返します。
	 * 
	 * @param entityClass 区切り文字形式データ注釈要素 {@link CsvEntity} で注釈付けされた Java プログラム要素の型
	 * @return 新しい {@link CsvEntityTemplate} のインスタンス
	 * @throws IllegalArgumentException {@code entityClass} が {@code null} または不正な場合
	 */
	public static  CsvEntityTemplate newInstance(final Class entityClass) {
		return new CsvEntityTemplate(entityClass);
	}

	// -----------------------------------------------------------------------
	// コンストラクタ

	/**
	 * コンストラクタです。
	 * 
	 * @param entityClass 区切り文字形式データ注釈要素 {@link com.orangesignal.csv.annotation.CsvEntity} で注釈付けされた Java プログラム要素の型
	 * @throws IllegalArgumentException {@code entityClass} が {@code null} または不正な場合
	 */
	public CsvEntityTemplate(final Class entityClass) {
		super(entityClass);
		if (entityClass.getAnnotation(CsvEntity.class) == null) {
			throw new IllegalArgumentException(String.format("No CsvEntity is available %s", entityClass.getName()));
		}
	}

	// -----------------------------------------------------------------------
	// オーバーライド メソッド

	@Override
	public CsvEntityTemplate filter(final CsvNamedValueFilter filter) {
		this.filter = filter;
		return this;
	}

	// ------------------------------------------------------------------------
	// パブリック メソッド

	/**
	 * 指定された区切り文字形式データの値リストが含まれる必要があるかどうかを判定します。
	 * 
	 * @param columnNames 区切り文字形式データの項目名リスト
	 * @param values 区切り文字形式データの項目値のリスト
	 * @return {@code values} が含まれる必要がある場合は {@code true}
	 */
	public boolean isAccept(final List columnNames, final List values) {
		return (filter != null && !filter.accept(columnNames, values));
	}

	/**
	 * 項目名のリストを作成して返します。
	 * 
	 * @return 項目名のリスト
	 */
	public List createColumnNames() {
		final SortedMap positionMap = new TreeMap();
		final List adding = new ArrayList();

		for (final Field f : getType().getDeclaredFields()) {
			final CsvColumns columns = f.getAnnotation(CsvColumns.class);
			if (columns != null) {
				for (final CsvColumn column : columns.value()) {
					final int pos = column.position();
					final String name = defaultIfEmpty(column.name(), f.getName());
					if (pos >= 0) {
						if (positionMap.containsKey(pos)) {
							continue;
						}
						positionMap.put(pos, name);
					} else {
						adding.add(name);
					}
				}
			}
			final CsvColumn column = f.getAnnotation(CsvColumn.class);
			if (column != null) {
				final int pos = column.position();
				final String name = defaultIfEmpty(column.name(), f.getName());
				if (pos >= 0) {
					if (positionMap.containsKey(pos)) {
						continue;
					}
					positionMap.put(pos, name);
				} else {
					adding.add(name);
				}
			}
		}

		final int max = positionMap.size() > 0 ? positionMap.lastKey().intValue() + 1 : 0;
		final String[] names = new String[max];
		for (final Map.Entry entry : positionMap.entrySet()) {
			names[entry.getKey().intValue()] = entry.getValue();
		}

		final List results = new ArrayList(Arrays.asList(names));
		if (adding.size() > 0) {
			results.addAll(adding);
		}
		return results;
	}

	public void prepare(final List names, final Field[] fields) {
		super.valueParserMapping(new HashMap(0));
		super.valueFormatterMapping(new HashMap(0));

		// 書式オブジェクトの準備を行います。
		for (final Field f : fields) {
			final CsvColumns columns = f.getAnnotation(CsvColumns.class);
			if (columns != null) {
				for (final CsvColumn column : columns.value()) {
					final Format format = createFormat(column, f);
					if (format != null) {
						setValueParser(f.getName(), format);
						setValueFormatter(getPosition(column, f, names), format);
					}
				}
			}
			final CsvColumn column = f.getAnnotation(CsvColumn.class);
			if (column != null) {
				final Format format = createFormat(column, f);
				if (format != null) {
					setValueParser(f.getName(), format);
					setValueFormatter(getPosition(column, f, names), format);
				}
			}
		}
	}

	// ------------------------------------------------------------------------

	public static int getPosition(final CsvColumn column, final Field f, final List names) {
		// 項目位置が指定されている場合は、項目位置から値を取得します。
		int pos = column.position();
		// 項目位置が指定されておらずヘッダ行が使用可能な場合は項目名から値を取得します。
		if (pos < 0 && names != null) {
			pos = names.indexOf(defaultIfEmpty(column.name(), f.getName()));
		}
		// 項目名と項目位置のどちらも使用できない場合は例外をスローします。
		if (pos == -1) {
			throw new IllegalStateException(String.format("Invalid CsvColumn field %s", f.getName()));
		}
		return pos;
	}

	public static String defaultIfEmpty(final String str, final String defaultStr) {
		return (str == null || str.isEmpty()) ? defaultStr : str;
	}

	// ------------------------------------------------------------------------

	private static Format createFormat(final CsvColumn column, final Field f) {
		final String pattern = column.format();
		if (pattern.isEmpty()) {
			return null;
		}

		final Locale locale = column.language().isEmpty() ? Locale.getDefault() : new Locale(column.language(), column.country());
		if (Date.class.isAssignableFrom(f.getType())) {
			final SimpleDateFormat format = new SimpleDateFormat(pattern, locale);
			if (!column.timezone().isEmpty()) {
				format.setTimeZone(TimeZone.getTimeZone(column.timezone()));
			}
			return format;
		}
		final DecimalFormat format = new DecimalFormat(pattern, DecimalFormatSymbols.getInstance(locale));
		if (!column.currency().isEmpty()) {
			format.setCurrency(Currency.getInstance(column.currency()));
		}
		return format;
	}

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy