mirror of https://github.com/M66B/FairEmail.git
Added support for favicons
This commit is contained in:
parent
4b96f6830f
commit
4ffed65da3
8
FAQ.md
8
FAQ.md
|
@ -276,7 +276,7 @@ Fonts, sizes, colors, etc should be material design whenever possible.
|
||||||
* [(151) Can you add backup/restore messages?](#user-content-faq151)
|
* [(151) Can you add backup/restore messages?](#user-content-faq151)
|
||||||
* [(152) How can I insert a contact group?](#user-content-faq152)
|
* [(152) How can I insert a contact group?](#user-content-faq152)
|
||||||
* [(153) Why does permanently deleting Gmail message not work?](#user-content-faq153)
|
* [(153) Why does permanently deleting Gmail message not work?](#user-content-faq153)
|
||||||
* [(154) Can you add favicons as contact photos?](#user-content-faq154)
|
* [~~(154) Can you add favicons as contact photos?~~](#user-content-faq154)
|
||||||
* [(155) What is a winmail.dat file?](#user-content-faq155)
|
* [(155) What is a winmail.dat file?](#user-content-faq155)
|
||||||
* [(156) How can I set up an Office365 account?](#user-content-faq156)
|
* [(156) How can I set up an Office365 account?](#user-content-faq156)
|
||||||
* [(157) How can I set up an Free.fr account?](#user-content-faq157)
|
* [(157) How can I set up an Free.fr account?](#user-content-faq157)
|
||||||
|
@ -3045,10 +3045,10 @@ Some background: Gmail seems to have an additional message view for IMAP, which
|
||||||
<br />
|
<br />
|
||||||
|
|
||||||
<a name="faq154"></a>
|
<a name="faq154"></a>
|
||||||
**(154) Can you add favicons as contact photos?**
|
**~~(154) Can you add favicons as contact photos?~~**
|
||||||
|
|
||||||
Besides that a [favicon](https://en.wikipedia.org/wiki/Favicon) might be shared by many email addresses with the same domain name
|
~~Besides that a [favicon](https://en.wikipedia.org/wiki/Favicon) might be shared by many email addresses with the same domain name~~
|
||||||
and therefore is not directly related to an email address, favicons can be used to track you.
|
~~and therefore is not directly related to an email address, favicons can be used to track you.~~
|
||||||
|
|
||||||
<br />
|
<br />
|
||||||
|
|
||||||
|
|
|
@ -36,14 +36,24 @@ import android.text.TextUtils;
|
||||||
import androidx.annotation.NonNull;
|
import androidx.annotation.NonNull;
|
||||||
import androidx.preference.PreferenceManager;
|
import androidx.preference.PreferenceManager;
|
||||||
|
|
||||||
|
import org.jsoup.nodes.Document;
|
||||||
|
import org.jsoup.nodes.Element;
|
||||||
|
|
||||||
|
import java.io.BufferedOutputStream;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileOutputStream;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
|
import java.io.OutputStream;
|
||||||
import java.io.UnsupportedEncodingException;
|
import java.io.UnsupportedEncodingException;
|
||||||
import java.net.HttpURLConnection;
|
import java.net.HttpURLConnection;
|
||||||
|
import java.net.SocketTimeoutException;
|
||||||
import java.net.URL;
|
import java.net.URL;
|
||||||
import java.nio.charset.StandardCharsets;
|
import java.nio.charset.StandardCharsets;
|
||||||
|
import java.util.ArrayList;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.concurrent.ConcurrentHashMap;
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
|
@ -51,6 +61,7 @@ import java.util.concurrent.ExecutorService;
|
||||||
|
|
||||||
import javax.mail.Address;
|
import javax.mail.Address;
|
||||||
import javax.mail.internet.InternetAddress;
|
import javax.mail.internet.InternetAddress;
|
||||||
|
import javax.net.ssl.HttpsURLConnection;
|
||||||
|
|
||||||
public class ContactInfo {
|
public class ContactInfo {
|
||||||
private String email;
|
private String email;
|
||||||
|
@ -63,13 +74,20 @@ public class ContactInfo {
|
||||||
private static Map<String, Lookup> emailLookup = new ConcurrentHashMap<>();
|
private static Map<String, Lookup> emailLookup = new ConcurrentHashMap<>();
|
||||||
private static final Map<String, ContactInfo> emailContactInfo = new HashMap<>();
|
private static final Map<String, ContactInfo> emailContactInfo = new HashMap<>();
|
||||||
private static final Map<String, Avatar> emailGravatar = new HashMap<>();
|
private static final Map<String, Avatar> emailGravatar = new HashMap<>();
|
||||||
|
private static final List<String> emailFaviconBlacklist = new ArrayList<>();
|
||||||
|
|
||||||
private static final ExecutorService executor =
|
private static final ExecutorService executor =
|
||||||
Helper.getBackgroundExecutor(1, "contact");
|
Helper.getBackgroundExecutor(1, "contact");
|
||||||
|
|
||||||
private static final int GRAVATAR_TIMEOUT = 5 * 1000; // milliseconds
|
private static final int GRAVATAR_TIMEOUT = 5 * 1000; // milliseconds
|
||||||
|
private static final int FAVICON_TIMEOUT = 15 * 1000; // milliseconds
|
||||||
private static final long CACHE_CONTACT_DURATION = 2 * 60 * 1000L; // milliseconds
|
private static final long CACHE_CONTACT_DURATION = 2 * 60 * 1000L; // milliseconds
|
||||||
private static final long CACHE_GRAVATAR_DURATION = 2 * 60 * 60 * 1000L; // milliseconds
|
private static final long CACHE_GRAVATAR_DURATION = 2 * 60 * 60 * 1000L; // milliseconds
|
||||||
|
|
||||||
|
static {
|
||||||
|
emailFaviconBlacklist.add("gmail.com");
|
||||||
|
}
|
||||||
|
|
||||||
private ContactInfo() {
|
private ContactInfo() {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -151,10 +169,12 @@ public class ContactInfo {
|
||||||
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
|
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
|
||||||
boolean avatars = prefs.getBoolean("avatars", true);
|
boolean avatars = prefs.getBoolean("avatars", true);
|
||||||
boolean gravatars = prefs.getBoolean("gravatars", false);
|
boolean gravatars = prefs.getBoolean("gravatars", false);
|
||||||
|
boolean favicons = prefs.getBoolean("favicons", false);
|
||||||
boolean generated = prefs.getBoolean("generated_icons", true);
|
boolean generated = prefs.getBoolean("generated_icons", true);
|
||||||
boolean identicons = prefs.getBoolean("identicons", false);
|
boolean identicons = prefs.getBoolean("identicons", false);
|
||||||
boolean circular = prefs.getBoolean("circular", true);
|
boolean circular = prefs.getBoolean("circular", true);
|
||||||
|
|
||||||
|
// Contact photo
|
||||||
if (!TextUtils.isEmpty(info.email) &&
|
if (!TextUtils.isEmpty(info.email) &&
|
||||||
Helper.hasPermission(context, Manifest.permission.READ_CONTACTS)) {
|
Helper.hasPermission(context, Manifest.permission.READ_CONTACTS)) {
|
||||||
ContentResolver resolver = context.getContentResolver();
|
ContentResolver resolver = context.getContentResolver();
|
||||||
|
@ -195,6 +215,7 @@ public class ContactInfo {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Gravatar
|
||||||
if (info.bitmap == null) {
|
if (info.bitmap == null) {
|
||||||
if (gravatars && !TextUtils.isEmpty(info.email)) {
|
if (gravatars && !TextUtils.isEmpty(info.email)) {
|
||||||
String gkey = info.email.toLowerCase(Locale.ROOT);
|
String gkey = info.email.toLowerCase(Locale.ROOT);
|
||||||
|
@ -243,6 +264,78 @@ public class ContactInfo {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Favicon
|
||||||
|
if (info.bitmap == null) {
|
||||||
|
int at = (info.email == null ? -1 : info.email.indexOf('@'));
|
||||||
|
String domain = (at < 0 ? null : info.email.substring(at + 1).toLowerCase(Locale.ROOT));
|
||||||
|
synchronized (emailFaviconBlacklist) {
|
||||||
|
if (emailFaviconBlacklist.contains(domain)) {
|
||||||
|
Log.i("Favicon blacklisted domain=" + domain);
|
||||||
|
domain = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (favicons && domain != null) {
|
||||||
|
try {
|
||||||
|
File dir = new File(context.getCacheDir(), "favicons");
|
||||||
|
if (!dir.exists())
|
||||||
|
dir.mkdir();
|
||||||
|
File file = new File(dir, domain);
|
||||||
|
if (file.exists())
|
||||||
|
info.bitmap = BitmapFactory.decodeFile(file.getAbsolutePath());
|
||||||
|
else {
|
||||||
|
URL base = new URL("https://" + domain);
|
||||||
|
|
||||||
|
info.bitmap = getFavicon(new URL(base, "favicon.ico"));
|
||||||
|
if (info.bitmap == null) {
|
||||||
|
Log.i("GET " + base);
|
||||||
|
HttpsURLConnection connection = (HttpsURLConnection) base.openConnection();
|
||||||
|
connection.setRequestMethod("GET");
|
||||||
|
connection.setReadTimeout(FAVICON_TIMEOUT);
|
||||||
|
connection.setConnectTimeout(FAVICON_TIMEOUT);
|
||||||
|
connection.connect();
|
||||||
|
|
||||||
|
String response;
|
||||||
|
try {
|
||||||
|
response = Helper.readStream(connection.getInputStream(), StandardCharsets.UTF_8.name());
|
||||||
|
} finally {
|
||||||
|
connection.disconnect();
|
||||||
|
}
|
||||||
|
|
||||||
|
Document doc = JsoupEx.parse(response);
|
||||||
|
|
||||||
|
Element link = doc.head().select("link[href~=.*\\.(ico|png)]").first();
|
||||||
|
String favicon = (link == null ? null : link.attr("href"));
|
||||||
|
|
||||||
|
if (TextUtils.isEmpty(favicon)) {
|
||||||
|
Element meta = doc.head().select("meta[itemprop=image]").first();
|
||||||
|
favicon = (meta == null ? null : meta.attr("content"));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!TextUtils.isEmpty(favicon)) {
|
||||||
|
URL url = new URL(base, favicon);
|
||||||
|
if ("https".equals(url.getProtocol()))
|
||||||
|
info.bitmap = getFavicon(url);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (info.bitmap != null)
|
||||||
|
try (OutputStream os = new BufferedOutputStream(new FileOutputStream(file))) {
|
||||||
|
info.bitmap.compress(Bitmap.CompressFormat.PNG, 90, os);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (Throwable ex) {
|
||||||
|
Log.w(ex);
|
||||||
|
} finally {
|
||||||
|
if (info.bitmap == null)
|
||||||
|
synchronized (emailFaviconBlacklist) {
|
||||||
|
emailFaviconBlacklist.add(domain);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generated
|
||||||
boolean identicon = false;
|
boolean identicon = false;
|
||||||
if (info.bitmap == null) {
|
if (info.bitmap == null) {
|
||||||
int dp = Helper.dp2pixels(context, 96);
|
int dp = Helper.dp2pixels(context, 96);
|
||||||
|
@ -277,6 +370,29 @@ public class ContactInfo {
|
||||||
return info;
|
return info;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static Bitmap getFavicon(URL url) throws IOException {
|
||||||
|
try {
|
||||||
|
Log.i("GET favicon " + url);
|
||||||
|
|
||||||
|
HttpURLConnection connection = (HttpURLConnection) url.openConnection();
|
||||||
|
connection.setRequestMethod("GET");
|
||||||
|
connection.setReadTimeout(FAVICON_TIMEOUT);
|
||||||
|
connection.setConnectTimeout(FAVICON_TIMEOUT);
|
||||||
|
connection.connect();
|
||||||
|
|
||||||
|
try {
|
||||||
|
return BitmapFactory.decodeStream(connection.getInputStream());
|
||||||
|
} finally {
|
||||||
|
connection.disconnect();
|
||||||
|
}
|
||||||
|
} catch (IOException ex) {
|
||||||
|
Log.w(ex);
|
||||||
|
if (ex instanceof SocketTimeoutException)
|
||||||
|
throw ex;
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void init(final Context context) {
|
static void init(final Context context) {
|
||||||
if (Helper.hasPermission(context, Manifest.permission.READ_CONTACTS)) {
|
if (Helper.hasPermission(context, Manifest.permission.READ_CONTACTS)) {
|
||||||
Handler handler = new Handler(Looper.getMainLooper());
|
Handler handler = new Handler(Looper.getMainLooper());
|
||||||
|
|
|
@ -80,7 +80,7 @@ public class FragmentOptions extends FragmentBase {
|
||||||
"subscriptions",
|
"subscriptions",
|
||||||
"landscape", "landscape3", "startup", "cards", "indentation", "date", "threading", "threading_unread",
|
"landscape", "landscape3", "startup", "cards", "indentation", "date", "threading", "threading_unread",
|
||||||
"highlight_unread", "color_stripe",
|
"highlight_unread", "color_stripe",
|
||||||
"avatars", "gravatars", "generated_icons", "identicons", "circular", "saturation", "brightness", "threshold",
|
"avatars", "gravatars", "favicons", "generated_icons", "identicons", "circular", "saturation", "brightness", "threshold",
|
||||||
"name_email", "prefer_contact", "distinguish_contacts", "show_recipients", "authentication",
|
"name_email", "prefer_contact", "distinguish_contacts", "show_recipients", "authentication",
|
||||||
"subject_top", "font_size_sender", "font_size_subject", "subject_italic", "highlight_subject", "subject_ellipsize",
|
"subject_top", "font_size_sender", "font_size_subject", "subject_italic", "highlight_subject", "subject_ellipsize",
|
||||||
"keywords_header", "labels_header", "flags", "flags_background", "preview", "preview_italic", "preview_lines",
|
"keywords_header", "labels_header", "flags", "flags_background", "preview", "preview_italic", "preview_lines",
|
||||||
|
|
|
@ -75,6 +75,7 @@ public class FragmentOptionsDisplay extends FragmentBase implements SharedPrefer
|
||||||
private SwitchCompat swAvatars;
|
private SwitchCompat swAvatars;
|
||||||
private TextView tvGravatarsHint;
|
private TextView tvGravatarsHint;
|
||||||
private SwitchCompat swGravatars;
|
private SwitchCompat swGravatars;
|
||||||
|
private SwitchCompat swFavicons;
|
||||||
private SwitchCompat swGeneratedIcons;
|
private SwitchCompat swGeneratedIcons;
|
||||||
private SwitchCompat swIdenticons;
|
private SwitchCompat swIdenticons;
|
||||||
private SwitchCompat swCircular;
|
private SwitchCompat swCircular;
|
||||||
|
@ -122,7 +123,7 @@ public class FragmentOptionsDisplay extends FragmentBase implements SharedPrefer
|
||||||
"theme", "startup", "cards", "date", "navbar_colorize", "landscape", "landscape3",
|
"theme", "startup", "cards", "date", "navbar_colorize", "landscape", "landscape3",
|
||||||
"threading", "threading_unread", "indentation", "seekbar", "actionbar", "actionbar_color",
|
"threading", "threading_unread", "indentation", "seekbar", "actionbar", "actionbar_color",
|
||||||
"highlight_unread", "color_stripe",
|
"highlight_unread", "color_stripe",
|
||||||
"avatars", "gravatars", "generated_icons", "identicons", "circular", "saturation", "brightness", "threshold",
|
"avatars", "gravatars", "favicons", "generated_icons", "identicons", "circular", "saturation", "brightness", "threshold",
|
||||||
"name_email", "prefer_contact", "distinguish_contacts", "show_recipients",
|
"name_email", "prefer_contact", "distinguish_contacts", "show_recipients",
|
||||||
"subject_top", "font_size_sender", "font_size_subject", "subject_italic", "highlight_subject", "subject_ellipsize",
|
"subject_top", "font_size_sender", "font_size_subject", "subject_italic", "highlight_subject", "subject_ellipsize",
|
||||||
"keywords_header", "labels_header", "flags", "flags_background",
|
"keywords_header", "labels_header", "flags", "flags_background",
|
||||||
|
@ -163,6 +164,7 @@ public class FragmentOptionsDisplay extends FragmentBase implements SharedPrefer
|
||||||
swAvatars = view.findViewById(R.id.swAvatars);
|
swAvatars = view.findViewById(R.id.swAvatars);
|
||||||
swGravatars = view.findViewById(R.id.swGravatars);
|
swGravatars = view.findViewById(R.id.swGravatars);
|
||||||
tvGravatarsHint = view.findViewById(R.id.tvGravatarsHint);
|
tvGravatarsHint = view.findViewById(R.id.tvGravatarsHint);
|
||||||
|
swFavicons = view.findViewById(R.id.swFavicons);
|
||||||
swGeneratedIcons = view.findViewById(R.id.swGeneratedIcons);
|
swGeneratedIcons = view.findViewById(R.id.swGeneratedIcons);
|
||||||
swIdenticons = view.findViewById(R.id.swIdenticons);
|
swIdenticons = view.findViewById(R.id.swIdenticons);
|
||||||
swCircular = view.findViewById(R.id.swCircular);
|
swCircular = view.findViewById(R.id.swCircular);
|
||||||
|
@ -344,6 +346,14 @@ public class FragmentOptionsDisplay extends FragmentBase implements SharedPrefer
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
swFavicons.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
|
||||||
|
@Override
|
||||||
|
public void onCheckedChanged(CompoundButton compoundButton, boolean checked) {
|
||||||
|
prefs.edit().putBoolean("favicons", checked).apply();
|
||||||
|
ContactInfo.clearCache();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
tvGravatarsHint.getPaint().setUnderlineText(true);
|
tvGravatarsHint.getPaint().setUnderlineText(true);
|
||||||
tvGravatarsHint.setOnClickListener(new View.OnClickListener() {
|
tvGravatarsHint.setOnClickListener(new View.OnClickListener() {
|
||||||
@Override
|
@Override
|
||||||
|
@ -757,6 +767,7 @@ public class FragmentOptionsDisplay extends FragmentBase implements SharedPrefer
|
||||||
swColorStripe.setChecked(prefs.getBoolean("color_stripe", true));
|
swColorStripe.setChecked(prefs.getBoolean("color_stripe", true));
|
||||||
swAvatars.setChecked(prefs.getBoolean("avatars", true));
|
swAvatars.setChecked(prefs.getBoolean("avatars", true));
|
||||||
swGravatars.setChecked(prefs.getBoolean("gravatars", false));
|
swGravatars.setChecked(prefs.getBoolean("gravatars", false));
|
||||||
|
swFavicons.setChecked(prefs.getBoolean("favicons", false));
|
||||||
swGeneratedIcons.setChecked(prefs.getBoolean("generated_icons", true));
|
swGeneratedIcons.setChecked(prefs.getBoolean("generated_icons", true));
|
||||||
swIdenticons.setChecked(prefs.getBoolean("identicons", false));
|
swIdenticons.setChecked(prefs.getBoolean("identicons", false));
|
||||||
swIdenticons.setEnabled(swGeneratedIcons.isChecked());
|
swIdenticons.setEnabled(swGeneratedIcons.isChecked());
|
||||||
|
|
|
@ -349,6 +349,29 @@
|
||||||
app:layout_constraintStart_toStartOf="parent"
|
app:layout_constraintStart_toStartOf="parent"
|
||||||
app:layout_constraintTop_toBottomOf="@id/swGravatars" />
|
app:layout_constraintTop_toBottomOf="@id/swGravatars" />
|
||||||
|
|
||||||
|
<androidx.appcompat.widget.SwitchCompat
|
||||||
|
android:id="@+id/swFavicons"
|
||||||
|
android:layout_width="0dp"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_marginTop="12dp"
|
||||||
|
android:text="@string/title_advanced_favicons"
|
||||||
|
app:layout_constraintEnd_toEndOf="parent"
|
||||||
|
app:layout_constraintStart_toStartOf="parent"
|
||||||
|
app:layout_constraintTop_toBottomOf="@id/tvGravatarsHint"
|
||||||
|
app:switchPadding="12dp" />
|
||||||
|
|
||||||
|
<eu.faircode.email.FixedTextView
|
||||||
|
android:id="@+id/tvFaviconsHint"
|
||||||
|
android:layout_width="0dp"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_marginEnd="48dp"
|
||||||
|
android:text="@string/title_advanced_gravatars_hint"
|
||||||
|
android:textAppearance="@style/TextAppearance.AppCompat.Small"
|
||||||
|
android:textColor="?attr/colorWarning"
|
||||||
|
app:layout_constraintEnd_toEndOf="parent"
|
||||||
|
app:layout_constraintStart_toStartOf="parent"
|
||||||
|
app:layout_constraintTop_toBottomOf="@id/swFavicons" />
|
||||||
|
|
||||||
<androidx.appcompat.widget.SwitchCompat
|
<androidx.appcompat.widget.SwitchCompat
|
||||||
android:id="@+id/swGeneratedIcons"
|
android:id="@+id/swGeneratedIcons"
|
||||||
android:layout_width="0dp"
|
android:layout_width="0dp"
|
||||||
|
@ -358,7 +381,7 @@
|
||||||
android:text="@string/title_advanced_generated_icons"
|
android:text="@string/title_advanced_generated_icons"
|
||||||
app:layout_constraintEnd_toEndOf="parent"
|
app:layout_constraintEnd_toEndOf="parent"
|
||||||
app:layout_constraintStart_toStartOf="parent"
|
app:layout_constraintStart_toStartOf="parent"
|
||||||
app:layout_constraintTop_toBottomOf="@id/tvGravatarsHint"
|
app:layout_constraintTop_toBottomOf="@id/tvFaviconsHint"
|
||||||
app:switchPadding="12dp" />
|
app:switchPadding="12dp" />
|
||||||
|
|
||||||
<androidx.appcompat.widget.SwitchCompat
|
<androidx.appcompat.widget.SwitchCompat
|
||||||
|
|
|
@ -331,6 +331,7 @@
|
||||||
<string name="title_advanced_color_stripe">Show color stripe</string>
|
<string name="title_advanced_color_stripe">Show color stripe</string>
|
||||||
<string name="title_advanced_avatars">Show contact photos</string>
|
<string name="title_advanced_avatars">Show contact photos</string>
|
||||||
<string name="title_advanced_gravatars">Show Gravatars</string>
|
<string name="title_advanced_gravatars">Show Gravatars</string>
|
||||||
|
<string name="title_advanced_favicons">Show favicons</string>
|
||||||
<string name="title_advanced_generated_icons">Show generated icons</string>
|
<string name="title_advanced_generated_icons">Show generated icons</string>
|
||||||
<string name="title_advanced_identicons">Show identicons</string>
|
<string name="title_advanced_identicons">Show identicons</string>
|
||||||
<string name="title_advanced_circular">Show round icons</string>
|
<string name="title_advanced_circular">Show round icons</string>
|
||||||
|
|
Loading…
Reference in New Issue