0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

基于安卓的滑动拼图验证组件实现鸿蒙化迁移和重构

OpenHarmony技术社区 来源:鸿蒙技术社区 作者:朱伟ISRC 2021-11-01 14:23 次阅读

基于安卓平台的滑动拼图验证组件 SwipeCaptcha,实现了鸿蒙化迁移和重构,代码已经开源,目前已经获得了很多人的 Star 和 Fork ,欢迎各位下载使用并提出宝贵意见!

开源地址:
https://gitee.com/isrc_ohos/swipe-captcha_ohos

在页面登录或者注册的时候,为了确保不是机器人操作,会让用户手动验证。

验证方式分为滑动拼图验证和滑动验证两种:

  • 滑动拼图验证:有图片作为背景,通过图块拼接实现安全验证。

  • 滑动验证:无图片背景,只拖动滑块便可实现安全验证。

本文的 SwipeCaptcha_ohos2.0 组件属于滑动拼图验证,操作简单,安全性强,可被应用于各种网站的登录、注册、找回密码或投票等场景中。

我们之前已经实现了滑动拼图验证组件 SwipeCaptcha_ohos,相关文章可查看《鸿蒙页面滑动组件,代码已开源!》 。

本次 SwipeCaptcha_ohos2.0 是基于之前移植的项目进行了相关功能的优化,具体优化内容将在下文中详细介绍。

组件效果展示

SwipeCaptcha_ohos2.0 的主要功能和之前的 SwipeCaptcha_ohos 基本一致,组件在使用时,有两个较为重要的元素:滑块和原图。

二者被放置于同一水平线上,用户拖动滑块至原图处使二者重合,误差小于提前设定的验证阈值,即可验证成功。每次调用组件,滑块和原图的位置都会发生随机变化。

SwipeCaptcha_ohos2.0 相较于之前的版本,大幅提升了组件功能的完整性以及使用体验。

下面将依次从组件验证失败和验证成功两个状态,展示 SwipeCaptcha_ohos2.0 与之前版本的效果对比。

①验证失败

通过图 1(a) 和图 1(b) 的对比可以看出,新版本移除了旧版本中“当前进度值预览”的不必要功能以及下方的状态栏。

取而代之的功能如下:
  • 验证滑块由正方形小块升级为“拼图块”样式。

  • 待验证背景图块增加了阴影遮罩效果。

  • 验证失败后增加了滑块闪烁效果以及“验证失败,请重新验证!”的弹窗提醒。

②验证成功

通过图 2(a) 和图 2(b) 的对比可以看出,新版本移除了旧版本中“当前进度值预览”的不必要功能以及下方的状态栏。

取而代之的功能如下:
  • 点击“重新生成验证码”按钮后,滑块和原图的位置都会发生随机变化。

  • 验证成功后增加了反光条划过的动画效果以及“验证成功!”的弹窗提醒。

除了上述直观的功能优化外,SwipeCaptcha_ohos2.0 还实现了以下功能:
  • 滑块大小和容错阈值的用户自定义:滑块大小自定义是指用户可以通过代码自定义滑块的宽高;容错阈值自定义是指用户可以通过代码自定义匹配时的容错率,即相差多少视作匹配成功。

  • 拼图背景在指定范围内的自适应填充:原组件的图片不能在指定组件宽高的前提下自动填充图片,如果强行适配宽高会出现拼图块内容错位的情况;经过改进后,验证图片已经能够适配布局中规定的组件宽高。

Sample 解析

通过上文相信大家已经了解 SwipeCaptcha_ohos2.0 组件的使用效果,下面将具体讲解 SwipeCaptcha_ohos2.0 组件的使用方法。

共分为如下 5 个步骤:
  • 步骤 1:导入 SwipeCaptchaView 类并声明类对象。

  • 步骤 2:在 xml 文件中添加 SwipeCaptchaView 控件。

  • 步骤 3:绑定 SwipeCaptchaView 控件。

  • 步骤 4:设置回调处理函数。

  • 步骤 5:设置 Button 控件监听事件,重新生成验证区域。

①导入 SwipeCaptchaView 类并声明类对象

在 MainAbilitySlice.java 文件中,通过 import 关键字导入 SwipeCaptchaView 类。

//导入SwipeCaptchaView类
importcom.huawei.swipecaptchaview.lib.SwipeCaptchaView;
publicclassMainAbilitySliceextendsAbilitySlice{
//声明SwipeCaptchaView类对象
SwipeCaptchaViewswipeCaptchaView;
......
}

②在 xml 文件中添加 SwipeCaptchaView 控件

在 xml 文件中添加 SwipeCaptchaView 控件,用于显示滑动验证的背景图和动态效果。设置控件高和宽、滑块的高和宽以及验证阈值等属性。

"http://schemas.huawei.com/res/ohos-auto"//声明一个用于传输自定义参数的命名空间
ohos:id="$+id:swipeCaptchaView"//规定控件id
ohos:height="220vp"//控件的高
ohos:width="330vp"//控件的宽
captcha:captchaHeight="30vp"//拼图滑块高
captcha:captchaWidth="30vp"//拼图滑块宽
captcha:matchDeviation="9"/>//验证失败的阈值

③绑定 SwipeCaptchaView 控件

在 MainAbilitySlice.java 的 onStart() 方法中,使用 findComponentById() 方法将 xml 文件中 SwipeCaptchaView 控件与 SwipeCaptchaView 类对象绑定;再调用 setImageId() 方法设置组件的背景图片。

//根据id找到相应的控件
swipeCaptchaView=(SwipeCaptchaView)findComponentById(ResourceTable.Id_swipeCaptchaView);
...
button=(Button)findComponentById(ResourceTable.Id_btn_change);

//设置背景图片
swipeCaptchaView.setImageId(ResourceTable.Media_pic01);

④设置回调处理函数

设置 SwipeCaptchaView 组件的回调处理函数,来提示用户滑动验证结果。

以提示用户验证成功为例:首先重写 matchSuccess() 方法,设置验证成功后的提示信息,然后实例化一个 ToastDialog 提示框对象,使用 setText() 方法设置显示文字为“验证成功!”。

setAlignment() 方法设置提示框的布局位置在整体布局的中央;show() 方法用于显示提示框。

设置验证失败的情况和验证成功同理,只需重写 matchFailed() 方法将文字信息设置为“验证失败!”即可。

//每次滑动结束后会根据判定结果回调
swipeCaptchaView.setOnCaptchaMatchCallback(newSwipeCaptchaView.OnCaptchaMatchCallback(){
@Override
publicvoidmatchSuccess(SwipeCaptchaViewswipeCaptchaView){
newToastDialog(getContext())
.setText("验证成功!")
.setAlignment(LayoutAlignment.CENTER)
.show();
}
});

⑤设置 Button 控件监听事件,重新生成验证区域

绑定 button 对象和 xml 文件中“重新生成验证码”Button控件;为 button 设置监听事件,每次点击按钮,都会调用 createCaptcha() 方法随机生成滑块和原图的位置。

button=(Button)findComponentById(ResourceTable.Id_btn_change);//绑定Button
button.setClickedListener(newComponent.ClickedListener(){//设置监听
@Override
publicvoidonClick(Componentcomponent){
swipeCaptchaView.createCaptcha();//随机生成滑块和原图的位置
...
}
});

Library解析

本部分将要重点介绍的类是图 3 中框出的 2 个类,分别是 DrawHelperUtils 和 SwipeCaptchaView。

它们向开发者提供设置 SwipeCaptcha_ohos2.0 组件相关属性的具体执行方法,其中 DrawHelperUtils 是工具类,SwipeCaptchaView 是具体实现滑块滑动效果的类,本节将分别讲解这两个类的内部逻辑实现。

①DrawHelperUtils 类

Swipeptcha_ohos2.0 升级实现的拼图滑块的原理是在方块的左、右两条竖边中点处分别绘制一个凸半圆或凹半圆(随机),可参考图 4。

DrawHelperUtils 类的 drawPartCircle() 方法具体用于绘制拼图滑块两条竖边上的半圆。

先来解释一下该方法涉及变量和参数的含义:
  • 起点坐标:开始绘制半圆的起点坐标,在图中由A表示,规定为方块竖边的前 1/3 处,由入参传入。

  • 终点坐标:开始绘制半圆的起点坐标,在图中由 C 表示,规定为方块竖边的后 1/3 处,由入参传入。

  • 中点坐标:半圆直径的中点坐标,在图中由 B 表示,由起点 A 和终点 C 的 X、Y 坐标计算得到。

  • r1:半圆半径 = AB 长度 = AC 长度/2 = 1/6 方块竖边长度。

  • gap1:由 r1 乘以贝塞尔曲线(cubicTo() 方法)系数 c 得到,用于确定控制点 D 和 F 的坐标,控制点作用是控制半圆绘制的轨迹。

  • flag:半圆的旋转系数,用来控制凹、凸半圆的绘制。当为 1 时,A、B、C 坐标与变量相加,绘制向外的凸半圆;当为 -1 时,其坐标与变量相减,得到向内的凹半圆。

以竖直绘制一个凸半圆为例,根据 A、B、C 点计算得到上述变量后,调用两次贝塞尔曲线 cubicTo(x1,y1,x2,y2,x3,y3)分别绘制前 1/2 和后 1/2 半圆。

此方法中需要使用到两个控制点,共有 6 个参数,分别表示控制点 1(x1,y1)、控制点 2(x2,y2)和绘制终点(x3,y3)。

如图 4-1,绘制前 1/2 半圆时以起点 A 右侧平行 gap1flag1 距离处作为第一个控制点 D、中点 B 右侧平行 r1 距离的半圆顶点第二个控制点 E、中点 B 作为绘制终点。

绘制后 1/2 半圆同理,以 E 点作为第一个控制点,终点 C 右侧平行 gap1flag1 距离处作为第二个控制点 F、终点 C 作为绘制终点。

其他绘制方向同理,若为从下向上绘制,则将 flag 设为 -1;若绘制凹半圆,则在计算坐标时横坐标反方向计算即可可参考图 4-2。

publicstaticvoiddrawPartCircle(Pointstart,Pointend,Pathpath,booleanouter){
floatc=0.551915024494f;
Pointmiddle=newPoint(start.getPointX()+(end.getPointX()-start.getPointX())/2,start.getPointY()+(end.getPointY()-start.getPointY())/2);//根据起点坐标A和终点坐标C算出中点B坐标
//半径
floatr1=(float)Math.sqrt(Math.pow((middle.getPointX()-start.getPointX()),2)+Math.pow((middle.getPointY()-start.getPointY()),2));
floatgap1=r1*c;//距离gap

if(start.getPointX()==end.getPointX()){//绘制竖直方向
booleantopToBottom=end.getPointY()-start.getPointY()>0;
intflag;//旋转系数
if(topToBottom){//若从上到下绘制
flag=1;//旋转系数设为1
}else{flag=-1;}//若从下到上绘制,设为-1
if(outer){//若为凸半圆,相加
path.cubicTo(start.getPointX()+gap1*flag,start.getPointY(),middle.getPointX()+r1*flag,middle.getPointY()-gap1*flag,middle.getPointX()+r1*flag,middle.getPointY());
path.cubicTo(middle.getPointX()+r1*flag,middle.getPointY()+gap1*flag,end.getPointX()+gap1*flag,end.getPointY(),end.getPointX(),end.getPointY());
}...}//若为凹半圆,则相减
}

②SwipteCaptchaView 类

SwipeCaptchaView 是具体实现滑块滑动效果的类,下文将从初始化滑动条并设置滑动条监听、初始化验证区域背景、设置验证后的动画效果、生成滑动验证区域四个方面具体讲解实现逻辑。

接下来将按类型讲解类中各方法间的调用逻辑,可参考图 4:

①初始化滑动条并设置滑动条监听

在 SwipteCaptchaView 类的构造函数中,调用 init() 方法进行初始化。其中,获取 xml 文件中控件参数即宽、高和系统屏幕宽度;通过 switch-case 判断来获取滑块的宽、高和滑动误差值。

mHeight=getHeight();//获取控件高和款
mWidth=getWidth();//获取系统屏幕宽度
if(mWidth==0){//mWidth=0为设置了match_parent的情况
mWidth=DisplayManager.getInstance().getDefaultDisplay(context).get().getAttributes().width;
}

for(inti=0;i< attrSet.getLength(); i++) {
    Optionalattr=attrSet.getAttr(i);
if(attr.isPresent()){
switch(attr.get().getName()){
case"captchaHeight"://获取滑块高度
mCaptchaHeight=attr.get().getDimensionValue();
break;
case"captchaWidth"://获取滑块宽度
...
case"matchDeviation"://获取滑动误差值
...
}
}
}

实例化 Image 类得到验证区域图片对象,并为其设置图片缩放模式以及位图格式等属性;实例化 Slider 类得到拖动条对象,为其设置宽、高、进度值、进度颜色等属性,以及监听事件。

mImage=newImage(context);//表示验证区域图片
...
mImage.setScaleMode(Image.ScaleMode.CLIP_CENTER);
mImage.setPixelMap(ResourceTable.Media_no_resource);
...
mSlider=newSlider(mLayout.getContext());//实例化Slider类表示拖动条
mSlider.setWidth(mWidth);//设置宽、高
mSlider.setHeight(SLIDER_HEIGHT);
mSlider.setMarginTop(mHeight-SLIDER_HEIGHT);
mSlider.setMinValue(0);//进度最小、最大值、当前进度值、进度颜色
mSlider.setMaxValue(10000);
mSlider.setProgressValue(0)
mSlider.setProgressColor(Color.BLACK);
setSlideListener();//设置拖动条的监听事件
...

在拖动条监听事件 setSlideListener() 方法中,重写 onTouchEnd() 方法,判断滑动结束后滑块位置的误差值是否小于规定误差值。

若小于则验证成功,取消滑块的阴影并设置回调;否则验证失败,直接设置回调。

@Override
publicvoidonTouchEnd(Sliderslider){
if(onCaptchaMatchCallback!=null){
if(Math.abs(mSlider.getProgress()*(mWidth-mCaptchaWidth)/10000-mCaptchaX)< mMatchDeviation) {//滑动结束后滑块位置误差值小于规定误差值验证成功
mCaptchaPaint.setMaskFilter(null);//取消滑块的阴影
slider.setEnabled(false);
onCaptchaMatchCallback.matchSuccess(SwipeCaptchaView.this);//设置验证成功后的回调
mSuccessAnim.start();//播放验证成功动画
}else{//滑动误差值大于规定误差值验证失败
slider.setProgressValue(0);
onCaptchaMatchCallback.matchFailed(SwipeCaptchaView.this);//设置验证失败后的回调
mFailAnim.start();//播放验证失败动画
}
}
}

②初始化滑动验证区域

在通过 Image 类对象调用 setPixelMap() 方法设置完验证图片后,由 initCaptcha() 方法完成验证区域的初始化。 实例化两个 Paint 类分别得到画笔对象和滑块目标区域对象,为其设置画笔抗锯齿和阴影、滑块样式和颜色等属性。

再分别调用 createMatchAnim() 和 createCaptcha() 方法设置验证后的动画效果和生成滑动验证区域。

privatevoidinitCaptcha(){
mRandom=newRandom(System.nanoTime());
//设置画笔
mCaptchaPaint=newPaint();//画笔对象
mCaptchaPaint.setAntiAlias(true);//抗锯齿
mCaptchaPaint.setDither(true);//使位图进行有利的抖动的位掩码标志
mCaptchaPaint.setStyle(Paint.Style.FILL_STYLE);
mCaptchaPaint.setMaskFilter(newMaskFilter(10,MaskFilter.Blur.SOLID));//阴影
//滑块目标区域
mMaskPaint=newPaint();//滑块目标区域对象
mMaskPaint.setAntiAlias(true);
mMaskPaint.setDither(true);
mMaskPaint.setStyle(Paint.Style.FILL_STYLE);//填充样式
mMaskPaint.setColor(newColor(Color.argb(188,0,0,0)));//填充颜色
mMaskPaint.setMaskFilter(newMaskFilter(20,MaskFilter.Blur.INNER));//阴影
mCaptchaPath=newPath();

createMatchAnim();//设置验证后的动画效果
createCaptcha();//生成验证码区域
}

③设置验证后的动画效果

由 createMatchAnim() 方法实现,能够设置验证成功或失败后的动画效果。

验证成功:通过 AnimatorValue 类对象设置动画间隔时间为 500 毫秒;并为其设置当值更新时的监听事件,重写 onUpdate() 方法,设置成功动画中拼图的偏移量。

//成功动画
intwidth=AttrHelper.vp2px(60,mLayout.getContext());
mSuccessAnim=newAnimatorValue();
mSuccessAnim.setDuration(500);//间隔时间为500毫秒
mSuccessAnim.setValueUpdateListener(newAnimatorValue.ValueUpdateListener(){
@Override//设置监听
publicvoidonUpdate(AnimatorValueanimatorValue,floatv){
mSuccessAnimOffset=(int)(v*(mWidth+width));//拼图偏移量
invalidate();
}
});

通过 Paint 类和 Path 类对象分别调用相关函数来完成阴影效果和动画路径的绘制。

mSuccessPaint=newPaint();
mSuccessPaint.setShader(newLinearShader(//设置阴影
newPoint[]{newPoint(0,0),newPoint(width*3/2,mHeight)},
newfloat[]{0,0.5f},
newColor[]{newColor(0x00FFFFFF),newColor(0x66FFFFFF)},
Shader.TileMode.MIRROR_TILEMODE),Paint.ShaderType.LINEAR_SHADER);
mSuccessPath=newPath();//绘制动画路径
mSuccessPath.moveTo(0,0);
mSuccessPath.rLineTo(width,0);
mSuccessPath.rLineTo(width/2,mHeight-SLIDER_HEIGHT);
mSuccessPath.rLineTo(-width,0);
mSuccessPath.close();//关闭

验证失败:与设置验证成功的前半部分流程相似,不同之处是将动画间隔设为 200 毫秒、还要设置画圈次数为 2 次。

在值更新时的监听事件中,需要判断当更新值小于 0.5f 时,将 isDrawMask 置为 false 即不绘制滑块,反之为 true 则绘制。

//设置验证失败动画
mFailAnim=newAnimatorValue();//实例化验证失败的动画对象
mFailAnim.setDuration(200);//设置间隔时间为200毫秒
mFailAnim.setLoopedCount(2);//设置画圈次数为2次
mFailAnim.setValueUpdateListener(newAnimatorValue.ValueUpdateListener(){
@Override
publicvoidonUpdate(AnimatorValueanimatorValue,floatv){
if(v< 0.5f){
isDrawMask=false;//不绘制滑块
}else{isDrawMask=true;}//绘制滑块
invalidate();
}});
}

④生成滑动验证区域

由 createCaptcha() 方法实现。先调用 createCaptchaPath() 方法绘制拼图块的轮廓路径。

其中通过 Random 类的 nextInt() 方法随机生成验证区域坐标,使滑块和原图位置随机变化。

再使用工具类 DrawHelperUtils 的 DrawPartCircle() 方法绘制拼图块左上角、右上角、右下角和左下角的图形。

privatevoidcreateCaptchaPath(){//绘制拼图块轮廓路径path
intgap=mCaptchaWidth/3;//拼图缺口的位置,设置在中间1/3处
mCaptchaX=mRandom.nextInt(mWidth-(mCaptchaWidth*3)-gap)+(mCaptchaWidth*2);//随机生成验证区域左上角的坐标
mCaptchaY=mRandom.nextInt(mHeight-SLIDER_HEIGHT-mCaptchaHeight-gap);
mCaptchaPath.reset();
mCaptchaPath.lineTo(0,0);
//开始绘制图形
mCaptchaPath.moveTo(mCaptchaX,mCaptchaY);//左上角
mCaptchaPath.lineTo(mCaptchaX+gap,mCaptchaY);
drawPartCircle(newPoint(mCaptchaX+gap,mCaptchaY),newPoint(mCaptchaX+gap*2,mCaptchaY),
mCaptchaPath,mRandom.nextBoolean());
...//右上角、右下角和左下角同理
mCaptchaPath.close();//绘制完成后及时关闭
}

接着生成滑动验证区域,前面介绍过,SwipeCaptcha_ohos2.0 版升级实现了验证区域背景图片自适应填充的效果。

其实现原理是先获取位图;根据图片的宽高和控件实际的宽高分别计算出水平方向和竖直方向上的缩放比例,两者中较大的是图片真实的缩放比例。

这是由于上文介绍的 Image 控件将图片缩放模式设为了 CLIP_CENTER,该模式会将图片的短边缩放至合适的大小并对长边进行裁剪。

因此较小的缩放比例代表被裁剪的边,较大的即在填充进控件时的真实缩放比例;接着绘制滑块目标区域的阴影,其不随拖动条的移动而更新。

最后绘制滑块区域,根据拖动条的数值计算画布偏移量,调用 drawPath() 方法绘制边框,获取图片 PixelMapHolder。

根据路径裁剪并将画布缩放至跟图片缩放程度一致,根据比例计算出垂直方向上由于 CLIP_CENTER 裁剪掉的图片的高度以及水平方向上被裁掉的宽度,即可绘制内容。

publicvoidcreateCaptcha(){//生成验证区域
if(mImage.getPixelMap()!=null){
createCaptchaPath();//绘制拼图块轮廓路径Path
...}

PixelMapmCaptchaPixelMap=mImage.getPixelMap();//getPixelMap(mLayout.getContext(),ResourceTable.Media_pic01);
//根据图片的原宽度和控件宽度算出缩放比例
intoriginWidth=mCaptchaPixelMap.getImageInfo().size.width;
intoriginHeight=mCaptchaPixelMap.getImageInfo().size.height;
floatratioWidth=(float)mWidth/originWidth;
floatratioHeight=(float)(mHeight-SLIDER_HEIGHT)/originHeight;
floatratio=Math.max(ratioWidth,ratioHeight);//更大的ratio

mImage.addDrawTask((component,canvas)->{//滑块目标区域阴影的绘制
canvas.drawPath(mCaptchaPath,mMaskPaint);
});
mLayout.addDrawTask((component,canvas)->{//滑块区域的绘制
if(isDrawMask){
canvas.translate(mSlider.getProgress()*(mWidth-mCaptchaWidth)/10000-mCaptchaX,0);//根据拖动条的数值计算画布的偏移量
canvas.drawPath(mCaptchaPath,mCaptchaPaint);//绘制边框
PixelMapHoldermCaptchaPixelMapHolder=newPixelMapHolder(mCaptchaPixelMap);//获取图片的PixelMapHolder
canvas.clipPath(mCaptchaPath,Canvas.ClipOp.INTERSECT);//根据路径裁剪
canvas.scale(ratio,ratio);//画布缩放至跟图片缩放程度一致
if(ratio==ratioWidth){
floatheightErr=(originHeight*ratio-(mHeight-SLIDER_HEIGHT))/2;//根据比例计算出垂直方向上由于CLIP_CENTER裁剪掉的图片的高度
canvas.drawPixelMapHolder(mCaptchaPixelMapHolder,0,-heightErr/ratio,mCaptchaPaint);//绘制内容
}
else{
floatwidthErr=(originWidth*ratio-mWidth)/2;//根据比例计算出水平方向上由于CLIP_CENTER裁剪掉的图片的宽度
canvas.drawPixelMapHolder(mCaptchaPixelMapHolder,-widthErr/ratio,0,mCaptchaPaint);//绘制内容
}
}});
}
责任编辑:haq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 鸿蒙系统
    +关注

    关注

    183

    文章

    2634

    浏览量

    66206
  • HarmonyOS
    +关注

    关注

    79

    文章

    1967

    浏览量

    30004

原文标题:鸿蒙滑动拼图验证组件,已开源!

文章出处:【微信号:gh_834c4b3d87fe,微信公众号:OpenHarmony技术社区】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    鸿蒙ArkTS容器组件:Swiper

    滑块视图容器,提供子组件滑动轮播显示的能力。
    的头像 发表于 07-15 09:51 525次阅读
    <b class='flag-5'>鸿蒙</b>ArkTS容器<b class='flag-5'>组件</b>:Swiper

    鸿蒙ArkTS声明式组件:TextPicker

    滑动选择文本内容的组件
    的头像 发表于 07-03 15:07 368次阅读
    <b class='flag-5'>鸿蒙</b>ArkTS声明式<b class='flag-5'>组件</b>:TextPicker

    最新开源代码证实!“鸿蒙原生版”微信正在积极开发中

    立刻保存,不需要同步或刷新调用。共享内存一般用于后台开发。 HarmonyOS NEXT 也就是鸿蒙星河版——俗称“纯血鸿蒙”,其采用华为自研内核,抛弃了传统的 Linux 内核以及 AOSP
    发表于 05-08 17:08

    华为鸿蒙操作系统发展战略:2024年完成安应用全面迁移

    徐直军指出,2024 年,鸿蒙操作系统的主要任务是打造原生应用生态。目前,鸿蒙操作系统已基本适配各种终端设备,但在应用开发方面仍依赖生态。
    的头像 发表于 04-18 15:38 747次阅读

    鸿蒙OS开发实例:【组件模式】

    组件一直是移动端比较流行的开发方式,有着编译运行快,业务逻辑分明,任务划分清晰等优点,针对Android端的组件;与Android端的组件
    的头像 发表于 04-07 17:44 588次阅读
    <b class='flag-5'>鸿蒙</b>OS开发实例:【<b class='flag-5'>组件</b><b class='flag-5'>化</b>模式】

    深圳市24年,实现鸿蒙原生应用数占全国总量10%以上

    操作系统。部分的高校已经取消了课程,从而开设鸿蒙课程;企业纷纷跟进启动了鸿蒙研发。 并且鸿蒙是完全具备无与伦比的机遇和潜力的。
    发表于 03-04 21:42

    面对如今的就业环境,vs鸿蒙……

    对于鸿蒙如今的发展势头,可谓是如火如荼。无论是最近发布的鸿蒙操作系统HarmonyOS NEXT,摒弃了Linux内核和AOSP开放源代码项目的代码, 不再兼容
    的头像 发表于 03-01 15:49 537次阅读
    面对如今的就业环境,<b class='flag-5'>安</b><b class='flag-5'>卓</b>vs<b class='flag-5'>鸿蒙</b>……

    没有“成份“的鸿蒙还能行吗?

    一、鸿蒙的前世今生 早前,就有关鸿蒙系统不兼容的消息引发了许多人的关注。由于2019年
    的头像 发表于 02-26 13:56 472次阅读
    没有“<b class='flag-5'>安</b><b class='flag-5'>卓</b>成份“的<b class='flag-5'>鸿蒙</b>还能行吗?

    华为鸿蒙系统怎么样 华为鸿蒙系统和系统的区别

    和稳定性。与此同时,鸿蒙系统与系统在架构、功能和生态系统方面都存在一些区别和特点,下面将详细探讨这些方面的差异。 首先,华为鸿蒙系统在架构上与
    的头像 发表于 02-02 14:54 1630次阅读

    鸿蒙OS和开源鸿蒙什么关系?

    内核,其他功能都以模块的形式存在。     华为用的是鸿蒙OS 我们都知道,华为手机的鸿蒙OS是可以运行软件的,是因为系统中有
    的头像 发表于 01-30 15:44 1079次阅读
    <b class='flag-5'>鸿蒙</b>OS和开源<b class='flag-5'>鸿蒙</b>什么关系?

    鸿蒙ArkUI开发-Tabs组件的使用

    鸿蒙ArkUI开发-Tabs组件的使用
    的头像 发表于 01-19 16:01 1759次阅读
    <b class='flag-5'>鸿蒙</b>ArkUI开发-Tabs<b class='flag-5'>组件</b>的使用

    鸿蒙系统和的区别 鸿蒙系统有什么特别之处

    了分布式架构,可以在不同设备上实现无缝连接和协同工作。而系统采用的是集中式架构,设备之间的连接和协同工作相对较为困难。 鸿蒙系统具备高度灵活性和可扩展性,支持设备与设备之间的直接通
    的头像 发表于 01-18 11:45 1.1w次阅读

    鸿蒙开发OpenHarmony组件复用案例

    复用能力。 环境准备 准备一个DevEco Studio,使用真机或者Simulator模拟器来验证组件复用接口 OpenHarmony SDK文件ets\\\\component
    发表于 01-15 17:37

    鸿蒙开发基础-Web组件之cookie操作

    }) ... } ... 本文章主要是对鸿蒙开发当中ArkTS语言的基础应用实战,Web组件里的cookie操作。更多的鸿蒙应用开发技术,可以前往我的主页学习更多,下面是一张鸿蒙
    发表于 01-14 21:31

    鸿蒙系统和的区别哪个好用

    的一些问题,如性能、隐私安全等。而系统是由谷歌开发和推广的移动设备操作系统,目前在全球范围内占据主导地位。 鸿蒙系统的架构更为先进和高效。鸿蒙系统采用了分布式架构,可以在不同设备之
    的头像 发表于 01-11 11:15 1868次阅读