Last active
August 12, 2024 07:31
-
-
Save JakeWharton/11274467 to your computer and use it in GitHub Desktop.
Extremely simple wrapper around SpannableStringBuilder to make the API more logical and less awful. Apache 2 licensed.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import android.text.SpannableStringBuilder; | |
import java.util.ArrayDeque; | |
import java.util.Deque; | |
import static android.text.Spanned.SPAN_INCLUSIVE_EXCLUSIVE; | |
/** A {@link SpannableStringBuilder} wrapper whose API doesn't make me want to stab my eyes out. */ | |
public class Truss { | |
private final SpannableStringBuilder builder; | |
private final Deque<Span> stack; | |
public Truss() { | |
builder = new SpannableStringBuilder(); | |
stack = new ArrayDeque<>(); | |
} | |
public Truss append(String string) { | |
builder.append(string); | |
return this; | |
} | |
public Truss append(CharSequence charSequence) { | |
builder.append(charSequence); | |
return this; | |
} | |
public Truss append(char c) { | |
builder.append(c); | |
return this; | |
} | |
public Truss append(int number) { | |
builder.append(String.valueOf(number)); | |
return this; | |
} | |
/** Starts {@code span} at the current position in the builder. */ | |
public Truss pushSpan(Object span) { | |
stack.addLast(new Span(builder.length(), span)); | |
return this; | |
} | |
/** End the most recently pushed span at the current position in the builder. */ | |
public Truss popSpan() { | |
Span span = stack.removeLast(); | |
builder.setSpan(span.span, span.start, builder.length(), SPAN_INCLUSIVE_EXCLUSIVE); | |
return this; | |
} | |
/** Create the final {@link CharSequence}, popping any remaining spans. */ | |
public CharSequence build() { | |
while (!stack.isEmpty()) { | |
popSpan(); | |
} | |
return builder; // TODO make immutable copy? | |
} | |
private static final class Span { | |
final int start; | |
final Object span; | |
public Span(int start, Object span) { | |
this.start = start; | |
this.span = span; | |
} | |
} | |
} |
It's available in Kotlin here: https://developer.android.com/reference/kotlin/androidx/core/text/package-summary#buildSpannedString(kotlin.Function1) along with the builder extension functions below.
See: https://medium.com/exploring-android/exploring-ktx-for-android-13a369795b51#b524 or the original tests for the DSL: https://github.com/android/android-ktx/blob/1.0.0-alpha1/src/androidTest/java/androidx/core/text/SpannableStringBuilderTest.kt
Thanks @JakeWharton. Those links are helpful.
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Rewrote the same in Kotlin.
https://gist.github.com/captswag/7a6f077c2a0eea52e047fa275aeb67ec