android - 带椭圆大小的垂直 TextView 和所有其他默认 TextView 功能支持

标签 android android-layout android-fragments

如何创建支持椭圆大小的垂直 TextView(旋转 90° 或 -90°)(这意味着如果 View 中没有足够的空间,文本将被截断,附加的“...”表示截断),可以使用 dp 值、wrap_contentmatch_parent?

调整大小

所有其他 TextView 选项也应该受到尊重。

它也不需要任何额外的代码来执行旋转或其他,并且应该在 UI 编辑器中正确显示。

This solution效果很好,但是,重写 onDraw() 会导致 ellipsize 和其他功能不再起作用。如果文本太大而无法显示,它就会消失,否则会发生其他奇怪的事情。

下面的旋转就ok了,但是ellipsize是根据view的width计算的,如果旋转90°应该是根据height计算的。

protected void onDraw(Canvas canvas) {
    canvas.save();

    canvas.rotate(90, 0, getPaint().getTextSize() +5);
    super.onDraw(canvas);
    canvas.restore();
}
TextView 上的

android:rotation 不是好的解决方案,因为定位和调整大小无法正常工作。

作为一个选项,View 应该允许文本旋转任何角度。

最佳答案

您可以使用类似这样的东西,这可能会对您有所帮助。

public class VerticalLabelView extends View {
    private TextPaint mTextPaint;
    private String mText;
    private int mAscent;
    private Rect text_bounds = new Rect();

    final static int DEFAULT_TEXT_SIZE = 15;

    public VerticalLabelView(Context context) {
        super(context);
        initLabelView();
    }

    public VerticalLabelView(Context context, AttributeSet attrs) {
        super(context, attrs);
        initLabelView();

        TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.VerticalLabelView);

        CharSequence s = a.getString(R.styleable.VerticalLabelView_text);
        if (s != null) setText(s.toString());

        setTextColor(a.getColor(R.styleable.VerticalLabelView_textColor, 0xFF000000));

        int textSize = a.getDimensionPixelOffset(R.styleable.VerticalLabelView_textSize, 0);
        if (textSize > 0) setTextSize(textSize);

        a.recycle();
    }

    private final void initLabelView() {
        mTextPaint = new TextPaint();
        mTextPaint.setAntiAlias(true);
        mTextPaint.setTextSize(DEFAULT_TEXT_SIZE);
        mTextPaint.setColor(0xFF000000);
        mTextPaint.setTextAlign(Align.CENTER);
        setPadding(3, 3, 3, 3);
    }

    public void setText(String text) {
        mText = text;
        requestLayout();
        invalidate();
    }

    public void setTextSize(int size) {
        mTextPaint.setTextSize(size);
        requestLayout();
        invalidate();
    }

    public void setTextColor(int color) {
        mTextPaint.setColor(color);
        invalidate();
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {

        mTextPaint.getTextBounds(mText, 0, mText.length(), text_bounds);
        setMeasuredDimension(
                measureWidth(widthMeasureSpec),
                measureHeight(heightMeasureSpec));
    }

    private int measureWidth(int measureSpec) {
        int result = 0;
        int specMode = MeasureSpec.getMode(measureSpec);
        int specSize = MeasureSpec.getSize(measureSpec);

        if (specMode == MeasureSpec.EXACTLY) {
            // We were told how big to be
            result = specSize;
        } else {
            // Measure the text
            result = text_bounds.height() + getPaddingLeft() + getPaddingRight();

            if (specMode == MeasureSpec.AT_MOST) {
                // Respect AT_MOST value if that was what is called for by measureSpec
                result = Math.min(result, specSize);
            }
        }
        return result;
    }

    private int measureHeight(int measureSpec) {
        int result = 0;
        int specMode = MeasureSpec.getMode(measureSpec);
        int specSize = MeasureSpec.getSize(measureSpec);

        mAscent = (int) mTextPaint.ascent();
        if (specMode == MeasureSpec.EXACTLY) {
            // We were told how big to be
            result = specSize;
        } else {
            // Measure the text
            result = text_bounds.width() + getPaddingTop() + getPaddingBottom();

            if (specMode == MeasureSpec.AT_MOST) {
                // Respect AT_MOST value if that was what is called for by measureSpec
                result = Math.min(result, specSize);
            }
        }
        return result;
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);

        float text_horizontally_centered_origin_x = getPaddingLeft() + text_bounds.width()/2f;
        float text_horizontally_centered_origin_y = getPaddingTop() - mAscent;

        canvas.translate(text_horizontally_centered_origin_y, text_horizontally_centered_origin_x);
        canvas.rotate(-90);
        canvas.drawText(mText, 0, 0, mTextPaint);
    }
}

attrs.xml 中:

<resources>
     <declare-styleable name="VerticalLabelView">
        <attr name="text" format="string" />
        <attr name="textColor" format="color" />
        <attr name="textSize" format="dimension" />
    </declare-styleable>
</resources>

关于android - 带椭圆大小的垂直 TextView 和所有其他默认 TextView 功能支持,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36313638/

相关文章:

android - Fragment 的 onActivityCreated() 在 Activity 的 onDestroy() 之后被调用

android - PhoneGap 2.2 的奇怪问题,Android 4.0.3 上的预测文本和输入类型 ="text"

java - "fill_parent"和 "wrap_content"之间的区别

android - 如何在 CollapsingToolbarLayout 中使用 PercentRelativeLayout?

android - 我们可以为不同的 ldpi 屏幕创建不同的布局 XML 文件吗?

android - 另一个 fragment 内的谷歌地图 fragment

java - Android 中的堆转储和内存使用差异?

android - 如何将 AirPush 广告集成到我的 Android 应用程序中?

java - Android 最高效的列表容器

java - 滑动时选项卡内容不会重叠