이 포스팅은 Butter Knifeの紹介 을 기본으로 번역하여 작성했습니다
제 일본어 실력으로 인하여 오역이나 오타가 발생할수 있습니다.
Android용 View Injection 라이브러리인 Butter Knife에 대해서 설명합니다(라고하지만, 사이트에 적혀있는 것을 대부분 그대로 일본어로 번역한 것 뿐입니다.)
Butter Knife은 ActionBarSherlock등 square의 Jake Wharton씨에 의한 Android용 View Injection 라이브러리입니다.
이 라이브러리의 목적은
라는 심플한것이므로, 간단하게 이해되리라고 생각합니다. 아래는 익숙한 Activity 코드입니다.
public class MainActivity extends Activity {
TextView mTextView;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mTextView = (TextView) findViewById(R.id.activity_main_textview);
}
}
View가 한두개라면 좋지만, 늘어갈수록 setContentView 이후에 findViewById가 길게 나열되어 보기어려운 코드가 됩니다.
이것을 아래와 같이 작성하는 것이 됩니다.
public class MainActivity extends Activity {
@InjectView(R.id.activity_main_textview)
TextView mTextView;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ButterKnife.inject(this);
// 아래에 mTextView를 사용한 코드 (mTextView.setText() 등)
}
}
@InjectView(int id)
으로 View의 Resource Id를 지정하면 ButterKnife.inject(Activity)
에서 mTextView = (TextView) findViewById(R.id.activity_main_textview)
와 같은 처리가 이루어집니다.
또한,
public class MainActivity extends Activity {
Button mButton;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mButton = (Button) findViewById(R.id.activity_main_button);
mButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
onClickButton();
}
});
}
void onClickButton() {
// 버튼을 눌렀을때의 처리
}
}
findViewById하여, OnClickListener를 설정하고… 이런 ButtonButton처리도, Button의 수가 늘어갈수록 대단히 보기어려운 코드가 됩니다만, 이것을
public class MainActivity extends Activity {
@OnClick(R.id.activity_main_button)
void onClickButton() {
// 버튼을 눌렀을때의 처리
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ButterKnife.inject(this);
}
}
Button의 onClick시에 호출되는 메소드에 @OnClick(int id)
으로 View의 Resource Id를 지정하는 것만으로도 좋게됩니다.
복수의 Button의 onClick 이벤트 공통처리를 작성하고 싶을때에도,
@OnClick({ R.id.activity_main_button1,
R.id.activity_main_button2,
R.id.activity_main_button3 })
void onClickButton(CustomButton button) {
if (button.isHoge()) {
// ...
}
}
이렇게 작성할수 있으므로 편리합니다. @InjectView
도 @OnClick
도 View를 상속받은 클래스에서도 사용가능하므로 커스텀 View도 가능합니다.
또한, Injection의 대상인 View를 지정할수 있으므로, Fragment에서도,
public class MainFragment extends Fragment {
@InjectView(R.id.fragment_main_textview)
TextView mTextView;
@Override
View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.fragment_main, container, false);
ButterKnife.inject(this, view);
// 아래는 mTextView를 사용한 처리
return view;
}
@Override
void onDestroyView() {
super.onDestroyView();
ButterKnife.reset(this);
}
}
와 같이 사용할수 있습니다. (Fragment의 경우 onDestroyView에서 ButterKnife.reset를 호출해주세요).
이것과 같이, ListView등에서 사용하는 ViewHolder 패턴도 아래와 같이 간결하게 작성할수 있습니다.
public class MyListAdapter extends BaseAdapter {
@Override
public View getView(int position, View view, ViewGroup parent) {
ViewHolder holder;
if (view != null) {
holder = (ViewHolder) view.getTag();
} else {
view = inflater.inflate(R.layout.list_item, parent, false);
holder = new ViewHolder(view);
view.setTag(holder);
}
// 아래에 ViewHolder의 변수를 사용한 처리
holder.mTitleView.setText("title");
holder.mNameView.setText("my name");
return convertView;
}
static class ViewHolder {
@InjectView(R.id.title)
TextView mTitleView;
@InjectView(R.id.name)
TextView mNameView;
public ViewHolder(View view) {
ButterKnife.inject(this, view);
}
}
}
이밖에도 아래 내용을 사용할수 있습니다.
같은 코드를 BaseActvity나 HogeUtil 안에 적는 사람들도 많을것입니다. View#findViewById(int id)
나 Activity#findViewById(int id)
의 Helper도 있습니다.
View childView = getLayoutInflater().inflate(R.layout.child, null);
TextView childText = ButterKnife.findById(childView, R.id.child_text);
ImageView childImage = ButterKnife.findById(childeView, R.id.child_image);
ProGuard를 사용하는 경우에는 proguard-project.txt 등에 아래와 같이 설정해주세요.
-dontwarn butterknife.internal.**
-keep class **$$ViewInjector { *; }
-keepnames class * { @butterknife.InjectView *;}
Jake Wharton씨가 Google+에 투고. Mar 6, 2013
People really like the concept of view "injection" for Android
and it often comes up when talking about Dagger.
Last night, I wrote a view injection library which uses
annotation processing as a little experiment and half-joke.
Dagger에서 다른 개발자와 대화할때에 View의 Injection 문제가 자주 나타나므로, Annotation Processing로 시험삼아 만들어봤습니다, 라는 정도로 시작한듯 합니다.
RoboGuice uses an annotation in a different package and reflection
at runtime whereas this uses code generation along with a static method call.
It uses the same annotation name so people feel at home (similar to how @Inject is a standard).
This is for people who use Dagger or Guice (or no dependency injection)
but can't bring themselves to write findViewById calls.
Guice (RoboGuice)와의 차이는 Guice는 Annotation의 LookUp이나 필드의 값 대입에 Reflection을 사용하는데(그런데, 느림), Butter Knife는 그렇지않고 빌드시에 코드를 생성합니다. 라고 (Reflection 하는건 ButterKnife.inject
정도).
Jake Wharton씨가 Google+에 투고. Mar 26, 2013
Ondřej Kroupa:
Seems nice, but what is main advantage to android-annotations library?
Jake Wharton:
This is a tiny, purpose-built library whereas AA is an application framework.
AndroidAnnotations에서도 @ViewById
나 @Click
을 사용해 동일하게 쓸수있지만 느리네? 라는 질문에 대해서는, AndroidAnnotations는 여러가지 가능한 어플리케이션 프레임워크이지만, Butter Knife는 View의 Injection만을 특화시킨 심플라이브러리이다. 라고
다른 사람도 Butter Knife의 소개 기사를 작성하고 있습니다만,
5 Reasons You Should Use Butterknife For Android
3. You find AndroidAnnotations to be overkill for your project.
AndroidAnnotations is a cool idea. It has tons of features
and more are being added all the time.
Depending on which camp you are in, adding features can be both a blessing and curse.
If you want to use annotations to inject everything imaginable
then definitely give AndroidAnnotations a look.
I am not 100% sold on the annotating and injecting of everything just yet.
For now views, viewholders, and OnClickListeners is enough for me.
AndroidAnnotations을 사용하고 있다면 그걸로도 충분하고, 혹시, 이 프로젝트는 거기까지는 필요 없다라고 생각한다면 Butter Knife을 써보자라고, 잘 분간해서 사용하면 되잖아, 라고.
AndroidAnnotations을 사용하면, HogeActivity에 대해서 Injection된 HogeActivity_ 클래스가 생성되니깐, startActivity(new Intent(this, HogeActivity_.class))
라고 작성하거나 AndroidManifest.xml에 <activity android:name=".HogeActivity_" />
라고 작성하지 않으면 안되는것이 개인적으로는 조금 불편합니다.(그러고 보니 AndroidAnnotations을 만든 Pierre-Yves Ricau씨도 Square 소속입니다)
나도 findViewById를 무시할정도라면 굳이 사용할 필요는 없다고 생각하고 있었기때문에, 사용하기 시작해보면 정말로 편리하므로 추천합니다.
최후로 여담입니다만, 이 “Butter Knife”라는 이름은 재치가 있다고 생각합니다.
그러므로, 뭐든지 되는(Dependency Injector) Dagger 에 비해, 빵에 버터를 바르는 (View를 Inject한다) 것밖에 안되지만, 그 용도만으로도 편리하다라는 의미로 Butter Knife라는 이름으로 했지않을까라고 생각한다
Butter Knife - View “injection” library for Android
comments powered by Disqus
Android Android-Library Dependency Injection Injection ButterKnife
Subscribe to this blog via RSS.
LazyColumn/Row에서 동일한 Key를 사용하면 크래시가 발생하는 이유
Posted on 30 Nov 2024