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変数は、<td> から <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}