/* * Copyright (C) 2021 The Android Open Source Project * * Licensed 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 android.app; import android.annotation.FlaggedApi; import android.annotation.IntDef; import android.annotation.NonNull; import android.annotation.Nullable; import android.annotation.SuppressLint; import android.content.Context; import android.content.res.Resources; import android.content.res.TypedArray; import android.content.res.XmlResourceParser; import android.os.LocaleList; import android.os.Parcel; import android.os.Parcelable; import android.ravenwood.annotation.RavenwoodKeepWholeClass; import android.ravenwood.annotation.RavenwoodThrow; import android.util.AttributeSet; import android.util.Slog; import android.util.Xml; import com.android.internal.R; import com.android.internal.util.XmlUtils; import org.xmlpull.v1.XmlPullParserException; import java.io.IOException; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.util.Arrays; import java.util.Collections; import java.util.HashSet; import java.util.List; import java.util.Locale; import java.util.Set; /** * The LocaleConfig of an application. * There are two sources. One is from an XML resource file with an {@code } element * and referenced in the manifest via {@code android:localeConfig} on {@code }. The * other is that the application dynamically provides an override version which is persisted in * {@link LocaleManager#setOverrideLocaleConfig(LocaleConfig)}. * *

For more information about the LocaleConfig from an XML resource file, see * * the section on per-app language preferences. * * @attr ref android.R.styleable#LocaleConfig_Locale_name * @attr ref android.R.styleable#AndroidManifestApplication_localeConfig */ // Add following to last Note: when guide is written: // For more information about the LocaleConfig overridden by the application, see TODO(b/261528306): // add link to guide @RavenwoodKeepWholeClass public class LocaleConfig implements Parcelable { private static final String TAG = "LocaleConfig"; public static final String TAG_LOCALE_CONFIG = "locale-config"; public static final String TAG_LOCALE = "locale"; private LocaleList mLocales; private Locale mDefaultLocale; private int mStatus = STATUS_NOT_SPECIFIED; /** * succeeded reading the LocaleConfig structure stored in an XML file. */ public static final int STATUS_SUCCESS = 0; /** * No android:localeConfig tag on . */ public static final int STATUS_NOT_SPECIFIED = 1; /** * Malformed input in the XML file where the LocaleConfig was stored. */ public static final int STATUS_PARSING_FAILED = 2; /** @hide */ @IntDef(prefix = { "STATUS_" }, value = { STATUS_SUCCESS, STATUS_NOT_SPECIFIED, STATUS_PARSING_FAILED }) @Retention(RetentionPolicy.SOURCE) public @interface Status{} /** * Returns an override LocaleConfig if it has been set via * {@link LocaleManager#setOverrideLocaleConfig(LocaleConfig)}. Otherwise, returns the * LocaleConfig from the application resources. * * @param context the context of the application. * * @see Context#createPackageContext(String, int). */ @RavenwoodThrow(blockedBy = LocaleManager.class) public LocaleConfig(@NonNull Context context) { this(context, true); } /** * Returns a LocaleConfig from the application resources regardless of whether any LocaleConfig * is overridden via {@link LocaleManager#setOverrideLocaleConfig(LocaleConfig)}. * * @param context the context of the application. * * @see Context#createPackageContext(String, int). */ @NonNull @RavenwoodThrow(blockedBy = LocaleManager.class) public static LocaleConfig fromContextIgnoringOverride(@NonNull Context context) { return new LocaleConfig(context, false); } @RavenwoodThrow(blockedBy = LocaleManager.class) private LocaleConfig(@NonNull Context context, boolean allowOverride) { if (allowOverride) { LocaleManager localeManager = context.getSystemService(LocaleManager.class); if (localeManager == null) { Slog.w(TAG, "LocaleManager is null, cannot get the override LocaleConfig"); mStatus = STATUS_NOT_SPECIFIED; return; } LocaleConfig localeConfig = localeManager.getOverrideLocaleConfig(); if (localeConfig != null) { Slog.d(TAG, "Has the override LocaleConfig"); mStatus = localeConfig.getStatus(); mLocales = localeConfig.getSupportedLocales(); return; } } Resources res = context.getResources(); int resId = context.getApplicationInfo().getLocaleConfigRes(); if (resId == 0) { mStatus = STATUS_NOT_SPECIFIED; return; } try (XmlResourceParser parser = res.getXml(resId)) { parseLocaleConfig(parser, res); } catch (Resources.NotFoundException e) { Slog.w(TAG, "The resource file pointed to by the given resource ID isn't found."); mStatus = STATUS_NOT_SPECIFIED; } catch (XmlPullParserException | IOException e) { Slog.w(TAG, "Failed to parse XML configuration from " + res.getResourceEntryName(resId), e); mStatus = STATUS_PARSING_FAILED; } } /** * Return the LocaleConfig with any sequence of locales combined into a {@link LocaleList}. * *

Note: Applications seeking to create an override LocaleConfig via * {@link LocaleManager#setOverrideLocaleConfig(LocaleConfig)} should use this constructor to * first create the LocaleConfig they intend the system to see as the override. * *

Note: The creation of this LocaleConfig does not automatically mean it will * become the override config for an application. Any LocaleConfig desired to be the override * must be passed into the {@link LocaleManager#setOverrideLocaleConfig(LocaleConfig)}, * otherwise it will not persist or affect the system's understanding of app-supported * resources. * * @param locales the desired locales for a specified application */ public LocaleConfig(@NonNull LocaleList locales) { mStatus = STATUS_SUCCESS; mLocales = locales; } /** * Instantiate a new LocaleConfig from the data in a Parcel that was * previously written with {@link #writeToParcel(Parcel, int)}. * * @param in The Parcel containing the previously written LocaleConfig, * positioned at the location in the buffer where it was written. */ private LocaleConfig(@NonNull Parcel in) { mStatus = in.readInt(); mLocales = in.readTypedObject(LocaleList.CREATOR); } /** * Parse the XML content and get the locales supported by the application */ private void parseLocaleConfig(XmlResourceParser parser, Resources res) throws IOException, XmlPullParserException { XmlUtils.beginDocument(parser, TAG_LOCALE_CONFIG); int outerDepth = parser.getDepth(); AttributeSet attrs = Xml.asAttributeSet(parser); String defaultLocale = null; if (android.content.res.Flags.defaultLocale()) { // Read the defaultLocale attribute of the LocaleConfig element try (TypedArray att = res.obtainAttributes( attrs, com.android.internal.R.styleable.LocaleConfig)) { defaultLocale = att.getString( R.styleable.LocaleConfig_defaultLocale); } } Set localeNames = new HashSet<>(); while (XmlUtils.nextElementWithin(parser, outerDepth)) { if (TAG_LOCALE.equals(parser.getName())) { try (TypedArray attributes = res.obtainAttributes( attrs, com.android.internal.R.styleable.LocaleConfig_Locale)) { String nameAttr = attributes.getString( com.android.internal.R.styleable.LocaleConfig_Locale_name); localeNames.add(nameAttr); } } else { XmlUtils.skipCurrentTag(parser); } } mStatus = STATUS_SUCCESS; mLocales = LocaleList.forLanguageTags(String.join(",", localeNames)); if (defaultLocale != null) { if (localeNames.contains(defaultLocale)) { mDefaultLocale = Locale.forLanguageTag(defaultLocale); } else { Slog.w(TAG, "Default locale specified that is not contained in the list: " + defaultLocale); mStatus = STATUS_PARSING_FAILED; } } } /** * Returns the locales supported by the specified application. * *

Note: The locale format should follow the * IETF BCP47 regular expression * * @return the {@link LocaleList} */ public @Nullable LocaleList getSupportedLocales() { return mLocales; } /** * Returns the locale the strings in values/strings.xml (the default strings in the directory * with no locale qualifier) are in if specified, otherwise null * * @return The default Locale or null */ @SuppressLint("UseIcu") @FlaggedApi(android.content.res.Flags.FLAG_DEFAULT_LOCALE) public @Nullable Locale getDefaultLocale() { return mDefaultLocale; } /** * Get the status of reading the resource file where the LocaleConfig was stored. * *

Distinguish "the application didn't provide the resource file" from "the application * provided malformed input" if {@link #getSupportedLocales()} returns {@code null}. * * @return {@code STATUS_SUCCESS} if the LocaleConfig structure existed in an XML file was * successfully read, or {@code STATUS_NOT_SPECIFIED} if no android:localeConfig tag on * pointing to an XML file that stores the LocaleConfig, or * {@code STATUS_PARSING_FAILED} if the application provided malformed input for the * LocaleConfig structure. * * @see #STATUS_SUCCESS * @see #STATUS_NOT_SPECIFIED * @see #STATUS_PARSING_FAILED * */ public @Status int getStatus() { return mStatus; } @Override public int describeContents() { return 0; } @Override public void writeToParcel(@NonNull Parcel dest, int flags) { dest.writeInt(mStatus); dest.writeTypedObject(mLocales, flags); } public static final @NonNull Parcelable.Creator CREATOR = new Parcelable.Creator() { @Override public LocaleConfig createFromParcel(Parcel source) { return new LocaleConfig(source); } @Override public LocaleConfig[] newArray(int size) { return new LocaleConfig[size]; } }; /** * Compare whether the LocaleConfig is the same. * *

If the elements of {@code mLocales} in LocaleConfig are the same but arranged in different * positions, they are also considered to be the same LocaleConfig. * * @param other The {@link LocaleConfig} to compare for. * * @return true if the LocaleConfig is the same, false otherwise. * * @hide */ public boolean isSameLocaleConfig(@Nullable LocaleConfig other) { if (other == this) { return true; } if (other != null) { if (mStatus != other.mStatus) { return false; } LocaleList otherLocales = other.mLocales; if (mLocales == null && otherLocales == null) { return true; } else if (mLocales != null && otherLocales != null) { List hostStrList = Arrays.asList(mLocales.toLanguageTags().split(",")); List targetStrList = Arrays.asList( otherLocales.toLanguageTags().split(",")); Collections.sort(hostStrList); Collections.sort(targetStrList); return hostStrList.equals(targetStrList); } } return false; } /** * Compare whether the locale is existed in the {@code mLocales} of the LocaleConfig. * * @param locale The {@link Locale} to compare for. * * @return true if the locale is existed in the {@code mLocales} of the LocaleConfig, false * otherwise. * * @hide */ public boolean containsLocale(Locale locale) { if (mLocales == null) { return false; } for (int i = 0; i < mLocales.size(); i++) { if (LocaleList.matchesLanguageAndScript(mLocales.get(i), locale)) { return true; } } return false; } }