Android - Butter Knife¶
Spis treści¶
Wstęp
Dodanie biblioteki do projektu
Wiązanie w aktywności
Wiązanie we fragmentach
Dodatek Android ButterKnife Zelezny
Wstęp¶
Podczas pisania aplikacji Android, programista jest zmuszony do powtarzania tych samych fragmentów kodu podczas tworzenia powiązania elementu GUI czy zasobu z odpowiednim polem klasy. Butter Knife eliminuje ten problem - odpowiednie wiązania tworzy się używając adnotacji. Na ich podstawie, w trakcie procesu kompilacji, generowany jest odpowiedni kod. Wykorzystanie biblioteki nie wpływa znacząco na wydajność aplikacji.
Dodanie biblioteki do projektu¶
Bibliotekę dodamy do projektu poprzez modyfikację pliku build.gradle znajdującego się w katalogu app. W sekcji dependencies dodajemy te dwie linie (X.Y.Z należy zamienić na numer konkretnej wersji):
compile 'com.jakewharton:butterknife:X.Y.Z'
annotationProcessor 'com.jakewharton:butterknife-compiler:X.Y.Z'
W nowej wersji Gradle (od 3.0) są to:
implementation 'com.jakewharton:butterknife:X.Y.Z'
annotationProcessor 'com.jakewharton:butterknife-compiler:X.Y.Z'
Aktualny numer wersji można sprawdzić na stronie projektu, w momencie pisania tego artykułu jest to 8.8.1. Poniżej przykładowy plik konfiguracyjny:
apply plugin: 'com.android.application'
android {
compileSdkVersion 26
defaultConfig {
applicationId "org.chyla.butterknifetestapp"
minSdkVersion 14
targetSdkVersion 26
versionCode 1
versionName "1.0"
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}
}
dependencies {
implementation fileTree(dir: 'libs', include: ['*.jar'])
implementation 'com.android.support:appcompat-v7:26.1.0'
implementation 'com.android.support.constraint:constraint-layout:1.0.2'
testImplementation 'junit:junit:4.12'
androidTestImplementation 'com.android.support.test:runner:1.0.1'
androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.1'
implementation 'com.jakewharton:butterknife:8.8.1'
annotationProcessor 'com.jakewharton:butterknife-compiler:8.8.1'
}
Wiązanie w aktywności¶
Tradycyjnie do utworzenia wiązania między widokiem a składnikiem klasy należy użyć metody findViewById, jak w poniższym przykładzie:
1public class LoginActivity extends AppCompatActivity {
2
3 private EditText mPasswordView;
4
5 @Override
6 protected void onCreate(Bundle savedInstanceState) {
7 super.onCreate(savedInstanceState);
8 setContentView(R.layout.activity_login);
9
10 mPasswordView = (EditText) findViewById(R.id.password);
11 }
12}
Każdej deklaracji (linia 3) odpowiada przypisanie widoku (linia 10). Zapis ten możemy skrócić używając BindView:
1public class LoginActivity extends AppCompatActivity {
2
3 @BindView(R.id.password) EditText mPasswordView;
4
5 @Override
6 protected void onCreate(Bundle savedInstanceState) {
7 super.onCreate(savedInstanceState);
8 setContentView(R.layout.activity_login);
9
10 ButterKnife.bind(this);
11 }
12
13}
Zmiany nastąpiły w linii 3 i 10. W linii 3 użyta zostałą adnotacja BindView, usunięty został również modyfikator dostępu private. W linii 10 następuje jednorazowa (dla danej klasy) inicjalizacja wiązań.
Oprócz widoków dowiązane mogą zostać zasoby:
public class LoginActivity extends AppCompatActivity {
@BindString(R.string.title) String title;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_login);
ButterKnife.bind(this);
}
}
Biblioteka pozwala także powiązać przycisk z odpowiednią funkcją:
public class LoginActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_login);
ButterKnife.bind(this);
}
@OnClick(R.id.email_sign_in_button)
void attemptLogin() {
// action
}
// ...
}
Powyżej przedstawiłem, moim zdaniem, najczęściej wykorzystywane rodzaje wiązań. Zachęcam do przejrzenia strony domowej projektu, gdzie zostały przedstawione pozostałe rodzaje (np. wiązania grupowe).
Wiązanie we fragmentach¶
We fragmentach również istnieje możliwość stworzenia powiązania. W tym celu należy:
do metody bind przekazać dodatkowy argument wskazujący na widok (na przykładzie poniżej, linia 13),
podczas niszczenia widoku usunąć powiązanie (na przykładzie poniżej, linia 20).
Przykład wiązania utworzonego we fragmencie:
1public class SearchPhotosFragment extends Fragment {
2
3 private Unbinder unbinder;
4
5 @BindView(R.id.edittext_tags)
6 EditText tagsEditText;
7
8 @Override
9 public View onCreateView(LayoutInflater inflater, ViewGroup container,
10 Bundle savedInstanceState) {
11 View view = inflater.inflate(R.layout.fragment_inspect_photos, container, false);
12
13 unbinder = ButterKnife.bind(this, view);
14
15 return view;
16 }
17
18 @Override public void onDestroyView() {
19 super.onDestroyView();
20 unbinder.unbind();
21 }
22
23 @OnClick(R.id.button_search)
24 void searchPhotos() {
25 }
26
27 // ...
28}
Powód, przez który wymagana jest operacja samodzielnego usunięcia wiązania, został wyjaśniony przez egor-n:
There can be a case when fragment’s view is destroyed, but the fragment instance is still present. For example, when fragment goes into the back stack - onDestroyView() is called, but onDestroy() is not. [egor-n]
oraz artem-zinnatullin:
„System” will not clear view references in onDestroy(), they’re just regular references and they would be freed only if GC will collect them. So for example if you have long running async code with strong reference to the Fragment, GC won’t collect bonded view objects until there won’t be strong references to the Fragment and you’ll have memory leak which would keep not only Fragment itself but also bounded Views.
Calling `unbind()` in `onDestroyView()` is not required, but recommended.
But you should also keep in mind that if you don’t prevent async callbacks that work with bounded Views after onDestroyView() app could be crashed by NullPointerException because of nulled View references. [artem-zinnatullin]
Dodatek Android ButterKnife Zelezny¶
Twórcy Avasta stworzyli dodatek do IntelliJ IDEA, który automatycznie generuje pola wraz z odpowiednimi adnotacjami dla Butter Knife. Działa to bardzo podobnie, jak generowanie metod (np. setterów i getterów dla pól klasy).
W celu instalacji przechodzimy do File → Settings → Plugins → Browse Repositories, wyszukujemy Android ButterKnife Zelezny i instalujemy.
Poniżej animacja prezentująca działanie dodatku: