Как сделать ссылки в TextView кликабельными?

у меня определен следующий TextView:

<TextView android:layout_width="wrap_content"
    android:layout_height="wrap_content" android:text="@string/txtCredits"
    android:autoLink="web" android:id="@+id/infoTxtCredits"
    android:layout_centerInParent="true"
    android:linksClickable="true"></TextView>

здесь @string/txtCredits является строковым ресурсом, который содержит <a href="some site">Link text</a>.

Android выделяет ссылки в TextView, но они не реагируют на щелчки. Кто-нибудь может сказать мне, что я делаю не так? Должен ли я установить onClickListener для TextView в моей деятельности для чего-то такого простого?

похоже, это связано с тем, как я определяю мой строковый ресурс. Это не работа:

<string name="txtCredits"><a href="http://www.google.com">Google</a></string>

а это:

<string name="txtCredits">www.google.com</string>

что облом, потому что я бы предпочел показать текстовую ссылку, чем показать полный URL.

30 ответов


похороненный в демонстрациях API, я нашел решение своей проблемы:

ссылка.java:

    // text2 has links specified by putting <a> tags in the string
    // resource.  By default these links will appear but not
    // respond to user input.  To make them active, you need to
    // call setMovementMethod() on the TextView object.

    TextView t2 = (TextView) findViewById(R.id.text2);
    t2.setMovementMethod(LinkMovementMethod.getInstance());

Я удалил большинство атрибутов в моем TextView, чтобы соответствовать тому, что было в демо.

<TextView
    android:id="@+id/text2"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="@string/txtCredits"/>

это решено. Довольно трудно обнаружить и исправить.

важно: не забудьте удалить autoLink="web" Если вы звоните setMovementMethod().


Я использую только android:autoLink="web" и он отлично работает. Щелчок по ссылке открывает браузер и показывает правильную страницу.

одна вещь, о которой я мог догадаться, заключается в том, что над ссылкой находится какое-то другое представление. Что-то прозрачное заполняет весь родитель, но не отображает ничего выше ссылки. В этом случае щелчок переходит к этому представлению вместо ссылки.


проведя некоторое время с этим, я обнаружил, что:

  • android:autoLink="web" работает, если у вас есть полные ссылки в вашем HTML. Ниже будут выделены синим цветом и доступны клики:
  • текст <a href="http://www.google.com">http://www.google.com</a>
  • текст http://www.google.com
  • view.setMovementMethod(LinkMovementMethod.getInstance()); будет работать со следующим (будет выделено и кликабельно):
  • некоторые текст <a href="http://www.google.com">http://www.google.com</a>
  • текст http://www.google.com
  • текст <a href="http://www.google.com">Go to Google</a>

обратите внимание, что третий вариант имеет гиперссылку, но описание ссылки (части между тегами) само по себе не является ссылкой. android:autoLink="web" тут не работы с такими ссылками.

  • android:autoLink="web" Если установлено в XML, переопределит view.setMovementMethod(LinkMovementMethod.getInstance()); (т. е. ссылки третьего рода будут выделены, но не кликабельный.)

мораль этой истории-использование view.setMovementMethod(LinkMovementMethod.getInstance()); в вашем коде и убедитесь, что у вас нет android:autoLink="web" в вашем XML-макете, если вы хотите все ссылки должны быть кликабельны.


вышеуказанные решения не работали для меня, но следующее (И это кажется немного чище).
Во-первых, в строковом ресурсе определите свой тег, открывающий шевроны, используя кодировку сущности HTML, т. е.:

&lt;a href="http://www.google.com">Google&lt;/a>

и

<a href="http://www.google.com">Google</a>

В общем, Закодируйте все шевроны в строке таким образом. Кстати, ссылка должна начинаться с http://

затем (как было предложено здесь) установите этот параметр на свой Виджет TextView:

 android:linksClickable="true"

наконец, в коде сделать:

((TextView) findViewById(R.id.your_text_view)).setMovementMethod(LinkMovementMethod.getInstance());
((TextView) findViewById(R.id.your_text_view)).setText(Html.fromHtml(getResources().getString(R.string.string_with_links)));

вот и все, никаких регулярных выражений или других ручных хаков не требуется.


Если вы хотите добавить HTML-ссылку, Все, что вам нужно сделать, это:

  • добавить HTML-строку ресурса:

     <string name="link"><a href="https://www.google.pl/">Google</a></string>
    
  • добавьте свой вид в макет без какой-либо конкретной конфигурации ссылки на всех:

     <TextView
        android:id="@+id/link"
        android:text="@string/link" />`
    
  • добавьте соответствующий MovementMethod программно в свой TextView:

     mLink = (TextView) findViewById(R.id.link);
     if (mLink != null) {
       mLink.setMovementMethod(LinkMovementMethod.getInstance());
     }
    

вот именно! И да, имея такие опции, как" autoLink "и" linksClickable", работающие только явные ссылки (не завернутые в html-теги) тоже вводят меня в заблуждение...


я использовал это просто

Linkify.addLinks(TextView, Linkify.ALL);

делает ссылки кликабельными, учитывая здесь


я добавил эту строку с TextView: android:autoLink="web"
Ниже приведен пример использования в файле макета.

layout.xml пример

    <TextView
        android:id="@+id/txtLostpassword"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center"
        android:autoLink="email"
        android:gravity="center"
        android:padding="20px"
        android:text="@string/lostpassword"
        android:textAppearance="?android:attr/textAppearanceSmall" />

    <TextView
        android:id="@+id/txtDefaultpassword"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center"
        android:autoLink="web"
        android:gravity="center"
        android:padding="20px"
        android:text="@string/defaultpassword"
        android:textAppearance="?android:attr/textAppearanceSmall" />

string.xml

<string name="lostpassword">If you lost your password please contact <a href="mailto:support@cleverfinger.com.au?Subject=Lost%20Password" target="_top">support@cleverfinger.com.au</a></string>

<string name="defaultpassword">User Guide <a href="http://www.cleverfinger.com.au/user-guide/">http://www.cleverfinger.com.au/user-guide/</a></string>

Я надеюсь, что это поможет вам;

String value = "<html>Visit my blog <a href=\"http://www.maxartists.com\">mysite</a> View <a href=\"sherif-activity://myactivity?author=sherif&nick=king\">myactivity</a> callback</html>";
    TextView text = (TextView) findViewById(R.id.text);


    text.setText(Html.fromHtml(value));
    text.setMovementMethod(LinkMovementMethod.getInstance());

только что вам нужно добавить это в текстовом представлении в xml

<TextView
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:autoLink="web"/>

Ричард, в следующий раз, вы должны добавить этот код под TextView в XML макета вместо этого.

android:autoLink="all"

это должно быть так.

<TextView 
    android:layout_width="wrap_content"
    android:layout_height="wrap_content" 
    android:text="@string/txtCredits"
    android:id="@+id/infoTxtCredits"
    android:autoLink="all"
    android:linksClickable="true">
</TextView>

вам не нужно использовать этот код (t2.setMovementMethod(LinkMovementMethod.getInstance());) для того, чтобы сделать кликабельную ссылку.

кроме того, вот правда: пока вы выберите автоссылка и linksClickable, не забудьте добавить это строку.в XML файл, так что кликабельная ссылка будет работа.

<string name="txtCredits"><a href="http://www.google.com">Google</a></string>

С помощью linkify: Linkify возьмите кусок текста и регулярное выражение и превращает все регулярные выражения в тексте в интерактивные ссылки

TextView textView = (TextView) findViewById(R.id.textView);
textView.setText("http://www.domain.com");
Linkify.addLinks(textView, Linkify.WEB_URLS);

Не забудьте

import android.widget.TextView;

самое простое, что сработало для меня, - это использовать Linkify

TextView txt_Message = (TextView) view.findViewById(R.id.txt_message);
txt_Message.setText("This is link https://www.google.co.in/");
Linkify.addLinks(txt_Message, Linkify.WEB_URLS);

и он автоматически обнаружит веб-адреса из текста в textview.


Я заметил, что с помощью

<TextView
    android:layout_width="match_parent"
    android:layout_height="wrap_content" 
    android:autoLink="web"/>

работал нормально для URL-адресов, но так как у меня был адрес электронной почты и номер телефона, который я хотел связать, я в конечном итоге использовал эту строку android:autoLink="all" такой

<TextView
        android:layout_width="match_parent"
        android:layout_height="wrap_content" 
        android:autoLink="all"/>

и это сработало как заклинание.


вот очень одна строка android код, чтобы сделать телефон и url выбирается из textView независимо от того, что строка и что данные. Для этого вам не нужно использовать HTML-теги.

TextView textView = (TextView)findViewById(R.id.textView1);
textView.setText("some url is www.google.com phone 7504567890 another url lkgndflg.com ");

// Makes the textView's Phone and URL (hyperlink) select and go.
Linkify.addLinks(textView, Linkify.WEB_URLS | Linkify.PHONE_NUMBERS);

обязательно не использовать setAutoLinkMask(Linkify.ALL) при использовании setMovementMethod(LinkMovementMethod.getInstance()) и Html.fromHTML() на правильно отформатированную HTML ссылки (например, <a href="http://www.google.com/">Google</a>).


следующее должно работать для тех, кто ищет комбинацию текста и гиперссылки в приложении для Android.

на string.xml:

<string name="applink">Looking for the regular Zesteve App? 
<a href="https://play.google.com/store/apps/details?id=zesteve.com.myapplication">Get it here</a>
</string>

теперь вы можете использовать это string в любой момент View такой:

<TextView
    android:id="@+id/getapp"
    android:layout_width="match_parent"
    android:layout_height="50dp"
    android:gravity="center"
    android:textColor="@color/main_color_grey_600"
    android:textSize="15sp"
    android:text="@string/applink"/>

теперь, в вашей деятельности или фрагменте, сделайте следующее:

TextView getapp =(TextView) findViewById(R.id.getapp);
getapp.setMovementMethod(LinkMovementMethod.getInstance());

к настоящему времени вам не требуется устанавливать android:autoLink="web" или android:linksClickable="true" используя этот подход.

Я надеюсь, вы найдете это полезным.


использовать это...

TextView.setOnClickListener(new OnClickListener() {

                    @Override
                    public void onClick(View v) {
                        // TODO Auto-generated method stub
                        Intent in=new Intent(Intent.ACTION_VIEW,Uri.parse("http://www.twitter.com/"));
                        startActivity(in);
                    }

                });

и добавьте разрешение в файл манифеста

<uses-permission android:name="android.permission.INTERNET"/>

вам нужно только это:

android:autoLink="web"

вставьте эту строку в TextView, которую можно щелкнуть со ссылкой на веб-сайт. URL-адрес, заданный как текст этого TextView.

пример:

 <TextView
    android:id="@+id/textViewWikiURL"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:textSize="20sp"
    android:textStyle="bold"
    android:text="http://www.wikipedia.org/"
    android:autoLink="web" />

принятый ответ правильный, но это будет означать, что телефонные номера, карты, адреса электронной почты и регулярные ссылки например http://google.com без тегов href больше не будет кликабельна, так как вы не можете иметь автолинк в xml.

единственное полное решение для всего, что я нашел, это следующее:

Spanned text = Html.fromHtml(myString);
URLSpan[] currentSpans = text.getSpans(0, text.length(), URLSpan.class);
SpannableString buffer = new SpannableString(text);
Linkify.addLinks(buffer, Linkify.ALL);
for (URLSpan span : currentSpans) {
    int end = text.getSpanEnd(span);
    int start = text.getSpanStart(span);
    buffer.setSpan(span, start, end, 0);
}
textView.setText(buffer);
textView.setMovementMethod(LinkMovementMethod.getInstance());

и TextView не должен иметь android:autolink. Нет необходимости android:linksClickable="true", и это правда по умолчанию.


вот как я решил кликабельные и видимые ссылки в TextView (по коду)

private void setAsLink(TextView view, String url){
        Pattern pattern = Pattern.compile(url);
        Linkify.addLinks(view, pattern, "http://");
        view.setText(Html.fromHtml("<a href='http://"+url+"'>http://"+url+"</a>"));
    }

использовать ниже код:

String html = "<a href=\"http://yourdomain.com\">Your Domain Name</a>"
TextView textview = (TextView) findViewById(R.id.your_textview_id);
textview.setMovementMethod(LinkMovementMethod.getInstance());
textview.setText(Html.fromHtml(html));

причина, по которой у вас проблема, заключается в том, что он пытается соответствовать только "голым" адресам. такие вещи, как "www.google.com" или "http://www.google.com".

запуск текста через HTML-код.fromHtml() следует сделать трюк. Вы должны делать это программно, но это работает.


Автоссылка телефон не работал для меня. Следующее сработало как заклинание,

TextView tv = (TextView) findViewById(R.id.emergencynos);
String html2="<br><br>Fire - <b><a href=tel:997>997</a> </b></br></br>";        
tv.append(Html.fromHtml(html2));
tv.setMovementMethod(LinkMovementMethod.getInstance());

добавьте CDATA в свой строковый ресурс

строки.в XML

<string name="txtCredits"><![CDATA[<a href=\"http://www.google.com\">Google</a>]]></string>

не знаю, стоит ли добавлять еще один ответ, но на всякий случай...

Я должен был выследить это в нескольких местах, но, наконец, получил эту версию кода для работы.

строки.XML-код:

<string name="name1">&lt;a href="http://www.google.com">link text1&lt;/a></string>
<string name="name2">&lt;a href="http://www.google.com">link text2&lt;/a></string>

myactivity.XML-код:

<TextView 
    android:id="@+id/textview1"
    android:layout_height="wrap_content"
    android:layout_width="wrap_content"
    android:layout_marginTop="5dp" />

<TextView 
    android:id="@+id/textview2"
    android:layout_height="wrap_content"
    android:layout_width="wrap_content"
    android:layout_marginTop="5dp" />

myactivty.java (в onCreate ()):

TextView tv1 = (TextView)findViewById(R.id.textview1);
TextView tv2 = (TextView)findViewById(R.id.textview2);

tv1.setText(Html.fromHtml(getResources().getString(R.string.name1)));
tv2.setText(Html.fromHtml(getResources().getString(R.string.name2)));
tv1.setMovementMethod(LinkMovementMethod.getInstance());
tv2.setMovementMethod(LinkMovementMethod.getInstance());

это создаст две интерактивные гиперссылки с текстом link text1 и link text2 которые перенаправляют пользователя в google.


Если вы используете TextView на основе XML, для вашего требования вам нужно сделать только две вещи:

  1. определите свою ссылку в строке, например: "это моя Веб-страница." Вы можете добавить его в XML или в коде.

  2. в xml, который имеет TextView, добавьте следующее:


android:linksClickable="true"

android:autoLink="web"

Я использую автолинк для "автоматического подчеркивания" текста, но только что сделал "onClick", который управляет им. (Я сам столкнулся с этой проблемой)

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_marginRight="10dp"
            android:textSize="18dp"
            android:autoLink="all"
            android:text="@string/twitter"
            android:onClick="twitter"/>

public void twitter (View view)
    {
        try
        {
            Intent browserIntent = new Intent(Intent.ACTION_VIEW, Uri.parse("http://twitter.com/onaclovtech"));
            startActivity(browserIntent);

        }
        finally
        {
        }
    }

не требует никаких разрешений, поскольку вы передаете намерение приложениям, которые управляют этими ресурсами (т. е. браузером).

Это было то, что работал для меня. Удача.


[Tested in Pre-lollipop as well as in Lollipop and above]

вы можете получить строку HTML из бэкэнда или из файлов ресурсов. Если вы помещаете текст в строку ресурса, обязательно добавьте CDATA tag:

<string name="your_text">![CDATA[...<a href="your_link">Link Title</a>  ...]]</string>

затем в коде вам нужно получить строку и назначить ее как HTML и установить метод перемещения ссылки:

String yourText = getString(R.string.your_text);

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
   textView.setText(Html.fromHtml(yourText, Html.FROM_HTML_MODE_COMPACT));
} else {
   textView.setText(Html.fromHtml(yourText));
}

try {
   subtext.setMovementMethod(LinkMovementMethod.getInstance());
} catch (Exception e) {
   //This code seems to crash in some Samsung devices.
   //You can handle this edge case base on your needs.
}

просто потратил столько времени, чтобы понять, что вам нужно использовать getText(R. string.что угодно) вместо getString (R. string.все.)..

в любом случае, вот как я получил свой рабочий. С несколькими гиперссылками в том же текстовом представлении тоже.

    TextView termsTextView = (TextView) getActivity().findViewById(R.id.termsTextView);
    termsTextView.append("By registering your account, you agree to our ");
    termsTextView.append(getText(R.string.terms_of_service));
    termsTextView.append(", ");
    termsTextView.append(getText(R.string.fees));
    termsTextView.append(", and the ");
    termsTextView.append(getText(R.string.stripe_connected_account_agreement));

    termsTextView.setMovementMethod(LinkMovementMethod.getInstance());



            <TextView
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:id="@+id/termsTextView"/>

пример строки

    <string name="stripe_connected_account_agreement"><a href="https://stripe.com/connect/account-terms">Stripe Connected Account Agreement</a></string>

мой код был такой:

<TextView
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:id="@+id/link"
    android:text="@string/forgot"
    android:layout_marginTop="16dp"
    android:gravity="center"
    android:linksClickable="true"/>

мой Java-код был такой:

/*TextView action*/
        TextView textView = (TextView) findViewById(R.id.link);
        textView.setMovementMethod(LinkMovementMethod.getInstance());
        textView.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                startActivity(new Intent(LoginActivity.this,forgot.class));
            }
        });  

Это просто указывает на ссылку на другое действие. Но эта ссылка является кликабельной и работает плавно. Протестировано в Android Studio 1.5 (предварительный просмотр)