/*
* Copyright (C) 2013-2014 Dominik Schürmann <dominik@dominikschuermann.de>
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.sufficientlysecure.htmltextview;
import android.content.Context;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.support.annotation.RawRes;
import android.text.Html;
import android.util.AttributeSet;
import android.view.MotionEvent;
import java.io.InputStream;
import java.util.Scanner;
public class HtmlTextView extends JellyBeanSpanFixTextView {
public static final String TAG = "HtmlTextView";
public static final boolean DEBUG = false;
boolean linkHit;
@Nullable
private ClickableTableSpan clickableTableSpan;
@Nullable
private DrawTableLinkSpan drawTableLinkSpan;
boolean dontConsumeNonUrlClicks = true;
private boolean removeFromHtmlSpace = true;
public HtmlTextView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
}
public HtmlTextView(Context context, AttributeSet attrs) {
super(context, attrs);
}
public HtmlTextView(Context context) {
super(context);
}
/**
* @see org.sufficientlysecure.htmltextview.HtmlTextView#setHtml(int)
*/
public void setHtml(@RawRes int resId) {
setHtml(resId, null);
}
/**
* @see org.sufficientlysecure.htmltextview.HtmlTextView#setHtml(String)
*/
public void setHtml(@NonNull String html) {
setHtml(html, null);
}
/**
* Loads HTML from a raw resource, i.e., a HTML file in res/raw/.
* This allows translatable resource (e.g., res/raw-de/ for german).
* The containing HTML is parsed to Android's Spannable format and then displayed.
*
* @param resId for example: R.raw.help
* @param imageGetter for fetching images. Possible ImageGetter provided by this library:
* HtmlLocalImageGetter and HtmlRemoteImageGetter
*/
public void setHtml(@RawRes int resId, @Nullable Html.ImageGetter imageGetter) {
InputStream inputStreamText = getContext().getResources().openRawResource(resId);
setHtml(convertStreamToString(inputStreamText), imageGetter);
}
/**
* Parses String containing HTML to Android's Spannable format and displays it in this TextView.
* Using the implementation of Html.ImageGetter provided.
*
* @param html String containing HTML, for example: "<b>Hello world!</b>"
* @param imageGetter for fetching images. Possible ImageGetter provided by this library:
* HtmlLocalImageGetter and HtmlRemoteImageGetter
*/
public void setHtml(@NonNull String html, @Nullable Html.ImageGetter imageGetter) {
final HtmlTagHandler htmlTagHandler = new HtmlTagHandler(getPaint());
htmlTagHandler.setClickableTableSpan(clickableTableSpan);
htmlTagHandler.setDrawTableLinkSpan(drawTableLinkSpan);
html = htmlTagHandler.overrideTags(html);
if (removeFromHtmlSpace) {
setText(removeHtmlBottomPadding(Html.fromHtml(html, imageGetter, htmlTagHandler)));
} else {
setText(Html.fromHtml(html, imageGetter, htmlTagHandler));
}
// make links work
setMovementMethod(LocalLinkMovementMethod.getInstance());
}
/**
* Note that this must be called before setting text for it to work
*/
public void setRemoveFromHtmlSpace(boolean removeFromHtmlSpace) {
this.removeFromHtmlSpace = removeFromHtmlSpace;
}
public void setClickableTableSpan(@Nullable ClickableTableSpan clickableTableSpan) {
this.clickableTableSpan = clickableTableSpan;
}
public void setDrawTableLinkSpan(@Nullable DrawTableLinkSpan drawTableLinkSpan) {
this.drawTableLinkSpan = drawTableLinkSpan;
}
/**
* http://stackoverflow.com/questions/309424/read-convert-an-inputstream-to-a-string
*/
@NonNull
static private String convertStreamToString(@NonNull InputStream is) {
Scanner s = new Scanner(is).useDelimiter("\\A");
return s.hasNext() ? s.next() : "";
}
/**
* Html.fromHtml sometimes adds extra space at the bottom.
* This methods removes this space again.
* See https://github.com/SufficientlySecure/html-textview/issues/19
*/
@Nullable
static private CharSequence removeHtmlBottomPadding(@Nullable CharSequence text) {
if (text == null) {
return null;
}
while (text.length() > 0 && text.charAt(text.length() - 1) == '\n') {
text = text.subSequence(0, text.length() - 1);
}
return text;
}
@Override
public boolean onTouchEvent(MotionEvent event) {
linkHit = false;
boolean res = super.onTouchEvent(event);
if (dontConsumeNonUrlClicks) {
return linkHit;
}
return res;
}
}