字符样式/跨度设置字母间距

时间:2015-02-17 13:24:28

标签: android textview

TextView.setLetterSpacing允许设置字母间距/字符间距。

是否有相应的CharacterStyle / span class允许在TextView的文字子集上设置字母间距?

3 个答案:

答案 0 :(得分:6)

所以我通过拧自己的span类来解决这个问题,它适用于API级别21及以上。

import android.annotation.TargetApi;
import android.content.Context;
import android.os.Parcel;
import android.text.TextPaint;
import android.text.style.MetricAffectingSpan;

/**
 * Created by alex on 19/02/2015.
 */
@TargetApi(21)
public class LetterSpacingSpan extends MetricAffectingSpan {
    private float letterSpacing;

    /**
     * @param letterSpacing
     */
    public LetterSpacingSpan(float letterSpacing) {
        this.letterSpacing = letterSpacing;
    }

    public float getLetterSpacing() {
        return letterSpacing;
    }

    @Override
    public void updateDrawState(TextPaint ds) {
        apply(ds);
    }

    @Override
    public void updateMeasureState(TextPaint paint) {
        apply(paint);
    }

    private void apply(TextPaint paint) {
        paint.setLetterSpacing(letterSpacing);
    }

}

答案 1 :(得分:5)

您可以查看实现所需内容的this custom class实现。

编辑:好的,如果你想申请文本的一小部分,你可以这样做:

CharSequence firstPart = "First Part";
CharSequence thirdPart = "Third Part";
SpannableStringBuilder middlePart = new SpannableStringBuilder("Middle Part");

int spacing = 2;
final String nonBreakingSpace = "\u00A0";


for (int i = middlePart.length() - 1; i >= 1; i--){
    middlePart.insert(i, nonBreakingSpace);
    middlePart.setSpan(new ScaleXSpan(spacing), i, i + 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
}

((TextView) rootView.findViewById(R.id.text_view)).setText(TextUtils.concat(firstPart, " ", middlePart, " ", thirdPart));
但是,它并不完美。

编辑:

要清楚,这是上面代码的结果

第一行(" Custom Spannable")是(+/-)上述代码的结果。第二行(" Letter Spacing")是属性android:letterSpacing="1"的结果。

enter image description here

正如你所看到的,就像我说的那样,它并不完美,但却是我为棒棒糖前设备找到的唯一解决方案。

我还认为你的问题的重点是使其与棒棒糖前设备兼容。我的坏。

干杯

答案 2 :(得分:0)

您可以使用我在Kotlin中编写的此功能。它将允许您修改一个TextView中两个字母之间的字母间距,而不会影响其他字母。

fun SpannableStringBuilder.setLetterSpacingBetweenTwoLettersSpans(
    firstLetter: Char,
    secondLetter: Char,
    spacing: Float
) {
    forEachIndexed { charIndex, startingChar ->
        if (startingChar == firstLetter && get(charIndex + 1) == secondLetter) {
            var negative = spacing < 0
            if (get(lastIndex) != ' ') append(' ')
            forEachIndexed { index, _ ->
                if (index > charIndex) {
                    val appliedSpacing = if (index == charIndex + 1) {
                        if (negative) -abs(spacing) else spacing
                    } else {
                        if (negative) spacing else -spacing
                    }
                    if (index != lastIndex) {
                        setSpan(object : MetricAffectingSpan() {
                            override fun updateMeasureState(textPaint: TextPaint) { apply(textPaint) }
                            override fun updateDrawState(tp: TextPaint?) { tp?.let { apply(tp) } }
                            private fun apply(paint: TextPaint) { paint.letterSpacing = appliedSpacing }
                        }, index, index + 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE)
                    }
                    negative = !negative
                }
            }
        }
    }
}

因此您可以像这样使用它:

val spannableStringBuilder = SpannableStringBuilder()
spannableStringBuilder.append("Bear")
spannableStringBuilder.setLetterSpacingBetweenTwoLettersSpans(
    firstLetter = "e",
    secondLetter = "a"
    spacing = 1f // Negative works aswell (that's why I initially did it)
)
yourTextView.text = spannableStringBuilder

1f看起来好像您在我认为的两个字母之间放置了一个空格。负数也会起作用。