설정 앱 > 앱 정보에 나와있는 리스트와 같이
리스트뷰를 이용해 단말에 설치된 앱 리스트를 보여주도록 해보겠습니다.
소스 전 레이아웃 부터 살펴보겠습니다.
일단 필요한 항목이 무엇이 있을까요?
일단 전체 리스트 뷰가 있는 레이아웃 과 안에 들어가는 리스트 하나하나의 레이아웃이 필요 해보입니다.
저는 버튼을 누르면 설치된 앱 리스트를 보여주는 샘플 앱으로 만들어보도록 하겠습니다.!
select_app_dialog.xml
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
tools:context=".MainActivity" >
<TextView
android:id="@+id/textMessage"
android:layout_width="match_parent"
android:layout_height="30dp"
android:layout_marginStart="10dp"
android:layout_marginTop="10dp"
android:layout_marginEnd="10dp"
android:layout_marginBottom="10dp"
android:gravity="center"
android:text="Select"
android:textSize="20sp" />
<ListView
android:id="@+id/installedApplicationListView"
android:layout_width="wrap_content"
android:layout_height="500dp"
android:layout_below="@+id/textMessage"
android:gravity="center_vertical"/>
</RelativeLayout>
application_list_item.xml
앱 아이콘 | 패키지 이름 |
저는 앱 아이콘 + 패키지 이름 형식으로 레이아웃을 구성하였습니다.
<?xml version="1.0" encoding="utf-8"?>
<!-- Copyright(c) 2013 Bluebird Inc. All rights reserved. -->
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal">
<ImageView
android:id="@+id/imageView"
android:layout_width="40dp"
android:layout_height="60dp"
android:layout_gravity="center"
android:layout_marginLeft="10dp"
app:srcCompat="@android:mipmap/sym_def_app_icon" />
<TextView
android:id="@+id/applicationName"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginLeft="10dp"
android:textSize="18sp"
android:gravity="center_vertical"/>
</LinearLayout>
이제 java 소스를 살펴봅시다
Java 소스에는 레이아웃과 같이 크게 두 가지가 필요합니다.
리스트뷰 레이아웃에 대한 관리를 해주는 리스트뷰 어댑터 와
리스트뷰 안에 들어가는 데이터(앱 아이콘 + 패키지 이름) 정의 가 필요합니다.
데이터에 대한 정의
class SelectListData {
private Drawable iconDrawable;
private String packageName;
public SelectListData(Drawable icon, String title) {
this.iconDrawable = icon;
this.packageName = title;
}
public Drawable getIcon() {
return this.iconDrawable;
}
public String getPackageName() {
return this.packageName;
}
}
데이터를 가지고 있는 Class 선언시 get 과 set은 꼭 만들어주셔야 하는거 다들 아시죠?
필요 없어도 만들어주는게 좋다고 합니다.
저는 데이터를 만들때 set을 해주기 때문에 따로 set하는 부분은 생성해주지 않았지만, 필요할 것 같아요.!
나중에 추가하도록 하겠습니다.
리스트뷰 레이아웃에 대한 관리를 해주는 리스트뷰 어댑터
class SelectApplicationViewAdapter extends BaseAdapter {
private ArrayList<SelectListData> SelectApplicationViewItemList = new ArrayList<SelectListData>();
public SelectApplicationViewAdapter() {
}
@Override
public int getCount() {
return SelectApplicationViewItemList.size();
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
final int pos = position;
final Context context = parent.getContext();
if (convertView == null) {
LayoutInflater inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
convertView = inflater.inflate(R.layout.application_list_item, parent, false);
}
ImageView iconImageView = (ImageView) convertView.findViewById(R.id.imageView);
TextView packageNameTextView = (TextView) convertView.findViewById(R.id.applicationName);
SelectListData listViewItem = SelectApplicationViewItemList.get(position);
iconImageView.setImageDrawable(listViewItem.getIcon());
packageNameTextView.setText(listViewItem.getPackageName());
return convertView;
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public Object getItem(int position) {
return SelectApplicationViewItemList.get(position);
}
public void addItem(SelectListData listViewItem) {
SelectApplicationViewItemList.add(listViewItem);
}
public void resetItems() {
SelectApplicationViewItemList.clear();
}
}
하나하나 자세히 살펴보면
private ArrayList<SelectListData> SelectApplicationViewItemList = new ArrayList<SelectListData>();
리스트뷰에 들어갈 전체 데이터를 가지고 있을 변수가 필요합니다.
HashMap 이나 Array, List 등 다양한 변수로 표현이 가능한데, 저는 사용하기 편한 ArrayList로 정의했습니다.
@Override
public int getCount() { // list view 에 들어가는 data 총 갯수
return 0;
}
@Override
public Object getItem(int i) { // list view 에 들어가는 data 중 i 번째 데이터
return null;
}
@Override
public long getItemId(int i) { // list view 에 들어가는 data 중 i 번째 데이터의 id
return 0;
}
@Override
public View getView(int i, View view, ViewGroup viewGroup) { // 보여지는 뷰 표현
return null;
}
BaseAdapter를 상속받으면 이렇게 4가지를 필수적으로 재정의 해줘야합니다.
각 함수를 제가 정의하면 주석에 있는 것과 같고, 실제 정의는 아래 개발자 사이트를 참고하시면 될 것 같습니다.!
https://developer.android.com/reference/android/widget/BaseAdapter
여기서 가장 어려웠던 것은 개인적으로 아래 부분이였는데.
실제 view를 그려주는 부분입니다.
@Override
public View getView(int position, View convertView, ViewGroup parent) {
final int pos = position;
final Context context = parent.getContext();
if (convertView == null) { // view에 대한 정의가 없으면, 새로 정의 해줘야함 -> layout 에서 가지고 오기
LayoutInflater inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
convertView = inflater.inflate(R.layout.application_list_item, parent, false);
}
// 가지고 온 layout에 데이터에 해당하는 layout 그려주기
ImageView iconImageView = (ImageView) convertView.findViewById(R.id.imageView);
TextView packageNameTextView = (TextView) convertView.findViewById(R.id.applicationName);
SelectListData listViewItem = SelectApplicationViewItemList.get(position);
iconImageView.setImageDrawable(listViewItem.getIcon());
packageNameTextView.setText(listViewItem.getPackageName());
return convertView;
}
설치되어 있는 앱 리스트 가져오기
packageManager의 getInstalledApplications() 함수를 사용하면 설치 되어 있는 앱 리스트를 쉽게 가져올 수 있다.
PackageManager packageManager = this.getPackageManager();
List<ApplicationInfo> appList = packageManager.getInstalledApplications(0);
받아온 ApplicationInfo List를 내가 보여주고 싶은 데이터의 형식으로 저장을 해주면 된다.
for (ApplicationInfo applicationInfo : appList) {
String packageName = applicationInfo.packageName;
Drawable icon = applicationInfo.loadIcon(packageManager);
if (!packageName.isEmpty() && packageName != null && icon != null) {
appNameList.add(new SelectListData(icon, packageName));
}
}
해당 부분의 전체 소스
private ArrayList<SelectListData> getInstalledApplicationsList() {
ArrayList<SelectListData> appNameList = new ArrayList<SelectListData>();
PackageManager packageManager = this.getPackageManager();
List<ApplicationInfo> appList = packageManager.getInstalledApplications(0);
for (ApplicationInfo applicationInfo : appList) {
String packageName = applicationInfo.packageName;
Drawable icon = applicationInfo.loadIcon(packageManager);
if (!packageName.isEmpty() && packageName != null && icon != null) {
appNameList.add(new SelectListData(icon, packageName));
}
}
return appNameList;
}
리스트뷰 어댑터 및 데이터 추가
리스트 뷰에 어댑터 추가
ListView installedApplicationListView = dialog.findViewById(R.id.installedApplicationListView);
SelectApplicationViewAdapter activityAdapter = new SelectApplicationViewAdapter();
installedApplicationListView.setAdapter(activityAdapter);
리스트뷰에 보여줄 아이템 추가
ArrayList<SelectListData> appNameList = getInstalledApplicationsList();
for (SelectListData listViewItem : appNameList) {
activityAdapter.addItem(listViewItem);
}
리스트 뷰 클릭 이벤트
리스트 뷰에 보여진 데이터를 클릭했을 때의 이벤트를 정의.
log에서 package 이름을 보여주는 로그를 추가해보았습니다.
installedApplicationListView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
final SelectListData selectedItem = (SelectListData) parent.getItemAtPosition(position);
Log.d(TAG, "selectedItem = " + selectedItem.getPackageName());
dialog.dismiss();
}
});
전체소스
전체 소스는 아래와 같습니다.
class 마다 따로 파일을 만들어 줘야하는데,..
그냥 한 곳에 만들어주었습니다..
package com.young.youngtest;
import androidx.appcompat.app.AppCompatActivity;
import android.app.Dialog;
import android.content.Context;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageManager;
import android.graphics.drawable.Drawable;
import android.os.Bundle;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.AdapterView;
import android.widget.BaseAdapter;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.TextView;
import java.util.ArrayList;
import java.util.List;
public class MainActivity extends AppCompatActivity {
private static final String TAG = "MainActivity";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Button btn = findViewById(R.id.button);
btn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
showNewApplicationSelectDialog();
}
});
}
private ArrayList<SelectListData> getInstalledApplicationsList() {
ArrayList<SelectListData> appNameList = new ArrayList<SelectListData>();
PackageManager packageManager = this.getPackageManager();
List<ApplicationInfo> appList = packageManager.getInstalledApplications(0);
for (ApplicationInfo applicationInfo : appList) {
String packageName = applicationInfo.packageName;
Drawable icon = applicationInfo.loadIcon(packageManager);
if (!packageName.isEmpty() && packageName != null && icon != null) {
appNameList.add(new SelectListData(icon, packageName));
}
}
return appNameList;
}
private void showNewApplicationSelectDialog() {
Log.i(TAG, "========== showNewApplicationSelectDialog ======== ");
Dialog dialog = new Dialog(this);
dialog.setContentView(R.layout.select_app_dialog);
ListView installedApplicationListView = dialog.findViewById(R.id.installedApplicationListView);
SelectApplicationViewAdapter activityAdapter = new SelectApplicationViewAdapter();
installedApplicationListView.setAdapter(activityAdapter);
ArrayList<SelectListData> appNameList = getInstalledApplicationsList();
for (SelectListData listViewItem : appNameList) {
activityAdapter.addItem(listViewItem);
}
installedApplicationListView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
final SelectListData selectedItem = (SelectListData) parent.getItemAtPosition(position);
Log.d(TAG, "selectedItem = " + selectedItem.getPackageName());
dialog.dismiss();
}
});
dialog.show();
}
}
class SelectListData {
private Drawable iconDrawable;
private String packageName;
public SelectListData(Drawable icon, String title) {
this.iconDrawable = icon;
this.packageName = title;
}
public Drawable getIcon() {
return this.iconDrawable;
}
public String getPackageName() {
return this.packageName;
}
}
class SelectApplicationViewAdapter extends BaseAdapter {
private ArrayList<SelectListData> SelectApplicationViewItemList = new ArrayList<SelectListData>();
public SelectApplicationViewAdapter() {
}
@Override
public int getCount() {
return SelectApplicationViewItemList.size();
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
final int pos = position;
final Context context = parent.getContext();
if (convertView == null) {
LayoutInflater inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
convertView = inflater.inflate(R.layout.application_list_item, parent, false);
}
ImageView iconImageView = (ImageView) convertView.findViewById(R.id.imageView);
TextView packageNameTextView = (TextView) convertView.findViewById(R.id.applicationName);
SelectListData listViewItem = SelectApplicationViewItemList.get(position);
iconImageView.setImageDrawable(listViewItem.getIcon());
packageNameTextView.setText(listViewItem.getPackageName());
return convertView;
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public Object getItem(int position) {
return SelectApplicationViewItemList.get(position);
}
public void addItem(SelectListData listViewItem) {
SelectApplicationViewItemList.add(listViewItem);
}
public void resetItems() {
SelectApplicationViewItemList.clear();
}
}
실제 동작 영상