裸眼 3D 效果 Compose 版本

简介: 裸眼 3D 效果 Compose 版本

前因后果

今天在博客中看见自如客APP研发团队发的一篇文章:自如客APP裸眼3D效果的实现,看了看效果感觉很炫,然后又看到已经有人使用 Flutter 实现了这个效果,Flutter 能实现,Compose 肯定也没问题,来吧,说干就干。

实现思路

先来看看实现的效果吧,没有人家的那么好看,大家将就着看吧😂。

b6ac22e946784c2a95c6fc2f823851ae.gif

实现思路其实很简单,就是放了两层布局(当然自如客APP中实现是三层布局,原理是一样的),姑且叫背景和前景吧,背景不动,前景跟随设备旋转的角度来进行位置偏移就可以了。

码代码

实现思路有了,下面就来码代码吧,先来注册下传感器吧:

mSensorManager = getSystemService(Context.SENSOR_SERVICE) as SensorManager
// 陀螺仪传感器
mMagneticSensor = mSensorManager.getDefaultSensor(Sensor.TYPE_MAGNETIC_FIELD)
mSensorManager.registerListener(this, mMagneticSensor, SensorManager.SENSOR_DELAY_GAME)

再来在回调中获取下需要偏移的坐标:

override fun onSensorChanged(event: SensorEvent?) {
    if (event == null) return
    when (event.sensor.type) {
        Sensor.TYPE_ACCELEROMETER -> {
            // x,y,z分别存储坐标轴x,y,z上的加速度
            val x = event.values[0]
            val y = event.values[1]
            val z = event.values[2]
            refreshState(x, y)
            Log.d(TAG, "TYPE_ACCELEROMETER x:$x y:$y z:$z")
        }
        Sensor.TYPE_MAGNETIC_FIELD -> {
            // 三个坐标轴方向上的电磁强度,单位是微特拉斯(micro-Tesla),用uT表示,也可以是高斯(Gauss),1Tesla=10000Gauss
            val x = event.values[0]
            val y = event.values[1]
            val z = event.values[2]
            refreshState(x, y)
            Log.d(TAG, "TYPE_MAGNETIC_FIELD x:$x y:$y z:$z")
        }
    }
}

获取完坐标后就需要思考一个问题,在 Compose 中我们一直说数据驱动 UI 改变,这里也同样是,先来创建一个 ViewModel:

class MainViewModel : ViewModel() {
    private val _xState = MutableLiveData(0f)
    val xState: LiveData<Float> = _xState
    fun onxStateChanged(position: Float) {
        _xState.value = position
    }
    private val _yState = MutableLiveData(0f)
    val yState: LiveData<Float> = _yState
    fun onyStateChanged(position: Float) {
        _yState.value = position
    }
}

ViewModel 中提供了两个方法,分别来修改 x 和 y 的坐标,上面的 refreshState 方法中就调用的是这两个方法:

private fun refreshState(x: Float, y: Float) {
    viewModel.onxStateChanged(x)
    viewModel.onyStateChanged(y)
}

好了,万事具备,只欠东风,下面就来画布局吧:

@Composable
fun ThreeDImage(x: Float, y: Float) {
    Box(modifier = Modifier.fillMaxSize()) {
        Image(
            modifier = Modifier
                .fillMaxSize(),
            contentScale = ContentScale.Crop,
            painter = painterResource(id = R.drawable.icon_three_bg),
            contentDescription = "背景",
        )
        Image(
            modifier = Modifier
                .fillMaxSize()
                .offset(x = x.dp, y = y.dp),
            painter = painterResource(id = R.drawable.icon_three_small),
            contentDescription = "前景",
        )
    }
}

ThreeDImage 可组合项中接受两个参数,分别是前景需要偏移的 x 和 y 值,整个布局使用 Box 进行包裹,里面依次放了背景和前景,然后根据传进来需要偏移的 x 和 y 值来对前景做偏移。

是不是非常简单,这还是多亏了 Compose 中万能的 Modifier ,调用 Modifier 的扩展方法 offset 即可调整布局的偏移量。

最后把 ViewModel 中的 LiveData 数据转为 Compose 中可观察的 State 类型即可:

setContent {
    BannerTheme {
        val xState by viewModel.xState.observeAsState(0f)
        val yState by viewModel.yState.observeAsState(0f)
        Surface(color = MaterialTheme.colors.background) {
            ThreeDImage(xState, yState)
        }
    }
}

OK,到这里代码就写完了,是不是很简单,核心代码只有一行: Modifier .offset(x = x.dp, y = y.dp)。

写在最后

本文只是给大家提供一个实现的思路,大家跟着文中的代码再优化下效果会更好的。

本文中所有代码都在 Github 中进行托管:https://github.com/zhujiang521/Banner ,有需要的可以进行查看。



目录
相关文章
|
容器
docker-compose构建项目
docker-compose构建项目
|
网络协议 Linux 网络安全
最新版 docker-compose安装和使用
docker-compose安装和使用
3373 0
最新版 docker-compose安装和使用
|
Ubuntu 安全 Linux
Docker的安装和版本详细介绍
Docker的安装和版本详细介绍
2275 0
Docker的安装和版本详细介绍
|
存储 Linux Docker
Docker 更新版本
Docker 更新版本原来版本 1.10 更新后的版本 19.03.1 更新 Docker 版本需要注意的问题: 注意系统是否支持新版本的储存驱动。 19.03.01 版本默认使用的储存驱动是 overlay2。
3005 0
|
3月前
|
Docker 容器
利用Docker Compose优化开发环境的配置
在现代软件开发中,环境一致性至关重要。开发人员常需在不同机器间复制环境配置,而Docker Compose提供了一种简便有效的方法来定义和运行多容器Docker应用程序,确保开发、测试和生产环境一致,简化团队协作,提高开发效率。通过YAML文件配置服务、网络和卷,使用简单命令即可启动和停止服务。本文将介绍Docker Compose的核心优势、基本使用方法及高级功能,帮助你更好地管理和优化开发环境。
|
Docker 容器
Excalidraw 简介及 Docker Compose 部署指南
家人们好,我们在工作生活中经常需要画些图,我们往期了已经出过draw-io私有化部署的文章了,今天我要向大家介绍一款名为 Excalidraw 的绘图工具,这款工具了我个人非常喜欢使用,是因为它可以修改成类似于手写体的字体,并且可以直接绘画,这篇文章我将分享如何使用 Docker Compose 轻松部署 Excalidraw。
739 0
Excalidraw 简介及 Docker Compose 部署指南
|
8月前
|
关系型数据库 MySQL 应用服务中间件
Docker Compose 使用方法
Docker Compose 是一个用于定义和运行多个 Docker 容器的工具。它允许您通过一个单独的配置文件来定义多个容器、网络设置、存储卷等,从而简化了多容器应用的部署和管理过程。使用 Docker Compose,您可以轻松地创建和管理复杂的容器化应用程序,而无需手动管理每个容器。 Docker Compose 使用 yml文件来描述应用程序的配置。在这个yml 文件中,您可以定义各个服务(容器)的镜像、端口映射、环境变量、依赖关系等信息。然后,您只需要运行一个命令,Docker Compose 就会根据配置文件自动创建、启动和连接这些容器。
163 1
|
8月前
|
IDE JavaScript 开发工具
Compose Multiplatform 1.0 发布
Compose Multiplatform 1.0 发布
125 0
|
Docker 容器
Docker compose v1 与 v2版本区别
Docker Compose是用于定义和管理多容器Docker应用程序的工具。下面是Docker Compose版本1和版本2之间的一些区别:
1158 0