Подтвердить что ты не робот

Как добавить тень в TextView для выбора/фокуса

Мой вопрос заключается в том, как добавить тень к тексту при выборе TextView или отобразить выбранный TextView. Например, у меня есть CheckedTextView, который меняет фон в соответствии с типом выбора. Я также сделал селектор текста, который меняет цвет на разные состояния. Теперь я хотел бы добавить тень, когда, например, выбран параметр "Вид". Таким образом, он меняет цвет фона, цвет текста и создает тень. Это мой селектор текста:

<selector 
xmlns:android="http://schemas.android.com/apk/res/android">

<item 
    android:state_focused="true" 
    android:state_pressed="false"       
    android:color="@android:color/white"
    style="@style/DarkShadow"/>

<item 
    android:state_focused="true" 
    android:state_pressed="true"            
    android:color="@android:color/white"
    style="@style/DarkShadow"/>

<item 
    android:state_focused="false" 
    android:state_pressed="true" 
    android:color="@android:color/white"
    style="@style/DarkShadow"/>

<item 
    android:color="@color/primary_text_light_disable_only"/>

и стиль:

<style name="DarkShadow">   
    <item name="android:shadowColor">#BB000000</item>
    <item name="android:shadowRadius">2.75</item>
</style>

Теперь текст будет правильно выделен, но тени не появятся. Кто-нибудь знает, как это решить?

4b9b3361

Ответ 1

Это текущее ограничение Android SDK. Я продлил TextView, чтобы он работал, вы можете использовать его свободно:

CustomTextView.java:

import android.widget.TextView;
import android.util.AttributeSet;
import android.content.res.TypedArray;
import android.content.Context;

import com.client.R;


public class CustomTextView extends TextView
{

    private static String TAG = "CustomTextView";

    private ColorStateList mShadowColors;
    private float mShadowDx;
    private float mShadowDy;
    private float mShadowRadius;


    public CustomTextView(Context context)
    {
        super(context);
    }


    public CustomTextView(Context context, AttributeSet attrs)
    {
        super(context, attrs);
        init(context, attrs);
    }


    public CustomTextView(Context context, AttributeSet attrs, int defStyle)
    {
        super(context, attrs, defStyle);
        init(context, attrs);
    }


    /**
     * Initialization process
     * 
     * @param context
     * @param attrs
     * @param defStyle
     */
    private void init(Context context, AttributeSet attrs, int defStyle)
    {
        TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.CustomTextView, defStyle, 0);

        final int attributeCount = a.getIndexCount();
        for (int i = 0; i < attributeCount; i++) {
            int curAttr = a.getIndex(i);

            switch (curAttr) {                  
                case R.styleable.CustomTextView_shadowColors:
                    mShadowColors = a.getColorStateList(curAttr);
                    break;

                case R.styleable.CustomTextView_android_shadowDx:
                    mShadowDx = a.getFloat(curAttr, 0);
                    break;

                case R.styleable.CustomTextView_android_shadowDy:
                    mShadowDy = a.getFloat(curAttr, 0);
                    break;

                case R.styleable.CustomTextView_android_shadowRadius:
                    mShadowRadius = a.getFloat(curAttr, 0);
                    break;  

                default:
                break;
        }
    }

        a.recycle();

        updateShadowColor();
    }

    private void updateShadowColor()
    {
        if (mShadowColors != null) {
            setShadowLayer(mShadowRadius, mShadowDx, mShadowDy, mShadowColors.getColorForState(getDrawableState(), 0));
            invalidate();
        }
    }

    @Override
    protected void drawableStateChanged()
    {
        super.drawableStateChanged();
        updateShadowColor();
    }
}

Вам также нужно добавить это в свой attr.xml(или создать): attr.xml:

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <declare-styleable name="Theme">
        <attr format="reference" name="CustomTextView"/>
    </declare-styleable>

    <declare-styleable name="CustomTextView">
        <attr name="shadowColors" format="color|reference"/>
        <attr name="android:shadowDx"/>
        <attr name="android:shadowDy"/>
        <attr name="android:shadowRadius"/>

    </declare-styleable>
</resources>

Итак, вы сможете использовать его в своих xmls, например:

<com.client.ui.textviews.CustomTextView
 xmlns:client="http://schemas.android.com/apk/res/com.client"
        android:id="@+id/join_text"
        android:shadowDx="1"
        android:shadowDy="1"
        android:shadowRadius="1"
        client:shadowColors="@color/btn_green_shadow_color"/>

Где @color/btn_green_shadow_color указывает на селектор, такой как:

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">

    <item android:state_enabled="false" android:color="@android:color/white"/>
    <item android:state_pressed="true" android:color="@color/BzDarkGray"/>
    <item android:color="@android:color/black"/>

</selector>

Если вы не знакомы с использованием пользовательских атрибутов (с использованием собственного пространства имен xml, которое я использовал), обратитесь к fooobar.com/questions/16076/....

Ответ 2

Да, я столкнулся с той же проблемой, вы можете изменить цвет текста, используя селектор в xml, но не shadowcolor. Поэтому, чтобы решить проблему, вам может потребоваться расширение CheckedTextView или любого вида, который вам нужен, а затем переопределить onDraw(Canvas canvas) в соответствии с состоянием представления Таким образом, вам необходимо использовать public void setShadowLayer (float radius, float dx, float dy, int color), определенный в здесь

например:

@Override
protected void onDraw(Canvas canvas) {
    if(isPressed()){
        setShadowLayer(1, 0, 1, Color.RED);
    }else{
        if(isFocused()){
            setShadowLayer(1, 0, 1, Color.WHITE);
        }else{
            setShadowLayer(1, 0, 1, Color.BLACK);
        }
    }
    super.onDraw(canvas);
}

Я надеюсь, что работает

Ответ 3

Вот что я сделал:

@Override
protected void drawableStateChanged() {
    super.drawableStateChanged();
    if(isPressed()) {
        setShadowLayer(15, 0, 0, getTextColors().getDefaultColor());
    } else {
        setShadowLayer(0, 0, 0, Color.TRANSPARENT);
    }
}