写一个 android launcher界面

本文介绍如何创建一个简单的Launcher应用,包括设置主页类别、定义界面布局、使用GridView展示应用图标及名称,并实现点击启动应用的功能。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

看了网上 PackageManager 和 ActivityManager ,看到有一个列出所有App的例子,就做了下面的launcher


第一步骤:

首先新建一个工程,默认就一个Activity 就是MainActivity

在AndroidManifest.xml的MainActivity节点下新添加

<category android:name="android.intent.category.HOME" />


第二步骤:

主界面只有一个 GridView

<pre name="code" class="html"><RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="${relativePackage}.${activityClass}" >

    <GridView
        android:id="@+id/apps_list"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:numColumns="4" >
    </GridView>

</RelativeLayout>


第三步骤:

定义每一个app的界面,app只显示icon和name

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:padding="30dp"
    android:orientation="vertical" >

    <ImageView
        android:id="@+id/app_icon"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center" >
    </ImageView>

    <TextView
        android:id="@+id/app_name"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center" >
    </TextView>

</LinearLayout>

第四步骤:

定义GridView的适配器


public class AppAdapter extends BaseAdapter {
	private Context mContext;
	private LayoutInflater inflater;
	ArrayList<ResolveInfo> appList;

	public final class AppItem {	
		public ImageView appIcon;
		public TextView appName;		
	}

	public AppAdapter(Context context, List<ResolveInfo> apps) {
		mContext = context;
		inflater = LayoutInflater.from(mContext);		
		appList = (ArrayList<ResolveInfo>)apps;
	}

	@Override
	public int getCount() {
		return appList.size();
	}

	@Override
	public Object getItem(int position) {
		return appList.get(position); 
	}

	@Override
	public long getItemId(int position) {
		return 0;
	}

	@Override
	public View getView(int position, View convertView, ViewGroup parent) {
		AppItem appItem = null;

		if(convertView == null)
		{  
			convertView = inflater.inflate(R.layout.app_item, null);
			appItem = new AppItem();
			appItem.appIcon = (ImageView) convertView.findViewById(R.id.app_icon);		
			appItem.appName = (TextView) convertView.findViewById(R.id.app_name);	
			convertView.setTag(appItem);
		} else {  		
			appItem = (AppItem) convertView.getTag();
		}  

		<span style="color:#990000;">ResolveInfo info = appList.get(position); </span>		
		appItem.appIcon.setImageDrawable<span style="color:#cc0000;">(info.loadIcon(mContext.getPackageManager())</span>);
		appItem.appName.setText(<span style="color:#cc0000;">info.loadLabel(mContext.getPackageManager())</span>);

		return convertView;
	}
}

通过 ResolveInfo 获取具体信息方法:

包名获取方法:resolve.activityInfo.packageName

icon获取获取方法:resolve.loadIcon(packageManager)

应用名称获取方法:resolve.loadLabel(packageManager).toString()


第五步骤:

主界面代码:

public class MainActivity extends Activity {

	private GridView appsGridView;  
	private List<ResolveInfo> appList;  

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);        
		appsGridView = (GridView) findViewById(R.id.apps_list);         

		//获取app列表
		appList = getAppList();
		appsGridView.setAdapter(new AppAdapter(MainActivity.this, appList));
		appsGridView.setOnItemClickListener(new OnItemClickListener(){
			@Override
			public void onItemClick(AdapterView<?> parent, View view,
					int position, long id) {			
				ResolveInfo info = appList.get(position); 
				openAppLication(info);
			}
		});
	}

	//获取APP列表
	private List<ResolveInfo> getAppList(){    
		Intent mainIntent = new Intent(Intent.ACTION_MAIN, null);  
		mainIntent.addCategory(Intent.CATEGORY_LAUNCHER);  
		return getPackageManager().queryIntentActivities(mainIntent, 0);	
	}

	//打开App
	private void openAppLication(ResolveInfo info){
		
		//该应用的包名  
		String pkg = info.activityInfo.packageName; 

		//应用的主activity类(MAIN)  
		String cls = info.activityInfo.name;  

		//ComponentName(组件名称)是用来打开其他应用程序中的Activity或服务的。
		ComponentName componet = new ComponentName(pkg, cls);  
		Intent intent = new Intent();  
		intent.setComponent(componet);  

		startActivity(intent);
	}
}

获取app列表

Intent mainIntent = new Intent(Intent.ACTION_MAIN, null);  
mainIntent.addCategory(Intent.CATEGORY_LAUNCHER);  
getPackageManager().queryIntentActivities(mainIntent, 0);


打开对应的应用

//该应用的包名  
String pkg = info.activityInfo.packageName; 

//应用的主activity类(MAIN)  
String cls = info.activityInfo.name;  

//ComponentName(组件名称)是用来打开其他应用程序中的Activity或服务的。
ComponentName componet = new ComponentName(pkg, cls);  
Intent intent = new Intent();  
intent.setComponent(componet);  

startActivity(intent);









### Android Launcher 滑动响应机制及实现流程 #### 1. 滑动事件的捕获与分发 Launcher 的滑动响应机制依赖于 `DragLayer` 和 `Workspace` 类。当用户在屏幕上执行滑动手势时,触摸事件会传递到 `DragLayer` 中。`DragLayer` 是一个继承自 `FrameLayout` 的容器视图,用于承载所有的子组件(如工作区、快捷栏等),并负责处理触控事件。 具体来说,`DragLayer` 使用 `onInterceptTouchEvent()` 方法拦截触摸事件,并将其交给内部的 `GestureDetector` 或者其他手势处理器进行进一步解析[^2]。如果检测到水平方向上的滑动,则触发相应的滚动逻辑。 #### 2. 工作区 (`Workspace`) 的滚动行为 `Workspace` 是 Launcher 中的主要内容区域,包含了多个页面(Page)。每个页面是一个独立的布局单元,通常由 `CellLayout` 实现。为了支持平滑的页面切换效果,`Workspace` 基于 `HorizontalScrollView` 构建,并重了部分方法以优化性能和用户体验。 以下是关键步骤: - **计算目标屏幕索引** 当接收到滑动事件后,`Workspace` 需要判断当前手指移动的距离以及速度,从而决定应该滚动到哪个页面。这一步骤通过调用 `snapToScreen(int whichScreen)` 完成[^4]。该函数接收一个整数参数表示目标屏幕编号,并确保最终停靠位置位于指定页签处。 - **动画过渡** 如果当前位置偏离了理想值(即未完全对齐至某一页中心点),则需借助 Scroller 对象创建惯性滑动动画。此过程涉及多次刷新 UI 来逐步调整偏移量直至达到终点坐标[(x,y)]=(target_x,0)[^3]。 ```java if (getScrollX() != (whichScreen * getWidth())) { final int delta = whichScreen * getWidth() - getScrollX(); mScroller.startScroll(getScrollX(), 0, delta, 0, Math.abs(delta) * 2); } invalidate(); // 引发重新绘制请求以便更新视觉表现形式 ``` #### 3. 回调通知监听器 除了基本的滚动控制外,还需要向外部模块报告状态变化情况。为此引入了一个接口——`OnViewChangeListener` ,允许开发者注册自己的观察者实例。每当发生显著改变(比如切换到了新页面)都会主动唤起对应的回调方法: ```java if (mOnViewChangeListener != null) { mOnViewChangeListener.OnViewChange(mCurScreen); } ``` 这种设计模式有助于增强系统的可扩展性和灵活性,使得不同业务场景下的需求能够得到满足而无需修改核心代码结构本身。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值