จัดการลิงค์ข้อความคลิกในแอป Android ของฉัน


148

ฉันกำลังแสดงอินพุต HTML ใน TextView เช่น:

tv.setText(Html.fromHtml("<a href='test'>test</a>"));

HTML ที่ถูกแสดงถูกจัดเตรียมไว้ให้ฉันผ่านทางแหล่งข้อมูลภายนอกดังนั้นฉันไม่สามารถเปลี่ยนแปลงสิ่งต่าง ๆ ได้ตามที่ฉันต้องการ แต่แน่นอนว่าฉันสามารถทำ regex ยุ่งกับ HTML เพื่อเปลี่ยนค่า href พูดเป็นอย่างอื่น

สิ่งที่ฉันต้องการคือสามารถจัดการลิงก์คลิกโดยตรงจากภายในแอปแทนที่จะให้ลิงก์เปิดหน้าต่างเบราว์เซอร์ สามารถทำได้ทั้งหมดหรือไม่ ฉันเดาว่ามันจะเป็นไปได้ที่จะตั้งค่าโปรโตคอลของ href-value เป็นอย่างเช่น "myApp: //" แล้วลงทะเบียนสิ่งที่จะให้แอพของฉันจัดการโปรโตคอลนั้น หากนี่เป็นวิธีที่ดีที่สุดฉันต้องการทราบวิธีการทำ แต่ฉันหวังว่าจะมีวิธีที่ง่ายกว่าในการพูดว่า "เมื่อมีการคลิกลิงก์ในมุมมองข้อความนี้ฉันต้องการเพิ่มกิจกรรมที่ได้รับ ค่า href ของลิงค์เป็นพารามิเตอร์อินพุต "


ฉันพบสิ่งอื่นที่ [ที่นี่] [1] [1]: stackoverflow.com/questions/7255249/… หวังว่าจะสามารถช่วยคุณได้ ^^
Mr_DK

1
เดวิดฉันมีกรณีที่คล้ายกับของคุณฉันได้รับ html จากแหล่งภายนอก (เว็บ) แต่ฉันจะ regex ยุ่งเกี่ยวกับค่า href ได้อย่างไรเพื่อให้ฉันสามารถใช้โซลูชันนี้ได้
X09

คำตอบ:


182

มาที่นี่เกือบปีต่อมามีวิธีที่แตกต่างกันซึ่งฉันแก้ไขปัญหาเฉพาะของฉันได้ เนื่องจากฉันต้องการให้ลิงก์จัดการโดยแอพของฉันเองจึงมีวิธีแก้ปัญหาที่ค่อนข้างง่ายกว่า

นอกเหนือจากตัวกรองเจตนาเริ่มต้นฉันเพียงให้กิจกรรมเป้าหมายของฉันฟังการแสดงความ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ค่าว่างได้เพราะเมื่อใดก็ตามที่ไม่เป็นโมฆะฉันจะรู้ว่ามันถูกเรียกใช้โดยลิงก์ดังกล่าว จากนั้นฉันแยกคำแนะนำที่ฉันต้องการจาก URL เพื่อให้สามารถแสดงข้อมูลที่เหมาะสม


1
เฮ้คำตอบของคุณสมบูรณ์แบบ ทำงานได้ดี แต่เราสามารถส่งข้อมูลการส่งผ่านข้อมูลเพิ่มเติมได้อีกหรือไม่
user861973

4
@ user861973: ใช่getDataมอบ URI แบบเต็มให้กับคุณคุณสามารถใช้getDataStringที่ให้การแสดงข้อความแทนได้ คุณสามารถสร้าง URL เพื่อเก็บข้อมูลทั้งหมดที่คุณต้องการcom.package.name://my-action/1/2/3/4และคุณสามารถดึงข้อมูลจากสตริงนั้นได้
David Hedlund

3
ฉันใช้เวลาทั้งวันเพื่อทำความเข้าใจความคิดนี้ แต่ฉันบอกคุณว่าอะไร - มันก็คุ้มค่า โซลูชันที่ออกแบบมาอย่างดี
Dennis

7
ทางออกที่ดี อย่าลืมเพิ่มสิ่งนี้ลงใน Textview เพื่อให้มันเปิดใช้งานลิงก์ tv.setMovementMethod (LinkMovementMethod.getInstance ());
Daniel Benedykt

3
ขออภัยในวิธีการในกิจกรรมที่ฉันควรจะพูดว่าUri data = getIntent().getData();? ฉันได้รับActivity not found to handle intentข้อผิดพลาด - ขอบคุณ
rgv

62

อีกวิธีหนึ่งที่ยืมมาจาก Linkify แต่ช่วยให้คุณสามารถปรับแต่งการจัดการของคุณ

คลาส Span ที่กำหนดเอง:

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
        }
    });

อีกวิธีหนึ่งคือการแทนที่ Spans ด้วย Spans ที่กำหนดเองของคุณดูstackoverflow.com/a/11417498/792677
A-Live

ทางออกที่ดีเยี่ยม
Mario Zderic

ฉันพบว่านี่เป็นทางออกที่ง่ายที่สุดในการติดตั้ง Google ไม่เคยทำความสะอาดระเบียบนี้เพื่อให้สามารถสร้างลิงก์ใน textviews ที่คลิกได้ มันเป็นวิธีการที่โหดร้ายในการบังคับให้มีช่วง แต่มันทำงานได้ดีกับระบบปฏิบัติการรุ่นต่าง ๆ .. +1
angryITguy

55

หากมีหลายลิงค์ในมุมมองข้อความ ตัวอย่างเช่น 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;
}

สิ่งนี้ควรถูกเรียกจาก textview ในลักษณะดังต่อไปนี้:

textViewObject.setMovementMethod(CustomLinkMovementMethod.getInstance(context));

8
คุณไม่จำเป็นต้องผ่านบริบทดังนั้น "ด้านหลัง" ในตัวแปรสแตติกแยกต่างหากนั่นเป็นกลิ่นเหม็น เพียงใช้widget.getContext()แทน
Sergej Koščejev

ใช่บริบทสามารถลบได้ ขอบคุณสำหรับการชี้ให้เห็น Sergej
อรุณ

6
วิธีนี้ใช้งานได้ดี แต่คุณต้องเรียกใช้ setMovementMethod หลังจาก setText มิฉะนั้นจะเขียนทับด้วย LinkMovementMethod เริ่มต้น
Ray Britton

ใช้งานได้ แต่ไม่จำเป็นต้องวนซ้ำในช่วงการคลิกแต่ละครั้ง ตัวกรองตามตำแหน่งยังมีแนวโน้มที่จะเกิดข้อผิดพลาด วิธีการเตรียมใช้งานแบบครั้งเดียวที่คล้ายกันจะแสดงในคำตอบนี้
มิสเตอร์สมิ ธ

@Arun กรุณาอัปเดตคำตอบของคุณตามความคิดเห็น
Behrouz.M

45

นี่คือวิธีแก้ปัญหาทั่วไปเพิ่มเติมตาม @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();
        }
    });

2
ใช้งานได้ดี หมายเหตุ: อย่าลืมเพิ่ม android: autoLink = "web" ลงใน textview ของคุณ หากไม่มีแอตทริบิวต์การเชื่อมโยงอัตโนมัติสิ่งนี้จะไม่ทำงาน
okarakose

ฉันได้ลองวิธีแก้ไขปัญหาทั้งหมดที่ระบุไว้ที่นี่แล้ว อันนี้ดีที่สุดสำหรับฉัน มันชัดเจนใช้งานง่ายและทรงพลัง คำแนะนำ: คุณต้องทำงานกับ Html.fromHtml เพื่อให้ได้ประโยชน์สูงสุด
ไก่วัง

1
คำตอบที่ดีที่สุด! ขอบคุณ! อย่าลืมเพิ่ม android: autoLink = "web" ใน xml หรือในรหัส LinkifyCompat.addLinks (textView, Linkify.WEB_URLS);
Nikita Axyonov

1
สิ่งที่จะเป็นความต้องการง่ายๆที่ดูเหมือนจะเป็นเรื่องค่อนข้างซับซ้อนใน Android วิธีแก้ปัญหานี้ช่วยลดความเจ็บปวดได้มาก พบว่า autoLink = "web" ไม่จำเป็นสำหรับโซลูชันนี้เพื่อทำงานบน Android N .. +1
angryITguy

1
ทำไมในโลกชนชั้นแบบนี้จึงไม่มีอยู่จริง - หลังจากทุกปีเหล่านี้ - อยู่นอกเหนือฉัน ขอบคุณ Android สำหรับ LinkMovementMethod แต่โดยปกติฉันต้องการควบคุม UI ของตัวเอง คุณสามารถรายงานกิจกรรม UI ให้ฉันและเพื่อให้ผู้ควบคุมของฉันสามารถจัดการได้
ระเบียบการ

10

มันง่ายมากที่จะเพิ่มบรรทัดนี้ในรหัสของคุณ:

tv.setMovementMethod(LinkMovementMethod.getInstance());

1
ขอบคุณสำหรับคำตอบของคุณโจนาธาน ใช่ฉันรู้เกี่ยวกับ MovementMethod; สิ่งที่ฉันไม่แน่ใจเกี่ยวกับวิธีระบุว่าแอปของฉันเองควรจัดการกับการคลิกลิงก์แทนที่จะเปิดเบราว์เซอร์เนื่องจากวิธีการเคลื่อนไหวเริ่มต้น (ดูคำตอบที่ยอมรับ) ขอบคุณอยู่ดี
David Hedlund

5

สารละลาย

ฉันได้ติดตั้งคลาสเล็ก ๆ ด้วยความช่วยเหลือซึ่งคุณสามารถจัดการกับการคลิกแบบยาวบน TextView เองและ Taps บนลิงก์ใน 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));

การเชื่อมโยง

หวังว่านี่จะช่วยได้! คุณสามารถค้นหารหัสที่นี่


3

เพียงเพื่อแชร์โซลูชันทางเลือกโดยใช้ไลบรารีที่ฉันสร้างขึ้น ด้วย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);

นี่ควรเป็นคำตอบที่ดีที่สุด ขอขอบคุณสำหรับการบริจาค
Marian Pavel

3

สำหรับผู้ที่มองหาตัวเลือกเพิ่มเติมที่นี่เป็นหนึ่ง

// 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


2
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());
}

1

ฉันเปลี่ยนสีของ TextView เป็นสีน้ำเงินโดยใช้ตัวอย่าง:

android:textColor="#3399FF"

ในไฟล์ xml วิธีที่จะทำให้มันขีดเส้นใต้จะมีการอธิบายที่นี่

จากนั้นใช้คุณสมบัติ onClick เพื่อระบุวิธี (ฉันคาดเดาว่าคุณสามารถโทรหาsetOnClickListener(this)เป็นวิธีอื่นได้) เช่น:

myTextView.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
    doSomething();
}
});

ในวิธีการนั้นฉันสามารถทำสิ่งที่ฉันต้องการได้ตามปกติเช่นเปิดตัวเจตนา โปรดทราบว่าคุณยังคงต้องทำmyTextView.setMovementMethod(LinkMovementMethod.getInstance());สิ่งปกติเช่นเดียวกับวิธี onCreate () ของ acitivity


1

คำตอบนี้ขยายโซลูชันที่ยอดเยี่ยมของ 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...
}

});

0

วิธีที่ดีที่สุดที่ฉันใช้และมันก็ใช้ได้ผลเสมอสำหรับฉัน

android:autoLink="web"

8
นี่ไม่ได้ตอบคำถามเลย
Tom

0

ตัวอย่าง: สมมติว่าคุณได้ตั้งค่าข้อความในมุมมองข้อความและคุณต้องการระบุลิงก์ในนิพจน์ข้อความเฉพาะ: "คลิกที่#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();
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.