001/*
002 * Licensed to the Apache Software Foundation (ASF) under one or more
003 * contributor license agreements.  See the NOTICE file distributed with
004 * this work for additional information regarding copyright ownership.
005 * The ASF licenses this file to You under the Apache License, Version 2.0
006 * (the "License"); you may not use this file except in compliance with
007 * the License.  You may obtain a copy of the License at
008 *
009 *      http://www.apache.org/licenses/LICENSE-2.0
010 *
011 * Unless required by applicable law or agreed to in writing, software
012 * distributed under the License is distributed on an "AS IS" BASIS,
013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
014 * See the License for the specific language governing permissions and
015 * limitations under the License.
016 */
017package org.apache.commons.math4.legacy.util;
018
019import java.text.FieldPosition;
020import java.text.NumberFormat;
021import java.text.ParsePosition;
022import java.util.Locale;
023
024/**
025 * Base class for formatters of composite objects (complex numbers, vectors ...).
026 *
027 */
028public final class CompositeFormat {
029
030    /**
031     * Class contains only static methods.
032     */
033    private CompositeFormat() {}
034
035    /**
036     * Create a default number format.  The default number format is based on
037     * {@link NumberFormat#getInstance()} with the only customizing that the
038     * maximum number of fraction digits is set to 10.
039     * @return the default number format.
040     */
041    public static NumberFormat getDefaultNumberFormat() {
042        return getDefaultNumberFormat(Locale.getDefault());
043    }
044
045    /**
046     * Create a default number format.  The default number format is based on
047     * {@link NumberFormat#getInstance(java.util.Locale)} with the only
048     * customizing that the maximum number of fraction digits is set to 10.
049     * @param locale the specific locale used by the format.
050     * @return the default number format specific to the given locale.
051     */
052    public static NumberFormat getDefaultNumberFormat(final Locale locale) {
053        final NumberFormat nf = NumberFormat.getInstance(locale);
054        nf.setMaximumFractionDigits(10);
055        return nf;
056    }
057
058    /**
059     * Parses <code>source</code> until a non-whitespace character is found.
060     *
061     * @param source the string to parse
062     * @param pos input/output parsing parameter.  On output, <code>pos</code>
063     *        holds the index of the next non-whitespace character.
064     */
065    public static void parseAndIgnoreWhitespace(final String source,
066                                                final ParsePosition pos) {
067        parseNextCharacter(source, pos);
068        pos.setIndex(pos.getIndex() - 1);
069    }
070
071    /**
072     * Parses <code>source</code> until a non-whitespace character is found.
073     *
074     * @param source the string to parse
075     * @param pos input/output parsing parameter.
076     * @return the first non-whitespace character.
077     */
078    public static char parseNextCharacter(final String source,
079                                          final ParsePosition pos) {
080         int index = pos.getIndex();
081         final int n = source.length();
082         char ret = 0;
083
084         if (index < n) {
085             char c;
086             do {
087                 c = source.charAt(index++);
088             } while (Character.isWhitespace(c) && index < n);
089             pos.setIndex(index);
090
091             if (index < n) {
092                 ret = c;
093             }
094         }
095
096         return ret;
097    }
098
099    /**
100     * Parses <code>source</code> for special double values.  These values
101     * include Double.NaN, Double.POSITIVE_INFINITY, Double.NEGATIVE_INFINITY.
102     *
103     * @param source the string to parse
104     * @param value the special value to parse.
105     * @param pos input/output parsing parameter.
106     * @return the special number.
107     */
108    private static Number parseNumber(final String source, final double value,
109                                      final ParsePosition pos) {
110        Number ret = null;
111
112        StringBuilder sb = new StringBuilder();
113        sb.append('(');
114        sb.append(value);
115        sb.append(')');
116
117        final int n = sb.length();
118        final int startIndex = pos.getIndex();
119        final int endIndex = startIndex + n;
120        if (endIndex < source.length() &&
121            source.substring(startIndex, endIndex).compareTo(sb.toString()) == 0) {
122            ret = Double.valueOf(value);
123            pos.setIndex(endIndex);
124        }
125
126        return ret;
127    }
128
129    /**
130     * Parses <code>source</code> for a number.  This method can parse normal,
131     * numeric values as well as special values.  These special values include
132     * Double.NaN, Double.POSITIVE_INFINITY, Double.NEGATIVE_INFINITY.
133     *
134     * @param source the string to parse
135     * @param format the number format used to parse normal, numeric values.
136     * @param pos input/output parsing parameter.
137     * @return the parsed number.
138     */
139    public static Number parseNumber(final String source, final NumberFormat format,
140                                     final ParsePosition pos) {
141        final int startIndex = pos.getIndex();
142        Number number = format.parse(source, pos);
143        final int endIndex = pos.getIndex();
144
145        // check for error parsing number
146        if (startIndex == endIndex) {
147            // try parsing special numbers
148            final double[] special = {
149                Double.NaN, Double.POSITIVE_INFINITY, Double.NEGATIVE_INFINITY
150            };
151            for (int i = 0; i < special.length; ++i) {
152                number = parseNumber(source, special[i], pos);
153                if (number != null) {
154                    break;
155                }
156            }
157        }
158
159        return number;
160    }
161
162    /**
163     * Parse <code>source</code> for an expected fixed string.
164     * @param source the string to parse
165     * @param expected expected string
166     * @param pos input/output parsing parameter.
167     * @return true if the expected string was there
168     */
169    public static boolean parseFixedstring(final String source,
170                                           final String expected,
171                                           final ParsePosition pos) {
172
173        final int startIndex = pos.getIndex();
174        final int endIndex = startIndex + expected.length();
175        if (startIndex >= source.length() ||
176            endIndex > source.length() ||
177            source.substring(startIndex, endIndex).compareTo(expected) != 0) {
178            // set index back to start, error index should be the start index
179            pos.setIndex(startIndex);
180            pos.setErrorIndex(startIndex);
181            return false;
182        }
183
184        // the string was here
185        pos.setIndex(endIndex);
186        return true;
187    }
188
189    /**
190     * Formats a double value to produce a string.  In general, the value is
191     * formatted using the formatting rules of <code>format</code>.  There are
192     * three exceptions to this:
193     * <ol>
194     * <li>NaN is formatted as '(NaN)'</li>
195     * <li>Positive infinity is formatted as '(Infinity)'</li>
196     * <li>Negative infinity is formatted as '(-Infinity)'</li>
197     * </ol>
198     *
199     * @param value the double to format.
200     * @param format the format used.
201     * @param toAppendTo where the text is to be appended
202     * @param pos On input: an alignment field, if desired. On output: the
203     *            offsets of the alignment field
204     * @return the value passed in as toAppendTo.
205     */
206    public static StringBuffer formatDouble(final double value, final NumberFormat format,
207                                            final StringBuffer toAppendTo,
208                                            final FieldPosition pos) {
209        if( Double.isNaN(value) || Double.isInfinite(value) ) {
210            toAppendTo.append('(');
211            toAppendTo.append(value);
212            toAppendTo.append(')');
213        } else {
214            format.format(value, toAppendTo, pos);
215        }
216        return toAppendTo;
217    }
218}