众所周知,安卓应用开发经过这么多年的发展相对成熟和稳定,鸿蒙OS作为后来者兼容一个成熟的开发体系会节省很多推广和开发成本。但在实际开发中,代码层面仍然有很多细节上的差异,会给初次开发人员造成困扰。
本宝典旨在汇总实际开发中第三方件接入时的代码差异,以期帮助开发人员更好的进行开发作业,由于目前接触的开发类型有限,所汇总的内容多少会有疏漏,后期我们会进一步完善和补全。
欢迎关注我们以及我们的专栏,方便您及时获得相关内容的更新。
※基础功能
1.获取屏幕分辨率
安卓:
- getWindowManager().getDefaultDisplay();
鸿蒙:
- Optional<Display>
- display = DisplayManager.getInstance().getDefaultDisplay(this.getContext());
- Point pt = new Point();
- display.get().getSize(pt);
2.隐藏标题栏TitleBar
安卓:
略
鸿蒙:
confi.json中添加如下描述:
- ""metaData"":{
- ""customizeData"":[
- {
- ""name"": ""hwc-theme"",
- ""value"": ""androidhwext:style/Theme.Emui.NoTitleBar"",
- ""extra"":""""
- }
- ]
- }
3.获取屏幕密度
安卓:
- Resources.getSystem().getDisplayMetrics().density
鸿蒙:
- // 获取屏幕密度
- Optional<Display>
- display = DisplayManager.getInstance().getDefaultDisplay(this.getContext());
- DisplayAttributes displayAttributes = display.get().getAttributes();
- //displayAttributes.xDpi;
- //displayAttributes.yDpi;
4.获取上下文
安卓:
- context
鸿蒙:
- getContext()
5.组件的父类
安卓:
- android.view.View; class ProgressBar extends View
鸿蒙:
- class ProgressBar extends Component
6.沉浸式显示
安卓:
略
鸿蒙:
A:在config.json ability 中添加
- "metaData"": {
- ""customizeData"": [
- {
- ""extra"": """",
- ""name"": ""hwc-theme"",
- ""value"": ""androidhwext:style/Theme.Emui.Light.NoTitleBar""
- }
- ]
- }
B:在AbilitySlice的onStart函数内增加如下代码,注意要在setUIContent之前。
- getWindow().addFlags(WindowManager.LayoutConfig.MARK_TRANSLUCENT_STATUS);
7.获取运行时权限
安卓:
- ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.CALL_PHONE}, 1)
鸿蒙:
- requestPermissionsFromUser(
- new String[]{""ohos.permission.READ_MEDIA"", ""ohos.permission.WRITE_MEDIA"", ""ohos.permission.READ_USER_STORAGE"", ""ohos.permission.WRITE_USER_STORAGE"",}, 1);
※布局&组件
1.页面跳转(显示跳转)
安卓:
A.从A跳转至B,没有参数,并且不接收返回值
- Intent intent = new Intent();
- intent.setClass(A.this, B.class);
- startActivity(intent);
B.从A跳转至B,有参数,不接收返回值
- Intent intent = new Intent(this, B.class);
- intent.putExtra(""name"", ""lily"");
- startActivity(intent);
C.从A跳转至B,有参数,接收返回值
- Intent intent = new Intent(this, B.class);
- intent.putExtra(""name"", ""lily"");
- startActivityForResult(intent, 2);
鸿蒙:
A.从A跳转至B,没有参数,并且不接收返回值
- present(new BSlice(), new Intent());
B.从A跳转至B,有参数,不接收返回值
- Intent intent = new Intent();
- Operation operation = new Intent.OperationBuilder()
- .withDeviceId("""") .withBundleName(""com.test"") .withAbilityName(""com.test.BAbility"")
- .build();
- intent.setParam(""name"",""lily"");
- intent.setOperation(operation);
- startAbility(intent);
C.从A跳转至B,有参数,接收返回值
- Intent intent = new Intent();
- Operation operation = new Intent.OperationBuilder()
- .withDeviceId("""") .withBundleName(""com.test"") .withAbilityName(""com.test.BAbility"")
- .build();
- intent.setParam(""name"",""lily"");
- intent.setOperation(operation);
- startAbilityForResult(intent,100);
2.页面跳转(隐式跳转)
安卓:
A.配置
- <activity android:name="".B"">
- <intent-filter>
- <action android:name=""com.hly.view.fling""/>
- </intent-filter>
- </activity>
B.启动
- Intent intent = new Intent(); intent.setAction(""com.hly.view.fling""); intent.putExtra(""key"", ""name""); startActivity(intent);
鸿蒙:
A.在config.json文件ability 中添加以下信息
- "skills"":[
- {
- ""actions"":[
- ""ability.intent.gotopage""
- ]
- }
- ]
B.在MainAbility的onStart函数中,增加页面路由
- addActionRoute( ""ability.intent.gotopage"", BSlice.class.getName());
C.跳转
- Intent intent = new Intent();
- intent.setAction(""ability.intent.gotopage"");
- startAbility(intent);
3.页面碎片
安卓:
- Fragment
鸿蒙:
- Fraction
A:Ability继承FractionAbility
B:获取Fraction调度器
- getFractionManager().startFractionScheduler()
C:构造Fraction
D:调用调度器管理Fraction
- FractionScheduler.add()
- FractionScheduler.remove()
- FractionScheduler.replace()
备注:
参考demo
https://www.jianshu.com/p/58558dc6673a"
4.从xml文件创建一个组件实例
安卓:
- LayoutInflater.from(mContext).inflate(R.layout.banner_viewpager_layout, null);
鸿蒙:
- LayoutScatter.getInstance(getContext()).parse(ResourceTable.Layout_ability_main, null, false);
5.组件自定义绘制
安卓:
- ImageView.setImageDrawable(Drawable drawable);
并重写Drawable 的draw函数
鸿蒙:
- Component.addDrawTask(Component.DrawTask task);
并实现Component.DrawTask接口的onDraw函数
6.自定义组件的自定义属性(在xml中使用)
安卓:
需要3步
A.在 values/attrs.xml,在其中编写 styleable 和 item 等标签元素。
B.在layout.xml中,增加
- xmln:app= ""http://schemas.android.com/apk/res/-auto""
C.在自定义组件的构造函数中,调用array.getInteger(R.styleable.***, 100);获取属性
鸿蒙:
只需2步
A. 在组件定义的layout.xml中增加 xmlns:app=""http://schemas.huawei.com/apk/res/ohos""
然后就可以使用app:***(***为任意字符串)来增加自定义属性了,为了区分建议加上组件名前缀。
B. 在自定义组件的带AttrSet参数的构造函数中,使用下面代码获取属性。attrSet.getAttr(""***"").get().getStringValue();
7.触摸事件
安卓:
- android.view.MotionEvent
鸿蒙:
- ohos.multimodalinput.event.TouchEvent
8.事件处理
安卓:
- android.os.Handler
鸿蒙:
- ohos.eventhandler.EventHandler
9.控件触摸事件回调
安卓:
- android.view.View.OnTouchListener
鸿蒙:
- ohos.agp.components.Component.
- TouchEventListener
10.轮播图继承的父类
安卓:
- extends ViewPager
鸿蒙:
- extends PageSlider
11.实现监听轮播图组件事件
安卓:
- implements PageSlider.PageChangedListener
鸿蒙:
- Implements OnPageChangedListener
12.touch事件监听
安卓:
- 直接重写onTouchEvent
鸿蒙:
- 继承 Component.TouchEventListener然后在构造方法中设置监听 setTouchEventListener(this::onTouchEvent);实现onTouchEvent
13.获取点击事件的坐标点
安卓:
- event.getX(), event.getY()
鸿蒙:
- MmiPoint point = touchEvent.getPointerPosition(touchEvent.getIndex());
14.调节滚轮中内容间距
安卓:
- setLineSpacingMultiplier(float f)
鸿蒙:
- setSelectedNormalTextMarginRatio(float f)
15.滚轮定位
安卓:
- setPosition
鸿蒙:
- setValue
16.Layout布局改变监听
安卓:
- View.OnLayoutChangeListener
鸿蒙:
- Component.LayoutRefreshedListener
17.组件容器
安卓:
- ViewGroup
鸿蒙:
- ComponentContainer
18.添加组件
安卓:
- addView()
鸿蒙:
- addComponent()
19.动态列表的适配器
安卓:
- extends RecyclerView.Adapter<>
鸿蒙:
- extends RecycleItemProvider
20.动态列表
安卓:
- RecyclerView
鸿蒙:
- ListContainer
21.文本域动态监听
安卓:
- TextWatcher
鸿蒙:
- Component.ComponentStateChangedListener
22.组件绘制自定义布局
安卓:
- 重写onLayout(boolean changed, int left, int top, int right, int bottom)
鸿蒙:
- 重写Component.LayoutRefreshedListener的onRefreshed方法
23.List组件
安卓:
- ListView
鸿蒙:
- ListContainer
24.设置背景颜色
安卓:
- setBackgroundColor(maskColor);
鸿蒙:
- // 创建背景元素
- ShapeElement shapeElement = new ShapeElement();
- // 设置颜色
- shapeElement.setRgbColor(new RgbColor(255, 0, 0));
- view.setBackground(shapeElement);
25.可以在控件上、下、左、右设置图标,大小按比例自适应
安卓:
- setCompoundDrawablesWithIntrinsicBounds
鸿蒙:
- setAroundElements
26.RadioButton组件在xml中如何设置checked属性
安卓:
在xml中可以设置
鸿蒙:
- radioButton = findComponentById();
- radioButton.setChecked(true);
备注:
sdk2.0后 xml中没有了checked属性,如果使用,可以在java代码中实现"
27.文本域动态监听
安卓:
- TextWatcher
鸿蒙:
- Component.ComponentStateChangedListener
28.颜色类
安卓:
- java.awt.Color
鸿蒙:
- ohos.agb.colors.rgbcolor
29.为ckeckbox或者Switch按钮设置资源图片
安卓:
略
鸿蒙:
- VectorElement vectorElement = new VectorElement(this, ResourceTable.Graphic_candy);
- setBackground(vectorElement)
30.子组件将拖拽事件传递给父组件
安卓:
略
鸿蒙:
注册setDraggedListener侦听,实现onDragPreAccept方法,再方法内根据拖拽方向判断是否需要父组件处理,如果需要则返回false,否则返回true
※资源管理
1.管理资源
安卓:
- AssertManager
鸿蒙:
- ResourceManager
2.获取应用的资源文件rawFile,并返回InputStream
安卓:
- getResources()
- AssetManager类
鸿蒙:
- ResourceManager resourceManager = getContext().getResourceManager();
- RawFileEntry rawFileEntry = resourceManager.getRawFileEntry(jsonFile);
- Resource resource = null;
- try {
- resource = rawFileEntry.openRawFile();
- } catch (IOException e) {
- e.printStackTrace();
- }
备注:
- Resource是InputStream的子类,可以直接作为InputStream使用。"
3.获取文件路径
安卓:
- Environment.getExternalStorageDirectory().getAbsolutePath()
鸿蒙:
- 获取文档(DIRECTORY_DOCUMENTS)、下载(DIRECTORY_DOWNLOADS)、视频(DIRECTORY_MOVIES)、音乐(DIRECTORY_MUSIC)、图片(DIRECTORY_PICTURES)
- GetExternalFilesDir(Environment.DIRECTORY_PICTURES).getAbsolutePath()
※消息&多线程
1.事件循环器
安卓:
- android.os.Looper
鸿蒙:
- EventRunner.create(true)
备注:
有参数且为true,表示队列被托管,参数为false,或无参表示不被托管,需要eventRunner.run()调用
2.消息
安卓:
- android.os.Message
鸿蒙:
- InnerEvent
3.休眠
安卓:
- android.os.SystemClock.sleep()
鸿蒙:
- ohos.miscservices.timeutility.time;
- Time.sleep(int millesend)
4.事件通知延迟消息
安卓:
- Handler.postDelayed(MESSAGE_LOGIN, 5000);
鸿蒙:
- Handler.postTask(task, 5000);
5.Intent传递参数
安卓:
- Intent.putExtra or add Bundle
鸿蒙:
- Intent.setParam
6.消息发送
安卓:
- Handler handler = new Handler,通过handlerMsg发消息
鸿蒙:
- InnerEvent event1 = InnerEvent.get(eventId1, param, object);
- myHandler.sendEvent(event1, 0, Priority.IMMEDIATE);
7.更新UI
安卓:
- class MyHandle extends Handler{
- @Override
- public void handleMessage(Message msg) {
- super.handleMessage(msg);
- switch (msg.what) {
- case 1:
- //更新UI的操作
- break;
- default:
- Break;
- }
- }
- }
鸿蒙:
- abilitySlice.getUITaskDispatcher().asyncDispatch(() -> {
- //更新UI的操作
- });
※图片处理
1.位图资源
安卓:
- Bitmap
鸿蒙:
- PixelMap
2.图像缩放,拉伸到视图边界
安卓:
- ImageView.ScaleType
- image.setScaleType(ScaleType.FXY);
鸿蒙:
- Image.ScaleMode
- image.setScaleMode(Image.ScaleMode.STRETCH);
3.List组件&内容适配器
安卓:
- ListView
- extends BaeAdapter
- ViewPage.setAdapter(BaeAdapter);
鸿蒙:
- ListContainer
- extends PageSlider
- PageSlider.setProvider(PageSlider);
4.图片显示组件
安卓:
- androidx.appcompat.widget.AppCompatImageView
- Bitmap bitmap = BitmapFactory.decodeResource(context.getResources(),drawableId);
- Image.setImageBitmap(bitmap);
鸿蒙:
- ohos.agp.components.Image
根据实际情况可传递其他参数
- ImageSource imageSource = ImageSource.create(file, new ImageSource.SourceOptions());
- pixelMap = imageSource.createPixelmap(new ImageSource.DecodingOptions());
- image.setPixelMap(pixelMap);
5.图片填充整个控件
安卓:
- image.setScaleType(ScaleType.FXY);
鸿蒙:
- image.setScaleMode(Image.ScaleMode.STRETCH);
6.通过资源id获取位图
安卓:
- getBitmapFromDrawable
鸿蒙:
- /**
- * 通过资源ID获取位图对象
- **/
- private PixelMap getPixelMap(int resId) {
- InputStream drawableInputStream = null;
- try {
- drawableInputStream = getResourceManager().getResource(resId);
- ImageSource.SourceOptions sourceOptions = new ImageSource.SourceOptions();
- sourceOptions.formatHint = ""image/png"";
- ImageSource imageSource = ImageSource.create(drawableInputStream, null);
- ImageSource.DecodingOptions decodingOptions = new ImageSource.DecodingOptions();
- decodingOptions.desiredSize = new Size(0, 0);
- decodingOptions.desiredRegion = new Rect(0, 0, 0, 0);
- decodingOptions.desiredPixelFormat = PixelFormat.ARGB_8888;
- PixelMap pixelMap = imageSource.createPixelmap(decodingOptions);
- return pixelMap;
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- try {
- if (drawableInputStream != null) {
- drawableInputStream.close();
- }
- } catch (Exception e) {
- e.printStackTrace();
- }
- }
- return null;
- }
7.获取Gif图片帧
安卓:
需要自定frame类,通过decoder获取
鸿蒙:
- ImageSource.createPixelmap(int index, ImageSource.DecodingOptions opts)
8.BMP位图裁剪
安卓:
- Bitmap.createBitmap(Bitmap source, int x, int y, int width, int height)
鸿蒙:
- PixelMap.create(PixelMap source, Rect srcRegion, PixelMap.InitializationOptions opts)
※视频播放
在视频播放窗口上层增加控件
安卓:
略
鸿蒙:
A.pinToTop设置false,保证其他控件与surfaceProvider在同一layout下,并且不能设置背景
B.增加以下代码设置顶部窗口透明
- WindowManager.getInstance().getTopWindow().get().setTransparent(true);
※数据库
数据库获取索引
安卓:
- android.database.Cursor;
- cursor.getString()/cursor.getColumnIndex()
鸿蒙:
- ohos.data.resultset
※数据结构
1.应用程序数据共享
安卓:
- context.getContentResolver();
- resolver.getType(uri)
鸿蒙:
- ohos.aafwk.ability.DataAbilityHelper
2.JSON解析
安卓:
- import org.json.JSONArray;
- import org.json.JSONException;
- import org.json.JSONObject;
- import org.json.JSONTokener;
鸿蒙:
- Gson,fastJson
3.对象序列化
安卓:
- android.os.Parcel;
- parcel.readParcelable();parcel.writeParcelable()
鸿蒙:
- ohos.utils.parcel
4.浮点数矩形,获取中心点
安卓:
- RectF.centerX()
鸿蒙:
- RectFloat.getCenter().getPointX()
5.数据结构类
安卓:
- LongSparseArray
- SparseArrayCompat
鸿蒙:
使用HashMap
备注:
内存使用和查找性能会有影响。"
6.浮点数矩形
安卓:
- RectF
鸿蒙:
- RectFloat
7.浮点坐标
安卓:
- PointF
鸿蒙:
- 可使用Point
※对话框
1.对话框销毁
安卓:
- mDialog.dismiss()
鸿蒙:
- mDialog.destroy();
2.对话框中加载布局
安卓:
- mDialog.setContentView(ViewGroup dialogView)
鸿蒙:
- setContentCustomComponent(Componnet comp)
3.点击对话框外部关闭对话框
安卓:
- mDialog.setCancelable(mPickerOptions.cancelable)
鸿蒙:
- mDialog.setDialogListener(new BaseDialog.DialogListener() {
- @Override
- public boolean isTouchOutside() {
- mDialog.destroy(); dialogView.getComponentParent().removeComponent(dialogView);
- return true;
- }
- });
备注:
鸿蒙对话框销毁之后需移除对话框中加载的布局,否则再次加载会报错"
※动画
1.旋转动画
安卓:
- android.view.animation.RotateAnimation
鸿蒙:
- ohos.agp.animation.AnimatorProperty
2.值动画及相关回调
安卓:
- android.animation.ValueAnimator
- ValueAnimator.AnimatorUpdateListener
- Animator.AnimatorListener
- Animator.AnimatorPauseListener
鸿蒙:
- ohos.agp.animation.AnimatorValue
- AnimatorValue.ValueUpdateListener
- Animator.StateChangedListener
- Animator.LoopedListener
备注:
启动动画时,AnimatorValue必须作为类的成员变量,而不能时函数局部变量,否则动画不会启动"
3.线性插值器
安卓:
- LinearInterpolator
鸿蒙:
自己写一个
- public interface Interpolator {
- float getInterpolation(float input);
- }
- public class LinearInterpolator implements Interpolator {
- public LinearInterpolator() {
- }
- public LinearInterpolator(Context context, AttrSet attrs) {
- }
- public float getInterpolation(float input) {
- return input;
- }
- }
4.设置动画循环次数
安卓:
- animation.setRepeatCount(Animation.INFINITE)
鸿蒙:
- animator.setLoopedCount(Animator.INFINITE)
※存储
获取存储根路径
安卓:
- Environment.getExternalStorageDirectory().getAbsolutePath();
鸿蒙:
- System.getProperty("user.dir")
※Canvas绘图
1.绘制圆弧
安卓:
- Android canvas.drawArc()
鸿蒙:
- ohos.agp.render; class Arc
2.绘制圆形的两种方式
安卓:
- canvas.drawCircle(float x, float y, float radius, Paint paint)
鸿蒙:
- A.canvas.drawPixelMapHolderRoundRectShape(PixelMapHolder holder, RectFloat rectSrc, RectFloat rectDst, float radiusX, float radiusY)
- B.canvas.drawCircle(float x, float y, float radius, Paint paint)
3.绘制文本的方法
安卓:
- drawText (String text, float x, float y, Paint paint)
鸿蒙:
- drawText(Paint paint, String text, float x, float y)
4.获取text 的宽度
安卓:
- text.getWidth();
鸿蒙:
- Paint paint = new Paint();
- paint.setTextSize(text.getTextSize());
- float childWidth = paint.measureText(text.getText());
欢迎交流:HWIS-HOS@isoftstone.com