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.view;
017
018import org.opengion.hayabusa.common.HybsSystem;
019import org.opengion.hayabusa.db.DBTableModel;
020import org.opengion.hayabusa.html.AbstractViewForm;
021import org.opengion.hayabusa.html.TableFormatter;                       // 6.3.9.0 (2015/11/06)
022import org.opengion.fukurou.model.Formatter;
023import org.opengion.fukurou.util.StringUtil;
024
025/**
026 * 検索結果を自動的に表形式に変換する、テーブル作成クラスです。
027 *
028 * AbstractViewForm により、setter/getterメソッドのデフォルト実装を提供しています。
029 * 各HTMLのタグに必要な setter/getterメソッドのみ、追加定義しています。
030 *
031 * AbstractViewForm を継承している為、ロケールに応じたラベルを出力させる事が出来ます。
032 *
033 * @og.group 画面表示
034 *
035 * @version  4.0
036 * @author       Kazuhiko Hasegawa
037 * @since    JDK5.0,
038 */
039public class ViewForm_HTMLTable extends AbstractViewForm {
040        /** このプログラムのVERSION文字列を設定します。   {@value} */
041        private static final String VERSION = "8.5.6.1 (2024/03/29)" ;
042
043        /** 3.6.0.0 (2004/09/17) チェック済みの行の先頭に、フォーカスを当てる処理 */
044        private static final String LAYER_FOCUS = "<a href=\"#top\" name=\"h_fcs\" id=\"h_fcs\" ></a>" ;
045
046        // 6.4.4.1 (2016/03/18) static final 定数化します。
047        // 7.0.4.0 (2019/05/31) colgroup 廃止
048//      protected static final String NUMBER_DISPLAY = "<colgroup class=\"X\" ><!-- --></colgroup><colgroup class=\"BIT\" ><!-- --></colgroup><colgroup class=\"S9\" ><!-- --></colgroup>" + CR ;
049
050        // 5.2.3.0 (2010/12/01) テーブル罫線対応
051//      private final String tableParam         = HybsSystem.sys( "HTML_TABLE_PARAM" ) ;        // 6.9.5.0 (2018/04/23) 廃止
052        private final boolean useRowId          = HybsSystem.sysBool( "USE_CHECKBOX_ROW_ID" ) ; // 3.6.0.0 (2004/09/17)
053
054//      // 6.1.2.0 (2015/01/24) HTML5 で colgroup が効かない対応で、protected化します。
055//      // 6.9.5.0 (2018/04/23) USE_IE7_HEADER 廃止(false固定)
056//      protected final boolean useIE7Header    = HybsSystem.sysBool( "USE_IE7_HEADER" );               // 5.7.5.0 (2014/04/04) HTML5 で colgroup が効かない対応
057
058        /** ヘッダー部分のキャッシュ変数 */
059        protected String        headerLine      ;               // 3.5.2.0 (2003/10/20)
060
061        private String  cacheTag                ;               // 3.5.5.5 (2004/04/23)
062        private int[]   popupClmNo              ;               // 3.8.6.1 (2006/10/20)
063
064        /**
065         * デフォルトコンストラクター
066         *
067         * @og.rev 6.4.2.0 (2016/01/29) PMD refactoring. Each class should declare at least one constructor.
068         */
069        public ViewForm_HTMLTable() { super(); }                // これも、自動的に呼ばれるが、空のメソッドを作成すると警告されるので、明示的にしておきます。
070
071        /**
072         * 内容をクリア(初期化)します。
073         *
074         * @og.rev 3.1.1.0 (2003/03/28) 同期メソッド(synchronized付き)を非同期に変更する。
075         * @og.rev 3.5.2.0 (2003/10/20) ヘッダー繰り返し属性( headerSkipCount )を採用
076         * @og.rev 3.5.5.5 (2004/04/23) cacheTag,isCache 属性追加(チェックボックス作成用)
077         * @og.rev 3.5.6.2 (2004/07/05) bgColorCycle 変数の削除。
078         * @og.rev 3.8.6.1 (2006/10/20) popup の値を返す為のカラム番号配列(popupClmNo)追加
079         * @og.rev 5.1.6.0 (2010/05/01) caption 属性が使われていないため、削除する。
080         */
081        @Override
082        public void clear() {
083                super.clear();
084                cacheTag        = null;         // 3.5.5.5 (2004/04/23)
085                popupClmNo      = null;         // 3.8.6.1 (2006/10/20)
086        }
087
088        /**
089         * DBTableModel から HTML文字列を作成して返します。
090         * startNo(表示開始位置)から、pageSize(表示件数)までのView文字列を作成します。
091         * 表示残りデータが pageSize 以下の場合は、残りのデータをすべて出力します。
092         *
093         * @og.rev 3.5.2.0 (2003/10/20) ヘッダー繰り返し属性( headerSkipCount )を採用
094         * @og.rev 3.5.3.1 (2003/10/31) skip属性を採用。headerLine のキャッシュクリア
095         * @og.rev 3.5.5.0 (2004/03/12) No 欄そのものの作成判断ロジックを追加
096         * @og.rev 3.5.6.4 (2004/07/16) ヘッダーとボディー部をJavaScriptで分離
097         * @og.rev 3.7.0.3 (2005/03/01) getBgColorCycleClass に、選択行マーカーを採用
098         * @og.rev 4.3.1.0 (2008/09/08) 編集行のみを表示する属性(isSkipNoEdit)追加
099         * @og.rev 4.3.3.0 (2008/10/01) noTransition属性対応
100         * @og.rev 6.8.1.1 (2017/07/22) ckboxTD変数は、&lt;td&gt; から &lt;td に変更します(タグの最後が記述されていない状態でもらう)。
101         * @og.rev 8.0.1.0 (2021/11/02) tdClassClms,tdClass追加
102         *
103         * @param  startNo        表示開始位置
104         * @param  pageSize   表示件数
105         *
106         * @return      DBTableModelから作成された HTML文字列
107         * @og.rtnNotNull
108         */
109        @Override       // ViewForm
110        public String create( final int startNo, final int pageSize )  {
111                if( getRowCount() == 0 ) { return ""; } // 暫定処置
112
113                headerLine       = null;
114                final int lastNo = getLastNo( startNo, pageSize );
115                final int blc = getBackLinkCount();
116                final int hsc = getHeaderSkipCount();           // 3.5.2.0 (2003/10/20)
117                int hscCnt = 1;                                                         // 3.5.2.0 (2003/10/20)
118
119                final StringBuilder out = new StringBuilder( BUFFER_LARGE )
120                                .append( getCountForm( startNo,pageSize ) )
121                                .append( getHeader() );
122
123                final String ckboxTD = "  <td";                         // 6.8.1.1 (2017/07/22)
124
125                out.append("<tbody>").append( CR );
126                int bgClrCnt = 0;
127                final int clmCnt = getColumnCount();            // 3.5.5.7 (2004/05/10)
128                for( int row=startNo; row<lastNo; row++ ) {
129                        if( isSkip( row ) || isSkipNoEdit( row ) ) { continue; } // 4.3.1.0 (2008/09/08)
130                        out.append(" <tr").append( getBgColorCycleClass( bgClrCnt++,row ) );
131                        if( isNoTransition() ) {                        // 4.3.3.0 (2008/10/01)
132                                out.append( getHiddenRowValue( row ) );
133                        }
134                        out.append('>').append( CR );           // 6.0.2.5 (2014/10/31) char を append する。
135                        // 3.5.5.0 (2004/03/12) No 欄そのものの作成判断追加
136                        if( isNumberDisplay() ) {
137                                out.append( makeCheckbox( ckboxTD, row, blc ) ).append( CR );
138                        }
139                        for( int column=0; column<clmCnt; column++ ) {
140                                if( isColumnDisplay( column ) ) {
141                                        // 8.0.1.0 (2021/11/02) tdClassClms,tdClass追加
142                                        out.append("  <td" ).append( getTdClass( column ) ).append( '>' )
143                //                      out.append("  <td>")
144                                                .append( getValueLabel(row,column) )
145                                                .append("</td>").append( CR );
146                                }
147                        }
148                        out.append(" </tr>").append( CR );
149
150                // 3.5.2.0 (2003/10/20) ヘッダー繰り返し属性( headerSkipCount )を採用
151                        if( hsc > 0 && hscCnt % hsc == 0 ) {
152                                out.append( getHeadLine() );
153                                hscCnt = 1;
154                        }
155                        else {
156                                hscCnt ++ ;
157                        }
158                }
159                out.append("</tbody>").append( CR )
160                        .append("</table>").append( CR )
161                        .append( getScrollBarEndDiv() );        // 3.8.0.3 (2005/07/15)
162
163                return out.toString();
164        }
165
166        /**
167         * 選択用のチェックボックスと行番号と変更タイプ(A,C,D)を表示します。
168         *
169         * 行番号の個別に、class="S9" を入れる処理を行うための新しいメソッドを作成しました。
170         * 互換性を保つためのメソッドを追加します。
171         *
172         * @og.rev 6.8.1.1 (2017/07/22) 互換性を保つためのメソッド
173         *
174         * @param       ckboxTD チェックボックスのタグ(マルチカラム時のrowspan対応)
175         * @param       row      行番号
176         * @param       blc      バックラインカウント(先頭へ戻るリンク間隔)
177         *
178         * @return      tdタグで囲まれたチェックボックスのHTML文字列
179         * @og.rtnNotNull
180         */
181        protected String makeCheckbox( final String ckboxTD,final int row,final int blc ) {
182                return makeCheckbox( ckboxTD, row, blc, false );
183        }
184
185        /**
186         * 選択用のチェックボックスと行番号と変更タイプ(A,C,D)を表示します。
187         *
188         * フォーマット系のViewに、makeNthChild を適用すると、思わぬ所で、副作用が発生します。
189         * その場合は、データ量は増えますが、個々の行番号欄に、class="S9" を入れていくのが
190         * もっとも確実な方法です。その切替を、useS9 変数で行います。
191         *
192         * @og.rev 3.5.1.0 (2003/10/03) Noカラムに、numberType 属性を追加
193         * @og.rev 3.5.4.0 (2003/11/25) Formatter をローカル変数に変更
194         * @og.rev 3.5.4.1 (2003/12/01) table オブジェクトのセット廃止
195         * @og.rev 3.5.5.5 (2004/04/23) Attributes オブジェクトのセット廃止
196         * @og.rev 3.6.0.0 (2004/09/17) ガントチャートの移動時にチェックするためのIDを追加
197         * @og.rev 3.6.0.0 (2004/09/17) チェック済みの行の先頭に、フォーカスを当てる処理を追加
198         * @og.rev 3.8.6.1 (2006/10/20) popup 用の値を設定する機能を追加
199         * @og.rev 6.4.3.4 (2016/03/11) Formatterに新しいコンストラクターを追加する。
200         * @og.rev 6.8.1.1 (2017/07/22) ckboxTD変数は、&lt;td&gt; から &lt;td に変更します(タグの最後が記述されていない状態でもらう)。
201         * @og.rev 6.8.1.1 (2017/07/22) 行番号のtdに、個別に class="S9" を追加するための useS9 変数を追加。
202         * @og.rev 7.0.1.0 (2018/10/15) XHTML → HTML5 対応(空要素の、"/>" 止めを、">" に変更します)。
203         *
204         * @param       ckboxTD チェックボックスのタグ(マルチカラム時のrowspan対応)
205         * @param       row      行番号
206         * @param       blc      バックラインカウント(先頭へ戻るリンク間隔)
207         * @param       useS9    trueにすると、行番号のtdに、class="S9" を個別に入れます。
208         *
209         * @return      tdタグで囲まれたチェックボックスのHTML文字列
210         * @og.rtnNotNull
211         */
212        protected String makeCheckbox( final String ckboxTD,final int row,final int blc,final boolean useS9 ) {
213
214                final DBTableModel table = getDBTableModel() ;
215
216                // 6.3.9.0 (2015/11/06) Found 'DD'-anomaly for variable(PMD)
217                final String opAtt;
218                final String fmt = getOptionTypeAttributes();
219                if( fmt == null ) {
220                        opAtt = null;                   // 6.3.9.0 (2015/11/06)
221                }
222                else {
223                        final Formatter format = new Formatter( table,fmt );    // 6.4.3.4 (2016/03/11)
224                        opAtt = format.getFormatString( row );
225                }
226
227                final StringBuilder out = new StringBuilder( BUFFER_MIDDLE )
228                                        .append( ckboxTD ).append( '>' )                                // 6.8.1.1 (2017/07/22)
229                                        .append( table.getModifyType( row ) )
230                                        .append("</td>")
231                                        .append( ckboxTD ).append( '>' );                               // 6.8.1.1 (2017/07/22)
232
233                // 3.5.5.5 (2004/04/23) Attributes オブジェクトのセット廃止
234                final String inputTag = makeChboxTag();
235                if( isWritable( row ) && inputTag != null ) {
236                        out.append( inputTag );
237                        if( opAtt != null ) { out.append( opAtt ); }
238                        if( isChecked( row ) ) {
239                                out.append( " checked=\"checked\"" );
240                        }
241                        if( useRowId ) {        // 3.6.0.0 (2004/09/17)
242                                out.append( " id=\"" )
243                                        .append( HybsSystem.ROW_ID_KEY )
244                                        .append( row ).append( '"' );           // 6.0.2.5 (2014/10/31) char を append する。
245                        }
246                        if( popupClmNo != null ) {                                      // 3.8.6.1 (2006/10/20)
247                                makePopupReturn( out,row );                             // 内部で StringBuilder に append しています。
248                        }
249//                      out.append( " value=\"" ).append( row ).append( "\" />" );
250                        out.append( " value=\"" ).append( row ).append( "\" >" );               // 7.0.1.0 (2018/10/15)
251                }
252                out.append("</td>" );
253                // 6.8.1.1 (2017/07/22) useS9 変数を追加
254                if( useS9 ) {
255                        final int ad = ckboxTD.indexOf( "class=" );             // class がすでに登録済みの場合
256                        if( ad >= 0 ) {
257                                out.append( ckboxTD.substring( 0,ad+7 ) ).append( "S9 " )       // +7 は、( class=" ) の文字数
258                                        .append( ckboxTD.substring( ad+7 ) ).append( '>' );
259                        }
260                        else {
261                                out.append( ckboxTD ).append( " class=\"S9\">");
262                        }
263                }
264                else {
265                        out.append( ckboxTD ).append( '>' );
266                }
267
268                // 3.5.1.0 (2003/10/03) Noカラムに、numberType 属性を追加
269                if( blc != 0 && (row+1) % blc == 0 ) {
270                        out.append( "<a href=\"#top\">" ).append( getNumberData( row ) ).append( "</a>");
271                } else {
272                        out.append( getNumberData( row ) );
273                }
274
275                // 4.3.5.3 (2008/02/22) Focus2のアンカー位置をtdタグ内に変更
276                if( isFirstChecked( row ) ) {
277                        out.append( LAYER_FOCUS );
278                }
279                out.append("</td>");
280                // if( isFirstChecked( row ) ) {
281                //      out.insert( 0,LAYER_FOCUS );
282                // }
283
284                return out.toString();
285        }
286
287        /**
288         * DBTableModel から テーブルのヘッダータグ文字列を作成して返します。
289         *
290         * @og.rev 3.5.2.0 (2003/10/20) ヘッダーそのもののキャッシュはしない。
291         *
292         * @return      テーブルのヘッダータグ文字列
293         * @og.rtnNotNull
294         */
295        protected String getHeader() {
296                return getTableTag() + getTableHead() ;
297        }
298
299        /**
300         * DBTableModel から テーブルのタグ文字列を作成して返します。
301         *
302         * @og.rev 3.5.0.0 (2003/09/17) &lt;tr&gt;属性は、元のフォーマットのまま使用します。
303         * @og.rev 3.5.1.0 (2003/10/03) Noカラムに、numberType 属性を追加
304         * @og.rev 3.5.4.0 (2003/11/25) TableFormatter クラスを使用するように変更。
305         * @og.rev 3.5.4.7 (2004/02/06) ヘッダーにソート機能用のリンクを追加します。
306         * @og.rev 3.5.5.0 (2004/03/12) systemFormat(例:[KEY.カラム名]形式等)の対応
307         * @og.rev 3.5.5.0 (2004/03/12) No 欄そのものの作成判断ロジックを追加
308         * @og.rev 6.2.0.0 (2015/02/27) フォーマット系の noDisplay 対応
309         * @og.rev 6.3.9.0 (2015/11/06) 引数にTableFormatterを渡して、処理の共有化を図る。
310         *
311         * @param       footerFormat TableFormatterオブジェクト
312         * @return      テーブルのタグ文字列
313         * @og.rtnNotNull
314         */
315        protected String getTableFoot( final TableFormatter footerFormat ) {
316                footerFormat.makeFormat( getDBTableModel() );
317                // 6.2.0.0 (2015/02/27) フォーマット系の noDisplay 対応
318                setFormatNoDisplay( footerFormat );
319
320                final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE )
321                        .append("<tfoot>").append( CR )
322                        .append( footerFormat.getTrTag() ).append( CR );
323
324                // 3.5.5.0 (2004/03/12) No 欄そのものの作成判断追加
325                if( isNumberDisplay() ) {
326                        buf.append("<th colspan=\"3\"")
327                                .append( footerFormat.getRowspan() )
328                                .append("></th>");
329                }
330
331                int cl = 0;
332                for( ; cl<footerFormat.getLocationSize(); cl++ ) {
333                        final int loc = footerFormat.getLocation(cl);
334                        if( loc >= 0 ) { buf.append( getSortedColumnLabel(loc) ); }
335                }
336                buf.append( footerFormat.getFormat(cl) ).append( CR )
337                        .append("</tfoot>").append( CR );
338
339                return buf.toString();
340        }
341
342        /**
343         * DBTableModel から テーブルのタグ文字列を作成して返します。
344         *
345         * @og.rev 3.5.6.4 (2004/07/16) ヘッダーとボディー部をJavaScriptで分離
346         * @og.rev 3.6.0.0 (2004/09/17) ヘッダー固定スクロールの簡素化(スクロールバーを右に出す)
347         * @og.rev 3.6.0.5 (2004/10/18) 印刷時の罫線出力関連機能の追加。id 属性を出力します。
348         * @og.rev 4.0.0.0 (2005/08/31) テーブル表示の CSSファイル利用の有無
349         * @og.rev 5.1.6.0 (2010/05/01) caption 属性が使われていないため、削除する。
350         * @og.rev 5.2.3.0 (2010/12/01) テーブル罫線対応
351         * @og.rev 5.3.4.0 (2011/04/01) テーブル罫線の初期値チェック変更
352         * @og.rev 6.9.5.0 (2018/04/23) tableParam 廃止
353         *
354         * @return      テーブルのタグ文字列
355         * @og.rtnNotNull
356         */
357        protected String getTableTag() {
358                return new StringBuilder( BUFFER_MIDDLE )
359                                        .append( getScrollBarStartDiv() )                                       // 3.8.0.3 (2005/07/15)
360                                        .append( "<table id=\"viewTable\" class=\"" )
361                                        .append( getTableClass() )
362                                        .append( "\">" ).append( CR )
363                                        .toString();
364
365//              // 6.9.7.0 (2018/05/14) 整理します。
366//              final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE )
367//                      .append( getScrollBarStartDiv() )                                       // 3.8.0.3 (2005/07/15)
368//                      .append("<table ");
369//              // 5.2.3.0 (2010/12/01) テーブル罫線対応
370////            if( tableParam != null && !tableParam.isEmpty() ) {             // 6.9.5.0 (2018/04/23) tableParam 廃止
371////                    buf.append( tableParam ).append( ' ' );                         // 6.0.2.5 (2014/10/31) char を append する。
372////            }
373//              buf.append( "id=\"viewTable\" class=\"" )
374//                      .append( getTableClass() )
375//                      .append( "\">" ).append( CR );
376//
377//              return buf.toString();
378        }
379
380        /**
381         * HTML5 で colgroup が効かない暫定対応の、処理をまとめるメソッド
382         *
383         * #viewTable td:nth-child(n) { text-align:right; } CSSを出力するメソッドをまとめました。
384         * 呼び出し元の、getTableHead() のコードをすっきりさせるためのメソッドです。
385         *
386         * @og.rev 5.7.5.0 (2014/04/04) 新規追加
387         * @og.rev 6.1.2.0 (2015/01/24) 他のクラスでも使用する為、protected化します。
388         * @og.rev 5.9.5.3 (2016/02/26) viewTableにtableIdのクラスを追加する対応
389         * @og.rev 6.4.4.1 (2016/03/18) StringBuilderのappend部分の見直し。
390         * @og.rev 6.4.4.2 (2016/04/01) className に BIT が指定された場合の対応。
391         * @og.rev 6.4.6.1 (2016/06/03) tableId 廃止(利用目的を明確にするため、パラメータ名をviewClassに変更)
392         * @og.rev 6.4.6.1 (2016/06/03) DbType とClassName が複雑化しているため、とりあえずの暫定対策。
393         *
394         * @param buf           タグ文字列をappendしていくStringBuilder
395         * @param ad            TDタグの番号 nth-child(n)設置する場合の n の値
396         * @param dbType        text-alignを決めるためのDBTYPE
397         *                  (R,S9,X9→text-align:right; , BIT→text-align:center; , left)
398         *
399         * @return      テーブルのタグ文字列(入力と同じStringBuilder)
400         * @og.rtnNotNull
401         * @see         #getTableHead()
402         */
403        protected StringBuilder makeNthChild( final StringBuilder buf , final int ad , final String dbType ) {
404                // 6.4.6.1 (2016/06/03) DbType とClassName が複雑化しているため、とりあえずの暫定対策。
405                // 従来の dbType には、R,S9,X9 などが、単独で来ます。
406                // BIT は、ClassName に含まれており、R,S9,X9 も、スペース連結で含まれます。
407                // 先に、BIT判定を行います。
408                int hantei = 0;                 // 0:は素通りする。
409                if( dbType != null ) {  // たぶん、不要
410                        if( dbType.contains( "BIT" ) ) { hantei = 2; }
411                        else {
412                                final String type = " " + dbType + " " ;
413                                if( type.contains( " R " ) || type.contains( " S9 " ) || type.contains( " X9 " ) ) { hantei = 1; }
414                        }
415                }
416
417                if( hantei == 1 ) {                             // 6.4.6.1 (2016/06/03)
418                        buf.append( "  #viewTable." ).append( getViewClass() ).append( " td:nth-child(" ).append( ad )
419                                .append( ") { text-align:right; }" ).append( CR );
420                }
421                else if( hantei == 2 ) {                // 6.4.6.1 (2016/06/03)
422                        buf.append( "  #viewTable." ).append( getViewClass() ).append( " td:nth-child(" ).append( ad )
423                                .append( ") { text-align:center; }" ).append( CR );
424                }
425                // 上記(R,S9,X9,BIT)以外のdbTypeの場合は、素通りします。(入力のStringBuilderをそのまま返すだけ)
426
427                return buf ;
428        }
429
430        /**
431         * DBTableModel から テーブルのタグ文字列を作成して返します。
432         *
433         * @og.rev 3.5.1.0 (2003/10/03) Noカラムに、numberType 属性を追加
434         * @og.rev 3.5.2.0 (2003/10/20) ヘッダー繰り返し部をgetHeadLine()へ移動
435         * @og.rev 3.5.3.1 (2003/10/31) VERCHAR2 を VARCHAR2 に修正。
436         * @og.rev 3.5.5.0 (2004/03/12) No 欄そのものの作成判断ロジックを追加
437         * @og.rev 3.5.6.5 (2004/08/09) thead に、id="header" を追加
438         * @og.rev 4.0.0.0 (2005/01/31) DBColumn の 属性(CLS_NM)から、DBTYPEに変更
439         * @og.rev 4.0.0.0 (2005/01/31) 新規作成(getColumnClassName ⇒ getColumnDbType)
440         * @og.rev 5.7.5.0 (2014/04/04) HTML5 で colgroup が効かない暫定対応(真の原因不明)
441         * @og.rev 5.9.1.2 (2015/10/23) 自己終了警告対応
442         * @og.rev 6.4.4.1 (2016/03/18) NUMBER_DISPLAYを、static final 定数化します。
443         * @og.rev 6.4.5.0 (2016/04/08) メソッド変更( getColumnDbType(int) → getClassName(int) )
444         * @og.rev 6.4.6.1 (2016/06/03) DbType とClassName が複雑化しているため、とりあえずの暫定対策。
445         * @og.rev 6.8.1.0 (2017/07/14) HTML5対応ヘッダー出力設定時に、ブラウザを互換設定したときの対応。
446         * @og.rev 6.9.5.0 (2018/04/23) USE_IE7_HEADER 廃止(false固定)
447         * @og.rev 7.0.4.0 (2019/05/31) colgroup 廃止
448         * @og.rev 8.5.6.1 (2024/03/29) thead に、固定の id と class 属性を共通に定義します。
449         *
450         * @return      テーブルのタグ文字列
451         * @og.rtnNotNull
452         */
453        protected String getTableHead() {
454                final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE );
455
456                // 5.7.5.0 (2014/04/04) HTML5 で colgroup が効かない対応
457                // 本当は、tableタグの前に入れたかったが、ややこしいので table タグの直後に入れます。
458                // 互換モードでない場合専用。処理速度を気にするより、処理を一か所にまとめておきます。
459                // 6.9.5.0 (2018/04/23) USE_IE7_HEADER 廃止(false固定)
460//              if( !useIE7Header ) {
461                        // 8.1.0.0 (2021/12/28) HTML5 準拠に見直し(type="text/css" 不要)
462//                      buf.append( "<style type=\"text/css\">" ).append( CR );
463                        buf.append( "<style>" ).append( CR );
464                        // 6.3.9.0 (2015/11/06) Found 'DD'-anomaly for variable(PMD)
465                        int ad = 1;
466        //              int ad ;
467                        if( isNumberDisplay() ) {
468                                makeNthChild( buf,2,"BIT" );
469                                makeNthChild( buf,3,"S9"  );
470                                ad = 4;
471                        }
472        //              else {
473        //                      ad = 1;         // 6.3.9.0 (2015/11/06)
474        //              }
475
476                        final int clmCnt = getColumnCount();
477                        for( int column=0; column<clmCnt; column++ ) {
478                                if( isColumnDisplay( column ) ) {
479                                        // 6.4.6.1 (2016/06/03) DbType とClassName が複雑化しているため、とりあえずの暫定対策。
480                                        makeNthChild( buf,ad,getClassName(column) );                    // 6.4.6.1 (2016/06/03)
481                                        ad++ ;                  // tdタグの順番なので、表示する場合のみカウントする。
482                                }
483                        }
484                        buf.append( "</style>" ).append( CR );
485//              }
486
487//              // 3.5.5.0 (2004/03/12) No 欄そのものの作成判断追加
488//              // 7.0.4.0 (2019/05/31) colgroup 廃止
489//              if( isNumberDisplay() ) {
490//                      // 5.9.1.2 (2015/10/23) 自己終了警告対応
491//                      // 6.4.4.1 (2016/03/18) NUMBER_DISPLAYを、static final 定数化します。
492//                      buf.append( NUMBER_DISPLAY );                                   // 6.8.1.0 (2017/07/14) HTML5ネイティブ時でも、出力します。
493//              }
494
495//              final int clmCnt = getColumnCount();                            // 3.5.5.7 (2004/05/10) 6.9.5.0 (2018/04/23) useIE7Header の if を削除したため、宣言不要
496                // 7.0.4.0 (2019/05/31) colgroup 廃止
497//              for( int column=0; column<clmCnt; column++ ) {
498//                      if( isColumnDisplay( column ) ) {
499//                              buf.append("<colgroup class=\"" )
500//                                      .append( getClassName(column) )                 // 6.4.5.0 (2016/04/08)
501//                                      // 5.9.1.2 (2015/10/23) 自己終了警告対応
502//                                      .append("\"><!-- --></colgroup>")
503//                                      .append( CR );
504//                      }
505//              }
506
507        // 3.5.2.0 (2003/10/20) ヘッダー繰り返し部をgetHeadLine()へ移動
508//              buf.append("<thead id=\"header\">").append( CR )        // 3.5.6.5 (2004/08/09)
509                buf.append( THEAD_TAG )                                                         // 8.5.6.1 (2024/03/29)
510                        .append( getHeadLine() )
511                        .append("</thead>").append( CR );
512
513                return buf.toString();
514        }
515
516        /**
517         * ヘッダー繰り返し部を、getTableHead()メソッドから分離。
518         *
519         * @og.rev 3.5.4.5 (2004/01/23) 実装をgetHeadLine( String thTag )に移動
520         * @og.rev 6.1.2.0 (2015/01/24) キャッシュを返すのを、#getHeadLine() に移動。
521         *
522         * @return      テーブルのタグ文字列
523         * @og.rtnNotNull
524         */
525        protected String getHeadLine() {
526                if( headerLine == null ) {                                      // キャッシュになければ、設定する。
527                        headerLine = getHeadLine( "<th" ) ;
528                }
529
530                return headerLine ;
531        }
532
533        /**
534         * ヘッダー繰り返し部を、getTableHead()メソッドから分離。
535         *
536         * @og.rev 3.5.2.0 (2003/10/20) 新規作成
537         * @og.rev 3.5.4.3 (2004/01/05) useCheckControl 属性の機能を追加
538         * @og.rev 3.5.4.5 (2004/01/23) thタグの属性設定出来る様に新規追加。
539         * @og.rev 3.5.4.6 (2004/01/30) numberType="none" 時の処理を追加(Noラベルを出さない)
540         * @og.rev 3.5.4.7 (2004/02/06) ヘッダーにソート機能用のリンクを追加します。
541         * @og.rev 3.7.0.1 (2005/01/31) 全件チェックコントロール処理変更
542         * @og.rev 6.1.2.0 (2015/01/24) キャッシュを返すのを、#getHeadLine() に移動。
543         * @og.rev 6.9.8.1 (2018/06/11) 読み取り専用(writable="false") の場合のNo欄対応
544         * @og.rev 8.0.1.0 (2021/11/02) tdClassClms,tdClass追加
545         * @og.rev 8.5.6.1 (2024/03/29) thead に、固定の id と class 属性を共通に定義します。
546         *
547         * @param       thTag タグの文字列
548         *
549         * @return      テーブルのタグ文字列
550         * @og.rtnNotNull
551         */
552        protected String getHeadLine( final String thTag ) {
553
554                final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE )
555//                      .append("<tr class=\"row_h\">").append( CR );
556                        .append("<tr>").append( CR );                                                   // 8.5.6.1 (2024/03/29) row_h は、theadタグへ
557
558                // 3.5.5.0 (2004/03/12) No 欄そのものの作成判断追加
559                if( isNumberDisplay() ) {
560                        // 3.5.4.3 (2004/01/05) 追加分
561                        if( isUseCheckControl() && "checkbox".equals( getSelectedType() ) ) {
562                                // 3.5.4.5 (2004/01/23) thタグの属性設定出来る様に変更。
563                                // 6.0.2.5 (2014/10/31) char を append する。
564                                buf.append(  thTag ).append( "></th>" )
565                                        .append( thTag ).append( '>' ).append( getAllCheckControl() ).append( "</th>" )
566                                        .append( thTag ).append( '>' ).append( getNumberHeader()    ).append( "</th>" );        // 3.5.4.6 (2004/01/30)
567                        }
568                        else {
569                                // 3.5.4.5 (2004/01/23) thタグの属性設定出来る様に変更。
570                                // 6.9.8.1 (2018/06/11) 読み取り専用(writable="false") の場合のNo欄対応
571//                              buf.append( thTag ).append( " colspan='3'>" ).append( getNumberHeader() ).append( "</th>" );    // 3.5.4.6 (2004/01/30)
572                                buf.append(  thTag ).append( "></th>" )
573                                        .append( thTag ).append( "></th>" )
574                                        .append( thTag ).append( '>' ).append( getNumberHeader()    ).append( "</th>" );                        // 3.5.4.6 (2004/01/30)
575                        }
576                }
577
578                buf.append( CR );
579                final int clmCnt = getColumnCount();                    // 3.5.5.7 (2004/05/10)
580                for( int column=0; column<clmCnt; column++ ) {
581                        if( isColumnDisplay( column ) ) {
582                                // 3.5.4.5 (2004/01/23) thタグの属性設定出来る様に変更。
583                                // 8.0.1.0 (2021/11/02) tdClassClms,tdClass追加
584                                buf.append( thTag ).append( getTdClass( column ) ).append( '>' )
585        //                      buf.append( thTag ).append('>')                 // 6.0.2.5 (2014/10/31) char を append する。
586                                        .append( getSortedColumnLabel(column) )
587                                        .append("</th>").append( CR );
588                        }
589                }
590                buf.append("</tr>").append( CR );
591
592                return buf.toString();                          // 6.1.2.0 (2015/01/24)
593        }
594
595        /**
596         * フォーマットメソッドを使用できるかどうかを問い合わせます。
597         *
598         * @return  使用可能(true)/ 使用不可能(false)
599         */
600        public boolean canUseFormat() {
601                return false;
602        }
603
604        /**
605         * ogPopup で検索結果の値を返すキーを、CSV形式で指定します。
606         *
607         * popup の検索結果を返す画面で、結果のラジオボタンにイベントセットします。
608         * この場合、オープンもとのwindow に値を返しますが、そのキーをCSV形式で
609         * 指定します。なお、設定は、init 以降(つまり、DBTableModelは設定済み)の
610         * 状態で呼び出してください。(エラーにしません)
611         * なお、このメソッドは、一覧表示(HTMLTable)関係のビューのみでサポートして
612         * いますが、チェックメソッドの関係で、それ以外のビューに適用しても素通り
613         * するようにします。(エラーにしません)
614         *
615         * @og.rev 3.8.6.1 (2006/10/20) 新規追加
616         *
617         * @param  rtnKeys ogPopupで値を返すカラム文字列(CSV形式)
618         */
619        @Override
620        public void setPopupReturnKeys( final String rtnKeys ) {
621                final DBTableModel table = getDBTableModel() ;
622                if( table != null && table.getRowCount() > 0 && rtnKeys != null ) {
623                        final String[] clmNames = StringUtil.csv2Array( rtnKeys );
624                        popupClmNo = new int[clmNames.length];
625                        for( int i=0; i<clmNames.length; i++ ) {
626                                final int no = table.getColumnNo( clmNames[i] );
627                                if( no >= 0 ) { popupClmNo[ i ] = no; }
628                        }
629                }
630        }
631
632        /**
633         * ogPopup で検索結果の値を返すキーを、CSV形式で指定します。
634         *
635         * popup の検索結果を返す画面で、結果のラジオボタンにイベントセットします。
636         * この場合、オープンもとのwindow に値を返しますが、そのキーをCSV形式で
637         * 指定します。なお、設定は、init 以降(つまり、DBTableModelは設定済み)の
638         * 状態で呼び出してください。(エラーにしません)
639         * なお、このメソッドは、一覧表示(HTMLTable)関係のビューのみでサポートして
640         * いますが、チェックメソッドの関係で、それ以外のビューに適用しても素通り
641         * するようにします。(エラーにしません)
642         * rtnPopup に値を渡す場合に、ダブルクオート(")、シングルクオート(')は、
643         * それぞれ、ASCII コード(¥x22、¥x27)に置き換えます。
644         *
645         * @og.rev 3.8.6.1 (2006/10/20) 新規追加
646         *
647         * @param  buf StringBuilder 追加するStringBuilderオブジェクト
648         * @param       rowNo   列番号
649         *
650         * @return      引数にデータを追加した後の同一オブジェクト
651         * @og.rtnNotNull
652         */
653        private StringBuilder makePopupReturn( final StringBuilder buf,final int rowNo ) {
654                int clmNo = popupClmNo[ 0 ];
655                String val = StringUtil.quoteFilter( getValue( rowNo,clmNo ) );
656                buf.append( " onclick=\"rtnPopup(new Array('" ).append( val ).append( '\'' );
657                for( int i=1; i<popupClmNo.length; i++ ) {
658                        clmNo = popupClmNo[ i ];
659                        val = StringUtil.quoteFilter( getValue( rowNo,clmNo ) );
660                        buf.append( ",'" ).append( val ).append( '\'' );
661                }
662                buf.append( "));\"" );
663
664                return buf;
665        }
666
667        /**
668         * 選択用のチェックボックスの input タグを作成します。
669         *
670         * @og.rev 3.5.5.5 (2004/04/23) 新規作成
671         *
672         * @return      チェックボックスのinputタグ
673         */
674        private String makeChboxTag() {
675                // 6.3.9.1 (2015/11/27) A method should have only one exit point, and that should be the last statement in the method.(PMD)
676                if( cacheTag == null ) {
677                        final String type = getSelectedType();          // "checkbox"/"radio"/"hidden"
678
679                        if( type != null ) {
680                                // 8.5.4.2 (2024/01/12) PMD 7.0.0 ConsecutiveLiteralAppends 対応
681                                final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE )
682                                        .append( "<input type=\"" ).append( type )
683                                        .append( "\" name=\"" ).append( HybsSystem.ROW_SEL_KEY ).append( '"' );         // 6.0.2.5 (2014/10/31) char を append する。
684
685                                cacheTag = buf.toString();
686                        }
687                }
688                return cacheTag ;
689        }
690
691        /**
692         * カラムのラベル名(短)を返します。
693         * カラムの項目名に対して、見える形の文字列を返します。
694         * 一般には、リソースバンドルと組合せて、各国ロケール毎にラベルを
695         * 切替えます。
696         *
697         * @og.rev 4.0.0.0 (2005/01/31) 新規追加( shortLabel を返します。)
698         *
699         * @param       column カラム番号
700         *
701         * @return      カラムのラベル名(短)
702         */
703        @Override
704        protected String getColumnLabel( final int column ) {
705                return getDBColumn( column ).getShortLabel();
706        }
707}