Merge remote-tracking branch 'remotes/upstream/material_fab' into beta
[pub/Android/ownCloud.git] / src / com / owncloud / android / utils / DisplayUtils.java
1 /**
2 * ownCloud Android client application
3 *
4 * @author Bartek Przybylski
5 * @author David A. Velasco
6 * Copyright (C) 2011 Bartek Przybylski
7 * Copyright (C) 2015 ownCloud Inc.
8 *
9 * This program is free software: you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License version 2,
11 * as published by the Free Software Foundation.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program. If not, see <http://www.gnu.org/licenses/>.
20 *
21 */
22
23 package com.owncloud.android.utils;
24
25 import java.io.File;
26 import java.net.IDN;
27 import java.text.DateFormat;
28 import java.util.Arrays;
29 import java.util.Calendar;
30 import java.util.Date;
31 import java.util.HashMap;
32 import java.util.HashSet;
33 import java.util.Set;
34 import java.util.Vector;
35
36 import android.annotation.TargetApi;
37 import android.app.Activity;
38 import android.content.Context;
39 import android.content.SharedPreferences;
40 import android.graphics.Point;
41 import android.graphics.PorterDuff;
42 import android.os.Build;
43 import android.text.format.DateUtils;
44 import android.view.Display;
45 import android.widget.ProgressBar;
46 import android.widget.SeekBar;
47
48 import com.owncloud.android.MainApp;
49 import com.owncloud.android.R;
50 import com.owncloud.android.datamodel.FileDataStorageManager;
51 import com.owncloud.android.datamodel.OCFile;
52
53 import java.math.BigDecimal;
54 import java.net.IDN;
55 import java.text.DateFormat;
56 import java.util.Calendar;
57 import java.util.Date;
58 import java.util.HashMap;
59 import java.util.Map;
60
61 /**
62 * A helper class for some string operations.
63 */
64 public class DisplayUtils {
65
66 private static final String OWNCLOUD_APP_NAME = "ownCloud";
67
68 private static final String[] sizeSuffixes = { "B", "KB", "MB", "GB", "TB", "PB", "EB", "ZB", "YB" };
69 private static final int[] sizeScales = { 0, 0, 0, 1, 1, 2, 2, 2, 2 };
70
71 private static Map<String, String> mimeType2HumanReadable;
72
73 static {
74 mimeType2HumanReadable = new HashMap<String, String>();
75 // images
76 mimeType2HumanReadable.put("image/jpeg", "JPEG image");
77 mimeType2HumanReadable.put("image/jpg", "JPEG image");
78 mimeType2HumanReadable.put("image/png", "PNG image");
79 mimeType2HumanReadable.put("image/bmp", "Bitmap image");
80 mimeType2HumanReadable.put("image/gif", "GIF image");
81 mimeType2HumanReadable.put("image/svg+xml", "JPEG image");
82 mimeType2HumanReadable.put("image/tiff", "TIFF image");
83 // music
84 mimeType2HumanReadable.put("audio/mpeg", "MP3 music file");
85 mimeType2HumanReadable.put("application/ogg", "OGG music file");
86 }
87
88 /**
89 * Converts the file size in bytes to human readable output.
90 * <ul>
91 * <li>appends a size suffix, e.g. B, KB, MB etc.</li>
92 * <li>rounds the size based on the suffix to 0,1 or 2 decimals</li>
93 * </ul>
94 *
95 * @param bytes Input file size
96 * @return Like something readable like "12 MB"
97 */
98 public static String bytesToHumanReadable(long bytes) {
99 double result = bytes;
100 int attachedSuff = 0;
101 while (result > 1024 && attachedSuff < sizeSuffixes.length) {
102 result /= 1024.;
103 attachedSuff++;
104 }
105
106 return new BigDecimal(result).setScale(
107 sizeScales[attachedSuff], BigDecimal.ROUND_HALF_UP) + " " + sizeSuffixes[attachedSuff];
108 }
109
110 /**
111 * Converts MIME types like "image/jpg" to more end user friendly output
112 * like "JPG image".
113 *
114 * @param mimetype MIME type to convert
115 * @return A human friendly version of the MIME type
116 */
117 public static String convertMIMEtoPrettyPrint(String mimetype) {
118 if (mimeType2HumanReadable.containsKey(mimetype)) {
119 return mimeType2HumanReadable.get(mimetype);
120 }
121 if (mimetype.split("/").length >= 2)
122 return mimetype.split("/")[1].toUpperCase() + " file";
123 return "Unknown type";
124 }
125
126 /**
127 * Converts Unix time to human readable format
128 * @param milliseconds that have passed since 01/01/1970
129 * @return The human readable time for the users locale
130 */
131 public static String unixTimeToHumanReadable(long milliseconds) {
132 Date date = new Date(milliseconds);
133 DateFormat df = DateFormat.getDateTimeInstance();
134 return df.format(date);
135 }
136
137 public static int getSeasonalIconId() {
138 if (Calendar.getInstance().get(Calendar.DAY_OF_YEAR) >= 354 &&
139 MainApp.getAppContext().getString(R.string.app_name).equals(OWNCLOUD_APP_NAME)) {
140 return R.drawable.winter_holidays_icon;
141 } else {
142 return R.drawable.icon;
143 }
144 }
145
146 /**
147 * Converts an internationalized domain name (IDN) in an URL to and from ASCII/Unicode.
148 * @param url the URL where the domain name should be converted
149 * @param toASCII if true converts from Unicode to ASCII, if false converts from ASCII to Unicode
150 * @return the URL containing the converted domain name
151 */
152 @TargetApi(Build.VERSION_CODES.GINGERBREAD)
153 public static String convertIdn(String url, boolean toASCII) {
154
155 String urlNoDots = url;
156 String dots="";
157 while (urlNoDots.startsWith(".")) {
158 urlNoDots = url.substring(1);
159 dots = dots + ".";
160 }
161
162 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.GINGERBREAD) {
163 // Find host name after '//' or '@'
164 int hostStart = 0;
165 if (urlNoDots.indexOf("//") != -1) {
166 hostStart = url.indexOf("//") + "//".length();
167 } else if (url.indexOf("@") != -1) {
168 hostStart = url.indexOf("@") + "@".length();
169 }
170
171 int hostEnd = url.substring(hostStart).indexOf("/");
172 // Handle URL which doesn't have a path (path is implicitly '/')
173 hostEnd = (hostEnd == -1 ? urlNoDots.length() : hostStart + hostEnd);
174
175 String host = urlNoDots.substring(hostStart, hostEnd);
176 host = (toASCII ? IDN.toASCII(host) : IDN.toUnicode(host));
177
178 return dots + urlNoDots.substring(0, hostStart) + host + urlNoDots.substring(hostEnd);
179 } else {
180 return dots + url;
181 }
182 }
183
184 /**
185 * Get the file extension if it is on path as type "content://.../DocInfo.doc"
186 * @param filepath: Content Uri converted to string format
187 * @return String: fileExtension (type '.pdf'). Empty if no extension
188 */
189 public static String getComposedFileExtension(String filepath) {
190 String fileExtension = "";
191 String fileNameInContentUri = filepath.substring(filepath.lastIndexOf("/"));
192
193 // Check if extension is included in uri
194 int pos = fileNameInContentUri.lastIndexOf('.');
195 if (pos >= 0) {
196 fileExtension = fileNameInContentUri.substring(pos);
197 }
198 return fileExtension;
199 }
200
201 @SuppressWarnings("deprecation")
202 public static CharSequence getRelativeDateTimeString (
203 Context c, long time, long minResolution, long transitionResolution, int flags
204 ){
205
206 CharSequence dateString = "";
207
208 // in Future
209 if (time > System.currentTimeMillis()){
210 return DisplayUtils.unixTimeToHumanReadable(time);
211 }
212 // < 60 seconds -> seconds ago
213 else if ((System.currentTimeMillis() - time) < 60 * 1000) {
214 return c.getString(R.string.file_list_seconds_ago);
215 } else {
216 // Workaround 2.x bug (see https://github.com/owncloud/android/issues/716)
217 if ( Build.VERSION.SDK_INT <= Build.VERSION_CODES.HONEYCOMB &&
218 (System.currentTimeMillis() - time) > 24 * 60 * 60 * 1000 ) {
219 Date date = new Date(time);
220 date.setHours(0);
221 date.setMinutes(0);
222 date.setSeconds(0);
223 dateString = DateUtils.getRelativeDateTimeString(
224 c, date.getTime(), minResolution, transitionResolution, flags
225 );
226 } else {
227 dateString = DateUtils.getRelativeDateTimeString(c, time, minResolution, transitionResolution, flags);
228 }
229 }
230
231 String[] parts = dateString.toString().split(",");
232 if (parts.length == 2) {
233 if (parts[1].contains(":") && !parts[0].contains(":")) {
234 return parts[0];
235 } else if (parts[0].contains(":") && !parts[1].contains(":")) {
236 return parts[1];
237 }
238 }
239 //dateString contains unexpected format. use localized, absolute date.
240 return DisplayUtils.unixTimeToHumanReadable(time);
241 }
242
243 /**
244 * Update the passed path removing the last "/" if it is not the root folder
245 * @param path
246 */
247 public static String getPathWithoutLastSlash(String path) {
248
249 // Remove last slash from path
250 if (path.length() > 1 && path.charAt(path.length()-1) == OCFile.PATH_SEPARATOR.charAt(0)) {
251 path = path.substring(0, path.length()-1);
252 }
253 return path;
254 }
255
256
257 /**
258 * Gets the screen size in pixels in a backwards compatible way
259 *
260 * @param caller Activity calling; needed to get access to the {@link android.view.WindowManager}
261 * @return Size in pixels of the screen, or default {@link Point} if caller is null
262 */
263 public static Point getScreenSize(Activity caller) {
264 Point size = new Point();
265 if (caller != null) {
266 Display display = caller.getWindowManager().getDefaultDisplay();
267 if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.HONEYCOMB_MR2) {
268 display.getSize(size);
269 } else {
270 size.set(display.getWidth(), display.getHeight());
271 }
272 }
273 return size;
274 }
275
276 /**
277 * Determines if user set folder to grid or list view. If folder is not set itself,
278 * it finds a parent that is set (at least root is set).
279 * @param file
280 * @param storageManager
281 * @return
282 */
283 public static boolean isGridView(OCFile file, FileDataStorageManager storageManager){
284 if (file != null) {
285 OCFile fileToTest = file;
286 OCFile parentDir = null;
287 String parentPath = null;
288
289 SharedPreferences setting = MainApp.getAppContext().getSharedPreferences(
290 "viewMode", Context.MODE_PRIVATE);
291
292 if (setting.contains(fileToTest.getRemoteId())) {
293 return setting.getBoolean(fileToTest.getRemoteId(), false);
294 } else {
295 do {
296 if (fileToTest.getParentId() != FileDataStorageManager.ROOT_PARENT_ID) {
297 parentPath = new File(fileToTest.getRemotePath()).getParent();
298 parentPath = parentPath.endsWith(OCFile.PATH_SEPARATOR) ? parentPath :
299 parentPath + OCFile.PATH_SEPARATOR;
300 parentDir = storageManager.getFileByPath(parentPath);
301 } else {
302 parentDir = storageManager.getFileByPath(OCFile.ROOT_PATH);
303 }
304
305 while (parentDir == null) {
306 parentPath = new File(parentPath).getParent();
307 parentPath = parentPath.endsWith(OCFile.PATH_SEPARATOR) ? parentPath :
308 parentPath + OCFile.PATH_SEPARATOR;
309 parentDir = storageManager.getFileByPath(parentPath);
310 }
311 fileToTest = parentDir;
312 } while (endWhile(parentDir, setting));
313 return setting.getBoolean(fileToTest.getRemoteId(), false);
314 }
315 } else {
316 return false;
317 }
318 }
319
320 private static boolean endWhile(OCFile parentDir, SharedPreferences setting) {
321 if (parentDir.getRemotePath().compareToIgnoreCase(OCFile.ROOT_PATH) == 0) {
322 return false;
323 } else {
324 return !setting.contains(parentDir.getRemoteId());
325 }
326 }
327
328 public static void setViewMode(OCFile file, boolean setGrid){
329 SharedPreferences setting = MainApp.getAppContext().getSharedPreferences(
330 "viewMode", Context.MODE_PRIVATE);
331
332 SharedPreferences.Editor editor = setting.edit();
333 editor.putBoolean(file.getRemoteId(), setGrid);
334 editor.commit();
335 }
336
337 /**
338 * sets the coloring of the given progress bar to color_accent.
339 *
340 * @param progressBar the progress bar to be colored
341 */
342 public static void colorPreLollipopHorizontalProgressBar(ProgressBar progressBar) {
343 if (progressBar != null && Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) {
344 int color = progressBar.getResources().getColor(R.color.color_accent);
345 progressBar.getIndeterminateDrawable().setColorFilter(color, PorterDuff.Mode.SRC_IN);
346 progressBar.getProgressDrawable().setColorFilter(color, PorterDuff.Mode.SRC_IN);
347 }
348 }
349
350 /**
351 * sets the coloring of the given seek bar to color_accent.
352 *
353 * @param seekBar the seek bar to be colored
354 */
355 public static void colorPreLollipopHorizontalSeekBar(SeekBar seekBar) {
356 if (seekBar != null && Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) {
357 colorPreLollipopHorizontalProgressBar(seekBar);
358
359 if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) {
360 int color = seekBar.getResources().getColor(R.color.color_accent);
361 seekBar.getThumb().setColorFilter(color, PorterDuff.Mode.SRC_IN);
362 seekBar.getThumb().setColorFilter(color, PorterDuff.Mode.SRC_IN);
363 }
364 }
365 }
366 }