1 /* 2 * Copyright (C) 2008 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.Nullable; 20 import android.content.Context; 21 import android.content.res.ColorStateList; 22 import android.content.res.TypedArray; 23 import android.graphics.LeakyTypefaceStorage; 24 import android.graphics.Typeface; 25 import android.graphics.fonts.FontStyle; 26 import android.os.LocaleList; 27 import android.os.Parcel; 28 import android.text.ParcelableSpan; 29 import android.text.TextPaint; 30 import android.text.TextUtils; 31 32 /** 33 * Sets the text appearance using the given 34 * {@link android.R.styleable#TextAppearance TextAppearance} attributes. 35 * By default {@link TextAppearanceSpan} only changes the specified attributes in XML. 36 * {@link android.R.styleable#TextAppearance_textColorHighlight textColorHighlight}, 37 * {@link android.R.styleable#TextAppearance_textColorHint textColorHint}, 38 * {@link android.R.styleable#TextAppearance_textAllCaps textAllCaps} and 39 * {@link android.R.styleable#TextAppearance_fallbackLineSpacing fallbackLineSpacing} 40 * are not supported by {@link TextAppearanceSpan}. 41 * 42 * {@see android.widget.TextView#setTextAppearance(int)} 43 * 44 * @attr ref android.R.styleable#TextAppearance_fontFamily 45 * @attr ref android.R.styleable#TextAppearance_textColor 46 * @attr ref android.R.styleable#TextAppearance_textColorLink 47 * @attr ref android.R.styleable#TextAppearance_textFontWeight 48 * @attr ref android.R.styleable#TextAppearance_textSize 49 * @attr ref android.R.styleable#TextAppearance_textStyle 50 * @attr ref android.R.styleable#TextAppearance_typeface 51 * @attr ref android.R.styleable#TextAppearance_shadowColor 52 * @attr ref android.R.styleable#TextAppearance_shadowDx 53 * @attr ref android.R.styleable#TextAppearance_shadowDy 54 * @attr ref android.R.styleable#TextAppearance_shadowRadius 55 * @attr ref android.R.styleable#TextAppearance_elegantTextHeight 56 * @attr ref android.R.styleable#TextAppearance_letterSpacing 57 * @attr ref android.R.styleable#TextAppearance_fontFeatureSettings 58 * @attr ref android.R.styleable#TextAppearance_fontVariationSettings 59 * 60 */ 61 public class TextAppearanceSpan extends MetricAffectingSpan implements ParcelableSpan { 62 private final String mFamilyName; 63 private final int mStyle; 64 private final int mTextSize; 65 private final ColorStateList mTextColor; 66 private final ColorStateList mTextColorLink; 67 private final Typeface mTypeface; 68 69 private final int mTextFontWeight; 70 private final LocaleList mTextLocales; 71 72 private final float mShadowRadius; 73 private final float mShadowDx; 74 private final float mShadowDy; 75 private final int mShadowColor; 76 77 private final boolean mHasElegantTextHeight; 78 private final boolean mElegantTextHeight; 79 private final boolean mHasLetterSpacing; 80 private final float mLetterSpacing; 81 82 private final String mFontFeatureSettings; 83 private final String mFontVariationSettings; 84 85 /** 86 * Uses the specified TextAppearance resource to determine the 87 * text appearance. The <code>appearance</code> should be, for example, 88 * <code>android.R.style.TextAppearance_Small</code>. 89 */ TextAppearanceSpan(Context context, int appearance)90 public TextAppearanceSpan(Context context, int appearance) { 91 this(context, appearance, -1); 92 } 93 94 /** 95 * Uses the specified TextAppearance resource to determine the 96 * text appearance, and the specified text color resource 97 * to determine the color. The <code>appearance</code> should be, 98 * for example, <code>android.R.style.TextAppearance_Small</code>, 99 * and the <code>colorList</code> should be, for example, 100 * <code>android.R.styleable.Theme_textColorPrimary</code>. 101 */ TextAppearanceSpan(Context context, int appearance, int colorList)102 public TextAppearanceSpan(Context context, int appearance, int colorList) { 103 ColorStateList textColor; 104 105 TypedArray a = 106 context.obtainStyledAttributes(appearance, 107 com.android.internal.R.styleable.TextAppearance); 108 109 textColor = a.getColorStateList(com.android.internal.R.styleable. 110 TextAppearance_textColor); 111 mTextColorLink = a.getColorStateList(com.android.internal.R.styleable. 112 TextAppearance_textColorLink); 113 mTextSize = a.getDimensionPixelSize(com.android.internal.R.styleable. 114 TextAppearance_textSize, -1); 115 116 mStyle = a.getInt(com.android.internal.R.styleable.TextAppearance_textStyle, 0); 117 if (!context.isRestricted() && context.canLoadUnsafeResources()) { 118 mTypeface = a.getFont(com.android.internal.R.styleable.TextAppearance_fontFamily); 119 } else { 120 mTypeface = null; 121 } 122 if (mTypeface != null) { 123 mFamilyName = null; 124 } else { 125 String family = a.getString(com.android.internal.R.styleable.TextAppearance_fontFamily); 126 if (family != null) { 127 mFamilyName = family; 128 } else { 129 int tf = a.getInt(com.android.internal.R.styleable.TextAppearance_typeface, 0); 130 131 switch (tf) { 132 case 1: 133 mFamilyName = "sans"; 134 break; 135 136 case 2: 137 mFamilyName = "serif"; 138 break; 139 140 case 3: 141 mFamilyName = "monospace"; 142 break; 143 144 default: 145 mFamilyName = null; 146 break; 147 } 148 } 149 } 150 151 mTextFontWeight = a.getInt(com.android.internal.R.styleable 152 .TextAppearance_textFontWeight, /*defValue*/ FontStyle.FONT_WEIGHT_UNSPECIFIED); 153 154 final String localeString = a.getString(com.android.internal.R.styleable 155 .TextAppearance_textLocale); 156 if (localeString != null) { 157 LocaleList localeList = LocaleList.forLanguageTags(localeString); 158 if (!localeList.isEmpty()) { 159 mTextLocales = localeList; 160 } else { 161 mTextLocales = null; 162 } 163 } else { 164 mTextLocales = null; 165 } 166 167 mShadowRadius = a.getFloat(com.android.internal.R.styleable 168 .TextAppearance_shadowRadius, 0.0f); 169 mShadowDx = a.getFloat(com.android.internal.R.styleable 170 .TextAppearance_shadowDx, 0.0f); 171 mShadowDy = a.getFloat(com.android.internal.R.styleable 172 .TextAppearance_shadowDy, 0.0f); 173 mShadowColor = a.getInt(com.android.internal.R.styleable 174 .TextAppearance_shadowColor, 0); 175 176 mHasElegantTextHeight = a.hasValue(com.android.internal.R.styleable 177 .TextAppearance_elegantTextHeight); 178 mElegantTextHeight = a.getBoolean(com.android.internal.R.styleable 179 .TextAppearance_elegantTextHeight, false); 180 181 mHasLetterSpacing = a.hasValue(com.android.internal.R.styleable 182 .TextAppearance_letterSpacing); 183 mLetterSpacing = a.getFloat(com.android.internal.R.styleable 184 .TextAppearance_letterSpacing, 0.0f); 185 186 mFontFeatureSettings = a.getString(com.android.internal.R.styleable 187 .TextAppearance_fontFeatureSettings); 188 189 mFontVariationSettings = a.getString(com.android.internal.R.styleable 190 .TextAppearance_fontVariationSettings); 191 192 a.recycle(); 193 194 if (colorList >= 0) { 195 a = context.obtainStyledAttributes(com.android.internal.R.style.Theme, 196 com.android.internal.R.styleable.Theme); 197 198 textColor = a.getColorStateList(colorList); 199 a.recycle(); 200 } 201 202 mTextColor = textColor; 203 } 204 205 /** 206 * Makes text be drawn with the specified typeface, size, style, 207 * and colors. 208 */ TextAppearanceSpan(String family, int style, int size, ColorStateList color, ColorStateList linkColor)209 public TextAppearanceSpan(String family, int style, int size, 210 ColorStateList color, ColorStateList linkColor) { 211 mFamilyName = family; 212 mStyle = style; 213 mTextSize = size; 214 mTextColor = color; 215 mTextColorLink = linkColor; 216 mTypeface = null; 217 218 mTextFontWeight = FontStyle.FONT_WEIGHT_UNSPECIFIED; 219 mTextLocales = null; 220 221 mShadowRadius = 0.0f; 222 mShadowDx = 0.0f; 223 mShadowDy = 0.0f; 224 mShadowColor = 0; 225 226 mHasElegantTextHeight = false; 227 mElegantTextHeight = false; 228 mHasLetterSpacing = false; 229 mLetterSpacing = 0.0f; 230 231 mFontFeatureSettings = null; 232 mFontVariationSettings = null; 233 } 234 TextAppearanceSpan(Parcel src)235 public TextAppearanceSpan(Parcel src) { 236 mFamilyName = src.readString(); 237 mStyle = src.readInt(); 238 mTextSize = src.readInt(); 239 if (src.readInt() != 0) { 240 mTextColor = ColorStateList.CREATOR.createFromParcel(src); 241 } else { 242 mTextColor = null; 243 } 244 if (src.readInt() != 0) { 245 mTextColorLink = ColorStateList.CREATOR.createFromParcel(src); 246 } else { 247 mTextColorLink = null; 248 } 249 mTypeface = LeakyTypefaceStorage.readTypefaceFromParcel(src); 250 251 mTextFontWeight = src.readInt(); 252 mTextLocales = src.readParcelable(LocaleList.class.getClassLoader(), android.os.LocaleList.class); 253 254 mShadowRadius = src.readFloat(); 255 mShadowDx = src.readFloat(); 256 mShadowDy = src.readFloat(); 257 mShadowColor = src.readInt(); 258 259 mHasElegantTextHeight = src.readBoolean(); 260 mElegantTextHeight = src.readBoolean(); 261 mHasLetterSpacing = src.readBoolean(); 262 mLetterSpacing = src.readFloat(); 263 264 mFontFeatureSettings = src.readString(); 265 mFontVariationSettings = src.readString(); 266 } 267 getSpanTypeId()268 public int getSpanTypeId() { 269 return getSpanTypeIdInternal(); 270 } 271 272 /** @hide */ getSpanTypeIdInternal()273 public int getSpanTypeIdInternal() { 274 return TextUtils.TEXT_APPEARANCE_SPAN; 275 } 276 describeContents()277 public int describeContents() { 278 return 0; 279 } 280 writeToParcel(Parcel dest, int flags)281 public void writeToParcel(Parcel dest, int flags) { 282 writeToParcelInternal(dest, flags); 283 } 284 285 /** @hide */ writeToParcelInternal(Parcel dest, int flags)286 public void writeToParcelInternal(Parcel dest, int flags) { 287 dest.writeString(mFamilyName); 288 dest.writeInt(mStyle); 289 dest.writeInt(mTextSize); 290 if (mTextColor != null) { 291 dest.writeInt(1); 292 mTextColor.writeToParcel(dest, flags); 293 } else { 294 dest.writeInt(0); 295 } 296 if (mTextColorLink != null) { 297 dest.writeInt(1); 298 mTextColorLink.writeToParcel(dest, flags); 299 } else { 300 dest.writeInt(0); 301 } 302 LeakyTypefaceStorage.writeTypefaceToParcel(mTypeface, dest); 303 304 dest.writeInt(mTextFontWeight); 305 dest.writeParcelable(mTextLocales, flags); 306 307 dest.writeFloat(mShadowRadius); 308 dest.writeFloat(mShadowDx); 309 dest.writeFloat(mShadowDy); 310 dest.writeInt(mShadowColor); 311 312 dest.writeBoolean(mHasElegantTextHeight); 313 dest.writeBoolean(mElegantTextHeight); 314 dest.writeBoolean(mHasLetterSpacing); 315 dest.writeFloat(mLetterSpacing); 316 317 dest.writeString(mFontFeatureSettings); 318 dest.writeString(mFontVariationSettings); 319 } 320 321 /** 322 * Returns the typeface family specified by this span, or <code>null</code> 323 * if it does not specify one. 324 */ getFamily()325 public String getFamily() { 326 return mFamilyName; 327 } 328 329 /** 330 * Returns the text color specified by this span, or <code>null</code> 331 * if it does not specify one. 332 */ getTextColor()333 public ColorStateList getTextColor() { 334 return mTextColor; 335 } 336 337 /** 338 * Returns the link color specified by this span, or <code>null</code> 339 * if it does not specify one. 340 */ getLinkTextColor()341 public ColorStateList getLinkTextColor() { 342 return mTextColorLink; 343 } 344 345 /** 346 * Returns the text size specified by this span, or <code>-1</code> 347 * if it does not specify one. 348 */ getTextSize()349 public int getTextSize() { 350 return mTextSize; 351 } 352 353 /** 354 * Returns the text style specified by this span, or <code>0</code> 355 * if it does not specify one. 356 */ getTextStyle()357 public int getTextStyle() { 358 return mStyle; 359 } 360 361 /** 362 * Returns the text font weight specified by this span, or 363 * <code>FontStyle.FONT_WEIGHT_UNSPECIFIED</code> if it does not specify one. 364 */ getTextFontWeight()365 public int getTextFontWeight() { 366 return mTextFontWeight; 367 } 368 369 /** 370 * Returns the {@link android.os.LocaleList} specified by this span, or <code>null</code> 371 * if it does not specify one. 372 */ 373 @Nullable getTextLocales()374 public LocaleList getTextLocales() { 375 return mTextLocales; 376 } 377 378 /** 379 * Returns the typeface specified by this span, or <code>null</code> 380 * if it does not specify one. 381 */ 382 @Nullable getTypeface()383 public Typeface getTypeface() { 384 return mTypeface; 385 } 386 387 /** 388 * Returns the color of the text shadow specified by this span, or <code>0</code> 389 * if it does not specify one. 390 */ getShadowColor()391 public int getShadowColor() { 392 return mShadowColor; 393 } 394 395 /** 396 * Returns the horizontal offset of the text shadow specified by this span, or <code>0.0f</code> 397 * if it does not specify one. 398 */ getShadowDx()399 public float getShadowDx() { 400 return mShadowDx; 401 } 402 403 /** 404 * Returns the vertical offset of the text shadow specified by this span, or <code>0.0f</code> 405 * if it does not specify one. 406 */ getShadowDy()407 public float getShadowDy() { 408 return mShadowDy; 409 } 410 411 /** 412 * Returns the blur radius of the text shadow specified by this span, or <code>0.0f</code> 413 * if it does not specify one. 414 */ getShadowRadius()415 public float getShadowRadius() { 416 return mShadowRadius; 417 } 418 419 /** 420 * Returns the font feature settings specified by this span, or <code>null</code> 421 * if it does not specify one. 422 */ 423 @Nullable getFontFeatureSettings()424 public String getFontFeatureSettings() { 425 return mFontFeatureSettings; 426 } 427 428 /** 429 * Returns the font variation settings specified by this span, or <code>null</code> 430 * if it does not specify one. 431 */ 432 @Nullable getFontVariationSettings()433 public String getFontVariationSettings() { 434 return mFontVariationSettings; 435 } 436 437 /** 438 * Returns the value of elegant height metrics flag specified by this span, 439 * or <code>false</code> if it does not specify one. 440 */ isElegantTextHeight()441 public boolean isElegantTextHeight() { 442 return mElegantTextHeight; 443 } 444 445 /** 446 * Returns the value of letter spacing to be added in em unit. 447 * @return a letter spacing amount 448 */ getLetterSpacing()449 public float getLetterSpacing() { 450 return mLetterSpacing; 451 } 452 453 @Override updateDrawState(TextPaint ds)454 public void updateDrawState(TextPaint ds) { 455 updateMeasureState(ds); 456 457 if (mTextColor != null) { 458 ds.setColor(mTextColor.getColorForState(ds.drawableState, 0)); 459 } 460 461 if (mTextColorLink != null) { 462 ds.linkColor = mTextColorLink.getColorForState(ds.drawableState, 0); 463 } 464 465 if (mShadowColor != 0) { 466 ds.setShadowLayer(mShadowRadius, mShadowDx, mShadowDy, mShadowColor); 467 } 468 } 469 470 @Override updateMeasureState(TextPaint ds)471 public void updateMeasureState(TextPaint ds) { 472 final Typeface styledTypeface; 473 int style = 0; 474 475 if (mTypeface != null) { 476 style = mStyle; 477 styledTypeface = Typeface.create(mTypeface, style); 478 } else if (mFamilyName != null || mStyle != 0) { 479 Typeface tf = ds.getTypeface(); 480 481 if (tf != null) { 482 style = tf.getStyle(); 483 } 484 485 style |= mStyle; 486 487 if (mFamilyName != null) { 488 styledTypeface = Typeface.create(mFamilyName, style); 489 } else if (tf == null) { 490 styledTypeface = Typeface.defaultFromStyle(style); 491 } else { 492 styledTypeface = Typeface.create(tf, style); 493 } 494 } else { 495 styledTypeface = null; 496 } 497 498 if (styledTypeface != null) { 499 final Typeface readyTypeface; 500 if (mTextFontWeight >= 0) { 501 final int weight = Math.min(FontStyle.FONT_WEIGHT_MAX, mTextFontWeight); 502 final boolean italic = (style & Typeface.ITALIC) != 0; 503 readyTypeface = ds.setTypeface(Typeface.create(styledTypeface, weight, italic)); 504 } else { 505 readyTypeface = styledTypeface; 506 } 507 508 int fake = style & ~readyTypeface.getStyle(); 509 510 if ((fake & Typeface.BOLD) != 0) { 511 ds.setFakeBoldText(true); 512 } 513 514 if ((fake & Typeface.ITALIC) != 0) { 515 ds.setTextSkewX(-0.25f); 516 } 517 518 ds.setTypeface(readyTypeface); 519 } 520 521 if (mTextSize > 0) { 522 ds.setTextSize(mTextSize); 523 } 524 525 if (mTextLocales != null) { 526 ds.setTextLocales(mTextLocales); 527 } 528 529 if (mHasElegantTextHeight) { 530 ds.setElegantTextHeight(mElegantTextHeight); 531 } 532 533 if (mHasLetterSpacing) { 534 ds.setLetterSpacing(mLetterSpacing); 535 } 536 537 if (mFontFeatureSettings != null) { 538 ds.setFontFeatureSettings(mFontFeatureSettings); 539 } 540 541 if (mFontVariationSettings != null) { 542 ds.setFontVariationSettings(mFontVariationSettings); 543 } 544 } 545 546 @Override toString()547 public String toString() { 548 return "TextAppearanceSpan{" 549 + "familyName='" + getFamily() + '\'' 550 + ", style=" + getTextStyle() 551 + ", textSize=" + getTextSize() 552 + ", textColor=" + getTextColor() 553 + ", textColorLink=" + getLinkTextColor() 554 + ", typeface=" + getTypeface() 555 + ", textFontWeight=" + getTextFontWeight() 556 + ", textLocales=" + getTextLocales() 557 + ", shadowRadius=" + getShadowRadius() 558 + ", shadowDx=" + getShadowDx() 559 + ", shadowDy=" + getShadowDy() 560 + ", shadowColor=" + String.format("#%08X", getShadowColor()) 561 + ", elegantTextHeight=" + isElegantTextHeight() 562 + ", letterSpacing=" + getLetterSpacing() 563 + ", fontFeatureSettings='" + getFontFeatureSettings() + '\'' 564 + ", fontVariationSettings='" + getFontVariationSettings() + '\'' 565 + '}'; 566 } 567 } 568