162

I was wondering if there is an easy way to strike text within an app widget in Android. In a normal activity, it is pretty easy, using textview flags:

textView.setPaintFlags(textView.getPaintFlags() | Paint.STRIKE_THRU_TEXT_FLAG);

But since in an app widget, I can use only remoteviews... I do not know if this is possible

Anyone know something about this?

Thanks!

Antonio
  • 3,128
  • 2
  • 22
  • 14

21 Answers21

215

To do it programatically in a textview, untested in other views >>

TextView tv = (TextView) findViewById(R.id.mytext);
tv.setText("This is strike-thru");
tv.setPaintFlags(tv.getPaintFlags() | Paint.STRIKE_THRU_TEXT_FLAG);
ruhalde
  • 3,521
  • 3
  • 24
  • 28
  • 14
    This is not possible for App-Widgets. They only work with RemoteViews and there is no method to read the paint flags. – Ridcully Mar 17 '14 at 17:53
88

Another way to do it programmatically which looks a bit less like a hack than the Paint way:

Instead of doing:

tv.setText(s);

do:

private static final StrikethroughSpan STRIKE_THROUGH_SPAN = new StrikethroughSpan();
...
tv.setText(s, TextView.BufferType.SPANNABLE);
Spannable spannable = (Spannable) tv.getText();
spannable.setSpan(STRIKE_THROUGH_SPAN, 0, s.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
BoD
  • 10,838
  • 6
  • 63
  • 59
  • Thanks works like a charm Specially when you need some part of TextView to have strikethrough line – Diljeet May 21 '13 at 19:37
  • +1 Solution with Spannable worked, thank you so much. You saved my day. Thanks man ;) – Simon Dorociak Sep 01 '14 at 14:36
  • This does not answer the question as this does not apply to `RemoteView` – Alex.F Jan 22 '15 at 13:12
  • @Alex.F I am a bit surprised since this does work for instance in Notifications (which are remote views). I must admit I haven't tested with AppWidgets though. – BoD Jan 22 '15 at 17:19
  • 1
    It does work with App Widgets (any framework span that implements ParcelableSpan should work) – Karsten Apr 17 '15 at 01:18
  • this worked with recyclerView's adapter, unlike the other answer(@ruhalde) which didn't work for it so up vote this. – M. Erfan Mowlaei Aug 10 '16 at 23:26
75

You can use this:

remoteviews.setInt(R.id.YourTextView, "setPaintFlags", Paint.STRIKE_THRU_TEXT_FLAG | Paint.ANTI_ALIAS_FLAG);

Of course you can also add other flags from the android.graphics.Paint class.

agirardello
  • 2,895
  • 22
  • 22
  • 8
    Don't know how this hasn't get more votes, it perfectly answers the question. Thanks. – James Cross Jan 04 '15 at 21:26
  • 4
    First it is unclear what is remoteviews , plus doesn't work for all android versions. – akshat tailang Nov 28 '18 at 18:26
  • This is the answer we are looking for, working with app-widgets on androidTextViews are only accessible through removeViews. Accessing TextView directly is not possible at this point. – endevour Mar 19 '20 at 14:52
61

2015 Update: Folks, this is for very old versions of Android. See other answers for modern solutions!


To strike through the entire text view, you can use a specific background image to simulate the strikethrough effect:

android:background="@drawable/bg_strikethrough"

Where the bg_strikethrough drawable is a 9-patch that keeps a solid line through the middle, growing either side, with however much padding you think is reasonable. I've used one like this:

alt text

(enlarged for clarity.. 1300% !)

alt text

That is my HDPI version, so save it (the first one https://i.stack.imgur.com/nt6BK.png) as res/drawable-hdpi/bg_strikethrough.9.png and the configuration will work as so:

alt text

ohhorob
  • 11,695
  • 7
  • 41
  • 50
40

For doing this you can use this

 title.setPaintFlags(title.getPaintFlags() | Paint.STRIKE_THRU_TEXT_FLAG);

and for remove you can use this

 title.setPaintFlags(title.getPaintFlags() & (~Paint.STRIKE_THRU_TEXT_FLAG));
pavel
  • 1,603
  • 22
  • 19
26

Here is an extension for all you Kotlin folks

fun TextView.showStrikeThrough(show: Boolean) {
    paintFlags =
            if (show) paintFlags or Paint.STRIKE_THRU_TEXT_FLAG
            else paintFlags and Paint.STRIKE_THRU_TEXT_FLAG.inv()
}

Usage

textView.showStrikeThrough(true)

Limitation

Strikethroughs can only be the same color as the text

(i.e. Red text and blue strikethrough is not possible)

Michael
  • 9,639
  • 3
  • 64
  • 69
11

It is really easy if you are using strings:

<string name="line"> Not crossed <strike> crossed </strike> </string>

And then just:

<TextView 
        ...
         android:text="@string/line"
 />
Ignacio Alorre
  • 7,307
  • 8
  • 57
  • 94
  • 5
    @war_Hero Worked for me, doesn't render in Android Studio's renderer though. Had to actually build it to see it. – Mic Fok Jan 25 '16 at 14:39
9

For multiline TextView you should use android.text.style.CharacterStyle like this:

SpannableString spannable = new SpannableString(text);
spannable.setSpan(new StrikethroughSpan(), 0, text.length(), Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
remoteViews.setTextViewText(R.id.itemText, spannable);
user3768270
  • 99
  • 1
  • 2
9

Make a drawable file, striking_text.xml

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:state_pressed="false">
        <shape android:shape="line">
            <stroke android:width="1dp" android:color="@android:color/holo_red_dark" />
        </shape>
    </item>
</selector>

layout xml

 <TextView
            ....
            ....
            android:background="@drawable/striking_text"
            android:foreground="@drawable/striking_text"
            android:text="69$"
           />

If your min SDK below API level 23 just use the background or just put the background and foreground in the Textview then the android studio will show an error that says create a layout file for API >23 after that remove the android:foreground="@drawable/stricking_text" from the main layout

Output look like this: enter image description here

d-feverx
  • 1,424
  • 3
  • 16
  • 31
8

Add the line below:-

TextView tv=(TextView) v.findViewById(android.R.id.text1);
tv.setPaintFlags(tv.getPaintFlags() | Paint.STRIKE_THRU_TEXT_FLAG);

use your reference instead of "tv"

Prafulla Kumar Sahu
  • 9,321
  • 11
  • 68
  • 105
  • 1
    This does not reference the question, he's asking about a RemoteViews object in an AppWidget. – Matthew Dec 27 '15 at 02:21
6

try this code

textview.setText((Html.fromHtml("<strike>hello world!</strike>")));
Abhishek c
  • 539
  • 6
  • 16
3

Kotlin way

val tv = findViewById<View>(R.id.mytext) as TextView
tv.text = "This is strike-through"
tv.paintFlags = tv.paintFlags or Paint.STRIKE_THRU_TEXT_FLAG
Mujahid Khan
  • 1,712
  • 1
  • 18
  • 24
2

For Lollipop and above. create a drawable

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:state_pressed="false">
        <shape android:shape="line">
            <stroke android:width="1dp"
                android:color="@color/onePlusRed" />
        </shape>
    </item>
</selector>

and use it as foreground.
android:foreground="@drawable/strike_through"

Akshay More
  • 416
  • 4
  • 9
2

Android resources have pretty good HTML markup support
The below HTML elements are supported:

Bold: <b>, <em>
Italic: <i>, <cite>, <dfn>
25% larger text: <big>
20% smaller text: <small>
Setting font properties: <font face=”font_family“ color=”hex_color”>. Examples of possible font families include monospace, serif, and sans_serif.
Setting a monospace font family: <tt>
Strikethrough: <s>, <strike>, <del>
Underline: <u>
Superscript: <sup>
Subscript: <sub>
Bullet points: <ul>, <li>
Line breaks: <br>
Division: <div>
CSS style: <span style=”color|background_color|text-decoration”>
Paragraphs: <p dir=”rtl | ltr” style=”…”>

Note however that it's not rendered in android studio layouts preview. Last tested on Android Studio 3.3.1

For example, the Strikethrough will look like that:

<string name="cost"><strike>$10</strike> $5 a month</string>
Ilya Gazman
  • 31,250
  • 24
  • 137
  • 216
1

You Can use data binding! In XML layout just do app:strike="@{true}"

@BindingAdapter("strike")
fun bindTextView(view: TextView, strike: Boolean) {
    if (strike) {
        view.paintFlags = view.paintFlags or Paint.STRIKE_THRU_TEXT_FLAG or Paint.ANTI_ALIAS_FLAG
    } else {
        view.paintFlags = 0 or Paint.ANTI_ALIAS_FLAG
    }
}
Papa Yev
  • 618
  • 7
  • 10
1

Kotlin way

averagePrice.paintFlags = Paint.STRIKE_THRU_TEXT_FLAG

  • You should not set flag but add it to existing, because there can be other flags set (for antialiasing, for example) – artem May 21 '23 at 11:22
0

you add in :

TextView variableTv = (TextView) findViewById(R.id.yourText);

you set/add in You variable :

variableTv.setText("It's Text use Style Strike");

and then add .setPaintFlags in variableTv :

variableTv.setPaintFlags(variableTv.getPaintFlags() | Paint.STRIKE_THRU_TEXT_FLAG);
Vivek Mishra
  • 5,669
  • 9
  • 46
  • 84
0

I tried few options but, this works best for me:

String text = "<strike><font color=\'#757575\'>Some text</font></strike>";
textview.setText(Html.fromHtml(text));

cheers

0

Another solution as a Kotlin extension:

var TextView.isStrikeThrough: Boolean
    get() = (paintFlags and Paint.STRIKE_THRU_TEXT_FLAG) == Paint.STRIKE_THRU_TEXT_FLAG
    set(value) {
        paintFlags = if (value) {
            paintFlags or Paint.STRIKE_THRU_TEXT_FLAG
        } else {
            paintFlags and Paint.STRIKE_THRU_TEXT_FLAG.inv()
        }
    }

artem
  • 16,382
  • 34
  • 113
  • 189
-1

I've done this on a regular (local) TextView, and it should work on the remote variety since the docs list the method as equivalent between the two:

remote_text_view.setText(Html.fromHtml("This is <del>crossed off</del>."));

Blrfl
  • 6,817
  • 1
  • 25
  • 25
-1

I know, I am answering an old question, it might be helpful for someone else, for striking out a particular portion of TextView programmatically.

TextView textView = (TextView) findViewById(R.id.textView);
textView.setText("Text need to be set here", TextView.BufferType.SPANNABLE);
Spannable spannable = (Spannable) textView.getText();
spannable.setSpan(new StrikethroughSpan(), 5, 8, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);

So, in this example code, 5 is starting position and 8 is ending position of text need to be stricken out, so after running this code we can get the text "need" with striked out. Hope it will helpful for someone else.

Ram Koti
  • 2,203
  • 7
  • 26
  • 36