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.column;
017
018import java.text.DecimalFormat;
019
020import org.opengion.hayabusa.db.AbstractRenderer;
021import org.opengion.hayabusa.db.CellRenderer;
022import org.opengion.hayabusa.db.DBColumn;
023import org.opengion.fukurou.util.StringUtil;
024
025/**
026 * DECIMAL レンデラーは、カラムのデータをDecimal(10進数、小数)表示する場合に
027 * 使用するクラスです。
028 * 負数の場合はspanタグclass="minus"を付けて出力します。
029 *
030 * 表示パラメータに与えられた文字列は、java.text.DecimalFormat を
031 * 使用してフォーマットされます。
032 * フォーマット変換前に、カンマなどの数値変換時にエラーになる情報を削除しておきます。
033 *
034 * フォーマットルールは、{@link java.text.DecimalFormat} を参照願います。
035 * 標準のフォーマットは"#,##0.#"です。
036 *
037 *  カラムの表示に必要な属性は、DBColumn オブジェクト より取り出します。
038 * このクラスは、DBColumn オブジェクト毎に1つ作成されます。
039 *
040 * @og.rev 3.5.0.0 (2003/09/17) 新規作成
041 * @og.rev 5.4.3.6 (2012/01/19) コメント修正
042 * @og.group データ表示
043 *
044 * @version  4.0
045 * @author       Kazuhiko Hasegawa
046 * @since    JDK5.0,
047 */
048public class Renderer_DECIMAL extends AbstractRenderer {
049        /** このプログラムのVERSION文字列を設定します。   {@value} */
050        private static final String VERSION = "6.8.3.1 (2017/12/01)" ;
051
052        private final DecimalFormat format ;
053        private final String defValue ;
054        private final String noDisplayVal ;             // 5.6.2.3 (2013/03/22)
055
056        /** 6.8.3.1 (2017/12/01) 標準オブジェクトのキャッシュ */
057        private static final CellRenderer DB_CELL = new Renderer_DECIMAL();
058
059        /**
060         * デフォルトコンストラクター。
061         * このコンストラクターで、基本オブジェクトを作成します。
062         *
063         * @og.rev 5.6.2.3 (2013/03/22) noDisplayVal 変数初期化
064         * @og.rev 6.8.3.1 (2017/12/01) 標準オブジェクトとして、利用できる形にして、キャッシュします。
065         */
066        public Renderer_DECIMAL() {
067                super();                // 6.4.1.1 (2016/01/16) PMD refactoring. It is a good practice to call super() in a constructor
068                format                  = new DecimalFormat( "#,##0.#" );       // 6.8.3.1 (2017/12/01)
069                defValue                = "";                   // 5.1.1.0 (2009/12/01)
070                noDisplayVal    = null;                 // 5.5.1.0 (2012/04/03)
071        }
072
073        /**
074         * DBColumnオブジェクトを指定したprivateコンストラクター。
075         *
076         * @param       clm     DBColumnオブジェクト
077         *
078         * @og.rev 5.1.1.0 (2009/12/01) 初期値がnullの場合は、defValueはnullとする。
079         * @og.rev 5.6.2.3 (2013/03/22) noDisplayVal 変数追加
080         */
081        private Renderer_DECIMAL( final DBColumn clm ) {
082                super();                // 6.4.1.1 (2016/01/16) PMD refactoring. It is a good practice to call super() in a constructor
083
084                String fm = clm.getRendererParam();
085//              if( fm == null || fm.isEmpty() || fm.equals( "_" ) ) {
086                if( fm == null || fm.isEmpty() || "_".equals( fm ) ) {  // 8.5.4.2 (2024/01/12) PMD 7.0.0 LiteralsFirstInComparisons
087                        fm = "#,##0.#";
088                }
089                format = new DecimalFormat( fm );
090
091                // 5.1.1.0 (2009/12/01)
092                final String defv = clm.getDefault();
093                if( defv == null || defv.isEmpty() ) {
094                        defValue = "";
095                }
096                else {
097                        final double dd = StringUtil.parseDouble( defv );
098                        defValue = format.format( dd );
099                }
100                noDisplayVal = clm.getNoDisplayVal();   // 5.6.2.3 (2013/03/22)
101        }
102
103        /**
104         * 各オブジェクトから自分のインスタンスを返します。
105         * 自分自身をキャッシュするのか、新たに作成するのかは、各サブクラスの実装に
106         * まかされます。
107         *
108         * @og.rev 6.8.3.1 (2017/12/01) 標準オブジェクトとして、利用できる形にして、キャッシュします。
109         *
110         * @param       clm     DBColumnオブジェクト
111         * @return      CellRendererオブジェクト
112         * @og.rtnNotNull
113         */
114        public CellRenderer newInstance( final DBColumn clm ) {
115                // 6.8.3.1 (2017/12/01) 標準オブジェクトとして、利用できる形にして、キャッシュします。
116
117                final String fm   = clm.getRendererParam();
118                final String defv = clm.getDefault();
119
120                // 8.5.5.1 (2024/02/29) PMD 7.0.0 OnlyOneReturn メソッドには終了ポイントが 1 つだけ必要
121////            if( ( fm == null || fm.isEmpty() || fm.equals( "_" ) ) &&
122////                    ( defv == null || defv.isEmpty() || defv.equals( "_" ) ) ) {
123//              if( ( fm == null || fm.isEmpty() || "_".equals( fm ) ) &&                       // 8.5.4.2 (2024/01/12) PMD 7.0.0 LiteralsFirstInComparisons
124//                      ( defv == null || defv.isEmpty() || "_".equals( defv ) ) ) {    // 8.5.4.2 (2024/01/12) PMD 7.0.0 LiteralsFirstInComparisons
125//                              return DB_CELL;
126//              }
127//
128//              return new Renderer_DECIMAL( clm );
129
130                return  ( fm == null || fm.isEmpty() || "_".equals( fm ) ) &&                   // 8.5.4.2 (2024/01/12) PMD 7.0.0 LiteralsFirstInComparisons
131                                ( defv == null || defv.isEmpty() || "_".equals( defv ) )                // 8.5.4.2 (2024/01/12) PMD 7.0.0 LiteralsFirstInComparisons
132                        ? DB_CELL
133                        : new Renderer_DECIMAL( clm );
134        }
135
136        /**
137         * データの表示用文字列を返します。
138         *
139         * ここでは、値が 0 の場合に、初期値を使うロジックが組み込まれており、
140         * 従来はこれを利用して、初期値にゼロ文字列を設定することで、"0" を
141         * 非表示文字として扱っていました。
142         * 互換性の問題があるので、既存の機能は残しますが、この処理は、noDisplayVal を
143         * 利用した処理に置き換えてください。
144         * 他の処理(NUMBER,MONEY,DBMENUなど)は、noDisplayVal を使います。
145         *
146         * @og.rev 3.8.5.2 (2006/05/31) カンマ編集された数字の対応
147         * @og.rev 5.3.1.0 (2009/12/01) 値が0の場合は、初期値を適用する。
148         * @og.rev 5.6.2.3 (2013/03/22) noDisplayVal 変数追加
149         * @og.rev 6.0.4.0 (2014/11/28) ロジックの共通化
150         *
151         * @param       value 入力値
152         * @return      データの表示用文字列
153         */
154        @Override
155        public String getValue( final String value ) {
156                return getValue( value , true );
157        }
158
159        /**
160         * データ出力用の文字列を作成します。
161         * ファイル等に出力する形式を想定しますので、HTMLタグを含まない
162         * データを返します。
163         * 基本は、#getValue( String ) をそのまま返します。
164         *
165         * @og.rev 6.0.4.0 (2014/11/28) データ出力用のレンデラー
166         *
167         * @param   value 入力値
168         * @return  データ出力用の文字列
169         * @see         #getValue( String )
170         */
171        @Override
172        public String getWriteValue( final String value ) {
173                return getValue( value , false );
174        }
175
176        /**
177         * データ表示用/出力用の文字列を作成します。
178         * 第二引数の isView == true で、データ表示用文字列を、false で
179         * データ出力用の文字列を作成します。
180         * 処理の共通化を行うためのメソッドです。
181         *
182         * @og.rev 6.0.4.0 (2014/11/28) ロジックの共通化
183         *
184         * @param   value  入力値
185         * @param   isView データ表示用かどうか(true:表示用/false:出力用)
186         * @return  データ表示用/出力用の文字列
187         * @see         #getValue( String )
188         */
189        private String getValue( final String value , final boolean isView ) {
190                // 8.5.5.1 (2024/02/29) PMD 7.0.0 OnlyOneReturn メソッドには終了ポイントが 1 つだけ必要
191                // 5.6.2.3 (2013/03/22) noDisplayVal 変数追加
192//              if( noDisplayVal != null && noDisplayVal.equalsIgnoreCase( value ) ) { return "" ; }
193
194                final String rtnVal;
195                if( noDisplayVal != null && noDisplayVal.equalsIgnoreCase( value ) ) {
196                        rtnVal = "";
197                }
198
199                // 8.5.4.2 (2024/01/12) PMD 7.0.0 InefficientEmptyStringCheck 対応
200//              if( value == null || value.trim().isEmpty() || "0".equals( value ) ) {
201                else if( StringUtil.isNull( value ) || "0".equals( value ) ) {
202//                      return defValue;
203                        rtnVal = defValue;
204                }
205                else {
206                        final double dd = StringUtil.parseDouble( value );
207
208                        // 6.4.2.1 (2016/02/05) PMD refactoring.
209                        final String rtn;
210                        synchronized( format ) {
211                                rtn = format.format( dd );
212                        }
213
214//                      return isView && dd < 0.0
215                        rtnVal = isView && dd < 0.0
216                                                ? ("<span class=\"minus\">" + rtn + "</span>")
217                                                : rtn;
218                }
219                return rtnVal;
220        }
221
222        /**
223         * name属性を変えた、データ表示用のHTML文字列を作成します。
224         * レンデラーのため、row(行番号)は使いません。
225         * 第3引数に、パラメータを渡すことが出来ます。これは、viewMarker で
226         * [$XXXX param] 形式を渡すことで、行単位に表示形式を変更できます。
227         *
228         * @og.rev 6.8.3.1 (2017/12/01) パラメータを渡せるようにします。
229         *
230         * @param   row   行番号
231         * @param   value 値
232         * @param   param パラメータ
233         * @return  データ表示/編集用の文字列
234         */
235        @Override
236        public String getValue( final int row,final String value,final String param ) {
237                // 8.5.5.1 (2024/02/29) PMD 7.0.0 OnlyOneReturn メソッドには終了ポイントが 1 つだけ必要
238                // 5.6.2.3 (2013/03/22) noDisplayVal 変数追加
239//              if( noDisplayVal != null && noDisplayVal.equalsIgnoreCase( value ) ) { return "" ; }
240
241                final String rtnVal;
242                if( noDisplayVal != null && noDisplayVal.equalsIgnoreCase( value ) ) {
243                        rtnVal = "";
244                }
245
246                // 8.5.4.2 (2024/01/12) PMD 7.0.0 InefficientEmptyStringCheck 対応
247//              if( value == null || value.trim().isEmpty() || "0".equals( value ) ) {
248                else if( StringUtil.isNull( value ) || "0".equals( value ) ) {
249//                      return defValue;
250                        rtnVal = defValue;
251                }
252                else {
253                        final double dd = StringUtil.parseDouble( value );
254
255                        String fm = param;
256        //              if( fm == null || fm.isEmpty() || fm.equals( "_" ) ) {
257                        if( fm == null || fm.isEmpty() || "_".equals( fm ) ) {  // 8.5.4.2 (2024/01/12) PMD 7.0.0 LiteralsFirstInComparisons
258                                fm = "#,##0.#";
259                        }
260                        final DecimalFormat format = new DecimalFormat( fm );
261                        final String rtn = format.format( dd );
262
263//                      return dd < 0.0
264                        rtnVal = dd < 0.0
265                                                ? ("<span class=\"minus\">" + rtn + "</span>")
266                                                : rtn;
267                }
268                return rtnVal;
269        }
270}