1 /* ownCloud Android client application
2 * Copyright (C) 2011 Bartek Przybylski
3 * Copyright (C) 2012-2013 ownCloud Inc.
5 * This program is free software: you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License version 2,
7 * as published by the Free Software Foundation.
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with this program. If not, see <http://www.gnu.org/licenses/>.
19 package com
.owncloud
.android
.utils
;
22 import java
.text
.DateFormat
;
23 import java
.util
.Arrays
;
24 import java
.util
.Calendar
;
25 import java
.util
.Date
;
26 import java
.util
.HashMap
;
27 import java
.util
.HashSet
;
29 import java
.util
.Vector
;
31 import android
.annotation
.TargetApi
;
32 import android
.content
.Context
;
33 import android
.os
.Build
;
34 import android
.text
.format
.DateUtils
;
35 import android
.webkit
.MimeTypeMap
;
37 import com
.owncloud
.android
.MainApp
;
38 import com
.owncloud
.android
.R
;
39 import com
.owncloud
.android
.datamodel
.OCFile
;
42 * A helper class for some string operations.
44 * @author Bartek Przybylski
45 * @author David A. Velasco
47 public class DisplayUtils
{
49 private static final String OWNCLOUD_APP_NAME
= "ownCloud";
51 //private static String TAG = DisplayUtils.class.getSimpleName();
53 private static final String
[] sizeSuffixes
= { "B", "KB", "MB", "GB", "TB", "PB", "EB", "ZB", "YB" };
55 private static HashMap
<String
, String
> mimeType2HUmanReadable
;
57 mimeType2HUmanReadable
= new HashMap
<String
, String
>();
59 mimeType2HUmanReadable
.put("image/jpeg", "JPEG image");
60 mimeType2HUmanReadable
.put("image/jpg", "JPEG image");
61 mimeType2HUmanReadable
.put("image/png", "PNG image");
62 mimeType2HUmanReadable
.put("image/bmp", "Bitmap image");
63 mimeType2HUmanReadable
.put("image/gif", "GIF image");
64 mimeType2HUmanReadable
.put("image/svg+xml", "JPEG image");
65 mimeType2HUmanReadable
.put("image/tiff", "TIFF image");
67 mimeType2HUmanReadable
.put("audio/mpeg", "MP3 music file");
68 mimeType2HUmanReadable
.put("application/ogg", "OGG music file");
72 private static final String TYPE_APPLICATION
= "application";
73 private static final String TYPE_AUDIO
= "audio";
74 private static final String TYPE_IMAGE
= "image";
75 private static final String TYPE_TXT
= "text";
76 private static final String TYPE_VIDEO
= "video";
78 private static final String SUBTYPE_PDF
= "pdf";
79 private static final String SUBTYPE_XML
= "xml";
80 private static final String
[] SUBTYPES_DOCUMENT
= {
82 "vnd.openxmlformats-officedocument.wordprocessingml.document",
83 "vnd.oasis.opendocument.text",
87 private static Set
<String
> SUBTYPES_DOCUMENT_SET
= new HashSet
<String
>(Arrays
.asList(SUBTYPES_DOCUMENT
));
88 private static final String
[] SUBTYPES_SPREADSHEET
= {
91 "vnd.openxmlformats-officedocument.spreadsheetml.sheet",
92 "vnd.oasis.opendocument.spreadsheet"
94 private static Set
<String
> SUBTYPES_SPREADSHEET_SET
= new HashSet
<String
>(Arrays
.asList(SUBTYPES_SPREADSHEET
));
95 private static final String
[] SUBTYPES_PRESENTATION
= {
98 "vnd.openxmlformats-officedocument.presentationml.presentation",
99 "vnd.oasis.opendocument.presentation"
101 private static Set
<String
> SUBTYPES_PRESENTATION_SET
= new HashSet
<String
>(Arrays
.asList(SUBTYPES_PRESENTATION
));
102 private static final String
[] SUBTYPES_COMPRESSED
= {"x-tar", "x-gzip", "zip"};
103 private static final Set
<String
> SUBTYPES_COMPRESSED_SET
= new HashSet
<String
>(Arrays
.asList(SUBTYPES_COMPRESSED
));
104 private static final String SUBTYPE_OCTET_STREAM
= "octet-stream";
105 private static final String EXTENSION_RAR
= "rar";
106 private static final String EXTENSION_RTF
= "rtf";
107 private static final String EXTENSION_3GP
= "3gp";
108 private static final String EXTENSION_PY
= "py";
109 private static final String EXTENSION_JS
= "js";
112 * Converts the file size in bytes to human readable output.
114 * @param bytes Input file size
115 * @return Like something readable like "12 MB"
117 public static String
bytesToHumanReadable(long bytes
) {
118 double result
= bytes
;
119 int attachedsuff
= 0;
120 while (result
> 1024 && attachedsuff
< sizeSuffixes
.length
) {
124 result
= ((int) (result
* 100)) / 100.;
125 return result
+ " " + sizeSuffixes
[attachedsuff
];
129 * Converts MIME types like "image/jpg" to more end user friendly output
132 * @param mimetype MIME type to convert
133 * @return A human friendly version of the MIME type
135 public static String
convertMIMEtoPrettyPrint(String mimetype
) {
136 if (mimeType2HUmanReadable
.containsKey(mimetype
)) {
137 return mimeType2HUmanReadable
.get(mimetype
);
139 if (mimetype
.split("/").length
>= 2)
140 return mimetype
.split("/")[1].toUpperCase() + " file";
141 return "Unknown type";
146 * Returns the resource identifier of an image to use as icon associated to a known MIME type.
148 * @param mimetype MIME type string; if NULL, the method tries to guess it from the extension in filename
149 * @param filename Name, with extension.
150 * @return Identifier of an image resource.
152 public static int getFileTypeIconId(String mimetype
, String filename
) {
154 if (mimetype
== null
) {
155 String fileExtension
= getExtension(filename
);
156 mimetype
= MimeTypeMap
.getSingleton().getMimeTypeFromExtension(fileExtension
);
157 if (mimetype
== null
) {
158 mimetype
= TYPE_APPLICATION
+ "/" + SUBTYPE_OCTET_STREAM
;
162 if ("DIR".equals(mimetype
)) {
163 return R
.drawable
.ic_menu_archive
;
166 String
[] parts
= mimetype
.split("/");
167 String type
= parts
[0];
168 String subtype
= (parts
.length
> 1) ? parts
[1] : "";
170 if(TYPE_TXT
.equals(type
)) {
171 return R
.drawable
.file_doc
;
173 } else if(TYPE_IMAGE
.equals(type
)) {
174 return R
.drawable
.file_image
;
176 } else if(TYPE_VIDEO
.equals(type
)) {
177 return R
.drawable
.file_movie
;
179 } else if(TYPE_AUDIO
.equals(type
)) {
180 return R
.drawable
.file_sound
;
182 } else if(TYPE_APPLICATION
.equals(type
)) {
184 if (SUBTYPE_PDF
.equals(subtype
)) {
185 return R
.drawable
.file_pdf
;
187 } else if (SUBTYPE_XML
.equals(subtype
)) {
188 return R
.drawable
.file_doc
;
190 } else if (SUBTYPES_DOCUMENT_SET
.contains(subtype
)) {
191 return R
.drawable
.file_doc
;
193 } else if (SUBTYPES_SPREADSHEET_SET
.contains(subtype
)) {
194 return R
.drawable
.file_xls
;
196 } else if (SUBTYPES_PRESENTATION_SET
.contains(subtype
)) {
197 return R
.drawable
.file_ppt
;
199 } else if (SUBTYPES_COMPRESSED_SET
.contains(subtype
)) {
200 return R
.drawable
.file_zip
;
202 } else if (SUBTYPE_OCTET_STREAM
.equals(subtype
) ) {
203 if (getExtension(filename
).equalsIgnoreCase(EXTENSION_RAR
)) {
204 return R
.drawable
.file_zip
;
206 } else if (getExtension(filename
).equalsIgnoreCase(EXTENSION_RTF
)) {
207 return R
.drawable
.file_doc
;
209 } else if (getExtension(filename
).equalsIgnoreCase(EXTENSION_3GP
)) {
210 return R
.drawable
.file_movie
;
212 } else if ( getExtension(filename
).equalsIgnoreCase(EXTENSION_PY
) ||
213 getExtension(filename
).equalsIgnoreCase(EXTENSION_JS
)) {
214 return R
.drawable
.file_doc
;
221 return R
.drawable
.file
;
225 private static String
getExtension(String filename
) {
226 String extension
= filename
.substring(filename
.lastIndexOf(".") + 1).toLowerCase();
231 * Converts Unix time to human readable format
232 * @param milliseconds that have passed since 01/01/1970
233 * @return The human readable time for the users locale
235 public static String
unixTimeToHumanReadable(long milliseconds
) {
236 Date date
= new Date(milliseconds
);
237 DateFormat df
= DateFormat
.getDateTimeInstance();
238 return df
.format(date
);
242 public static int getSeasonalIconId() {
243 if (Calendar
.getInstance().get(Calendar
.DAY_OF_YEAR
) >= 354 &&
244 MainApp
.getAppContext().getString(R
.string
.app_name
).equals(OWNCLOUD_APP_NAME
)) {
245 return R
.drawable
.winter_holidays_icon
;
247 return R
.drawable
.icon
;
252 * Converts an internationalized domain name (IDN) in an URL to and from ASCII/Unicode.
253 * @param url the URL where the domain name should be converted
254 * @param toASCII if true converts from Unicode to ASCII, if false converts from ASCII to Unicode
255 * @return the URL containing the converted domain name
257 @TargetApi(Build
.VERSION_CODES
.GINGERBREAD
)
258 public static String
convertIdn(String url
, boolean toASCII
) {
260 if (Build
.VERSION
.SDK_INT
>= Build
.VERSION_CODES
.GINGERBREAD
) {
261 // Find host name after '//' or '@'
263 if (url
.indexOf("//") != -1) {
264 hostStart
= url
.indexOf("//") + "//".length();
265 } else if (url
.indexOf("@") != -1) {
266 hostStart
= url
.indexOf("@") + "@".length();
269 int hostEnd
= url
.substring(hostStart
).indexOf("/");
270 // Handle URL which doesn't have a path (path is implicitly '/')
271 hostEnd
= (hostEnd
== -1 ? url
.length() : hostStart
+ hostEnd
);
273 String host
= url
.substring(hostStart
, hostEnd
);
274 host
= (toASCII ? IDN
.toASCII(host
) : IDN
.toUnicode(host
));
276 return url
.substring(0, hostStart
) + host
+ url
.substring(hostEnd
);
283 * Get the file extension if it is on path as type "content://.../DocInfo.doc"
284 * @param filepath: Content Uri converted to string format
285 * @return String: fileExtension (type '.pdf'). Empty if no extension
287 public static String
getComposedFileExtension(String filepath
) {
288 String fileExtension
= "";
289 String fileNameInContentUri
= filepath
.substring(filepath
.lastIndexOf("/"));
291 // Check if extension is included in uri
292 int pos
= fileNameInContentUri
.lastIndexOf('.');
294 fileExtension
= fileNameInContentUri
.substring(pos
);
296 return fileExtension
;
299 @SuppressWarnings("deprecation")
300 public static CharSequence
getRelativeDateTimeString (
301 Context c
, long time
, long minResolution
, long transitionResolution
, int flags
304 CharSequence dateString
= "";
307 if (time
> System
.currentTimeMillis()){
308 return DisplayUtils
.unixTimeToHumanReadable(time
);
310 // < 60 seconds -> seconds ago
311 else if ((System
.currentTimeMillis() - time
) < 60 * 1000) {
312 return c
.getString(R
.string
.file_list_seconds_ago
);
314 // Workaround 2.x bug (see https://github.com/owncloud/android/issues/716)
315 if ( Build
.VERSION
.SDK_INT
<= Build
.VERSION_CODES
.HONEYCOMB
&&
316 (System
.currentTimeMillis() - time
) > 24 * 60 * 60 * 1000 ) {
317 Date date
= new Date(time
);
321 dateString
= DateUtils
.getRelativeDateTimeString(
322 c
, date
.getTime(), minResolution
, transitionResolution
, flags
325 dateString
= DateUtils
.getRelativeDateTimeString(c
, time
, minResolution
, transitionResolution
, flags
);
329 return dateString
.toString().split(",")[0];
333 * Update the passed path removing the last "/" if it is not the root folder
336 public static String
getPathWithoutLastSlash(String path
) {
338 // Remove last slash from path
339 if (path
.length() > 1 && path
.charAt(path
.length()-1) == OCFile
.PATH_SEPARATOR
.charAt(0)) {
340 path
= path
.substring(0, path
.length()-1);