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}