001/* 002 * Copyright (c) 2009 The openGion Project. 003 * 004 * Licensed under the Apache License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.apache.org/licenses/LICENSE-2.0 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, 013 * either express or implied. See the License for the specific language 014 * governing permissions and limitations under the License. 015 */ 016package org.opengion.hayabusa.report2; 017 018import java.io.BufferedReader; 019import java.io.BufferedWriter; 020import java.io.File; 021// import java.io.FileInputStream; // 8.5.4.2 (2024/01/12) 022import java.io.FileNotFoundException; 023// import java.io.FileOutputStream; // 8.5.4.2 (2024/01/12) 024import java.io.IOException; 025// import java.io.OutputStreamWriter; // 8.5.4.2 (2024/01/12) 026import java.io.UnsupportedEncodingException; 027import java.nio.channels.FileChannel; 028import java.nio.charset.CharacterCodingException; // 6.3.1.0 (2015/06/28) 029import java.nio.file.Paths; // 8.5.4.2 (2024/01/12) PMD 7.0.0 AvoidFileStream 対応 030import java.nio.file.Files; // 8.5.4.2 (2024/01/12) PMD 7.0.0 AvoidFileStream 対応 031import java.nio.file.OpenOption; // 8.5.4.2 (2024/01/12) PMD 7.0.0 AvoidFileStream 対応 032import java.nio.file.StandardOpenOption; // 8.5.4.2 (2024/01/12) PMD 7.0.0 AvoidFileStream 対応 033import static java.nio.charset.StandardCharsets.UTF_8; // 8.5.4.2 (2024/01/12) PMD 7.0.0 AvoidFileStream 対応 034 035import java.util.List; 036import java.util.ArrayList; 037import java.util.Map; 038import java.util.HashMap; 039import java.util.concurrent.ConcurrentMap; // 6.4.3.3 (2016/03/04) 040import java.util.concurrent.ConcurrentHashMap; // 6.4.3.1 (2016/02/12) refactoring 041import java.util.Locale; 042import java.util.Set; 043 044import org.opengion.fukurou.system.OgCharacterException; // 6.5.0.1 (2016/10/21) 045import org.opengion.fukurou.model.NativeType; 046import org.opengion.fukurou.util.StringUtil; // 6.2.0.0 (2015/02/27) 047import org.opengion.fukurou.system.Closer; 048import org.opengion.fukurou.util.FileUtil; 049import org.opengion.fukurou.util.QrcodeImage; 050import org.opengion.hayabusa.common.HybsSystem; 051import org.opengion.hayabusa.common.HybsSystemException; 052import org.opengion.hayabusa.db.DBTableModel; // 6.1.1.0 (2015/01/17) 053import static org.opengion.fukurou.system.HybsConst.CR; // 6.1.0.0 (2014/12/26) 054import static org.opengion.fukurou.system.HybsConst.FS; // 8.0.3.0 (2021/12/17) 055import static org.opengion.fukurou.system.HybsConst.BUFFER_MIDDLE; // 6.1.0.0 (2014/12/26) refactoring 056 057import org.opengion.hayabusa.report2.TagParser.SplitKey; // 8.0.3.0 (2021/12/17) 058 059/** 060 * 指定されたパスに存在するODSの各XMLファイルをパースし、帳票定義及び 061 * 帳票データから書き換えます。 062 * 書き換えは読み取り先と同じファイルであるため、一旦読み取った各XMLを 063 * メモリ上に格納したからパース後のXMLファイルの書き込みを行います。 064 * 065 * パース対象となるファイルは以下の3つです。 066 * content.xml シートの中身を定義 067 * meta.xml メタデータを定義 068 * style.xml 帳票ヘッダーフッターを定義 069 * 070 * content.xmlのパース処理として、まずxmlファイルをシート+行単位に分解します。 071 * その後、分解された行毎に帳票データを埋め込み、出力先のXMLに書き込みを行います。 072 * 書き込みは行単位に行われます。 073 * 074 * また、Calcの特性として、関数の引数に不正な引数が指定された場合、(Text関数の 075 * 引数にnullが指定された場合等)、エラー:XXXという文字が表示されます。 076 * ここでは、これを回避するため、全ての関数にisError関数を埋め込み、エラー表示を 077 * 行わないようにしています。 078 * 079 * @og.group 帳票システム 080 * 081 * @version 4.0 082 * @author Hiroki.Nakamura 083 * @since JDK1.6 084 */ 085class OdsContentParser { 086 // 8.5.4.2 (2024/01/12) PMD 7.0.0 AvoidFileStream 対応 087 private static final OpenOption[] CREATE = { StandardOpenOption.WRITE , StandardOpenOption.CREATE , StandardOpenOption.TRUNCATE_EXISTING }; 088 private static final OpenOption[] APPEND = { StandardOpenOption.WRITE , StandardOpenOption.CREATE , StandardOpenOption.APPEND }; 089 private static final OpenOption[] READ = { StandardOpenOption.READ }; 090 091 //======== content.xmlのパースで使用 ======================================== 092 /* シートの開始終了タグ */ 093 private static final String BODY_START_TAG = "<table:table "; 094 private static final String BODY_END_TAG = "</table:table>"; 095 096 /* 行の開始終了タグ */ 097 private static final String ROW_START_TAG = "<table:table-row "; 098 099 /* ページエンドカットの際に、行を非表示にするためのテーブル宣言 */ 100 private static final String ROW_START_TAG_INVISIBLE = "<table:table-row table:visibility=\"collapse\" "; 101 102 /* セルの開始タグ */ 103 private static final String TABLE_CELL_START_TAG = "<table:table-cell"; 104 private static final String TABLE_CELL_END_TAG = "</table:table-cell>"; 105 106 /* シート名を取得するための開始終了文字 */ 107 private static final String SHEET_NAME_START = "table:name=\""; 108 109 /* オブジェクトの終了位置(シート名)を見つけるための開始文字 */ 110 private static final String OBJECT_SEARCH_STR = "table:end-cell-address=\""; 111 112 /* 印刷範囲指定の開始終了文字 */ 113 // 4.3.3.5 (2008/11/08) 空白ページ対策で追加 114 private static final String PRINT_RANGE_START = "table:print-ranges=\""; 115// private static final String PRINT_RANGE_END = "\""; 116 private static final String END_KEY = "\""; // 8.0.3.0 (2021/12/17) 117 118 /* 表紙印刷用のページ名称 */ 119 private static final String FIRST_PAGE_NAME = "FIRST"; 120 121 /* シートブレイク用のキー 5.1.7.0 (2010/06/01) */ 122 private static final String SHEET_BREAK = "SHEETBREAK"; 123 124 /* 変数定義の開始終了文字及び区切り文字 */ 125 private static final String VAR_START = "{@"; 126 private static final String VAR_END = "}"; 127// private static final String VAR_CON = "_"; // 8.0.3.0 (2021/12/17) '_' で、キーと行番号の分離を、インナークラス化します。 128 129 /* ページエンドカットのカラム文字列 */ 130 private static final String PAGE_END_CUT = "PAGEENDCUT"; 131 132 /* ページブレイクのカラム文字列 */ 133 private static final String PAGE_BREAK = "PAGEBREAK"; 134 135 /* ページ番号出力用文字列 5.1.6.0 (2010/05/01) */ 136 private static final String PAGE_NO= "PAGENO"; 137 138 /* 行番号出力用文字列 5.1.6.0 (2010/05/01) */ 139 private static final String ROW_NO= "ROWNO"; 140 141 /* 画像のリンクを取得するための開始終了文字 */ 142 private static final String DRAW_IMG_START_TAG = "<draw:image xlink:href=\""; 143 private static final String DRAW_IMG_END_TAG = "</draw:image>"; 144// private static final String DRAW_IMG_HREF_END = "\""; 145 146 /* 画像ファイルを保存するためのパス */ 147 private static final String IMG_DIR = "Pictures"; 148 149 /* QRコードを処理するためのカラム名 */ 150 private static final String QRCODE_PREFIX = "QRCODE."; 151 152 /* 作成したQRコードのフォルダ名及び拡張子 */ 153 private static final String QRCODE_FILETYPE = ".png"; 154 155 /* 7.0.5.1 (2019/09/27) QRコードのパラメータをシステムリソースで設定できるようにします(ただし、staticとします) */ 156 private static final int QR_VERSION = HybsSystem.sysInt( "REPORT_QR_VERSION" ); // 7.0.5.1 (2019/09/27) バージョン 157// private static final char QR_ENCMODE_CH = HybsSystem.sys( "REPORT_QR_ENCMODE" ).charAt(0); // 7.0.5.1 (2019/09/27) エンコードモード 8.4.1.0 (2023/02/10) Delete 158 private static final char QR_ERRCRCT_CH = HybsSystem.sys( "REPORT_QR_ERRCRCT" ).charAt(0); // 7.0.5.1 (2019/09/27) エラー訂正レベル 159 private static final String QR_IMAGE_TYPE = "PNG"; // 7.0.5.1 (2019/09/27) 出力イメージのタイプ(PNG/JPEG) 160 private static final int QR_PIXEL = HybsSystem.sysInt( "REPORT_QR_PIXEL" ); // 7.0.5.1 (2019/09/27) 1セル辺りの塗りつぶしピクセル数 161// private static final QrcodeImage.EncMode QR_ENCMODE = QrcodeImage.EncMode.get( QR_ENCMODE_CH ); // 7.0.5.1 (2019/09/27) 8.4.1.0 (2023/02/10) Delete 162 private static final QrcodeImage.ErrCrct QR_ERRCRCT = QrcodeImage.ErrCrct.get( QR_ERRCRCT_CH ); // 7.0.5.1 (2019/09/27) 163 private static final String QR_TXT_ENC = HybsSystem.sys( "REPORT_QR_TEXT_ENCODE" ); // 7.2.3.0 (2020/04/10) 帳票出力のQRコード作成時のテキストのエンコード指定 164 165 /* 4.3.3.5 (2008/11/08) 動的に画像を入れ替えるためのパスを記述するカラム名 */ 166 private static final String IMG_PREFIX = "IMG."; 167 168 /* ファンクション定義を見つけるための開始終了文字 */ 169 private static final String OOOC_FUNCTION_START = "oooc:="; 170 private static final String OOOC_FUNCTION_START_3 = "of:="; // 4.3.7.2 (2009/06/15) ODS仕様変更につき追加 171 private static final String OOOC_FUNCTION_END = ")\" "; 172 173 /* セル内の改行を定義する文字列 5.0.2.0 (2009/11/01) */ 174 private static final String OOO_CR = "</text:p><text:p>"; 175 176 /* グラフオブジェクトの書き換えを行うための開始終了文字 5.1.8.0 (2010/07/01) */ 177 private static final String GRAPH_START_TAG = "<draw:frame "; 178 private static final String GRAPH_END_TAG = "</draw:frame>"; 179 180 /* グラフの範囲指定の書き換えを行うための開始終了文字 5.1.8.0 (2010/07/01) */ 181 private static final String GRAPH_UPDATE_RANGE_START = "draw:notify-on-update-of-ranges=\""; 182// private static final String GRAPH_UPDATE_RANGE_END = "\""; 183 184 /* グラフのオブジェクトへのリンクの書き換えを行うための開始終了文字 5.1.8.0 (2010/07/01) */ 185 private static final String GRAPH_HREF_START = "xlink:href=\"./"; 186// private static final String GRAPH_HREF_END = "\""; 187 private static final String GRAPH_OBJREPL = "ObjectReplacements"; 188 189 /* グラフのオブジェクト毎のcontent.xmlに記述してあるシート名の書き換えを行うための開始終了文字 5.1.8.0 (2010/07/01) */ 190 private static final String GRAPH_CONTENT_START = "-address=\""; 191// private static final String GRAPH_CONTENT_END = "\""; 192 193 /* 生成したグラフのオブジェクトをMETA-INF/manifest.xmlに登録するための開始終了文字列 5.1.8.0 (2010/07/01) */ 194 private static final String MANIFEST_START_TAG = "<manifest:file-entry "; 195 private static final String MANIFEST_END_TAG = "/>"; // XML なので、このまま。 196 197 /* 数値タイプ置き換え用の文字列 5.1.8.0 (2010/07/01) */ 198 // 8.5.4.2 (2024/01/12) PMD 7.0.0 LongVariable 対応 199// private static final String TABLE_CELL_STRING_TYPE = "office:value-type=\"string\""; 200// private static final String TABLE_CELL_FLOAT_TYPE = "office:value-type=\"float\""; 201// private static final String TABLE_CELL_FLOAT_VAL_START = "office:value=\""; 202 private static final String OFC_VALUE_STRING = "office:value-type=\"string\""; 203 private static final String OFC_VALUE_FLOAT = "office:value-type=\"float\""; 204 private static final String OFC_VALUE_START = "office:value=\""; 205// private static final String TABLE_CELL_FLOAT_VAL_END = "\""; 206 207 /* テキスト文字列の開始終了文字列 5.1.8.0 (2010/07/01) */ 208 private static final String TEXT_START_TAG = "<text:p>"; 209 private static final String TEXT_END_TAG = "</text:p>"; 210 211 /* コメント(アノテーション)を処理するためのカラム名 5.1.8.0 (2010/07/01) */ 212 private static final String ANNOTATION_PREFIX = "ANO."; 213 private static final String TEXT_START_ANO_TAG = "<text:p"; // アノテーションの場合の置き換えを 214 private static final String TEXT_START_END_ANO_TAG = ">"; // アノテーションの場合の置き換えを 215 216 /* コメント(アノテーション)の開始・終了タグ 5.1.8.0 (2010/07/01) */ 217 private static final String ANNOTATION_START_TAG = "<office:annotation"; 218 private static final String ANNOTATION_END_TAG = "</office:annotation>"; 219 220 /* オブジェクトを検索するための文字列 5.1.8.0 (2010/07/01) */ 221// private static final String DRAW_START_KEY = "<draw:"; 222// private static final String DRAW_END_KEY = "</draw:"; 223 private static final String DRAW_START_TAG = "<draw:"; 224 private static final String DRAW_END_TAG = "</draw:"; 225 226 /* シートの開始終了タグ 5.2.2.0 (2010/11/01) */ 227 private static final String STYLE_START_TAG = "<style:style "; 228 private static final String STYLE_END_TAG = "</style:style>"; 229 230 /* シート名称 5.2.2.0 (2010/11/01) */ 231 private static final String STYLE_NAME_START_TAG = "style:name=\""; 232// private static final String STYLE_NAME_END_TAG = "\""; 233 234 /* テーブル内シート名称 5.2.2.0 (2010/11/01) */ 235 // 8.5.4.2 (2024/01/12) PMD 7.0.0 LongVariable 対応 236// private static final String TABLE_STYLE_NAME_START_TAG = "table:style-name=\""; 237 private static final String TABLE_STYLE_NAME_START = "table:style-name=\""; 238// private static final String TABLE_STYLE_NAME_END_TAG = "\""; // 5.6.3.1 (2013/04/05) 239 240 /* LibreOffice対応(数字が文字扱いの対応) 8.1.2.1 (2022/03/25) */ 241 private static final String XMLNS_CALCEXT = "xmlns:calcext=\""; 242 //private static final String TABLE_CELL_CALCEXT = "calcext:value-type=\""; // 一時的にコメント化 243 244 //=========================================================================== 245 246 //======== meta.xmlのパースで使用 =========================================== 247 /* 総シートカウント数 */ 248 private static final String TABLE_COUNT_START_TAG = "meta:table-count=\""; 249// private static final String TABLE_COUNT_END_TAG = "\""; 250 251 /* 総セルカウント数 */ 252 private static final String CELL_COUNT_START_TAG = "meta:cell-count=\""; 253// private static final String CELL_COUNT_END_TAG = "\""; 254 255 /* 総オブジェクトカウント数 */ 256 private static final String OBJECT_COUNT_START_TAG = "meta:object-count=\""; 257// private static final String OBJECT_COUNT_END_TAG = "\""; 258 //=========================================================================== 259 260 /* 261 * 処理中の行番号の状態 262 * NORMAL : 通常 263 * LASTROW : 最終行 264 * OVERFLOW : 終了 265 */ 266 private static final int NORMAL = 0; 267 private static final int LASTROW = 1; 268 private static final int OVERFLOW = 2; 269 private int status = NORMAL; 270 271 /* 272 * 各雛形ファイルを処理する際の基準となる行数 273 * 初期>0 2行({@XXX_1}まで)処理後>2 ・・・ 274 * 各雛形で定義されている行番号 + [baseRow] の値がDBTableModel上の行番号に相当する 275 * currentMaxRowは各シート処理後の[baseRow]と同じ 276 */ 277 private int currentBaseRow ; 278 private int currentMaxRow ; 279 280 /* 処理したページ数 */ 281 private int pages ; 282 283 /* 処理行がページエンドカットの対象かどうか */ 284 private boolean isPageEndCut ; // 4.3.1.1 (2008/08/23) ローカル変数化 285 286 /* ページブレイクの処理中かどうか */ 287 private boolean isPageBreak ; 288 289 /* XML宣言の文字列。各XMLで共通なのでクラス変数として定義 */ 290 private String xmlHeader ; 291 292 /* シートブレイク対象かどうか 5.1.7.0 (2010/06/01) */ 293 private int sheetBreakClm = -1; 294 295 /* シート名カラム 5.7.6.2 (2014/05/16) */ 296 private int sheetNameClm = -1; // 今は、ページブレイクカラムと同じカラムを使用しています。 297 298 /* シートのヘッダー部分の再パースを行うかどうか 5.2.2.0 (2010/11/01) */ 299 private boolean isNeedsReparse ; 300 301 /* ページ名のマッピング(元のシート名に対する新しいシート名) 5.2.2.0 (2010/11/01) */ 302 /** 6.4.3.1 (2016/02/12) PMD refactoring. HashMap → ConcurrentHashMap に置き換え。 */ 303 private final ConcurrentMap<String,List<String>> pageNameMap = new ConcurrentHashMap<>(); 304 305 /* ページ名に依存しているスタイル名称のリスト 5.2.2.0 (2010/11/01) */ 306 private final List<String> repStyleList = new ArrayList<>(); 307 308 /* manifest.xmlに追加が必要なオブジェクトのマップ 5.3.1.0 (2011/01/01) */ 309 /** 6.4.3.1 (2016/02/12) PMD refactoring. HashMap → ConcurrentHashMap に置き換え。 */ 310 private final ConcurrentMap<String,String> addObjMap = new ConcurrentHashMap<>(); 311 312 private final ExecQueue queue; 313 private final String path; 314 315 private final boolean useChangeType ; // 6.8.3.1 (2017/12/01) 316 317// /* 8.0.3.0 (2021/12/17) ods→xlsx変換時のシート毎の行数 */ 318// 【保留】private final List<Integer> sheetRows = new ArrayList<>(); 319 320 /** 321 * コンストラクタ 322 * 323 * @og.rev 5.1.2.0 (2010/01/01) 処理した行数をQueueオブジェクトから取得(シート数が256を超えた場合の対応) 324 * @og.rev 6.8.3.1 (2017/12/01) ローカルリソースの文字型⇒数値型変換の処理の有無 325 * 326 * @param qu ExecQueueオブジェクト 327 * @param pt パス 328 */ 329 /* default */ OdsContentParser( final ExecQueue qu, final String pt ) { 330 queue = qu; 331 path = pt; 332 333 currentBaseRow = queue.getExecRowCnt(); 334 useChangeType = !queue.isFglocal() || HybsSystem.sysBool( "REPORT_USE_CHANGETYPE" ); // 6.8.3.1 (2017/12/01) 335 } 336 337 /** 338 * パース処理を実行します。(1) 339 * 340 * @og.rev 5.2.2.0 (2010/11/01) 条件付書式対応 341 * @og.rev 5.3.1.0 (2011/01/01) OpenOffice3.2対応 追加した画像をmanifest.xmlに登録する 342 */ 343 public void exec() { 344 /* 345 * 雛形ヘッダーフッターの定義 346 * OOoではページ毎にヘッダーフッターが設定できないよう。 347 * なので、全てヘッダー扱いで処理 348 */ 349 execStyles(); // (2) 350 351 /* 中身の変換 */ 352 execContent(); // (3) 353 354 /* ヘッダー部分にシート情報がある場合に書き換え */ 355 if( isNeedsReparse ) { 356 /* ヘッダーファイルの再パース */ 357 execContentHeader(); // (4) 358 /* ヘッダーファイルとそれ以降のファイルの連結 */ 359 execMergeContent(); // (5) 360 } 361 362 /* メタデータの変換 */ 363 execMeta(); // (6) 364 365 // 5.3.1.0 (2011/01/01) OpenOffice3.2対応 追加した画像をmanifest.xmlに登録する 366 /* 追加した画像、オブジェクトをmanifest.xmlに追加 */ 367 if( !addObjMap.isEmpty() ) { // 6.1.1.0 (2015/01/17) refactoring 368 execManifest(); // (7) 369 } 370 } 371 372// /** 373// * 【保留】シート毎の行数をリストで返します。 374// * 375// * @og.rev 8.0.3.0 (2021/12/17) ods→xlsx変換時のシート毎の行数 376// * 377// * @return シート毎の行数リスト 378// */ 379// public List<Integer> getSheetRowsList() { 380// return sheetRows; 381// } 382 383 /** 384 * 帳票処理キューを元に、content.xmlを書き換えます。(3) 385 * まず、XMLを一旦メモリ上に展開した後、シート単位に分解し、データの埋め込みを行います。 386 * 387 * @og.rev 4.3.0.0 (2008/07/18) ページ数が256を超えた場合のエラー処理 388 * @og.rev 5.0.0.2 (2009/09/15) LINECOPY機能追加 389 * @og.rev 5.1.2.0 (2010/01/01) 処理したページ数、行数をQueueオブジェクトにセット(シート数が256を超えた場合の対応) 390 * @og.rev 5.1.7.0 (2010/06/01) 複数シート対応 391 * @og.rev 5.2.2.0 (2010/11/01) 条件付書式対応 392 * @og.rev 5.7.6.2 (2014/05/16) PAGEBREAKカラムの値を、シート名として使う場合の処理追加 393 * @og.rev 5.7.6.3 (2014/05/23) PAGEBREAKカラムの値を、シート名として使う場合の、FIRST雛形への適用 394 * @og.rev 6.1.1.0 (2015/01/17) 内部ロジックの見直し。queue.getBody() を、ローカル変数で定義他。 395 * @og.rev 6.4.3.3 (2016/03/04) 気になったので、sheet.getOrigSheetName() を、変数に入れておきます。 396 * @og.rev 8.0.3.0 (2021/12/17) TagParser.tag2Array を使用します。 397 * @og.rev 8.1.2.1 (2022/03/25) LibreOffice対応(数字が文字扱いの対応) 398 */ 399 private void execContent() { 400 final String fileName = path + "content.xml"; 401 final String content = readOOoXml( fileName ); 402 // ファイルを解析し、シート+行単位に分解 403// final String[] tags = tag2Array( content, BODY_START_TAG, BODY_END_TAG ); 404 final String[] tags = TagParser.tag2Array( content, BODY_START_TAG, BODY_END_TAG ); // 405 406 // 5.2.2.0 (2010/11/01) 条件付書式対応 407 // content.xmlのヘッダー部分のみ書き出し 408// final String contentHeader = tags[0]; // 8.1.2.1 (2022/03/25) Modify 409 String contentHeader = tags[0]; 410 411 // LibreOffice対応 8.1.2.1 (2022/03/25) 412 final int calcSt = contentHeader.indexOf( XMLNS_CALCEXT ); 413 if( calcSt >= 0 ) { 414 // 8.5.4.2 (2024/01/12) PMD 7.0.0 ConfusingTernary 対応 415 final int calcEd = contentHeader.indexOf( '"', calcSt + XMLNS_CALCEXT.length() ); 416 contentHeader = contentHeader.replace( contentHeader.substring( calcSt - 1, calcEd + 1 ) , "" ); 417 } 418 419 BufferedWriter bw = null; 420 try { 421 bw = getWriter( fileName ); 422 bw.write( xmlHeader ); 423 bw.write( '\n' ); 424 bw.write( contentHeader ); 425 bw.flush(); 426 } 427 catch( final IOException ex ) { 428 queue.addMsg( "[ERROR]PARSE:error occurer while content.xml(header) " + fileName ); 429 throw new HybsSystemException( ex ); 430 } 431 finally { 432 Closer.ioClose( bw ); 433 bw = null; 434 } 435 436 final String contentFooter = tags[1]; 437 438 final List<OdsSheet> firstSheets = new ArrayList<>(); 439 final Map<String, OdsSheet> sheetMap = new HashMap<>(); 440 441 final DBTableModel bodyModel = queue.getBody(); // 6.1.1.0 (2015/01/17) 442// final int rowCount = bodyModel.getRowCount(); // 6.1.1.0 (2015/01/17) 443 444 OdsSheet defaultSheet = null; 445 for( int i=2; i<tags.length; i++ ) { 446 final OdsSheet sheet = new OdsSheet(); 447 448 // sheet.analyze( tags[i] ); 449// sheet.analyze( tags[i],rowCount ); // 6.1.1.0 (2015/01/17) ループから出す。 450 final String[] bodyTypes = queue.getBodyTypes(); // 8.0.3.0 (2021/12/17) 451 sheet.analyze( tags[i],bodyTypes ); // 8.0.3.0 (2021/12/17) 452 // 5.1.7.0 (2010/06/01) 複数シート対応 453 final String sheetName = sheet.getSheetName(); 454 if( sheetName.startsWith( FIRST_PAGE_NAME ) ) { 455 firstSheets.add( sheet ); 456 } 457 else { 458 sheetMap.put( sheetName, sheet ); 459 // 一番初めに見つかった表紙以外のシートをデフォルトシートとして設定 460 if( defaultSheet == null ) { 461 defaultSheet = sheet; 462 } 463 } 464 465 // 6.4.3.3 (2016/03/04) 気になったので、sheet.getOrigSheetName() を、変数に入れておきます。 466 final String orgShtName = sheet.getOrigSheetName(); 467 468 // 5.2.2.0 (2010/11/01) 条件付書式対応 469 if( !isNeedsReparse && contentHeader.indexOf( "=\"" + orgShtName + "." ) >= 0 ) { 470 isNeedsReparse = true; 471 } 472 473 // 5.2.2.0 (2010/11/01) 条件付書式対応 474 pageNameMap.put( orgShtName, new ArrayList<>() ); 475 } 476 477 // content.xmlの書き出し 478 try { 479 // 5.2.2.0 (2010/11/01) 条件付書式対応 480 if( isNeedsReparse ) { 481 // ヘッダーを再パースする場合は、ボディ部分を 482 // content.xml.tmpに書き出して、後でマージする 483 bw = getWriter( fileName + ".tmp" ); 484// getRepStyleList( contentHeader ); 485 makeRepStyleList( contentHeader ); // 8.5.4.2 (2024/01/12) PMD 7.0.0 LinguisticNaming 486 } 487 else { 488 // ヘッダーを再パースしない場合は、ボディ部分を 489 // content.xml追加モードで書き込みする 490 bw = getWriter( fileName, true ); 491 } 492 493 // 5.7.6.3 (2014/05/23) PAGEBREAKカラムの値を、シート名として使うかどうか。 494 if( queue.isUseSheetName() ) { 495 sheetNameClm = bodyModel.getColumnNo( PAGE_BREAK, false ); // 6.1.1.0 (2015/01/17) 496 } 497 498 final int rowCount = bodyModel.getRowCount(); // 6.1.1.0 (2015/01/17) 499 500 // 表紙ページの出力 501 if( queue.getExecPagesCnt() == 0 ) { 502 for( final OdsSheet firstSheet : firstSheets ) { 503 if( currentBaseRow >= rowCount ) { // 6.1.1.0 (2015/01/17) ループから出す。 504 break; 505 } 506 writeParsedSheet( firstSheet, bw ); 507 } 508 } 509 510 // 5.1.7.0 (2010/06/01) 複数シート対応 511 sheetBreakClm = bodyModel.getColumnNo( SHEET_BREAK, false ); // 6.1.1.0 (2015/01/17) 512 513 // 5.7.6.3 (2014/05/23) 表紙ページも、PAGEBREAKカラムの値を、シート名として使えるようにする。 514 515 // 繰り返しページの出力 516 while( currentBaseRow < rowCount ) { // 6.1.1.0 (2015/01/17) ループから出す。 517 // 4.3.0.0 (2008/07/18) ページ数が256を超えた場合にエラーとする 518 // 5.1.2.0 (2010/01/01) 256シートを超えた場合の対応 519 if( pages >= ExecQueue.MAX_SHEETS_PER_FILE ) { 520 queue.setEnd( false ); 521 break; 522 } 523 524 OdsSheet sheet = null; 525 if( sheetBreakClm >= 0 ) { 526 final String sheetName = bodyModel.getValue( currentBaseRow, sheetBreakClm ); // 6.1.1.0 (2015/01/17) 527 if( sheetName != null && sheetName.length() > 0 ) { 528 sheet = sheetMap.get( sheetName ); 529 } 530 } 531 if( sheet == null ) { sheet = defaultSheet; } 532 533 writeParsedSheet( sheet, bw ); 534 } 535 536 // 5.1.2.0 (2010/01/01) 256シートを超えた場合の対応 537 queue.addExecPageCnt( pages ); 538 queue.setExecRowCnt( currentBaseRow ); 539 540 // フッター 541 bw.write( contentFooter ); 542 bw.flush(); 543 } 544 catch( final IOException ex ) { 545 queue.addMsg( "[ERROR]PARSE:error occurer while write Parsed Sheet " + fileName ); 546 throw new HybsSystemException( ex ); 547 } 548 finally { 549 Closer.ioClose( bw ); 550 } 551 } 552 553 /** 554 * シート単位にパースされた文書データを書き込みます 555 * 出力されるシート名には、ページ番号と基底となる行番号をセットします。 556 * 557 * @og.rev 4.2.4.0 (2008/07/04) 行単位にファイルに書き込むように変更 558 * @og.rev 5.2.0.0 (2010/09/01) 表紙の場合は、BODY部分のデータが含まれていなくてもOK 559 * @og.rev 5.2.1.0 (2010/10/01) シート名定義対応 560 * @og.rev 5.2.2.0 (2010/11/01) 条件付書式対応 561 * @og.rev 5.7.6.2 (2014/05/16) PAGEBREAKカラムの値を、シート名として使う場合の処理追加 562 * @og.rev 5.7.6.3 (2014/05/23) FIRST雛形シート名が、FIRST**** の場合、**** 部分をシート名に使う。 563 * @og.rev 6.4.3.3 (2016/03/04) 気になったので、sheet.getOrigSheetName() を、変数に入れておきます。 564 * @og.rev 7.3.0.1 (2021/01/22) 画像ファイルの置き方によって、ヘッダー部に {@IMG.XXX} が書かれることがある。 565 * @og.rev 8.0.3.0 (2021/12/17) COPY_LINE機能の追加 566 * @og.rev 8.3.0.2 (2022/08/20) 条件付書式のシート名変換対応(ただし、xmlns:calcext 定義なので、現在未使用)(問合・トラブル 43100-220819-01) 567 * 568 * @param sheet シート 569 * @param bw BufferedWriterオブジェクト 570 * @throws IOException 書き込みに失敗した場合 571 */ 572 private void writeParsedSheet( final OdsSheet sheet, final BufferedWriter bw ) throws IOException { 573 // シート名 574 String outputSheetName = null; 575 576 // 5.7.6.2 (2014/05/16) PAGEBREAKカラムの値を、シート名として使うかどうか。 577 if( sheetNameClm >= 0 ) { 578 final String sheetName = queue.getBody().getValue( currentBaseRow, sheetNameClm ); 579 if( sheetName != null ) { 580 outputSheetName = sheetName; 581 } 582 } 583 584 // 5.7.6.3 (2014/05/23) FIRST雛形シート名が、FIRST**** の場合、**** 部分をシート名に使う。 585 if( outputSheetName == null ) { 586 String sheetName = sheet.getSheetName(); 587 if( sheetName.startsWith( FIRST_PAGE_NAME ) ) { 588 sheetName = sheetName.substring( FIRST_PAGE_NAME.length() ).trim(); 589 // 小細工。"FIRST_****" の場合は、"_" を外す。長さ0判定の前に行う。 590 if( StringUtil.startsChar( sheetName , '_' ) ) { // 6.2.0.0 (2015/02/27) 1文字 String.startsWith 591 sheetName = sheetName.substring( 1 ); 592 } 593 594 // 長さ0の場合(例えば、FIRSTだけとか)は、設定しない。 595 if( sheetName.length() > 0 ) { outputSheetName = sheetName; } 596 } 597 } 598 599 // 従来からあるシート名の値 600 if( outputSheetName == null ) { 601 if( sheet.getConfSheetName() == null ) { 602 outputSheetName = "Page" + ( queue.getExecPagesCnt() + pages ) + "_Row" + currentBaseRow ; 603 } 604 else { 605 outputSheetName = sheet.getConfSheetName() + ( queue.getExecPagesCnt() + pages + 1 ) ; 606 } 607 } 608 // ページブレイク変数を初期化 609 isPageBreak = false; 610 611 // 6.4.3.3 (2016/03/04) 気になったので、sheet.getOrigSheetName() を、変数に入れておきます。 612 final String orgShtName = sheet.getOrigSheetName(); 613 614 // シートのヘッダー部分を書き込み(シート名も書き換え) 615 String headerStr = sheet.getHeader().replace( SHEET_NAME_START + orgShtName, SHEET_NAME_START + outputSheetName ); 616 // 印刷範囲指定部分のシート名を変更 617 // 4.3.3.5 (2008/11/08) 空白ページ出力の対策。印刷範囲のシート名書き換えを追加 618 final int printRangeStart = headerStr.indexOf( PRINT_RANGE_START ); 619 if( printRangeStart >= 0 ) { 620// final int printRangeEnd = headerStr.indexOf( PRINT_RANGE_END, printRangeStart + PRINT_RANGE_START.length() ); 621 final int printRangeEnd = headerStr.indexOf( END_KEY, printRangeStart + PRINT_RANGE_START.length() ); 622 String rangeStr = headerStr.substring( printRangeStart, printRangeEnd ); 623 rangeStr = rangeStr.replace( orgShtName, outputSheetName ); 624 headerStr = headerStr.substring( 0, printRangeStart ) + rangeStr + headerStr.substring( printRangeEnd ); 625 } 626 627 // 7.3.0.1 (2021/01/22) 画像ファイルの置き方によって、ヘッダー部に {@IMG.XXX} が書かれることがある。 628 writeParsedRow( headerStr, bw, orgShtName, outputSheetName ); 629// bw.write( headerStr ); 630 631 // 8.0.3.0 (2021/12/17) COPYLINE機能の追加 シートのボディ部分を書き込み 632 // 【保留】sheetRows.add( sheet.getRowCnt() ); 633 for( int i=0; i<sheet.getRowCnt(); i++ ) { // i はシートの行数 634 final String row = sheet.getRow( i,currentBaseRow ); 635 writeParsedRow( row, bw, orgShtName, outputSheetName ); 636 } 637 638// // シートのボディ部分を書き込み 639// for( final String row : sheet.getRows() ) { 640// writeParsedRow( row, bw, orgShtName, outputSheetName ); 641// } 642 643// final String[] rows = sheet.getRows(); 644// for( int i=0; i<rows.length; i++ ) { 645// // 4.3.4.4 (2009/01/01) 646// writeParsedRow( rows[i], bw, orgShtName, outputSheetName ); 647// } 648 // {@XXXX}が埋め込まれていない場合はエラー 649 // 5.2.0.0 (2010/09/01) 表紙の場合は、BODY部分のデータが含まれていなくてもOK 650 if( currentBaseRow == currentMaxRow && !orgShtName.startsWith( FIRST_PAGE_NAME ) ) { 651 queue.addMsg( "[ERROR]PARSE:No Data defined on Template ODS(" + queue.getListId() + ")" ); 652 throw new HybsSystemException(); 653 } 654 currentBaseRow = currentMaxRow; 655 656 // 8.3.0.2 (2022/08/20) 条件付書式のシート名変換対応(ただし、xmlns:calcext 定義なので、現在未使用) 657 // シートのフッター部分を書き込み 658// bw.write( sheet.getFooter() ); 659 String footerStr = sheet.getFooter(); 660 final String sheetNew = outputSheetName; // final 化しないと、無名クラスで使えない。 661 662 // シート名の置き換えの検索に、GRAPH_CONTENT_START を使っているが、"-address=" を使いたかっただけ。 663 if( footerStr.indexOf( GRAPH_CONTENT_START ) >= 0 ) { 664 footerStr = new TagParser() { 665 /** 666 * 開始タグから終了タグまでの文字列の処理を定義します。 667 * 668 * この実装では、何も処理を行いません。(切り出した文字列はアペンドされません) 669 * サブクラスでオーバーライドして実際の処理を実装して下さい。 670 * 671 * @param str 開始タグから終了タグまでの文字列(開始タグ・終了タグを含む) 672 * @param buf 出力を行う文字列バッファ 673 * @param offset 終了タグのオフセット(ここでは使っていません) 674 */ 675 @Override 676 protected void exec( final String str, final StringBuilder buf, final int offset ) { 677 buf.append( str.replace( orgShtName, sheetNew ) ); 678 } 679 }.doParse( footerStr, GRAPH_CONTENT_START, END_KEY ); 680 } 681 682 bw.write( footerStr ); 683 684 pages++; 685 686 // 5.2.2.0 (2010/11/01) 条件付書式対応 687 pageNameMap.get( orgShtName ).add( outputSheetName ); 688 } 689 690 /** 691 * 行単位にパースされた文書データを書き込みます。 692 * 693 * @og.rev 4.2.3.1 (2008/06/19) 関数エラーを表示させないため、ISERROR関数を埋め込み 694 * @og.rev 4.2.4.0 (2008/07/04) 行単位にファイルに書き込むように変更 695 * @og.rev 4.3.0.0 (2008/07/17) {@と}の整合性チェック追加 696 * @og.rev 4.3.0.0 (2008/07/22) 行最後の{@}整合性エラーハンドリング追加 697 * @og.rev 4.3.3.5 (2008/11/08) 画像の動的な入れ替えに対応 698 * @og.rev 5.1.8.0 (2010/07/01) パース方法の内部実装変更 699 * @og.rev 5.2.2.0 (2010/11/01) 条件付書式対応 700 * @og.rev 5.4.2.0 (2011/12/01) ページブレイク、シートブレイク中でもページエンドカットが適用されるようにする。 701 * @og.rev 5.6.3.1 (2013/04/05) 条件付書式の属性終了文字対応 702 * @og.rev 6.8.3.1 (2017/12/01) ローカルリソースの文字型⇒数値型変換の処理の有無 703 * @og.rev 8.0.3.0 (2021/12/17) "\n" 文字列を実際の改行コードに置換して設定します。 704 * 705 * @param row 行データ 706 * @param bw BufferedWriterオブジェクト 707 * @param sheetNameOrig 元シート名 708 * @param sheetNameNew 新シート名 709 * @throws IOException 書き込みに失敗した場合 710 */ 711 private void writeParsedRow( final String row, final BufferedWriter bw, final String sheetNameOrig, final String sheetNameNew ) throws IOException { 712 isPageEndCut = false; 713 714 String rowStr = new TagParser() { 715 /** 716 * 開始タグから終了タグまでの文字列の処理を定義します。 717 * 718 * @param str 開始タグから終了タグまでの文字列(開始タグ・終了タグを含む) 719 * @param buf 出力を行う文字列バッファ 720 * @param offset 終了タグのオフセット(ここでは使っていません) 721 */ 722 @Override 723 protected void exec( final String str, final StringBuilder buf, final int offset ) { 724 final String key = TagParser.checkKey( str, buf ); 725 726 // 4.3.0.0 (2008/07/15) "<"が入っていた場合には{@不整合}エラー 727 if( key.indexOf( '<' ) >= 0 ){ 728 queue.addMsg( "[ERROR]PARSE:{@と}の整合性が不正です。" + CR 729 + "変数内の特定の文字列に書式設定がされている可能性があります。キー=" + key ); 730 throw new HybsSystemException(); 731 } 732 733 // QRコードの処理、処理後はoffsetが進むため、offsetを再セット 734 if( key.startsWith( QRCODE_PREFIX ) ) { 735 setOffset( makeQRImage( row, offset, key.substring( QRCODE_PREFIX.length() ), buf ) ); 736 } 737 // 画像置き換えの処理、処理後はoffsetが進むため、offsetを再セット 738 else if( key.startsWith( IMG_PREFIX ) ) { 739 setOffset( changeImage( row, offset, key.substring( IMG_PREFIX.length() ), buf ) ); 740 } 741 // コメント(アノテーション)による置き換え処理、処理後はoffsetが進むため、offsetを再セット 742 else if( key.startsWith( ANNOTATION_PREFIX ) ) { 743 setOffset( parseByAnnotation( row, offset, key.substring( ANNOTATION_PREFIX.length() ), buf ) ); 744 } 745 else { 746 String val = getValue( key ); 747 // 5.5.2.4 (2012/05/16) String key は使われていないので、削除します。 748 if( useChangeType ) { // 6.8.3.1 (2017/12/01) 749 // 5.5.2.4 (2012/05/16) String key は使われていないので、削除します。 750 changeType( row, offset, val, getNativeType( key, val ), buf ); 751 } 752 // 8.0.3.0 (2021/12/17) "\n" 文字列を実際の改行コードに置換して設定します。 753 if( val.indexOf( "\\n" ) >= 0 ) { 754 val = val.replace( "\\n" , "\n" ); 755 } 756 buf.append( val ); 757 } 758 759 // 処理行がページエンドカットの対象か 760 if( queue.isFgcut() && PAGE_END_CUT.equals( key ) ) { 761 isPageEndCut = true; 762 } 763 } 764 }.doParse( row, VAR_START, VAR_END, false ); 765 766 //==== ここからは後処理 ========================================================= 767 /* 768 * ページエンドカットの判定は最後で処理する。 769 * {@PAGEENDCUT}が行の最初に書かれている場合は、OVERFLOWになっていない可能性が 770 * あるため行の途中では判断できない 771 */ 772 // 5.4.2.0 (2011/12/01) シートブレイク中でもページエンドカットが適用されるようにする。 773 // (通常のページブレイクは先読み判定のためページエンドカットすると、ブレイク発生行自身が 774 // 削除されてしまうため現時点では未対応) 775// if( isPageEndCut && ( status == OVERFLOW || ( sheetBreakClm >= 0 && isPageBreak ) ) ) { 776 if( isPageEndCut && ( status == OVERFLOW || sheetBreakClm >= 0 && isPageBreak ) ) { // 6.9.7.0 (2018/05/14) PMD Useless parentheses. 777 // ページエンドカットの場合は、非表示状態にする。 778 rowStr = rowStr.replace( ROW_START_TAG, ROW_START_TAG_INVISIBLE ) ; 779 } 780 781 /* 782 * オブジェクトで定義されているテーブル名を変更 783 */ 784 if( rowStr.indexOf( OBJECT_SEARCH_STR ) >= 0 ) { 785 rowStr = rowStr.replace( OBJECT_SEARCH_STR + sheetNameOrig, OBJECT_SEARCH_STR + sheetNameNew ); 786 } 787 788 /* 789 * 関数エラーを表示されないため、ISERROR関数を埋め込み 4.2.3.1 (2008/06/19) 790 */ 791 rowStr = replaceOoocError( rowStr ); 792 793 /* 794 * グラフをシート毎にコピー 5.1.8.0 (2010/07/01) 795 */ 796 rowStr = replaceGraphInfo( rowStr, sheetNameOrig, sheetNameNew ); 797 798 /* 799 * アノテーション(コメント)を削除 5.1.8.0 (2010/07/01) 800 * (コメントが存在すると起動が異常に遅くなる) 801 */ 802 if( rowStr.indexOf( ANNOTATION_START_TAG ) >= 0 ) { 803 rowStr = new TagParser() {}.doParse( rowStr, ANNOTATION_START_TAG, ANNOTATION_END_TAG ); 804 } 805 806 /* 807 * 条件付書式対応 5.2.2.0 (2010/11/01) 808 * テーブル内に存在するスタイル名称を書き換え 809 */ 810 if( isNeedsReparse ) { 811 for( final String name : repStyleList ) { 812 // 5.6.3.1 (2013/04/05) 属性終了追加 813 // 8.5.4.2 (2024/01/12) PMD 7.0.0 LongVariable 対応 814// final String from = TABLE_STYLE_NAME_START_TAG + name + END_KEY ; 815// final String to = TABLE_STYLE_NAME_START_TAG + name + "_" + sheetNameNew + END_KEY ; 816 final String from = TABLE_STYLE_NAME_START + name + END_KEY ; 817 final String to = TABLE_STYLE_NAME_START + name + "_" + sheetNameNew + END_KEY ; 818 819// if( rowStr.indexOf( TABLE_STYLE_NAME_START_TAG + name + TABLE_STYLE_NAME_END_TAG ) >= 0 ) { 820// rowStr = rowStr.replace( TABLE_STYLE_NAME_START_TAG + name + TABLE_STYLE_NAME_END_TAG, 821// TABLE_STYLE_NAME_START_TAG + name + "_" + sheetNameNew + TABLE_STYLE_NAME_END_TAG ); 822// } 823 824 if( rowStr.indexOf( from ) >= 0 ) { 825 rowStr = rowStr.replace( from, to ); 826 } 827 } 828 } 829 //============================================================================== 830 831 bw.write( rowStr ); 832 } 833 834 /** 835 * 帳票データに応じて、カラムの属性を変更(文字型⇒数値型)に変更します。 836 * 837 * @og.rev 5.1.8.0 (2010/07/01) 新規作成 838 * @og.rev 5.5.2.4 (2012/05/16) String key は使われていないので、削除します。 839 * @og.rev 8.1.2.1 (2022/03/25) LibreOffice対応(数字が文字扱いの対応) 840 * 841 * @param row 行データ 842 * @param curOffset オフセット 843 * @param val 設定値 844 * @param type ネイティブタイプ 845 * @param sb StringBuilderオブジェクト 846 */ 847 private void changeType( final String row, final int curOffset 848 , final String val, final NativeType type, final StringBuilder sb ) { 849 // 8.5.5.1 (2024/02/29) PMD 7.0.0 OnlyOneReturn メソッドには終了ポイントが 1 つだけ必要 850 if( val == null || val.isEmpty() 851 // 書き換え対象は数値型のみ 852 || ( type != NativeType.INT && type != NativeType.LONG && type != NativeType.DOUBLE ) 853 // 処理対象がセルでない(オブジェクト)は書き換えしない 854 || !isCell( row, curOffset ) ) { 855 return; 856 } 857 858// if( val == null || val.isEmpty() ) { 859// return; 860// } 861// // 書き換え対象は数値型のみ 862// if( type != NativeType.INT && type != NativeType.LONG && type != NativeType.DOUBLE ) { 863// return; 864// } 865// // 処理対象がセルでない(オブジェクト)は書き換えしない 866// if( !isCell( row, curOffset ) ) { 867// return; 868// } 869 870 // セルの文字が{@xxxx_n}のみであった場合だけ、数値定義の判定を行う。 871 // (関数内に{@xxxx_n}等があった場合は、判定しない(<text:p>{@xxxx_n}</text:p>の場合のみ)) 872 if( sb.lastIndexOf( TEXT_START_TAG ) + TEXT_START_TAG.length() == sb.length() 873 && row.indexOf( TEXT_END_TAG, curOffset ) == curOffset ) { 874// final int typeIdx = sb.lastIndexOf( TABLE_CELL_STRING_TYPE ); 875 final int typeIdx = sb.lastIndexOf( OFC_VALUE_STRING ); // 8.5.4.2 (2024/01/12) PMD 7.0.0 LongVariable 対応 876 final int cellIdx = sb.lastIndexOf( TABLE_CELL_START_TAG ); 877 if( typeIdx >= 0 && cellIdx >= 0 && typeIdx > cellIdx ) { 878 // office:value-type="string" を office:value-type="float" office:value="xxx" に変換 879// final int endIdx = typeIdx + TABLE_CELL_STRING_TYPE.length() ; 880 final int endIdx = typeIdx + OFC_VALUE_STRING.length() ; // 8.5.4.2 (2024/01/12) PMD 7.0.0 LongVariable 対応 881// final String repStr = TABLE_CELL_FLOAT_TYPE + " " + TABLE_CELL_FLOAT_VAL_START + val + END_KEY ; // 8.1.2.1 (2022/03/25) Modify 882 final String valNew = val.replace( ",", "" ); // カンマ除去 8.1.2.1 (2022/03/25) Add 883// final String repStr = TABLE_CELL_FLOAT_TYPE + " " + TABLE_CELL_FLOAT_VAL_START + valNew + END_KEY ; 884 final String repStr = OFC_VALUE_FLOAT + " " + OFC_VALUE_START + valNew + END_KEY ; // 8.5.4.2 (2024/01/12) PMD 7.0.0 LongVariable 対応 885 886// sb.replace( typeIdx, typeIdx + TABLE_CELL_STRING_TYPE.length() 887// ,TABLE_CELL_FLOAT_TYPE + " " + TABLE_CELL_FLOAT_VAL_START + val + TABLE_CELL_FLOAT_VAL_END ); 888 sb.replace( typeIdx, endIdx, repStr ); 889 890 // // LibreOffice対応 8.1.2.1 (2022/03/25) 一時的にコメント化 891 // final int calcSt = sb.lastIndexOf( TABLE_CELL_CALCEXT ); 892 // if( calcSt >= 0 ) { 893 // final int calcEd = sb.indexOf( "\"", calcSt + TABLE_CELL_CALCEXT.length() ); 894 // sb.delete( calcSt - 1, calcEd + 1 ); 895 // } 896 } 897 } 898 } 899 900 /** 901 * 引数に指定された文字列のNativeタイプを返します。 902 * 903 * リソース使用時は、各DBTypeで定義されたNativeタイプを、 904 * 未使用時は、値からNativeタイプを取得して返します。 905 * 906 * @og.rev 5.1.8.0 (2010/07/01) NativeType#getType(String) のメソッドを使用するように変更。 907 * @og.rev 5.10.5.1 (2018/11/09) intだけでなくlongの0始まりも文字列として扱う 908 * @og.rev 8.0.3.0 (2021/12/17) アンダーバーで、キーと行番号の分離を、インナークラス化します。 909 * 910 * @param key キー 911 * @param val 文字列 912 * @return NATIVEの型の識別コード 913 * @og.rtnNotNull 914 * @see org.opengion.fukurou.model.NativeType 915 */ 916 private NativeType getNativeType( final String key, final String val ) { 917 if( val == null || val.isEmpty() ) { 918 return NativeType.STRING; 919 } 920 921 NativeType type = null; 922 if( queue.isFglocal() ) { 923// String name = key; 924// final int conOffset = key.lastIndexOf( VAR_CON ); 925// if( conOffset >= 0 ) { 926// int rownum = -1; 927// try { 928// rownum = Integer.parseInt( name.substring( conOffset + VAR_CON.length(), name.length() ) ); // 6.0.2.4 (2014/10/17) メソッド間違い 929// } 930// // '_'以降の文字が数字でない場合は、'_'以降の文字もカラム名の一部として扱う 931// catch( final NumberFormatException ex ) { 932// // 6.4.1.1 (2016/01/16) PMD refactoring. Avoid empty catch blocks 933// final String errMsg = "'_'以降の文字をカラム名の一部として扱います。カラム名=[" + key + "]" + CR + ex.getMessage() ; 934// System.err.println( errMsg ); 935// } 936// if( rownum >= 0 ) { 937// name = name.substring( 0, conOffset ); 938// } 939// } 940// final int col = queue.getBody().getColumnNo( name, false ); 941 final SplitKey spKey = new SplitKey( key ); // 8.0.3.0 (2021/12/17) 942 final int col = queue.getBody().getColumnNo( spKey.name, false ); 943 if( col >= 0 ) { 944 type = queue.getBody().getDBColumn( col ).getNativeType(); 945 } 946 } 947 948 if( type == null ) { 949 // ,は削除した状態で判定 950 final String repVal = val.replace( ",", "" ); 951 type = NativeType.getType( repVal ); // 5.1.8.0 (2010/07/01) NativeType#getType(String) のメソッドを使用 952 // 整数型で、0nnnとなっている場合は、文字列をして扱う 953// if( type == NativeType.INT && repVal.length() >= 2 && repVal.charAt(0) == '0' ) { 954 if( ( type == NativeType.INT || type == NativeType.LONG ) && repVal.length() >= 2 && repVal.charAt(0) == '0' ) { // 5.10.5.1 (2018/11/09) LONGを含む 955 type = NativeType.STRING; 956 } 957 } 958 959 return type; 960 } 961 962 /** 963 * コメント(アノテーションによる置き換え処理を行います) 964 * この処理では、offsetを進めるため、戻り値として処理後のoffsetを返します。 965 * 966 * @og.rev 5.1.8.0 (2010/07/01) 新規作成 967 * @og.rev 6.8.3.1 (2017/12/01) ローカルリソースの文字型⇒数値型変換の処理の有無 968 * 969 * @param row 行データ 970 * @param curOffset オフセット 971 * @param key キー 972 * @param sb StringBuilderオブジェクト 973 * @return 処理後のオフセット 974 */ 975 private int parseByAnnotation( final String row, final int curOffset, final String key, final StringBuilder sb ) { 976 int offset = curOffset; 977 final boolean isCell = isCell( row, offset ); 978 979 // セルの場合のみ置き換えの判定を行う(オブジェクトの場合は判定しない) 980 if( isCell ) { 981 final int cellStrIdx = sb.lastIndexOf( TABLE_CELL_START_TAG, offset ); 982 // office:value-type="float" office:value="xxx" を office:value-type="string" に変換 983 // 数値型の場合は、後で再度変換を行う。 984 // (文字型に変換しておかないと、値がnullの場合でも"0"が表示されてしまうため) 985// final int floatIdx = sb.indexOf( TABLE_CELL_FLOAT_TYPE, cellStrIdx ); 986 final int floatIdx = sb.indexOf( OFC_VALUE_FLOAT, cellStrIdx ); // 8.5.4.2 (2024/01/12) PMD 7.0.0 LongVariable 対応 987 if( floatIdx >= 0 ) { 988// sb.replace( floatIdx, floatIdx + TABLE_CELL_FLOAT_TYPE.length(), TABLE_CELL_STRING_TYPE ); 989 sb.replace( floatIdx, floatIdx + OFC_VALUE_FLOAT.length(), OFC_VALUE_STRING ); // 8.5.4.2 (2024/01/12) PMD 7.0.0 LongVariable 対応 990 991// final int floatStrIdx = sb.indexOf( TABLE_CELL_FLOAT_VAL_START, floatIdx ); 992 final int floatStrIdx = sb.indexOf( OFC_VALUE_START, floatIdx ); // 8.5.4.2 (2024/01/12) PMD 7.0.0 LongVariable 対応 993 if( floatStrIdx >= 0 ) { 994// final int floatEndIdx = sb.indexOf( TABLE_CELL_FLOAT_VAL_END, floatStrIdx + TABLE_CELL_FLOAT_VAL_START.length() ); 995 final int floatEndIdx = sb.indexOf( END_KEY, floatStrIdx + OFC_VALUE_START.length() ); // 8.5.4.2 (2024/01/12) PMD 7.0.0 LongVariable 対応 996 if( floatEndIdx >= 0 ) { 997// sb.replace( floatStrIdx, floatEndIdx + TABLE_CELL_FLOAT_VAL_END.length(), "" ); 998 sb.replace( floatStrIdx, floatEndIdx + 1, "" ); 999 1000 // // LibreOffice対応 8.1.2.1 (2022/03/25) 1001 // final int calcSt = sb.indexOf( TABLE_CELL_CALCEXT, floatEndIdx ); 1002 // if( calcSt >= 0 ) { 1003 // final int calcEd = sb.indexOf( "\"", calcSt + TABLE_CELL_CALCEXT.length() ); 1004 // sb.delete( calcSt - 1, calcEd + 1 ); 1005 // } 1006 } 1007 } 1008 } 1009 } 1010 1011 // アノテーションの値から、セルの文字列部分を置き換え 1012 final int endIdx = isCell ? row.indexOf( TABLE_CELL_END_TAG, offset ) : row.indexOf( DRAW_END_TAG, offset ); 1013 if( endIdx >= 0 ) { 1014 int textStrIdx = row.indexOf( TEXT_START_ANO_TAG, offset ); 1015 // セルのコメントの場合、<text:pで検索すると、オブジェクトのテキストが検索されている可能性がある。 1016 // このため、セルの<text:pが見つかるまで検索を繰り返す 1017 if( isCell ) { 1018 while( !isCell( row, textStrIdx ) && textStrIdx >= 0 ) { 1019 textStrIdx = row.indexOf( TEXT_START_ANO_TAG, textStrIdx + 1 ); 1020 } 1021 } 1022 if( textStrIdx >= 0 && textStrIdx < endIdx ) { 1023 // セルのコメントの場合、</text:p>で検索すると、オブジェクトのテキストが検索されている可能性がある。 1024 // このため、セルの</text:p>が見つかるまで検索を繰り返す 1025 int textEndIdx = row.lastIndexOf( TEXT_END_TAG, endIdx ); 1026 if( isCell ) { 1027 while( !isCell( row, textEndIdx ) && textEndIdx >= 0 ) { 1028 textEndIdx = row.lastIndexOf( TEXT_END_TAG, textEndIdx - 1 ); 1029 } 1030 } 1031 if( textEndIdx >= 0 && textStrIdx < textEndIdx && textEndIdx < endIdx ) { 1032 // <text:p xxxx> の xxxx> の部分(style定義など)を書き込み 1033 final int textStyleEnd = row.indexOf( TEXT_START_END_ANO_TAG, textStrIdx + TEXT_START_ANO_TAG.length() ) + TEXT_START_END_ANO_TAG.length(); 1034 sb.append( row.substring( offset, textStyleEnd ) ); 1035 1036 // <text:pの中身(spanタグなどを取り除いた状態の文字列 1037 final String textVal = TagParser.checkKey( row.substring( textStyleEnd, textEndIdx ), sb ); 1038 // 取得したテキスト内にタグ文字が含まれている場合は、処理しない 1039 if( textVal.indexOf( '<' ) < 0 && textVal.indexOf( '>' ) < 0 ) { 1040 // <text:p xxxx>を書き出し 1041 final String val = getValue( key ); 1042 if( useChangeType ) { // 6.8.3.1 (2017/12/01) 1043 // 5.5.2.4 (2012/05/16) String key は使われていないので、削除します。 1044 changeType( row, textEndIdx, val, getNativeType( key, textVal ), sb ); 1045 } 1046 sb.append( val ); 1047 } 1048 offset = textEndIdx; 1049 } 1050 } 1051 } 1052 1053 return offset; 1054 } 1055 1056 /** 1057 * 現在のオフセットがセルかどうかを返します。 1058 * 1059 * trueの場合はセルを、falseの場合はオブジェクトを意味します。 1060 * 1061 * セルとして判定されるための条件は以下の通りです。 1062 * 現在のoffsetを基準として、 1063 * ①前に<draw:(オブジェクトの開始)が見つからない 1064 * ②前に<table:table-cell(セルの始まり)が<draw:(オブジェクトの始まり)より後方にある 1065 * ③後に</draw:(オブジェクトの終わり)が見つからない 1066 * ④後に</draw:(オブジェクトの終わり)が</table:table-cell>(セルの終わり)より後方にある 1067 * 1068 * @param row 行データ 1069 * @param offset オフセット 1070 * @return 現在のオフセットがセルかどうか(falseの場合はオブジェクト) 1071 */ 1072 private boolean isCell( final String row, final int offset ) { 1073 final int drawStartOffset = row.lastIndexOf( DRAW_START_TAG, offset ); 1074 // 8.5.5.1 (2024/02/29) PMD 7.0.0 OnlyOneReturn メソッドには終了ポイントが 1 つだけ必要 1075 final boolean flag; 1076 if( drawStartOffset < 0 ) { 1077// return true; 1078 flag = true; 1079 } 1080 else { 1081 final int cellStartOffset = row.lastIndexOf( TABLE_CELL_START_TAG, offset ); 1082 if( drawStartOffset < cellStartOffset ) { 1083// return true; 1084 flag = true; 1085 } 1086 else { 1087 final int drawEndOffset = row.indexOf( DRAW_END_TAG, offset ); 1088 if( drawEndOffset < 0 ) { 1089// return true; 1090 flag = true; 1091 } 1092 else { 1093 final int cellEndOffset = row.indexOf( TABLE_CELL_END_TAG, offset ); 1094 // 5.1.8.0 (2010/07/01) Avoid unnecessary if..then..else statements when returning a boolean 1095// return cellEndOffset >= 0 && cellEndOffset < drawEndOffset ; 1096 flag = cellEndOffset >= 0 && cellEndOffset < drawEndOffset ; 1097 } 1098 } 1099 } 1100 return flag; 1101 } 1102 1103 /** 1104 * QRコードを作成します。 1105 * この処理では、offsetを進めるため、戻り値として処理後のoffsetを返します。 1106 * 1107 * @og.rev 4.3.1.1 (2008/08/23) mkdirs の戻り値判定 1108 * @og.rev 4.3.3.5 (2008/11/08) ↑の判定は存在チェックを行ってから処理する。ファイル名に処理行を付加 1109 * @og.rev 5.3.1.0 (2011/01/01) OpenOffice3.2対応 追加した画像をmanifest.xmlに登録する 1110 * @og.rev 7.0.5.1 (2019/09/27) QRコードのパラメータをシステムリソースで設定できるようにします(ただし、staticとします) 1111 * @og.rev 7.2.3.0 (2020/04/10) QRコードのパラメータをシステムリソースで設定(帳票出力のQRコード作成時のテキストのエンコード指定) 1112 * @og.rev 8.3.0.0 (2022/08/01) '<','>','&'のメタ文字変換をするかどうかを指定します。 1113 * @og.rev 8.4.1.0 (2023/02/10) QRコードを swetake から ZXing への置換(encodeMode廃止) 1114 * 1115 * @param row 行データ 1116 * @param curOffset オフセット 1117 * @param key キー 1118 * @param sb StringBuilderオブジェクト 1119 * @return 処理後のオフセット 1120 */ 1121 private int makeQRImage( final String row, final int curOffset, final String key, final StringBuilder sb ) { 1122 int offset = curOffset; 1123 1124 // {@QRCODE.XXXX}から実際に画像のパスが書かれている部分までを書き込む 1125 offset = row.indexOf( DRAW_IMG_START_TAG, offset ) + DRAW_IMG_START_TAG.length(); 1126 sb.append( row.substring( curOffset, offset ) ); 1127 // 画像のパスの終了インデックスを求める 1128// offset = row.indexOf( DRAW_IMG_HREF_END, offset ) + DRAW_IMG_HREF_END.length(); 1129 offset = row.indexOf( END_KEY, offset ) + 1; 1130 1131 // QRCODEの画像ファイル名を求め書き込む 1132 // 4.3.3.5 (2008/11/08) ファイル名に処理行を付加 1133 final String fileName = IMG_DIR + '/' + key + "_" + currentBaseRow + QRCODE_FILETYPE; 1134// sb.append( fileName ).append( DRAW_IMG_HREF_END ); 1135 sb.append( fileName ).append( END_KEY ); 1136 1137// // QRCODEに書き込む値を求める … 8.3.0.0 (2022/08/01) QrcodeImage 作成直前に移動 1138// final String value = getValue( key ); 1139 1140 // QRCODEの作成 1141 // 4.3.3.5 (2008/11/08) ファイル名に処理行を付加 1142 final String fileNameAbs = 1143// new File( path ).getAbsolutePath() + File.separator + IMG_DIR + File.separator + key + "_" + currentBaseRow + QRCODE_FILETYPE; 1144 new File( path ).getAbsolutePath() + FS + IMG_DIR + FS + key + "_" + currentBaseRow + QRCODE_FILETYPE; 1145 1146 // 画像リンクが無効となっている場合は、Picturesのフォルダが作成されていない可能性がある 1147 // 4.3.1.1 (2008/08/23) mkdirs の戻り値判定 1148 // 4.3.3.5 (2008/11/08) 存在チェック追加 1149 // 6.9.7.0 (2018/05/14) PMD These nested if statements could be combined 1150// if( !new File( fileNameAbs ).getParentFile().exists() ) { 1151// if( new File( fileNameAbs ).getParentFile().mkdirs() ) { 1152 if( !new File( fileNameAbs ).getParentFile().exists() 1153 && new File( fileNameAbs ).getParentFile().mkdirs() ) { 1154 System.err.println( fileNameAbs + " の ディレクトリ作成に失敗しました。" ); 1155// } 1156 } 1157 1158 // QRCODEに書き込む値を求める 1159 final String value = getValue( key,false ); // 8.3.0.0 (2022/08/01) メタ文字変換 しない 1160 1161 final QrcodeImage qrImage = new QrcodeImage(); 1162// qrImage.init( value, fileNameAbs ); 1163// qrImage.init( value, fileNameAbs, QR_VERSION, QR_ENCMODE, QR_ERRCRCT, QR_IMAGE_TYPE, QR_PIXEL ); // 7.0.5.1 (2019/09/27) 1164// qrImage.init( value, fileNameAbs, QR_VERSION, QR_ENCMODE, QR_ERRCRCT, QR_IMAGE_TYPE, QR_PIXEL, QR_TXT_ENC ); // 7.2.3.0 (2020/04/10) 1165 qrImage.init( value, fileNameAbs, QR_VERSION, QR_ERRCRCT, QR_IMAGE_TYPE, QR_PIXEL, QR_TXT_ENC ); // 8.4.1.0 (2023/02/10) 1166 qrImage.saveImage(); 1167 1168 // 5.3.1.0 (2011/01/01) OpenOffice3.2対応 追加した画像をmanifest.xmlに登録する 1169 addObjMap.put( fileName, QRCODE_FILETYPE.substring( 1 ) ); 1170 1171 // 読み込みOffsetを返します 1172 return offset; 1173 } 1174 1175 /** 1176 * DBTableModelに設定されたパスから画像データを取得し、内部に取り込みます 1177 * この処理では、offsetを進めるため、戻り値として処理後のoffsetを返します。 1178 * 1179 * @og.rev 4.3.3.5 (2008/11/08) 新規追加 1180 * @og.rev 4.3.3.6 (2008/11/15) 画像パスが存在しない場合は、リンクタグ(draw:image)自体を削除 1181 * @og.rev 5.3.1.0 (2011/01/01) OpenOffice3.2対応 追加した画像をmanifest.xmlに登録する 1182 * @og.rev 7.3.0.1 (2021/01/22) 画像ファイル名が漢字の場合、うまくいかないので、置き換える。 1183 * 1184 * @param row 行データ 1185 * @param curOffset オフセット 1186 * @param key キー 1187 * @param sb StringBuilderオブジェクト 1188 * @return 処理後のオフセット 1189 */ 1190 private int changeImage( final String row, final int curOffset, final String key, final StringBuilder sb ) { 1191 int offset = curOffset; 1192 File imgFile = null; 1193 1194 // 画像ファイルを読み込むパスを求める 1195 final String value = getValue( key ); 1196 1197 if( value != null && value.length() > 0 ) { 1198 imgFile = new File( HybsSystem.url2dir( value ) ); 1199 } 1200 1201 // 画像ファイルのパスが入っていて、実際に画像が存在する場合 1202 if( imgFile != null && imgFile.exists() ) { 1203 // {@IMG.XXXX}から実際に画像のパスが書かれている部分までを書き込む 1204 offset = row.indexOf( DRAW_IMG_START_TAG, offset ) + DRAW_IMG_START_TAG.length(); 1205 sb.append( row.substring( curOffset, offset ) ); 1206 1207 // 画像のパスの終了インデックスを求める 1208// offset = row.indexOf( DRAW_IMG_HREF_END, offset ) + DRAW_IMG_HREF_END.length(); 1209 offset = row.indexOf( END_KEY, offset ) + 1; 1210 1211 // 7.3.0.1 (2021/01/22) 画像ファイル名が漢字の場合、うまくいかないので、置き換える。 1212// final String fileNameOut = IMG_DIR + '/' + imgFile.getName(); 1213 final String extension = value.substring( value.lastIndexOf('.') ); // 7.3.0.1 (2021/01/22) 拡張子( .付き ) 1214 // 7.3.0.1 (2021/01/22) 同一ファイルは同一名にしておきます。マイナスが気持ち悪いのでハッシュ値は絶対値にしておきます。 1215 // 8.0.0.0 (2021/07/31) spotbugs:ハッシュコードが Integer.MIN_VALUE なら結果は同様に負です (Math.abs(Integer.MIN_VALUE) == Integer.MIN_VALUE なので)。 1216// final String fileNameOut = IMG_DIR + '/' + Math.abs( imgFile.hashCode() ) + extension; 1217 final String fileNameOut = IMG_DIR + '/' + Integer.toUnsignedString( imgFile.hashCode() ) + extension; 1218 1219// sb.append( fileNameOut ).append( DRAW_IMG_HREF_END ); 1220 sb.append( fileNameOut ).append( END_KEY ); 1221 1222 final File fileOutAbs = new File( path,fileNameOut ); 1223 if( !fileOutAbs.getParentFile().exists() && fileOutAbs.getParentFile().mkdirs() ) { 1224 System.err.println( fileOutAbs + " の ディレクトリ作成に失敗しました。" ); 1225 } 1226 1227 // final String fileNameOutAbs = 1228// new File( path ).getAbsolutePath() + File.separator + IMG_DIR + File.separator + imgFile.getName(); 1229 // new File( path ).getAbsolutePath() + '/' + fileNameOut; 1230 // // 6.9.7.0 (2018/05/14) PMD These nested if statements could be combined 1231// if( !new File( fileNameOutAbs ).getParentFile().exists() ) { 1232// if( new File( fileNameOutAbs ).getParentFile().mkdirs() ) { 1233 // if( !new File( fileNameOutAbs ).getParentFile().exists() 1234 // && new File( fileNameOutAbs ).getParentFile().mkdirs() ) { 1235 // System.err.println( fileNameOutAbs + " の ディレクトリ作成に失敗しました。" ); 1236// } 1237 // } 1238 // FileUtil.copy( imgFile, new File( fileNameOutAbs ) ); // imgFile → fileNameOutAbs copy 1239 FileUtil.copy( imgFile, fileOutAbs ); // imgFile → fileOutAbs copy 1240 1241 // 5.3.1.0 (2011/01/01) OpenOffice3.2対応 追加した画像をmanifest.xmlに登録する 1242 addObjMap.put( fileNameOut, getSuffix( imgFile.getName() ) ); 1243 } 1244 // 画像パスが設定されていない、又は画像が存在しない場合 1245 else { 1246 // {@IMG.XXXX}から見て、<draw:image> ... </draw:image>までをスキップする 1247 offset = row.indexOf( DRAW_IMG_START_TAG, offset ); 1248 sb.append( row.substring( curOffset, offset ) ); 1249 1250 offset = row.indexOf( DRAW_IMG_END_TAG, offset ) + DRAW_IMG_END_TAG.length(); 1251 } 1252 1253 // 読み込みOffsetを返します 1254 return offset; 1255 } 1256 1257 /** 1258 * 変換後の行データで定義されている関数にISERROR関数を埋め込みます。 1259 * 1260 * これは、OOoの関数の動作として、不正な引数等が入力された場合(null値など)に、 1261 * エラー:xxxと表示されてしまうため、これを防ぐために関数エラーのハンドリングを行い、 1262 * エラーの場合は、空白文字を返すようにします。 1263 * 1264 * @og.rev 4.3.7.2 (2009/06/15) 開始文字が変更になったため対応 1265 * @og.rev 5.0.2.0 (2009/11/01) 関数内の"(quot)は、メタ文字に変換する 1266 * @og.rev 5.1.7.0 (2010/06/01) 関数の終わりが)出ない場合にエラーとなるバグを修正 1267 * @og.rev 5.1.8.0 (2010/07/01) パース方法の内部実装変更 1268 * 1269 * @param row 行データ 1270 * @return 変換後の行データ 1271 */ 1272 private String replaceOoocError( final String row ) { 1273 // 4.3.7.2 (2009/06/15) OOOC_FUNCTION_START3の条件判定追加。どちらか分からないので変数で受ける。 1274 final String functionStart; 1275 if( row.indexOf( OOOC_FUNCTION_START_3 ) >= 0 ) { functionStart = OOOC_FUNCTION_START_3; } 1276 else if( row.indexOf( OOOC_FUNCTION_START ) >= 0 ) { functionStart = OOOC_FUNCTION_START; } 1277 else { return row; } 1278 1279 final String rowStr = new TagParser() { 1280 /** 1281 * 開始タグから終了タグまでの文字列の処理を実行するかどうかを定義します。 1282 * 1283 * @param strOffset 開始タグのオフセット 1284 * @param endOffset 終了タグのオフセット 1285 * @return 処理を行うかどうか(true:処理を行う false:処理を行わない) 1286 */ 1287 @Override 1288 protected boolean checkIgnore( final int strOffset, final int endOffset ) { 1289 // 5.1.7.0 (2010/06/01) 関数の終わりが)出ない場合にエラーとなるバグを修正 1290 // 単なる行参照でも、of:=で始まるがこの場合は、関数でないため終わりが)でない 1291 // このため、)が見つからないまたは、タグの終わり(>)が先に見つかった場合は、エラー関数を 1292 // 埋め込まないようにする。 1293 // 8.5.4.2 (2024/01/12) PMD 7.0.0 ConfusingTernary 対応 1294 final int tmpOffset = row.indexOf( '>', strOffset + 1 ); // 8.5.4.2 (2024/01/12) PMD 7.0.0 LocalVariableCouldBeFinal 1295 return endOffset >= 0 && endOffset < tmpOffset ; 1296 } 1297 1298 /** 1299 * 開始タグから終了タグまでの文字列の処理を定義します。 1300 * 1301 * @param str 開始タグから終了タグまでの文字列(開始タグ・終了タグを含む) 1302 * @param buf 出力を行う文字列バッファ 1303 * @param offset 終了タグのオフセット(ここでは使っていません) 1304 */ 1305 @Override 1306 protected void exec( final String str, final StringBuilder buf, final int offset ) { 1307 String key = str.substring( functionStart.length(), str.length() - OOOC_FUNCTION_END.length() ) + ")"; 1308 key = key.replace( "\"", """" ).replace( OOO_CR, "" ); 1309 // 6.4.2.1 (2016/02/05) PMD refactoring. 1310 buf.append( functionStart ).append( "IF(ISERROR(" ).append( key ) 1311 .append( ");"";" ).append( key ).append( OOOC_FUNCTION_END ); 1312 } 1313 }.doParse( row, functionStart, OOOC_FUNCTION_END ); 1314 1315 return rowStr; 1316 } 1317 1318 /** 1319 * グラフ表示データ部分を更新します。 1320 * 1321 * @og.rev 5.1.8.0 (2010/07/01) 新規作成 1322 * @og.rev 5.3.1.0 (2011/01/01) OpenOffice3.2対応 追加した画像をmanifest.xmlに登録する 1323 * 1324 * @param row 行データ 1325 * @param sheetOrig 元シート 1326 * @param sheetNew 新シート 1327 * @return 変換後の行データ 1328 */ 1329 private String replaceGraphInfo( final String row, final String sheetOrig, final String sheetNew ) { 1330 if( row.indexOf( GRAPH_START_TAG ) < 0 || row.indexOf( GRAPH_UPDATE_RANGE_START ) < 0 ) { return row; } 1331 1332 // 7.2.9.5 (2020/11/28) PMD:Consider simply returning the value vs storing it in local variable 'XXXX' 1333 return new TagParser() { 1334// final String rowStr = new TagParser() { 1335 /** 1336 * 開始タグから終了タグまでの文字列の処理を定義します。 1337 * 1338 * @param str 開始タグから終了タグまでの文字列(開始タグ・終了タグを含む) 1339 * @param buf 出力を行う文字列バッファ 1340 * @param offset 終了タグのオフセット(ここでは使っていません) 1341 */ 1342 @Override 1343 protected void exec( final String str, final StringBuilder buf, final int offset ) { 1344 // <draw:object ... /> の部分 1345 String graphTag = str; 1346 1347 if( graphTag.indexOf( GRAPH_UPDATE_RANGE_START ) >= 0 ) { 1348// final String nameOrig = TagParser.getValueFromTag( graphTag, GRAPH_HREF_START, GRAPH_HREF_END ); 1349 final String nameOrig = TagParser.getValueFromTag( graphTag, GRAPH_HREF_START, END_KEY ); // 8.0.3.0 (2021/12/17) 1350 if( new File( path + nameOrig ).exists() ) { 1351 final String nameNew = nameOrig + "_" + pages; 1352 1353 // グラフオブジェクトの定義ファイルをコピー(./Object X/* ⇒ ./Object X_n/*) 1354 FileUtil.copyDirectry( path + nameOrig, path + nameNew ); 1355 graphTag = graphTag.replace( GRAPH_HREF_START + nameOrig, GRAPH_HREF_START + nameNew ); 1356 1357 // グラフオブジェクトの画像イメージをコピー(./ObjectReplacements/Object X ⇒ ./ObjectReplacements/Object X_n) 1358 // ※実体はコピーしない(リンクの参照を無効にしておくことで、次回起動時にグラフの再描画が行われる) 1359 graphTag = graphTag.replace( GRAPH_HREF_START + GRAPH_OBJREPL + "/" + nameOrig, GRAPH_HREF_START + GRAPH_OBJREPL + "/" + nameNew ); 1360 1361 // OpenOffice3.2対応 追加した画像をmanifest.xmlに登録する 1362 addObjMap.put( nameNew, "graph" ); 1363 1364 // グラフオブジェクトの定義ファイルに記述されている定義ファイルをパースし、シート名と{@XXXX}を置き換え 1365// parseGraphContent( path + nameNew + File.separator + "content.xml", sheetOrig, sheetNew ); 1366 parseGraphContent( path + nameNew + FS + "content.xml", sheetOrig, sheetNew ); 1367 1368 // グラフの参照範囲のシート名を置き換え 1369// final String range = TagParser.getValueFromTag( str, GRAPH_UPDATE_RANGE_START, GRAPH_UPDATE_RANGE_END ); 1370 final String range = TagParser.getValueFromTag( str, GRAPH_UPDATE_RANGE_START, END_KEY ); // 8.0.3.0 (2021/12/17) 1371 graphTag = graphTag.replace( GRAPH_UPDATE_RANGE_START + range, GRAPH_UPDATE_RANGE_START + range.replace( sheetOrig, sheetNew ) ); 1372 } 1373 } 1374 1375 buf.append( graphTag ); 1376 } 1377 }.doParse( row, GRAPH_START_TAG, GRAPH_END_TAG ); 1378 1379// return rowStr; 1380 } 1381 1382 /** 1383 * グラフデータのcontent.xmlをパースします。 1384 * 1385 * @og.rev 5.1.8.0 (2010/07/01) 新規作成 1386 * 1387 * @param fileName ファイル名 1388 * @param sheetOrig 元シート 1389 * @param sheetNew 新シート 1390 */ 1391 private void parseGraphContent( final String fileName, final String sheetOrig, final String sheetNew ) { 1392 String graphContent = readOOoXml( fileName ); 1393 1394 // シート名の置き換え 1395 if( graphContent.indexOf( GRAPH_CONTENT_START ) >= 0 ) { 1396 graphContent = new TagParser() { 1397 /** 1398 * 開始タグから終了タグまでの文字列の処理を定義します。 1399 * 1400 * この実装では、何も処理を行いません。(切り出した文字列はアペンドされません) 1401 * サブクラスでオーバーライドして実際の処理を実装して下さい。 1402 * 1403 * @param str 開始タグから終了タグまでの文字列(開始タグ・終了タグを含む) 1404 * @param buf 出力を行う文字列バッファ 1405 * @param offset 終了タグのオフセット(ここでは使っていません) 1406 */ 1407 @Override 1408 protected void exec( final String str, final StringBuilder buf, final int offset ) { 1409 buf.append( str.replace( sheetOrig, sheetNew ) ); 1410 } 1411// }.doParse( graphContent, GRAPH_CONTENT_START, GRAPH_CONTENT_END ); 1412 }.doParse( graphContent, GRAPH_CONTENT_START, END_KEY ); 1413 } 1414 1415 // {@XXXX}の置き換え 1416 if( graphContent.indexOf( VAR_START ) >= 0 ) { 1417 graphContent = new TagParser() { 1418 /** 1419 * 開始タグから終了タグまでの文字列の処理を定義します。 1420 * 1421 * この実装では、何も処理を行いません。(切り出した文字列はアペンドされません) 1422 * サブクラスでオーバーライドして実際の処理を実装して下さい。 1423 * 1424 * @param str 開始タグから終了タグまでの文字列(開始タグ・終了タグを含む) 1425 * @param buf 出力を行う文字列バッファ 1426 * @param offset 終了タグのオフセット(ここでは使っていません) 1427 */ 1428 @Override 1429 public void exec( final String str, final StringBuilder buf, final int offset ) { 1430 buf.append( getHeaderFooterValue( str ) ); 1431 } 1432 }.doParse( graphContent, VAR_START, VAR_END, false ); 1433 } 1434 1435 writeOOoXml( fileName, graphContent ); 1436 } 1437 1438 /** 1439 * 指定されたキーの値を返します。 1440 * 1441 * @og.rev 4.3.0.0 (2008/07/18) アンダースコアの処理変更 1442 * @og.rev 4.3.5.0 (2008/02/01) カラム名と行番号文字の位置は最後から検索する 4.3.3.4 (2008/11/01) 修正分 1443 * @og.rev 8.0.3.0 (2021/12/17) アンダーバーで、キーと行番号の分離を、インナークラス化します。 1444 * @og.rev 8.3.0.0 (2022/08/01) '<','>','&'のメタ文字変換をデフォルト実行します。 1445 * 1446 * @param key キー 1447 * @return 値 1448 */ 1449 private String getValue( final String key ) { 1450 return getValue( key,true ); 1451 } 1452 1453 /** 1454 * 指定されたキーの値を返します。 1455 * 1456 * @og.rev 4.3.0.0 (2008/07/18) アンダースコアの処理変更 1457 * @og.rev 4.3.5.0 (2008/02/01) カラム名と行番号文字の位置は最後から検索する 4.3.3.4 (2008/11/01) 修正分 1458 * @og.rev 8.0.3.0 (2021/12/17) アンダーバーで、キーと行番号の分離を、インナークラス化します。 1459 * @og.rev 8.3.0.0 (2022/08/01) '<','>','&'のメタ文字変換をするかどうかを指定します。 1460 * 1461 * @param key キー 1462 * @param useChange タ文字変換するかどうか[true:する] 1463 * @return 値 1464 */ 1465// private String getValue( final String key ) { 1466 private String getValue( final String key,final boolean useChange ) { 1467// final int conOffset = key.lastIndexOf( VAR_CON ); 1468 1469// String value = null; 1470 final String value ; 1471 1472// if( conOffset < 0 ) { 1473 final SplitKey spKey = new SplitKey( key ); // 8.0.3.0 (2021/12/17) 1474 if( spKey.rownum < 0 ) { 1475 value = getHeaderFooterValue( key ); 1476 } 1477 else { 1478// final String name = key.substring( 0, conOffset ); 1479// int rownum = -1; 1480// try { 1481// rownum = Integer.parseInt( key.substring( conOffset + VAR_CON.length(), key.length() ) ) + currentBaseRow; // 6.0.2.4 (2014/10/17) メソッド間違い 1482// } 1483// catch( final NumberFormatException ex ) { 1484// // 4.3.0.0 (2008/07/18) エラーが起きてもなにもしない。 1485// // queue.addMsg( "[ERROR]雛形の変数定義が誤っています。カラム名=" + name + CR ); 1486// // throw new Exception( ex ); 1487// // 6.4.1.1 (2016/01/16) PMD refactoring. Avoid empty catch blocks 1488// final String errMsg = "雛形の変数定義で、行番号文字が取得できません。カラム名=[" + key + "]" + CR + ex.getMessage() ; 1489// System.err.println( errMsg ); 1490// } 1491// 1492// // 4.3.0.0 (2008/07/18) アンダースコア後が数字に変換できない場合はヘッダフッタとして認識 1493// if( rownum < 0 ){ 1494// value = getHeaderFooterValue( key ); 1495// } 1496// else{ 1497// value = getBodyValue( name, rownum ); 1498 value = getBodyValue( spKey.name, spKey.rownum + currentBaseRow ); 1499// } 1500 } 1501 1502// return checkValue( value ); 1503 return checkValue( value,useChange ); // 8.3.0.0 (2022/08/01) 1504 } 1505 1506 /** 1507 * 指定されたキーのヘッダー、フッター値を返します。 1508 * 1509 * @og.rev 4.3.6.0 (2009/04/01) レンデラー適用されていないバグを修正 1510 * @og.rev 5.0.2.0 (2009/11/01) ローカルリソースフラグを使用しない場合は、リソース変換を行わない。 1511 * @og.rev 5.1.6.0 (2010/05/01) ページNO出力対応 1512 * @og.rev 6.1.1.0 (2015/01/17) getRendererValue の代わりに、getWriteValue を使うように変更。 1513 * @og.rev 6.1.1.0 (2015/01/17) 内部ロジックの見直し。queue.getFooter(),queue.getHeader() を、ローカル変数で定義。 1514 * 1515 * @param key キー 1516 * @return 値 1517 */ 1518 private String getHeaderFooterValue( final String key ) { 1519 String value = ""; 1520 1521 // 5.1.6.0 (2010/05/01) ページNO出力対応 1522 if( PAGE_NO.equals( key ) ) { 1523 value = String.valueOf( pages + 1 ); 1524 } 1525 // 6.1.1.0 (2015/01/17) 内部ロジックの見直し。queue.getFooter(),queue.getHeader() を、ローカル変数で定義。 1526 else { 1527 // 最後の行かオーバーフロー時はフッター。最後の行にきていない場合はヘッダー 1528 final DBTableModel headerFooterModel = status >= LASTROW ? queue.getFooter() : queue.getHeader() ; 1529 1530 if( headerFooterModel != null ) { 1531 final int clmno = headerFooterModel.getColumnNo( key, false ); 1532 if( clmno >= 0 ) { 1533 value = headerFooterModel.getValue( 0, clmno ); 1534 // 5.0.2.0 (2009/11/01) ローカルリソースフラグを使用しない場合は、リソース変換を行わない。 1535 if( queue.isFglocal() ) { 1536 // 4.3.6.0 (2009/04/01) 1537 // 6.1.1.0 (2015/01/17) getRendererValue の代わりに、getWriteValue を使うように変更。 1538 value = headerFooterModel.getDBColumn( clmno ).getWriteValue( value ); 1539 } 1540 } 1541 } 1542 } 1543 1544 return value; 1545 } 1546 1547 /** 1548 * 指定された行番号、キーのボディー値を返します。 1549 * 1550 * @og.rev 4.3.6.0 (2009/04/01) レンデラー適用されていないバグを修正 1551 * @og.rev 4.3.6.2 (2009/04/15) 行番号のより小さいカラム定義を読んだ際に、内部カウンタがクリアされてしまうバグを修正 1552 * @og.rev 4.3.6.2 (2009/04/15) 一度オーバーフローした場合に移行が全て空文字で返ってしまうバグを修正 1553 * @og.rev 5.0.2.0 (2009/11/01) ローカルリソースフラグを使用しない場合は、リソース変換を行わない。 1554 * @og.rev 5.1.6.0 (2010/05/01) 行番号出力対応 1555 * @og.rev 5.1.7.0 (2010/06/01) 複数シート対応 1556 * @og.rev 5.1.9.0 (2010/08/01) 最終行で正しくシートブレイクされないバグを修正 1557 * @og.rev 6.1.1.0 (2015/01/17) getRendererValue の代わりに、getWriteValue を使うように変更。 1558 * @og.rev 6.1.1.0 (2015/01/17) 内部ロジックの見直し。queue.getBody() を、ローカル変数で定義他。 1559 * 1560 * @param key キー 1561 * @param rownum 行番号 1562 * @return キーのボディー値 1563 * @og.rtnNotNull 1564 */ 1565 private String getBodyValue( final String key, final int rownum ) { 1566 // if( status == OVERFLOW || isPageBreak ) { return ""; } 1567 if( isPageBreak ) { return ""; } // 4.3.6.2 (2009/04/15) OVERFLOW時バグ修正 1568 1569 final DBTableModel bodyModel = queue.getBody(); // 6.1.1.0 (2015/01/17) 1570 1571 final int clmno = bodyModel.getColumnNo( key, false ); // 6.1.1.0 (2015/01/17) 1572 if( clmno < 0 && !ROW_NO.equals( key ) ) { return ""; } 1573 1574 // 6.4.1.1 (2016/01/16) PMD refactoring. Avoid declaring a variable if it is unreferenced before a possible exit point. 1575 final int rowCount = bodyModel.getRowCount(); // 6.1.1.0 (2015/01/17) 1576 // ページブレイク判定、先読みして判断 1577 if( PAGE_BREAK.equals( key ) ) { 1578 // 6.9.7.0 (2018/05/14) PMD These nested if statements could be combined 1579// if( rownum < rowCount - 1 ) { // 6.1.1.0 (2015/01/17) 1580// if( !( bodyModel.getValue( rownum, clmno ).equals( bodyModel.getValue( rownum + 1, clmno ) ) ) ) { 1581 if( rownum < rowCount - 1 // 6.1.1.0 (2015/01/17) 1582 && !bodyModel.getValue( rownum, clmno ).equals( bodyModel.getValue( rownum + 1, clmno ) ) ) { 1583 isPageBreak = true; 1584// } 1585 } 1586 return ""; 1587 } 1588 1589 // 5.1.7.0 (2010/06/01) 複数シート対応 1590 // シートブレイクは後読みして判断(前の行と異なっていた場合にブレイク) 1591 // 8.5.4.2 (2024/01/12) PMD 7.0.0 CollapsibleIfStatements 1592// if( sheetBreakClm >= 0 ) { 1593// // 5.1.9.0 (2010/08/01) 最終行で正しくシートブレイクされないバグを修正 1594// // 6.9.7.0 (2018/05/14) PMD These nested if statements could be combined 1595//// if( rownum < rowCount && currentBaseRow != rownum ) { 1596//// if( !( bodyModel.getValue( currentBaseRow, sheetBreakClm ).equals( bodyModel.getValue( rownum, sheetBreakClm ) ) ) ) { 1597// if( rownum < rowCount && currentBaseRow != rownum 1598// && !( bodyModel.getValue( currentBaseRow, sheetBreakClm ).equals( bodyModel.getValue( rownum, sheetBreakClm ) ) ) ) { 1599// isPageBreak = true; 1600// return ""; 1601//// } 1602// } 1603// } 1604 if( sheetBreakClm >= 0 1605 && rownum < rowCount && currentBaseRow != rownum 1606 && !bodyModel.getValue( currentBaseRow, sheetBreakClm ).equals( bodyModel.getValue( rownum, sheetBreakClm ) ) ) { 1607 isPageBreak = true; 1608 return ""; 1609 } 1610 1611 if( rownum >= rowCount ) { // 6.1.1.0 (2015/01/17) 1612 status = OVERFLOW; 1613 return ""; 1614 } 1615 1616 if( rownum == rowCount - 1 ) { // 6.1.1.0 (2015/01/17) 1617 // status = LASTROW; 1618 status = Math.max( LASTROW, status ); // 4.3.6.2 (2009/04/15) 自身のステータスと比べて大きい方を返す 1619 } 1620 1621 String value = null; 1622 // 5.1.6.0 (2010/05/01) ページNO出力対応 1623 if( ROW_NO.equals( key ) ) { 1624 value = String.valueOf( rownum + 1 ); 1625 } 1626 else { 1627 value = bodyModel.getValue( rownum, clmno ); // 6.1.1.0 (2015/01/17) 1628 // 5.0.2.0 (2009/11/01) ローカルリソースフラグを使用しない場合は、リソース変換を行わない。 1629 if( queue.isFglocal() ) { 1630 // 4.3.6.0 (2009/04/01) 1631 // 6.1.1.0 (2015/01/17) getRendererValue の代わりに、getWriteValue を使うように変更。 1632 value = bodyModel.getDBColumn( clmno ).getWriteValue( value ); // 6.1.1.0 (2015/01/17) 1633 } 1634 } 1635 1636 // 4.3.6.2 (2009/04/15) 1637 if( currentMaxRow < rownum + 1 ) { 1638 currentMaxRow = rownum + 1; 1639 } 1640 1641 return value; 1642 } 1643 1644 /** 1645 * 値に'<','>','&'が含まれていた場合にメタ文字に変換します。 1646 * 1647 * @og.rev 5.0.2.0 (2009/11/01) 改行Cの変換ロジックを追加 1648 * @og.rev 5.0.2.0 (2009/11/01) リソース変換時のspanタグを除去 1649 * @og.rev 8.3.0.0 (2022/08/01) '<','>','&'のメタ文字変換をするかどうかを指定します。 1650 * 1651 * @param value 変換前の値 1652 * @param useChange タ文字変換するかどうか[true:する] 1653 * @return 変換後の値 1654 * @og.rtnNotNull 1655 */ 1656// private String checkValue( final String value ) { 1657 private String checkValue( final String value,final boolean useChange ) { 1658 String rtn = value; 1659 1660 // 5.0.2.0 (2009/11/01) 1661 if( queue.isFglocal() ) { 1662 // 6.0.2.5 (2014/10/31) refactoring 1663 final int idx = rtn.indexOf( "<span" ); 1664 if( idx >= 0 ) { 1665 final String spanStart = rtn.substring( idx, rtn.indexOf( '>', idx ) + 1 ); 1666 rtn = rtn.replace( spanStart, "" ).replace( "</span>", "" ); 1667 } 1668 } 1669 1670 if( useChange ) { // 8.3.0.0 (2022/08/01) 1671 if( rtn.indexOf( '&' ) >= 0 ) { 1672 rtn = rtn.replace( "&", "&" ); 1673 } 1674 if( rtn.indexOf( '<' ) >= 0 ) { 1675 rtn = rtn.replace( "<", "<" ); 1676 } 1677 if( rtn.indexOf( '>' ) >= 0 ) { 1678 rtn = rtn.replace( ">", ">" ); 1679 } 1680 if( rtn.indexOf( '\n' ) >= 0 ) { 1681 rtn = rtn.replace( "\r\n", "\n" ).replace( "\n", OOO_CR ); 1682 } 1683 } 1684 1685 return rtn; 1686 } 1687 1688// /** 1689// * 引数の文字列を指定された開始タグ、終了タグで解析し配列として返します。 1690// * 開始タグより前の文字列は0番目に、終了タグより後の文字列は1番目に格納されます。 1691// * 2番目以降に、開始タグ、終了タグの部分が格納されます。 1692// * 1693// * @og.rev 8.0.3.0 (2021/12/17) TagParser.tag2Array を使用します。 1694// * 1695// * @param str 文字列 1696// * @param startTag 開始タグ 1697// * @param endTag 終了タグ 1698// * @return 解析結果の配列 1699// * @og.rtnNotNull 1700// */ 1701// private static String[] tag2Array( final String str, final String startTag, final String endTag ) { 1702// String header = null; 1703// String footer = null; 1704// final List<String> body = new ArrayList<>(); 1705// 1706// int preOffset = -1; 1707// int curOffset = 0; 1708// 1709// while( true ) { 1710// curOffset = str.indexOf( startTag, preOffset + 1 ); 1711// if( curOffset < 0 ) { 1712// curOffset = str.lastIndexOf( endTag ) + endTag.length(); 1713// body.add( str.substring( preOffset, curOffset ) ); 1714// 1715// footer = str.substring( curOffset ); 1716// break; 1717// } 1718// else if( preOffset == -1 ) { 1719// header = str.substring( 0, curOffset ); 1720// } 1721// else { 1722// body.add( str.substring( preOffset, curOffset ) ); 1723// } 1724// preOffset = curOffset; 1725// } 1726// 1727// String[] arr = new String[body.size()+2]; 1728// arr[0] = header; 1729// arr[1] = footer; 1730// for( int i=0; i<body.size(); i++ ) { 1731// arr[i+2] = body.get(i); 1732// } 1733// 1734// return arr; 1735// } 1736 1737 /** 1738 * 帳票処理キューを元に、style.xml(ヘッダー、フッター)を書き換えます。(2) 1739 * 1740 * @og.rev 5.1.8.0 (2010/07/01) パース方法の内部実装変更 1741 */ 1742 private void execStyles() { 1743 final String fileName = path + "styles.xml"; 1744 String content = readOOoXml( fileName ); 1745 1746 if( content.indexOf( VAR_START ) < 0 ) { return; } 1747 1748 content = new TagParser() { 1749 /** 1750 * 開始タグから終了タグまでの文字列の処理を定義します。 1751 * 1752 * この実装では、何も処理を行いません。(切り出した文字列はアペンドされません) 1753 * サブクラスでオーバーライドして実際の処理を実装して下さい。 1754 * 1755 * @param str 開始タグから終了タグまでの文字列(開始タグ・終了タグを含む) 1756 * @param buf 出力を行う文字列バッファ 1757 * @param offset 終了タグのオフセット(ここでは使っていません) 1758 */ 1759 @Override 1760 public void exec( final String str, final StringBuilder buf, final int offset ) { 1761 buf.append( getHeaderFooterValue( str ) ); 1762 } 1763 }.doParse( readOOoXml( fileName ), VAR_START, VAR_END, false ); 1764 1765 writeOOoXml( fileName, content ); 1766 } 1767 1768 /** 1769 * 帳票処理キューを元に、meta.xmlを書き換えます。(6) 1770 * 1771 * @og.rev 5.1.6.0 (2010/05/01) 画面帳票作成機能対応(API経由では出力されないことがある) 1772 */ 1773 private void execMeta() { 1774 final String fileName = path + "meta.xml"; 1775 1776 String meta = readOOoXml( fileName ); 1777 1778 // シート数書き換え 1779 // 5.1.6.0 (2010/05/01) 1780 if( meta.indexOf( TABLE_COUNT_START_TAG ) >=0 ){ 1781// final String tableCount = TagParser.getValueFromTag( meta, TABLE_COUNT_START_TAG, TABLE_COUNT_END_TAG ); 1782 final String tableCount = TagParser.getValueFromTag( meta, TABLE_COUNT_START_TAG, END_KEY ); // 8.0.3.0 (2021/12/17) 1783 meta = meta.replace( TABLE_COUNT_START_TAG + tableCount, TABLE_COUNT_START_TAG + pages ); 1784 } 1785 1786 // セル数書き換え 1787 // 5.1.6.0 (2010/05/01) 1788 if( meta.indexOf( CELL_COUNT_START_TAG ) >=0 ){ 1789// final String cellCount = TagParser.getValueFromTag( meta, CELL_COUNT_START_TAG, CELL_COUNT_END_TAG ); 1790 final String cellCount = TagParser.getValueFromTag( meta, CELL_COUNT_START_TAG, END_KEY ); // 8.0.3.0 (2021/12/17) 1791 meta = meta.replace( CELL_COUNT_START_TAG + cellCount, CELL_COUNT_START_TAG + ( Integer.parseInt( cellCount ) * pages ) ); 1792 } 1793 1794 // オブジェクト数書き換え 1795 // 5.1.6.0 (2010/05/01) 1796 if( meta.indexOf( OBJECT_COUNT_START_TAG ) >= 0 ){ 1797// final String objectCount = TagParser.getValueFromTag( meta, OBJECT_COUNT_START_TAG, OBJECT_COUNT_END_TAG ); 1798 final String objectCount = TagParser.getValueFromTag( meta, OBJECT_COUNT_START_TAG, END_KEY ); // 8.0.3.0 (2021/12/17) 1799 //4.2.4.0 (2008/06/02) 存在しない場合はnullで帰ってくるので無視する 1800 if( objectCount != null){ 1801 meta = meta.replace( OBJECT_COUNT_START_TAG + objectCount, OBJECT_COUNT_START_TAG + ( Integer.parseInt( objectCount ) * pages ) ); 1802 } 1803 } 1804 1805 writeOOoXml( fileName, meta ); 1806 } 1807 1808 /** 1809 * 書き換え対象のスタイルリストを取得します。 1810 * 1811 * @og.rev 5.2.2.0 (2010/11/01) 条件付書式対応 1812 * @og.rev 8.0.3.0 (2021/12/17) TagParser.tag2Array を使用します。 1813 * @og.rev 8.5.4.2 (2024/01/12) PMD 7.0.0 LinguisticNaming 対応 1814 * 1815 * @param header content.xmlのヘッダー 1816 */ 1817// private void getRepStyleList( final String header ) { 1818 private void makeRepStyleList( final String header ) { // 8.5.4.2 (2024/01/12) 1819// final String[] tags = tag2Array( header, STYLE_START_TAG, STYLE_END_TAG ); 1820 final String[] tags =TagParser.tag2Array( header, STYLE_START_TAG, STYLE_END_TAG ); 1821 final Set<String> origNameSet = pageNameMap.keySet(); 1822 for( int i=2; i<tags.length; i++ ) { 1823 for( final String origName : origNameSet ) { 1824 if( tags[i].indexOf( "=\"" + origName + "." ) >= 0 ) { 1825// final String styleName = TagParser.getValueFromTag( tags[i], STYLE_NAME_START_TAG, STYLE_NAME_END_TAG ); 1826 final String styleName = TagParser.getValueFromTag( tags[i], STYLE_NAME_START_TAG, END_KEY ); // 8.0.3.0 (2021/12/17) 1827 repStyleList.add( styleName ); 1828 break; 1829 } 1830 } 1831 } 1832 } 1833 1834 /** 1835 * 帳票処理キューを元に、content.xmlを書き換えます。(4) 1836 * まず、XMLを一旦メモリ上に展開した後、シート単位に分解し、データの埋め込みを行います。 1837 * 1838 * @og.rev 5.2.2.0 (2010/11/01) 条件付書式対応 1839 * @og.rev 8.0.3.0 (2021/12/17) TagParser.tag2Array を使用します。 1840 * @og.rev 8.5.4.2 (2024/01/12) PMD 7.0.0 CloseResource 対応 1841 */ 1842 private void execContentHeader() { 1843 final String fileName = path + "content.xml"; 1844 final String content = readOOoXml( fileName ); 1845 1846 // ファイルの解析し、シート+行単位に分解 1847// final String[] tags = tag2Array( content, STYLE_START_TAG, STYLE_END_TAG ); 1848 final String[] tags = TagParser.tag2Array( content, STYLE_START_TAG, STYLE_END_TAG ); 1849 final String header = tags[0]; 1850 final String footer = tags[1]; 1851 1852 // 8.5.4.2 (2024/01/12) PMD 7.0.0 CloseResource 対応 1853// BufferedWriter bw = null; 1854// try { 1855// bw = getWriter( fileName ); 1856 try ( BufferedWriter bw = getWriter( fileName ) ) { 1857 bw.write( xmlHeader ); 1858 bw.write( '\n' ); 1859 bw.write( header ); 1860 1861 // スタイル情報にシート依存の情報がある場合は、ページ分だけコピーする。 1862 // 6.3.9.0 (2015/11/06) entrySet イテレータではなく効率が悪い keySet イテレータを使用している 1863 for( int i=2; i<tags.length; i++ ) { 1864 boolean isReplace = false; 1865 for( final Map.Entry<String,List<String>> entry : pageNameMap.entrySet() ) { 1866 final String origName = entry.getKey(); 1867 if( tags[i].indexOf( "=\"" + origName + "." ) >= 0 ) { 1868 for( final String newName : entry.getValue() ) { 1869 String styleStr = tags[i].replace( "=\"" + origName + "." , "=\"" + newName + "." ); 1870 // シート名の書き換え 1871// final String styleName = TagParser.getValueFromTag( styleStr, STYLE_NAME_START_TAG, STYLE_NAME_END_TAG ); 1872 final String styleName = TagParser.getValueFromTag( styleStr, STYLE_NAME_START_TAG, END_KEY ); // 8.0.3.0 (2021/12/17) 1873 styleStr = styleStr.replace( STYLE_NAME_START_TAG + styleName, STYLE_NAME_START_TAG + styleName + "_" + newName ); 1874 bw.write( styleStr ); 1875 isReplace = true; 1876 } 1877 break; 1878 } 1879 } 1880 if( !isReplace ) { 1881 bw.write( tags[i] ); 1882 } 1883 } 1884 1885 bw.write( footer ); 1886 bw.flush(); 1887 } 1888 catch( final IOException ex ) { 1889 queue.addMsg( "[ERROR]PARSE:error occurer while write ReParsed Sheet " + fileName ); 1890 throw new HybsSystemException( ex ); 1891 } 1892// finally { 1893// Closer.ioClose( bw ); 1894// } 1895 } 1896 1897 /** 1898 * content.xmlのヘッダー部分を出力したcontent.xmlに、ヘッダー部分以降を出力した content.xml.bakをマージします。(5) 1899 * 1900 * @og.rev 5.2.2.0 (2010/11/01) 条件付書式対応 1901 */ 1902 private void execMergeContent() { 1903 FileChannel srcChannel = null; 1904 FileChannel destChannel = null; 1905 try { 1906 // 8.5.4.2 (2024/01/12) PMD 7.0.0 AvoidFileStream 対応 1907// srcChannel = new FileInputStream( path + "content.xml.tmp" ).getChannel(); 1908// destChannel = new FileOutputStream( path + "content.xml", true ).getChannel(); 1909 srcChannel = FileChannel.open( Paths.get(path + "content.xml.tmp"),READ ); 1910 destChannel = FileChannel.open(Paths.get(path + "content.xml"),APPEND ); 1911 1912 srcChannel.transferTo(0, srcChannel.size(), destChannel); 1913 } 1914 catch( final IOException ex ) { 1915 queue.addMsg( "[ERROR]PARSE:error occurer while merge content.xml" ); 1916 throw new HybsSystemException( ex ); 1917 } 1918 finally { 1919 Closer.ioClose( srcChannel ); 1920 Closer.ioClose( destChannel ); 1921 } 1922 FileUtil.deleteFiles( new File( path + "content.xml.tmp" ) ); 1923 } 1924 1925 /** 1926 * META-INF/manifest.xmlに、追加したオブジェクト(グラフ、画像)を登録します。(7) 1927 * 1928 * @og.rev 5.3.1.0 (2011/12/01) 新規作成 1929 */ 1930 private void execManifest() { 1931// final String fileName = path + "META-INF" + File.separator + "manifest.xml"; 1932 final String fileName = path + "META-INF" + FS + "manifest.xml"; // 8.0.3.0 (2021/12/17) 1933 final String[] conArr = TagParser.tag2Array( readOOoXml( fileName ), MANIFEST_START_TAG, MANIFEST_END_TAG ); 1934 1935 final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE ); 1936 buf.append( conArr[0] ); 1937 for( int i=2; i<conArr.length; i++ ) { 1938 buf.append( conArr[i] ); 1939 } 1940 for( final Map.Entry<String,String> entry : addObjMap.entrySet() ) { 1941 if( "graph".equals( entry.getValue() ) ) { 1942 buf.append( "<manifest:file-entry manifest:media-type=\"text/xml\" manifest:full-path=\"" ) 1943 .append( entry.getKey() ) 1944 .append( "/content.xml\"/><manifest:file-entry manifest:media-type=\"text/xml\" manifest:full-path=\"" ) 1945 .append( entry.getKey() ) 1946 .append( "/styles.xml\"/><manifest:file-entry manifest:media-type=\"text/xml\" manifest:full-path=\"" ) 1947 .append( entry.getKey() ) 1948 .append( "/meta.xml\"/><manifest:file-entry manifest:media-type=\"application/vnd.oasis.opendocument.chart\" manifest:full-path=\"" ) 1949 .append( entry.getKey() ).append( "/\"/>" ); // XML なので、このまま。 1950 } 1951 else { 1952 buf.append( "<manifest:file-entry manifest:media-type=\"image/" ) 1953 .append( entry.getValue() ).append( "\" manifest:full-path=\"" ) 1954 .append( entry.getKey() ).append( "\"/>" ); // XML なので、このまま。 1955 } 1956 } 1957 buf.append( conArr[1] ); 1958 1959 writeOOoXml( fileName, buf.toString() ); 1960 } 1961 1962 /** 1963 * XMLファイルを読み取り、結果を返します。 1964 * OOoのXMLファイルは全て1行めがxml宣言で、2行目が内容全体という形式であるため、 1965 * ここでは、2行目の内容部分を返します。 1966 * 1967 * @og.rev 4.3.6.0 (2009/04/01) meta.xmlでコンテンツの部分が改行されている場合があるため、ループを回して読込み 1968 * @og.rev 6.2.0.0 (2015/02/27) new BufferedReader … を、FileUtil.getBufferedReader … に変更。 1969 * @og.rev 6.3.1.0 (2015/06/28) nioを使用すると UTF-8とShuft-JISで、エラーになる。 1970 * @og.rev 6.5.0.1 (2016/10/21) CharacterCodingException は、OgCharacterException に変換する。 1971 * 1972 * @param fileName ファイル名 1973 * @return 読み取った文字列 1974 * @og.rtnNotNull 1975 */ 1976 private String readOOoXml( final String fileName ) { 1977 final File file = new File ( fileName ); 1978 1979 BufferedReader br = null; 1980 String tmp = null; 1981 final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE ); 1982 try { 1983 br = FileUtil.getBufferedReader( file, "UTF-8" ); // 6.2.0.0 (2015/02/27) 1984 xmlHeader = br.readLine(); 1985 while( ( tmp = br.readLine() ) != null ) { // 4.3.6.0 (2009/04/01) 1986 buf.append( tmp ); 1987 } 1988 } 1989 // 6.3.1.0 (2015/06/28) nioを使用すると UTF-8とShuft-JISで、エラーになる。 1990 catch( final CharacterCodingException ex ) { 1991 final String errMsg = "文字のエンコード・エラーが発生しました。" + CR 1992 + " ファイルのエンコードが指定のエンコードと異なります。" + CR 1993 + " [" + fileName + "] , Encode=[UTF-8]" ; 1994 throw new OgCharacterException( errMsg,ex ); // 6.5.0.1 (2016/10/21) 1995 } 1996 catch( final IOException ex ) { 1997 queue.addMsg( "[ERROR]PARSE:Failed to read " + fileName ); 1998 throw new HybsSystemException( ex ); 1999 } 2000 finally { 2001 Closer.ioClose( br ); 2002 } 2003 2004 final String str = buf.toString(); 2005 if( xmlHeader == null || xmlHeader.isEmpty() || str == null || str.isEmpty() ) { 2006 queue.addMsg( "[ERROR]PARSE:Maybe " + fileName + " is Broken!" ); 2007 throw new HybsSystemException(); 2008 } 2009 2010 return str; 2011 } 2012 2013 /** 2014 * XMLファイルを書き込みます。 2015 * OOoのXMLファイルは全て1行めがxml宣言で、2行目が内容全体という形式であるため、 2016 * ここでは、2行目の内容部分を渡すことで、XMLファイルを作成します。 2017 * 2018 * @og.rev 8.5.4.2 (2024/01/12) PMD 7.0.0 CloseResource 対応 2019 * 2020 * @param fileName 書き込むXMLファイル名 2021 * @param str 書き込む文字列 2022 */ 2023 private void writeOOoXml( final String fileName, final String str ) { 2024 // 8.5.4.2 (2024/01/12) PMD 7.0.0 CloseResource 対応 2025// BufferedWriter bw = null; 2026// try { 2027// bw = getWriter( fileName ); 2028 try ( BufferedWriter bw = getWriter( fileName ) ) { 2029 bw.write( xmlHeader ); 2030 bw.write( '\n' ); 2031 bw.write( str ); 2032 bw.flush(); 2033 } 2034 catch( final IOException ex ) { 2035 queue.addMsg( "[ERROR]PARSE:Failed to write " + fileName ); 2036 throw new HybsSystemException( ex ); 2037 } 2038// finally { 2039// Closer.ioClose( bw ); 2040// } 2041 } 2042 2043 /** 2044 * XMLファイル書き込み用のライターを返します。 2045 * 2046 * @param fileName ファイル名 2047 * @return ライター 2048 * @og.rtnNotNull 2049 */ 2050 private BufferedWriter getWriter( final String fileName ) { 2051 return getWriter( fileName, false ); 2052 } 2053 2054 /** 2055 * XMLファイル書き込み用のライターを返します。 2056 * 2057 * @param fileName ファイル名 2058 * @param append アベンドするか 2059 * @return ライター 2060 * @og.rtnNotNull 2061 */ 2062 private BufferedWriter getWriter( final String fileName, final boolean append ) { 2063 final File file = new File ( fileName ); 2064 BufferedWriter bw; 2065 try { 2066 // 8.5.4.2 (2024/01/12) PMD 7.0.0 AvoidFileStream 対応 2067// bw = new BufferedWriter( new OutputStreamWriter( new FileOutputStream( file, append ), "UTF-8" ) ); 2068 bw = Files.newBufferedWriter(file.toPath(),UTF_8,append ? APPEND : CREATE); 2069 } 2070 catch( final UnsupportedEncodingException ex ) { 2071 queue.addMsg( "[ERROR]PARSE:Input File is written by Unsupported Encoding" ); 2072 throw new HybsSystemException( ex ); 2073 } 2074 catch( final FileNotFoundException ex ) { 2075 queue.addMsg( "[ERROR]PARSE:File not Found" ); 2076 throw new HybsSystemException( ex ); 2077 } 2078 // 8.5.4.2 (2024/01/12) PMD 7.0.0 AvoidFileStream 対応 2079 catch( final IOException ex ) { 2080 queue.addMsg( "[ERROR]PARSE:File I/O error" ); 2081 throw new HybsSystemException( ex ); 2082 } 2083 return bw; 2084 } 2085 2086 /** 2087 * ファイル名から拡張子(小文字)を求めます。 2088 * 2089 * @param fileName 拡張子を取得する為のファイル名 2090 * @return 拡張子(小文字) 2091 */ 2092 public static String getSuffix( final String fileName ) { 2093 String suffix = null; 2094 if( fileName != null ) { 2095 final int sufIdx = fileName.lastIndexOf( '.' ); 2096 if( sufIdx >= 0 ) { 2097 suffix = fileName.substring( sufIdx + 1 ).toLowerCase( Locale.JAPAN ); 2098 } 2099 } 2100 return suffix; 2101 } 2102}