Linkify.java

 1/*
 2 * Copyright (c) 2018, Daniel Gultsch All rights reserved.
 3 *
 4 * Redistribution and use in source and binary forms, with or without modification,
 5 * are permitted provided that the following conditions are met:
 6 *
 7 * 1. Redistributions of source code must retain the above copyright notice, this
 8 * list of conditions and the following disclaimer.
 9 *
10 * 2. Redistributions in binary form must reproduce the above copyright notice,
11 * this list of conditions and the following disclaimer in the documentation and/or
12 * other materials provided with the distribution.
13 *
14 * 3. Neither the name of the copyright holder nor the names of its contributors
15 * may be used to endorse or promote products derived from this software without
16 * specific prior written permission.
17 *
18 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
19 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
20 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
21 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
22 * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
23 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
24 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
25 * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
27 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28 */
29
30package de.gultsch.common;
31
32import android.net.Uri;
33import android.text.Spannable;
34import com.google.common.base.Splitter;
35import com.google.common.collect.ImmutableList;
36import com.google.common.collect.Iterables;
37import eu.siacs.conversations.utils.XmppUri;
38import java.util.List;
39import java.util.Objects;
40
41public class Linkify {
42
43    private static final android.text.util.Linkify.MatchFilter MATCH_FILTER =
44            (s, start, end) -> isPassAdditionalValidation(s.subSequence(start, end).toString());
45
46    private static boolean isPassAdditionalValidation(final String match) {
47        final var scheme = Iterables.getFirst(Splitter.on(':').limit(2).splitToList(match), null);
48        if (scheme == null) {
49            return false;
50        }
51        return switch (scheme) {
52            case "tel" -> Patterns.URI_TEL.matcher(match).matches();
53            case "http", "https" -> Patterns.URI_HTTP.matcher(match).matches();
54            case "geo" -> Patterns.URI_GEO.matcher(match).matches();
55            case "xmpp" -> new XmppUri(Uri.parse(match)).isValidJid();
56            case "web+ap" -> {
57                if (Patterns.URI_WEB_AP.matcher(match).matches()) {
58                    final var webAp = new MiniUri(match);
59                    // TODO once we have fragment support check that there aren't any
60                    yield Objects.nonNull(webAp.getAuthority()) && webAp.getParameter().isEmpty();
61                } else {
62                    yield false;
63                }
64            }
65            default -> true;
66        };
67    }
68
69    public static void addLinks(final Spannable body) {
70        android.text.util.Linkify.addLinks(body, Patterns.URI_GENERIC, null, MATCH_FILTER, null);
71    }
72
73    public static List<MiniUri> getLinks(final String body) {
74        final var builder = new ImmutableList.Builder<MiniUri>();
75        final var matcher = Patterns.URI_GENERIC.matcher(body);
76        while (matcher.find()) {
77            final var match = matcher.group();
78            if (isPassAdditionalValidation(match)) {
79                builder.add(new MiniUri(match));
80            }
81        }
82        return builder.build();
83    }
84}