001/*
002 * Copyright (c) 2009 The openGion Project.
003 *
004 * Licensed under the Apache License, Version 2.0 (the "License");
005 * you may not use this file except in compliance with the License.
006 * You may obtain a copy of the License at
007 *
008 *     http://www.apache.org/licenses/LICENSE-2.0
009 *
010 * Unless required by applicable law or agreed to in writing, software
011 * distributed under the License is distributed on an "AS IS" BASIS,
012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
013 * either express or implied. See the License for the specific language
014 * governing permissions and limitations under the License.
015 */
016package org.opengion.fukurou.model;
017
018import java.util.Locale;                                                                                // 6.2.0.0 (2015/02/27)
019import java.util.Date;                                                                                  // 6.2.0.0 (2015/02/27)
020import java.util.concurrent.ConcurrentMap;                                              // 6.4.3.4 (2016/03/11)
021import java.util.concurrent.ConcurrentHashMap;                                  // 6.4.3.1 (2016/02/12) refactoring
022import java.text.DateFormat;                                                                    // 6.2.0.0 (2015/02/27)
023import java.text.SimpleDateFormat;                                                              // 6.2.0.0 (2015/02/27)
024
025import org.apache.poi.ss.usermodel.BuiltinFormats;                              // 6.2.0.0 (2015/02/27)
026import org.apache.poi.ss.usermodel.DateUtil;                                    // 6.2.0.0 (2015/02/27)
027import org.apache.poi.ss.util.NumberToTextConverter;                    // 6.2.0.0 (2015/02/27)
028import org.apache.poi.hssf.record.ExtendedFormatRecord;                 // 6.2.0.0 (2015/02/27)
029import org.apache.poi.hssf.record.FormatRecord;                                 // 6.2.0.0 (2015/02/27)
030import org.apache.poi.hssf.record.NumberRecord;                                 // 6.2.0.0 (2015/02/27)
031import org.apache.poi.xssf.model.StylesTable;                                   // 6.2.0.0 (2015/02/27)
032import org.apache.poi.xssf.usermodel.XSSFCellStyle;                             // 6.2.0.0 (2015/02/27)
033
034/**
035 * POI による、Excel(xlsx)の読み取りクラスです。
036 *
037 * xlsx形式のEXCELを、イベント方式でテキストデータを読み取ります。
038 * このクラスでは、XSSF(.xlsx)形式のファイルを、TableModelHelper を介したイベントで読み取ります。
039 * TableModelHelperイベントは、openGion形式のファイル読み取りに準拠した方法をサポートします。
040 * ※ openGion形式のEXCELファイルとは、#NAME 列に、カラム名があり、#で始まる
041 *    レコードは、コメントとして判断し、読み飛ばす処理の事です。
042 *
043 * @og.rev 6.0.3.0 (2014/11/13) 新規作成
044 * @og.rev 6.2.0.0 (2015/02/27) パッケージ変更(util → model),クラス名変更(ExcelReader_XSSF → EventReader_XLSX)
045 * @og.group ファイル入力
046 *
047 * @version  6.0
048 * @author   Kazuhiko Hasegawa
049 * @since    JDK7.0,
050 */
051public final class ExcelStyleFormat {
052        /** このプログラムのVERSION文字列を設定します。   {@value} */
053        private static final String VERSION = "6.4.3.3 (2016/03/04)" ;
054
055        /** 6.2.0.0 (2015/02/27) 日付型の処理(DATE=0,DATETIME=1,TIME=2) */
056        private static final String[] DATE_TYPE = { "yyyyMMdd" , "yyyyMMddHHmmss" , "HHmmss" };
057        /** 6.3.1.0 (2015/06/28) 外部からも使えるように、static化します。 */
058        private static final DateFormat[] DT_FORMAT = new DateFormat[DATE_TYPE.length];                 // 6.4.1.1 (2016/01/16) dtFormat → DT_FORMAT refactoring
059        private final StylesTable                       stylesTable ;
060
061        // 6.2.0.0 (2015/02/27) 日付型の処理(DATE=0,DATETIME=1,TIME=2)
062        // private final List<Integer>                  extFmtIdx = new ArrayList<>();  // ExtendedFormatRecord のアドレス(順番) の順番に、FormatIndexを設定する。
063        /** 6.4.3.1 (2016/02/12) PMD refactoring. HashMap → ConcurrentHashMap に置き換え。 */
064        private final ConcurrentMap<Integer,String> fmtStrMap = new ConcurrentHashMap<>();      // FormatIndex をキーに、Format文字列 を管理
065        /** 6.4.3.1 (2016/02/12) PMD refactoring. HashMap → ConcurrentHashMap に置き換え。 */
066        private final ConcurrentMap<Integer,Integer> extFmtIdxMap = new ConcurrentHashMap<>();  // ExtendedFormatRecord のアドレス(順番) をキーに、FormatIndexを設定する。
067        private int   extFmtCnt ;
068
069        /**
070         * XSL系 コンストラクター
071         *
072         * XSL 処理では、HSSFListener のイベント処理のうち、NumberRecord の値取得に
073         * 必要な内部処理を、実行します。
074         * 具体的には、ExtendedFormatRecord レコードから、FormatIndex と 並び順を
075         * 管理するMapと、FormatRecord レコードから、IndexCode と フォーマット文字列を
076         * 管理するMap を作成し、NumberRecordレコードの XFIndex から、ExtendedFormatRecord を
077         * 経由して、FormatRecord のフォーマット文字列 を取得し、日付フォーマットの場合は、
078         * 日付文字列に、それ以外は、数字文字列に変換する手助けを行います。
079         *
080         * @og.rev 6.2.0.0 (2015/02/27) 日付型の処理(DATE=0,DATETIME=1,TIME=2)
081         */
082        public ExcelStyleFormat() {
083                stylesTable = null;
084        }
085
086        /**
087         * XSLX系 コンストラクター
088         *
089         * StylesTable は、日付型をはじめとする、EXCELのスタイルのフォーマットを管理しています。
090         * XSLX形式のEXCELをパースする場合に、このコンストラクタを使用して、StylesTableオブジェクトを
091         * 設定します。
092         *
093         * @og.rev 6.2.0.0 (2015/02/27) 日付型の処理(DATE=0,DATETIME=1,TIME=2)
094         *
095         * @param       styles StylesTableオブジェクト
096         */
097        public ExcelStyleFormat( final StylesTable styles ) {
098                stylesTable = styles;
099        }
100
101        /**
102         * XSL系 ExtendedFormatRecordレコードの設定。
103         *
104         * @og.rev 6.2.0.0 (2015/02/27) 日付型の処理(DATE=0,DATETIME=1,TIME=2)
105         *
106         * @param       extFmtRec ExtendedFormatRecordレコード
107         */
108        public void addExtFmtRec( final ExtendedFormatRecord extFmtRec ) {
109                final short fmtIdx = extFmtRec.getFormatIndex();
110                final short xfType = extFmtRec.getXFType();
111                // Listに アドレス(順番) の順番に、FormatIndexを設定する。
112        //      extFmtIdx.add( Integer.valueOf( fmtIdx ) );
113
114                // タイプを判別して、アドレス(順番)をキーに、Mapに登録することで、データ件数を削減します。
115                if( xfType == ExtendedFormatRecord.XF_CELL ) {
116                        //                              アドレス(順番)               FormatIndex
117                        // 2.0.0 (2024/01/12) PMD 7.0.0 UnnecessaryBoxing
118//                      extFmtIdxMap.put( Integer.valueOf( extFmtCnt ),Integer.valueOf( fmtIdx ) );
119                        extFmtIdxMap.put( extFmtCnt , (int)fmtIdx );
120        //              System.out.println( "fmtIdx=[" + fmtIdx + "] , xfType=[" + xfType + "] , CNT=" + extFmtCnt );
121                }
122                extFmtCnt++;
123        }
124
125        /**
126         * XSL系 FormatRecordレコードの設定。
127         *
128         * @og.rev 6.2.0.0 (2015/02/27) 日付型の処理(DATE=0,DATETIME=1,TIME=2)
129         *
130         * @param       fmtRec FormatRecordレコード
131         */
132        public void addFmtRec( final FormatRecord fmtRec ) {
133                final int    idxc = fmtRec.getIndexCode();
134                final String fmt  = fmtRec.getFormatString();
135
136                // IndexCode をキーに、Format文字列を登録する。
137                // 2.0.0 (2024/01/12) PMD 7.0.0 UnnecessaryBoxing
138//              fmtStrMap.put( Integer.valueOf( idxc ) , fmt );
139                fmtStrMap.put( idxc , fmt );
140        //      System.out.println( "fmtRec=[" + idxc + "], fmt=[" + fmt + "]" );
141        }
142
143        /**
144         * XSLX系 セルスタイル文字列(スタイル番号)から、データフォーマットを取得します。
145         *
146         * StylesTable は、日付型をはじめとする、EXCELのスタイルのフォーマットを管理しています。
147         * XSLX形式のEXCELのフォーマット文字列を取得する場合に、使用します。
148         *
149         * @og.rev 6.2.0.0 (2015/02/27) 日付型の処理(DATE=0,DATETIME=1,TIME=2)
150         *
151         * @param       cellStyleStr    セルスタイル文字列(スタイル番号)
152         * @param       val                             endElement時の値文字列
153         * @return      日付データか、数値データかを判別した結果の文字列
154         */
155        public String getNumberValue( final String cellStyleStr , final String val ) {
156                String fmtStr = null;
157
158                if( stylesTable != null && cellStyleStr != null && !cellStyleStr.isEmpty() ) {
159                        final int stIdx = Integer.parseInt( cellStyleStr );
160                        final XSSFCellStyle style = stylesTable.getStyleAt( stIdx );
161                        fmtStr = style.getDataFormatString();
162
163                        // 必要かどうか不明。テスト時は、ユーザー定義フォーマットも、上記処理で取得できていた。
164                        if( fmtStr == null ) {
165                                final int fmtIdx = style.getDataFormat();
166                                fmtStr = BuiltinFormats.getBuiltinFormat( fmtIdx );
167                        }
168
169        //              if( fmtStr != null ) {
170        //                      System.out.println( "style=[" + cellStyleStr + "], stIdx=[" + stIdx + "],  fmtStr=[" + fmtStr + "]" );
171        //              }
172                }
173
174                return getNumberValue( fmtStr , Double.parseDouble( val ) ) ;
175        }
176
177        /**
178         * XSL系 Numberレコードから、日付データか、数値データかを判別して返します。
179         *
180         * 日付フォーマットの判定処理を #isDateFormat(String) で行い、日付の場合は、
181         * 各タイプ(日付、日時、時刻)に応じた、文字列を返します。
182         * 日付フォーマットでない場合は、数字化文字列を返します。
183         *
184         * @og.rev 6.2.0.0 (2015/02/27) 日付型の処理(DATE=0,DATETIME=1,TIME=2)
185         *
186         * @param       numrec  NumberRecordレコード
187         * @return      日付データか、数値データかを判別した結果の文字列
188         */
189        public String getNumberValue( final NumberRecord numrec ) {
190                final int xfIdx  = numrec.getXFIndex();                                                         // extFmtCnt の事
191                // 2.0.0 (2024/01/12) PMD 7.0.0 UnnecessaryBoxing
192//              final int fmtIdx = extFmtIdxMap.get( Integer.valueOf( xfIdx ) );
193                final int fmtIdx = extFmtIdxMap.get( xfIdx );
194
195        //      final String fmtStr = fmtIdx < HSSFDataFormat.getNumberOfBuiltinBuiltinFormats()
196        //                                                      ? HSSFDataFormat.getBuiltinFormat( (short)fmtIdx )
197        //                                                      : fmtStrMap.get( Integer.valueOf( fmtIdx ) );
198
199                final String fmtStr = fmtIdx < BuiltinFormats.FIRST_USER_DEFINED_FORMAT_INDEX
200                                                                ? BuiltinFormats.getBuiltinFormat( fmtIdx )
201                                                                // 2.0.0 (2024/01/12) PMD 7.0.0 UnnecessaryBoxing
202//                                                              : fmtStrMap.get( Integer.valueOf( fmtIdx ) );
203                                                                : fmtStrMap.get( fmtIdx );
204
205        //              if( fmtStr != null ) {
206        //                      System.out.println( "xfIdx=[" + xfIdx + "], fmtIdx=[" + fmtIdx + "],  fmtStr=[" + fmtStr + "]" );
207        //              }
208
209                return getNumberValue( fmtStr , numrec.getValue() ) ;
210        }
211
212        /**
213         * フォーマット情報と値から、日付データか、数値データかを判別して返します。
214         *
215         * 日付フォーマットの判定処理を #isDateFormat(String) で行い、日付の場合は、
216         * 各タイプ(日付、日時、時刻)に応じた、文字列を返します。
217         * 日付フォーマットでない場合は、数字文字列を返します。
218         *
219         * @og.rev 6.2.0.0 (2015/02/27) 日付型の処理(DATE=0,DATETIME=1,TIME=2)
220         * @og.rev 6.3.1.0 (2015/06/28) 外部からも使えるように、static化します。
221         *
222         * @param       fmtStr  フォーマット情報
223         * @param       val             Numberレコードのデータ
224         * @return      日付データか、数値データかを判別した結果の文字列
225         */
226        public static String getNumberValue( final String fmtStr , final double val ) {
227                return isDateFormat( fmtStr )
228                                        ? dateFormat( val )                                                     // 日付
229                                        : NumberToTextConverter.toText( val ) ;         // 数字
230        }
231
232        /**
233         * フォーマット文字列から、日付型フォーマットかどうかの判定を行います。
234         *
235         * ここでは、日本式のフォーマットや、ユーザー定義の日付フォーマットでも、
236         * ある程度判定できるように、処理しています。
237         * 以下の文字列を含む場合は、true を返し、それ以外は、false です。
238         *   "年","月","日","yy","y/m","m/d","h:m"
239         * ただし、General(標準) は、除外しています。
240         *
241         * @og.rev 6.2.0.0 (2015/02/27) 日付型の処理(DATE=0,DATETIME=1,TIME=2)
242         * @og.rev 6.3.1.0 (2015/06/28) 外部からも使えるように、static化します。
243         * @og.rev 6.4.3.3 (2016/03/04) PMD refactoring. These nested if statements could be combined
244         *
245         * @param       fmt     フォーマット文字列
246         * @return      判定結果 [true:日付型/false:それ以外]
247         */
248        private static boolean isDateFormat( final String fmt ) {
249                return  fmt != null && !fmt.isEmpty() && !"General".equalsIgnoreCase( fmt ) &&
250                        (       fmt.contains( "年" ) || fmt.contains( "月"  ) || fmt.contains( "日"  ) ||
251                                fmt.contains( "yy" ) || fmt.contains( "y/m" ) || fmt.contains( "m/d" ) ||
252                                fmt.contains( "h:m" ) ) ;
253        }
254
255        /**
256         * 日付型の値を、最適なフォーマットで変換して返します。
257         *
258         * 日付データは、(DATE=0,DATETIME=1,TIME=2) に分類できます。
259         * DATE とは、日付のみの状態で、引数の val は、整数に変換できます。
260         * その場合、"yyyyMMdd" フォーマットで変換します。
261         * DATETIME とは、日付+時刻なので、"yyyyMMddHHmmss" に変換します。
262         * TIME は、日付情報を持っていないため、"HHmmss" に変換します。
263         *
264         * @og.rev 6.2.0.0 (2015/02/27) 日付型の処理(DATE=0,DATETIME=1,TIME=2)
265         * @og.rev 6.3.1.0 (2015/06/28) 外部からも使えるように、static化します。
266         *
267         * @param       val     日付型の値
268         * @return      日付型の変換結果
269         */
270        public static String dateFormat( final double val ) {
271                int dtType = 0;         // 日付型の処理(DATE=0,DATETIME=1,TIME=2)
272                if( val < 1.0d ) {                                                                                              // 日付部が無い → TIME=2
273                        dtType = 2;
274                }
275                // 6.3.9.0 (2015/11/06) Avoid if (x != y) ..; else ..; (PMD)
276                else if( Double.compare( val , Math.floor( val ) ) == 0 ) {             // 整数(Long 相当) → DATE=0
277                        dtType = 0;
278                }
279                else {                                                                                                                  // 整数でない → DATETIME=1
280                        dtType = 1;
281                }
282
283                DateFormat dtfmt = DT_FORMAT[dtType];                   // 各タイプ別にキャッシュしている。
284                if( dtfmt == null ) {
285                        dtfmt = new SimpleDateFormat( DATE_TYPE[dtType] , Locale.JAPAN );               // 初めての場合は新規作成
286                        DT_FORMAT[dtType] = dtfmt;
287                }
288
289                final Date dt = DateUtil.getJavaDate( val );
290                return dtfmt.format( dt );
291        }
292}