[android] 여러 글꼴 / 텍스트 크기 등으로 문자열을 만드는 SpannableStringBuilder 예?

다음과 같은 문자열을 표시 할 TextView에 배치 된 문자열을 만들어야합니다.

첫 번째 부분은 굵게하지 BOLD 나머지 대담하지

그래서 어떻게 이것을 할 수 있는지 알고 싶습니다 SpannableStringBuilder.

이 작업을 수행하기 위해 세 개의 텍스트 편집기를 사용할 수 있지만 최상의 솔루션을 사용하고 싶습니다.



답변

First Part Not Bold   BOLD  rest not bold

@Rajesh가 제안한대로 또는 이것에 의해 이것을 할 수 있습니다.

String normalBefore= "First Part Not Bold ";
String normalBOLD=  "BOLD ";
String normalAfter= "rest not bold";
String finalString= normalBefore+normalBOLD+normalAfter;
Spannable sb = new SpannableString( finalString );
sb.setSpan(new StyleSpan(android.graphics.Typeface.BOLD), finalString.indexOf(normalBOLD)+ normalBOLD.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE); //bold
sb.setSpan(new AbsoluteSizeSpan(intSize), finalString.indexOf(normalBOLD)+ normalBOLD.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);//resize size

이것을 TextView에 표시하려면

textview.setText(sb,  TextView.BufferType.SPANNABLE);


답변

받아 들여지는 대답은 괜찮지 만 (그리고 나는 그것을 찬성했습니다) 제출자가 요청 한대로 SpannableStringBuilder를 사용하지 못했습니다. 빌더가 가장 이해하기 쉬운 경우가 있었기 때문에 여기에 코드가 있습니다 (다른 사람에게 도움이 될 경우 텍스트의 색상을 변경하는 보너스 사용 포함). SpannableStringBuilder 생성자에 초기 문자열을 제공 할 수도 있지만 여기에서 “append”를 사용하여 원하는 “굵은”텍스트 앞에 많은 것을 추가 한 다음 표시된대로 시작을 기록 할 수 있음을 명확히하기 위해 여기에서 설정했습니다. 나는 이것이 허용되는 답변보다 더 빠른 코드라고 의심합니다.

SpannableStringBuilder longDescription = new SpannableStringBuilder();
longDescription.append("First Part Not Bold ");
int start = longDescription.length();
longDescription.append("BOLD");
longDescription.setSpan(new ForegroundColorSpan(0xFFCC5500), start, longDescription.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
longDescription.setSpan(new StyleSpan(android.graphics.Typeface.BOLD), start, longDescription.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
longDescription.append(" rest not bold");


답변

Kotlin을 사용하는 경우 android-ktx 라이브러리를 사용하여 다음을 수행 할 수 있습니다.

val s = SpannableStringBuilder()
        .append("First Part Not Bold ")
        .bold { append("BOLD") }
        .append("Rest not bold")

bold확장 기능입니다 SpannableStringBuilder. 사용할 수있는 작업 목록은 여기 에서 설명서를 참조하십시오 .

다른 예시:

val s = SpannableStringBuilder()
            .color(green, { append("Green text ") })
            .append("Normal text ")
            .scale(0.5, { append("Text at half size " })
            .backgroundColor(green, { append("Background green") })

green해결 된 RGB 색상은 어디에 있습니까 ?

스팬을 중첩하여 임베디드 DSL로 끝낼 수도 있습니다.

bold { underline { italic { append("Bold and underlined") } } }

build.gradle작동하려면 앱 모듈 수준에서 다음이 필요합니다 .

repositories {
    google()
}

dependencies {
    implementation "androidx.core:core-ktx:1.2.0"
}


답변

API 21에서 SpannableStringBuilder에는이를 수행하는 간단한 방법이 포함되어 있습니다. 다음은 솔루션 예입니다.

SpannableStringBuilder builder= new SpannableStringBuilder();
StyleSpan boldSpan = new StyleSpan(android.graphics.Typeface.BOLD);
builder.append("First Part Not Bold ")
              .append("BOLD ", boldSpan, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE)
              .append("rest not bold");

API 21을 지원하지 않는 좋은 기회이므로 해당 메서드에서 코드를 복제 할 수 있습니다.

public SpannableStringBuilder append(CharSequence text, Object what, int flags) {
        int start = length();
        append(text);
        setSpan(what, start, length(), flags);
        return this;
}


답변

Html클래스를 사용하여 TextView에서 HTML 코드를 사용하십시오 .

Spanned styledText = Html.fromHtml("First Part Not Bold <b>BOLD</b> rest not bold");
textView.setText(styledText);


답변

이 코드는 html bold 태그 안에있는 모든 것을 굵게 설정해야합니다. 또한 태그를 삭제하여 내부 내용 만 표시합니다.

        SpannableStringBuilder sb = new SpannableStringBuilder("this is <b>bold</b> and this is <b>bold too</b>  and this is <b>bold too, again</b>.");

        Pattern p = Pattern.compile("<b>.*?</b>", Pattern.CASE_INSENSITIVE);
        boolean stop = false;
        while (!stop)
        {
            Matcher m = p.matcher(sb.toString());
            if (m.find()) {
                sb.setSpan(new StyleSpan(android.graphics.Typeface.BOLD), m.start(), m.end(), Spannable.SPAN_INCLUSIVE_INCLUSIVE);
                sb.delete(m.end()-4, m.end());
                sb.delete(m.start(), m.start() + 3);
            }
            else
                stop = true;
        }

이 코드는 Superscript (sup 태그) 등과 같은 다른 html 스타일 태그에도 적용 할 수 있습니다.

        SpannableStringBuilder sb = new SpannableStringBuilder("text has <sup>superscript</sup> tag");

        Pattern p = Pattern.compile("<sup>.*?</sup>", Pattern.CASE_INSENSITIVE);
        boolean stop = false;
        while (!stop)
        {
            Matcher m = p.matcher(sb.toString());
            if (m.find()) {
                sb.setSpan(new SuperscriptSpan(), m.start(), m.end(), Spannable.SPAN_INCLUSIVE_INCLUSIVE);
                sb.delete(m.end()-6, m.end());
                sb.delete(m.start(), m.start() + 5);
            }
            else
                stop = true;
        }

색상을 설정하려면 setSpan과 함께 ForegroundColorSpan을 사용하십시오.

sb.setSpan(new ForegroundColorSpan(Color.rgb(255, 0, 0)), m.start(), m.end(), Spannable.SPAN_INCLUSIVE_INCLUSIVE);

도움이 되었기를 바랍니다.


답변

그래서 나는 이것이 해결되었다는 것을 알고 있으며 SpannableStringBuilder로 요청한 것처럼 문자열을 더 동적으로 만들고 싶을 때 나는 이것을 올릴 것이라고 생각했습니다.

// Stuff needed
TextView DataTextView = (TextView)rootView.findViewById(R.id.DataView);
String Fields[] = {...database column names as strings... "x","y"};

String DataString = new String();

int start,stop;     // Start and Stop of formatting

// Final Result
SpannableStringBuilder coloredString = new SpannableStringBuilder();

SpannableString temp;       // Small segment of colored string
for (int i =0; i < Fields.length; i++)
{
    if (database_result.containsKey(Fields[i]))  // Be sure a field exists in the ContentValues
    {
            DataString = Fields[i]+": ";
        start = DataString.length();
        DataString = DataString+ +database_result.getAsInteger(Fields[i])+" ";
        stop= DataString.length();
        temp = new SpannableString(DataString);
        temp.setSpan(new ForegroundColorSpan(Color.WHITE),start, stop, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
        coloredString.append(temp);
    }
}
DataTextView.setText(coloredString);

database_result는 SQL 쿼리의 반환 된 Cursor 유형에서 생성 한 ContentValues ​​유형입니다. 내가 가진 유일한 문제는 처음에는 ColorSpaning이 첫 번째 세그먼트였습니다. 루프에서 하나 (또는 ​​다른 종류의 범위)를 사용할 때마다 새 ForegroundColorSpan을 선언해야한다는 점이 이음새가됩니다.