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.taglib;
017
018import java.util.Set;                                                                                   // 6.4.3.4 (2016/03/11)
019import java.io.IOException;
020import java.io.ObjectInputStream;
021
022import org.opengion.fukurou.util.Options;
023import org.opengion.fukurou.util.StringUtil;
024import org.opengion.fukurou.util.TagBuffer;
025import org.opengion.fukurou.util.XHTMLTag;
026import org.opengion.fukurou.util.ToString;                                              // 6.1.1.0 (2015/01/17)
027import org.opengion.fukurou.util.ArraySet;                                              // 6.4.3.4 (2016/03/11)
028import org.opengion.hayabusa.common.HybsSystem;
029import org.opengion.hayabusa.common.HybsSystemException;
030
031import static org.opengion.fukurou.util.StringUtil.nval;
032
033/**
034 * プルダウンメニューを作成するHTML拡張タグです。
035 *
036 * プルダウンメニュー内の選択肢は、selectタグの内容であるoptionタグ、
037 * またはqueryOptionタグによって指定します。
038 * optionタグがセットされない場合は、name属性の、コードリソース からoption文字列より、
039 * プルダウンメニューの選択肢を作成します。
040 * name 属性に ラベルリソース のキーを与えることで、 ロケールにあわせたリソースを使用して、
041 * 画面に表示します。ロケールは、ユーザー情報の lang 属性を初期値で使用し、
042 * セットされていない場合は、リクエスト情報のロケールから取得します。
043 * language 属性で定義されている場合は、そちらを優先します。
044 *
045 * @og.formSample
046 * ●形式:<og:select name="…" />
047 * ●body:あり(EVAL_BODY_INCLUDE:BODYをインクルードし、{@XXXX} は解析しません)
048 *
049 * ●Tag定義:
050 *   <og:select
051 *       name             ○【TAG】メニューの名称を指定します(必須)。
052 *       value              【TAG】値を指定します
053 *       defaultVal         【TAG】value属性に値がセットされていないとき使用する、初期値を指定します
054 *       lbl                【TAG】ラベルリソースのラベルIDを指定します
055 *       must               【TAG】必須入力を表す色に変えるかどうか[true/false]を指定します(初期値:false)
056 *       mustAny            【TAG】選択必須入力(どれかひとつ必須)を表す色[true/mustAny/その他]を指定します(初期値:無指定)
057 *       useMustHidden      【TAG】必須の自動チェック用Hiddenを出力するかどうか[true/false]を指定します(初期値:true)
058 *       addNoValue         【TAG】値なしのOptionを含めるかどうか[true/false]を指定します(初期値:false)
059 *       addKeyLabel        【TAG】セレクトメニューの場合、キー:ラベル形式で表示するかどうか[true/false/null]を指定します(初期値:null)
060 *       td                 【TAG】テーブル形式の <td> タグを使用するかどうか[yes/no/false]を指定します(初期値:yes)
061 *       colspan            【TAG】フィールド部分の colspan を指定します
062 *       rowspan            【TAG】ラベル、フィールド共に rowspan を指定します
063 *       optionAttributes   【TAG】JavaScript などの HTML基本タグ以外の属性を、そのままタグとして使用します
064 *       size               【TAG】リストボックスとして表示する場合の表示行数を指定します
065 *       multiple           【TAG】複数選択を可能(multiple)にします(初期値:未設定)
066 *       id                 【HTML】要素に対して固有の名前(id)をつける場合に設定します
067 *       caseKey            【TAG】このタグ自体を利用するかどうかの条件キーを指定します(初期値:null)
068 *       caseVal            【TAG】このタグ自体を利用するかどうかの条件値を指定します(初期値:null)
069 *       caseNN             【TAG】指定の値が、null/ゼロ文字列 でない場合(Not Null=NN)は、このタグは使用されます(初期値:判定しない)
070 *       caseNull           【TAG】指定の値が、null/ゼロ文字列 の場合は、このタグは使用されます(初期値:判定しない)
071 *       caseIf             【TAG】指定の値が、true/TRUE文字列の場合は、このタグは使用されます(初期値:判定しない)
072 *       lang               【HTML】要素の内容と他の属性値の言語(lang,xml:lang)を指定します
073 *       dir                【HTML】文字表記の方向(dir)を指定します
074 *       title              【HTML】要素に対する補足的情報(title)を設定します
075 *       style              【HTML】この要素に対して適用させるスタイルシート(style)を設定します
076 *       disabled           【TAG】その部品に対して、選択や変更が出来ないように(disabled)指定します(サーバーに送信されない)
077 *       tabindex           【HTML】タブの移動順(tabindex)を指定します(0 ~ 32767)
078 *       clazz              【HTML】要素に対して class 属性を設定します
079 *       language           【TAG】タグ内部で使用する言語コード[ja/en/zh/…]を指定します
080 *       onClick            【HTML】JavaScriptのイベント onClick を設定します(例:onClick="renew('query.jsp','QUERY');")
081 *       onChange           【HTML】JavaScriptのイベント onChange を設定します(例:onChange="renew('query.jsp','QUERY');")
082 *       onBlur             【HTML】JavaScriptのイベント onBlur を設定します(例:onBlur="this.value=value.toUpperCase();")
083 *       onFocus            【HTML】JavaScriptのイベント onFocus を設定します
084 *       ondblClick         【HTML】JavaScriptのイベント ondblClick を設定します
085 *       onMouseDown        【HTML】JavaScriptのイベント onMouseDown を設定します
086 *       onMouseUp          【HTML】JavaScriptのイベント onMouseUp を設定します
087 *       onMouseMove        【HTML】JavaScriptのイベント onMouseMove を設定します
088 *       onMouseOut         【HTML】JavaScriptのイベント onMouseOut を設定します
089 *       onMouseOver        【HTML】JavaScriptのイベント onMouseOver を設定します
090 *       onSelect           【HTML】JavaScriptのイベント onSelect を設定します
091 *       onKeydown          【HTML】JavaScriptのイベント onKeydown を設定します
092 *       onKeypress         【HTML】JavaScriptのイベント onKeypress を設定します
093 *       onKeyup            【HTML】JavaScriptのイベント onKeyup を設定します
094 *       autofocus          【HTML5】指定した入力欄にカーソルが当たって自動的にフォーカスされます。
095 *       roles              【TAG】ロールをセットします
096 *       eventColumn        【TAG】イベントカラム(親カラム)を指定します
097 *       eventValue         【TAG】イベントValueのSQL分
098 *       eventURL           【TAG】イベントカラム指定時に呼び出すURL
099 *       eventCallback      【TAG】eventColumn実行後のcallbak関数指定
100 *       debug              【TAG】デバッグ情報を出力するかどうか[true/false]を指定します(初期値:false)
101 *   >   ... Body ...
102 *   </og:select>
103 *
104 * ●使用例
105 *    <og:select name="CDC">                       CodeResource プロパティの値を使用したいときはcode属性を使います。
106 *        <og:option code="CDC" />
107 *    </og:select>
108 *
109 *    <og:select name="CDC">                       選択項目の一番上に項目を付け足すことも可能です。
110 *        <og:option lbl="MSG0001" />              MessageResource プロパティの値を使用したいとききはlbl属性を使います。
111 *        <og:option code="CDC" />
112 *    </og:select>
113 *
114 *    <og:select name="CDC">                       選択項目の一番上に項目を付け足すことも可能です。
115 *        <og:option lbl="CDC1" />                 LabelResource プロパティの値を使用したいとききはlbl属性を使います。
116 *        <og:option code="CDC" />
117 *    </og:select>
118 *
119 *    <og:select name="CDC" multiple="multiple">   選択肢の中から複数選択できるようにするときはmultiple属性を使います。
120 *        <og:option code="CDC" />
121 *    </og:select>
122 *
123 *    <og:select name="CDC" disabled="disabled">   選択不可にするときはdisabled属性を使います。
124 *        <og:option code="CDC" />
125 *    </og:select>
126 *
127 *    <og:select name="CDC">                       選択肢をSELECT文の結果から作成したいときはqueryOptionタグと組み合わせて使います。
128 *        <og:queryOption>
129 *                    select NOSYN,NOSYN,':',NMSYN from DB01 ORDER BY 1
130 *        </og:queryOption>
131 *    </og:select>
132 *
133 *    <og:select name="CDC" eventColumn="OYA" >    親子関係のプルダウンを作る場合
134 *        <og:queryOption>
135 *                    select NOSYN,NOSYN,':',NMSYN from DB01
136 *                     where CLM = "{@OYA}" ORDER BY 1
137 *        </og:queryOption>
138 *    </og:select>
139 *
140 * @og.group 選択データ制御
141 *
142 * @version  4.0
143 * @author       Kazuhiko Hasegawa
144 * @since    JDK5.0,
145 */
146public class SelectTag extends HTMLTagSupport implements OptionAncestorIF {
147        /** このプログラムのVERSION文字列を設定します。   {@value} */
148        private static final String VERSION = "6.8.0.0 (2017/06/02)" ;
149        private static final long serialVersionUID = 680020170602L ;
150
151        // 6.4.3.4 (2016/03/11) String配列 から、Setに置き換えます。
152        private static final Set<String> TD_SET = new ArraySet<>( "yes" , "no" , "false" );
153
154        // 4.0.0 (2005/01/31) HTML_LABEL_SEPARATOR を boolean 変数として取得します。
155        private final String CLM = HybsSystem.sysBool( "HTML_LABEL_SEPARATOR" ) ? ":" : "" ;
156
157        // 4.0.0 (2005/01/31) USE_MULTI_KEY_SELECT を定義しておきます。
158        private final boolean USE_MULTI_KEY_SELECT =  HybsSystem.sysBool( "USE_MULTI_KEY_SELECT" ) ;
159
160        private transient Options option                = new Options();
161        private String  tdFlag          = "yes";        // yes:<td> タグを使用 / false:フィールドのみ / no:ラベルとフィールド
162
163        // 3.0.1.3 (2003/03/11) colspan 属性を追加。
164        private String  colspan     = "";               // 6.1.0.0 (2014/12/26) refactoring
165        private String  rowspan         = "";           // 6.2.0.0 (2015/02/27) rowspan 属性を追加。
166
167        // 3.5.4.0 (2003/11/25) value ,defaultVal  属性を追加。
168        private String  value       ;
169        private String  defaultVal      ;               // value がセットされていないときの初期値
170        private boolean multipleAll     ;               // 3.8.0.9 (2005/10/17) 複数選択可能時に全選択を設定する。
171        private String eventColumn      ;               // 4.3.6.0 (2009/04/01) イベントカラム
172        private String  eventValue      ;               // 6.3.4.0 (2015/08/01) eventValue 追加
173        private String eventURL         ;               // 5.1.9.0 (2010/08/01) 動的プルダウン不具合対応
174
175        private String  rawParam        ;               // 5.1.7.0 (2010/06/01) 動的プルダウン実装見直し
176
177        // 5.1.9.0 (2010/08/01) 値なしのOptionをSelectに含めるかどうか 新規追加
178        private boolean addNoValue  ;           // 互換性のため、初期値は、false になります。
179        private String  addKeyLabel     ;               // 6.0.4.0 (2014/11/28) キー:ラベル形式で表示するかどうか
180
181        /**
182         * デフォルトコンストラクター
183         *
184         * @og.rev 6.4.2.0 (2016/01/29) PMD refactoring. Each class should declare at least one constructor.
185         */
186        public SelectTag() { super(); }         // これも、自動的に呼ばれるが、空のメソッドを作成すると警告されるので、明示的にしておきます。
187
188        /**
189         * Taglibの開始タグが見つかったときに処理する doStartTag() を オーバーライドします。
190         *
191         * @og.rev 5.0.2.0 (2009/11/01) 複数パラメーターの選択に対応
192         * @og.rev 5.1.7.0 (2010/06/01) command=RENEWの場合はdefaultValを無視する。
193         * @og.rev 5.1.9.0 (2010/08/01) addNoValue 対応
194         * @og.rev 5.2.2.0 (2010/11/01) caseKey 、caseVal 属性対応
195         * @og.rev 5.3.7.0 (2011/07/01) 5.1.7.0 (2010/06/01)の対応で判定条件が逆転しているバグを修正
196         * @og.rev 6.3.1.0 (2015/06/28) 個人リソースのUSE_ADD_KEY_LABELで初期値設定
197         *
198         * @return      後続処理の指示( EVAL_BODY_INCLUDE )
199         */
200        @Override
201        public int doStartTag() {
202                // 5.2.2.0 (2010/11/01) caseKey 、caseVal 属性対応
203                if( useTag() ) {
204                        // 6.3.1.0 (2015/06/28) 個人リソースのUSE_ADD_KEY_LABELで初期値設定
205                        if( addKeyLabel == null ) { addKeyLabel = nval( sys( "USE_ADD_KEY_LABEL" ) , null ); }          // 4:個人設定可
206
207                        if( value == null ) {
208        //                      value = nval( getRequestValue( get( "name" ) ),defaultVal );
209                                // 5.1.7.0 (2010/06/01) command=RENEWの場合はdefaultValを無視する。
210                                final String cmd = pageContext.getRequest().getParameter( "command" );
211                                if( "RENEW".equals( cmd ) ) {
212                                        value = StringUtil.array2line( getRequestValues( get( "name" ) ), "|" );
213                                }
214                                else {
215                                        value = nval( StringUtil.array2line( getRequestValues( get( "name" ) ), "|" ),defaultVal );
216                                }
217                        }
218
219                        // 5.1.9.0 (2010/08/01) addNoValue 対応:option を最初にセットする。
220                        if( addNoValue ) {
221                                option.add( "<option value=\"\" ></option>" );
222                        }
223
224                        return EVAL_BODY_INCLUDE ;      // Body インクルード( extends TagSupport 時)
225                }
226                return SKIP_BODY ;                              // Body を評価しない
227        }
228
229        /**
230         * タグリブオブジェクトをリリースします。
231         * キャッシュされて再利用されるので、フィールドの初期設定を行います。
232         *
233         * @og.rev 2.0.0.4 (2002/09/27) カスタムタグの release() メソッドを、追加
234         * @og.rev 2.0.0.8 (2002/10/09) yes/no/false で指定するように変更
235         * @og.rev 3.0.1.3 (2003/03/11) colspan 属性を追加。
236         * @og.rev 3.1.1.2 (2003/04/04) Tomcat4.1 対応。release2() を doEndTag()で呼ぶ。
237         * @og.rev 3.5.4.0 (2003/11/25) value ,defaultVal  属性を追加。
238         * @og.rev 3.8.0.9 (2005/10/17) 複数選択可能時に全選択を設定する。
239         * @og.rev 4.1.2.1 (2008/03/13) must , mustAny 属性を自動化します。
240         * @og.rev 4.3.6.0 (2009/04/01) eventColumn,eventURL追加
241         * @og.rev 5.1.7.0 (2010/06/01) 動的プルダウン実装見直し
242         * @og.rev 5.1.9.0 (2010/08/01) columnタグで動的プルダウンが動作しない不具合に対応
243         * @og.rev 5.1.9.0 (2010/08/01) addNoValue  属性を追加。
244         * @og.rev 5.2.1.0 (2010/10/01) must , mustAny 属性の処理を、HTMLTagSupport に移します。
245         * @og.rev 6.0.4.0 (2014/11/28) addKeyLabel 属性を追加
246         * @og.rev 6.2.0.0 (2015/02/27) rowspan 属性を追加
247         * @og.rev 6.3.4.0 (2015/08/01) eventValue 追加
248         */
249        @Override
250        protected void release2() {
251                super.release2();
252                option          = new Options();
253                tdFlag     = "yes";     // table形式の <td> タグを使用するかどうか
254                colspan         = "";           // 6.1.0.0 (2014/12/26) refactoring
255                rowspan         = "";           // 6.2.0.0 (2015/02/27) rowspan 属性を追加
256                value       = null;
257                defaultVal      = null;         // value がセットされていないときの初期値
258                multipleAll     = false;        // 3.8.0.9 (2005/10/17) 複数選択可能時に全選択を設定する。
259                eventColumn     = null;         // 4.3.6.0 (2009/04/01)
260                eventValue  = null;     // 6.3.4.0 (2015/08/01)
261                eventURL        = null;         // 5.1.9.0 (2010/08/01) columnタグで動的プルダウンが動作しない不具合に対応
262                rawParam        = null;         // 5.1.7.0 (2010/06/01) 動的プルダウン実装見直し
263                addNoValue  = false;    // 5.1.9.0 (2010/08/01) 互換性のため、初期値は、false になります。
264                addKeyLabel = null;             // 6.0.4.0 (2014/11/28) キー:ラベル形式で表示するかどうか
265        }
266
267        /**
268         * 入力用のプルダウンメニューを作成します。
269         *
270         * 表示順の項目名タグは、あらかじめ作成済みの値をここでは受け取ります。
271         * hasSelect 属性の値に応じて,タグの作成方法は変わります。
272         *
273         * @og.rev 2.0.0.8 (2002/10/09) yes/no/false で指定するように変更
274         * @og.rev 2.0.1.0 (2002/10/10) ラベルとフィールドのセパレーターとして、コロン(:)を使用するかどうかを指定できる
275         * @og.rev 3.0.1.3 (2003/03/11) colspan 属性を追加。
276         * @og.rev 3.5.4.0 (2003/11/25) value ,defaultVal  属性を追加。
277         * @og.rev 4.0.0.0 (2005/01/31) マルチセレクションのON/OFFをここで指定。
278         * @og.rev 4.1.2.1 (2008/03/13) must , mustAny 属性を自動化します。
279         * @og.rev 4.3.6.0 (2009/04/01) eventColumn対応
280         * @og.rev 4.3.7.1 (2009/06/08) id=labelのclass化
281         * @og.rev 5.1.7.0 (2010/06/01) 動的プルダウン実装見直し
282         * @og.rev 5.2.1.0 (2010/10/01) must , mustAny 属性の処理を、HTMLTagSupport に移します。
283         * @og.rev 5.5.3.0 (2012/06/01) idのセット見直し
284         * @og.rev 5.6.2.2 (2013/03/15) 自動must処理の出力位置を変更
285         * @og.rev 6.2.0.0 (2015/02/27) rowspan 属性を追加。makeMustHidden(String)対応
286         * @og.rev 6.3.4.0 (2015/08/01) eventValue 追加
287         *
288         * @return      表示順選択用のプルダウンメニュータグ
289         * @og.rtnNotNull
290         */
291        @Override
292        protected String makeTag() {
293
294                final String name = get( "name" );
295                if( getMsglbl() == null ) { setLbl( name ); }
296
297                // 6.1.0.0 (2014/12/26) refactoring
298
299                // 4.0.0 (2005/01/31) XHTMLTag.java より移動)
300                if( USE_MULTI_KEY_SELECT ) {
301                        set( "onkeydown", "setKeySelect(this);" );
302                }
303
304                // 5.5.3.0 既にIDが指定されている場合はnameをセットしない
305                if( get("id")==null || get("id").isEmpty() ){
306                        set( "id", name );
307                }
308
309                // 4.3.6.0 (2009/04/01) eventColumnの対応
310                // 5.1.7.0 (2010/06/01) 動的プルダウン実装見直し
311                String selTag = XHTMLTag.select( getAttributes(), option );
312
313                // 5.1.7.0 (2010/06/01) 動的プルダウン実装見直し
314                if( eventColumn != null && eventColumn.length() > 0 ) {
315                        String editor = null;
316                        if( rawParam != null && rawParam.length() > 0 ) {
317                                editor = "DBMENU";
318                        }
319                        else {
320                                editor = "MENU";
321                        }
322                        addEventColumn( name, eventColumn, eventValue , eventURL, "", editor, "", rawParam );   // 6.3.4.0 (2015/08/01)
323
324                        // 6.1.1.0 (2015/01/17) TagBufferの連結記述
325                        selTag = new TagBuffer( "span" )
326                                                .add( "class"                                                   , HybsSystem.EVENT_COLUMN_CLASS )
327                                                .add( HybsSystem.EVENT_COLUMN_ID                , name )
328                                                .add( HybsSystem.EVENT_COLUMN_INITVAL   , value )
329                                                .add( HybsSystem.EVENT_COLUMN_WRITABLE  , "true" )
330                                                .addBody( selTag )
331                                                .makeTag();
332
333                }
334
335                final StringBuilder rtn = new StringBuilder( BUFFER_MIDDLE );
336                if( "yes".equals( tdFlag ) ) {
337                        // 4.3.7.1 (2009/06/08) id=labelのclass化
338                        // rtn.append( "<td id=\"label\">" );
339                        rtn.append( "<td class=\"label\"" )
340                                .append( rowspan ).append( '>' )                // 6.2.0.0 (2015/02/27)
341                                .append( getLongLabel() )                               // 4.0.0 (2005/01/31)
342                                .append( CLM )
343                                .append( "</td><td" )
344                                .append( colspan )                                              // 6.1.0.0 (2014/12/26) refactoring
345                                .append( rowspan )                                              // 6.2.0.0 (2015/02/27)
346                                .append( '>' )                                                  // 6.0.2.5 (2014/10/31) char を append する。
347                                .append( selTag )
348                                .append( makeMustHidden( name ) )               // 6.2.0.0 (2015/02/27) , 5.6.2.2 (2013/03/15)
349                                .append( "</td>" );
350                }
351                else if( "no".equals( tdFlag ) ) {
352                        // 4.3.7.1 (2009/06/08) id=labelのclass化
353                        // rtn.append( "<span id=\"label\">" );
354                        rtn.append( "<span class=\"label\">" )
355                                .append( getLongLabel() )               // 4.0.0 (2005/01/31)
356                                .append( CLM )
357                                .append( "</span>" )
358                                .append( selTag )
359                                .append( makeMustHidden( name ) );              // 6.2.0.0 (2015/02/27) , 5.6.2.2 (2013/03/15)
360                }
361                // "false".equals( tdFlag ) の条件
362                else {
363                        rtn.append( selTag )
364                                .append( makeMustHidden( name ) );              // 6.2.0.0 (2015/02/27) , 5.6.2.2 (2013/03/15)
365                }
366
367                return rtn.toString() ;
368        }
369
370        /**
371         * メニュー項目の選択項目を追加します。
372         *
373         * select タグのBODY要素の OptionTag よりアクセスされます。
374         *
375         * @og.rev 3.1.2.0 (2003/04/07) taglib パッケージ内部で使用している箇所を protected 化する。
376         * @param       opt      オプションタグ文字列
377         * @see         org.opengion.hayabusa.taglib.OptionAncestorIF#addOption( String )
378         */
379        public void addOption( final String opt ) {
380                option.add( opt );
381        }
382
383        /**
384         * メニュー項目の最後の項目を削除します。
385         *
386         * select タグのBODY要素の OptionTag よりアクセスされます。
387         *
388         * @og.rev 6.8.0.0 (2017/06/02) メニュー項目の最後の項目を削除。
389         * @see         org.opengion.hayabusa.taglib.OptionAncestorIF#removeLast()
390         */
391        public void removeLast() {
392                option.removeLast();
393        }
394
395        /**
396         * 【TAG】メニューの名称を指定します。
397         *
398         * @og.tag メニューの名称を指定します。
399         *
400         * @param       name 名称
401         */
402        public void setName( final String name ) {
403                set( "name", getRequestParameter( name ) );
404        }
405
406        /**
407         * 【TAG】リストボックスとして表示する場合の表示行数を指定します。
408         *
409         * @og.tag 表示行数を指定します。
410         *
411         * @param       size 表示行数
412         */
413        public void setSize( final String size ) {
414                set( "size",getRequestParameter( size ) );
415        }
416
417        /**
418         * 【TAG】複数選択を可能(multiple)にします(初期値:未設定)。
419         *
420         * @og.tag
421         * multiple 値が "multiple" または "true" の場合、複数選択状態になります。
422         * また、"ALL" を設定する場合は、OptionTag上で、すべてのOption が選択状態に
423         * 初期設定されます。
424         * 大文字、小文字の種別はなくしますが、指定以外の文字が設定された場合は、エラーになります。
425         *
426         * @og.rev 3.8.0.9 (2005/10/17) 複数選択可能時に全選択を設定する。
427         *
428         * @param       multi 複数選択指定 [multiple:複数選択/true:複数選択/ALL:初期値全選択]
429         */
430        public void setMultiple( final String multi ) {
431                final String multiple = nval( getRequestParameter( multi ),null );
432                if( "multiple".equalsIgnoreCase( multiple ) || "true".equalsIgnoreCase( multiple ) ) {
433                        set( "multiple","multiple" );
434                }
435                else if( "ALL".equalsIgnoreCase( multiple ) ) {
436                        set( "multiple","multiple" );
437                        multipleAll = true;
438                }
439                else if( ! ( "false".equalsIgnoreCase( multiple ) || multiple == null ) ) {
440                        final String errMsg = "multiple 属性には、[multiple,true,ALL] 以外は設定できません。"
441                                                + "multiple=" + multi ;
442                        throw new HybsSystemException( errMsg );
443                }
444        }
445
446        /**
447         * 【TAG】テーブル形式の &lt;td&gt; タグを使用するかどうか[yes/no/false]を指定します(初期値:yes)。
448         *
449         * @og.tag
450         * 初期値は、使用する("yes") です。
451         *
452         * @og.rev 2.0.0.8 (2002/10/09) yes/no/false で指定するように変更
453         * @og.rev 5.5.1.0 (2012/04/03) エラーメッセージ変更
454         * @og.rev 6.3.4.0 (2015/08/01) Arrays.toString から String.join に置き換え。
455         * @og.rev 6.4.3.4 (2016/03/11) String配列 から、Setに置き換えます。
456         *
457         * @param   flag TDタグ使用 [yes:tdタグを使用/no:ラベルとフィールド/false:フィールドのみ]
458         */
459        public void setTd( final String flag ) {
460                final String td = nval( getRequestParameter( flag ),tdFlag );
461
462                if( check( td , TD_SET ) ) {
463                        tdFlag = td;
464                }
465                else {
466                        final String errMsg = "Error in SelectTag [td]: "
467                                                 + td + "  in [" + String.join( ", " , TD_SET ) + "] only used.";
468                        throw new HybsSystemException( errMsg );
469                }
470        }
471
472        /**
473         * 【TAG】フィールド部分の colspan を指定します。
474         *
475         * @og.tag
476         * 通常は、ラベルとフィールドが、別のtdで囲われます。しかし、場合によっては、
477         * フィールド部の長いカラムがある場合、上下のカラムとの位置関係がおかしくなり、
478         * ラベルとフィールドが離れて表示されるケースがあります。
479         * この場合、長いフィールド部を持つカラムに、colspan 属性を指定すると、
480         * その分のカラムを割り当てることにより、上下のカラムとの位置関係を
481         * 調節することが可能になります。
482         * 通常は、3 を指定して、tdが、フィールド、ラベル、フィールドの
483         * 組み合わせ部分に、長いフィールドが適用されるように使用します。
484         *
485         *  &lt;og:select name="CDK" colspan="3" /&gt;
486         *
487         * @og.rev 3.0.1.3 (2003/03/11) colspan 属性を追加。
488         *
489         * @param   sp フィールド部分のcolspan
490         */
491        public void setColspan( final String sp ) {
492                colspan = nval( getRequestParameter( sp ),colspan );
493                // 6.1.0.0 (2014/12/26) refactoring
494                if( colspan.length() > 0 ) {
495                        colspan = " colspan=\"" + colspan + "\" ";
496                }
497        }
498
499        /**
500         * 【TAG】ラベル、フィールド共に rowspan を指定します。
501         *
502         * @og.tag
503         * 通常は、ラベルとフィールドが、別のtdで囲われます。rowspan は、
504         * そのどちらの TDフィールドにも、設定します。
505         * 初期値は、rowspan を付けないです。
506         *
507         *  &lt;og:select name="CDK" rowspan="3" /&gt;
508         *
509         * @og.rev 6.2.0.0 (2015/02/27) rowspan 属性を追加
510         *
511         * @param   sp ラベル、フィールド部分のrowspan
512         */
513        public void setRowspan( final String sp ) {
514                rowspan = nval( getRequestParameter( sp ),rowspan );
515                if( rowspan.length() > 0 ) {
516                        rowspan = " rowspan=\"" + rowspan + "\" ";
517                }
518        }
519
520        /**
521         * 【TAG】値を指定します。
522         *
523         * @og.tag
524         *   ここで指定された値が、optionタグの値と同じであれば、選択状態になります。
525         *
526         *    &lt;og:select name="CDC" value="{&#064;ABC}" defaultVal="XYZ"  &gt;
527         *         &lt;og:option value="ABC" lbl="CDC1" /&gt;
528         *         &lt;og:option value="XYZ" lbl="CDC2" /&gt;
529         *    &lt;/og:select&gt;
530         *
531         * @og.rev 3.5.4.0 (2003/11/25) 新規作成
532         * @og.rev 5.0.2.0 (2009/11/01) 複数パラメーターの選択に対応
533         *
534         * @param   val 値
535         */
536        public void setValue( final String val ) {
537                value = nval( StringUtil.array2line( getRequestParameterValues( val ), "|" ),value );
538        }
539
540        /**
541         * 値を外部から取り出します。
542         *
543         * OptionTag で、value を取り出して、内部の値と同じ場合は、選択状態にします。
544         *
545         * @og.rev 3.5.4.0 (2003/11/25) 新規作成
546         * @og.rev 5.7.1.0 (2013/12/06) OptionAncestorIF にする為、protected ⇒ public 化する。
547         *
548         * @return      内部に設定された値
549         * @see         org.opengion.hayabusa.taglib.OptionAncestorIF#getValue()
550         */
551        public String getValue() {
552                return value;
553        }
554
555        /**
556         * 【TAG】イベントカラム(親カラム)を設定します。
557         *
558         * @og.tag
559         *   親子関係のカラムを作成する場合に利用します。
560         *   queryOptionタグとセットで利用して下さい。
561         *
562         *   ※詳細はcolumnTagのeventColumn属性の説明をご覧下さい。
563         *
564         * @og.rev 4.3.6.0 (2009/04/01)
565         *
566         * @param   col 親カラム名
567         */
568        public void setEventColumn( final String col ) {
569                eventColumn = nval( getRequestParameter( col ), eventColumn );
570        }
571
572        /**
573         * 【TAG】eventColumn使用時の値を、SQL文から求めるためのqueryをセットします。
574         *
575         * @og.tag
576         * eventColumn利用時は{&#064;XXXX}はリクエストパラメータではなく
577         * xxxカラムの実際の値が利用されます。
578         *
579         * @og.rev 6.3.4.0 (2015/08/01) eventValue 追加
580         * @param       val     パラメータ
581         */
582        public void setEventValue( final String val ) {
583                eventValue = nval( getReservedParameter( val ), eventValue );   // ここでは{@XXXX}変換をしない
584        }
585
586        /**
587         * 【TAG】イベントカラム指定時に呼び出すURLを指定します。
588         *
589         * @og.tag
590         *   イベントカラム指定時に部品を作成するJSPを指定します。
591         *   初期値はシステムリソースのEVENT_COLUMN_URLです。
592         *   (例:eventURL="makeColumn_custom.jsp")
593         *
594         *   ※詳細はcolumnTagのeventURL属性の説明をご覧下さい。
595         *
596         * @og.rev 4.3.6.0 (2009/04/01)
597         *
598         * @param       url     呼び出すURL
599         */
600        public void setEventURL( final String url ) {
601                eventURL = nval( getRequestParameter( url ), eventURL );
602        }
603
604        /**
605         * 複数選択可能時に全選択を設定するかどうかを返します。
606         *
607         * これは、上位入れ子のタグの OptionTag で、multipleAll を取り出して、
608         * true であれば、全選択に設定します。
609         *
610         * @og.rev 3.8.0.9 (2005/10/17) 新規作成
611         * @og.rev 5.7.1.0 (2013/12/06) OptionAncestorIF にする為、protected ⇒ public 化する。
612         *
613         * @return      全選択:true / 通常:false
614         * @see         org.opengion.hayabusa.taglib.OptionAncestorIF#isMultipleAll()
615         */
616        public boolean isMultipleAll() {
617                return multipleAll;
618        }
619
620        /**
621         * 【TAG】value属性に値がセットされていないとき使用する、初期値を指定します。
622         *
623         * @og.tag
624         * value属性に値がセットされていないときに、この初期値を使用します。
625         *
626         * なお、commandがRENEWの場合は、このdefaultValは適用されません。
627         * これは、defaultValの値が埋め込まれた項目が存在する画面に戻った際に、
628         * defaultValの項目の値がnullとなっていた場合に、defaultValの値が復元されると、
629         * 検索結果との不一致が発生するためです。
630         *
631         * @og.rev 3.5.4.0 (2003/11/25) 新規作成
632         * @og.rev 5.0.2.0 (2009/11/01) 複数パラメーターの選択に対応
633         *
634         * @param       dv 初期値
635         */
636        public void setDefaultVal( final String dv ) {
637                defaultVal = nval( StringUtil.array2line( getRequestParameterValues( dv ), "|" ),defaultVal );
638        }
639
640        /**
641         * 【TAG】値なしのOptionを含めるかどうか[true/false]を指定します(初期値:false)。
642         *
643         * @og.tag
644         * カラムがプルダウンメニュー形式の場合、選択肢には通常データベース登録できる候補が
645         * 並びます。しかし、検索時には、候補を絞りたくない(指定しない)場合があり、
646         * その様な場合には、値なしのOptionを含める(true)ことで、対応可能です。
647         * 初期値:false(値なしのOptionを含めない)は、過去の互換性を重視しているため、column タグと異なります。
648         *
649         * @og.rev 5.1.9.0 (2010/08/01) 新規追加
650         *
651         * @param       flag  値なしのOptionの有無 [true:含める/それ以外:含めない]
652         */
653        public void setAddNoValue( final String flag ) {
654                addNoValue = nval( getRequestParameter( flag ),addNoValue );
655        }
656
657        /**
658         * 【TAG】セレクトメニューの場合、キー:ラベル形式で表示するかどうか[true/false/null]を指定します(初期値:null)。
659         *
660         * @og.tag
661         * カラムがプルダウンメニュー形式の場合、キー:ラベル形式で表示するか、ラベルだけで表示するかを
662         * 選択できます。
663         * true を指定すると、「強制的に」キー:ラベル形式で表示します。
664         * false の場合は、「強制的に」ラベルのみで表示されます。
665         * 初期値の null の場合、コードリソースや、SQL文に準拠した指定通りの形式で表示します。
666         *
667         * &lt;og:column name="CDC" addKeyLabel="true" /&gt;
668         *
669         * @og.rev 6.0.4.0 (2014/11/28) キー:ラベル形式で表示するかどうか。新規追加
670         *
671         * @param   flag  キー:ラベル形式表示 [true:キー:ラベル形式/false:ラベルのみ/null:指定通り]
672         */
673        public void setAddKeyLabel( final String flag ) {
674                addKeyLabel = nval( getRequestParameter( flag ),addKeyLabel );
675        }
676
677        /**
678         * セレクトメニューの場合、キー:ラベル形式で表示するかどうか[true/false/null]を返します。
679         *
680         * これは、上位入れ子のタグの OptionTag で、addKeyLabel を取り出して、
681         * true であれば、キー:ラベル形式 のオプションを、#addOption( String ) で
682         * 登録させます。
683         *
684         * @og.rev 6.0.4.0 (2014/11/28) キー:ラベル形式で表示するかどうか。新規追加
685         *
686         * @return      true:キー:ラベル形式/false:ラベルのみ/null:指定通り
687         * @see         #addOption( String )
688         * @see         org.opengion.hayabusa.taglib.OptionAncestorIF#getAddKeyLabel()
689         */
690        public String getAddKeyLabel() {
691                return addKeyLabel;
692        }
693
694        /**
695         * パラメーター変換({&#064;XXXX}の置き換えをしない状態のパラメーターをセットします。
696         *
697         * @og.rev 5.1.7.0 (2010/06/01) 新規作成(動的プルダウン実装見直し)
698         * @og.rev 5.5.4.0 (2012/07/02) 予約語対応
699         *
700         * @param   param パラメーター
701         * @see         org.opengion.hayabusa.taglib.OptionAncestorIF#setRawParam( String )
702         */
703        public void setRawParam( final String param ) {
704                rawParam = nval( getReservedParameter(param), rawParam ); // 5.5.4.0 (2012/07/02)
705        }
706
707        /**
708         * 【TAG】eventColumn実行後のcallbak関数を指定します。
709         *
710         * @og.tag
711         * eventColumnの動作終了後に実行するCallback関数の指定を行います。
712         * 関数名のみを指定して下さい。
713         *  ex) sampleCallback()を実行する場合はeventCallback="sampleCallback"
714         * 実行する関数の第一引数には自動的に次の要素が入った配列が渡されます
715         *  [0] 変更された子カラムのID属性値
716         *  [1] evnetColumn実行後の子カラムの値
717         *  [2] eventColumn実行前の子カラムの値
718         *
719         * この属性は、optionAttributesへの設定と同様の動作を行います。
720         *
721         * @og.rev 5.5.4.0 (2012/07/02) 新規追加
722         *
723         * @param   callback callbak関数
724         */
725        public void setEventCallback( final String callback ) {
726                // optionAttributes扱いで登録します。
727                if( callback != null && callback.length() > 0 ){
728                        add( "optionAttributes", "eventCallback='"+getRequestParameter( callback )+"'" );
729                }
730        }
731
732        /**
733         * シリアライズ用のカスタムシリアライズ読み込みメソッド
734         *
735         * ここでは、transient 宣言された内部変数の内、初期化が必要なフィールドのみ設定します。
736         *
737         * @og.rev 4.0.0.0 (2006/09/31) 新規追加
738         * @serialData 一部のオブジェクトは、シリアライズされません。
739         *
740         * @param       strm    ObjectInputStreamオブジェクト
741         * @see #release2()
742         * @throws IOException  シリアライズに関する入出力エラーが発生した場合
743         * @throws ClassNotFoundException       クラスを見つけることができなかった場合
744         */
745        private void readObject( final ObjectInputStream strm ) throws IOException , ClassNotFoundException {
746                strm.defaultReadObject();
747                option = new Options();
748        }
749
750        /**
751         * このオブジェクトの文字列表現を返します。
752         * 基本的にデバッグ目的に使用します。
753         *
754         * @return このクラスの文字列表現
755         * @og.rtnNotNull
756         */
757        @Override
758        public String toString() {
759                return ToString.title( this.getClass().getName() )
760                                .println( "VERSION"             ,VERSION                )
761                                .println( "tdFlag"              ,tdFlag         )
762                                .println( "colspan"             ,colspan                )
763                                .println( "rowspan"             ,rowspan                )
764                                .println( "value"               ,value                  )
765                                .println( "defaultVal"  ,defaultVal             )
766                                .println( "multipleAll" ,multipleAll    )
767                                .println( "Other..."    ,getAttributes().getAttribute() )
768                                .fixForm().toString() ;
769        }
770}