当前位置: 首页 > news >正文

UniApp自定义Android基座原理及流程

殴木釉乇? 本文将介绍基于 SurfaceControlViewHost 实现跨进程渲染普通 View 和 GlSurfaceView,力求用最简单的 Demo,介绍 SurfaceControlViewHost 的应用,方便读者轻松扣出核心代码应用到自己的业务中。

? 核心代码片段如下。

? 1)服务端

public SurfaceControlViewHost.SurfacePackage getSurfacePackage(int displayId, IBinder hostToken, int width, int height) {

// 创建SurfaceControlViewHost

Display display = mContext.getSystemService(DisplayManager.class).getDisplay(displayId);

mSurfaceControlViewHost = new SurfaceControlViewHost(mContext, display, hostToken);

// 创建要渲染的View

mView = new CustomView(mContext);

// 将View附加到SurfaceControlViewHost

mSurfaceControlViewHost.setView(mView, width, height);

SurfacePackage surfacePackage = mSurfaceControlViewHost.getSurfacePackage();

return surfacePackage;

}

? 2)客户端

IBinder hostToken = mSurfaceView.getHostToken();

SurfaceControlViewHost.SurfacePackage surfacePackage = mRemoteRender.getSurfacePackage(0, hostToken, 1000, 2000);

mSurfaceView.setChildSurfacePackage(surfacePackage);

? 本文案例项目结构如下,完整资源见 → 基于SurfaceControlViewHost实现跨进程渲染。

img

2 AIDL 配置

? Android 跨进程通信可以使用 AIDL 或 messenger,它们本质都是 Binder,本文使用 AIDL 实现跨进程通信。

? 1)aidl 文件

// IRemoteRender.aidl

package com.zhyan8.remoterender;

import android.view.SurfaceControlViewHost.SurfacePackage;

import android.os.IBinder;

interface IRemoteRender {

SurfacePackage getSurfacePackage(int displayId, IBinder hostToken, int width, int height);

}

? 2)gradle 配置

sourceSets {

main {

aidl.srcDirs = ['src/main/aidl']

}

}

buildFeatures.aidl true

? 3)manifest 配置

? 客户端配置如下。

? 服务端配置如下。

android:name=".RemoteRenderService"

android:exported="true">

android:name=".RemoteGLRenderService"

android:exported="true">

3 客户端

? MainActivity.java

package com.zhyan8.client;

import android.content.ComponentName;

import android.content.Context;

import android.content.Intent;

import android.content.ServiceConnection;

import android.os.Bundle;

import android.os.IBinder;

import android.os.RemoteException;

import android.util.Log;

import android.view.SurfaceControlViewHost.SurfacePackage;

import android.view.SurfaceView;

import android.view.View;

import androidx.appcompat.app.AppCompatActivity;

import com.zhyan8.remoterender.IRemoteRender;

public class MainActivity extends AppCompatActivity {

private static final String TAG = "MainActivity";

private IRemoteRender mRemoteRender;

private IBinder mService;

private SurfaceView mSurfaceView;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

mSurfaceView = findViewById(R.id.surface_view);

startService();

}

public void onClickDraw(View view) {

try {

IBinder hostToken = mSurfaceView.getHostToken();

SurfacePackage surfacePackage = mRemoteRender.getSurfacePackage(0, hostToken, 1000, 2000);

mSurfaceView.setChildSurfacePackage(surfacePackage);

} catch (RemoteException e) {

e.printStackTrace();

}

}

@Override

protected void onDestroy() {

super.onDestroy();

unbindService(mConnection);

}

private void startService() {

Log.d(TAG, "startService");

Intent intent = new Intent("com.zhyan8.remoterender.IRemoteRender");

//intent.setPackage("com.zhyan8.service"); // 渲染普通View的服务

intent.setPackage("com.zhyan8.glservice"); // 基于OpenGL ES渲染的服务

bindService(intent, mConnection, Context.BIND_AUTO_CREATE);

}

private void clearBind() {

Log.d(TAG, "clearBind");

if (mService != null) {

mService.unlinkToDeath(mDeathRecipient, 0);

}

mRemoteRender = null;

mService = null;

}

private ServiceConnection mConnection = new ServiceConnection() {

@Override

public void onServiceConnected(ComponentName name, IBinder service) {

Log.d(TAG, "onServiceConnected");

mRemoteRender = IRemoteRender.Stub.asInterface(service);

mService = service;

try {

mService.linkToDeath(mDeathRecipient, 0);

} catch (RemoteException e) {

Log.e(TAG, "e=" + e.getMessage());

}

}

@Override

public void onServiceDisconnected(ComponentName name) {

Log.d(TAG, "onServiceDisconnected");

clearBind();

}

};

private IBinder.DeathRecipient mDeathRecipient = new IBinder.DeathRecipient() {

@Override

public void binderDied() {

Log.d(TAG, "binderDied");

clearBind();

}

};

}

? activity_main.xml

android:layout_width="match_parent"

android:layout_height="match_parent"

android:orientation="vertical"

android:padding="16dp">

http://www.dtcms.com/a/592557.html

相关文章:

  • Ganache-CLI以太坊私网JSON-RPC接口执行环境搭建
  • Android 系统超级实用的分析调试命令
  • 【ZeroRange WebRTC】WebRTC 加密安全总览:对称/非对称、数字签名、证书、SHA/HMAC、随机数
  • 【ZeroRange WebRTC】数字签名与 WebRTC 的应用(从原理到实践)
  • 承德网站制作公司做国外的网站有什么不用钱的
  • 破解遗留数据集成难题:基于AWS Glue的无服务器ETL实践
  • Rust 的所有权系统,是一场对“共享即混乱”的编程革命
  • 【Rust 探索之旅】Rust 库开发实战教程:从零构建高性能 HTTP 客户端库
  • API 设计哲学:构建健壮、易用且符合惯用语的 Rust 库
  • 横沥镇做网站wordpress中文说明书
  • 先做个在线电影网站该怎么做贵阳做网站软件
  • 【字符串String类大集合】构造创建_常量池情况_获取方法_截取方法_转换方法_String和基本数据类型互转方法
  • Http请求中Accept的类型详细解析以及应用场景
  • 升鲜宝 供应链SCM 一体化自动化部署体系说明
  • grafana配置redis数据源预警误报问题(database is locked)
  • 拒绝繁琐,介绍一款简洁易用的项目管理工具-Kanass
  • 测试自动化新突破:金仓KReplay助力金融核心系统迁移周期缩减三周
  • 大语言模型入门指南:从科普到实战的技术笔记(1)
  • 大模型原理之Transformer进化历程与变种
  • 2025-简单点-ultralytics之LetterBox
  • 网站开发经济可行性分析石龙做网站
  • wordpress中国优化网络优化的目的
  • 【Linux网络】Socket编程TCP-实现Echo Server(下)
  • 路由协议的基础
  • ios 26的tabbar 背景透明
  • Hadoop大数据平台在中国AI时代的后续发展趋势研究CMP(类Cloudera CDP 7.3 404版华为鲲鹏Kunpeng)
  • Apache Jena:利用 SPARQL 查询与推理机深度挖掘知识图谱
  • Regression vs. Classification|回归vs分类
  • Nine.fun × AIOT重磅联手,打造健康娱乐新经济
  • The Life of a Read/Write Query for Apache Iceberg Tables