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 java.util.List;
019// import java.util.regex.Pattern;                                                              // 6.4.3.3 (2016/03/04)
020
021import org.opengion.hayabusa.common.HybsSystemException;
022// import org.opengion.fukurou.util.StringUtil;
023import org.opengion.hayabusa.html.TableFormatter;
024
025/**
026 * ヘッダ、フッタ、ボディを指定して作成する、自由レイアウトが可能な、カスタムテーブル表示クラスです。
027 * このクラスは、ViewForm_HTMLFormatTable クラスの代替えとしても使用できます。
028 * その場合は、thead のみ指定すれば、同じフォームが tbody にも適用されます。
029 * これは、まさに、ViewForm_HTMLFormatTable と同じです。
030 *
031 * AbstractViewForm により、setter/getterメソッドのデフォルト実装を提供しています。
032 * 各HTMLのタグに必要な setter/getterメソッドのみ、追加定義しています。
033 *
034 * AbstractViewForm を継承している為、ロケールに応じたラベルを出力させる事が出来ます。
035 *
036 * @og.group 画面表示
037 *
038 * @version  4.0
039 * @author       Kazuhiko Hasegawa
040 * @since    JDK5.0,
041 */
042public class ViewForm_HTMLCustomTable extends ViewForm_HTMLTable        {
043        /** このプログラムのVERSION文字列を設定します。   {@value} */
044        private static final String VERSION = "8.5.6.1 (2024/03/29)" ;
045
046        // 6.4.3.3 (2016/03/04) class属性(ColumnのDBType)置換で、td属性に[カラム]があると、誤ってそちらがセットされてしまう対応。
047        // ※ 正規表現の「.」は、改行を除く任意の文字なので、改行を含みたい場合は、「[\\s\\S]」とします。
048        // 8.5.6.1 (2024/03/29) AbstractViewForm に移動します。
049//      private static final Pattern TD_BODY = Pattern.compile( "[\\s\\S]*<td[^>]*>[\\s]*" );           // 6.4.4.2 (2016/04/01) tdタグが完了して、BODY部分の解析に入る。
050//      private static final Pattern TD_END  = Pattern.compile( "[^<]*>[^<>]*" );                                       // 何らかのタグが完了して、BODY部分の解析に入る。
051
052        // 3.5.4.0 (2003/11/25) TableFormatter クラス追加
053        // 4.3.1.0 (2008/09/08) protectedに変更
054        /** ヘッダーフォーマット変数 */
055        protected TableFormatter                headerFormat    ;
056        /** ボディーフォーマット配列変数 */
057        protected TableFormatter[]              bodyFormats             ;
058        /** フッターフォーマット変数 */
059        protected TableFormatter                footerFormat    ;
060        /** ボディーフォーマット数 */
061        protected int                                   bodyFormatsCount;
062
063        // 3.5.4.6 (2004/01/30) 初期値変更
064        /** ボディーフォーマット最大数 初期値:{@value} */
065        protected static final int BODYFORMAT_MAX_COUNT = 15;           // 5.7.6.2 (2014/05/16)
066
067        /**
068         * デフォルトコンストラクター
069         *
070         * @og.rev 6.4.2.0 (2016/01/29) PMD refactoring. Each class should declare at least one constructor.
071         */
072        public ViewForm_HTMLCustomTable() { super(); }          // これも、自動的に呼ばれるが、空のメソッドを作成すると警告されるので、明示的にしておきます。
073
074        /**
075         * DBTableModel から HTML文字列を作成して返します。
076         * startNo(表示開始位置)から、pageSize(表示件数)までのView文字列を作成します。
077         * 表示残りデータが pageSize 以下の場合は、残りのデータをすべて出力します。
078         *
079         * @og.rev 3.5.0.0 (2003/09/17) BODY要素の noClass 属性を追加。
080         * @og.rev 3.5.0.0 (2003/09/17) &lt;tr&gt;属性は、元のフォーマットのまま使用します。
081         * @og.rev 3.5.2.0 (2003/10/20) ヘッダー繰り返し属性( headerSkipCount )を採用
082         * @og.rev 3.5.3.1 (2003/10/31) skip属性を採用。headerLine のキャッシュクリア
083         * @og.rev 3.5.4.0 (2003/11/25) TableFormatter クラスを使用するように変更。
084         * @og.rev 3.5.5.0 (2004/03/12) systemFormat(例:[KEY.カラム名]形式等)の対応
085         * @og.rev 3.5.5.0 (2004/03/12) No 欄そのものの作成判断ロジックを追加
086         * @og.rev 3.5.5.7 (2004/05/10) [#カラム名] , [$カラム名] に対応
087         * @og.rev 3.5.6.0 (2004/06/18) '!' 値のみ 追加 既存の '$' は、レンデラー
088         * @og.rev 3.5.6.2 (2004/07/05) makeFormat 処理をgetTableHead メソッドから移動
089         * @og.rev 3.5.6.4 (2004/07/16) ヘッダーとボディー部をJavaScriptで分離
090         * @og.rev 4.0.0.0 (2005/01/31) 新規作成(getColumnClassName ⇒ getColumnDbType)
091         * @og.rev 3.7.0.3 (2005/03/01) getBgColorCycleClass に、選択行マーカーを採用
092         * @og.rev 4.3.1.0 (2008/09/08) フォーマットが設定されていない場合のエラー追加・編集行のみを表示する属性(isSkipNoEdit)追加
093         * @og.rev 4.3.3.0 (2008/10/01) noTransition属性対応
094         * @og.rev 4.3.7.4 (2009/07/01) tbodyタグの入れ子を解消(FireFox対応)
095         * @og.rev 6.2.0.0 (2015/02/27) フォーマット系の noDisplay 対応
096         * @og.rev 6.4.3.3 (2016/03/04) class属性(ColumnのDBType)置換で、td属性に[カラム]があると、誤ってそちらがセットされてしまう対応。
097         * @og.rev 6.4.3.4 (2016/03/11) tdに、[カラム]が無いケースで、次の[カラム]のクラス属性が、前方すべてのtdにセットされてしまう対応。
098         * @og.rev 6.4.4.2 (2016/04/01) TableFormatterのタイプ別値取得処理の共通部をまとめる。
099         * @og.rev 6.4.5.0 (2016/04/08) メソッド変更( getColumnDbType(int) → getClassName(int) )
100         * @og.rev 6.8.1.1 (2017/07/22) ckboxTD変数は、&lt;td&gt; から &lt;td に変更します(タグの最後が記述されていない状態でもらう)。
101         * @og.rev 6.8.1.1 (2017/07/22) 行番号のtdに、個別に class="S9" を追加するための useS9 変数を追加。
102         * @og.rev 8.5.4.2 (2024/01/12) class 属性がフォーマット中に存在する場合、追記になる。
103         * @og.rev 8.5.6.1 (2024/03/29) PMD 7.0.0 Finding duplicated code with CPD
104         *
105         * @param  startNo        表示開始位置
106         * @param  pageSize   表示件数
107         *
108         * @return      DBTableModelから作成された HTML文字列
109         * @og.rtnNotNull
110         */
111        @Override
112        public String create( final int startNo, final int pageSize )  {
113                if( getRowCount() == 0 ) { return ""; } // 暫定処置
114
115                // 4.3.1.0 (2008/09/08)
116                if( headerFormat == null ) {
117                        final String errMsg = "ViewTagで canUseFormat() = true の場合、Formatter は必須です。";
118                        throw new HybsSystemException( errMsg );
119                }
120
121                headerLine       = null;                // 3.5.3.1 (2003/10/31) キャッシュクリア
122
123                final int lastNo = getLastNo( startNo, pageSize );
124                final int blc = getBackLinkCount();
125                final int hsc = getHeaderSkipCount();           // 3.5.2.0 (2003/10/20)
126                int hscCnt = 1;                                         // 3.5.2.0 (2003/10/20)
127
128                headerFormat.makeFormat( getDBTableModel() );   // 3.5.6.2 (2004/07/05) 移動
129                // 6.2.0.0 (2015/02/27) フォーマット系の noDisplay 対応
130                setFormatNoDisplay( headerFormat );
131
132                final StringBuilder out = new StringBuilder( BUFFER_LARGE )
133                        .append( getCountForm( startNo,pageSize ) )
134                        .append( getHeader() );
135
136                if( bodyFormatsCount == 0 ) {
137                        bodyFormats[0] = headerFormat ;
138                        bodyFormatsCount ++ ;
139                }
140                else {
141                        for( int i=0; i<bodyFormatsCount; i++ ) {
142                                bodyFormats[i].makeFormat( getDBTableModel() );
143                                // 6.2.0.0 (2015/02/27) フォーマット系の noDisplay 対応
144                                setFormatNoDisplay( bodyFormats[i] );
145                        }
146                }
147
148                int bgClrCnt = 0;
149                for( int row=startNo; row<lastNo; row++ ) {
150                        if( isSkip( row ) || isSkipNoEdit( row ) ) { continue; }                                // 4.3.1.0 (2008/09/08)
151                        for( int i=0; i<bodyFormatsCount; i++ ) {
152                                final TableFormatter bodyFormat = bodyFormats[i];
153                                if( ! bodyFormat.isUse( row,getDBTableModel() ) ) { continue; }         // 3.5.4.0 (2003/11/25)
154                                out.append("<tbody").append( getBgColorCycleClass( bgClrCnt++,row ) );
155                                if( isNoTransition() ) { // 4.3.3.0 (2008/10/01)
156                                        out.append( getHiddenRowValue( row ) );
157                                }
158                                out.append('>')         // 6.0.2.5 (2014/10/31) char を append する。
159                                        .append( bodyFormat.getTrTag() );
160
161                                // 3.5.5.0 (2004/03/12) No 欄そのものの作成判断追加
162                                if( isNumberDisplay() ) {
163                                        final String ckboxTD = "<td" + bodyFormat.getRowspan();         // 6.8.1.1 (2017/07/22)
164                                        out.append( makeCheckbox( ckboxTD,row,blc,true ) );                     // 6.8.1.1 (2017/07/22) 行番号のtdに、個別に class="S9" を追加
165                                }
166
167                                // 8.5.6.1 (2024/03/29) PMD 7.0.0 Finding duplicated code with CPD
168                                addFormatBody( out, bodyFormat, row );
169//                              int cl = 0;
170//                              boolean isTdBody = false;                                                                               // 6.4.3.3 (2016/03/04) tdBody処理を行ってよいかどうかのフラグ
171//                              for( ; cl<bodyFormat.getLocationSize(); cl++ ) {
172//                                      String fmt = bodyFormat.getFormat(cl);
173//                                      final int loc = bodyFormat.getLocation(cl);                                     // 3.5.5.0 (2004/03/12)
174//                                      if( ! bodyFormat.isNoClass() && loc >= 0 ) {                            // 3.5.5.7 (2004/05/10)
175//                                              if( isTdBody && TD_END.matcher( fmt ).matches() ) {             // 6.4.3.3 (2016/03/04) tdBody許可があり、TDが閉じた場合。
176//                                                      isTdBody = false;
177//                                                      final int idx = out.lastIndexOf( "<td" );
178//                                                      if( idx >= 0 ) {
179//                                                              // 8.5.4.2 (2024/01/12) class 属性がフォーマット中に存在する場合、追記になる。
180//                                                              insertClassName( out,loc,idx );                                 // 8.5.4.2 (2024/01/12)
181//
182//                                      //                      final String tdclass = " class=\"" + getClassName(loc) + "\" ";                 // 6.4.5.0 (2016/04/08)
183//                                      //                      out.insert( idx+3, tdclass );                                   // +3 は、"<td" の後ろという意味。fmtでなくそれ以前に戻る必要がある。
184//                                                      }
185//                                              }
186//
187//                                              if( TD_BODY.matcher( fmt ).matches() ) {                                // 6.4.3.3 (2016/03/04) TDが閉じた場合。
188//                                                      // 6.4.3.4 (2016/03/11) tdに、[カラム]が無いケースで、次の[カラム]のクラス属性が、前方すべてのtdにセットされてしまう対応。
189//                                                      // 上の処理と統一しなければいけないが、とりあえず暫定対応
190//                                                      final int idx = fmt.lastIndexOf( "<td" );
191//                                                      if( idx >= 0 ) {        // matchしてるので、あるはず
192//                                                              // 8.5.4.2 (2024/01/12) class 属性がフォーマット中に存在する場合、追記になる。
193//                                                              fmt = insertClassName( fmt,loc,idx );                   // 8.5.4.2 (2024/01/12)
194//
195//                                      //                      final String tdclass = " class=\"" + getClassName(loc) + "\" ";                 // 6.4.5.0 (2016/04/08)
196//                                      //                      fmt = fmt.substring( 0,idx+3 ) + tdclass + fmt.substring( idx+3 ) ;
197//                                                      }
198//                                                      isTdBody = false;                                                                       // 6.4.3.3 (2016/03/04) これは、要らない・・・はず。
199//                                              }
200//                                              else {
201//                                                      isTdBody = true;                                                                        // TDが閉じていない。
202//                                              }
203//                                      }
204//                                      out.append( fmt );                      // 3.5.0.0
205//
206//                                      // 3.5.5.7 (2004/05/10) #,$ 対応
207//                                      if( loc >= 0 ) {
208//                                              // 6.4.4.2 (2016/04/01) 処理の共通部をまとめる。
209//                                              out.append( getTypeCaseValue( bodyFormat.getType(cl),row,loc ) );
210//                                      }
211//                                      else {
212//                                              out.append( bodyFormat.getSystemFormat(row,loc) );
213//                                      }
214//                              }
215//                              out.append( bodyFormat.getFormat(cl) )
216//                                      .append("</tbody>").append( CR );
217                        }
218
219                // 3.5.2.0 (2003/10/20) ヘッダー繰り返し属性( headerSkipCount )を採用
220                        if( hsc > 0 && hscCnt % hsc == 0 ) {
221                                out.append("<tbody class=\"row_h\" >")
222                                        .append( getHeadLine() )
223                                        .append("</tbody>");
224                                hscCnt = 1;
225                        }
226                        else {
227                                hscCnt ++ ;
228                        }
229                }
230
231                if( footerFormat != null ) {
232                // 6.3.9.0 (2015/11/06) 引数にTableFormatterを渡して、処理の共有化を図る。
233                        out.append( getTableFoot( footerFormat ) );
234                }
235
236                out.append("</table>").append( CR )
237                        .append( getScrollBarEndDiv() );        // 3.8.0.3 (2005/07/15)
238
239                return out.toString();
240        }
241
242        /**
243         * 内容をクリア(初期化)します。
244         *
245         * @og.rev 3.1.1.0 (2003/03/28) 同期メソッド(synchronized付き)を非同期に変更する。
246         * @og.rev 3.5.0.0 (2003/09/17) Noカラムに、表示を全て消せるように、class 属性を追加。
247         * @og.rev 3.5.4.0 (2003/11/25) TableFormatter クラスを使用するように変更。
248         *
249         */
250        @Override
251        public void clear() {
252                super.clear();
253                headerFormat                    = null;
254                bodyFormats                             = null;
255                footerFormat                    = null;
256                bodyFormatsCount                = 0;
257        }
258
259        /**
260         * DBTableModel から テーブルのタグ文字列を作成して返します。
261         *
262         * @og.rev 3.5.0.0 (2003/09/17) &lt;tr&gt;属性は、元のフォーマットのまま使用します。
263         * @og.rev 3.5.1.0 (2003/10/03) Noカラムに、numberType 属性を追加
264         * @og.rev 3.5.2.0 (2003/10/20) ヘッダー繰り返し部をgetHeadLine()へ移動
265         * @og.rev 3.5.3.1 (2003/10/31) VERCHAR2 を VARCHAR2 に修正。
266         * @og.rev 3.5.4.0 (2003/11/25) TableFormatter クラスを使用するように変更。
267         * @og.rev 3.5.5.0 (2004/03/12) No 欄そのものの作成判断ロジックを追加
268         * @og.rev 3.5.6.2 (2004/07/05) makeFormat 処理をcreate メソッドの頭に移動
269         * @og.rev 3.5.6.5 (2004/08/09) thead に、id="header" を追加
270         * @og.rev 4.0.0.0 (2005/01/31) DBColumn の 属性(CLS_NM)から、DBTYPEに変更
271         * @og.rev 5.9.1.2 (2015/10/23) 自己終了警告対応
272         * @og.rev 6.4.4.1 (2016/03/18) NUMBER_DISPLAYを、static final 定数化します。
273         * @og.rev 6.4.9.0 (2016/07/23) colgroupのHTML5対応(No欄)
274         * @og.rev 6.4.9.1 (2016/08/05) colgroupのHTML5対応(No欄)時の対応ミス修正
275         * @og.rev 6.8.1.0 (2017/07/14) HTML5対応ヘッダー出力設定時に、ブラウザを互換設定したときの対応。
276         * @og.rev 6.8.2.0 (2017/10/13) makeNthChildの廃止と、makeCheckboxで、個別にclass指定するように変更。
277         * @og.rev 7.0.4.0 (2019/05/31) colgroup 廃止
278         * @og.rev 8.5.6.1 (2024/03/29) thead に、class="row_h" 属性を追加します。
279         * @og.rev 8.5.6.1 (2024/03/29) thead に、固定の id と class 属性を共通に定義します。
280         *
281         * @return      テーブルのタグ文字列
282         * @og.rtnNotNull
283         */
284        @Override
285        protected String getTableHead() {
286                final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE );
287
288//              // 3.5.5.0 (2004/03/12) No 欄そのものの作成判断追加
289//              // 7.0.4.0 (2019/05/31) colgroup 廃止
290//              if( isNumberDisplay() ) {
291//                      // 6.4.9.0 (2016/07/23) colgroupのHTML5対応(No欄)
292//                              buf.append( NUMBER_DISPLAY );           // 6.8.1.0 (2017/07/14) HTML5ネイティブ時でも、出力します。
293//                      // 6.8.1.0 (2017/07/14) HTML5対応ヘッダー出力設定時に、ブラウザを互換設定したときの対応。
294//                      // 6.8.2.0 (2017/10/13) makeNthChildの廃止と、makeCheckboxで、個別にclass指定するように変更。
295//      //              if( !useIE7Header ) {
296//      //                      buf.append( "<style type=\"text/css\">" ).append( CR );
297//      //                      makeNthChild( buf,2,"BIT" );
298//      //                      makeNthChild( buf,3,"S9"  );
299//      //                      buf.append( "</style>" ).append( CR );          // 6.4.9.1 (2016/08/05)
300//      //              }
301//                      // 5.9.1.2 (2015/10/23) 自己終了警告対応
302//                      // 6.4.4.1 (2016/03/18) NUMBER_DISPLAYを、static final 定数化します。
303//              }
304
305                // 3.5.2.0 (2003/10/20) ヘッダー繰り返し部をgetHeadLine()へ移動
306//              buf.append( "<thead id=\"header\">").append( CR )                                               // 3.5.6.5 (2004/08/09)
307                buf.append( THEAD_TAG )                                                                                                 // 8.5.6.1 (2024/03/29)
308                        .append( getHeadLine() )
309                        .append("</thead>").append( CR );
310
311                return buf.toString();
312        }
313
314        /**
315         * ヘッダー繰り返し部を、getTableHead()メソッドから分離。
316         *
317         * @og.rev 3.5.6.2 (2004/07/05) HTMLCustomTableScrollBar 用に新規作成
318         * @og.rev 6.1.2.0 (2015/01/24) キャッシュを返すのを、#getHeadLine() に移動。
319         * @og.rev 8.5.6.1 (2024/03/29) getHeadLine(String) → getFormatHeadLine(TableFormatter,String)
320         *
321         * @return      テーブルのタグ文字列
322         * @og.rtnNotNull
323         */
324        protected String getHeadLine() {
325                if( headerLine == null ) {                                      // キャッシュになければ、設定する。
326//                      headerLine = getHeadLine( "<th" + headerFormat.getRowspan() ) ;
327                        headerLine = getFormatHeadLine( headerFormat, "<th" + headerFormat.getRowspan() ) ;             // 8.5.6.1 (2024/03/29)
328                }
329
330                return headerLine ;
331        }
332
333//      /**
334//       * ヘッダー繰り返し部を、getTableHead()メソッドから分離。
335//       *
336//       * @og.rev 3.5.2.0 (2003/10/20) 新規作成
337//       * @og.rev 3.5.4.0 (2003/11/25) TableFormatter クラスを使用するように変更。
338//       * @og.rev 3.5.4.3 (2004/01/05) useCheckControl 属性の機能を追加
339//       * @og.rev 3.5.4.6 (2004/01/30) numberType="none" 時の処理を追加(Noラベルを出さない)
340//       * @og.rev 3.5.4.7 (2004/02/06) ヘッダーにソート機能用のリンクを追加します。
341//       * @og.rev 3.5.5.0 (2004/03/12) systemFormat(例:[KEY.カラム名]形式等)の対応
342//       * @og.rev 3.5.5.0 (2004/03/12) No 欄そのものの作成判断ロジックを追加
343//       * @og.rev 3.5.6.2 (2004/07/05) HTMLCustomTableScrollBar 用に引数追加
344//       * @og.rev 3.7.0.1 (2005/01/31) 全件チェックコントロール処理変更
345//       * @og.rev 6.1.2.0 (2015/01/24) キャッシュを返すのを、#getHeadLine() に移動。
346//       * @og.rev 6.4.3.4 (2016/03/11) ヘッダーでもTableFormatterのType(#,$,!)に対応した値を出すようにする。
347//       * @og.rev 6.4.4.2 (2016/04/01) TableFormatterのタイプ別値取得処理の共通部をまとめる。
348//       * @og.rev 8.5.6.1 (2024/03/29) getHeadLine(String) → getFormatHeadLine(TableFormatter,String)
349//       *
350//       * @param       thTag タグの文字列
351//       *
352//       * @return      テーブルのタグ文字列
353//       * @og.rtnNotNull
354//       */
355//      protected String getHeadLine( final String thTag ) {
356//              final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE )
357//                              .append( headerFormat.getTrTag() ).append( CR );
358//
359//              // 3.5.5.0 (2004/03/12) No 欄そのものの作成判断追加
360//              if( isNumberDisplay() ) {
361//                      // 6.1.2.0 (2015/01/24) thTag に、headerFormat.getRowspan() を含ませて受け取る。
362//                      if( isUseCheckControl() && "checkbox".equals( getSelectedType() ) ) {
363//                              buf.append( thTag ).append( "></th>" )
364//                                      // 6.0.2.5 (2014/10/31) char を append する。
365//                                      .append( thTag ).append( '>' ).append( getAllCheckControl() ).append( "</th>" )
366//                                      .append( thTag ).append( '>' ).append( getNumberHeader()    ).append( "</th>" );
367//                      }
368//                      else {
369//                              buf.append( thTag ).append( " colspan=\"3\">" ).append( getNumberHeader() ).append( "</th>" );  // 6.0.2.5 (2014/10/31) char を append する。
370//                      }
371//              }
372//
373//              int cl = 0;
374//              for( ; cl<headerFormat.getLocationSize(); cl++ ) {
375//                      buf.append( StringUtil.replace( headerFormat.getFormat(cl) ,"td","th" ));
376//                      final int loc = headerFormat.getLocation(cl);
377//                      // 6.4.3.4 (2016/03/11) ヘッダーでもTableFormatterのType(#,$,!)に対応した値を出すようにする。
378//      //              if( loc >= 0 ) { buf.append( getSortedColumnLabel(loc) ); }
379//                      if( loc >= 0 ) {
380//                              // 6.4.4.2 (2016/04/01) 処理の共通部をまとめる。
381//                              buf.append( getTypeCaseValue( headerFormat.getType(cl),-1,loc ) );
382//                      }
383//              }
384//              buf.append( StringUtil.replace( headerFormat.getFormat(cl) ,"td","th" ) ).append( CR );
385//
386//              return buf.toString();                          // 6.1.2.0 (2015/01/24)
387//      }
388
389        /**
390         * フォーマットを設定します。
391         *
392         * @og.rev 3.5.4.0 (2003/11/25) 新規作成
393         * @og.rev 3.5.4.4 (2004/01/16) 配列の最大数を変更
394         * @og.rev 3.5.5.5 (2004/04/23) headerFormat が定義されていない場合はエラー
395         * @og.rev 8.5.5.1 (2024/02/29) switch文にアロー構文を使用
396         *
397         * @param       list    TableFormatterのリスト
398         */
399        @Override
400        public void setFormatterList( final List<TableFormatter> list ) {               // 4.3.3.6 (2008/11/15) Generics警告対応
401                bodyFormats = new TableFormatter[BODYFORMAT_MAX_COUNT];
402
403                bodyFormatsCount = 0;
404                // 7.2.9.4 (2020/11/20) PMD:This for loop can be replaced by a foreach loop
405                for( final TableFormatter format : list ) {
406//              for( int i=0; i<list.size(); i++ ) {
407//                      final TableFormatter format = list.get( i );            // 4.3.3.6 (2008/11/15) Generics警告対応
408
409                        // 8.5.5.1 (2024/02/29) switch文にアロー構文を使用
410//                      switch( format.getFormatType() ) {
411//                              case TYPE_HEAD : headerFormat = format; break;
412//                              case TYPE_BODY : bodyFormats[bodyFormatsCount++] = format; break;
413//                              case TYPE_FOOT : footerFormat = format; break;
414//                              default : final String errMsg = "FormatterType の定義外の値が指定されました。";
415//                              // 4.3.4.4 (2009/01/01)
416//                                                throw new HybsSystemException( errMsg );
417//                      }
418                        switch( format.getFormatType() ) {
419                                case TYPE_HEAD -> headerFormat = format;
420                                case TYPE_BODY -> bodyFormats[bodyFormatsCount++] = format;
421                                case TYPE_FOOT -> footerFormat = format;
422                                default -> {
423                                        final String errMsg = "FormatterType の定義外の値が指定されました。";
424                                        // 4.3.4.4 (2009/01/01)
425                                        throw new HybsSystemException( errMsg );
426                                }
427                        }
428                }
429
430                // 3.5.5.5 (2004/04/23) headerFormat が定義されていない場合はエラー
431                if( headerFormat == null ) {
432                        final String errMsg = "h:thead タグの、フォーマットの指定は必須です。";
433                        throw new HybsSystemException( errMsg );
434                }
435        }
436
437        /**
438         * フォーマットメソッドを使用できるかどうかを問い合わせます。
439         *
440         * @return  使用可能(true)/ 使用不可能 (false)
441         */
442        @Override
443        public boolean canUseFormat() {
444                return true;
445        }
446
447        /**
448         * ビューで表示したカラムの一覧をCSV形式で返します。
449         *
450         * @og.rev 5.1.6.0 (2010/05/01) 新規追加
451         * @og.rev 6.2.0.1 (2015/03/06) TableFormatter#getLocation(int)の有効判定
452         * @og.rev 6.4.3.4 (2016/03/11) getViewClms(TableFormatter) を使用して表示されたカラム一覧を求めます。
453         *
454         * @return      ビューで表示したカラムの一覧
455         * @og.rtnNotNull
456         */
457        @Override
458        public String getViewClms() {
459                return getViewClms( headerFormat );
460        }
461}