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.plugin.column; 017 018import org.opengion.hayabusa.common.HybsSystem; 019import org.opengion.hayabusa.common.HybsSystemException; 020import org.opengion.hayabusa.db.AbstractEditor; 021import org.opengion.hayabusa.db.CellEditor; 022import org.opengion.hayabusa.db.DBColumn; 023import org.opengion.hayabusa.db.SelectionCellEditor; // 6.2.2.0 (2015/03/27) 024import org.opengion.hayabusa.db.Selection; 025import org.opengion.hayabusa.db.SelectionFactory; 026import org.opengion.fukurou.util.StringFormat; 027import org.opengion.fukurou.util.XHTMLTag; 028import org.opengion.fukurou.util.Attributes; 029import org.opengion.fukurou.util.TagBuffer; 030import org.opengion.fukurou.util.StringUtil; // 8.5.4.2 (2024/01/12) import static … を個別に記述 031 032// import static org.opengion.fukurou.util.StringUtil.isNull; // 6.1.1.0 (2015/01/17) 033 034/** 035 * カラムの編集パラメーターのSQL文の実行結果より、プルダウンメニューを作成して 036 * 編集する場合に使用するエディタークラスです。 037 * 038 * 編集パラメータには、プルダウンメニューを作成するための、SQL文を記述します。 039 * このSQL文は、select KEY,LABEL from xx ・・・ という構文で、KEY部分とLABEL部分が 040 * 選択されます。 041 * 第一カラムはキー、第二カラムはラベルでこの2つは必須です。第三カラムは短縮ラベル、 042 * 第四カラムはグループ(optgroup)、第五カラムは色付け等に使うクラスです。 043 * 第五カラムの指定方法には、3通りあります。 044 * ここでは、 045 * ① "=" を含む場合は、そのままセット。(style='AAAA' など) 046 * ② disabled 単体の場合は、disabled="disabled" をセット 047 * ③ それ以外は、class= の後に、引数をセット 048 * します。 049 * 短縮ラベルが設定されている場合、一覧でこのエディタが適用されると短縮ラベル表示を 050 * した上でマウスオーバー時はツールチップで通常のラベルを表示します。 051 * 052 * 各カラムの値(value値)に、AAA:BBB:CCC:DDD という値を設定できます。これは、 053 * $1,$2,$3,$4 に割り当てなおして、QUERYを実行します。また、$1 は、本来の値として、 054 * メニューの初期値設定等に使用します。上記の例では、AAA が値で、それ以降は、 055 * 引数になります。 056 * 又、$Cには自分自身のカラム名を割り当てます。 057 * この機能を使用すれば、動的メニューを行ごとに条件を変えて作成することが 058 * 可能になります。 059 * 例:select KEY,LABEL from xx where KUBUN='$2' and CDK='$3' 060 * さらに、元の文字列"AAA:BBB:CCC:DDD"は、$0 に割り当てられます。割り当てがない 061 * 変数は、""(ゼロ文字列)として、扱われます。 062 * 063 * 編集パラメータに"SEQ"と記述することで正方向にしか選べないシークメニューを実現できます。 064 * これにより、シーケンスにステータスを順に挙げていくような、プルダウンメニュー 065 * を作成することが出来ます。(逆に戻れないメニュー) 066 * 067 * カラムの表示に必要な属性は、DBColumn オブジェクト より取り出します。 068 * このクラスは、DBColumn オブジェクト毎に1つ作成されます。 069 * 070 * 8.0.0.1 (2021/10/08) 071 * 検索機能(絞り機能)付きのプルダウンメニューを作成する為には、文字パラメータに 072 * "class=sel2sel" と入力してください。 073 * 074 * @og.rev 3.2.3.0 (2003/06/06) 新規作成 075 * @og.rev 3.4.0.1 (2003/09/03) DB検索をリアルタイムに変更。 076 * @og.rev 4.3.6.0 (2009/04/01) eventColumn対応 077 * @og.rev 5.4.3.6 (2012/01/19) コメント変更 078 * @og.rev 6.2.2.0 (2015/03/27) SelectionCellEditor I/Fを追加 079 * @og.group データ編集 080 * 081 * @version 4.0 082 * @author Kazuhiko Hasegawa 083 * @since JDK5.0, 084 */ 085// 8.5.5.1 (2024/02/29) spotbugs CT_CONSTRUCTOR_THROW(コンストラクタで、Excweptionを出さない) class を final にすれば、警告は消える。 086// public class Editor_DBMENU extends AbstractEditor implements SelectionCellEditor { 087public final class Editor_DBMENU extends AbstractEditor implements SelectionCellEditor { 088 /** このプログラムのVERSION文字列を設定します。 {@value} */ 089 private static final String VERSION = "8.5.3.0 (2023/09/08)" ; 090 091 private static final String STR_CLS = "class=" ; // 8.0.0.1 (2021/10/08) 092 private static final int LEN_CLS = STR_CLS.length() ; // 8.0.0.1 (2021/10/08) 093 094 private final String query ; 095 private final String dbid ; 096 private final String lang ; // 4.0.0 (2006/11/15) 097 private final boolean addNoValue ; // 3.5.5.7 (2004/05/10) 098 private final boolean seqFlag ; // 3.6.0.6 (2004/10/22) 099 private final String useSLabel ; // 5.5.1.0 (2012/04/03) 100 private final String addKeyLabel ; // 6.2.0.0 (2015/02/27) キー:ラベル形式 101 102 /** 103 * デフォルトコンストラクター。 104 * このコンストラクターで、基本オブジェクトを作成します。 105 * 106 * @og.rev 3.4.0.1 (2003/09/03) 初期値でQUERY文をキープする。 107 * @og.rev 3.6.0.6 (2004/10/22) シーケンスアクセス機能(seqFlag)を追加します 108 * @og.rev 5.5.1.0 (2012/04/03) Slabel対応 109 * @og.rev 6.2.0.0 (2015/02/27) キー:ラベル形式で表示するかどうかを、指定できるようにします。 110 */ 111 public Editor_DBMENU() { 112 super(); // 6.4.1.1 (2016/01/16) PMD refactoring. It is a good practice to call super() in a constructor 113 // 4.3.4.4 (2009/01/01) 114 query = null; 115 dbid = null; 116 lang = null; // 4.0.0 (2006/11/15) 117 addNoValue = false; // 3.5.5.7 (2004/05/10) 118 seqFlag = false; // 3.6.0.6 (2004/10/22) 119 useSLabel = "auto"; // 5.5.1.0 (2012/04/03) 120 addKeyLabel = null; // 6.2.0.0 (2015/02/27) キー:ラベル形式 121 } 122 123 /** 124 * DBColumnオブジェクトを指定したprivateコンストラクター。 125 * 126 * @og.rev 3.3.1.1 (2003/07/03) name , attributes 属性を final にする。 127 * @og.rev 3.4.0.1 (2003/09/03) 継承の親元の変更に伴う実装の移動。 128 * @og.rev 3.5.5.7 (2004/05/10) addNoValue 属性を追加します。 129 * @og.rev 3.5.5.9 (2004/06/07) editorParam 属性が null の場合は、エラーとします。 130 * @og.rev 3.5.6.0 (2004/06/18) XHTMLTag の 内部配列 SELECT_KEY を隠蔽します。 131 * @og.rev 3.6.0.6 (2004/10/22) シーケンスアクセス機能(seqFlag)を追加します 132 * @og.rev 5.1.7.0 (2010/06/01) 動的プルダウン実装見直し 133 * @og.rev 5.5.1.0 (2012/04/03) Slabel対応 134 * @og.rev 6.0.4.0 (2014/11/28) optionAttributes は、コンストラクタで設定します。 135 * @og.rev 6.0.4.0 (2014/11/28) useMultiSelect は、selection ではなく、colomn から取得する。 136 * @og.rev 6.2.0.0 (2015/02/27) キー:ラベル形式で表示するかどうかを、指定できるようにします。 137 * @og.rev 6.4.4.2 (2016/04/01) nameのfinal化 138 * @og.rev 6.9.5.0 (2018/04/23) useMultiSelect 廃止 139 * @og.rev 7.0.5.1 (2019/09/27) optionAttributes が二重に設定されていたため、削除 140 * @og.rev 8.0.0.1 (2021/10/08) dbType パラメータのclass属性対応 141 * @og.rev 8.5.3.0 (2023/09/08) DynamicAttributes対応 142 * 143 * @param clm DBColumnオブジェクト 144 */ 145 private Editor_DBMENU( final DBColumn clm ) { 146 // super(); // 6.4.1.1 (2016/01/16) PMD refactoring. It is a good practice to call super() in a constructor 147 super( clm ); // 6.4.4.2 (2016/04/01) nameのfinal化 148 // name = clm.getName(); 149 addNoValue = clm.isAddNoValue() ; // 3.5.5.7 (2004/05/10) 150 query = clm.getEditorParam(); 151 dbid = clm.getDbid(); 152 lang = clm.getLang(); // 4.0.0.0 (2006/11/15) 153 seqFlag = false; // 3.6.0.6 (2004/10/22) 154 useSLabel = clm.getUseSLabel() ; // 5.5.1.0 (2012/04/03) 155 addKeyLabel = clm.getAddKeyLabel(); // 6.2.0.0 (2015/02/27) キー:ラベル形式 156 157 // 3.5.5.9 (2004/06/07) 158 if( query == null || query.isEmpty() ) { 159 final String errMsg = "DBMENU Editor では、編集パラメータは必須です。" 160 + " name=[" + name + "]" + CR ; 161 throw new HybsSystemException( errMsg ); 162 } 163 164 final String disabled = clm.isWritable() ? null : "disabled" ; 165 166 // 6.1.1.0 (2015/01/17) Attributesの連結記述 167 attributes = new Attributes() 168 .set( "disabled" ,disabled ) 169 .set( "size" ,clm.getRawSize() ) // 8.5.3.0 (2023/09/08) Add 170 .set( clm.getEditorAttributes() ); // #addAttributes( Attributes ) の代替え 171 172 // class設定 8.0.0.1 (2021/10/08) 173 final String dbTypeParam = clm.getDbTypeParam(); 174 if( dbTypeParam != null ) { 175 final int st = dbTypeParam.indexOf( STR_CLS ); 176 if( st >= 0 ) { 177 String newTypePram = dbTypeParam; 178 final int ed = newTypePram.indexOf( ';' , st ); 179 // 文字パラメータの文字列から class属性の文字列を取得する 180 if( ed >= 0 ) { 181 // [0-9]|class=sel2sel;[A-z] の場合、sel2sel を取得 182 newTypePram = newTypePram.substring( st + LEN_CLS , ed ); 183 } 184 else { 185 // [0-9A-z]class=sel2sel の場合、sel2sel を取得 186 newTypePram = newTypePram.substring( st + LEN_CLS ); 187 } 188 // attributes に class 付与 189 attributes.add( "class" , newTypePram ); 190 } 191 } 192 193// // 6.9.5.0 (2018/04/23) useMultiSelect 廃止 194// // 6.1.1.0 (2015/01/17) TagBufferの連結記述 195// tagBuffer.add( XHTMLTag.selectAttri( attributes ) ) 196// .add( attributes.get( "optionAttributes" ) ) // 6.0.4.0 (2014/11/28) 197// .add( "onkeydown" , "setKeySelect(this);" , clm.useMultiSelect() ); // 6.0.4.0 (2014/11/28) 198 199 // 6.1.1.0 (2015/01/17) TagBufferの連結記述 200 // 7.0.5.1 (2019/09/27) optionAttributes が二重に設定されていたため、削除 201 tagBuffer.add( XHTMLTag.selectAttri( attributes ) ); 202// .add( attributes.get( "optionAttributes" ) ); // 6.0.4.0 (2014/11/28) 203 } 204 205 /** 206 * 各オブジェクトから自分のインスタンスを返します。 207 * 自分自身をキャッシュするのか、新たに作成するのかは、各サブクラスの実装に 208 * まかされます。 209 * 210 * @param clm DBColumnオブジェクト 211 * 212 * @return CellEditorオブジェクト 213 * @og.rtnNotNull 214 */ 215 public CellEditor newInstance( final DBColumn clm ) { 216 return new Editor_DBMENU( clm ); 217 } 218 219 /** 220 * データの編集用文字列を返します。 221 * 222 * ここでは、AAA:BBB:CCC:DDD という値を、$1,$2,$3,$4 に割り当てなおして、 223 * QUERYを実行します。また、$1 は、本来の値として、メニューの初期値設定等に 224 * 使用します。上記の例では、AAA が値で、それ以降は、引数になります。 225 * さらに、元の文字列"AAA:BBB:CCC:DDD"は、$0 に割り当てられます。割り当てがない 226 * 変数は、""(ゼロ文字列)として、扱われます。 227 * 又、$Cには自分自身のカラム名を割り当てます。 228 * 229 * @og.rev 3.4.0.1 (2003/09/03) リアルタイムで値を作成する様に変更。 230 * @og.rev 3.5.5.5 (2004/04/23) 新規に Attributes オブジェクトを作成する方式を止めます。 231 * @og.rev 4.3.7.2 (2009/06/15) 属性でidが出力される場合は、idを出力しない 232 * @og.rev 5.1.3.0 (2010/02/01) 一覧表示のみで、ツールチップ表示を行う。 233 * @og.rev 5.1.7.0 (2010/06/01) 動的プルダウン実装見直し 234 * @og.rev 5.5.1.0 (2012/04/03) Slabel対応 235 * 236 * @param value 入力値 237 * 238 * @return データの編集用文字列 239 * @og.rtnNotNull 240 */ 241 @Override 242 public String getValue( final String value ) { 243 final boolean useSlbl = "true".equalsIgnoreCase( useSLabel ); // 5.5.1.0 (2012/04/03) 244 245 // select タグの作成 246 // 6.1.1.0 (2015/01/17) TagBufferの連結記述 247 return getOption( 248 new TagBuffer( "select" ) 249 .add( "name" , name ) // 4.3.6.0 (2009/04/01) 250 // 8.5.4.2 (2024/01/12) import static … を個別に記述 251 .add( "id" , name , StringUtil.isNull( attributes.get( "id" ) ) ) // 4.3.7.2 (2009/06/15) 252 .add( tagBuffer.makeTag() ) 253 , value 254 , useSlbl 255 ).makeTag() ; 256 } 257 258 /** 259 * name属性を変えた、データ表示/編集用のHTML文字列を作成します。 260 * テーブル上の name に 行番号を付加して、名前_行番号 で登録するキーを作成し、 261 * リクエスト情報を1つ毎のフィールドで処理できます。 262 * 263 * ここでは、AAA:BBB:CCC:DDD という値を、$1,$2,$3,$4 に割り当てなおして、 264 * QUERYを実行します。また、$1 は、本来の値として、メニューの初期値設定等に 265 * 使用します。上記の例では、AAA が値で、それ以降は、引数になります。 266 * さらに、元の文字列"AAA:BBB:CCC:DDD"は、$0 に割り当てられます。割り当てがない 267 * 変数は、""(ゼロ文字列)として、扱われます。 268 * 又、$Cには自分自身のカラム名を割り当てます。 269 * 270 * @og.rev 2.0.0.3 (2002/09/26) optionAttributes 属性に "$i" を使うとその行数に置き換る機能を追加。 271 * @og.rev 3.4.0.1 (2003/09/03) リアルタイムで値を作成する様に変更。 272 * @og.rev 3.5.5.0 (2004/03/12) 名前と行番号の区切り記号("__")を、HybsSystem.JOINT_STRING に変更。 273 * @og.rev 3.5.5.5 (2004/04/23) 新規に Attributes オブジェクトを作成する方式を止めます。 274 * @og.rev 4.3.7.2 (2009/06/15) 属性でidが出力される場合は、idを出力しない 275 * @og.rev 5.1.3.0 (2010/02/01) 一覧表示のみで、ツールチップ表示を行う。 276 * @og.rev 5.1.7.0 (2010/06/01) 動的プルダウン実装見直し 277 * @og.rev 5.5.1.0 (2012/04/03) Slabel対応 278 * @og.rev 6.4.5.3 (2016/05/13) value は、コロン区切りの先頭だけ分離する。 279 * 280 * @param row 行番号 281 * @param value 入力値 282 * 283 * @return データ表示/編集用の文字列 284 * @og.rtnNotNull 285 */ 286 @Override 287 public String getValue( final int row,final String value ) { 288 final boolean useSlbl = "auto".equalsIgnoreCase( useSLabel ) || "true".equalsIgnoreCase( useSLabel ); // 5.5.1.0 (2012/04/03) 289 final String newName = name + HybsSystem.JOINT_STRING + row; // 4.3.6.0 (2009/04/01) 290 final String newValue = StringFormat.getValue( value ); // 6.4.5.3 (2016/05/13) コロン区切りの先頭だけ 291 292 // select タグの作成 293 // 6.1.1.0 (2015/01/17) TagBufferの連結記述 294 return getOption( 295 new TagBuffer( "select" ) 296 .add( "name" , newName ) // 4.3.6.0 (2009/04/01) 297 // 8.5.4.2 (2024/01/12) import static … を個別に記述 298 .add( "id" , newName , StringUtil.isNull( attributes.get( "id" ) ) ) // 4.3.7.2 (2009/06/15) 299 .add( tagBuffer.makeTag() ) 300 , value 301 , useSlbl 302 ).makeTag( row,newValue ) ; // 6.4.5.3 (2016/05/13) ※ 元は、makeTag() でしたが、バグ? の可能性がある。 303 } 304 305 /** 306 * 初期値が選択済みの 選択肢(オプション)をTagBuffer に反映します。 307 * このオプションは、引数の値を初期値とするオプションタグ作成し、TagBuffer 308 * に値を設定して返します。 309 * 310 * ここでは、AAA:BBB:CCC:DDD という値を、$1,$2,$3,$4 に割り当てなおして、 311 * QUERYを実行します。また、$1 は、本来の値として、メニューの初期値設定等に 312 * 使用します。上記の例では、AAA が値で、それ以降は、引数になります。 313 * さらに、元の文字列"AAA:BBB:CCC:DDD"は、$0 に割り当てられます。割り当てがない 314 * 変数は、""(ゼロ文字列)として、扱われます。 315 * 又、$Cには自分自身のカラム名を割り当てます。 316 * 317 * @og.rev 3.5.5.7 (2004/05/10) getOption( String value )の廃止を受けて、新規作成 318 * @og.rev 3.6.0.6 (2004/10/22) シーケンスアクセス機能(seqFlag)を追加します 319 * @og.rev 4.0.0.0 (2006/11/15) SelectionFactory に lang 属性を追加します。 320 * @og.rev 4.3.4.0 (2008/12/01) $Cのカラム名置換えを追加 321 * @og.rev 5.1.3.0 (2010/02/01) 一覧表示のみで、ツールチップ表示を行う。 322 * @og.rev 6.0.4.0 (2014/11/28) useMultiSelect は、selection ではなく、colomn から取得する。 323 * @og.rev 6.2.0.0 (2015/02/27) キー:ラベル形式で表示するかどうかを、指定できるようにします。 324 * 325 * @param buf タグ文字列のバッファー 326 * @param value 選択されている値 327 * @param useSlbl ラベル(短)をベースとしたオプション表示を行うかどうか。 328 * 329 * @return オプションタグ 330 * @og.rtnNotNull 331 */ 332 private TagBuffer getOption( final TagBuffer buf,final String value,final boolean useSlbl ) { 333 334 // StringFormat format = new StringFormat( query,value); 335 final StringFormat format = new StringFormat( query, value, name ); // 4.3.4.0 (2008/12/01) 336 final String newQuery = format.format(); 337 final String newValue = format.getValue(); 338 339 // 6.2.0.0 (2015/02/27) キー:ラベル形式 340 final Selection selection = SelectionFactory.newDBSelection( newQuery, dbid, lang, addKeyLabel ); 341 // 6.0.4.0 (2014/11/28) useMultiSelect は、selection ではなく、colomn から取得する。 342 343 // 6.1.1.0 (2015/01/17) TagBufferの連結記述 344 return buf.addBody( Selection.NO_VALUE_OPTION , addNoValue ) // 5.5.1.0 (2012/04/03) 345 .addBody( selection.getOption( newValue, seqFlag, useSlbl ) ); 346 } 347}