Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/grivos/spanomatic
Automatically add spans to text from Android resources strings
https://github.com/grivos/spanomatic
android android-library kotlin-android spannable
Last synced: 4 months ago
JSON representation
Automatically add spans to text from Android resources strings
- Host: GitHub
- URL: https://github.com/grivos/spanomatic
- Owner: grivos
- License: apache-2.0
- Created: 2019-04-27T16:01:52.000Z (almost 6 years ago)
- Default Branch: master
- Last Pushed: 2021-06-08T19:57:07.000Z (over 3 years ago)
- Last Synced: 2024-10-31T15:38:07.690Z (4 months ago)
- Topics: android, android-library, kotlin-android, spannable
- Language: Kotlin
- Size: 523 KB
- Stars: 131
- Watchers: 3
- Forks: 5
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# Spanomatic
Spanomatic is an Android library that allows you to automatically add spans to text from resources strings.* [Getting Started](#getting-started)
* [Dependencies](#dependencies)
* [Initialization](#initialization)
* [String Resources Annotations](#string-resources-annotations)
* [Supported Annotation Keys](#supported-annotation-keys)
* [fgColor](#fgcolor)
* [bgColor](#bgcolor)
* [relativeTextSize](#relativetextsize)
* [absoluteTextSize](#absolutetextsize)
* [drawable](#drawable)
* [format](#format)
* [quote](#quote)
* [bullet](#bullet)
* [url](#url)
* [leadingMargin](#leadingmargin)
* [typeface](#typeface)
* [click](#click)
* [Adding Support For Custom User Annotation Keys](#adding-support-for-custom-user-annotation-keys)
* [Using Spanomatic Manually](#using-spanomatic-manually)
* [Format Arguments](#format-arguments)
* [Acknowledgements](#acknowledgements)## Getting Started
### Dependencies
Add Jitpack in your root build.gradle:
```groovy
allprojects {
repositories {
...
maven { url 'https://jitpack.io' }
}
}
```
Spanomatic uses the [ViewPump](https://github.com/InflationX/ViewPump) library, that provides an API for pre/post-inflation interceptors.
Add these dependencies to your app module's build.gradle file:
```groovy
dependencies {
...
implementation 'com.github.grivos:Spanomatic:1.2.1'
implementation 'io.github.inflationx:viewpump:2.0.3'
}
```### Initialization
Init ViewPump and add the SpanomaticInterceptor in your app class:
```kotlin
class MyApplication : Application() {override fun onCreate() {
super.onCreate()
ViewPump.init(
ViewPump.builder()
.addInterceptor(
SpanomaticInterceptor()
)
.build()
)
}
}
```Wrap your activities context:
```kotlin
class BaseActivity : AppCompatActivity() {override fun attachBaseContext(newBase: Context) {
super.attachBaseContext(ViewPumpContextWrapper.wrap(newBase))
}}
```## String Resources Annotations
In order for Spanomatic to translate string resources annotations into spans, you need to wrap your strings in special annotations.
The mechanic is like so:
```xml
This part isn\'t annotated, but this part is.
```
Spanomatic parses the string resource, looks for these annotations, and depending on the key and value, it adds the appropriate spans.
You can also nest annotations like so:
```xml
This part isn\'t annotated, but this part has two annotations.
```Spanomatic supports several annotation keys out of the box, but you can also register handlers for custom annotation keys.
### Supported Annotation Keys
#### fgColor
**Description:** Adds a `ForegroundSpan`.
**Possible Values:** Either a literal color hex (`"#9C27B0"`) or a color reference (`"@color/material_purple"`).
**Example:**
```xml
This text has purple foreground
```
data:image/s3,"s3://crabby-images/0839d/0839dbc261840a9083ad7f0fbae45a416dfc502b" alt="example_fg"#### bgColor
**Description:** Adds a `BackgroundSpan`.
**Possible Values:** Either a literal color hex or a color reference.
**Example:**
```xml
This text has green background
```
data:image/s3,"s3://crabby-images/2cb1e/2cb1efac9e6a4ee9894db0e2948b2cf5a98ffb42" alt="example_bg"#### relativeTextSize
**Description:** Adds a `RelativeSizeSpan`.
**Possible Values:** The size multiplier.
**Example:**
```xml
This text is two times bigger
```
data:image/s3,"s3://crabby-images/4c9b6/4c9b6257d029864f2479fb64f2dd3de56877201d" alt="example_relative_size"#### absoluteTextSize
**Description:** Adds an `AbsoluteSizeSpan`.
**Possible Values:** A size identifier. Either a literal (`18dp`, `16sp`, etc.) or a reference (`"@dimen/big_text_size"`).
**Example:**
```xml
This text size is 20dp
```
data:image/s3,"s3://crabby-images/39481/39481ce2ae79c919da6174f914ede5249eabeb0b" alt="example_absolute_size"#### drawable
**Description:** Adds an `ImageSpan`.
**Possible Values:** A drawable reference and an optional alignment (either `baseline` or `bottom`). If the alignment is missing, the default is `baseline`.
The optional alignment is separated from the drawable reference using the pipe character (`|`).
**Example:**
```xml
This is a cake drawable span.
```
data:image/s3,"s3://crabby-images/9e01c/9e01c4ddf09560353849bce86d0e5e242075e506" alt="example_drawable"#### format
**Description:** Adds one of `StrikethroughSpan`, `StyleSpan`, `SuperscriptSpan`, `SubscriptSpan`, `UnderlineSpan`.
**Possible Values:**| Value | Span |
|---|---|
| strikethrough | `StrikethroughSpan()` |
| bold | `StyleSpan(Typeface.BOLD)` |
| italic | `StyleSpan(Typeface.ITALIC)` |
| boldItalic | `StyleSpan(Typeface.BOLD_ITALIC)` |
| superscript | `SuperscriptSpa()` |
| subscript | `SubscriptSpan()` |
| underline | `UnderlineSpan()` |**Example:**
```xml
Spanomatic supports strikethrough, bold, italic, bold italic, superscript, subscript and underline spans.
```
data:image/s3,"s3://crabby-images/3bda4/3bda405be273de2b84f932803b328a396501b0db" alt="example_format"#### quote
**Description:** Adds a `QuoteSpan`.
**Possible Values:** Either empty or an optional literal color hex or a color reference.
**Example:**
```xml
This is a quote
```
```xml
This is a quote
```
data:image/s3,"s3://crabby-images/a1aa6/a1aa62d5a87ac33e3e3560e3b4a2ac1e82ea9145" alt="example_quote"#### bullet
**Description:** Adds a `BulletSpan`.
**Possible Values:** Either empty or an optional gap width (literal dimension or a dimension reference), and an optional literal color hex or a color reference.
**Example:**
```xml
This is a bullet
```
```xml
This is a bullet
```
```xml
This is a bullet
```
data:image/s3,"s3://crabby-images/952ea/952ea7bc0eb19afb37383e554d3437afc9d594a6" alt="example_bullet"#### url
**Description:** Adds a `URLSpan`.
**Possible Values:** A url.
**Example:**
```xml
This text has a UrlSpan
```
data:image/s3,"s3://crabby-images/3dd8c/3dd8ccd130c46ea76b16b7566f83a9de67ca9d8a" alt="example_url"#### leadingMargin
**Description:** Adds a `LeadingMarginSpan`.
**Possible Values:** The size of the leading margin (either a literal dimension or a dimension reference).
**Example:**
```xml
This text has a leading margin
```
data:image/s3,"s3://crabby-images/ebb61/ebb612436aee406131e6e808fe620e2870c664af" alt="example_leading_margin"#### typeface
**Description:** Adds a `CustomTypefaceSpan`.
**Possible Values:** The name of this typeface (either a literal string or a string reference).
**Example:**
First, you need to register a `TypefaceProvider` (the best place to do so is in your Application `onCreate()` method):
```kotlin
val latoRegular = ResourcesCompat.getFont(this, R.font.lato_regular)
val latoLight = ResourcesCompat.getFont(this, R.font.lato_light)
val latoBold = ResourcesCompat.getFont(this, R.font.lato_bold)
Spanomatic.typefaceProvider = object : TypefaceProvider {override fun getTypeface(name: String): Typeface? {
return when (name) {
"latoRegular" -> latoRegular
"latoLight" -> latoLight
"latoBold" -> latoBold
else -> null
}
}}
```
Then we can use these typefaces in a string resource:```xml
Here we have three different typefaces: Lato Regular, Lato Light and Lato Bold
```
data:image/s3,"s3://crabby-images/1a38d/1a38d4d8df9a4ee09c11780389418664dd7f0434" alt="example_typeface"#### click
**Description:** Adds a `ListenableClickableSpan` (a custom `ClickableSpan` that can be registered with a click callback).
Note that this span doesn't add an `UnderlineSpan`, so if that's what you want, you should wrap it in another `format="underline"` annotation.
**Possible Values:** The name of this click (so we can register a callback to the span from the code).
**Example:**```xml
This span is clickable: click me
```To catch the onClick event, you need to register a callback like so:
```kotlin
clickableTextView.addSpanClickListener("click1") {
Toast.makeText(this, "Span was clicked", Toast.LENGTH_SHORT).show()
}
```
data:image/s3,"s3://crabby-images/2ea9a/2ea9a22dad11d0ff4337e195c8c756ff1423d7f3" alt="example_click_toast"### Adding Support For Custom User Annotation Keys
You can register your own annotation span handlers like so:
```kotlin
Spanomatic.setAnnotationSpanHandler("myCustomKey") { annotationValue, context ->
val myParsedValue = parseValue(annotationValue, context)
MyCustomSpan(myParsedValue)
}
```Then you can use it in your string resources:
```xml
This span is custom
```## Using Spanomatic Manually
Spanomatic automatically adds spans in the layout inflation process, but you can also use it manually like so:
```kotlin
val spanned = addSpansFromAnnotations(R.string.annotated_string)
myTextView.text = spanned
```### Format Arguments
Spanomatic also supports format arguments:
```xml
This text has annotations with parameters: %1$s, %2$b, %3$d
``````kotlin
val param1 = "text"
val param2 = false
val param3 = 5
val spanned = addSpansFromAnnotations(R.string.annotated_string_with_parameters, param1, param2, param3)
myTextView.text = spanned
```
## AcknowledgementsSpanomatic was inspired by [this blog post](https://medium.com/androiddevelopers/styling-internationalized-text-in-android-f99759fb7b8f) by Florina Muntenescu and the [Rialto](https://github.com/StylingAndroid/Rialto) library by Mark Allison.