Bug fixes
This commit is contained in:
parent
5ddd8a0940
commit
4ddfda6340
File diff suppressed because it is too large
Load Diff
@ -18,6 +18,7 @@ import android.content.res.Configuration;
|
|||||||
import android.text.format.DateFormat;
|
import android.text.format.DateFormat;
|
||||||
import android.util.Xml;
|
import android.util.Xml;
|
||||||
|
|
||||||
|
import org.telegram.android.time.FastDateFormat;
|
||||||
import org.telegram.messenger.ConnectionsManager;
|
import org.telegram.messenger.ConnectionsManager;
|
||||||
import org.telegram.messenger.FileLog;
|
import org.telegram.messenger.FileLog;
|
||||||
import org.telegram.messenger.R;
|
import org.telegram.messenger.R;
|
||||||
|
@ -3886,8 +3886,13 @@ public class MessagesController implements NotificationCenter.NotificationCenter
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
if (chat.seq_in != layer.out_seq_no && chat.seq_in != layer.out_seq_no - 2) {
|
if (chat.seq_in != layer.out_seq_no && chat.seq_in != layer.out_seq_no - 2) {
|
||||||
/*TLRPC.Message decryptedMessage = processDecryptedObject(chat, message, layer.message);
|
ArrayList<TLRPC.TL_decryptedMessageHolder> arr = secretHolesQueue.get(chat.id);
|
||||||
if (decryptedMessage == null) {
|
if (arr == null) {
|
||||||
|
arr = new ArrayList<TLRPC.TL_decryptedMessageHolder>();
|
||||||
|
secretHolesQueue.put(chat.id, arr);
|
||||||
|
}
|
||||||
|
if (arr.size() >= 10) {
|
||||||
|
secretHolesQueue.remove(chat.id);
|
||||||
final TLRPC.TL_encryptedChatDiscarded newChat = new TLRPC.TL_encryptedChatDiscarded();
|
final TLRPC.TL_encryptedChatDiscarded newChat = new TLRPC.TL_encryptedChatDiscarded();
|
||||||
newChat.id = chat.id;
|
newChat.id = chat.id;
|
||||||
newChat.user_id = chat.user_id;
|
newChat.user_id = chat.user_id;
|
||||||
@ -3903,12 +3908,9 @@ public class MessagesController implements NotificationCenter.NotificationCenter
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
declineSecretChat(chat.id);
|
declineSecretChat(chat.id);
|
||||||
}*/
|
return null;
|
||||||
ArrayList<TLRPC.TL_decryptedMessageHolder> arr = secretHolesQueue.get(chat.id);
|
|
||||||
if (arr == null) {
|
|
||||||
arr = new ArrayList<TLRPC.TL_decryptedMessageHolder>();
|
|
||||||
secretHolesQueue.put(chat.id, arr);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TLRPC.TL_decryptedMessageHolder holder = new TLRPC.TL_decryptedMessageHolder();
|
TLRPC.TL_decryptedMessageHolder holder = new TLRPC.TL_decryptedMessageHolder();
|
||||||
holder.layer = layer;
|
holder.layer = layer;
|
||||||
holder.file = message.file;
|
holder.file = message.file;
|
||||||
|
@ -110,7 +110,7 @@ public class MessagesStorage {
|
|||||||
database.executeFast("CREATE TABLE download_queue(uid INTEGER, type INTEGER, date INTEGER, data BLOB, PRIMARY KEY (uid, type));").stepThis().dispose();
|
database.executeFast("CREATE TABLE download_queue(uid INTEGER, type INTEGER, date INTEGER, data BLOB, PRIMARY KEY (uid, type));").stepThis().dispose();
|
||||||
database.executeFast("CREATE TABLE dialog_settings(did INTEGER PRIMARY KEY, flags INTEGER);").stepThis().dispose();
|
database.executeFast("CREATE TABLE dialog_settings(did INTEGER PRIMARY KEY, flags INTEGER);").stepThis().dispose();
|
||||||
database.executeFast("CREATE TABLE messages_seq(mid INTEGER PRIMARY KEY, seq_in INTEGER, seq_out INTEGER);").stepThis().dispose();
|
database.executeFast("CREATE TABLE messages_seq(mid INTEGER PRIMARY KEY, seq_in INTEGER, seq_out INTEGER);").stepThis().dispose();
|
||||||
database.executeFast("CREATE TABLE secret_holes(uid INTEGER, seq_in INTEGER, seq_out INTEGER, data BLOB, PRIMARY KEY (uid, seq_in, seq_out));").stepThis().dispose();
|
//database.executeFast("CREATE TABLE secret_holes(uid INTEGER, seq_in INTEGER, seq_out INTEGER, data BLOB, PRIMARY KEY (uid, seq_in, seq_out));").stepThis().dispose();
|
||||||
|
|
||||||
//database.executeFast("CREATE TABLE attach_data(uid INTEGER, id INTEGER, data BLOB, PRIMARY KEY (uid, id))").stepThis().dispose();
|
//database.executeFast("CREATE TABLE attach_data(uid INTEGER, id INTEGER, data BLOB, PRIMARY KEY (uid, id))").stepThis().dispose();
|
||||||
|
|
||||||
@ -142,7 +142,7 @@ public class MessagesStorage {
|
|||||||
|
|
||||||
database.executeFast("CREATE INDEX IF NOT EXISTS seq_idx_messages_seq ON messages_seq(seq_in, seq_out);").stepThis().dispose();
|
database.executeFast("CREATE INDEX IF NOT EXISTS seq_idx_messages_seq ON messages_seq(seq_in, seq_out);").stepThis().dispose();
|
||||||
|
|
||||||
database.executeFast("PRAGMA user_version = 8").stepThis().dispose();
|
database.executeFast("PRAGMA user_version = 7").stepThis().dispose();
|
||||||
} else {
|
} else {
|
||||||
try {
|
try {
|
||||||
SQLiteCursor cursor = database.queryFinalized("SELECT seq, pts, date, qts, lsv, sg, pbytes FROM params WHERE id = 1");
|
SQLiteCursor cursor = database.queryFinalized("SELECT seq, pts, date, qts, lsv, sg, pbytes FROM params WHERE id = 1");
|
||||||
@ -174,7 +174,7 @@ public class MessagesStorage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
int version = database.executeInt("PRAGMA user_version");
|
int version = database.executeInt("PRAGMA user_version");
|
||||||
if (version < 8) {
|
if (version < 7) {
|
||||||
updateDbToLastVersion(version);
|
updateDbToLastVersion(version);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -304,11 +304,11 @@ public class MessagesStorage {
|
|||||||
database.executeFast("PRAGMA user_version = 7").stepThis().dispose();
|
database.executeFast("PRAGMA user_version = 7").stepThis().dispose();
|
||||||
version = 7;
|
version = 7;
|
||||||
}
|
}
|
||||||
if (version == 7 && version < 8) {
|
/*if (version == 7 && version < 8) {
|
||||||
database.executeFast("CREATE TABLE IF NOT EXISTS secret_holes(uid INTEGER, seq_in INTEGER, seq_out INTEGER, data BLOB, PRIMARY KEY (uid, seq_in, seq_out));").stepThis().dispose();
|
database.executeFast("CREATE TABLE IF NOT EXISTS secret_holes(uid INTEGER, seq_in INTEGER, seq_out INTEGER, data BLOB, PRIMARY KEY (uid, seq_in, seq_out));").stepThis().dispose();
|
||||||
database.executeFast("PRAGMA user_version = 8").stepThis().dispose();
|
database.executeFast("PRAGMA user_version = 8").stepThis().dispose();
|
||||||
version = 8;
|
version = 8;
|
||||||
}
|
}*/
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
FileLog.e("tmessages", e);
|
FileLog.e("tmessages", e);
|
||||||
}
|
}
|
||||||
@ -671,7 +671,7 @@ public class MessagesStorage {
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
database.executeFast("DELETE FROM enc_chats WHERE uid = " + high_id).stepThis().dispose();
|
database.executeFast("DELETE FROM enc_chats WHERE uid = " + high_id).stepThis().dispose();
|
||||||
database.executeFast("DELETE FROM secret_holes WHERE uid = " + high_id).stepThis().dispose();
|
//database.executeFast("DELETE FROM secret_holes WHERE uid = " + high_id).stepThis().dispose();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
database.executeFast("UPDATE dialogs SET unread_count = 0 WHERE did = " + did).stepThis().dispose();
|
database.executeFast("UPDATE dialogs SET unread_count = 0 WHERE did = " + did).stepThis().dispose();
|
||||||
@ -2805,7 +2805,7 @@ public class MessagesStorage {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public void getHoleMessages() {
|
/*public void getHoleMessages() {
|
||||||
storageQueue.postRunnable(new Runnable() {
|
storageQueue.postRunnable(new Runnable() {
|
||||||
@Override
|
@Override
|
||||||
public void run() {
|
public void run() {
|
||||||
@ -2857,7 +2857,7 @@ public class MessagesStorage {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}*/
|
||||||
|
|
||||||
public void setMessageSeq(final int mid, final int seq_in, final int seq_out) {
|
public void setMessageSeq(final int mid, final int seq_in, final int seq_out) {
|
||||||
storageQueue.postRunnable(new Runnable() {
|
storageQueue.postRunnable(new Runnable() {
|
||||||
|
@ -0,0 +1,107 @@
|
|||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package org.telegram.android.time;
|
||||||
|
|
||||||
|
import java.text.ParseException;
|
||||||
|
import java.text.ParsePosition;
|
||||||
|
import java.util.Date;
|
||||||
|
import java.util.Locale;
|
||||||
|
import java.util.TimeZone;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>DateParser is the "missing" interface for the parsing methods of
|
||||||
|
* {@link java.text.DateFormat}.</p>
|
||||||
|
*
|
||||||
|
* @since 3.2
|
||||||
|
*/
|
||||||
|
public interface DateParser {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Equivalent to DateFormat.parse(String).
|
||||||
|
* <p/>
|
||||||
|
* See {@link java.text.DateFormat#parse(String)} for more information.
|
||||||
|
*
|
||||||
|
* @param source A <code>String</code> whose beginning should be parsed.
|
||||||
|
* @return A <code>Date</code> parsed from the string
|
||||||
|
* @throws ParseException if the beginning of the specified string cannot be parsed.
|
||||||
|
*/
|
||||||
|
Date parse(String source) throws ParseException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Equivalent to DateFormat.parse(String, ParsePosition).
|
||||||
|
* <p/>
|
||||||
|
* See {@link java.text.DateFormat#parse(String, ParsePosition)} for more information.
|
||||||
|
*
|
||||||
|
* @param source A <code>String</code>, part of which should be parsed.
|
||||||
|
* @param pos A <code>ParsePosition</code> object with index and error index information
|
||||||
|
* as described above.
|
||||||
|
* @return A <code>Date</code> parsed from the string. In case of error, returns null.
|
||||||
|
* @throws NullPointerException if text or pos is null.
|
||||||
|
*/
|
||||||
|
Date parse(String source, ParsePosition pos);
|
||||||
|
|
||||||
|
// Accessors
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Get the pattern used by this parser.</p>
|
||||||
|
*
|
||||||
|
* @return the pattern, {@link java.text.SimpleDateFormat} compatible
|
||||||
|
*/
|
||||||
|
String getPattern();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Get the time zone used by this parser.
|
||||||
|
* </p>
|
||||||
|
* <p/>
|
||||||
|
* <p>
|
||||||
|
* The default {@link TimeZone} used to create a {@link Date} when the {@link TimeZone} is not specified by
|
||||||
|
* the format pattern.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* @return the time zone
|
||||||
|
*/
|
||||||
|
TimeZone getTimeZone();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Get the locale used by this parser.</p>
|
||||||
|
*
|
||||||
|
* @return the locale
|
||||||
|
*/
|
||||||
|
Locale getLocale();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses text from a string to produce a Date.
|
||||||
|
*
|
||||||
|
* @param source A <code>String</code> whose beginning should be parsed.
|
||||||
|
* @return a <code>java.util.Date</code> object
|
||||||
|
* @throws ParseException if the beginning of the specified string cannot be parsed.
|
||||||
|
* @see java.text.DateFormat#parseObject(String)
|
||||||
|
*/
|
||||||
|
Object parseObject(String source) throws ParseException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parse a date/time string according to the given parse position.
|
||||||
|
*
|
||||||
|
* @param source A <code>String</code> whose beginning should be parsed.
|
||||||
|
* @param pos the parse position
|
||||||
|
* @return a <code>java.util.Date</code> object
|
||||||
|
* @see java.text.DateFormat#parseObject(String, ParsePosition)
|
||||||
|
*/
|
||||||
|
Object parseObject(String source, ParsePosition pos);
|
||||||
|
}
|
@ -0,0 +1,126 @@
|
|||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package org.telegram.android.time;
|
||||||
|
|
||||||
|
import java.text.FieldPosition;
|
||||||
|
import java.util.Calendar;
|
||||||
|
import java.util.Date;
|
||||||
|
import java.util.Locale;
|
||||||
|
import java.util.TimeZone;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>DatePrinter is the "missing" interface for the format methods of
|
||||||
|
* {@link java.text.DateFormat}.</p>
|
||||||
|
*
|
||||||
|
* @since 3.2
|
||||||
|
*/
|
||||||
|
public interface DatePrinter {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Formats a millisecond {@code long} value.</p>
|
||||||
|
*
|
||||||
|
* @param millis the millisecond value to format
|
||||||
|
* @return the formatted string
|
||||||
|
* @since 2.1
|
||||||
|
*/
|
||||||
|
String format(long millis);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Formats a {@code Date} object using a {@code GregorianCalendar}.</p>
|
||||||
|
*
|
||||||
|
* @param date the date to format
|
||||||
|
* @return the formatted string
|
||||||
|
*/
|
||||||
|
String format(Date date);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Formats a {@code Calendar} object.</p>
|
||||||
|
*
|
||||||
|
* @param calendar the calendar to format
|
||||||
|
* @return the formatted string
|
||||||
|
*/
|
||||||
|
String format(Calendar calendar);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Formats a milliseond {@code long} value into the
|
||||||
|
* supplied {@code StringBuffer}.</p>
|
||||||
|
*
|
||||||
|
* @param millis the millisecond value to format
|
||||||
|
* @param buf the buffer to format into
|
||||||
|
* @return the specified string buffer
|
||||||
|
*/
|
||||||
|
StringBuffer format(long millis, StringBuffer buf);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Formats a {@code Date} object into the
|
||||||
|
* supplied {@code StringBuffer} using a {@code GregorianCalendar}.</p>
|
||||||
|
*
|
||||||
|
* @param date the date to format
|
||||||
|
* @param buf the buffer to format into
|
||||||
|
* @return the specified string buffer
|
||||||
|
*/
|
||||||
|
StringBuffer format(Date date, StringBuffer buf);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Formats a {@code Calendar} object into the
|
||||||
|
* supplied {@code StringBuffer}.</p>
|
||||||
|
*
|
||||||
|
* @param calendar the calendar to format
|
||||||
|
* @param buf the buffer to format into
|
||||||
|
* @return the specified string buffer
|
||||||
|
*/
|
||||||
|
StringBuffer format(Calendar calendar, StringBuffer buf);
|
||||||
|
|
||||||
|
// Accessors
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets the pattern used by this printer.</p>
|
||||||
|
*
|
||||||
|
* @return the pattern, {@link java.text.SimpleDateFormat} compatible
|
||||||
|
*/
|
||||||
|
String getPattern();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets the time zone used by this printer.</p>
|
||||||
|
* <p/>
|
||||||
|
* <p>This zone is always used for {@code Date} printing. </p>
|
||||||
|
*
|
||||||
|
* @return the time zone
|
||||||
|
*/
|
||||||
|
TimeZone getTimeZone();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets the locale used by this printer.</p>
|
||||||
|
*
|
||||||
|
* @return the locale
|
||||||
|
*/
|
||||||
|
Locale getLocale();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Formats a {@code Date}, {@code Calendar} or
|
||||||
|
* {@code Long} (milliseconds) object.</p>
|
||||||
|
* <p/>
|
||||||
|
* See {@link java.text.DateFormat#format(Object, StringBuffer, FieldPosition)}
|
||||||
|
*
|
||||||
|
* @param obj the object to format
|
||||||
|
* @param toAppendTo the buffer to append to
|
||||||
|
* @param pos the position - ignored
|
||||||
|
* @return the buffer passed in
|
||||||
|
*/
|
||||||
|
StringBuffer format(Object obj, StringBuffer toAppendTo, FieldPosition pos);
|
||||||
|
}
|
@ -0,0 +1,613 @@
|
|||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package org.telegram.android.time;
|
||||||
|
|
||||||
|
import java.text.DateFormat;
|
||||||
|
import java.text.FieldPosition;
|
||||||
|
import java.text.Format;
|
||||||
|
import java.text.ParseException;
|
||||||
|
import java.text.ParsePosition;
|
||||||
|
import java.util.Calendar;
|
||||||
|
import java.util.Date;
|
||||||
|
import java.util.Locale;
|
||||||
|
import java.util.TimeZone;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>FastDateFormat is a fast and thread-safe version of
|
||||||
|
* {@link java.text.SimpleDateFormat}.</p>
|
||||||
|
*
|
||||||
|
* <p>This class can be used as a direct replacement to
|
||||||
|
* {@code SimpleDateFormat} in most formatting and parsing situations.
|
||||||
|
* This class is especially useful in multi-threaded server environments.
|
||||||
|
* {@code SimpleDateFormat} is not thread-safe in any JDK version,
|
||||||
|
* nor will it be as Sun have closed the bug/RFE.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <p>All patterns are compatible with
|
||||||
|
* SimpleDateFormat (except time zones and some year patterns - see below).</p>
|
||||||
|
*
|
||||||
|
* <p>Since 3.2, FastDateFormat supports parsing as well as printing.</p>
|
||||||
|
*
|
||||||
|
* <p>Java 1.4 introduced a new pattern letter, {@code 'Z'}, to represent
|
||||||
|
* time zones in RFC822 format (eg. {@code +0800} or {@code -1100}).
|
||||||
|
* This pattern letter can be used here (on all JDK versions).</p>
|
||||||
|
*
|
||||||
|
* <p>In addition, the pattern {@code 'ZZ'} has been made to represent
|
||||||
|
* ISO8601 full format time zones (eg. {@code +08:00} or {@code -11:00}).
|
||||||
|
* This introduces a minor incompatibility with Java 1.4, but at a gain of
|
||||||
|
* useful functionality.</p>
|
||||||
|
*
|
||||||
|
* <p>Javadoc cites for the year pattern: <i>For formatting, if the number of
|
||||||
|
* pattern letters is 2, the year is truncated to 2 digits; otherwise it is
|
||||||
|
* interpreted as a number.</i> Starting with Java 1.7 a pattern of 'Y' or
|
||||||
|
* 'YYY' will be formatted as '2003', while it was '03' in former Java
|
||||||
|
* versions. FastDateFormat implements the behavior of Java 7.</p>
|
||||||
|
*
|
||||||
|
* @since 2.0
|
||||||
|
* @version $Id: FastDateFormat.java 1572877 2014-02-28 08:42:25Z britter $
|
||||||
|
*/
|
||||||
|
public class FastDateFormat extends Format implements DateParser, DatePrinter {
|
||||||
|
/**
|
||||||
|
* Required for serialization support.
|
||||||
|
*
|
||||||
|
* @see java.io.Serializable
|
||||||
|
*/
|
||||||
|
private static final long serialVersionUID = 2L;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* FULL locale dependent date or time style.
|
||||||
|
*/
|
||||||
|
public static final int FULL = DateFormat.FULL;
|
||||||
|
/**
|
||||||
|
* LONG locale dependent date or time style.
|
||||||
|
*/
|
||||||
|
public static final int LONG = DateFormat.LONG;
|
||||||
|
/**
|
||||||
|
* MEDIUM locale dependent date or time style.
|
||||||
|
*/
|
||||||
|
public static final int MEDIUM = DateFormat.MEDIUM;
|
||||||
|
/**
|
||||||
|
* SHORT locale dependent date or time style.
|
||||||
|
*/
|
||||||
|
public static final int SHORT = DateFormat.SHORT;
|
||||||
|
|
||||||
|
private static final FormatCache<FastDateFormat> cache = new FormatCache<FastDateFormat>() {
|
||||||
|
@Override
|
||||||
|
protected FastDateFormat createInstance(final String pattern, final TimeZone timeZone, final Locale locale) {
|
||||||
|
return new FastDateFormat(pattern, timeZone, locale);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
private final FastDatePrinter printer;
|
||||||
|
private final FastDateParser parser;
|
||||||
|
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a formatter instance using the default pattern in the
|
||||||
|
* default locale.</p>
|
||||||
|
*
|
||||||
|
* @return a date/time formatter
|
||||||
|
*/
|
||||||
|
public static FastDateFormat getInstance() {
|
||||||
|
return cache.getInstance();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a formatter instance using the specified pattern in the
|
||||||
|
* default locale.</p>
|
||||||
|
*
|
||||||
|
* @param pattern {@link java.text.SimpleDateFormat} compatible
|
||||||
|
* pattern
|
||||||
|
* @return a pattern based date/time formatter
|
||||||
|
* @throws IllegalArgumentException if pattern is invalid
|
||||||
|
*/
|
||||||
|
public static FastDateFormat getInstance(final String pattern) {
|
||||||
|
return cache.getInstance(pattern, null, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a formatter instance using the specified pattern and
|
||||||
|
* time zone.</p>
|
||||||
|
*
|
||||||
|
* @param pattern {@link java.text.SimpleDateFormat} compatible
|
||||||
|
* pattern
|
||||||
|
* @param timeZone optional time zone, overrides time zone of
|
||||||
|
* formatted date
|
||||||
|
* @return a pattern based date/time formatter
|
||||||
|
* @throws IllegalArgumentException if pattern is invalid
|
||||||
|
*/
|
||||||
|
public static FastDateFormat getInstance(final String pattern, final TimeZone timeZone) {
|
||||||
|
return cache.getInstance(pattern, timeZone, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a formatter instance using the specified pattern and
|
||||||
|
* locale.</p>
|
||||||
|
*
|
||||||
|
* @param pattern {@link java.text.SimpleDateFormat} compatible
|
||||||
|
* pattern
|
||||||
|
* @param locale optional locale, overrides system locale
|
||||||
|
* @return a pattern based date/time formatter
|
||||||
|
* @throws IllegalArgumentException if pattern is invalid
|
||||||
|
*/
|
||||||
|
public static FastDateFormat getInstance(final String pattern, final Locale locale) {
|
||||||
|
return cache.getInstance(pattern, null, locale);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a formatter instance using the specified pattern, time zone
|
||||||
|
* and locale.</p>
|
||||||
|
*
|
||||||
|
* @param pattern {@link java.text.SimpleDateFormat} compatible
|
||||||
|
* pattern
|
||||||
|
* @param timeZone optional time zone, overrides time zone of
|
||||||
|
* formatted date
|
||||||
|
* @param locale optional locale, overrides system locale
|
||||||
|
* @return a pattern based date/time formatter
|
||||||
|
* @throws IllegalArgumentException if pattern is invalid
|
||||||
|
* or {@code null}
|
||||||
|
*/
|
||||||
|
public static FastDateFormat getInstance(final String pattern, final TimeZone timeZone, final Locale locale) {
|
||||||
|
return cache.getInstance(pattern, timeZone, locale);
|
||||||
|
}
|
||||||
|
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a date formatter instance using the specified style in the
|
||||||
|
* default time zone and locale.</p>
|
||||||
|
*
|
||||||
|
* @param style date style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @return a localized standard date formatter
|
||||||
|
* @throws IllegalArgumentException if the Locale has no date
|
||||||
|
* pattern defined
|
||||||
|
* @since 2.1
|
||||||
|
*/
|
||||||
|
public static FastDateFormat getDateInstance(final int style) {
|
||||||
|
return cache.getDateInstance(style, null, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a date formatter instance using the specified style and
|
||||||
|
* locale in the default time zone.</p>
|
||||||
|
*
|
||||||
|
* @param style date style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @param locale optional locale, overrides system locale
|
||||||
|
* @return a localized standard date formatter
|
||||||
|
* @throws IllegalArgumentException if the Locale has no date
|
||||||
|
* pattern defined
|
||||||
|
* @since 2.1
|
||||||
|
*/
|
||||||
|
public static FastDateFormat getDateInstance(final int style, final Locale locale) {
|
||||||
|
return cache.getDateInstance(style, null, locale);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a date formatter instance using the specified style and
|
||||||
|
* time zone in the default locale.</p>
|
||||||
|
*
|
||||||
|
* @param style date style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @param timeZone optional time zone, overrides time zone of
|
||||||
|
* formatted date
|
||||||
|
* @return a localized standard date formatter
|
||||||
|
* @throws IllegalArgumentException if the Locale has no date
|
||||||
|
* pattern defined
|
||||||
|
* @since 2.1
|
||||||
|
*/
|
||||||
|
public static FastDateFormat getDateInstance(final int style, final TimeZone timeZone) {
|
||||||
|
return cache.getDateInstance(style, timeZone, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a date formatter instance using the specified style, time
|
||||||
|
* zone and locale.</p>
|
||||||
|
*
|
||||||
|
* @param style date style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @param timeZone optional time zone, overrides time zone of
|
||||||
|
* formatted date
|
||||||
|
* @param locale optional locale, overrides system locale
|
||||||
|
* @return a localized standard date formatter
|
||||||
|
* @throws IllegalArgumentException if the Locale has no date
|
||||||
|
* pattern defined
|
||||||
|
*/
|
||||||
|
public static FastDateFormat getDateInstance(final int style, final TimeZone timeZone, final Locale locale) {
|
||||||
|
return cache.getDateInstance(style, timeZone, locale);
|
||||||
|
}
|
||||||
|
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a time formatter instance using the specified style in the
|
||||||
|
* default time zone and locale.</p>
|
||||||
|
*
|
||||||
|
* @param style time style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @return a localized standard time formatter
|
||||||
|
* @throws IllegalArgumentException if the Locale has no time
|
||||||
|
* pattern defined
|
||||||
|
* @since 2.1
|
||||||
|
*/
|
||||||
|
public static FastDateFormat getTimeInstance(final int style) {
|
||||||
|
return cache.getTimeInstance(style, null, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a time formatter instance using the specified style and
|
||||||
|
* locale in the default time zone.</p>
|
||||||
|
*
|
||||||
|
* @param style time style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @param locale optional locale, overrides system locale
|
||||||
|
* @return a localized standard time formatter
|
||||||
|
* @throws IllegalArgumentException if the Locale has no time
|
||||||
|
* pattern defined
|
||||||
|
* @since 2.1
|
||||||
|
*/
|
||||||
|
public static FastDateFormat getTimeInstance(final int style, final Locale locale) {
|
||||||
|
return cache.getTimeInstance(style, null, locale);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a time formatter instance using the specified style and
|
||||||
|
* time zone in the default locale.</p>
|
||||||
|
*
|
||||||
|
* @param style time style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @param timeZone optional time zone, overrides time zone of
|
||||||
|
* formatted time
|
||||||
|
* @return a localized standard time formatter
|
||||||
|
* @throws IllegalArgumentException if the Locale has no time
|
||||||
|
* pattern defined
|
||||||
|
* @since 2.1
|
||||||
|
*/
|
||||||
|
public static FastDateFormat getTimeInstance(final int style, final TimeZone timeZone) {
|
||||||
|
return cache.getTimeInstance(style, timeZone, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a time formatter instance using the specified style, time
|
||||||
|
* zone and locale.</p>
|
||||||
|
*
|
||||||
|
* @param style time style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @param timeZone optional time zone, overrides time zone of
|
||||||
|
* formatted time
|
||||||
|
* @param locale optional locale, overrides system locale
|
||||||
|
* @return a localized standard time formatter
|
||||||
|
* @throws IllegalArgumentException if the Locale has no time
|
||||||
|
* pattern defined
|
||||||
|
*/
|
||||||
|
public static FastDateFormat getTimeInstance(final int style, final TimeZone timeZone, final Locale locale) {
|
||||||
|
return cache.getTimeInstance(style, timeZone, locale);
|
||||||
|
}
|
||||||
|
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a date/time formatter instance using the specified style
|
||||||
|
* in the default time zone and locale.</p>
|
||||||
|
*
|
||||||
|
* @param dateStyle date style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @param timeStyle time style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @return a localized standard date/time formatter
|
||||||
|
* @throws IllegalArgumentException if the Locale has no date/time
|
||||||
|
* pattern defined
|
||||||
|
* @since 2.1
|
||||||
|
*/
|
||||||
|
public static FastDateFormat getDateTimeInstance(final int dateStyle, final int timeStyle) {
|
||||||
|
return cache.getDateTimeInstance(dateStyle, timeStyle, null, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a date/time formatter instance using the specified style and
|
||||||
|
* locale in the default time zone.</p>
|
||||||
|
*
|
||||||
|
* @param dateStyle date style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @param timeStyle time style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @param locale optional locale, overrides system locale
|
||||||
|
* @return a localized standard date/time formatter
|
||||||
|
* @throws IllegalArgumentException if the Locale has no date/time
|
||||||
|
* pattern defined
|
||||||
|
* @since 2.1
|
||||||
|
*/
|
||||||
|
public static FastDateFormat getDateTimeInstance(final int dateStyle, final int timeStyle, final Locale locale) {
|
||||||
|
return cache.getDateTimeInstance(dateStyle, timeStyle, null, locale);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a date/time formatter instance using the specified style and
|
||||||
|
* time zone in the default locale.</p>
|
||||||
|
*
|
||||||
|
* @param dateStyle date style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @param timeStyle time style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @param timeZone optional time zone, overrides time zone of
|
||||||
|
* formatted date
|
||||||
|
* @return a localized standard date/time formatter
|
||||||
|
* @throws IllegalArgumentException if the Locale has no date/time
|
||||||
|
* pattern defined
|
||||||
|
* @since 2.1
|
||||||
|
*/
|
||||||
|
public static FastDateFormat getDateTimeInstance(final int dateStyle, final int timeStyle, final TimeZone timeZone) {
|
||||||
|
return getDateTimeInstance(dateStyle, timeStyle, timeZone, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a date/time formatter instance using the specified style,
|
||||||
|
* time zone and locale.</p>
|
||||||
|
*
|
||||||
|
* @param dateStyle date style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @param timeStyle time style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @param timeZone optional time zone, overrides time zone of
|
||||||
|
* formatted date
|
||||||
|
* @param locale optional locale, overrides system locale
|
||||||
|
* @return a localized standard date/time formatter
|
||||||
|
* @throws IllegalArgumentException if the Locale has no date/time
|
||||||
|
* pattern defined
|
||||||
|
*/
|
||||||
|
public static FastDateFormat getDateTimeInstance(
|
||||||
|
final int dateStyle, final int timeStyle, final TimeZone timeZone, final Locale locale) {
|
||||||
|
return cache.getDateTimeInstance(dateStyle, timeStyle, timeZone, locale);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Constructor
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Constructs a new FastDateFormat.</p>
|
||||||
|
*
|
||||||
|
* @param pattern {@link java.text.SimpleDateFormat} compatible pattern
|
||||||
|
* @param timeZone non-null time zone to use
|
||||||
|
* @param locale non-null locale to use
|
||||||
|
* @throws NullPointerException if pattern, timeZone, or locale is null.
|
||||||
|
*/
|
||||||
|
protected FastDateFormat(final String pattern, final TimeZone timeZone, final Locale locale) {
|
||||||
|
this(pattern, timeZone, locale, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Constructor
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Constructs a new FastDateFormat.</p>
|
||||||
|
*
|
||||||
|
* @param pattern {@link java.text.SimpleDateFormat} compatible pattern
|
||||||
|
* @param timeZone non-null time zone to use
|
||||||
|
* @param locale non-null locale to use
|
||||||
|
* @param centuryStart The start of the 100 year period to use as the "default century" for 2 digit year parsing. If centuryStart is null, defaults to now - 80 years
|
||||||
|
* @throws NullPointerException if pattern, timeZone, or locale is null.
|
||||||
|
*/
|
||||||
|
protected FastDateFormat(final String pattern, final TimeZone timeZone, final Locale locale, final Date centuryStart) {
|
||||||
|
printer = new FastDatePrinter(pattern, timeZone, locale);
|
||||||
|
parser = new FastDateParser(pattern, timeZone, locale, centuryStart);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Format methods
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Formats a {@code Date}, {@code Calendar} or
|
||||||
|
* {@code Long} (milliseconds) object.</p>
|
||||||
|
*
|
||||||
|
* @param obj the object to format
|
||||||
|
* @param toAppendTo the buffer to append to
|
||||||
|
* @param pos the position - ignored
|
||||||
|
* @return the buffer passed in
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public StringBuffer format(final Object obj, final StringBuffer toAppendTo, final FieldPosition pos) {
|
||||||
|
return printer.format(obj, toAppendTo, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Formats a millisecond {@code long} value.</p>
|
||||||
|
*
|
||||||
|
* @param millis the millisecond value to format
|
||||||
|
* @return the formatted string
|
||||||
|
* @since 2.1
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public String format(final long millis) {
|
||||||
|
return printer.format(millis);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Formats a {@code Date} object using a {@code GregorianCalendar}.</p>
|
||||||
|
*
|
||||||
|
* @param date the date to format
|
||||||
|
* @return the formatted string
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public String format(final Date date) {
|
||||||
|
return printer.format(date);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Formats a {@code Calendar} object.</p>
|
||||||
|
*
|
||||||
|
* @param calendar the calendar to format
|
||||||
|
* @return the formatted string
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public String format(final Calendar calendar) {
|
||||||
|
return printer.format(calendar);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Formats a millisecond {@code long} value into the
|
||||||
|
* supplied {@code StringBuffer}.</p>
|
||||||
|
*
|
||||||
|
* @param millis the millisecond value to format
|
||||||
|
* @param buf the buffer to format into
|
||||||
|
* @return the specified string buffer
|
||||||
|
* @since 2.1
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public StringBuffer format(final long millis, final StringBuffer buf) {
|
||||||
|
return printer.format(millis, buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Formats a {@code Date} object into the
|
||||||
|
* supplied {@code StringBuffer} using a {@code GregorianCalendar}.</p>
|
||||||
|
*
|
||||||
|
* @param date the date to format
|
||||||
|
* @param buf the buffer to format into
|
||||||
|
* @return the specified string buffer
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public StringBuffer format(final Date date, final StringBuffer buf) {
|
||||||
|
return printer.format(date, buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Formats a {@code Calendar} object into the
|
||||||
|
* supplied {@code StringBuffer}.</p>
|
||||||
|
*
|
||||||
|
* @param calendar the calendar to format
|
||||||
|
* @param buf the buffer to format into
|
||||||
|
* @return the specified string buffer
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public StringBuffer format(final Calendar calendar, final StringBuffer buf) {
|
||||||
|
return printer.format(calendar, buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parsing
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
|
/* (non-Javadoc)
|
||||||
|
* @see DateParser#parse(java.lang.String)
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public Date parse(final String source) throws ParseException {
|
||||||
|
return parser.parse(source);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (non-Javadoc)
|
||||||
|
* @see DateParser#parse(java.lang.String, java.text.ParsePosition)
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public Date parse(final String source, final ParsePosition pos) {
|
||||||
|
return parser.parse(source, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (non-Javadoc)
|
||||||
|
* @see java.text.Format#parseObject(java.lang.String, java.text.ParsePosition)
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public Object parseObject(final String source, final ParsePosition pos) {
|
||||||
|
return parser.parseObject(source, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Accessors
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets the pattern used by this formatter.</p>
|
||||||
|
*
|
||||||
|
* @return the pattern, {@link java.text.SimpleDateFormat} compatible
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public String getPattern() {
|
||||||
|
return printer.getPattern();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets the time zone used by this formatter.</p>
|
||||||
|
* <p/>
|
||||||
|
* <p>This zone is always used for {@code Date} formatting. </p>
|
||||||
|
*
|
||||||
|
* @return the time zone
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public TimeZone getTimeZone() {
|
||||||
|
return printer.getTimeZone();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets the locale used by this formatter.</p>
|
||||||
|
*
|
||||||
|
* @return the locale
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public Locale getLocale() {
|
||||||
|
return printer.getLocale();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets an estimate for the maximum string length that the
|
||||||
|
* formatter will produce.</p>
|
||||||
|
* <p/>
|
||||||
|
* <p>The actual formatted length will almost always be less than or
|
||||||
|
* equal to this amount.</p>
|
||||||
|
*
|
||||||
|
* @return the maximum formatted length
|
||||||
|
*/
|
||||||
|
public int getMaxLengthEstimate() {
|
||||||
|
return printer.getMaxLengthEstimate();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Basics
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Compares two objects for equality.</p>
|
||||||
|
*
|
||||||
|
* @param obj the object to compare to
|
||||||
|
* @return {@code true} if equal
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public boolean equals(final Object obj) {
|
||||||
|
if (!(obj instanceof FastDateFormat)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
final FastDateFormat other = (FastDateFormat) obj;
|
||||||
|
// no need to check parser, as it has same invariants as printer
|
||||||
|
return printer.equals(other.printer);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Returns a hashcode compatible with equals.</p>
|
||||||
|
*
|
||||||
|
* @return a hashcode compatible with equals
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
return printer.hashCode();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a debugging string version of this formatter.</p>
|
||||||
|
*
|
||||||
|
* @return a debugging string
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "FastDateFormat[" + printer.getPattern() + "," + printer.getLocale() + "," + printer.getTimeZone().getID() + "]";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Performs the formatting by applying the rules to the
|
||||||
|
* specified calendar.</p>
|
||||||
|
*
|
||||||
|
* @param calendar the calendar to format
|
||||||
|
* @param buf the buffer to format into
|
||||||
|
* @return the specified string buffer
|
||||||
|
*/
|
||||||
|
protected StringBuffer applyRules(final Calendar calendar, final StringBuffer buf) {
|
||||||
|
return printer.applyRules(calendar, buf);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,840 @@
|
|||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package org.telegram.android.time;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.ObjectInputStream;
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.text.DateFormatSymbols;
|
||||||
|
import java.text.ParseException;
|
||||||
|
import java.text.ParsePosition;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Calendar;
|
||||||
|
import java.util.Date;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Locale;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.SortedMap;
|
||||||
|
import java.util.TimeZone;
|
||||||
|
import java.util.TreeMap;
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
|
import java.util.concurrent.ConcurrentMap;
|
||||||
|
import java.util.regex.Matcher;
|
||||||
|
import java.util.regex.Pattern;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>FastDateParser is a fast and thread-safe version of
|
||||||
|
* {@link java.text.SimpleDateFormat}.</p>
|
||||||
|
*
|
||||||
|
* <p>This class can be used as a direct replacement for
|
||||||
|
* <code>SimpleDateFormat</code> in most parsing situations.
|
||||||
|
* This class is especially useful in multi-threaded server environments.
|
||||||
|
* <code>SimpleDateFormat</code> is not thread-safe in any JDK version,
|
||||||
|
* nor will it be as Sun has closed the
|
||||||
|
* <a href="http://bugs.sun.com/bugdatabase/view_bug.do?bug_id=4228335">bug</a>/RFE.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <p>Only parsing is supported, but all patterns are compatible with
|
||||||
|
* SimpleDateFormat.</p>
|
||||||
|
*
|
||||||
|
* <p>Timing tests indicate this class is as about as fast as SimpleDateFormat
|
||||||
|
* in single thread applications and about 25% faster in multi-thread applications.</p>
|
||||||
|
*
|
||||||
|
* @version $Id: FastDateParser.java 1572877 2014-02-28 08:42:25Z britter $
|
||||||
|
* @since 3.2
|
||||||
|
*/
|
||||||
|
public class FastDateParser implements DateParser, Serializable {
|
||||||
|
/**
|
||||||
|
* Required for serialization support.
|
||||||
|
*
|
||||||
|
* @see java.io.Serializable
|
||||||
|
*/
|
||||||
|
private static final long serialVersionUID = 2L;
|
||||||
|
|
||||||
|
static final Locale JAPANESE_IMPERIAL = new Locale("ja", "JP", "JP");
|
||||||
|
|
||||||
|
// defining fields
|
||||||
|
private final String pattern;
|
||||||
|
private final TimeZone timeZone;
|
||||||
|
private final Locale locale;
|
||||||
|
private final int century;
|
||||||
|
private final int startYear;
|
||||||
|
|
||||||
|
// derived fields
|
||||||
|
private transient Pattern parsePattern;
|
||||||
|
private transient Strategy[] strategies;
|
||||||
|
|
||||||
|
// dynamic fields to communicate with Strategy
|
||||||
|
private transient String currentFormatField;
|
||||||
|
private transient Strategy nextStrategy;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Constructs a new FastDateParser.</p>
|
||||||
|
*
|
||||||
|
* @param pattern non-null {@link java.text.SimpleDateFormat} compatible
|
||||||
|
* pattern
|
||||||
|
* @param timeZone non-null time zone to use
|
||||||
|
* @param locale non-null locale
|
||||||
|
*/
|
||||||
|
protected FastDateParser(final String pattern, final TimeZone timeZone, final Locale locale) {
|
||||||
|
this(pattern, timeZone, locale, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Constructs a new FastDateParser.</p>
|
||||||
|
*
|
||||||
|
* @param pattern non-null {@link java.text.SimpleDateFormat} compatible
|
||||||
|
* pattern
|
||||||
|
* @param timeZone non-null time zone to use
|
||||||
|
* @param locale non-null locale
|
||||||
|
* @param centuryStart The start of the century for 2 digit year parsing
|
||||||
|
* @since 3.3
|
||||||
|
*/
|
||||||
|
protected FastDateParser(final String pattern, final TimeZone timeZone, final Locale locale, final Date centuryStart) {
|
||||||
|
this.pattern = pattern;
|
||||||
|
this.timeZone = timeZone;
|
||||||
|
this.locale = locale;
|
||||||
|
|
||||||
|
final Calendar definingCalendar = Calendar.getInstance(timeZone, locale);
|
||||||
|
int centuryStartYear;
|
||||||
|
if (centuryStart != null) {
|
||||||
|
definingCalendar.setTime(centuryStart);
|
||||||
|
centuryStartYear = definingCalendar.get(Calendar.YEAR);
|
||||||
|
} else if (locale.equals(JAPANESE_IMPERIAL)) {
|
||||||
|
centuryStartYear = 0;
|
||||||
|
} else {
|
||||||
|
// from 80 years ago to 20 years from now
|
||||||
|
definingCalendar.setTime(new Date());
|
||||||
|
centuryStartYear = definingCalendar.get(Calendar.YEAR) - 80;
|
||||||
|
}
|
||||||
|
century = centuryStartYear / 100 * 100;
|
||||||
|
startYear = centuryStartYear - century;
|
||||||
|
|
||||||
|
init(definingCalendar);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initialize derived fields from defining fields.
|
||||||
|
* This is called from constructor and from readObject (de-serialization)
|
||||||
|
*
|
||||||
|
* @param definingCalendar the {@link java.util.Calendar} instance used to initialize this FastDateParser
|
||||||
|
*/
|
||||||
|
private void init(Calendar definingCalendar) {
|
||||||
|
|
||||||
|
final StringBuilder regex = new StringBuilder();
|
||||||
|
final List<Strategy> collector = new ArrayList<Strategy>();
|
||||||
|
|
||||||
|
final Matcher patternMatcher = formatPattern.matcher(pattern);
|
||||||
|
if (!patternMatcher.lookingAt()) {
|
||||||
|
throw new IllegalArgumentException(
|
||||||
|
"Illegal pattern character '" + pattern.charAt(patternMatcher.regionStart()) + "'");
|
||||||
|
}
|
||||||
|
|
||||||
|
currentFormatField = patternMatcher.group();
|
||||||
|
Strategy currentStrategy = getStrategy(currentFormatField, definingCalendar);
|
||||||
|
for (; ; ) {
|
||||||
|
patternMatcher.region(patternMatcher.end(), patternMatcher.regionEnd());
|
||||||
|
if (!patternMatcher.lookingAt()) {
|
||||||
|
nextStrategy = null;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
final String nextFormatField = patternMatcher.group();
|
||||||
|
nextStrategy = getStrategy(nextFormatField, definingCalendar);
|
||||||
|
if (currentStrategy.addRegex(this, regex)) {
|
||||||
|
collector.add(currentStrategy);
|
||||||
|
}
|
||||||
|
currentFormatField = nextFormatField;
|
||||||
|
currentStrategy = nextStrategy;
|
||||||
|
}
|
||||||
|
if (patternMatcher.regionStart() != patternMatcher.regionEnd()) {
|
||||||
|
throw new IllegalArgumentException("Failed to parse \"" + pattern + "\" ; gave up at index " + patternMatcher.regionStart());
|
||||||
|
}
|
||||||
|
if (currentStrategy.addRegex(this, regex)) {
|
||||||
|
collector.add(currentStrategy);
|
||||||
|
}
|
||||||
|
currentFormatField = null;
|
||||||
|
strategies = collector.toArray(new Strategy[collector.size()]);
|
||||||
|
parsePattern = Pattern.compile(regex.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Accessors
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
/* (non-Javadoc)
|
||||||
|
* @see org.apache.commons.lang3.time.DateParser#getPattern()
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public String getPattern() {
|
||||||
|
return pattern;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (non-Javadoc)
|
||||||
|
* @see org.apache.commons.lang3.time.DateParser#getTimeZone()
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public TimeZone getTimeZone() {
|
||||||
|
return timeZone;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (non-Javadoc)
|
||||||
|
* @see org.apache.commons.lang3.time.DateParser#getLocale()
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public Locale getLocale() {
|
||||||
|
return locale;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the generated pattern (for testing purposes).
|
||||||
|
*
|
||||||
|
* @return the generated pattern
|
||||||
|
*/
|
||||||
|
Pattern getParsePattern() {
|
||||||
|
return parsePattern;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Basics
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Compare another object for equality with this object.</p>
|
||||||
|
*
|
||||||
|
* @param obj the object to compare to
|
||||||
|
* @return <code>true</code>if equal to this instance
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public boolean equals(final Object obj) {
|
||||||
|
if (!(obj instanceof FastDateParser)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
final FastDateParser other = (FastDateParser) obj;
|
||||||
|
return pattern.equals(other.pattern)
|
||||||
|
&& timeZone.equals(other.timeZone)
|
||||||
|
&& locale.equals(other.locale);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Return a hashcode compatible with equals.</p>
|
||||||
|
*
|
||||||
|
* @return a hashcode compatible with equals
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
return pattern.hashCode() + 13 * (timeZone.hashCode() + 13 * locale.hashCode());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Get a string version of this formatter.</p>
|
||||||
|
*
|
||||||
|
* @return a debugging string
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "FastDateParser[" + pattern + "," + locale + "," + timeZone.getID() + "]";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Serializing
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create the object after serialization. This implementation reinitializes the
|
||||||
|
* transient properties.
|
||||||
|
*
|
||||||
|
* @param in ObjectInputStream from which the object is being deserialized.
|
||||||
|
* @throws IOException if there is an IO issue.
|
||||||
|
* @throws ClassNotFoundException if a class cannot be found.
|
||||||
|
*/
|
||||||
|
private void readObject(final ObjectInputStream in) throws IOException, ClassNotFoundException {
|
||||||
|
in.defaultReadObject();
|
||||||
|
|
||||||
|
final Calendar definingCalendar = Calendar.getInstance(timeZone, locale);
|
||||||
|
init(definingCalendar);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (non-Javadoc)
|
||||||
|
* @see org.apache.commons.lang3.time.DateParser#parseObject(java.lang.String)
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public Object parseObject(final String source) throws ParseException {
|
||||||
|
return parse(source);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (non-Javadoc)
|
||||||
|
* @see org.apache.commons.lang3.time.DateParser#parse(java.lang.String)
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public Date parse(final String source) throws ParseException {
|
||||||
|
final Date date = parse(source, new ParsePosition(0));
|
||||||
|
if (date == null) {
|
||||||
|
// Add a note re supported date range
|
||||||
|
if (locale.equals(JAPANESE_IMPERIAL)) {
|
||||||
|
throw new ParseException(
|
||||||
|
"(The " + locale + " locale does not support dates before 1868 AD)\n" +
|
||||||
|
"Unparseable date: \"" + source + "\" does not match " + parsePattern.pattern(), 0);
|
||||||
|
}
|
||||||
|
throw new ParseException("Unparseable date: \"" + source + "\" does not match " + parsePattern.pattern(), 0);
|
||||||
|
}
|
||||||
|
return date;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (non-Javadoc)
|
||||||
|
* @see org.apache.commons.lang3.time.DateParser#parseObject(java.lang.String, java.text.ParsePosition)
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public Object parseObject(final String source, final ParsePosition pos) {
|
||||||
|
return parse(source, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (non-Javadoc)
|
||||||
|
* @see org.apache.commons.lang3.time.DateParser#parse(java.lang.String, java.text.ParsePosition)
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public Date parse(final String source, final ParsePosition pos) {
|
||||||
|
final int offset = pos.getIndex();
|
||||||
|
final Matcher matcher = parsePattern.matcher(source.substring(offset));
|
||||||
|
if (!matcher.lookingAt()) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
// timing tests indicate getting new instance is 19% faster than cloning
|
||||||
|
final Calendar cal = Calendar.getInstance(timeZone, locale);
|
||||||
|
cal.clear();
|
||||||
|
|
||||||
|
for (int i = 0; i < strategies.length; ) {
|
||||||
|
final Strategy strategy = strategies[i++];
|
||||||
|
strategy.setCalendar(this, cal, matcher.group(i));
|
||||||
|
}
|
||||||
|
pos.setIndex(offset + matcher.end());
|
||||||
|
return cal.getTime();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Support for strategies
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Escape constant fields into regular expression
|
||||||
|
*
|
||||||
|
* @param regex The destination regex
|
||||||
|
* @param value The source field
|
||||||
|
* @param unquote If true, replace two success quotes ('') with single quote (')
|
||||||
|
* @return The <code>StringBuilder</code>
|
||||||
|
*/
|
||||||
|
private static StringBuilder escapeRegex(final StringBuilder regex, final String value, final boolean unquote) {
|
||||||
|
regex.append("\\Q");
|
||||||
|
for (int i = 0; i < value.length(); ++i) {
|
||||||
|
char c = value.charAt(i);
|
||||||
|
switch (c) {
|
||||||
|
case '\'':
|
||||||
|
if (unquote) {
|
||||||
|
if (++i == value.length()) {
|
||||||
|
return regex;
|
||||||
|
}
|
||||||
|
c = value.charAt(i);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case '\\':
|
||||||
|
if (++i == value.length()) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
* If we have found \E, we replace it with \E\\E\Q, i.e. we stop the quoting,
|
||||||
|
* quote the \ in \E, then restart the quoting.
|
||||||
|
*
|
||||||
|
* Otherwise we just output the two characters.
|
||||||
|
* In each case the initial \ needs to be output and the final char is done at the end
|
||||||
|
*/
|
||||||
|
regex.append(c); // we always want the original \
|
||||||
|
c = value.charAt(i); // Is it followed by E ?
|
||||||
|
if (c == 'E') { // \E detected
|
||||||
|
regex.append("E\\\\E\\"); // see comment above
|
||||||
|
c = 'Q'; // appended below
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
regex.append(c);
|
||||||
|
}
|
||||||
|
regex.append("\\E");
|
||||||
|
return regex;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the short and long values displayed for a field
|
||||||
|
*
|
||||||
|
* @param field The field of interest
|
||||||
|
* @param definingCalendar The calendar to obtain the short and long values
|
||||||
|
* @param locale The locale of display names
|
||||||
|
* @return A Map of the field key / value pairs
|
||||||
|
*/
|
||||||
|
private static Map<String, Integer> getDisplayNames(final int field, final Calendar definingCalendar, final Locale locale) {
|
||||||
|
return definingCalendar.getDisplayNames(field, Calendar.ALL_STYLES, locale);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Adjust dates to be within appropriate century
|
||||||
|
*
|
||||||
|
* @param twoDigitYear The year to adjust
|
||||||
|
* @return A value between centuryStart(inclusive) to centuryStart+100(exclusive)
|
||||||
|
*/
|
||||||
|
private int adjustYear(final int twoDigitYear) {
|
||||||
|
int trial = century + twoDigitYear;
|
||||||
|
return twoDigitYear >= startYear ? trial : trial + 100;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Is the next field a number?
|
||||||
|
*
|
||||||
|
* @return true, if next field will be a number
|
||||||
|
*/
|
||||||
|
boolean isNextNumber() {
|
||||||
|
return nextStrategy != null && nextStrategy.isNumber();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* What is the width of the current field?
|
||||||
|
*
|
||||||
|
* @return The number of characters in the current format field
|
||||||
|
*/
|
||||||
|
int getFieldWidth() {
|
||||||
|
return currentFormatField.length();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A strategy to parse a single field from the parsing pattern
|
||||||
|
*/
|
||||||
|
private static abstract class Strategy {
|
||||||
|
/**
|
||||||
|
* Is this field a number?
|
||||||
|
* The default implementation returns false.
|
||||||
|
*
|
||||||
|
* @return true, if field is a number
|
||||||
|
*/
|
||||||
|
boolean isNumber() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the Calendar with the parsed field.
|
||||||
|
* <p/>
|
||||||
|
* The default implementation does nothing.
|
||||||
|
*
|
||||||
|
* @param parser The parser calling this strategy
|
||||||
|
* @param cal The <code>Calendar</code> to set
|
||||||
|
* @param value The parsed field to translate and set in cal
|
||||||
|
*/
|
||||||
|
void setCalendar(final FastDateParser parser, final Calendar cal, final String value) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generate a <code>Pattern</code> regular expression to the <code>StringBuilder</code>
|
||||||
|
* which will accept this field
|
||||||
|
*
|
||||||
|
* @param parser The parser calling this strategy
|
||||||
|
* @param regex The <code>StringBuilder</code> to append to
|
||||||
|
* @return true, if this field will set the calendar;
|
||||||
|
* false, if this field is a constant value
|
||||||
|
*/
|
||||||
|
abstract boolean addRegex(FastDateParser parser, StringBuilder regex);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A <code>Pattern</code> to parse the user supplied SimpleDateFormat pattern
|
||||||
|
*/
|
||||||
|
private static final Pattern formatPattern = Pattern.compile(
|
||||||
|
"D+|E+|F+|G+|H+|K+|M+|S+|W+|Z+|a+|d+|h+|k+|m+|s+|w+|y+|z+|''|'[^']++(''[^']*+)*+'|[^'A-Za-z]++");
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Obtain a Strategy given a field from a SimpleDateFormat pattern
|
||||||
|
*
|
||||||
|
* @param formatField A sub-sequence of the SimpleDateFormat pattern
|
||||||
|
* @param definingCalendar The calendar to obtain the short and long values
|
||||||
|
* @return The Strategy that will handle parsing for the field
|
||||||
|
*/
|
||||||
|
private Strategy getStrategy(final String formatField, final Calendar definingCalendar) {
|
||||||
|
switch (formatField.charAt(0)) {
|
||||||
|
case '\'':
|
||||||
|
if (formatField.length() > 2) {
|
||||||
|
return new CopyQuotedStrategy(formatField.substring(1, formatField.length() - 1));
|
||||||
|
}
|
||||||
|
//$FALL-THROUGH$
|
||||||
|
default:
|
||||||
|
return new CopyQuotedStrategy(formatField);
|
||||||
|
case 'D':
|
||||||
|
return DAY_OF_YEAR_STRATEGY;
|
||||||
|
case 'E':
|
||||||
|
return getLocaleSpecificStrategy(Calendar.DAY_OF_WEEK, definingCalendar);
|
||||||
|
case 'F':
|
||||||
|
return DAY_OF_WEEK_IN_MONTH_STRATEGY;
|
||||||
|
case 'G':
|
||||||
|
return getLocaleSpecificStrategy(Calendar.ERA, definingCalendar);
|
||||||
|
case 'H':
|
||||||
|
return MODULO_HOUR_OF_DAY_STRATEGY;
|
||||||
|
case 'K':
|
||||||
|
return HOUR_STRATEGY;
|
||||||
|
case 'M':
|
||||||
|
return formatField.length() >= 3 ? getLocaleSpecificStrategy(Calendar.MONTH, definingCalendar) : NUMBER_MONTH_STRATEGY;
|
||||||
|
case 'S':
|
||||||
|
return MILLISECOND_STRATEGY;
|
||||||
|
case 'W':
|
||||||
|
return WEEK_OF_MONTH_STRATEGY;
|
||||||
|
case 'a':
|
||||||
|
return getLocaleSpecificStrategy(Calendar.AM_PM, definingCalendar);
|
||||||
|
case 'd':
|
||||||
|
return DAY_OF_MONTH_STRATEGY;
|
||||||
|
case 'h':
|
||||||
|
return MODULO_HOUR_STRATEGY;
|
||||||
|
case 'k':
|
||||||
|
return HOUR_OF_DAY_STRATEGY;
|
||||||
|
case 'm':
|
||||||
|
return MINUTE_STRATEGY;
|
||||||
|
case 's':
|
||||||
|
return SECOND_STRATEGY;
|
||||||
|
case 'w':
|
||||||
|
return WEEK_OF_YEAR_STRATEGY;
|
||||||
|
case 'y':
|
||||||
|
return formatField.length() > 2 ? LITERAL_YEAR_STRATEGY : ABBREVIATED_YEAR_STRATEGY;
|
||||||
|
case 'Z':
|
||||||
|
case 'z':
|
||||||
|
return getLocaleSpecificStrategy(Calendar.ZONE_OFFSET, definingCalendar);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("unchecked") // OK because we are creating an array with no entries
|
||||||
|
private static final ConcurrentMap<Locale, Strategy>[] caches = new ConcurrentMap[Calendar.FIELD_COUNT];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get a cache of Strategies for a particular field
|
||||||
|
*
|
||||||
|
* @param field The Calendar field
|
||||||
|
* @return a cache of Locale to Strategy
|
||||||
|
*/
|
||||||
|
private static ConcurrentMap<Locale, Strategy> getCache(final int field) {
|
||||||
|
synchronized (caches) {
|
||||||
|
if (caches[field] == null) {
|
||||||
|
caches[field] = new ConcurrentHashMap<Locale, Strategy>(3);
|
||||||
|
}
|
||||||
|
return caches[field];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct a Strategy that parses a Text field
|
||||||
|
*
|
||||||
|
* @param field The Calendar field
|
||||||
|
* @param definingCalendar The calendar to obtain the short and long values
|
||||||
|
* @return a TextStrategy for the field and Locale
|
||||||
|
*/
|
||||||
|
private Strategy getLocaleSpecificStrategy(final int field, final Calendar definingCalendar) {
|
||||||
|
final ConcurrentMap<Locale, Strategy> cache = getCache(field);
|
||||||
|
Strategy strategy = cache.get(locale);
|
||||||
|
if (strategy == null) {
|
||||||
|
strategy = field == Calendar.ZONE_OFFSET
|
||||||
|
? new TimeZoneStrategy(locale)
|
||||||
|
: new TextStrategy(field, definingCalendar, locale);
|
||||||
|
final Strategy inCache = cache.putIfAbsent(locale, strategy);
|
||||||
|
if (inCache != null) {
|
||||||
|
return inCache;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return strategy;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A strategy that copies the static or quoted field in the parsing pattern
|
||||||
|
*/
|
||||||
|
private static class CopyQuotedStrategy extends Strategy {
|
||||||
|
private final String formatField;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct a Strategy that ensures the formatField has literal text
|
||||||
|
*
|
||||||
|
* @param formatField The literal text to match
|
||||||
|
*/
|
||||||
|
CopyQuotedStrategy(final String formatField) {
|
||||||
|
this.formatField = formatField;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
boolean isNumber() {
|
||||||
|
char c = formatField.charAt(0);
|
||||||
|
if (c == '\'') {
|
||||||
|
c = formatField.charAt(1);
|
||||||
|
}
|
||||||
|
return Character.isDigit(c);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
boolean addRegex(final FastDateParser parser, final StringBuilder regex) {
|
||||||
|
escapeRegex(regex, formatField, true);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A strategy that handles a text field in the parsing pattern
|
||||||
|
*/
|
||||||
|
private static class TextStrategy extends Strategy {
|
||||||
|
private final int field;
|
||||||
|
private final Map<String, Integer> keyValues;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct a Strategy that parses a Text field
|
||||||
|
*
|
||||||
|
* @param field The Calendar field
|
||||||
|
* @param definingCalendar The Calendar to use
|
||||||
|
* @param locale The Locale to use
|
||||||
|
*/
|
||||||
|
TextStrategy(final int field, final Calendar definingCalendar, final Locale locale) {
|
||||||
|
this.field = field;
|
||||||
|
this.keyValues = getDisplayNames(field, definingCalendar, locale);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
boolean addRegex(final FastDateParser parser, final StringBuilder regex) {
|
||||||
|
regex.append('(');
|
||||||
|
for (final String textKeyValue : keyValues.keySet()) {
|
||||||
|
escapeRegex(regex, textKeyValue, false).append('|');
|
||||||
|
}
|
||||||
|
regex.setCharAt(regex.length() - 1, ')');
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
void setCalendar(final FastDateParser parser, final Calendar cal, final String value) {
|
||||||
|
final Integer iVal = keyValues.get(value);
|
||||||
|
if (iVal == null) {
|
||||||
|
final StringBuilder sb = new StringBuilder(value);
|
||||||
|
sb.append(" not in (");
|
||||||
|
for (final String textKeyValue : keyValues.keySet()) {
|
||||||
|
sb.append(textKeyValue).append(' ');
|
||||||
|
}
|
||||||
|
sb.setCharAt(sb.length() - 1, ')');
|
||||||
|
throw new IllegalArgumentException(sb.toString());
|
||||||
|
}
|
||||||
|
cal.set(field, iVal.intValue());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A strategy that handles a number field in the parsing pattern
|
||||||
|
*/
|
||||||
|
private static class NumberStrategy extends Strategy {
|
||||||
|
private final int field;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct a Strategy that parses a Number field
|
||||||
|
*
|
||||||
|
* @param field The Calendar field
|
||||||
|
*/
|
||||||
|
NumberStrategy(final int field) {
|
||||||
|
this.field = field;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
boolean isNumber() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
boolean addRegex(final FastDateParser parser, final StringBuilder regex) {
|
||||||
|
// See LANG-954: We use {Nd} rather than {IsNd} because Android does not support the Is prefix
|
||||||
|
if (parser.isNextNumber()) {
|
||||||
|
regex.append("(\\p{Nd}{").append(parser.getFieldWidth()).append("}+)");
|
||||||
|
} else {
|
||||||
|
regex.append("(\\p{Nd}++)");
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
void setCalendar(final FastDateParser parser, final Calendar cal, final String value) {
|
||||||
|
cal.set(field, modify(Integer.parseInt(value)));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Make any modifications to parsed integer
|
||||||
|
*
|
||||||
|
* @param iValue The parsed integer
|
||||||
|
* @return The modified value
|
||||||
|
*/
|
||||||
|
int modify(final int iValue) {
|
||||||
|
return iValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final Strategy ABBREVIATED_YEAR_STRATEGY = new NumberStrategy(Calendar.YEAR) {
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
void setCalendar(final FastDateParser parser, final Calendar cal, final String value) {
|
||||||
|
int iValue = Integer.parseInt(value);
|
||||||
|
if (iValue < 100) {
|
||||||
|
iValue = parser.adjustYear(iValue);
|
||||||
|
}
|
||||||
|
cal.set(Calendar.YEAR, iValue);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A strategy that handles a timezone field in the parsing pattern
|
||||||
|
*/
|
||||||
|
private static class TimeZoneStrategy extends Strategy {
|
||||||
|
|
||||||
|
private final String validTimeZoneChars;
|
||||||
|
private final SortedMap<String, TimeZone> tzNames = new TreeMap<String, TimeZone>(String.CASE_INSENSITIVE_ORDER);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Index of zone id
|
||||||
|
*/
|
||||||
|
private static final int ID = 0;
|
||||||
|
/**
|
||||||
|
* Index of the long name of zone in standard time
|
||||||
|
*/
|
||||||
|
private static final int LONG_STD = 1;
|
||||||
|
/**
|
||||||
|
* Index of the short name of zone in standard time
|
||||||
|
*/
|
||||||
|
private static final int SHORT_STD = 2;
|
||||||
|
/**
|
||||||
|
* Index of the long name of zone in daylight saving time
|
||||||
|
*/
|
||||||
|
private static final int LONG_DST = 3;
|
||||||
|
/**
|
||||||
|
* Index of the short name of zone in daylight saving time
|
||||||
|
*/
|
||||||
|
private static final int SHORT_DST = 4;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct a Strategy that parses a TimeZone
|
||||||
|
*
|
||||||
|
* @param locale The Locale
|
||||||
|
*/
|
||||||
|
TimeZoneStrategy(final Locale locale) {
|
||||||
|
final String[][] zones = DateFormatSymbols.getInstance(locale).getZoneStrings();
|
||||||
|
for (String[] zone : zones) {
|
||||||
|
if (zone[ID].startsWith("GMT")) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
final TimeZone tz = TimeZone.getTimeZone(zone[ID]);
|
||||||
|
if (!tzNames.containsKey(zone[LONG_STD])) {
|
||||||
|
tzNames.put(zone[LONG_STD], tz);
|
||||||
|
}
|
||||||
|
if (!tzNames.containsKey(zone[SHORT_STD])) {
|
||||||
|
tzNames.put(zone[SHORT_STD], tz);
|
||||||
|
}
|
||||||
|
if (tz.useDaylightTime()) {
|
||||||
|
if (!tzNames.containsKey(zone[LONG_DST])) {
|
||||||
|
tzNames.put(zone[LONG_DST], tz);
|
||||||
|
}
|
||||||
|
if (!tzNames.containsKey(zone[SHORT_DST])) {
|
||||||
|
tzNames.put(zone[SHORT_DST], tz);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
final StringBuilder sb = new StringBuilder();
|
||||||
|
sb.append("(GMT[+\\-]\\d{0,1}\\d{2}|[+\\-]\\d{2}:?\\d{2}|");
|
||||||
|
for (final String id : tzNames.keySet()) {
|
||||||
|
escapeRegex(sb, id, false).append('|');
|
||||||
|
}
|
||||||
|
sb.setCharAt(sb.length() - 1, ')');
|
||||||
|
validTimeZoneChars = sb.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
boolean addRegex(final FastDateParser parser, final StringBuilder regex) {
|
||||||
|
regex.append(validTimeZoneChars);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
void setCalendar(final FastDateParser parser, final Calendar cal, final String value) {
|
||||||
|
TimeZone tz;
|
||||||
|
if (value.charAt(0) == '+' || value.charAt(0) == '-') {
|
||||||
|
tz = TimeZone.getTimeZone("GMT" + value);
|
||||||
|
} else if (value.startsWith("GMT")) {
|
||||||
|
tz = TimeZone.getTimeZone(value);
|
||||||
|
} else {
|
||||||
|
tz = tzNames.get(value);
|
||||||
|
if (tz == null) {
|
||||||
|
throw new IllegalArgumentException(value + " is not a supported timezone name");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cal.setTimeZone(tz);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final Strategy NUMBER_MONTH_STRATEGY = new NumberStrategy(Calendar.MONTH) {
|
||||||
|
@Override
|
||||||
|
int modify(final int iValue) {
|
||||||
|
return iValue - 1;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
private static final Strategy LITERAL_YEAR_STRATEGY = new NumberStrategy(Calendar.YEAR);
|
||||||
|
private static final Strategy WEEK_OF_YEAR_STRATEGY = new NumberStrategy(Calendar.WEEK_OF_YEAR);
|
||||||
|
private static final Strategy WEEK_OF_MONTH_STRATEGY = new NumberStrategy(Calendar.WEEK_OF_MONTH);
|
||||||
|
private static final Strategy DAY_OF_YEAR_STRATEGY = new NumberStrategy(Calendar.DAY_OF_YEAR);
|
||||||
|
private static final Strategy DAY_OF_MONTH_STRATEGY = new NumberStrategy(Calendar.DAY_OF_MONTH);
|
||||||
|
private static final Strategy DAY_OF_WEEK_IN_MONTH_STRATEGY = new NumberStrategy(Calendar.DAY_OF_WEEK_IN_MONTH);
|
||||||
|
private static final Strategy HOUR_OF_DAY_STRATEGY = new NumberStrategy(Calendar.HOUR_OF_DAY);
|
||||||
|
private static final Strategy MODULO_HOUR_OF_DAY_STRATEGY = new NumberStrategy(Calendar.HOUR_OF_DAY) {
|
||||||
|
@Override
|
||||||
|
int modify(final int iValue) {
|
||||||
|
return iValue % 24;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
private static final Strategy MODULO_HOUR_STRATEGY = new NumberStrategy(Calendar.HOUR) {
|
||||||
|
@Override
|
||||||
|
int modify(final int iValue) {
|
||||||
|
return iValue % 12;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
private static final Strategy HOUR_STRATEGY = new NumberStrategy(Calendar.HOUR);
|
||||||
|
private static final Strategy MINUTE_STRATEGY = new NumberStrategy(Calendar.MINUTE);
|
||||||
|
private static final Strategy SECOND_STRATEGY = new NumberStrategy(Calendar.SECOND);
|
||||||
|
private static final Strategy MILLISECOND_STRATEGY = new NumberStrategy(Calendar.MILLISECOND);
|
||||||
|
}
|
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,265 @@
|
|||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package org.telegram.android.time;
|
||||||
|
|
||||||
|
import java.text.DateFormat;
|
||||||
|
import java.text.Format;
|
||||||
|
import java.text.SimpleDateFormat;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.Locale;
|
||||||
|
import java.util.TimeZone;
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
|
import java.util.concurrent.ConcurrentMap;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>FormatCache is a cache and factory for {@link Format}s.</p>
|
||||||
|
*
|
||||||
|
* @since 3.0
|
||||||
|
* @version $Id: FormatCache 892161 2009-12-18 07:21:10Z $
|
||||||
|
*/
|
||||||
|
// TODO: Before making public move from getDateTimeInstance(Integer,...) to int; or some other approach.
|
||||||
|
abstract class FormatCache<F extends Format> {
|
||||||
|
/**
|
||||||
|
* No date or no time. Used in same parameters as DateFormat.SHORT or DateFormat.LONG
|
||||||
|
*/
|
||||||
|
static final int NONE = -1;
|
||||||
|
|
||||||
|
private final ConcurrentMap<MultipartKey, F> cInstanceCache
|
||||||
|
= new ConcurrentHashMap<MultipartKey, F>(7);
|
||||||
|
|
||||||
|
private static final ConcurrentMap<MultipartKey, String> cDateTimeInstanceCache
|
||||||
|
= new ConcurrentHashMap<MultipartKey, String>(7);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a formatter instance using the default pattern in the
|
||||||
|
* default timezone and locale.</p>
|
||||||
|
*
|
||||||
|
* @return a date/time formatter
|
||||||
|
*/
|
||||||
|
public F getInstance() {
|
||||||
|
return getDateTimeInstance(DateFormat.SHORT, DateFormat.SHORT, TimeZone.getDefault(), Locale.getDefault());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a formatter instance using the specified pattern, time zone
|
||||||
|
* and locale.</p>
|
||||||
|
*
|
||||||
|
* @param pattern {@link java.text.SimpleDateFormat} compatible
|
||||||
|
* pattern, non-null
|
||||||
|
* @param timeZone the time zone, null means use the default TimeZone
|
||||||
|
* @param locale the locale, null means use the default Locale
|
||||||
|
* @return a pattern based date/time formatter
|
||||||
|
* @throws IllegalArgumentException if pattern is invalid
|
||||||
|
* or <code>null</code>
|
||||||
|
*/
|
||||||
|
public F getInstance(final String pattern, TimeZone timeZone, Locale locale) {
|
||||||
|
if (pattern == null) {
|
||||||
|
throw new NullPointerException("pattern must not be null");
|
||||||
|
}
|
||||||
|
if (timeZone == null) {
|
||||||
|
timeZone = TimeZone.getDefault();
|
||||||
|
}
|
||||||
|
if (locale == null) {
|
||||||
|
locale = Locale.getDefault();
|
||||||
|
}
|
||||||
|
final MultipartKey key = new MultipartKey(pattern, timeZone, locale);
|
||||||
|
F format = cInstanceCache.get(key);
|
||||||
|
if (format == null) {
|
||||||
|
format = createInstance(pattern, timeZone, locale);
|
||||||
|
final F previousValue = cInstanceCache.putIfAbsent(key, format);
|
||||||
|
if (previousValue != null) {
|
||||||
|
// another thread snuck in and did the same work
|
||||||
|
// we should return the instance that is in ConcurrentMap
|
||||||
|
format = previousValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return format;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Create a format instance using the specified pattern, time zone
|
||||||
|
* and locale.</p>
|
||||||
|
*
|
||||||
|
* @param pattern {@link java.text.SimpleDateFormat} compatible pattern, this will not be null.
|
||||||
|
* @param timeZone time zone, this will not be null.
|
||||||
|
* @param locale locale, this will not be null.
|
||||||
|
* @return a pattern based date/time formatter
|
||||||
|
* @throws IllegalArgumentException if pattern is invalid
|
||||||
|
* or <code>null</code>
|
||||||
|
*/
|
||||||
|
abstract protected F createInstance(String pattern, TimeZone timeZone, Locale locale);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a date/time formatter instance using the specified style,
|
||||||
|
* time zone and locale.</p>
|
||||||
|
*
|
||||||
|
* @param dateStyle date style: FULL, LONG, MEDIUM, or SHORT, null indicates no date in format
|
||||||
|
* @param timeStyle time style: FULL, LONG, MEDIUM, or SHORT, null indicates no time in format
|
||||||
|
* @param timeZone optional time zone, overrides time zone of
|
||||||
|
* formatted date, null means use default Locale
|
||||||
|
* @param locale optional locale, overrides system locale
|
||||||
|
* @return a localized standard date/time formatter
|
||||||
|
* @throws IllegalArgumentException if the Locale has no date/time
|
||||||
|
* pattern defined
|
||||||
|
*/
|
||||||
|
// This must remain private, see LANG-884
|
||||||
|
private F getDateTimeInstance(final Integer dateStyle, final Integer timeStyle, final TimeZone timeZone, Locale locale) {
|
||||||
|
if (locale == null) {
|
||||||
|
locale = Locale.getDefault();
|
||||||
|
}
|
||||||
|
final String pattern = getPatternForStyle(dateStyle, timeStyle, locale);
|
||||||
|
return getInstance(pattern, timeZone, locale);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a date/time formatter instance using the specified style,
|
||||||
|
* time zone and locale.</p>
|
||||||
|
*
|
||||||
|
* @param dateStyle date style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @param timeStyle time style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @param timeZone optional time zone, overrides time zone of
|
||||||
|
* formatted date, null means use default Locale
|
||||||
|
* @param locale optional locale, overrides system locale
|
||||||
|
* @return a localized standard date/time formatter
|
||||||
|
* @throws IllegalArgumentException if the Locale has no date/time
|
||||||
|
* pattern defined
|
||||||
|
*/
|
||||||
|
// package protected, for access from FastDateFormat; do not make public or protected
|
||||||
|
F getDateTimeInstance(final int dateStyle, final int timeStyle, final TimeZone timeZone, Locale locale) {
|
||||||
|
return getDateTimeInstance(Integer.valueOf(dateStyle), Integer.valueOf(timeStyle), timeZone, locale);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a date formatter instance using the specified style,
|
||||||
|
* time zone and locale.</p>
|
||||||
|
*
|
||||||
|
* @param dateStyle date style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @param timeZone optional time zone, overrides time zone of
|
||||||
|
* formatted date, null means use default Locale
|
||||||
|
* @param locale optional locale, overrides system locale
|
||||||
|
* @return a localized standard date/time formatter
|
||||||
|
* @throws IllegalArgumentException if the Locale has no date/time
|
||||||
|
* pattern defined
|
||||||
|
*/
|
||||||
|
// package protected, for access from FastDateFormat; do not make public or protected
|
||||||
|
F getDateInstance(final int dateStyle, final TimeZone timeZone, Locale locale) {
|
||||||
|
return getDateTimeInstance(Integer.valueOf(dateStyle), null, timeZone, locale);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a time formatter instance using the specified style,
|
||||||
|
* time zone and locale.</p>
|
||||||
|
*
|
||||||
|
* @param timeStyle time style: FULL, LONG, MEDIUM, or SHORT
|
||||||
|
* @param timeZone optional time zone, overrides time zone of
|
||||||
|
* formatted date, null means use default Locale
|
||||||
|
* @param locale optional locale, overrides system locale
|
||||||
|
* @return a localized standard date/time formatter
|
||||||
|
* @throws IllegalArgumentException if the Locale has no date/time
|
||||||
|
* pattern defined
|
||||||
|
*/
|
||||||
|
// package protected, for access from FastDateFormat; do not make public or protected
|
||||||
|
F getTimeInstance(final int timeStyle, final TimeZone timeZone, Locale locale) {
|
||||||
|
return getDateTimeInstance(null, Integer.valueOf(timeStyle), timeZone, locale);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Gets a date/time format for the specified styles and locale.</p>
|
||||||
|
*
|
||||||
|
* @param dateStyle date style: FULL, LONG, MEDIUM, or SHORT, null indicates no date in format
|
||||||
|
* @param timeStyle time style: FULL, LONG, MEDIUM, or SHORT, null indicates no time in format
|
||||||
|
* @param locale The non-null locale of the desired format
|
||||||
|
* @return a localized standard date/time format
|
||||||
|
* @throws IllegalArgumentException if the Locale has no date/time pattern defined
|
||||||
|
*/
|
||||||
|
// package protected, for access from test code; do not make public or protected
|
||||||
|
static String getPatternForStyle(final Integer dateStyle, final Integer timeStyle, final Locale locale) {
|
||||||
|
final MultipartKey key = new MultipartKey(dateStyle, timeStyle, locale);
|
||||||
|
|
||||||
|
String pattern = cDateTimeInstanceCache.get(key);
|
||||||
|
if (pattern == null) {
|
||||||
|
try {
|
||||||
|
DateFormat formatter;
|
||||||
|
if (dateStyle == null) {
|
||||||
|
formatter = DateFormat.getTimeInstance(timeStyle.intValue(), locale);
|
||||||
|
} else if (timeStyle == null) {
|
||||||
|
formatter = DateFormat.getDateInstance(dateStyle.intValue(), locale);
|
||||||
|
} else {
|
||||||
|
formatter = DateFormat.getDateTimeInstance(dateStyle.intValue(), timeStyle.intValue(), locale);
|
||||||
|
}
|
||||||
|
pattern = ((SimpleDateFormat) formatter).toPattern();
|
||||||
|
final String previous = cDateTimeInstanceCache.putIfAbsent(key, pattern);
|
||||||
|
if (previous != null) {
|
||||||
|
// even though it doesn't matter if another thread put the pattern
|
||||||
|
// it's still good practice to return the String instance that is
|
||||||
|
// actually in the ConcurrentMap
|
||||||
|
pattern = previous;
|
||||||
|
}
|
||||||
|
} catch (final ClassCastException ex) {
|
||||||
|
throw new IllegalArgumentException("No date time pattern for locale: " + locale);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return pattern;
|
||||||
|
}
|
||||||
|
|
||||||
|
// ----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>Helper class to hold multi-part Map keys</p>
|
||||||
|
*/
|
||||||
|
private static class MultipartKey {
|
||||||
|
private final Object[] keys;
|
||||||
|
private int hashCode;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs an instance of <code>MultipartKey</code> to hold the specified objects.
|
||||||
|
*
|
||||||
|
* @param keys the set of objects that make up the key. Each key may be null.
|
||||||
|
*/
|
||||||
|
public MultipartKey(final Object... keys) {
|
||||||
|
this.keys = keys;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public boolean equals(final Object obj) {
|
||||||
|
// Eliminate the usual boilerplate because
|
||||||
|
// this inner static class is only used in a generic ConcurrentHashMap
|
||||||
|
// which will not compare against other Object types
|
||||||
|
return Arrays.equals(keys, ((MultipartKey) obj).keys);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
if (hashCode == 0) {
|
||||||
|
int rc = 0;
|
||||||
|
for (final Object key : keys) {
|
||||||
|
if (key != null) {
|
||||||
|
rc = rc * 7 + key.hashCode();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
hashCode = rc;
|
||||||
|
}
|
||||||
|
return hashCode;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -11,7 +11,7 @@ package org.telegram.messenger;
|
|||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
|
|
||||||
import org.telegram.android.FastDateFormat;
|
import org.telegram.android.time.FastDateFormat;
|
||||||
import org.telegram.ui.ApplicationLoader;
|
import org.telegram.ui.ApplicationLoader;
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
|
@ -89,8 +89,8 @@ public class VideoEditorActivity extends BaseFragment implements TextureView.Sur
|
|||||||
private float videoDuration = 0;
|
private float videoDuration = 0;
|
||||||
private long startTime = 0;
|
private long startTime = 0;
|
||||||
private long endTime = 0;
|
private long endTime = 0;
|
||||||
private int audioFramesSize = 0;
|
private long audioFramesSize = 0;
|
||||||
private int videoFramesSize = 0;
|
private long videoFramesSize = 0;
|
||||||
private int estimatedSize = 0;
|
private int estimatedSize = 0;
|
||||||
private long esimatedDuration = 0;
|
private long esimatedDuration = 0;
|
||||||
private long originalSize = 0;
|
private long originalSize = 0;
|
||||||
@ -292,7 +292,9 @@ public class VideoEditorActivity extends BaseFragment implements TextureView.Sur
|
|||||||
name.equals("OMX.ST.VFM.H264Enc") ||
|
name.equals("OMX.ST.VFM.H264Enc") ||
|
||||||
name.equals("OMX.Exynos.avc.enc") ||
|
name.equals("OMX.Exynos.avc.enc") ||
|
||||||
name.equals("OMX.MARVELL.VIDEO.HW.CODA7542ENCODER") ||
|
name.equals("OMX.MARVELL.VIDEO.HW.CODA7542ENCODER") ||
|
||||||
name.equals("OMX.MARVELL.VIDEO.H264ENCODER")) {
|
name.equals("OMX.MARVELL.VIDEO.H264ENCODER") ||
|
||||||
|
name.equals("OMX.k3.video.encoder.avc") || //fix this later
|
||||||
|
name.equals("OMX.TI.DUCATI1.VIDEO.H264E")) { //fix this later
|
||||||
compressVideo.setVisibility(View.GONE);
|
compressVideo.setVisibility(View.GONE);
|
||||||
} else {
|
} else {
|
||||||
if (MediaController.selectColorFormat(codecInfo, MediaController.MIME_TYPE) == 0) {
|
if (MediaController.selectColorFormat(codecInfo, MediaController.MIME_TYPE) == 0) {
|
||||||
@ -721,8 +723,8 @@ public class VideoEditorActivity extends BaseFragment implements TextureView.Sur
|
|||||||
|
|
||||||
for (Box box : boxes) {
|
for (Box box : boxes) {
|
||||||
TrackBox trackBox = (TrackBox)box;
|
TrackBox trackBox = (TrackBox)box;
|
||||||
int sampleSizes = 0;
|
long sampleSizes = 0;
|
||||||
int trackBitrate = 0;
|
long trackBitrate = 0;
|
||||||
try {
|
try {
|
||||||
MediaBox mediaBox = trackBox.getMediaBox();
|
MediaBox mediaBox = trackBox.getMediaBox();
|
||||||
MediaHeaderBox mediaHeaderBox = mediaBox.getMediaHeaderBox();
|
MediaHeaderBox mediaHeaderBox = mediaBox.getMediaHeaderBox();
|
||||||
@ -738,7 +740,7 @@ public class VideoEditorActivity extends BaseFragment implements TextureView.Sur
|
|||||||
TrackHeaderBox headerBox = trackBox.getTrackHeaderBox();
|
TrackHeaderBox headerBox = trackBox.getTrackHeaderBox();
|
||||||
if (headerBox.getWidth() != 0 && headerBox.getHeight() != 0) {
|
if (headerBox.getWidth() != 0 && headerBox.getHeight() != 0) {
|
||||||
trackHeaderBox = headerBox;
|
trackHeaderBox = headerBox;
|
||||||
bitrate = trackBitrate / 100000 * 100000;
|
bitrate = (int)(trackBitrate / 100000 * 100000);
|
||||||
if (bitrate > 900000) {
|
if (bitrate > 900000) {
|
||||||
bitrate = 900000;
|
bitrate = 900000;
|
||||||
}
|
}
|
||||||
@ -768,7 +770,7 @@ public class VideoEditorActivity extends BaseFragment implements TextureView.Sur
|
|||||||
resultHeight *= scale;
|
resultHeight *= scale;
|
||||||
if (bitrate != 0) {
|
if (bitrate != 0) {
|
||||||
bitrate *= Math.max(0.5f, scale);
|
bitrate *= Math.max(0.5f, scale);
|
||||||
videoFramesSize = (int)(bitrate / 8 * videoDuration);
|
videoFramesSize = (long)(bitrate / 8 * videoDuration);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user