Como posso obter hiperlinks clicáveis ​​no AlertDialog a partir de um recurso de string?

134

O que estou tentando realizar é ter hiperlinks clicáveis ​​no texto da mensagem exibido por um AlertDialog. Enquanto a AlertDialogimplementação sublinha e colore alegremente quaisquer hiperlinks (definidos usando <a href="...">o recurso de cadeia passado para Builder.setMessage) fornecido, os links não se tornam clicáveis.

O código que estou usando atualmente se parece com isso:

new AlertDialog.Builder(MainActivity.this).setTitle(
        R.string.Title_About).setMessage(
        getResources().getText(R.string.about))
        .setPositiveButton(android.R.string.ok, null)
        .setIcon(R.drawable.icon).show();

Gostaria de evitar o uso de um WebViewpara exibir apenas um trecho de texto.

Thilo-Alexander Ginkel
fonte
Oi! Você realmente obtém resultados declarados ("sublinha alegremente e colore qualquer hiperlink")? Qual valor de string você está passando?
Maksym Gontar
1
Sim, a chave é exibir a mensagem em um recurso de sequência, que Resources.getText (...) retorna como um android.text.Spanned preservando a formatação HTML. Assim que você o converte em uma String, a mágica desaparece.
Thilo-Alexander Ginkel

Respostas:

128

Se você está apenas mostrando algum texto e URL (s) na sua caixa de diálogo, talvez a solução seja mais simples

public static class MyOtherAlertDialog {

 public static AlertDialog create(Context context) {
  final TextView message = new TextView(context);
  // i.e.: R.string.dialog_message =>
            // "Test this dialog following the link to dtmilano.blogspot.com"
  final SpannableString s = 
               new SpannableString(context.getText(R.string.dialog_message));
  Linkify.addLinks(s, Linkify.WEB_URLS);
  message.setText(s);
  message.setMovementMethod(LinkMovementMethod.getInstance());

  return new AlertDialog.Builder(context)
   .setTitle(R.string.dialog_title)
   .setCancelable(true)
   .setIcon(android.R.drawable.ic_dialog_info)
   .setPositiveButton(R.string.dialog_action_dismiss, null)
   .setView(message)
   .create();
 }
}

Como mostrado aqui http://picasaweb.google.com/lh/photo/up29wTQeK_zuz-LLvre9wQ?feat=directlink

Caixa de diálogo de alerta com links clicáveis

Diego Torres Milano
fonte
1
você provavelmente deseja criar um arquivo de layout, inflá-lo e usá-lo como visualização.
22612 Jeffrey Blattman
5
Como você definiria o estilo do textView para corresponder ao estilo usado por padrão?
android developer
3
Então, eu estou recebendo erroCalling startActivity() from outside of an Activity context requires the FLAG_ACTIVITY_NEW_TASK flag. Is this really what you want?
ViliusK
207

Eu realmente não gostei da resposta atualmente mais popular porque altera significativamente a formatação da mensagem na caixa de diálogo.

Aqui está uma solução que vinculará o texto da caixa de diálogo sem alterar o estilo do texto:

    // Linkify the message
    final SpannableString s = new SpannableString(msg); // msg should have url to enable clicking
    Linkify.addLinks(s, Linkify.ALL);

    final AlertDialog d = new AlertDialog.Builder(activity)
        .setPositiveButton(android.R.string.ok, null)
        .setIcon(R.drawable.icon)
        .setMessage( s )
        .create();

    d.show();

    // Make the textview clickable. Must be called after show()
    ((TextView)d.findViewById(android.R.id.message)).setMovementMethod(LinkMovementMethod.getInstance());
empapar
fonte
5
Saúde, Trabalhou para mim de dentro onCreateDialogde um DialogFragment. Só tinha de definir código clicável no onStartdado que showtinha sido chamado para invocar o DialogFragment
PJL
5
Isso parece tornar o TextView inteiro clicável, em vez de apenas os links ... Existe alguma maneira de contornar isso?
Kavi
1
Concordo que esta é uma opção muito melhor, pois a resposta original está atrapalhando visualmente o diálogo.
Hcpl 23/03/12
1
A exibição retornada por findViewById deve ser verificada com "instanceof TextView", porque não há garantia de que a implementação não seja alterada.
Denis Gladkiy
6
Como apontado em outro lugar, se estiver usando setMessage(R.string.something), não é necessário vincular explicitamente. Também não é necessário para create()o objeto AlertDialog antes da chamada show()(ele pode ser chamado no Builder) e, como show()retorna o objeto de diálogo, ele findViewById(android.R.id.message)pode ser encadeado. Envolva tudo em uma tentativa de captura, apenas no caso de a exibição da mensagem não ser um TextView, e você tenha uma formulação concisa.
Pierre-Luc Paour
50

Isso deve fazer com que as <a href>tags também sejam destacadas. Observe que acabei de adicionar algumas linhas ao código do empacotamento. então credito a ele

final AlertDialog d = new AlertDialog.Builder(this)
 .setPositiveButton(android.R.string.ok, null)
 .setIcon(R.drawable.icon)
 .setMessage(Html.fromHtml("<a href=\"http://www.google.com\">Check this link out</a>"))
 .create();
d.show();
// Make the textview clickable. Must be called after show()   
    ((TextView)d.findViewById(android.R.id.message)).setMovementMethod(LinkMovementMethod.getInstance());
DeRagan
fonte
10
Se você usa html em strings.xml, não precisa usar Html.fromHtml. setMessage(R.string.cool_link)trabalha com<string name="cool_link"><a href="http://www.google.com">Check this link out</a></string>
idbrii 17/06
2
Isso é verdade. Quando você combina os dois métodos (Html.fromHtml e tag HTML em strings.xml), ele não funciona.
JerabekJakub
Já faz um tempo e fromHtml está obsoleto, e agora?
Menasheh
Você ainda pode usar fromHtml: developer.android.com/reference/android/text/… , int) Simplesmente useHtml.fromHtml("string with links", Html.FROM_HTML_MODE_LEGACY)
BVB
2
setMovementMethod()é a parte importante aqui, caso contrário, o URL não será clicável.
Scai
13

Na verdade, se você quiser simplesmente usar uma string sem lidar com todas as visualizações, a maneira mais rápida é encontrar a visualização de texto da mensagem e vinculá-la:

d.setMessage("Insert your cool string with links and stuff here");
Linkify.addLinks((TextView) d.findViewById(android.R.id.message), Linkify.ALL);
vinc3m1
fonte
12

JFTR, aqui vem a solução que eu descobri depois de algum tempo:

View view = View.inflate(MainActivity.this, R.layout.about, null);
TextView textView = (TextView) view.findViewById(R.id.message);
textView.setMovementMethod(LinkMovementMethod.getInstance());
textView.setText(R.string.Text_About);
new AlertDialog.Builder(MainActivity.this).setTitle(
        R.string.Title_About).setView(view)
        .setPositiveButton(android.R.string.ok, null)
        .setIcon(R.drawable.icon).show();

O about.xml correspondente emprestado como um fragmento das fontes do Android se parece com isso:

<?xml version="1.0" encoding="utf-8"?>
<ScrollView xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/scrollView" android:layout_width="fill_parent"
    android:layout_height="wrap_content" android:paddingTop="2dip"
    android:paddingBottom="12dip" android:paddingLeft="14dip"
    android:paddingRight="10dip">
    <TextView android:id="@+id/message" style="?android:attr/textAppearanceMedium"
        android:layout_width="fill_parent" android:layout_height="wrap_content"
        android:padding="5dip" android:linksClickable="true" />
</ScrollView>

As partes importantes são definir linksClickable como true e setMovementMethod (LinkMovementMethod.getInstance ()).

Thilo-Alexander Ginkel
fonte
Obrigado, isso resolveu o problema para mim. No meu caso, não era necessário setLinksClickable(true)(acho que já era), mas setMovementMethod(...)fazia toda a diferença.
Larsh
10

Ao invés de ...

AlertDialog.Builder dialogBuilder = new AlertDialog.Builder(this);
dialogBuilder.setTitle(R.string.my_title);
dialogBuilder.setMessage(R.string.my_text);

... agora uso:

AlertDialog.Builder dialogBuilder = new AlertDialog.Builder(this);
dialogBuilder.setTitle(R.string.my_title);
TextView textView = new TextView(this);
textView.setMovementMethod(LinkMovementMethod.getInstance());
textView.setText(R.string.my_text);
dialogBuilder.setView(textView);
neu242
fonte
Ei, seu soln funciona. você sabe por que o texto inteiro pisca quando clica no link?
31412 aimango
Ele não rola como o padrão.
Meow Cat 2012
7

Maneira mais simples:

final AlertDialog dlg = new AlertDialog.Builder(this)
                .setTitle(R.string.title)
                .setMessage(R.string.message)
                .setNeutralButton(R.string.close_button, null)
                .create();
        dlg.show();
        // Important! android.R.id.message will be available ONLY AFTER show()
        ((TextView)dlg.findViewById(android.R.id.message)).setMovementMethod(LinkMovementMethod.getInstance());
Gabriele
fonte
6

Toda a resposta acima não removerá a tag html como, etc, se a string especificada contiver, tentei remover todas as tags, e isso funciona bem para mim

AlertDialog.Builder builder = new AlertDialog.Builder(ctx);
        builder.setTitle("Title");

        LayoutInflater inflater = (LayoutInflater) ctx.getSystemService(LAYOUT_INFLATER_SERVICE);
        View layout = inflater.inflate(R.layout.custom_dialog, null);

        TextView text = (TextView) layout.findViewById(R.id.text);
        text.setMovementMethod(LinkMovementMethod.getInstance());
        text.setText(Html.fromHtml("<b>Hello World</b> This is a test of the URL <a href=http://www.example.com> Example</a><p><b>This text is bold</b></p><p><em>This text is emphasized</em></p><p><code>This is computer output</code></p><p>This is<sub> subscript</sub> and <sup>superscript</sup></p>";));
        builder.setView(layout);
AlertDialog alert = builder.show();

e o custom_dialog seria como;

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:id="@+id/layout_root"
              android:orientation="horizontal"
              android:layout_width="fill_parent"
              android:layout_height="fill_parent"
              android:padding="10dp"
              >

    <TextView android:id="@+id/text"
              android:layout_width="wrap_content"
              android:layout_height="fill_parent"
              android:textColor="#FFF"
              />
</LinearLayout>

O código acima removerá toda a tag html e mostra Exemplo como URL capaz de clicar em todos os outros no texto de formatação html especificado.

Lakshmanan
fonte
5

Eu não estava realmente satisfeito com as respostas atuais. Há duas coisas importantes quando você deseja hiperlinks clicáveis ​​no estilo href dentro de um AlertDialog:

  1. Defina o conteúdo como Visualização e não com setMessage(…), pois somente as Visualizações permitem conteúdo HTML clicável
  2. Defina o método de movimento correto ( setMovementMethod(…))

Aqui está um exemplo mínimo de trabalho:

strings.xml

<string name="dialogContent">
    Cool Links:\n
    <a href="http://stackoverflow.com">Stackoverflow</a>\n
    <a href="http://android.stackexchange.com">Android Enthusiasts</a>\n
</string>

MyActivity.java


public void showCoolLinks(View view) {
   final TextView textView = new TextView(this);
   textView.setText(R.string.dialogContent);
   textView.setMovementMethod(LinkMovementMethod.getInstance()); // this is important to make the links clickable
   final AlertDialog alertDialog = new AlertDialog.Builder(this)
       .setPositiveButton("OK", null)
       .setView(textView)
       .create();
   alertDialog.show()
}
Fluxo
fonte
3

Eu verifiquei muitas perguntas e respostas, mas não funciona. Eu mesmo fiz. Este é o trecho de código em MainActivity.java.

private void skipToSplashActivity()
{

    final TextView textView = new TextView(this);
    final SpannableString str = new SpannableString(this.getText(R.string.dialog_message));

    textView.setText(str);
    textView.setMovementMethod(LinkMovementMethod.getInstance());

    ....
}

Coloque essa tag em res \ values ​​\ String.xml

<string name="dialog_message"><a href="http://www.nhk.or.jp/privacy/english/">NHK Policy on Protection of Personal Information</a></string>
Hiroto
fonte
2

Combinei algumas das opções discutidas acima para criar esta função que funciona para mim. passe o resultado para o método SetView () do construtor de diálogo.

public ScrollView LinkifyText(String message) 
{
    ScrollView svMessage = new ScrollView(this); 
    TextView tvMessage = new TextView(this);

    SpannableString spanText = new SpannableString(message);

    Linkify.addLinks(spanText, Linkify.ALL);
    tvMessage.setText(spanText);
    tvMessage.setMovementMethod(LinkMovementMethod.getInstance());

    svMessage.setPadding(14, 2, 10, 12);
    svMessage.addView(tvMessage);

    return svMessage;
}
Alex
fonte
2

Se você estiver usando um DialogFragment, esta solução deve ajudar.

public class MyDialogFragment extends DialogFragment {
    @Override
    public Dialog onCreateDialog(Bundle savedInstanceState) {

        // dialog_text contains "This is a http://test.org/"
        String msg = getResources().getString(R.string.dialog_text);
        SpannableString spanMsg = new SpannableString(msg);
        Linkify.addLinks(spanMsg, Linkify.ALL);

        AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());
        builder.setTitle(R.string.dialog_title)
            .setMessage(spanMsg)
            .setPositiveButton(R.string.ok, null);
        return builder.create();
    }

    @Override
    public void onStart() {
        super.onStart();

        // Make the dialog's TextView clickable
        ((TextView)this.getDialog().findViewById(android.R.id.message))
                .setMovementMethod(LinkMovementMethod.getInstance());
    }
}
tronman
fonte
Se você definir SpannableString como a mensagem da caixa de diálogo, o link será destacado, mas não poderá ser clicado.
bk138
@ bk138 A chamada para .setMovementMethod () em onStart () é o que torna o link clicável.
tronman
2

Para mim, a melhor solução para criar um diálogo de política de privacidade é:

    private void showPrivacyDialog() {
    if (!PreferenceManager.getDefaultSharedPreferences(getApplicationContext()).getBoolean(PRIVACY_DIALOG_SHOWN, false)) {

        String privacy_pol = "<a href='https://sites.google.com/view/aiqprivacypolicy/home'> Privacy Policy </a>";
        String toc = "<a href='https://sites.google.com/view/aiqprivacypolicy/home'> T&C </a>";
        AlertDialog dialog = new AlertDialog.Builder(this)
                .setMessage(Html.fromHtml("By using this application, you agree to " + privacy_pol + " and " + toc + " of this application."))
                .setPositiveButton("ACCEPT", new DialogInterface.OnClickListener() {
                    public void onClick(DialogInterface dialog, int which) {
                        PreferenceManager.getDefaultSharedPreferences(getApplicationContext()).edit().putBoolean(PRIVACY_DIALOG_SHOWN, true).apply();
                    }
                })
                .setNegativeButton("DECLINE", null)
                .setCancelable(false)
                .create();

        dialog.show();
        TextView textView = dialog.findViewById(android.R.id.message);
        textView.setLinksClickable(true);
        textView.setClickable(true);
        textView.setMovementMethod(LinkMovementMethod.getInstance());
    }
}

verifique o exemplo de trabalho: link do aplicativo

Minion
fonte
1

Eu faço isso especificando a caixa de alerta em um recurso XML e carregando isso. Veja, por exemplo, o about.xml (consulte o ID ABOUT_URL) que é instanciado perto do final do ChandlerQE.java . As partes relevantes do código java:

LayoutInflater inflater = 
    (LayoutInflater)getSystemService(Context.LAYOUT_INFLATER_SERVICE);
View view = (View) inflater.inflate(R.layout.about, null);

new AlertDialog.Builder(ChandlerQE.this)
.setTitle(R.string.about)
.setView(view)
Heikki Toivonen
fonte
link está morto, você pode consertar?
Bijoy Thangaraj
1

Esta é a minha solução. Ele cria um link normal sem as tags html envolvidas e sem qualquer URL visível. Ele também mantém o design intacto.

SpannableString s = new SpannableString("This is my link.");
s.setSpan(new URLSpan("http://www.google.com"), 11, 15, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);

AlertDialog.Builder builder;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
    builder = new AlertDialog.Builder(this, android.R.style.Theme_Material_Dialog_Alert);
} else {
    builder = new AlertDialog.Builder(this);
}

final AlertDialog d = builder
        .setPositiveButton("CLOSE", new DialogInterface.OnClickListener() {
            public void onClick(DialogInterface dialog, int which) {
                // Do nothing, just close
            }
        })
        .setNegativeButton("SHARE", new DialogInterface.OnClickListener() {
            public void onClick(DialogInterface dialog, int which) {
                // Share the app
                share("Subject", "Text");
            }
        })
        .setIcon(R.drawable.photo_profile)
        .setMessage(s)
        .setTitle(R.string.about_title)
        .create();

d.show();

((TextView)d.findViewById(android.R.id.message)).setMovementMethod(LinkMovementMethod.getInstance());
John T
fonte
1
Obrigado, Apenas para adicionar setSpan (URL, startPoint, endPoint, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE). Aqui startPoint e endPoint são palavras que serão destacados para clicar
Manish
0

A maneira mais fácil e mais curta é assim

Link para Android na caixa de diálogo

((TextView) new AlertDialog.Builder(this)
.setTitle("Info")
.setIcon(android.R.drawable.ic_dialog_info)
.setMessage(Html.fromHtml("<p>Sample text, <a href=\"http://google.nl\">hyperlink</a>.</p>"))
.show()
// Need to be called after show(), in order to generate hyperlinks
.findViewById(android.R.id.message))
.setMovementMethod(LinkMovementMethod.getInstance());
Javier Castellanos Cruz
fonte
Você pode me dizer como fazer isso no Kotlin?
Thomas Williams
Sinto muito. Não sei Kotlin
Javier Castellanos Cruz