31

我有一个EditText我想缩放它,并用setScaleX/滚动setScaleY,它工作正常 - 文本正在正确的位置进行编辑。

但是当我尝试选择文本时 - 它会将选择句柄绘制到诸如文本未缩放时的位置。这是已知的错误

这是预期的结果,因为句柄是在与视图大小相关的弹出窗口上绘制的。

上的所有操作android.widget.Editor都针对其字段private TextView mTextView;。如果我们通过反射设置自己的编辑器,我不知道如何处理私有方法,这是不可覆盖的。

选择句柄也是在布局中计算的弹出窗口坐标上绘制的android.widget.Editor.HandleView#HandleView,我只需要DynamicLayout,但它对我们的目的没有区别。

方法android.text.Layout#getPrimaryHorizontal(int, boolean)是公共的,它的价值可以成倍增加,但是我们需要扩展和覆盖私有方法android.widget.TextView#makeSingleLayout,但这是一个问题。

我们也可以使用所有必需的覆盖方法来实现我们自己的布局,但是我们可以覆盖的所有方法都标有@hide注释,并且没有可以通过反射访问的字段。

下一个屏幕截图显示为 2 倍缩放

在此处输入图像描述

PS:任务的上下文是一个带有捏到缩放编辑文本的编辑器。计算大小的文本重新布局不是解决方案。因为我需要每个屏幕尺寸的便携式文档。

4

2 回答 2

13

您可以使用MetricAffectingSpan来做到这一点。这是一个例子:

package android.text.style;

import android.os.Parcel;
import android.text.ParcelableSpan;
import android.text.TextPaint;
import android.text.TextUtils;

public class AbsoluteSizeSpan extends MetricAffectingSpan implements ParcelableSpan {

    private final int mSize;
    private boolean mDip;

    /**
     * Set the text size to <code>size physical pixels.
     */
    public AbsoluteSizeSpan(int size) {
        mSize = size;
    }

    /**
     * Set the text size to <code>size physical pixels,
     * or to <code>size device-independent pixels if
     * <code>dip is true.
     */
    public AbsoluteSizeSpan(int size, boolean dip) {
        mSize = size;
        mDip = dip;
    }

    public AbsoluteSizeSpan(Parcel src) {
        mSize = src.readInt();
        mDip = src.readInt() != 0;
    }

    public int getSpanTypeId() {
        return TextUtils.ABSOLUTE_SIZE_SPAN;
    }

    public int describeContents() {
        return 0;
    }

    public void writeToParcel(Parcel dest, int flags) {
        dest.writeInt(mSize);
        dest.writeInt(mDip ? 1 : 0);
    }

    public int getSize() {
        return mSize;
    }

    public boolean getDip() {
        return mDip;
    }

    @Override
    public void updateDrawState(TextPaint ds) {
        if (mDip) {
            ds.setTextSize(mSize * ds.density);
        } else {
            ds.setTextSize(mSize);
        }
    }

    @Override
    public void updateMeasureState(TextPaint ds) {
        if (mDip) {
            ds.setTextSize(mSize * ds.density);
        } else {
            ds.setTextSize(mSize);
        }
    }
}

参考:Java 源代码仓库项目

您需要使用MetricAffectingSpanwrap(CharacterStyle cs) - 它允许将CharacterStyle应用于给定 Spanned 的单个区域。

在您的子类中,覆盖onTouch并将其值传递给ScaleGestureDetector。将检测到的比例存储为成员变量。

覆盖onDraw,并在调用 super.onDraw之前使用您的比例值调用canvas.scale ()。正如您在AbsoluteSizeSpan中注意到的那样,使用AbsoluteSizeSpan(Parcel src)将获得您希望调整大小的文本,然后应用updateDrawState

于 2014-05-07T10:01:36.983 回答
2

您是否尝试对这些元素应用不同的样式?我认为它应该与一些条件有关。

<item name="android:textSelectHandle">@drawable/text_select_handle_middle</item>
<item name="android:textSelectHandleLeft">@drawable/text_select_handle_left</item>
<item name="android:textSelectHandleRight">@drawable/text_select_handle_right</item>
于 2014-05-10T21:49:22.053 回答