Use data binding to provide EditText input filters

Signed-off-by: Jason A. Donenfeld <Jason@zx2c4.com>
This commit is contained in:
Samuel Holland 2017-08-22 21:48:42 -05:00
parent 5d04714eea
commit fb919a7226
6 changed files with 71 additions and 22 deletions

View File

@ -4,7 +4,7 @@ import android.databinding.BindingAdapter;
import android.databinding.ObservableArrayMap; import android.databinding.ObservableArrayMap;
import android.databinding.ObservableList; import android.databinding.ObservableList;
import android.graphics.Typeface; import android.graphics.Typeface;
import android.widget.EditText; import android.text.InputFilter;
import android.widget.ListView; import android.widget.ListView;
import android.widget.TextView; import android.widget.TextView;
@ -62,8 +62,13 @@ public final class BindingAdapters {
} }
} }
@BindingAdapter({"filter"})
public static void setFilter(final TextView view, final InputFilter filter) {
view.setFilters(new InputFilter[]{filter});
}
@BindingAdapter({"android:textStyle"}) @BindingAdapter({"android:textStyle"})
public static void textStyleBinding(final TextView view, final Typeface typeface) { public static void setTextStyle(final TextView view, final Typeface typeface) {
view.setTypeface(typeface); view.setTypeface(typeface);
} }

View File

@ -2,8 +2,6 @@ package com.wireguard.android;
import android.content.Context; import android.content.Context;
import android.os.Bundle; import android.os.Bundle;
import android.text.InputFilter;
import android.text.LoginFilter;
import android.view.LayoutInflater; import android.view.LayoutInflater;
import android.view.Menu; import android.view.Menu;
import android.view.MenuInflater; import android.view.MenuInflater;
@ -11,7 +9,6 @@ import android.view.MenuItem;
import android.view.View; import android.view.View;
import android.view.ViewGroup; import android.view.ViewGroup;
import android.view.inputmethod.InputMethodManager; import android.view.inputmethod.InputMethodManager;
import android.widget.EditText;
import android.widget.Toast; import android.widget.Toast;
import com.wireguard.android.databinding.ConfigEditFragmentBinding; import com.wireguard.android.databinding.ConfigEditFragmentBinding;
@ -45,18 +42,6 @@ public class ConfigEditFragment extends BaseConfigFragment {
final Bundle savedInstanceState) { final Bundle savedInstanceState) {
final ConfigEditFragmentBinding binding = final ConfigEditFragmentBinding binding =
ConfigEditFragmentBinding.inflate(inflater, parent, false); ConfigEditFragmentBinding.inflate(inflater, parent, false);
final EditText configNameText = binding.getRoot().findViewById(R.id.config_name_text);
configNameText.setFilters(new InputFilter[]{
new InputFilter.LengthFilter(16),
new LoginFilter.UsernameFilterGeneric() {
@Override
public boolean isAllowed(final char c) {
return Character.isLetterOrDigit(c) || "_=+.-".indexOf(c) != -1;
}
}
});
final EditText privateKeyText = binding.getRoot().findViewById(R.id.private_key_text);
privateKeyText.setFilters(new InputFilter[]{new KeyInputFilter()});
binding.setConfig(localConfig); binding.setConfig(localConfig);
return binding.getRoot(); return binding.getRoot();
} }

View File

@ -9,7 +9,12 @@ import com.wireguard.crypto.KeyEncoding;
/** /**
* InputFilter for entering WireGuard private/public keys encoded with base64. * InputFilter for entering WireGuard private/public keys encoded with base64.
*/ */
class KeyInputFilter implements InputFilter {
public class KeyInputFilter implements InputFilter {
public static KeyInputFilter newInstance() {
return new KeyInputFilter();
}
@Override @Override
public CharSequence filter(final CharSequence source, public CharSequence filter(final CharSequence source,
final int sStart, final int sEnd, final int sStart, final int sEnd,

View File

@ -0,0 +1,46 @@
package com.wireguard.android;
import android.text.InputFilter;
import android.text.SpannableStringBuilder;
import android.text.Spanned;
import com.wireguard.config.Config;
/**
* InputFilter for entering WireGuard configuration names (Linux interface names).
*/
public class NameInputFilter implements InputFilter {
public static NameInputFilter newInstance() {
return new NameInputFilter();
}
@Override
public CharSequence filter(final CharSequence source,
final int sStart, final int sEnd,
final Spanned dest,
final int dStart, final int dEnd) {
SpannableStringBuilder replacement = null;
int rIndex = 0;
final int dLength = dest.length();
for (int sIndex = sStart; sIndex < sEnd; ++sIndex) {
final char c = source.charAt(sIndex);
final int dIndex = dStart + (sIndex - sStart);
// Restrict characters to those valid in interfaces.
// Ensure adding this character does not push the length over the limit.
if ((dIndex < Config.NAME_MAX_LENGTH && isAllowed(c)) &&
dLength + (sIndex - sStart) < Config.NAME_MAX_LENGTH) {
++rIndex;
} else {
if (replacement == null)
replacement = new SpannableStringBuilder(source, sStart, sEnd);
replacement.delete(rIndex, rIndex + 1);
}
}
return replacement;
}
private boolean isAllowed(final char c) {
return Character.isLetterOrDigit(c) || "_=+.-".indexOf(c) >= 0;
}
}

View File

@ -22,10 +22,11 @@ import java.util.regex.Pattern;
public class Config extends BaseObservable public class Config extends BaseObservable
implements Comparable<Config>, Copyable<Config>, Observable { implements Comparable<Config>, Copyable<Config>, Observable {
public static final int NAME_MAX_LENGTH = 16;
private static final Pattern PATTERN = Pattern.compile("^[a-zA-Z0-9_=+.-]{1,16}$"); private static final Pattern PATTERN = Pattern.compile("^[a-zA-Z0-9_=+.-]{1,16}$");
private static boolean isNameValid(final String name) { private static boolean isNameValid(final String name) {
return PATTERN.matcher(name).matches(); return name.length() <= NAME_MAX_LENGTH && PATTERN.matcher(name).matches();
} }
private final Interface iface = new Interface(); private final Interface iface = new Interface();

View File

@ -1,8 +1,13 @@
<?xml version="1.0" encoding="utf-8"?> <?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"> <layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<data> <data>
<import type="com.wireguard.android.KeyInputFilter" />
<import type="com.wireguard.android.NameInputFilter" />
<variable <variable
name="config" name="config"
type="com.wireguard.config.Config" /> type="com.wireguard.config.Config" />
@ -32,7 +37,8 @@
android:layout_height="wrap_content" android:layout_height="wrap_content"
android:layout_below="@+id/config_name_label" android:layout_below="@+id/config_name_label"
android:inputType="textCapWords" android:inputType="textCapWords"
android:text="@={config.name}" /> android:text="@={config.name}"
app:filter="@{NameInputFilter.newInstance()}" />
<TextView <TextView
android:id="@+id/private_key_label" android:id="@+id/private_key_label"
@ -50,7 +56,8 @@
android:layout_below="@+id/private_key_label" android:layout_below="@+id/private_key_label"
android:layout_toStartOf="@+id/generate_private_key_button" android:layout_toStartOf="@+id/generate_private_key_button"
android:inputType="textVisiblePassword" android:inputType="textVisiblePassword"
android:text="@={config.interface.privateKey}" /> android:text="@={config.interface.privateKey}"
app:filter="@{KeyInputFilter.newInstance()}" />
<Button <Button
android:id="@+id/generate_private_key_button" android:id="@+id/generate_private_key_button"