Как сравнить два чертежа, я делаю так, но не имею успеха
public void MyClick(View view)
{
Drawable fDraw = view.getBackground();
Drawable sDraw = getResources().getDrawable(R.drawable.twt_hover);
if(fDraw.equals(sDraw))
{
//Not coming
}
}
Ответы:
Обновите https://stackoverflow.com/a/36373569/1835650
getConstantState () работает не очень хорошо
Есть еще один способ сравнить:mRememberPwd.getDrawable().getConstantState().equals (getResources().getDrawable(R.drawable.login_checked).getConstantState());
mRemeberPwd
являетсяImageView
в этом примере. Если вы используетеTextView
, используйтеgetBackground().getConstantState
вместо него.источник
Полагаться
getConstantState()
только на одного может привести к ложным отрицаниям .Подход, который я использовал, заключается в том, чтобы попытаться сравнить ConstantState в первом экземпляре, но вернуться к сравнению Bitmap, если эта проверка не удалась.
Это должно работать во всех случаях (включая изображения, которые не являются ресурсами), но учтите, что это требует памяти.
public static boolean areDrawablesIdentical(Drawable drawableA, Drawable drawableB) { Drawable.ConstantState stateA = drawableA.getConstantState(); Drawable.ConstantState stateB = drawableB.getConstantState(); // If the constant state is identical, they are using the same drawable resource. // However, the opposite is not necessarily true. return (stateA != null && stateB != null && stateA.equals(stateB)) || getBitmap(drawableA).sameAs(getBitmap(drawableB)); } public static Bitmap getBitmap(Drawable drawable) { Bitmap result; if (drawable instanceof BitmapDrawable) { result = ((BitmapDrawable) drawable).getBitmap(); } else { int width = drawable.getIntrinsicWidth(); int height = drawable.getIntrinsicHeight(); // Some drawables have no intrinsic width - e.g. solid colours. if (width <= 0) { width = 1; } if (height <= 0) { height = 1; } result = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888); Canvas canvas = new Canvas(result); drawable.setBounds(0, 0, canvas.getWidth(), canvas.getHeight()); drawable.draw(canvas); } return result; }
источник
getConstantState()
сравнениеsetBounds
иdraw
на копии вместо оригинала stackoverflow.com/a/25462223/1916449Мой вопрос заключался в том, чтобы просто сравнить два чертежа, я попытался, но не смог получить какой-либо метод, который напрямую сравнивает два чертежа, однако для своего решения я изменил рисование на растровое изображение, а затем сравнил два растровых изображения, и это работает.
Bitmap bitmap = ((BitmapDrawable)fDraw).getBitmap(); Bitmap bitmap2 = ((BitmapDrawable)sDraw).getBitmap(); if(bitmap == bitmap2) { //Code blcok }
источник
для SDK 21+
это работает в SDK-21
mRememberPwd.getDrawable().getConstantState().equals (getResources().getDrawable(R.drawable.login_checked).getConstantState())
для SDK +21 android 5. установите выводимый идентификатор для просмотра изображений с тегом
img.setTag(R.drawable.xxx);
и сравните вот так
if ((Integer) img.getTag() == R.drawable.xxx) { ....your code }
это решение для тех, кто хочет сравнить
drawable
идентификаторimageview
с идентификаторомdrawable.xxx
.источник
Решение для Android 5:
if(image.getDrawable().getConstantState().equals(image.getContext().getDrawable(R.drawable.something).getConstantState()))
источник
getDrawable (int) теперь не рекомендуется. Используйте getDrawable (context, R.drawable.yourimageid)
Сравнить два фона
Boolean Condition1=v.getBackground().getConstantState().equals( ContextCompat.getDrawable(getApplicationContext(),R.drawable.***).getConstantState());
источник
context.getResources().getDrawable(R.drawable.***)
на Android 6+, но не был на Android 5. С этим небольшим изменением я могу безупречно сравнивать фоновые рисунки во всех версиях Android.возможно, попробуй вот так:
public void MyClick(View view) { Drawable fDraw = view.getBackground(); Drawable sDraw = getResources().getDrawable(R.drawable.twt_hover); if(fDraw.hashCode() == sDraw.hashCode()) { //Not coming } }
или подготовьте метод, который принимает два аргумента и возвращает логическое значение. В этом методе вы можете преобразовать drawable в байты и сравнить,
public boolean compareDrawable(Drawable d1, Drawable d2){ try{ Bitmap bitmap1 = ((BitmapDrawable)d1).getBitmap(); ByteArrayOutputStream stream1 = new ByteArrayOutputStream(); bitmap1.compress(Bitmap.CompressFormat.JPEG, 100, stream1); stream1.flush(); byte[] bitmapdata1 = stream1.toByteArray(); stream1.close(); Bitmap bitmap2 = ((BitmapDrawable)d2).getBitmap(); ByteArrayOutputStream stream2 = new ByteArrayOutputStream(); bitmap2.compress(Bitmap.CompressFormat.JPEG, 100, stream2); stream2.flush(); byte[] bitmapdata2 = stream2.toByteArray(); stream2.close(); return bitmapdata1.equals(bitmapdata2); } catch (Exception e) { // TODO: handle exception } return false; }
источник
e.g R.drawable.abc
обоих параметров?Хорошо, я думаю, что нашел окончательное решение для этого. Из-за AppCompat и друзей предоставляемый drawable иногда раздувается в разных формах, поэтому этого недостаточно
getResources().getBitmap(R.drawable.my_awesome_drawable)
.Итак, чтобы получить доступный для рисования экземпляр того же типа и формы, что и в представлении, можно сделать следующее:
public static Drawable drawableFrom(View view, @DrawableRes int drawableId) { Context context = view.getContext(); try { View dummyView = view.getClass().getConstructor(Context.class).newInstance(context); dummyView.setBackgroundResource(drawableId); return dummyView.getBackground(); } catch (Exception e) { return ResourcesCompat.getDrawable(context.getResources(), drawableId, null); } }
Это полезно при проведении тестов. Однако я бы не рекомендовал делать это в продакшене. Если вам нужно, было бы желательно дополнительное кеширование, чтобы избежать слишком большого количества отражений.
Для тестов Expresso вы можете использовать это довольно хорошо:
onView(withDrawable(R.drawable.awesome_drawable)) .check(matches(isDisplayed()));
или же
onView(withId(R.id.view_id)) .check(matches(withDrawable(R.drawable.awesome_drawable)));
Прежде чем вам нужно будет объявить этот вспомогательный класс:
public class CustomMatchers { public static Matcher<View> withDrawable(@DrawableRes final int drawableId) { return new DrawableViewMatcher(drawableId); } private static class DrawableViewMatcher extends TypeSafeMatcher<View> { private final int expectedId; private String resourceName; private enum DrawableExtractionPolicy { IMAGE_VIEW { @Override Drawable findDrawable(View view) { return view instanceof ImageView ? ((ImageView) view).getDrawable() : null; } }, TEXT_VIEW_COMPOUND { @Override Drawable findDrawable(View view) { return view instanceof TextView ? findFirstCompoundDrawable((TextView) view) : null; } }, BACKGROUND { @Override Drawable findDrawable(View view) { return view.getBackground(); } }; @Nullable private static Drawable findFirstCompoundDrawable(TextView view) { for (Drawable drawable : view.getCompoundDrawables()) { if (drawable != null) { return drawable; } } return null; } abstract Drawable findDrawable(View view); } private DrawableViewMatcher(@DrawableRes int expectedId) { this.expectedId = expectedId; } @Override protected boolean matchesSafely(View view) { resourceName = resources(view).getResourceName(expectedId); return haveSameState(actualDrawable(view), expectedDrawable(view)); } private boolean haveSameState(Drawable actual, Drawable expected) { return actual != null && expected != null && areEqual(expected.getConstantState(), actual.getConstantState()); } private Drawable actualDrawable(View view) { for (DrawableExtractionPolicy policy : DrawableExtractionPolicy.values()) { Drawable drawable = policy.findDrawable(view); if (drawable != null) { return drawable; } } return null; } private boolean areEqual(Object first, Object second) { return first == null ? second == null : first.equals(second); } private Drawable expectedDrawable(View view) { return drawableFrom(view, expectedId); } private static Drawable drawableFrom(View view, @DrawableRes int drawableId) { Context context = view.getContext(); try { View dummyView = view.getClass().getConstructor(Context.class).newInstance(context); dummyView.setBackgroundResource(drawableId); return dummyView.getBackground(); } catch (Exception e) { return ResourcesCompat.getDrawable(context.getResources(), drawableId, null); } } @NonNull private Resources resources(View view) { return view.getContext().getResources(); } @Override public void describeTo(Description description) { description.appendText("with drawable from resource id: "); description.appendValue(expectedId); if (resourceName != null) { description.appendValueList("[", "", "]", resourceName); } } } }
источник
источник
Я уже отвечал на аналогичную тему здесь: Получите идентификатор чертежа в ImageView . Подход основан на пометке представления с указанным идентификатором ресурса в custom
LayoutInflater
. Весь процесс автоматизирован простой библиотекой TagView .В результате вы можете сравнить два чертежа только по их идентификаторам:
TagViewUtils.getTag(view, ViewTag.VIEW_BACKGROUND.id) == R.drawable.twt_hover
источник
Расширяя ответ от @vaughandroid, следующий Matcher работает для окрашенного Vector Drawable. Вы должны предоставить оттенок, который использовался для Drawable.
public static Matcher<View> compareVectorDrawables(final int imageId, final int tintId) { return new TypeSafeMatcher<View>() { @Override protected boolean matchesSafely(View target) { if (!(target instanceof ImageView)) { return false; } ImageView imageView = (ImageView) target; if (imageId < 0) { return imageView.getDrawable() == null; } Resources resources = target.getContext().getResources(); Drawable expectedDrawable = resources.getDrawable(imageId, null); if (expectedDrawable == null) { return false; } Drawable imageDrawable = imageView.getDrawable(); ColorFilter imageColorFilter = imageDrawable.getColorFilter(); expectedDrawable.setColorFilter(imageColorFilter); expectedDrawable.setTintList(target.getResources() .getColorStateList(tintId, null)); boolean areSame = areDrawablesIdentical(imageDrawable, expectedDrawable); return areSame; } public boolean areDrawablesIdentical(Drawable drawableA, Drawable drawableB) { Drawable.ConstantState stateA = drawableA.getConstantState(); Drawable.ConstantState stateB = drawableB.getConstantState(); // If the constant state is identical, they are using the same drawable resource. // However, the opposite is not necessarily true. return (stateA != null && stateB != null && stateA.equals(stateB)) || getBitmap(drawableA).sameAs(getBitmap(drawableB)); } public Bitmap getBitmap(Drawable drawable) { Bitmap result; if (drawable instanceof BitmapDrawable) { result = ((BitmapDrawable) drawable).getBitmap(); } else { int width = drawable.getIntrinsicWidth(); int height = drawable.getIntrinsicHeight(); // Some drawables have no intrinsic width - e.g. solid colours. if (width <= 0) { width = 1; } if (height <= 0) { height = 1; } result = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888); Canvas canvas = new Canvas(result); drawable.setBounds(0, 0, canvas.getWidth(), canvas.getHeight()); drawable.draw(canvas); } return result; } @Override public void describeTo(Description description) { } }; }
источник
Сравните 2 чертежа:
drawable1.constantState == drawable2.constantState || drawable1.toBitmap().sameAs(drawable2.toBitmap())
Если вы не можете найти
Drawable.toBitmap(...)
вот это Drawable.ktисточник
если вы хотите напрямую сравнить два чертежа, используйте следующий код
Возможность рисования fDraw = getResources (). GetDrawable (R.drawable.twt_hover);
Drawable sDraw = getResources (). GetDrawable (R.drawable.twt_hover);
if (fDraw.getConstantState().equals(sDraw.getConstantState())) { //write your code. } else { //write your code. }
источник
Когда вы используете
equals()
метод, он используется для сравнения содержимого. вы должны попробовать==
сравнить два объекта.public void MyClick(View view) { Drawable fDraw = view.getBackground(); Drawable sDraw = getResources().getDrawable(R.drawable.twt_hover); if( fDraw == sDraw ) { // Coming } }
источник