내 안드로이드 앱에서 텍스트 뷰 링크 클릭 처리
현재 TextView에서 HTML 입력을 렌더링하고 있습니다.
tv.setText(Html.fromHtml("<a href='test'>test</a>"));
표시되는 HTML은 외부 리소스를 통해 나에게 제공되므로 원하는대로 변경할 수는 없지만 HTML을 변경하여 정규 표현식 값을 변경하여 다른 값으로 변경할 수는 있습니다.
내가 원하는 것은 링크가 브라우저 창을 열지 않고 앱 내에서 직접 링크 클릭을 처리 할 수 있도록하는 것입니다. 이것은 전혀 달성 가능한가? href-value의 프로토콜을 "myApp : //"와 같은 것으로 설정 한 다음 내 앱에서 해당 프로토콜을 처리 할 수있는 항목을 등록하는 것이 가능할 것 같습니다. 이것이 실제로 최선의 방법이라면, 그것이 어떻게 수행되는지 알고 싶습니다만, "이 텍스트 뷰에서 링크를 클릭하면 이벤트가 발생하기를 원합니다." 입력 매개 변수로서의 링크의 href 값 "
거의 1 년 후이 문제를 해결하기 위해 특정 문제를 해결 한 방식이 다릅니다. 링크가 내 앱에서 처리되기를 원했기 때문에 조금 더 간단한 솔루션이 있습니다.
기본 의도 필터 외에도 대상 활동에서 ACTION_VIEW
의도, 특히 체계가있는 의도를 청취하게 합니다.com.package.name
<intent-filter>
<category android:name="android.intent.category.DEFAULT" />
<action android:name="android.intent.action.VIEW" />
<data android:scheme="com.package.name" />
</intent-filter>
이것은 com.package.name://
내 활동에 의해 시작되는 링크 가 처리됨을 의미합니다 .
그래서 내가해야 할 일은 전달하려는 정보가 포함 된 URL을 작성하는 것입니다.
com.package.name://action-to-perform/id-that-might-be-needed/
대상 활동에서이 주소를 검색 할 수 있습니다.
Uri data = getIntent().getData();
내 예에서는 data
null 값을 확인할 수 있습니다. null이 아닌 경우 그러한 링크를 통해 호출 된 것을 알 수 있습니다. 거기에서 URL에서 필요한 지침을 추출하여 적절한 데이터를 표시 할 수 있습니다.
다른 방법으로, Linkify에서 약간 빌리지 만 처리를 사용자 정의 할 수 있습니다.
맞춤 스팬 클래스 :
public class ClickSpan extends ClickableSpan {
private OnClickListener mListener;
public ClickSpan(OnClickListener listener) {
mListener = listener;
}
@Override
public void onClick(View widget) {
if (mListener != null) mListener.onClick();
}
public interface OnClickListener {
void onClick();
}
}
도우미 기능 :
public static void clickify(TextView view, final String clickableText,
final ClickSpan.OnClickListener listener) {
CharSequence text = view.getText();
String string = text.toString();
ClickSpan span = new ClickSpan(listener);
int start = string.indexOf(clickableText);
int end = start + clickableText.length();
if (start == -1) return;
if (text instanceof Spannable) {
((Spannable)text).setSpan(span, start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
} else {
SpannableString s = SpannableString.valueOf(text);
s.setSpan(span, start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
view.setText(s);
}
MovementMethod m = view.getMovementMethod();
if ((m == null) || !(m instanceof LinkMovementMethod)) {
view.setMovementMethod(LinkMovementMethod.getInstance());
}
}
용법:
clickify(textView, clickText,new ClickSpan.OnClickListener()
{
@Override
public void onClick() {
// do something
}
});
텍스트보기에 여러 개의 링크가있는 경우 예를 들어 textview에는 "https : //"및 "tel no"가 있습니다. LinkMovement 메서드를 사용자 지정하고 패턴을 기준으로 단어 클릭을 처리 할 수 있습니다. 맞춤 링크 이동 방법이 첨부되어 있습니다.
public class CustomLinkMovementMethod extends LinkMovementMethod
{
private static Context movementContext;
private static CustomLinkMovementMethod linkMovementMethod = new CustomLinkMovementMethod();
public boolean onTouchEvent(android.widget.TextView widget, android.text.Spannable buffer, android.view.MotionEvent event)
{
int action = event.getAction();
if (action == MotionEvent.ACTION_UP)
{
int x = (int) event.getX();
int y = (int) event.getY();
x -= widget.getTotalPaddingLeft();
y -= widget.getTotalPaddingTop();
x += widget.getScrollX();
y += widget.getScrollY();
Layout layout = widget.getLayout();
int line = layout.getLineForVertical(y);
int off = layout.getOffsetForHorizontal(line, x);
URLSpan[] link = buffer.getSpans(off, off, URLSpan.class);
if (link.length != 0)
{
String url = link[0].getURL();
if (url.startsWith("https"))
{
Log.d("Link", url);
Toast.makeText(movementContext, "Link was clicked", Toast.LENGTH_LONG).show();
} else if (url.startsWith("tel"))
{
Log.d("Link", url);
Toast.makeText(movementContext, "Tel was clicked", Toast.LENGTH_LONG).show();
} else if (url.startsWith("mailto"))
{
Log.d("Link", url);
Toast.makeText(movementContext, "Mail link was clicked", Toast.LENGTH_LONG).show();
}
return true;
}
}
return super.onTouchEvent(widget, buffer, event);
}
public static android.text.method.MovementMethod getInstance(Context c)
{
movementContext = c;
return linkMovementMethod;
}
텍스트 뷰에서 다음과 같은 방식으로 호출해야합니다.
textViewObject.setMovementMethod(CustomLinkMovementMethod.getInstance(context));
@Arun 답변을 기반으로 한보 다 일반적인 솔루션은 다음과 같습니다.
public abstract class TextViewLinkHandler extends LinkMovementMethod {
public boolean onTouchEvent(TextView widget, Spannable buffer, MotionEvent event) {
if (event.getAction() != MotionEvent.ACTION_UP)
return super.onTouchEvent(widget, buffer, event);
int x = (int) event.getX();
int y = (int) event.getY();
x -= widget.getTotalPaddingLeft();
y -= widget.getTotalPaddingTop();
x += widget.getScrollX();
y += widget.getScrollY();
Layout layout = widget.getLayout();
int line = layout.getLineForVertical(y);
int off = layout.getOffsetForHorizontal(line, x);
URLSpan[] link = buffer.getSpans(off, off, URLSpan.class);
if (link.length != 0) {
onLinkClick(link[0].getURL());
}
return true;
}
abstract public void onLinkClick(String url);
}
그것을 사용하려면 단지 클래스의 구현 onLinkClick
입니다 TextViewLinkHandler
. 예를 들어 :
textView.setMovementMethod(new TextViewLinkHandler() {
@Override
public void onLinkClick(String url) {
Toast.makeText(textView.getContext(), url, Toast.LENGTH_SHORT).show();
}
});
그것의 매우 간단한 코드에 다음 줄을 추가하십시오 :
tv.setMovementMethod(LinkMovementMethod.getInstance());
해결책
TextView 자체를 길게 클릭하고 TextView의 링크를 탭 할 수있는 작은 클래스를 구현했습니다.
나열한 것
TextView android:id="@+id/text"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:autoLink="all"/>
TextViewClickMovement.java
import android.content.Context;
import android.text.Layout;
import android.text.Spannable;
import android.text.method.LinkMovementMethod;
import android.text.style.ClickableSpan;
import android.util.Patterns;
import android.view.GestureDetector;
import android.view.MotionEvent;
import android.widget.TextView;
public class TextViewClickMovement extends LinkMovementMethod {
private final String TAG = TextViewClickMovement.class.getSimpleName();
private final OnTextViewClickMovementListener mListener;
private final GestureDetector mGestureDetector;
private TextView mWidget;
private Spannable mBuffer;
public enum LinkType {
/** Indicates that phone link was clicked */
PHONE,
/** Identifies that URL was clicked */
WEB_URL,
/** Identifies that Email Address was clicked */
EMAIL_ADDRESS,
/** Indicates that none of above mentioned were clicked */
NONE
}
/**
* Interface used to handle Long clicks on the {@link TextView} and taps
* on the phone, web, mail links inside of {@link TextView}.
*/
public interface OnTextViewClickMovementListener {
/**
* This method will be invoked when user press and hold
* finger on the {@link TextView}
*
* @param linkText Text which contains link on which user presses.
* @param linkType Type of the link can be one of {@link LinkType} enumeration
*/
void onLinkClicked(final String linkText, final LinkType linkType);
/**
*
* @param text Whole text of {@link TextView}
*/
void onLongClick(final String text);
}
public TextViewClickMovement(final OnTextViewClickMovementListener listener, final Context context) {
mListener = listener;
mGestureDetector = new GestureDetector(context, new SimpleOnGestureListener());
}
@Override
public boolean onTouchEvent(final TextView widget, final Spannable buffer, final MotionEvent event) {
mWidget = widget;
mBuffer = buffer;
mGestureDetector.onTouchEvent(event);
return false;
}
/**
* Detects various gestures and events.
* Notify users when a particular motion event has occurred.
*/
class SimpleOnGestureListener extends GestureDetector.SimpleOnGestureListener {
@Override
public boolean onDown(MotionEvent event) {
// Notified when a tap occurs.
return true;
}
@Override
public void onLongPress(MotionEvent e) {
// Notified when a long press occurs.
final String text = mBuffer.toString();
if (mListener != null) {
Log.d(TAG, "----> Long Click Occurs on TextView with ID: " + mWidget.getId() + "\n" +
"Text: " + text + "\n<----");
mListener.onLongClick(text);
}
}
@Override
public boolean onSingleTapConfirmed(MotionEvent event) {
// Notified when tap occurs.
final String linkText = getLinkText(mWidget, mBuffer, event);
LinkType linkType = LinkType.NONE;
if (Patterns.PHONE.matcher(linkText).matches()) {
linkType = LinkType.PHONE;
}
else if (Patterns.WEB_URL.matcher(linkText).matches()) {
linkType = LinkType.WEB_URL;
}
else if (Patterns.EMAIL_ADDRESS.matcher(linkText).matches()) {
linkType = LinkType.EMAIL_ADDRESS;
}
if (mListener != null) {
Log.d(TAG, "----> Tap Occurs on TextView with ID: " + mWidget.getId() + "\n" +
"Link Text: " + linkText + "\n" +
"Link Type: " + linkType + "\n<----");
mListener.onLinkClicked(linkText, linkType);
}
return false;
}
private String getLinkText(final TextView widget, final Spannable buffer, final MotionEvent event) {
int x = (int) event.getX();
int y = (int) event.getY();
x -= widget.getTotalPaddingLeft();
y -= widget.getTotalPaddingTop();
x += widget.getScrollX();
y += widget.getScrollY();
Layout layout = widget.getLayout();
int line = layout.getLineForVertical(y);
int off = layout.getOffsetForHorizontal(line, x);
ClickableSpan[] link = buffer.getSpans(off, off, ClickableSpan.class);
if (link.length != 0) {
return buffer.subSequence(buffer.getSpanStart(link[0]),
buffer.getSpanEnd(link[0])).toString();
}
return "";
}
}
}
용법
TextView tv = (TextView) v.findViewById(R.id.textview);
tv.setText(Html.fromHtml("<a href='test'>test</a>"));
textView.setMovementMethod(new TextViewClickMovement(this, context));
연결
도움이 되었기를 바랍니다! 여기에서 코드를 찾을 수 있습니다 .
내가 만든 라이브러리를 사용하여 대체 솔루션을 공유하기 만하면됩니다. Textoo를 사용하면 다음 과 같이 달성 할 수 있습니다.
TextView locNotFound = Textoo
.config((TextView) findViewById(R.id.view_location_disabled))
.addLinksHandler(new LinksHandler() {
@Override
public boolean onClick(View view, String url) {
if ("internal://settings/location".equals(url)) {
Intent locSettings = new Intent(android.provider.Settings.ACTION_LOCATION_SOURCE_SETTINGS);
startActivity(locSettings);
return true;
} else {
return false;
}
}
})
.apply();
또는 동적 HTML 소스를 사용하는 경우 :
String htmlSource = "Links: <a href='http://www.google.com'>Google</a>";
Spanned linksLoggingText = Textoo
.config(htmlSource)
.parseHtml()
.addLinksHandler(new LinksHandler() {
@Override
public boolean onClick(View view, String url) {
Log.i("MyActivity", "Linking to google...");
return false; // event not handled. Continue default processing i.e. link to google
}
})
.apply();
textView.setText(linksLoggingText);
더 많은 옵션을 찾는 사람은 여기입니다.
// Set text within a `TextView`
TextView textView = (TextView) findViewById(R.id.textView);
textView.setText("Hey @sarah, where did @jim go? #lost");
// Style clickable spans based on pattern
new PatternEditableBuilder().
addPattern(Pattern.compile("\\@(\\w+)"), Color.BLUE,
new PatternEditableBuilder.SpannableClickedListener() {
@Override
public void onSpanClicked(String text) {
Toast.makeText(MainActivity.this, "Clicked username: " + text,
Toast.LENGTH_SHORT).show();
}
}).into(textView);
자원 : CodePath
public static void setTextViewFromHtmlWithLinkClickable(TextView textView, String text) {
Spanned result;
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.N) {
result = Html.fromHtml(text, Html.FROM_HTML_MODE_LEGACY);
} else {
result = Html.fromHtml(text);
}
textView.setText(result);
textView.setMovementMethod(LinkMovementMethod.getInstance());
}
예를 들어 TextView의 색상을 파란색으로 변경했습니다.
android:textColor="#3399FF"
xml 파일에서. 밑줄을 만드는 방법은 여기 에 설명되어 있습니다 .
그런 다음 onClick 속성을 사용하여 메소드를 지정하십시오 ( setOnClickListener(this)
다른 방법으로 호출 할 수 있다고 생각합니다 ). 예 :
myTextView.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
doSomething();
}
});
이 방법에서는 인 텐트를 시작하는 등 원하는대로 무엇이든 할 수 있습니다. myTextView.setMovementMethod(LinkMovementMethod.getInstance());
acitivity의 onCreate () 메소드에서와 같이 여전히 정상적인 작업 을 수행해야합니다 .
내가 사용한 가장 좋은 방법은 항상 나를 위해 일했다
android:autoLink="web"
이 답변은 Jonathan S의 탁월한 솔루션을 확장합니다.
다음 방법을 사용하여 텍스트에서 링크를 추출 할 수 있습니다.
private static ArrayList<String> getLinksFromText(String text) {
ArrayList links = new ArrayList();
String regex = "\(?\b((http|https)://www[.])[-A-Za-z0-9+&@#/%?=~_()|!:,.;]*[-A-Za-z0-9+&@#/%=~_()|]";
Pattern p = Pattern.compile(regex);
Matcher m = p.matcher(text);
while (m.find()) {
String urlStr = m.group();
if (urlStr.startsWith("(") && urlStr.endsWith(")")) {
urlStr = urlStr.substring(1, urlStr.length() - 1);
}
links.add(urlStr);
}
return links;
}
clickify()
메소드 에서 매개 변수 중 하나를 제거하는 데 사용할 수 있습니다 .
public static void clickify(TextView view,
final ClickSpan.OnClickListener listener) {
CharSequence text = view.getText();
String string = text.toString();
ArrayList<String> linksInText = getLinksFromText(string);
if (linksInText.isEmpty()){
return;
}
String clickableText = linksInText.get(0);
ClickSpan span = new ClickSpan(listener,clickableText);
int start = string.indexOf(clickableText);
int end = start + clickableText.length();
if (start == -1) return;
if (text instanceof Spannable) {
((Spannable) text).setSpan(span, start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
} else {
SpannableString s = SpannableString.valueOf(text);
s.setSpan(span, start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
view.setText(s);
}
MovementMethod m = view.getMovementMethod();
if ((m == null) || !(m instanceof LinkMovementMethod)) {
view.setMovementMethod(LinkMovementMethod.getInstance());
}
}
ClickSpan에 대한 몇 가지 변경 사항 :
public static class ClickSpan extends ClickableSpan {
private String mClickableText;
private OnClickListener mListener;
public ClickSpan(OnClickListener listener, String clickableText) {
mListener = listener;
mClickableText = clickableText;
}
@Override
public void onClick(View widget) {
if (mListener != null) mListener.onClick(mClickableText);
}
public interface OnClickListener {
void onClick(String clickableText);
}
}
이제 TextView에서 텍스트를 설정 한 다음 리스너를 추가 할 수 있습니다.
TextViewUtils.clickify(textWithLink,new TextUtils.ClickSpan.OnClickListener(){
@Override
public void onClick(String clickableText){
//action...
}
});
예 : textview에 일부 텍스트를 설정하고 특정 텍스트 표현식에 대한 링크를 제공하려고한다고 가정합니다. " #facebook 을 클릭 하면 facebook.com으로 이동합니다"
레이아웃 xml에서 :
<TextView
android:id="@+id/testtext"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
활동 중 :
String text = "Click on #facebook will take you to facebook.com";
tv.setText(text);
Pattern tagMatcher = Pattern.compile("[#]+[A-Za-z0-9-_]+\\b");
String newActivityURL = "content://ankit.testactivity/";
Linkify.addLinks(tv, tagMatcher, newActivityURL);
또한 다음과 같이 하나의 태그 제공자를 작성하십시오.
public class TagProvider extends ContentProvider {
@Override
public int delete(Uri arg0, String arg1, String[] arg2) {
// TODO Auto-generated method stub
return 0;
}
@Override
public String getType(Uri arg0) {
return "vnd.android.cursor.item/vnd.cc.tag";
}
@Override
public Uri insert(Uri arg0, ContentValues arg1) {
// TODO Auto-generated method stub
return null;
}
@Override
public boolean onCreate() {
// TODO Auto-generated method stub
return false;
}
@Override
public Cursor query(Uri arg0, String[] arg1, String arg2, String[] arg3,
String arg4) {
// TODO Auto-generated method stub
return null;
}
@Override
public int update(Uri arg0, ContentValues arg1, String arg2, String[] arg3) {
// TODO Auto-generated method stub
return 0;
}
}
매니페스트 파일에서 공급자 및 테스트 활동의 항목으로 다음을 수행하십시오.
<provider
android:name="ankit.TagProvider"
android:authorities="ankit.testactivity" />
<activity android:name=".TestActivity"
android:label = "@string/app_name">
<intent-filter >
<action android:name="android.intent.action.VIEW" />
<category android:name="android.intent.category.DEFAULT" />
<data android:mimeType="vnd.android.cursor.item/vnd.cc.tag" />
</intent-filter>
</activity>
이제 #facebook을 클릭하면 testactivtiy가 호출됩니다. 테스트 활동에서 데이터를 다음과 같이 얻을 수 있습니다.
Uri uri = getIntent().getData();
참고 URL : https://stackoverflow.com/questions/1697084/handle-textview-link-click-in-my-android-app
'Programing' 카테고리의 다른 글
Qt, C ++에 폴더가 있는지 확인하고 폴더 만들기 (0) | 2020.06.18 |
---|---|
파일 목록에서 입력을받는 Tar 아카이브 (0) | 2020.06.18 |
postgres에서 필드의 데이터 유형을 선택하십시오. (0) | 2020.06.18 |
C # Double-소수점 이하 두 자리이지만 반올림이없는 ToString () 형식 (0) | 2020.06.18 |
EF Code First“잘못된 열 이름 'Discriminator' '이지만 상속은 없습니다. (0) | 2020.06.18 |