1 /*
2  * Copyright (C) 2006 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package android.text.style;
18 
19 import android.annotation.DrawableRes;
20 import android.annotation.NonNull;
21 import android.annotation.Nullable;
22 import android.compat.annotation.UnsupportedAppUsage;
23 import android.content.Context;
24 import android.graphics.Bitmap;
25 import android.graphics.BitmapFactory;
26 import android.graphics.drawable.BitmapDrawable;
27 import android.graphics.drawable.Drawable;
28 import android.net.Uri;
29 import android.util.Log;
30 
31 import java.io.InputStream;
32 
33 /**
34  * Span that replaces the text it's attached to with a {@link Drawable} that can be aligned with
35  * the bottom or with the baseline of the surrounding text. The drawable can be constructed from
36  * varied sources:
37  * <ul>
38  * <li>{@link Bitmap} - see {@link #ImageSpan(Context, Bitmap)} and
39  * {@link #ImageSpan(Context, Bitmap, int)}
40  * </li>
41  * <li>{@link Drawable} - see {@link #ImageSpan(Drawable, int)}</li>
42  * <li>resource id - see {@link #ImageSpan(Context, int, int)}</li>
43  * <li>{@link Uri} - see {@link #ImageSpan(Context, Uri, int)}</li>
44  * </ul>
45  * The default value for the vertical alignment is {@link DynamicDrawableSpan#ALIGN_BOTTOM}
46  * <p>
47  * For example, an <code>ImagedSpan</code> can be used like this:
48  * <pre>
49  * SpannableString string = new SpannableString("Bottom: span.\nBaseline: span.");
50  * // using the default alignment: ALIGN_BOTTOM
51  * string.setSpan(new ImageSpan(this, R.mipmap.ic_launcher), 7, 8, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
52  * string.setSpan(new ImageSpan(this, R.mipmap.ic_launcher, DynamicDrawableSpan.ALIGN_BASELINE),
53  * 22, 23, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
54  * </pre>
55  * <img src="{@docRoot}reference/android/images/text/style/imagespan.png" />
56  * <figcaption>Text with <code>ImageSpan</code>s aligned bottom and baseline.</figcaption>
57  */
58 public class ImageSpan extends DynamicDrawableSpan {
59 
60     @Nullable
61     @UnsupportedAppUsage
62     private Drawable mDrawable;
63     @Nullable
64     private Uri mContentUri;
65     @DrawableRes
66     private int mResourceId;
67     @Nullable
68     private Context mContext;
69     @Nullable
70     private String mSource;
71 
72     /**
73      * @deprecated Use {@link #ImageSpan(Context, Bitmap)} instead.
74      */
75     @Deprecated
ImageSpan(@onNull Bitmap b)76     public ImageSpan(@NonNull Bitmap b) {
77         this(null, b, ALIGN_BOTTOM);
78     }
79 
80     /**
81      * @deprecated Use {@link #ImageSpan(Context, Bitmap, int)} instead.
82      */
83     @Deprecated
ImageSpan(@onNull Bitmap b, int verticalAlignment)84     public ImageSpan(@NonNull Bitmap b, int verticalAlignment) {
85         this(null, b, verticalAlignment);
86     }
87 
88     /**
89      * Constructs an {@link ImageSpan} from a {@link Context} and a {@link Bitmap} with the default
90      * alignment {@link DynamicDrawableSpan#ALIGN_BOTTOM}
91      *
92      * @param context context used to create a drawable from {@param bitmap} based on the display
93      *                metrics of the resources
94      * @param bitmap  bitmap to be rendered
95      */
ImageSpan(@onNull Context context, @NonNull Bitmap bitmap)96     public ImageSpan(@NonNull Context context, @NonNull Bitmap bitmap) {
97         this(context, bitmap, ALIGN_BOTTOM);
98     }
99 
100     /**
101      * Constructs an {@link ImageSpan} from a {@link Context}, a {@link Bitmap} and a vertical
102      * alignment.
103      *
104      * @param context           context used to create a drawable from {@param bitmap} based on
105      *                          the display metrics of the resources
106      * @param bitmap            bitmap to be rendered
107      * @param verticalAlignment one of {@link DynamicDrawableSpan#ALIGN_BOTTOM} or
108      *                          {@link DynamicDrawableSpan#ALIGN_BASELINE}
109      */
ImageSpan(@onNull Context context, @NonNull Bitmap bitmap, int verticalAlignment)110     public ImageSpan(@NonNull Context context, @NonNull Bitmap bitmap, int verticalAlignment) {
111         super(verticalAlignment);
112         mContext = context;
113         mDrawable = context != null
114                 ? new BitmapDrawable(context.getResources(), bitmap)
115                 : new BitmapDrawable(bitmap);
116         int width = mDrawable.getIntrinsicWidth();
117         int height = mDrawable.getIntrinsicHeight();
118         mDrawable.setBounds(0, 0, width > 0 ? width : 0, height > 0 ? height : 0);
119     }
120 
121     /**
122      * Constructs an {@link ImageSpan} from a drawable with the default
123      * alignment {@link DynamicDrawableSpan#ALIGN_BOTTOM}.
124      *
125      * @param drawable drawable to be rendered
126      */
ImageSpan(@onNull Drawable drawable)127     public ImageSpan(@NonNull Drawable drawable) {
128         this(drawable, ALIGN_BOTTOM);
129     }
130 
131     /**
132      * Constructs an {@link ImageSpan} from a drawable and a vertical alignment.
133      *
134      * @param drawable          drawable to be rendered
135      * @param verticalAlignment one of {@link DynamicDrawableSpan#ALIGN_BOTTOM} or
136      *                          {@link DynamicDrawableSpan#ALIGN_BASELINE}
137      */
ImageSpan(@onNull Drawable drawable, int verticalAlignment)138     public ImageSpan(@NonNull Drawable drawable, int verticalAlignment) {
139         super(verticalAlignment);
140         mDrawable = drawable;
141     }
142 
143     /**
144      * Constructs an {@link ImageSpan} from a drawable and a source with the default
145      * alignment {@link DynamicDrawableSpan#ALIGN_BOTTOM}
146      *
147      * @param drawable drawable to be rendered
148      * @param source   drawable's Uri source
149      */
ImageSpan(@onNull Drawable drawable, @NonNull String source)150     public ImageSpan(@NonNull Drawable drawable, @NonNull String source) {
151         this(drawable, source, ALIGN_BOTTOM);
152     }
153 
154     /**
155      * Constructs an {@link ImageSpan} from a drawable, a source and a vertical alignment.
156      *
157      * @param drawable          drawable to be rendered
158      * @param source            drawable's uri source
159      * @param verticalAlignment one of {@link DynamicDrawableSpan#ALIGN_BOTTOM} or
160      *                          {@link DynamicDrawableSpan#ALIGN_BASELINE}
161      */
ImageSpan(@onNull Drawable drawable, @NonNull String source, int verticalAlignment)162     public ImageSpan(@NonNull Drawable drawable, @NonNull String source, int verticalAlignment) {
163         super(verticalAlignment);
164         mDrawable = drawable;
165         mSource = source;
166     }
167 
168     /**
169      * Constructs an {@link ImageSpan} from a {@link Context} and a {@link Uri} with the default
170      * alignment {@link DynamicDrawableSpan#ALIGN_BOTTOM}. The Uri source can be retrieved via
171      * {@link #getSource()}
172      *
173      * @param context context used to create a drawable from {@param bitmap} based on the display
174      *                metrics of the resources
175      * @param uri     {@link Uri} used to construct the drawable that will be rendered
176      */
ImageSpan(@onNull Context context, @NonNull Uri uri)177     public ImageSpan(@NonNull Context context, @NonNull Uri uri) {
178         this(context, uri, ALIGN_BOTTOM);
179     }
180 
181     /**
182      * Constructs an {@link ImageSpan} from a {@link Context}, a {@link Uri} and a vertical
183      * alignment. The Uri source can be retrieved via {@link #getSource()}
184      *
185      * @param context           context used to create a drawable from {@param bitmap} based on
186      *                          the display
187      *                          metrics of the resources
188      * @param uri               {@link Uri} used to construct the drawable that will be rendered.
189      * @param verticalAlignment one of {@link DynamicDrawableSpan#ALIGN_BOTTOM} or
190      *                          {@link DynamicDrawableSpan#ALIGN_BASELINE}
191      */
ImageSpan(@onNull Context context, @NonNull Uri uri, int verticalAlignment)192     public ImageSpan(@NonNull Context context, @NonNull Uri uri, int verticalAlignment) {
193         super(verticalAlignment);
194         mContext = context;
195         mContentUri = uri;
196         mSource = uri.toString();
197     }
198 
199     /**
200      * Constructs an {@link ImageSpan} from a {@link Context} and a resource id with the default
201      * alignment {@link DynamicDrawableSpan#ALIGN_BOTTOM}
202      *
203      * @param context    context used to retrieve the drawable from resources
204      * @param resourceId drawable resource id based on which the drawable is retrieved
205      */
ImageSpan(@onNull Context context, @DrawableRes int resourceId)206     public ImageSpan(@NonNull Context context, @DrawableRes int resourceId) {
207         this(context, resourceId, ALIGN_BOTTOM);
208     }
209 
210     /**
211      * Constructs an {@link ImageSpan} from a {@link Context}, a resource id and a vertical
212      * alignment.
213      *
214      * @param context           context used to retrieve the drawable from resources
215      * @param resourceId        drawable resource id based on which the drawable is retrieved.
216      * @param verticalAlignment one of {@link DynamicDrawableSpan#ALIGN_BOTTOM} or
217      *                          {@link DynamicDrawableSpan#ALIGN_BASELINE}
218      */
ImageSpan(@onNull Context context, @DrawableRes int resourceId, int verticalAlignment)219     public ImageSpan(@NonNull Context context, @DrawableRes int resourceId,
220             int verticalAlignment) {
221         super(verticalAlignment);
222         mContext = context;
223         mResourceId = resourceId;
224     }
225 
226     @Override
getDrawable()227     public Drawable getDrawable() {
228         Drawable drawable = null;
229 
230         if (mDrawable != null) {
231             drawable = mDrawable;
232         } else if (mContentUri != null) {
233             Bitmap bitmap = null;
234             try {
235                 InputStream is = mContext.getContentResolver().openInputStream(
236                         mContentUri);
237                 bitmap = BitmapFactory.decodeStream(is);
238                 drawable = new BitmapDrawable(mContext.getResources(), bitmap);
239                 drawable.setBounds(0, 0, drawable.getIntrinsicWidth(),
240                         drawable.getIntrinsicHeight());
241                 is.close();
242             } catch (Exception e) {
243                 Log.e("ImageSpan", "Failed to loaded content " + mContentUri, e);
244             }
245         } else {
246             try {
247                 drawable = mContext.getDrawable(mResourceId);
248                 drawable.setBounds(0, 0, drawable.getIntrinsicWidth(),
249                         drawable.getIntrinsicHeight());
250             } catch (Exception e) {
251                 Log.e("ImageSpan", "Unable to find resource: " + mResourceId);
252             }
253         }
254 
255         return drawable;
256     }
257 
258     /**
259      * Returns the source string that was saved during construction.
260      *
261      * @return the source string that was saved during construction
262      * @see #ImageSpan(Drawable, String)
263      * @see #ImageSpan(Context, Uri)
264      */
265     @Nullable
getSource()266     public String getSource() {
267         return mSource;
268     }
269 
270     @Override
toString()271     public String toString() {
272         return "ImageSpan{"
273                 + "drawable=" + getDrawable()
274                 + ", source='" + getSource() + '\''
275                 + ", verticalAlignment=" + getVerticalAlignment()
276                 + '}';
277     }
278 }
279