WORK/Android

[Android] ListView를 이용하여 설치된 app list 보여주기 | BaseAdapter | ListView onClickListener | PackageManager | getInstalledApplications

호이호이호잇 2022. 8. 30. 20:30
728x90
반응형

설정 앱 > 앱 정보에 나와있는 리스트와 같이
리스트뷰를 이용해 단말에 설치된 앱 리스트를 보여주도록 해보겠습니다.

소스 전 레이아웃 부터 살펴보겠습니다.
일단 필요한 항목이 무엇이 있을까요?

일단 전체 리스트 뷰가 있는 레이아웃안에 들어가는 리스트 하나하나의 레이아웃이 필요 해보입니다.

저는 버튼을 누르면 설치된 앱 리스트를 보여주는 샘플 앱으로 만들어보도록 하겠습니다.!

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

 

BaseAdapter  |  Android Developers

android.net.wifi.hotspot2.omadm

developer.android.com


여기서 가장 어려웠던 것은 개인적으로 아래 부분이였는데.
실제 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() 함수를 사용하면 설치 되어 있는 앱 리스트를 쉽게 가져올 수 있다.

https://developer.android.com/reference/android/content/pm/PackageManager#getInstalledApplications(int) 

 

PackageManager  |  Android Developers

android.net.wifi.hotspot2.omadm

developer.android.com

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();
    }
}

 

실제 동작 영상

 

728x90
반응형