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.hayabusa.db;
017
018import java.util.Calendar;
019
020import org.opengion.fukurou.util.HybsDateUtil;                                                  // 8.5.4.2 (2024/01/12) Calendar インスタンスの作成方法変更
021// import org.opengion.fukurou.system.LogWriter;
022import static org.opengion.fukurou.system.HybsConst.CR ;                                // 6.1.0.0 (2014/12/26)
023import static org.opengion.fukurou.system.HybsConst.BUFFER_MIDDLE;              // 6.1.0.0 (2014/12/26) refactoring
024
025/**
026 * データのコード情報を取り扱うクラスです。
027 *
028 * 開始、終了、ステップの情報から、HTMLのメニューやリストを作成するための オプション
029 * タグを作成したり、与えられたキーをもとに、チェック済みのオプションタグを作成したりします。
030 *
031 * ここでは、時間(時:分)の自動生成を行います。パラメータで、開始、終了、ステップ、開始前設定値、終了後設定値
032 * を指定できます。
033 * キーは、4文字の HHMM 形式で与えられます。ラベルは、HH:MM になります。
034 * ステップは、分単位です。つまり、1時間の場合は、"60" と指定します。"0100"ではありません。
035 * 開始前設定値、終了後設定値はそれぞれ、開始の前と終了の後ろに特別に値を設定できます。
036 *
037 * 開始、または、終了に、現在時刻からの相対値を指定する事ができます。
038 * H1 ~ HXXX とすれば、現在時刻の時に数字部分を+-します。分は0に初期化されます。
039 *
040 * パラメータの初期値は、開始(0700)、終了(1900)、ステップ(30)、開始前設定値(null)、終了後設定値(null) です。
041 *
042 * 例:0800,2000,30  → 0800,0830,0900,0930,1000,・・・1900,1930,2000 のプルダウン
043 * 例:0800,2000,30,0000:△,2400:▽  → 0000,0800,0830,0900,0930,1000,・・・1900,1930,2000,2400 のプルダウン
044 *
045 * @og.group 選択データ制御
046 * @og.rev 5.6.1.1 (2013/02/08) 新規追加
047 * @og.rev 8.5.6.1 (2024/03/29) 継承で使えるように、一部修正します。
048 *
049 * @version  4.0
050 * @author   Kazuhiko Hasegawa
051 * @since    JDK5.0,
052 */
053// 8.5.5.1 (2024/02/29) spotbugs CT_CONSTRUCTOR_THROW(コンストラクタで、Excweptionを出さない) class を final にすれば、警告は消える。
054// public class Selection_HM extends Selection_NULL {
055public final class Selection_HM extends Selection_NULL {
056//      private final String    CACHE ;                 // 8.5.6.1 (2024/03/29) 親クラスで定義
057//      private final String    ST_ED_STEP ;    // 8.5.6.1 (2024/03/29) 未使用なので削除
058
059        private final long      maxCacheTime ;          // キャッシュの破棄タイミングを計るための最大有効時間
060
061        /**
062         * コンストラクター
063         *
064         * 引数は、開始、終了、ステップ、開始前設定値、終了後設定値 です。
065         * パラメータの初期値は、開始(0700)、終了(1900)、ステップ(30)、開始前設定値(null)、終了後設定値(null) です。
066         *
067         * @og.rev 6.2.6.0 (2015/06/19) type別Selectionの場合、ラベルリソースを使用する為、言語を引数で渡す。
068         * @og.rev 6.3.4.0 (2015/08/01) Selection_HM の引数から、lang 属性を削除します。
069         * @og.rev 7.2.9.4 (2020/11/20) String.format でゼロ埋め対応
070         * @og.rev 8.5.4.2 (2024/01/12) Calendar インスタンスの作成方法変更
071         *
072         * @param       editPrm 開始、終了、ステップ、開始前設定値、終了後設定値 を表す引数(例:0800,2000,30)
073         */
074        public Selection_HM( final String editPrm ) {
075//              super();                                // 6.4.1.1 (2016/01/16) PMD refactoring. It is a good practice to call super() in a constructor
076                super( editPrm );               // 8.5.6.1 (2024/03/29) 継承親で、引数付きコンストラクタで初期設定されます。
077        //      if( param.length < 3 ) {
078        //      final String errMsg = "引数は、開始、終了、ステップ、[開始前設定値]、[終了後設定値]です。最低でも3個必要です。";
079        //              throw new IllegalArgumentException( errMsg );
080        //      }
081
082                final String[] param = editPrm == null ? new String[0] : editPrm.split( "," ) ;
083                final String step  = param.length > 2 ? param[2].trim() : "30" ;
084
085                final int stepTime      = Integer.parseInt( step );
086                if( stepTime == 0 ) {
087                        final String errMsg = "ステップ に 0 は指定できません。無限ループします。";
088                        throw new IllegalArgumentException( errMsg );
089                }
090
091                final String start = param.length > 0 ? param[0].trim() : "0700" ;
092                final String end   = param.length > 1 ? param[1].trim() : "1900" ;
093                final String stOp  = param.length > 3 ? param[3].trim() : null ;
094                final String enOp  = param.length > 4 ? param[4].trim() : null ;
095
096                // 8.5.4.2 (2024/01/12) Calendar インスタンスの作成方法変更
097//              final Calendar cal = Calendar.getInstance();
098//              calendarCalc( cal, start );
099                final Calendar cal = HybsDateUtil.newCalendar( start );
100
101//              final Calendar endCal = Calendar.getInstance();
102//              calendarCalc( endCal, end );
103                final Calendar endCal = HybsDateUtil.newCalendar( end );
104
105                final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE );
106
107                addOption( buf,stOp );          // 開始前設定値 の追加
108
109                // cal.before( endCal ) では、同一時刻の場合に false になる為、ここの判定では使えません。
110                // sign を掛け算しているのは、逆順対応
111                final int sign = stepTime > 0 ? 1 : -1 ;        // ステップの符号。
112                while( endCal.compareTo( cal ) * sign >= 0 ) {
113                        final int hh = cal.get( Calendar.HOUR_OF_DAY ); // 時
114                        final int mm = cal.get( Calendar.MINUTE );              // 分
115
116                        // 7.2.9.4 (2020/11/20) String.format でゼロ埋め対応
117//                      final String hVal = hh < 10 ? "0" + hh : String.valueOf( hh ) ;
118//                      final String mVal = mm < 10 ? "0" + mm : String.valueOf( mm ) ;
119
120                        final String hVal = String.format( "%02d", hh );
121                        final String mVal = String.format( "%02d", mm );
122
123                        // 6.0.2.5 (2014/10/31) char を append する。
124                        buf.append( "<option value=\"" ).append( hVal ).append( mVal )
125                                .append( "\">" ).append( hVal ).append( ':' ).append( mVal ).append( "</option>" );
126
127                        cal.add( Calendar.MINUTE,stepTime );    // 時刻に加えるのは、分
128                }
129
130                addOption( buf,enOp );                                          // 終了後設定値 の追加
131
132                cache = buf.toString();                                         // 8.5.6.1 (2024/03/29) super.cache
133//              // 8.5.6.1 (2024/03/29) 未使用なので削除
134//              ST_ED_STEP = "Start=" + start + " , End=" + end + " , Step=" + step + " , StartBefore=" + stOp + " , EndAfter=" + enOp ;
135
136                // キャシュの有効期間を求めるための時刻を作成します。キャッシュは、時間指定があれば、同一時間内のみ有効です。
137                final Calendar now   = Calendar.getInstance();
138                final boolean nowBase = start.charAt(0) == 'H' || end.charAt(0) == 'H' ;
139                if( nowBase ) {
140                        now.add( Calendar.HOUR   , 1 );                         // 1時間進めます。
141                        now.set( Calendar.MINUTE , 0 );                         // 分、秒 をリセットします。
142                        now.set( Calendar.SECOND , 0 );
143                }
144                else {
145                        now.add( Calendar.YEAR   , 1 );                         // 1年間進めます。(現在時刻をベースに指定ない為、無制限キャッシュの意味)
146                }
147
148                maxCacheTime = now.getTimeInMillis() ;
149        }
150
151//      /**
152//       * 開始、または 終了の文字列から、カレンダオブジェクトを作成します。
153//       * 基準となる日付に計算した結果を反映させます。
154//       *
155//       * prmB は、日付についての加減算処理を行うためのコマンドを指定します。
156//       * ・数字       :HHMM 形式の時分です。
157//       * ・H1 ~ HXXX :現在時刻に数字部分を+-します。分は0に初期化されます。
158//       *
159//       * @og.rev 8.5.4.2 (2024/01/12) Calendar インスタンスの作成方法変更( HybsDateUtil.newCalendar( String ) に移行
160//       * @og.rev 8.5.4.2 (2024/01/12) Calendar インスタンスの作成方法変更
161//       *
162//       * @param   cal     基準となる日付(Calendarオブジェクト)
163//       *
164//       * @param   prmB        処理コマンド
165//       */
166//      private void calendarCalc( final Calendar cal,final String prmB ) {
167//              final boolean nowBase = prmB.charAt(0) == 'H' ;
168//
169//              if( nowBase ) {
170//                      final int hour  = Integer.parseInt( prmB.substring( 1 ) );
171//                      cal.add( Calendar.HOUR_OF_DAY,hour );
172//                      cal.set( Calendar.MINUTE     ,0 );
173//                      cal.set( Calendar.SECOND     ,0 );
174//              }
175//              else {
176//                      final int hour  = Integer.parseInt( prmB.substring( 0,2 ) );
177//                      final int minute        = Integer.parseInt( prmB.substring( 2,4 ) );
178//                      cal.set( Calendar.HOUR_OF_DAY,hour );
179//                      cal.set( Calendar.MINUTE     ,minute );
180//                      cal.set( Calendar.SECOND     ,0 );
181//              }
182//      }
183
184        /**
185         * 開始前設定値、または 終了後設定値の文字列から、オプション文字列を合成します。
186         * このオプションは、引数のStringBuilder に、オプションタグを追加して返します。
187         * optVal が null の場合は、処理しません。
188         *
189         * @param   buf     文字列連結する StringBuilderオブジェクト。このオブジェクトに追加します。
190         * @param   optVal  開始前設定値、または 終了後設定値 文字列("0000:△" 形式)
191         *
192         */
193        private void addOption( final StringBuilder buf,final String optVal ) {
194                if( optVal != null ) {
195                        final int adrs = optVal.indexOf( ':' );
196                        if( adrs > 0 ) {
197                                // 6.0.2.5 (2014/10/31) char を append する。
198                                buf.append( "<option value=\"" ).append( optVal.substring( 0,adrs ) )
199                                        .append( "\">" ).append( optVal.substring( adrs+1 ) ).append( "</option>" );
200                        }
201                        // 開始前設定値 が存在する場合、"0000:△" 形式必須
202                        else {
203                                final String errMsg = "引数は、0000:△ 形式です。";
204                                throw new IllegalArgumentException( errMsg );
205                        }
206                }
207        }
208
209//      /**
210//       * 初期値が選択済みの 選択肢(オプション)を返します。
211//       * このオプションは、引数の値を初期値とするオプションタグを返します。
212//       * このメソッドでは、引数のuseShortLabelがtrueに指定された場合に、ラベル(短)をベースとした
213//       * ツールチップ表示を行います。
214//       *
215//       * @og.rev 8.5.6.1 (2024/03/29) 継承元と同じなので削除
216//       *
217//       * @param   selectValue  選択されている値
218//       * @param   seqFlag  シーケンスアクセス機能 [true:ON/false:OFF]
219//       * @param   useShortLabel ラベル(短)をベースとしたオプション表示を行うかどうか。(未使用)
220//       *
221//       * @return  オプションタグ
222//       * @og.rtnNotNull
223//       */
224//      @Override
225//      public String getOption( final String selectValue,final boolean seqFlag, final boolean useShortLabel ) {
226//              // 8.5.5.1 (2024/02/29) PMD 7.0.0 OnlyOneReturn メソッドには終了ポイントが 1 つだけ必要
227//              final String rtn;
228//
229//              // マッチするアドレスを探す。
230//              final int selected = CACHE.indexOf( "\"" + selectValue + "\"" );
231//              if( selected < 0 ) {
232//                      if( selectValue != null && selectValue.length() > 0 ) {
233//                              final String errMsg = "時分範囲に存在しない値が指定されました。"
234//                                                      + " value=[" + selectValue + "]"
235//                                                      + CR + ST_ED_STEP ;
236//                              LogWriter.log( errMsg );
237//                      }
238////                    return CACHE;
239//                      rtn = CACHE;
240//              }
241//              else {
242//                      // "時分" 文字列の位置が、selected なので、時分の文字数+2までが、前半部分になる。(時分の文字数は4固定のはず)
243//                      final int indx = selected + selectValue.length() + 2 ;
244//
245//                      final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE );
246//                      // 3.6.0.6 (2004/10/22) シーケンスアクセス機能を指定する seqFlag を導入
247//                      if( seqFlag ) {
248//                              buf.append( "<option value=\"" ).append( selectValue ).append( '"' );           // 6.0.2.5 (2014/10/31) char を append する。
249//                      }
250//                      else {
251//                              buf.append( CACHE.substring( 0,indx ) );
252//                      }
253//                      buf.append( " selected=\"selected\"" )
254//                              .append( CACHE.substring( indx ) );
255////                    return buf.toString() ;
256//                      rtn = buf.toString() ;
257//              }
258//              return rtn;
259//      }
260
261        /**
262         * 選択肢(value)に対するラベルを返します。
263         * 選択肢(value)が、存在しなかった場合は、選択肢そのものを返します。
264         * このメソッドでは、短縮ラベルを返すかどうかを指定するフラグは使いません。
265         * getValueLabel( XX,false ) は、getValueLabel( XX ) と同じです。
266         *
267         * @param       selectValue     選択肢の値
268         * @param       isSLbl  短縮ラベルを [true:使用する/false:しない](未使用)
269         *
270         * @return  選択肢のラベル
271         * @see     #getValueLabel( String )
272         */
273        @Override
274        public String getValueLabel( final String selectValue,final boolean isSLbl ) {
275                // あろうがなかろうが、選択肢そのものを返します。
276                return selectValue;
277        }
278
279        /**
280         * オブジェクトのキャッシュが時間切れかどうかを返します。
281         * キャッシュが時間切れ(無効)であれば、true を、有効であれば、
282         * false を返します。
283         *
284         * @return  キャッシュが時間切れなら true
285         */
286        @Override
287        public boolean isTimeOver() {
288                return System.currentTimeMillis() > maxCacheTime ;
289        }
290}