RecyclerView in LayoutLib: better XML attrs.
- RecyclerView now supports XML attributes natively. Thus, remove the custom support via tools attribute. Users with older versions of RecyclerView should update. - Add Context.getPackageName() support used by RecyclerView. - Update SessionParamsFlags with the new changes and rename it to RenderParamsFlags. The attribute behaves slightly different from the original tools attribute. For usage, see commit 044b5b61e96 in frameworks/support. Change-Id: I12073e37a2ba411558ca1d3e30c399e3d9a0b144
This commit is contained in:
@ -16,19 +16,15 @@
|
|||||||
|
|
||||||
package android.view;
|
package android.view;
|
||||||
|
|
||||||
import com.android.ide.common.rendering.api.LayoutlibCallback;
|
|
||||||
import com.android.ide.common.rendering.api.LayoutLog;
|
import com.android.ide.common.rendering.api.LayoutLog;
|
||||||
|
import com.android.ide.common.rendering.api.LayoutlibCallback;
|
||||||
import com.android.ide.common.rendering.api.MergeCookie;
|
import com.android.ide.common.rendering.api.MergeCookie;
|
||||||
import com.android.ide.common.rendering.api.ResourceReference;
|
import com.android.ide.common.rendering.api.ResourceReference;
|
||||||
import com.android.ide.common.rendering.api.ResourceValue;
|
import com.android.ide.common.rendering.api.ResourceValue;
|
||||||
import com.android.layoutlib.bridge.Bridge;
|
import com.android.layoutlib.bridge.Bridge;
|
||||||
import com.android.layoutlib.bridge.BridgeConstants;
|
|
||||||
import com.android.layoutlib.bridge.android.BridgeContext;
|
import com.android.layoutlib.bridge.android.BridgeContext;
|
||||||
import com.android.layoutlib.bridge.android.BridgeXmlBlockParser;
|
import com.android.layoutlib.bridge.android.BridgeXmlBlockParser;
|
||||||
import com.android.layoutlib.bridge.android.support.RecyclerViewUtil;
|
|
||||||
import com.android.layoutlib.bridge.android.support.RecyclerViewUtil.LayoutManagerType;
|
|
||||||
import com.android.layoutlib.bridge.impl.ParserFactory;
|
import com.android.layoutlib.bridge.impl.ParserFactory;
|
||||||
import com.android.layoutlib.bridge.impl.RenderSessionImpl;
|
|
||||||
import com.android.resources.ResourceType;
|
import com.android.resources.ResourceType;
|
||||||
import com.android.util.Pair;
|
import com.android.util.Pair;
|
||||||
|
|
||||||
@ -233,22 +229,6 @@ public final class BridgeInflater extends LayoutInflater {
|
|||||||
if (viewKey != null) {
|
if (viewKey != null) {
|
||||||
bc.addViewKey(view, viewKey);
|
bc.addViewKey(view, viewKey);
|
||||||
}
|
}
|
||||||
if (RenderSessionImpl.isInstanceOf(view, RecyclerViewUtil.CN_RECYCLER_VIEW)) {
|
|
||||||
String type = attrs.getAttributeValue(BridgeConstants.NS_RESOURCES,
|
|
||||||
BridgeConstants.ATTR_LAYOUT_MANAGER_TYPE);
|
|
||||||
if (type != null) {
|
|
||||||
LayoutManagerType layoutManagerType = LayoutManagerType.getByLogicalName(type);
|
|
||||||
if (layoutManagerType == null) {
|
|
||||||
layoutManagerType = LayoutManagerType.getByClassName(type);
|
|
||||||
}
|
|
||||||
if (layoutManagerType == null) {
|
|
||||||
// add the classname itself.
|
|
||||||
bc.addCookie(view, type);
|
|
||||||
} else {
|
|
||||||
bc.addCookie(view, layoutManagerType);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -92,6 +92,8 @@ import java.util.IdentityHashMap;
|
|||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
|
import static com.android.layoutlib.bridge.android.RenderParamsFlags.FLAG_KEY_APPLICATION_PACKAGE;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Custom implementation of Context/Activity to handle non compiled resources.
|
* Custom implementation of Context/Activity to handle non compiled resources.
|
||||||
*/
|
*/
|
||||||
@ -305,7 +307,7 @@ public final class BridgeContext extends Context {
|
|||||||
// check if this is a style resource
|
// check if this is a style resource
|
||||||
if (value instanceof StyleResourceValue) {
|
if (value instanceof StyleResourceValue) {
|
||||||
// get the id that will represent this style.
|
// get the id that will represent this style.
|
||||||
outValue.resourceId = getDynamicIdByStyle((StyleResourceValue)value);
|
outValue.resourceId = getDynamicIdByStyle((StyleResourceValue) value);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -783,6 +785,14 @@ public final class BridgeContext extends Context {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getPackageName() {
|
||||||
|
if (mApplicationInfo.packageName == null) {
|
||||||
|
mApplicationInfo.packageName = mLayoutlibCallback.getFlag(FLAG_KEY_APPLICATION_PACKAGE);
|
||||||
|
}
|
||||||
|
return mApplicationInfo.packageName;
|
||||||
|
}
|
||||||
|
|
||||||
// ------------- private new methods
|
// ------------- private new methods
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -1189,12 +1199,6 @@ public final class BridgeContext extends Context {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public String getPackageName() {
|
|
||||||
// pass
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getBasePackageName() {
|
public String getBasePackageName() {
|
||||||
// pass
|
// pass
|
||||||
|
@ -0,0 +1,53 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2014 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 com.android.layoutlib.bridge.android;
|
||||||
|
|
||||||
|
import com.android.ide.common.rendering.api.RenderParams;
|
||||||
|
import com.android.ide.common.rendering.api.SessionParams.Key;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This contains all known keys for the {@link RenderParams#getFlag(Key)}.
|
||||||
|
* <p/>
|
||||||
|
* The IDE has its own copy of this class which may be newer or older than this one.
|
||||||
|
* <p/>
|
||||||
|
* Constants should never be modified or removed from this class.
|
||||||
|
*/
|
||||||
|
public final class RenderParamsFlags {
|
||||||
|
|
||||||
|
public static final Key<String> FLAG_KEY_ROOT_TAG =
|
||||||
|
new Key<String>("rootTag", String.class);
|
||||||
|
public static final Key<Boolean> FLAG_KEY_DISABLE_BITMAP_CACHING =
|
||||||
|
new Key<Boolean>("disableBitmapCaching", Boolean.class);
|
||||||
|
public static final Key<Boolean> FLAG_KEY_RENDER_ALL_DRAWABLE_STATES =
|
||||||
|
new Key<Boolean>("renderAllDrawableStates", Boolean.class);
|
||||||
|
/**
|
||||||
|
* To tell LayoutLib that the IDE supports RecyclerView.
|
||||||
|
* <p/>
|
||||||
|
* Default is false.
|
||||||
|
*/
|
||||||
|
public static final Key<Boolean> FLAG_KEY_RECYCLER_VIEW_SUPPORT =
|
||||||
|
new Key<Boolean>("recyclerViewSupport", Boolean.class);
|
||||||
|
/**
|
||||||
|
* The application package name. Used via
|
||||||
|
* {@link com.android.ide.common.rendering.api.LayoutlibCallback#getFlag(Key)}
|
||||||
|
*/
|
||||||
|
public static final Key<String> FLAG_KEY_APPLICATION_PACKAGE =
|
||||||
|
new Key<String>("applicationPackage", String.class);
|
||||||
|
|
||||||
|
// Disallow instances.
|
||||||
|
private RenderParamsFlags() {}
|
||||||
|
}
|
@ -1,37 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (C) 2014 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 com.android.layoutlib.bridge.android;
|
|
||||||
|
|
||||||
import com.android.ide.common.rendering.api.SessionParams;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This contains all known keys for the {@link SessionParams#getFlag(SessionParams.Key)}.
|
|
||||||
* <p/>
|
|
||||||
* The IDE has its own copy of this class which may be newer or older than this one.
|
|
||||||
* <p/>
|
|
||||||
* Constants should never be modified or removed from this class.
|
|
||||||
*/
|
|
||||||
public final class SessionParamsFlags {
|
|
||||||
|
|
||||||
public static final SessionParams.Key<String> FLAG_KEY_ROOT_TAG =
|
|
||||||
new SessionParams.Key<String>("rootTag", String.class);
|
|
||||||
public static final SessionParams.Key<Boolean> FLAG_KEY_RECYCLER_VIEW_SUPPORT =
|
|
||||||
new SessionParams.Key<Boolean>("recyclerViewSupport", Boolean.class);
|
|
||||||
|
|
||||||
// Disallow instances.
|
|
||||||
private SessionParamsFlags() {}
|
|
||||||
}
|
|
@ -23,15 +23,16 @@ import com.android.ide.common.rendering.api.LayoutlibCallback;
|
|||||||
import com.android.ide.common.rendering.api.SessionParams;
|
import com.android.ide.common.rendering.api.SessionParams;
|
||||||
import com.android.layoutlib.bridge.Bridge;
|
import com.android.layoutlib.bridge.Bridge;
|
||||||
import com.android.layoutlib.bridge.android.BridgeContext;
|
import com.android.layoutlib.bridge.android.BridgeContext;
|
||||||
import com.android.layoutlib.bridge.android.SessionParamsFlags;
|
import com.android.layoutlib.bridge.android.RenderParamsFlags;
|
||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
import android.widget.LinearLayout;
|
|
||||||
|
|
||||||
import java.lang.reflect.Method;
|
import java.lang.reflect.Method;
|
||||||
|
|
||||||
import static com.android.layoutlib.bridge.util.ReflectionUtils.*;
|
import static com.android.layoutlib.bridge.util.ReflectionUtils.ReflectionException;
|
||||||
|
import static com.android.layoutlib.bridge.util.ReflectionUtils.getMethod;
|
||||||
|
import static com.android.layoutlib.bridge.util.ReflectionUtils.invoke;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Utility class for working with android.support.v7.widget.RecyclerView
|
* Utility class for working with android.support.v7.widget.RecyclerView
|
||||||
@ -39,17 +40,15 @@ import static com.android.layoutlib.bridge.util.ReflectionUtils.*;
|
|||||||
@SuppressWarnings("SpellCheckingInspection") // for "recycler".
|
@SuppressWarnings("SpellCheckingInspection") // for "recycler".
|
||||||
public class RecyclerViewUtil {
|
public class RecyclerViewUtil {
|
||||||
|
|
||||||
/**
|
private static final String RV_PKG_PREFIX = "android.support.v7.widget.";
|
||||||
* Used by {@link LayoutManagerType}.
|
public static final String CN_RECYCLER_VIEW = RV_PKG_PREFIX + "RecyclerView";
|
||||||
* <p/>
|
|
||||||
* Not declared inside the enum, since it needs to be accessible in the constructor.
|
|
||||||
*/
|
|
||||||
private static final Object CONTEXT = new Object();
|
|
||||||
|
|
||||||
public static final String CN_RECYCLER_VIEW = "android.support.v7.widget.RecyclerView";
|
|
||||||
private static final String CN_LAYOUT_MANAGER = CN_RECYCLER_VIEW + "$LayoutManager";
|
private static final String CN_LAYOUT_MANAGER = CN_RECYCLER_VIEW + "$LayoutManager";
|
||||||
private static final String CN_ADAPTER = CN_RECYCLER_VIEW + "$Adapter";
|
private static final String CN_ADAPTER = CN_RECYCLER_VIEW + "$Adapter";
|
||||||
|
|
||||||
|
// LinearLayoutManager related constants.
|
||||||
|
private static final String CN_LINEAR_LAYOUT_MANAGER = RV_PKG_PREFIX + "LinearLayoutManager";
|
||||||
|
private static final Class<?>[] LLM_CONSTRUCTOR_SIGNATURE = new Class<?>[]{Context.class};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tries to create an Adapter ({@code android.support.v7.widget.RecyclerView.Adapter} and a
|
* Tries to create an Adapter ({@code android.support.v7.widget.RecyclerView.Adapter} and a
|
||||||
* LayoutManager {@code RecyclerView.LayoutManager} and assign these to the {@code RecyclerView}
|
* LayoutManager {@code RecyclerView.LayoutManager} and assign these to the {@code RecyclerView}
|
||||||
@ -71,39 +70,35 @@ public class RecyclerViewUtil {
|
|||||||
|
|
||||||
private static void setLayoutManager(@NonNull View recyclerView, @NonNull BridgeContext context,
|
private static void setLayoutManager(@NonNull View recyclerView, @NonNull BridgeContext context,
|
||||||
@NonNull LayoutlibCallback callback) throws ReflectionException {
|
@NonNull LayoutlibCallback callback) throws ReflectionException {
|
||||||
Object cookie = context.getCookie(recyclerView);
|
if (getLayoutManager(recyclerView) == null) {
|
||||||
assert cookie == null || cookie instanceof LayoutManagerType || cookie instanceof String;
|
// Only set the layout manager if not already set by the recycler view.
|
||||||
if (!(cookie instanceof LayoutManagerType)) {
|
Object layoutManager = createLayoutManager(context, callback);
|
||||||
if (cookie != null) {
|
setProperty(recyclerView, CN_LAYOUT_MANAGER, layoutManager, "setLayoutManager");
|
||||||
// TODO: When layoutlib API is updated, try to load the class with a null
|
|
||||||
// constructor or a constructor taking one argument - the context.
|
|
||||||
Bridge.getLog().warning(LayoutLog.TAG_UNSUPPORTED,
|
|
||||||
"LayoutManager (" + cookie + ") not found, falling back to " +
|
|
||||||
"LinearLayoutManager", null);
|
|
||||||
}
|
|
||||||
cookie = LayoutManagerType.getDefault();
|
|
||||||
}
|
}
|
||||||
Object layoutManager = createLayoutManager((LayoutManagerType) cookie, context, callback);
|
|
||||||
setProperty(recyclerView, CN_LAYOUT_MANAGER, layoutManager, "setLayoutManager");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Creates a LinearLayoutManager using the provided context. */
|
||||||
@Nullable
|
@Nullable
|
||||||
private static Object createLayoutManager(@Nullable LayoutManagerType type,
|
private static Object createLayoutManager(@NonNull Context context,
|
||||||
@NonNull Context context, @NonNull LayoutlibCallback callback)
|
@NonNull LayoutlibCallback callback)
|
||||||
throws ReflectionException {
|
throws ReflectionException {
|
||||||
if (type == null) {
|
|
||||||
type = LayoutManagerType.getDefault();
|
|
||||||
}
|
|
||||||
try {
|
try {
|
||||||
return callback.loadView(type.getClassName(), type.getSignature(), type.getArgs(context));
|
return callback.loadView(CN_LINEAR_LAYOUT_MANAGER, LLM_CONSTRUCTOR_SIGNATURE,
|
||||||
|
new Object[]{ context});
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw new ReflectionException(e);
|
throw new ReflectionException(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
private static Object getLayoutManager(View recyclerview) throws ReflectionException {
|
||||||
|
Method getLayoutManager = getMethod(recyclerview.getClass(), "getLayoutManager");
|
||||||
|
return getLayoutManager != null ? invoke(getLayoutManager, recyclerview) : null;
|
||||||
|
}
|
||||||
|
|
||||||
@Nullable
|
@Nullable
|
||||||
private static Object createAdapter(@NonNull SessionParams params) throws ReflectionException {
|
private static Object createAdapter(@NonNull SessionParams params) throws ReflectionException {
|
||||||
Boolean ideSupport = params.getFlag(SessionParamsFlags.FLAG_KEY_RECYCLER_VIEW_SUPPORT);
|
Boolean ideSupport = params.getFlag(RenderParamsFlags.FLAG_KEY_RECYCLER_VIEW_SUPPORT);
|
||||||
if (ideSupport != Boolean.TRUE) {
|
if (ideSupport != Boolean.TRUE) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@ -145,74 +140,4 @@ public class RecyclerViewUtil {
|
|||||||
}
|
}
|
||||||
throw new RuntimeException("invalid object/classname combination.");
|
throw new RuntimeException("invalid object/classname combination.");
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Supported LayoutManagers. */
|
|
||||||
public enum LayoutManagerType {
|
|
||||||
LINEAR_LAYOUT_MANGER("Linear",
|
|
||||||
"android.support.v7.widget.LinearLayoutManager",
|
|
||||||
new Class[]{Context.class}, new Object[]{CONTEXT}),
|
|
||||||
GRID_LAYOUT_MANAGER("Grid",
|
|
||||||
"android.support.v7.widget.GridLayoutManager",
|
|
||||||
new Class[]{Context.class, int.class}, new Object[]{CONTEXT, 2}),
|
|
||||||
STAGGERED_GRID_LAYOUT_MANAGER("StaggeredGrid",
|
|
||||||
"android.support.v7.widget.StaggeredGridLayoutManager",
|
|
||||||
new Class[]{int.class, int.class}, new Object[]{2, LinearLayout.VERTICAL});
|
|
||||||
|
|
||||||
private String mLogicalName;
|
|
||||||
private String mClassName;
|
|
||||||
private Class[] mSignature;
|
|
||||||
private Object[] mArgs;
|
|
||||||
|
|
||||||
LayoutManagerType(String logicalName, String className, Class[] signature, Object[] args) {
|
|
||||||
mLogicalName = logicalName;
|
|
||||||
mClassName = className;
|
|
||||||
mSignature = signature;
|
|
||||||
mArgs = args;
|
|
||||||
}
|
|
||||||
|
|
||||||
String getClassName() {
|
|
||||||
return mClassName;
|
|
||||||
}
|
|
||||||
|
|
||||||
Class[] getSignature() {
|
|
||||||
return mSignature;
|
|
||||||
}
|
|
||||||
|
|
||||||
@NonNull
|
|
||||||
Object[] getArgs(Context context) {
|
|
||||||
Object[] args = new Object[mArgs.length];
|
|
||||||
System.arraycopy(mArgs, 0, args, 0, mArgs.length);
|
|
||||||
for (int i = 0; i < args.length; i++) {
|
|
||||||
if (args[i] == CONTEXT) {
|
|
||||||
args[i] = context;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return args;
|
|
||||||
}
|
|
||||||
|
|
||||||
@NonNull
|
|
||||||
public static LayoutManagerType getDefault() {
|
|
||||||
return LINEAR_LAYOUT_MANGER;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
public static LayoutManagerType getByLogicalName(@NonNull String logicalName) {
|
|
||||||
for (LayoutManagerType type : values()) {
|
|
||||||
if (logicalName.equals(type.mLogicalName)) {
|
|
||||||
return type;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
public static LayoutManagerType getByClassName(@NonNull String className) {
|
|
||||||
for (LayoutManagerType type : values()) {
|
|
||||||
if (className.equals(type.mClassName)) {
|
|
||||||
return type;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -45,7 +45,7 @@ import com.android.layoutlib.bridge.Bridge;
|
|||||||
import com.android.layoutlib.bridge.android.BridgeContext;
|
import com.android.layoutlib.bridge.android.BridgeContext;
|
||||||
import com.android.layoutlib.bridge.android.BridgeLayoutParamsMapAttributes;
|
import com.android.layoutlib.bridge.android.BridgeLayoutParamsMapAttributes;
|
||||||
import com.android.layoutlib.bridge.android.BridgeXmlBlockParser;
|
import com.android.layoutlib.bridge.android.BridgeXmlBlockParser;
|
||||||
import com.android.layoutlib.bridge.android.SessionParamsFlags;
|
import com.android.layoutlib.bridge.android.RenderParamsFlags;
|
||||||
import com.android.layoutlib.bridge.android.support.RecyclerViewUtil;
|
import com.android.layoutlib.bridge.android.support.RecyclerViewUtil;
|
||||||
import com.android.layoutlib.bridge.bars.AppCompatActionBar;
|
import com.android.layoutlib.bridge.bars.AppCompatActionBar;
|
||||||
import com.android.layoutlib.bridge.bars.BridgeActionBar;
|
import com.android.layoutlib.bridge.bars.BridgeActionBar;
|
||||||
@ -403,7 +403,7 @@ public class RenderSessionImpl extends RenderAction<SessionParams> {
|
|||||||
// it can instantiate the custom Fragment.
|
// it can instantiate the custom Fragment.
|
||||||
Fragment_Delegate.setLayoutlibCallback(params.getLayoutlibCallback());
|
Fragment_Delegate.setLayoutlibCallback(params.getLayoutlibCallback());
|
||||||
|
|
||||||
String rootTag = params.getFlag(SessionParamsFlags.FLAG_KEY_ROOT_TAG);
|
String rootTag = params.getFlag(RenderParamsFlags.FLAG_KEY_ROOT_TAG);
|
||||||
boolean isPreference = "PreferenceScreen".equals(rootTag);
|
boolean isPreference = "PreferenceScreen".equals(rootTag);
|
||||||
View view;
|
View view;
|
||||||
if (isPreference) {
|
if (isPreference) {
|
||||||
|
Reference in New Issue
Block a user