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.Calendar;
019import java.util.Date;
020import java.util.List;
021
022import org.opengion.hayabusa.common.HybsSystemException;
023import org.opengion.hayabusa.db.DBTableModel;
024import org.opengion.fukurou.util.HybsDateUtil;                                          // 6.4.2.0 (2016/01/29)
025import org.opengion.fukurou.util.StringUtil;
026import org.opengion.hayabusa.html.TableFormatter;
027import org.opengion.hayabusa.html.ViewStackTableParam;
028
029/**
030 * 積上ガント表示専用のViewFormです。
031 * stackParamTagを利用する事でスタックガント用の行を出力する事が可能です。
032 * stackParamTagによりstackColumnsが指定された場合は、そのカラム毎にブレークして、
033 * stacklink属性により積上げ行の判別が可能なtbody行を出力します。
034 * その際、stackColumnsで指定されたカラム以外の[xxx]は処理されません(空白として出力)
035 * [xxx]以外で書かれた箇所、例えば<iGantBar>タグの本体部分等は出力されます。
036 *
037 * ヘッダの表示にはstackHeaderタグを利用します。
038 *
039 * [エンジン内部積上げを行わない場合]
040 * 積上の表示はJavaScriptによってiGantBarタグの箇所に作成されます。
041 * 積上げそのものもiGantBarによって出力されるガントを利用してJavaScriptで行っているため、
042 * 最大検索行数と表示行数に注意して下さい。
043 *
044 * [エンジン内部積上げを行う場合]
045 * 工数積上げをエンジン内部で行いdivタグとして出力します。
046 * その後の描画(位置調整や色等)はJavaScriptで行います。
047 * ガント部分は出力されません。
048 * スタック部分はbody部分の最後尾に新たにtd作成するため、注意してください。
049 * paramタグでの指定で、costColumnが必須です。
050 *
051 *
052 * AbstractViewForm により、setter/getterメソッドのデフォルト実装を提供しています。
053 * 各HTMLのタグに必要な setter/getterメソッドのみ、追加定義しています。
054 *
055 * AbstractViewForm を継承している為、ロケールに応じたラベルを出力させる事が出来ます。
056 *
057 * @og.rev 5.5.7.0 (2012/10/01) 新規作成
058 * @og.rev 5.5.8.3 (2012/11/17) 内部積上げ対応
059 * @og.rev 5.6.1.2 (2013/02/22) キャパシティ対応
060 * @og.rev 8.5.6.1 (2024/03/29) ViewForm_HTMLCustomTable を継承します。
061 * @og.group 画面表示
062 *
063 * @version  5.0
064 * @author       Takahashi Masakazu
065 * @since    JDK5.0,
066 */
067// public class ViewForm_HTMLStackedGanttTable extends ViewForm_HTMLTable       {
068public class ViewForm_HTMLStackedGanttTable extends ViewForm_HTMLCustomTable    {
069        /** このプログラムのVERSION文字列を設定します。   {@value} */
070        private static final String VERSION = "8.5.6.1 (2024/03/29)" ;
071
072//      8.5.6.1 (2024/03/29) ViewForm_HTMLCustomTable を継承します。
073//      /** ボディーフォーマット最大数 初期値:{@value} */
074//      protected static final int BODYFORMAT_MAX_COUNT = 10;
075        /** stack行の判定出力用 {@value} */
076        protected static final String STACK_TBODY               = " stackline='true'";
077        /** stack行の判定出力用 {@value} */
078        protected static final String GANTT_TBODY               = " stackline='false'";
079        /** stackのIDプレフィックス {@value} */
080        protected static final String STACK_ID_PREFIX   = " id='stack_";
081        /** stackの行プレフィックス {@value} */
082        protected static final String STACK_ROW_PREFIX  = " stackrow='";
083
084//      8.5.6.1 (2024/03/29) ViewForm_HTMLCustomTable を継承します。
085//      /** ヘッダーフォーマット変数 */
086//      protected TableFormatter                headerFormat    ;
087//      /** ボディーフォーマット配列変数 */
088//      protected TableFormatter[]              bodyFormats             ;
089//      /** フッターフォーマット変数 */
090//      protected TableFormatter                footerFormat    ;
091//      /** ボディーフォーマット数 */
092//      protected int                                   bodyFormatsCount;
093
094        // stack,gantt用
095        private int[]  stackCols                ;
096
097        // 5.5.8.3 (2012/11/17)
098        private int[]   costCols                ;               // 工数カラム、開始日カラム、終了日カラム
099        private boolean innerStack              = Boolean.parseBoolean( ViewStackTableParam.INNER_STACK_VALUE );
100        private boolean stackHoliday    = Boolean.parseBoolean( ViewStackTableParam.STACK_HOLIDAY_KEY );
101
102        private String[][] calArray                             ;               // headで作成されたカレンダーデータ                   // 6.3.9.1 (2015/11/27) 修飾子を、なし → private に変更。
103        private int capCol = -1                                 ;               // 5.6.1.2 (2013/02/22) 能力値カラム          // 6.3.9.1 (2015/11/27) 修飾子を、なし → private に変更。
104
105        /**
106         * デフォルトコンストラクター
107         *
108         * @og.rev 8.5.3.2 (2023/10/13) JDK21対応。警告: デフォルトのコンストラクタの使用で、コメントが指定されていません
109         */
110        public ViewForm_HTMLStackedGanttTable() { super(); }            // これも、自動的に呼ばれるが、空のメソッドを作成すると警告されるので、明示的にしておきます。
111
112        /**
113         * DBTableModel から HTML文字列を作成して返します。
114         * startNo(表示開始位置)から、pageSize(表示件数)までのView文字列を作成します。
115         * 表示残りデータが pageSize 以下の場合は、残りのデータをすべて出力します。
116         *
117         *
118         * @og.rev 5.5.8.3 (2012/11/17) 内部積上げ対応
119         * @og.rev 5.6.1.2 (2013/02/22) キャパシティ対応
120         * @og.rev 5.6.2.1 (2013/06/13) 積上不具合修正
121         * @og.rev 6.2.0.0 (2015/02/27) フォーマット系の noDisplay 対応
122         * @og.rev 6.4.2.0 (2016/01/29) HybsDateUtil.getCalendar( String ) を直接利用するように修正します。
123         * @og.rev 6.4.3.4 (2016/03/11) tdに、[カラム]が無いケースで、次の[カラム]のクラス属性が、前方すべてのtdにセットされてしまう対応。
124         * @og.rev 6.4.4.2 (2016/04/01) TableFormatterのタイプ別値取得処理の共通部をまとめる。
125         * @og.rev 6.4.5.0 (2016/04/08) メソッド変更( getColumnDbType(int) → getClassName(int) )
126         * @og.rev 6.8.1.1 (2017/07/22) ckboxTD変数は、<td> から <td に変更します(タグの最後が記述されていない状態でもらう)。
127         * @og.rev 6.8.2.0 (2017/10/13) makeNthChildの廃止と、makeCheckboxで、個別にclass指定するように変更。
128         *
129         * @param  sttNo          表示開始位置
130         * @param  pgSize   表示件数
131         *
132         * @return      DBTableModelから作成された HTML文字列
133         * @og.rtnNotNull
134         */
135        @Override
136        public String create( final int sttNo, final int pgSize )  {
137                // ガントは、キーブレイクがあるため、全件表示します。
138                final int pageSize = getRowCount() ;
139                if( pageSize == 0 ) { return ""; }      // 暫定処置
140
141                // 4.3.1.0 (2008/09/08)
142                if( headerFormat == null ) {
143                        final String errMsg = "ViewTagで canUseFormat() = true の場合、Formatter は必須です。";
144                        throw new HybsSystemException( errMsg );
145                }
146
147                headerLine       = null;                // 3.5.3.1 (2003/10/31) キャッシュクリア
148
149                // 6.4.1.1 (2016/01/16) PMD refactoring. Avoid declaring a variable if it is unreferenced before a possible exit point.
150                // ガントは、キーブレイクがあるため、全件表示します。
151                final int startNo  = 0;
152
153                final int lastNo = getLastNo( startNo, pageSize );
154                final int blc = getBackLinkCount();
155
156                // このビューの特有な属性を初期化
157                paramInit();
158
159                headerFormat.makeFormat( getDBTableModel() );   // 3.5.6.2 (2004/07/05) 移動
160                // 6.2.0.0 (2015/02/27) フォーマット系の noDisplay 対応
161                setFormatNoDisplay( headerFormat );
162
163                final StringBuilder out = new StringBuilder( BUFFER_LARGE )
164                        .append( getCountForm( startNo,pageSize ) )
165                        .append( getHeader() );
166
167                if( bodyFormatsCount == 0 ) {
168                        bodyFormats[0] = headerFormat ;
169                        bodyFormatsCount ++ ;
170                }
171                else {
172                        for( int i=0; i<bodyFormatsCount; i++ ) {
173                                bodyFormats[i].makeFormat( getDBTableModel() );
174                                // 6.2.0.0 (2015/02/27) フォーマット系の noDisplay 対応
175                                setFormatNoDisplay( bodyFormats[i] );
176                        }
177                }
178
179                final String[] astrOldStackKeys = new String[stackCols.length];         // 8.5.4.2 (2024/01/12) PMD 7.0.0 LocalVariableCouldBeFinal
180                for( int nIndex=0; nIndex<astrOldStackKeys.length; nIndex++ ) {
181                        astrOldStackKeys[nIndex] = "";
182                }
183
184                int bgClrCnt = 0;
185                int stackRow = 0;
186
187                // 5.5.8.3 (2012/11/17)
188                double[] costAry = null;
189                // 6.3.9.1 (2015/11/27) Found 'DD'-anomaly for variable(PMD)
190                final Calendar firstCalday;
191                final String   calZoom ;                // 6.3.9.1 (2015/11/27)
192                if( innerStack ){
193                        costAry = new double[calArray.length];
194                        final String[] firstCal = calArray[0];
195                        firstCalday = HybsDateUtil.getCalendar(firstCal[0]);                                                    // 6.4.2.0 (2016/01/29)
196                        final Calendar fstCalEnd = HybsDateUtil.getCalendar(firstCal[2]);                               // 6.4.2.0 (2016/01/29)
197
198                        if( differenceDays(firstCalday.getTime(),fstCalEnd.getTime()) == 1 ){
199                                calZoom = ViewStackTableParam.STACK_ZOOM_DAY;
200                        }
201                        else if( differenceDays(firstCalday.getTime(),fstCalEnd.getTime()) == 7 ){
202                                calZoom = ViewStackTableParam.STACK_ZOOM_WEEK;
203                        }
204                        else{
205                                calZoom = ViewStackTableParam.STACK_ZOOM_MONTH;
206                        }
207                }
208                else {                                                  // 6.3.9.1 (2015/11/27)
209                        firstCalday = null;
210                        calZoom     = null;
211                }
212
213                String capacity = null;                 // 5.6.1.2 (2013/02/22)
214                for( int row=startNo; row<lastNo; row++ ) {
215                        // データのスキップは行わない
216
217                        // ガントのブレイク
218                //      if(! isSameGroup(row, astrOldGroupKeys)) {
219                                // 6.9.7.0 (2018/05/14) PMD These nested if statements could be combined
220//                              if( !isSameStack(row, astrOldStackKeys) && stackCols.length > 0 ) { // 積上のブレイク
221//                                      if( !(innerStack && row == startNo) ) {                                                 // 5.5.8.3 (2012/11/17) 内部積上げは後から積上げるので、初回は出力しない
222                                if( !isSameStack(row, astrOldStackKeys) && stackCols.length > 0         // 積上のブレイク
223                                        && !(innerStack && row == startNo ) ) {                                                 // 5.5.8.3 (2012/11/17) 内部積上げは後から積上げるので、初回は出力しない
224                                                stackRow = row;
225
226                                                makeBodyTable( out,innerStack ? row -1 : row, stackRow, bgClrCnt, blc, costAry, capacity );                     // 6.1.0.0 (2014/12/26) refactoring
227
228                                                if( innerStack ){
229                                                        costAry = new double[calArray.length];
230                                                }
231//                                      }
232                                }
233                                // 6.1.0.0 (2014/12/26) refactoring : Avoid if(x != y) ..; else ..;
234                                if( innerStack ){       // 5.5.8.3 (2012/11/17) 内部積上げをする場合
235                                        final double costDbl = Double.parseDouble( getValue(row,costCols[0]) ); //工数
236                                        final Calendar startDay = HybsDateUtil.getCalendar(getValue(row,costCols[1]));          // 6.4.2.0 (2016/01/29)
237                                        final Calendar endDay   = HybsDateUtil.getCalendar(getValue(row,costCols[2]));          // 6.4.2.0 (2016/01/29)
238
239                                        final Date startDayDate = startDay.getTime();
240                                        Date endDayDate = endDay.getTime();
241
242                                        // 5.6.1.2 (2013/02/22)
243                                        if( capCol > -1 ){
244                                                capacity = getValue(row,capCol);
245                                        }
246                                        else{
247                                                capacity = "1";
248                                        }
249
250                                        // 枠はそのままで計算
251                                        final int fromCell = calNumber(startDayDate,calZoom,firstCalday.getTime());
252                                        final int toCell   = calNumber(endDayDate,calZoom,firstCalday.getTime());
253
254                                        endDay.add(Calendar.DATE, 1); // 終了日は範囲に入るので1つ進める
255                                        endDayDate = endDay.getTime();
256
257                                        int stackMother = differenceDays(startDayDate,endDayDate);
258                                        if( !stackHoliday ){
259                                                for( int cel=fromCell; cel<=toCell; cel++  ){
260                                                        if( "1".equals( calArray[cel][1] ) ){
261                                                                stackMother--;
262                                                        }
263                                                }
264                                        }
265
266                                        Date calFrom;
267                                        Date calTo;
268
269                                        for( int cel=fromCell; cel<=toCell; cel++ ){
270                                                calFrom = HybsDateUtil.getCalendar(calArray[cel][0]).getTime();                 // 6.4.2.0 (2016/01/29)
271                                                calTo   = HybsDateUtil.getCalendar(calArray[cel][2]).getTime();                 // 6.4.2.0 (2016/01/29)
272                                                if( calFrom.compareTo( startDayDate ) < 0 ){
273                                                        calFrom = startDayDate;
274                                                }
275                                                if( endDayDate.compareTo( calTo ) < 0 ){
276                                                        calTo = endDayDate;
277                                                }
278                                                // 6.3.9.1 (2015/11/27) Found 'DD'-anomaly for variable(PMD)
279                                                final int cellDays = differenceDays( calFrom, calTo );
280                                                if( stackHoliday ){
281                                                        costAry[cel] += (costDbl / stackMother) * cellDays;
282                                                }
283                                                else{
284                                                        // 休日のみの場合は積上げられない!
285                                                        if( !"1".equals( calArray[cel][1] ) ){
286                                                                costAry[cel] += (costDbl / stackMother) * cellDays;
287                                                        }
288                                                }
289                                        }
290                                }
291                                else{   // 5.5.8.3 (2012/11/17) 内部積上げの場合はガント部分は出力せずに積上げだけする。
292                                        for( int i=0; i<bodyFormatsCount; i++ ) {
293                                                final TableFormatter bodyFormat = bodyFormats[i];
294                                                if( ! bodyFormat.isUse( row,getDBTableModel() ) ) { continue; }         // 3.5.4.0 (2003/11/25)
295                                                out.append("<tbody").append( getBgColorCycleClass( bgClrCnt++,row ) );
296                                                if( isNoTransition() ) {        // 4.3.3.0 (2008/10/01)
297                                                        out.append( getHiddenRowValue( row ) );
298                                                }
299                                                out.append( GANTT_TBODY )
300                                                        .append( STACK_ROW_PREFIX ).append( stackRow ).append("'>")
301                                                        .append( bodyFormat.getTrTag() );
302
303                                                // 3.5.5.0 (2004/03/12) No 欄そのものの作成判断追加
304                                                if( isNumberDisplay() ) {
305                                                        final String ckboxTD = "<td" + bodyFormat.getRowspan();                 // 6.8.1.1 (2017/07/22)
306                                                        out.append( makeCheckbox( ckboxTD,row,blc,true ) );                             // 6.8.2.0 (2017/10/13)
307                                                }
308
309                                                int cl = 0;
310                                                for( ; cl<bodyFormat.getLocationSize(); cl++ ) {
311                                                        String fmt = bodyFormat.getFormat(cl);
312                                                        final int loc = bodyFormat.getLocation(cl);             // 3.5.5.0
313                                                        if( ! bodyFormat.isNoClass() && loc >= 0 ) {    // 3.5.5.7 (2004/05/10)
314                                                                // 6.4.3.4 (2016/03/11) tdに、[カラム]が無いケースで、次の[カラム]のクラス属性が、前方すべてのtdにセットされてしまう対応。
315                                                                final int idx = fmt.lastIndexOf( "<td" );
316                                                                if( idx >= 0 ) {        // matchしてるので、あるはず
317                                                                        // 8.5.4.2 (2024/01/12) class 属性がフォーマット中に存在する場合、追記になる。
318                                                                        fmt = insertClassName( fmt,loc,idx );                   // 8.5.4.2 (2024/01/12)
319
320                                                        //              final String tdclass = " class=\"" + getClassName(loc) + "\" ";                 // 6.4.5.0 (2016/04/08)
321                                                        //              fmt = fmt.substring( 0,idx+3 ) + tdclass + fmt.substring( idx+3 ) ;
322                                                                }
323                                                        }
324                                                        out.append( fmt );                      // 3.5.0.0
325                                                        // 3.5.5.7 (2004/05/10) #,$ 対応
326                                                        if( loc >= 0 ) {
327                                                                // 6.4.4.2 (2016/04/01) 処理の共通部をまとめる。
328                                                                out.append( getTypeCaseValue( bodyFormat.getType(cl),row,loc ) );
329                                                        }
330                                                        else {
331                                                                out.append( bodyFormat.getSystemFormat(row,loc) );
332                                                        }
333                                                }
334                                                out.append( bodyFormat.getFormat(cl) )
335                                                        .append("</tbody>").append( CR );
336                                        }
337                                }
338                }
339
340                // 6.0.2.5 (2014/10/31) たぶん、カッコのコメントする位置間違いで使われてないようなので、一旦コメントする。
341        //      } // 5.6.5.2 (2013/06/21) 括弧の位置間違いのため修正
342
343                // 内部積上げ時は最終行の出力を行う
344                if( innerStack ){
345                        makeBodyTable( out, lastNo-1, stackRow, bgClrCnt, blc, costAry, capacity );                     // 6.1.0.0 (2014/12/26) refactoring
346                }
347
348                if( footerFormat != null ) {
349                        // 6.3.9.0 (2015/11/06) 引数にTableFormatterを渡して、処理の共有化を図る。
350                        out.append( getTableFoot( footerFormat ) );
351                }
352
353                out.append("</table>").append( CR )
354                        .append( getScrollBarEndDiv() );        // 3.8.0.3 (2005/07/15)
355
356                return out.toString();
357        }
358
359        /**
360         * 内容をクリア(初期化)します。
361         *
362         * @og.rev 5.5.8.3 (2012/11/17) 内部積上げのための修正
363         * @og.rev 5.6.1.2 (2013/02/22) キャパシティ対応
364         *
365         */
366        @Override
367        public void clear() {
368                super.clear();
369//              headerFormat            = null;
370//              bodyFormats                     = null;
371//              footerFormat            = null;
372//              bodyFormatsCount        = 0;
373                stackCols                       = null;                 // 5.5.8.3 (2012/11/17)
374                costCols                        = null;                 // 5.5.8.3 (2012/11/17)
375                innerStack                      = Boolean.parseBoolean( ViewStackTableParam.INNER_STACK_VALUE ); // 5.5.8.3 (2012/11/17)
376                calArray                        = null;                 // 5.5.8.3 (2012/11/17)
377                stackHoliday            = Boolean.parseBoolean( ViewStackTableParam.STACK_HOLIDAY_KEY ); // 5.5.8.3 (2012/11/17)
378                capCol                          = -1;                   // 5.6.1.2 (2013/02/22)
379        }
380
381        /**
382         * このビューに対する特別な初期化を行う。
383         *
384         * @og.rev 5.5.8.3 (2012/11/17)
385         * @og.rev 5.5.9.0 (2012/12/03) objectではなくArrayList化
386         * @og.rev 5.6.1.2 (2013/02/22) キャパシティ対応
387         * @og.rev 5.6.2.1 (2013/03/08) stackHolidayが抜けていたので追加
388         */
389        private void paramInit() {
390                final String costCol            = getParam( ViewStackTableParam.COST_COLUMNS_KEY,       ViewStackTableParam.COST_COLUMNS_VALUE  ); // 5.5.8.3 (2012/11/17)
391                innerStack                                      = getBoolParam( ViewStackTableParam.INNER_STACK_KEY     ); // 5.5.8.3 (2012/11/17)
392                stackHoliday                            = getBoolParam( ViewStackTableParam.STACK_HOLIDAY_KEY   ); // 5.6.2.1 (2013/03/08)
393
394                if( innerStack ){
395                        // 6.1.0.0 (2014/12/26) findBugs: Bug type ITA_INEFFICIENT_TO_ARRAY (click for details)
396                        // 長さが0の配列の引数で Collection.toArray() を使用しています。
397                        final List<String[]> lst = getViewArrayList();
398                        // 8.5.5.0 (2024/02/02) OptimizableToArrayCall 対応 多重配列の場合の挙動が未確認のため、いったん戻します。
399                        calArray = lst.toArray( new String[lst.size()][3] );
400        //              calArray = lst.toArray( new String[0][0] );     // 8.5.4.2 (2024/01/12) PMD 7.0.0 OptimizableToArrayCall 対応
401                        if( calArray == null || costCol == null){
402                                final String errMsg = "ヘッダのカレンダデータ、costColumnsの設定は必須です。"+costCol;
403                                throw new HybsSystemException( errMsg );
404                        }
405                }
406
407                final DBTableModel table = getDBTableModel();
408
409                // 6.4.1.1 (2016/01/16) PMD refactoring. Avoid declaring a variable if it is unreferenced before a possible exit point.
410                final String strStackCols       = getParam( ViewStackTableParam.STACK_COLUMNS_KEY,      ViewStackTableParam.STACK_COLUMNS_VALUE );
411                final String[] stackKeys = StringUtil.csv2Array(strStackCols);
412                stackCols = new int[stackKeys.length];
413                for( int nIndex=0; nIndex<stackCols.length ; nIndex++ ) {
414                        stackCols[nIndex] = table.getColumnNo( stackKeys[nIndex] );
415                }
416
417                final String[] costKeys = StringUtil.csv2Array(costCol);
418                costCols = new int[costKeys.length];
419                for( int nIndex=0; nIndex<costCols.length ; nIndex++ ) {
420                        costCols[nIndex] = table.getColumnNo( costKeys[nIndex] );
421                }
422
423                // 5.6.1.2 (2013/02/22) キャパシティ
424                // 6.4.1.1 (2016/01/16) PMD refactoring. Avoid declaring a variable if it is unreferenced before a possible exit point.
425                final String capColName         = getParam( ViewStackTableParam.CAP_COLUMN_KEY, ViewStackTableParam.CAP_COLUMN_VALUE    ); // 5.6.1.2 (2013/02/22)
426                if( capColName != null && capColName.length() > 0 ){
427                        capCol = table.getColumnNo(capColName);
428                }
429        }
430
431//      /**
432//       * DBTableModel から テーブルのタグ文字列を作成して返します。
433//       *
434//       * @og.rev 5.9.1.2 (2015/10/23) 自己終了警告対応
435//       * @og.rev 6.4.4.1 (2016/03/18) NUMBER_DISPLAYを、static final 定数化します。
436//       * @og.rev 6.4.9.0 (2016/07/23) colgroupのHTML5対応(No欄)
437//       * @og.rev 6.4.9.1 (2016/08/05) colgroupのHTML5対応(No欄)時の対応ミス修正
438//       * @og.rev 6.8.1.0 (2017/07/14) HTML5対応ヘッダー出力設定時に、ブラウザを互換設定したときの対応。
439//       * @og.rev 6.8.2.0 (2017/10/13) makeNthChildの廃止と、makeCheckboxで、個別にclass指定するように変更。
440//       * @og.rev 7.0.4.0 (2019/05/31) colgroup 廃止
441//       * @og.rev 8.5.6.1 (2024/03/29) 継承元と同じなので削除
442//       *
443//       * @return      テーブルのタグ文字列
444//       * @og.rtnNotNull
445//       */
446//      @Override
447//      protected String getTableHead() {
448//              final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE );
449//
450////            // 3.5.5.0 (2004/03/12) No 欄そのものの作成判断追加
451////            // 7.0.4.0 (2019/05/31) colgroup 廃止
452////            if( isNumberDisplay() ) {
453////                    // 6.4.9.0 (2016/07/23) colgroupのHTML5対応(No欄)
454////                            buf.append( NUMBER_DISPLAY );           // 6.8.1.0 (2017/07/14) HTML5ネイティブ時でも、出力します。
455////                    // 6.8.1.0 (2017/07/14) HTML5対応ヘッダー出力設定時に、ブラウザを互換設定したときの対応。
456////                    // 6.8.2.0 (2017/10/13) makeNthChildの廃止と、makeCheckboxで、個別にclass指定するように変更。
457////    //              if( !useIE7Header ) {
458////    //                      buf.append( "<style type=\"text/css\">" ).append( CR );
459////    //                      makeNthChild( buf,2,"BIT" );
460////    //                      makeNthChild( buf,3,"S9"  );
461////    //                      buf.append( "</style>" ).append( CR );          // 6.4.9.1 (2016/08/05)
462////    //              }
463////            }
464//
465//              // 3.5.2.0 (2003/10/20) ヘッダー繰り返し部をgetHeadLine()へ移動
466//              buf.append("<thead id=\"header\">").append( CR )        // 3.5.6.5 (2004/08/09)
467//                      .append( getHeadLine() )
468//                      .append("</thead>").append( CR );
469//
470//              return buf.toString();
471//      }
472
473//      /**
474//       * ヘッダー繰り返し部を、getTableHead()メソッドから分離。
475//       *
476//       * @og.rev 6.1.2.0 (2015/01/24) キャッシュを返すのを、#getHeadLine() に移動。
477//       * @og.rev 8.5.6.1 (2024/03/29) 継承元と同じなので削除
478//       *
479//       * @return      テーブルのタグ文字列
480//       * @og.rtnNotNull
481//       */
482//      protected String getHeadLine() {
483//              if( headerLine == null ) {                                      // キャッシュになければ、設定する。
484//                      headerLine = getHeadLine( "<th" + headerFormat.getRowspan() ) ;
485//              }
486//
487//              return headerLine ;
488//      }
489
490//      /**
491//       * ヘッダー繰り返し部を、getTableHead()メソッドから分離。
492//       *
493//       * @og.rev 6.1.2.0 (2015/01/24) キャッシュを返すのを、#getHeadLine() に移動。
494//       * @og.rev 6.4.3.4 (2016/03/11) ヘッダーでもTableFormatterのType(#,$,!)に対応した値を出すようにする。
495//       * @og.rev 6.4.4.2 (2016/04/01) TableFormatterのタイプ別値取得処理の共通部をまとめる。
496//       * @og.rev 8.5.6.1 (2024/03/29) 継承元と同じなので削除
497//       *
498//       * @param       thTag タグの文字列
499//       *
500//       * @return      テーブルのタグ文字列
501//       * @og.rtnNotNull
502//       */
503//      protected String getHeadLine( final String thTag ) {
504//
505//              final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE )
506//                              .append( headerFormat.getTrTag() ).append( CR );
507//
508//              // 3.5.5.0 (2004/03/12) No 欄そのものの作成判断追加
509//              if( isNumberDisplay() ) {
510//                      // 6.1.2.0 (2015/01/24) thTag に、headerFormat.getRowspan() を含ませて受け取る。
511//                      if( isUseCheckControl() && "checkbox".equals( getSelectedType() ) ) {
512//                              buf.append( thTag ).append( "></th>" )
513//                                      .append( thTag ).append( '>' ).append( getAllCheckControl() ).append( "</th>" )         // 6.0.2.5 (2014/10/31) char を append する。
514//                                      .append( thTag ).append( '>' ).append( getNumberHeader()    ).append( "</th>" );        // 6.0.2.5 (2014/10/31) char を append する。
515//                      }
516//                      else {
517//                              buf.append( thTag ).append( " colspan=\"3\">" ).append( getNumberHeader() ).append( "</th>" );  // 6.0.2.5 (2014/10/31) char を append する。
518//                      }
519//              }
520//
521//              int cl = 0;
522//              for( ; cl<headerFormat.getLocationSize(); cl++ ) {
523//                      buf.append( StringUtil.replace( headerFormat.getFormat(cl) ,"td","th" ));
524//                      final int loc = headerFormat.getLocation(cl);
525//                      // 6.4.3.4 (2016/03/11) ヘッダーでもTableFormatterのType(#,$,!)に対応した値を出すようにする。
526//                      if( loc >= 0 ) {
527//                              // 6.4.4.2 (2016/04/01) 処理の共通部をまとめる。
528//                              buf.append( getTypeCaseValue( headerFormat.getType(cl),-1,loc ) );
529//                      }
530//              }
531//              buf.append( StringUtil.replace( headerFormat.getFormat(cl) ,"td","th" ) ).append( CR );
532//
533//              return buf.toString();                          // 6.1.2.0 (2015/01/24)
534//      }
535
536//      /**
537//       * フォーマットを設定します。
538//       *
539//       * @og.rev 8.5.5.1 (2024/02/29) switch文にアロー構文を使用
540//       * @og.rev 8.5.6.1 (2024/03/29) 継承元と同じなので削除
541//       *
542//       * @param       list    TableFormatterのリスト
543//       */
544//      @Override
545//      public void setFormatterList( final List<TableFormatter> list ) {               // 4.3.3.6 (2008/11/15) Generics警告対応
546//              bodyFormats = new TableFormatter[BODYFORMAT_MAX_COUNT];
547//
548//              bodyFormatsCount = 0;
549//              // 7.2.9.4 (2020/11/20) PMD:This for loop can be replaced by a foreach loop
550//              for( final TableFormatter format : list ) {
551////            for( int i=0; i<list.size(); i++ ) {
552////                    final TableFormatter format = list.get( i );            // 4.3.3.6 (2008/11/15) Generics警告対応
553//
554//                      // 8.5.5.1 (2024/02/29) switch文にアロー構文を使用
555////                    switch( format.getFormatType() ) {
556////                            case TYPE_HEAD : headerFormat = format; break;
557////                            case TYPE_BODY : bodyFormats[bodyFormatsCount++] = format; break;
558////                            case TYPE_FOOT : footerFormat = format; break;
559////                            default : final String errMsg = "FormatterType の定義外の値が指定されました。";
560////                            // 4.3.4.4 (2009/01/01)
561////                                              throw new HybsSystemException( errMsg );
562////                    }
563//                      switch( format.getFormatType() ) {
564//                              case TYPE_HEAD -> headerFormat = format;
565//                              case TYPE_BODY -> bodyFormats[bodyFormatsCount++] = format;
566//                              case TYPE_FOOT -> footerFormat = format;
567//                              default -> {
568//                                      final String errMsg = "FormatterType の定義外の値が指定されました。";
569//                                      // 4.3.4.4 (2009/01/01)
570//                                      throw new HybsSystemException( errMsg );
571//                              }
572//                      }
573//              }
574//
575//              // 3.5.5.5 (2004/04/23) headerFormat が定義されていない場合はエラー
576//              if( headerFormat == null ) {
577//                      final String errMsg = "h:thead タグの、フォーマットの指定は必須です。";
578//                      throw new HybsSystemException( errMsg );
579//              }
580//      }
581
582//      /**
583//       * フォーマットメソッドを使用できるかどうかを問い合わせます。
584//       *
585//       * @og.rev 8.5.6.1 (2024/03/29) 継承元と同じなので削除
586//       *
587//       * @return  使用可能(true)/ 使用不可能 (false)
588//       */
589//      @Override
590//      public boolean canUseFormat() {
591//              return true;
592//      }
593
594//      /**
595//       * ビューで表示したカラムの一覧をCSV形式で返します。
596//       *
597//       *    ※ 8.5.6.1 (2024/03/29) 継承元と処理が異なっているが、
598//       *       多分同じことをしていると思われるので、統一しておきます。
599//       *
600//       * @og.rev 5.1.6.0 (2010/05/01) 新規追加
601//       * @og.rev 6.2.0.1 (2015/03/06) TableFormatter#getLocation(int)の有効判定
602//       * @og.rev 8.5.6.1 (2024/03/29) 継承元と同じなので削除
603//       *
604//       * @return      ビューで表示したカラムの一覧
605//       * @og.rtnNotNull
606//       */
607//      @Override
608//      public String getViewClms() {
609//              final DBTableModel table = getDBTableModel();
610//              final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE );
611//              for( int i=0; i<headerFormat.getLocationSize(); i++ ) {
612//                      if( buf.length() > 0 ) { buf.append( ',' ); }
613//                      // 6.2.0.1 (2015/03/06) TableFormatter#getLocation(int)の有効判定
614//                      final int loc = headerFormat.getLocation(i);
615//                      if( loc >= 0 ) { buf.append( table.getColumnName( loc ) ); }
616//              }
617//              return buf.toString();
618//      }
619
620        /**
621         * 上下行のデータが同じ積上かどうかをチェックする。
622         *
623         * @param   nRowIndex テーブルモデルの行番号
624         * @param   astrOldValues 古いグルプーデータ配列
625         *
626         * @return  使用可能(true)/ 使用不可能 (false)
627         */
628        private boolean isSameStack(final int nRowIndex, final String[] astrOldValues) {
629                boolean bRet = stackCols.length > 0 ;
630                if( bRet ) {
631                        for( int nIndex=0; bRet && nIndex<stackCols.length ; nIndex++ ) {
632                                bRet = astrOldValues[nIndex].equals( getValue( nRowIndex, stackCols[nIndex] ) ) ;
633                        }
634
635                        // 不一致時に astrOldValues に 新しい値を設定しておきます。
636                        if( !bRet ) {
637                                for( int nIndex=0; nIndex<stackCols.length; nIndex++ ) {
638                                        astrOldValues[nIndex] = getValue(nRowIndex, stackCols[nIndex]);
639                                }
640                        }
641                }
642                return bRet;
643        }
644
645        /**
646         * 対象カラムが積上げカラムかどうか。
647         *
648         * @param   loc 列番号
649         *
650         * @return  対象(true)/ 非対象 (false)
651         */
652        private boolean isStackClm(final int loc) {
653                boolean rtn = false;
654                // 7.2.9.4 (2020/11/20) PMD:This for loop can be replaced by a foreach loop
655                for( final int stCol : stackCols ) {
656                        if( stCol == loc ) {
657                                rtn = true;
658                                break;                          // 6.3.9.1 (2015/11/27)
659                        }
660                }
661
662//              for( int nIndex=0; nIndex<stackCols.length ; nIndex++ ) {
663//                      if( stackCols[nIndex] == loc ) {
664//                              rtn = true;
665//                              break;                          // 6.3.9.1 (2015/11/27)
666//                      }
667//              }
668                return rtn;
669        }
670
671        /**
672         * 2つの日付の差を求めます。
673         * java.util.Date 型の日付 date1 - date2 が何日かを返します。
674         *
675         * @og.rev 5.5.8.3 (2012/11/17) 新規
676         *
677         * @param date1    日付
678         * @param date2    日付
679         * @return    2つの日付の差(日数 2-1) 同日なら0
680         */
681        public static int differenceDays(final Date date1,final Date date2) {
682                final long datetime1 = date1.getTime();
683                final long datetime2 = date2.getTime();
684                // 8.5.5.1 (2024/02/29) PMD 7.0.0 LocalVariableNamingConventions
685//              final long one_date_time = 1000 * 60 * 60 * 24L;                                // 1日のミリセカンド数
686//              return (int)((datetime2 - datetime1) / one_date_time);
687                final long oneDayMillisec = 1000 * 60 * 60 * 24L;                               // 1日のミリセカンド数
688                return (int)((datetime2 - datetime1) / oneDayMillisec);
689        }
690
691        /**
692         * 日付から枠番号を返す。
693         *
694         * @og.rev 5.5.8.3 (2012/11/17) 新規
695         *
696         * @param       date    日付(YYYY/MM/DD)
697         * @param       zoom    Zoom設定値
698         * @param       calFD   ヘッダ初日
699         *
700         * @return  枠番号
701         */
702        private int calNumber(final Date date, final String zoom, final Date calFD ) {
703                // 6.3.9.1 (2015/11/27) Found 'DD'-anomaly for variable(PMD)
704                final int rtn ;
705                if( zoom.equals( ViewStackTableParam.STACK_ZOOM_MONTH ) ){
706                        // 月だけは別の計算が必要
707                        final Calendar cal1 = Calendar.getInstance();
708                        cal1.setTime( calFD );
709                        final Calendar cal2 = Calendar.getInstance();
710                        cal2.setTime( date );
711                        rtn = ( cal2.get( Calendar.YEAR )-cal1.get( Calendar.YEAR ) ) * 12
712                                        + ( cal2.get( Calendar.MONTH ) - cal1.get( Calendar.MONTH ) );
713                }
714                else{
715                        final int diff = differenceDays( calFD, date );
716                        if( zoom.equals( ViewStackTableParam.STACK_ZOOM_WEEK )){
717                                rtn = diff/7;
718                        }
719                        else{
720                                rtn = diff;
721                        }
722                }
723                return rtn;
724        }
725
726        /**
727         * テーブル本体の作成。
728         *
729         * @og.rev 5.5.8.3 (2012/11/17) 繰り返し利用するため分離
730         * @og.rev 5.6.1.2 (2013/02/22) td終了が抜けていたので追加、キャパシティ対応
731         * @og.rev 6.1.0.0 (2014/12/26) 引数に StringBuffer を追加し、それに、データを追記していく。
732         * @og.rev 6.4.3.4 (2016/03/11) tdに、[カラム]が無いケースで、次の[カラム]のクラス属性が、前方すべてのtdにセットされてしまう対応。
733         * @og.rev 6.4.4.2 (2016/04/01) TableFormatterのタイプ別値取得処理の共通部をまとめる。
734         * @og.rev 6.4.5.0 (2016/04/08) メソッド変更( getColumnDbType(int) → getClassName(int) )
735         * @og.rev 6.8.1.1 (2017/07/22) ckboxTD変数は、&lt;td&gt; から &lt;td に変更します(タグの最後が記述されていない状態でもらう)。
736         * @og.rev 6.8.2.0 (2017/10/13) makeNthChildの廃止と、makeCheckboxで、個別にclass指定するように変更。
737         *
738         * @param       outBuf          データを追加するStringBuffer
739         * @param       row                     テーブルモデルのrow
740         * @param       stackRow        スタック行保存用
741         * @param       bgClrCnt        背景色カウンタ
742         * @param       blc                     チェックボックス用
743         * @param       costAry         コスト集計配列
744         * @param       cap                     能力
745         *
746         * @return      テーブル本体のHTML(入力の out オブジェクトそのもの)
747         * @og.rtnNotNull
748         */
749        private StringBuilder makeBodyTable( final StringBuilder outBuf, final int row, final int stackRow,
750                                                                                         final int bgClrCnt, final int blc, final double[] costAry, final String cap ) {
751                int bcCnt = bgClrCnt;           // 6.0.0.1 (2014/04/25) 引数を直接変更できなくする。
752                for( int i=0; i<bodyFormatsCount; i++ ) {
753                        final TableFormatter bodyFormat = bodyFormats[i];
754                        if( ! bodyFormat.isUse( row,getDBTableModel() ) ) { continue; }
755                        outBuf.append("<tbody").append( getBgColorCycleClass( bcCnt++,row ) );
756                        if( isNoTransition() ) {
757                                outBuf.append( getHiddenRowValue( row ) );
758                        }
759                        outBuf.append( STACK_TBODY )
760                                .append( STACK_ROW_PREFIX ).append( stackRow ).append( '\'' )
761                                .append( STACK_ID_PREFIX  ).append( stackRow ).append( "'>" )
762                                .append( bodyFormat.getTrTag() );
763
764                        //  No 欄そのものの作成判断追加
765                        if( isNumberDisplay() ) {
766                                final String ckboxTD = "<td" + bodyFormat.getRowspan();                         // 6.8.1.1 (2017/07/22)
767                                outBuf.append( makeCheckbox( ckboxTD,row,blc,true ) );                          // 6.8.2.0 (2017/10/13)
768                        }
769
770                        int cl = 0;
771                        for( ; cl<bodyFormat.getLocationSize(); cl++ ) {
772                                String fmt = bodyFormat.getFormat(cl);
773                                final int loc = bodyFormat.getLocation(cl);             // 3.5.5.0 (2004/03/12)
774                                if( ! bodyFormat.isNoClass() && loc >= 0 ) {    // 3.5.5.7 (2004/05/10)
775                                        // 6.4.3.4 (2016/03/11) tdに、[カラム]が無いケースで、次の[カラム]のクラス属性が、前方すべてのtdにセットされてしまう対応。
776                                        final int idx = fmt.lastIndexOf( "<td" );
777                                        if( idx >= 0 ) {        // matchしてるので、あるはず
778                                                final String tdclass = " class=\"" + getClassName(loc) + "\" ";                 // 6.4.5.0 (2016/04/08)
779                                                fmt = fmt.substring( 0,idx+3 ) + tdclass + fmt.substring( idx+3 ) ;
780                                        }
781                                }
782                                outBuf.append( fmt );
783
784                                // locがstackに入っていれば出力
785                                if( isStackClm(loc) ){
786                                        if( loc >= 0 ) {
787                                                // 6.4.4.2 (2016/04/01) 処理の共通部をまとめる。
788                                                outBuf.append( getTypeCaseValue( bodyFormat.getType(cl),row,loc ) );
789                                        }
790                                        else {
791                                                outBuf.append( bodyFormat.getSystemFormat(row,loc) );
792                                        }
793                                }
794                        }
795                        // 5.5.8.3 (2012/11/17)内部積上げの結果は出力場所の特定が難しいため一番最後尾にtd付きで出力しておきます
796                        if( innerStack ){
797                                // 8.5.4.2 (2024/01/12) PMD 7.0.0 InefficientStringBuffering 対応
798//                              outBuf.append("</td><td><div class='stackDivParent' capacity='"+ cap + "' style='width:100%; position:relative;'>"); // 5.6.1.2 (2013/02/22) td終了追加
799                                outBuf.append("</td><td><div class='stackDivParent' capacity='" )
800                                        .append( cap ).append( "' style='width:100%; position:relative;'>" ); // 5.6.1.2 (2013/02/22) td終了追加
801                                for( int cs=0; cs<costAry.length; cs++ ){
802                                        outBuf.append("<div class='stackDiv' style='position:absolute; top:0px;' num='")
803                                                .append( cs)
804                                                .append("' stackedCost='")
805                                                .append( costAry[cs] )
806                                                .append( "'></div>");
807                                }
808                                outBuf.append("</div>");
809                        }
810
811                        outBuf.append( bodyFormat.getFormat(cl) )
812                                .append("</tbody>").append( CR );
813                }
814                return outBuf;
815        }
816}