成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

HarmonyOS 實(shí)現(xiàn)一個(gè)手繪板

系統(tǒng) OpenHarmony
本篇文章分享一下我實(shí)現(xiàn)的一個(gè)手繪板,利用openHarmony內(nèi)置的API cnavas組件實(shí)現(xiàn)。

??想了解更多關(guān)于開(kāi)源的內(nèi)容,請(qǐng)?jiān)L問(wèn):??

??51CTO 開(kāi)源基礎(chǔ)軟件社區(qū)??

??https://ost.51cto.com??

前言

最近在學(xué)習(xí)openHarmony,恰好之前了解過(guò)canvas,所以本篇文章分享一下我實(shí)現(xiàn)的一個(gè)手繪板,利用openHarmony內(nèi)置的API cnavas組件實(shí)現(xiàn)。

介紹

這是一個(gè)手繪板,并且可以根據(jù)滑動(dòng)屏幕速度,動(dòng)態(tài)生成線條大小,當(dāng)用戶觸摸屏幕,會(huì)生成線條,并且速度越快,線條越細(xì)。

效果展示

#夏日挑戰(zhàn)賽# HarmonyOS 實(shí)現(xiàn)一個(gè)手繪板-開(kāi)源基礎(chǔ)軟件社區(qū)

原理分析

1、繪制原理

使用前,需要線了解canvas組件,可以參考harmonyOS開(kāi)發(fā)者文檔,文檔介紹的非常詳細(xì),這里就不多介紹了。

首先,我們需要將canvas上下文對(duì)象,需要在觸摸移動(dòng)事件中綁定,因?yàn)槲覀兪峭ㄟ^(guò)觸摸來(lái)生成對(duì)應(yīng)線條的。

然后,屬性選擇lineCap,屬性值有三種:butt、round、square,我嘗試了后發(fā)現(xiàn)round效果比較好。

  • 屬性值為butt時(shí)的效果

#夏日挑戰(zhàn)賽# HarmonyOS 實(shí)現(xiàn)一個(gè)手繪板-開(kāi)源基礎(chǔ)軟件社區(qū)

  • 屬性值為round

  • 屬性值為square

其實(shí)butt效果也還行,就是鋸齒太嚴(yán)重,雖然API中有內(nèi)置抗鋸齒屬性,但是不知道為啥設(shè)置了沒(méi)有效果,可能粒度太大了,現(xiàn)在這個(gè)粒度已經(jīng)有點(diǎn)卡了,如果把粒度小設(shè)置小一點(diǎn)估計(jì)更卡。
綜上還是選擇了round,它會(huì)將線端點(diǎn)以圓形結(jié)束,所以效果上更圓潤(rùn)。

最后將數(shù)組中的最后一個(gè)值取出,作為moveTo的坐標(biāo),將鼠標(biāo)移動(dòng)后的點(diǎn)作為lineTo的坐標(biāo),然后再通過(guò)stroke就能繪制出圖像。

繪制直線,通常使用moveTo ()與lineTo ()兩個(gè)方法。. moveTo ()方法用于將畫筆移至指定點(diǎn)并以改點(diǎn)為直線的開(kāi)始點(diǎn),lineTo ()則為結(jié)束點(diǎn)。

const el = this.$refs.canvas;
this.ctx = el.getContext('2d')
this.ctx.lineWidth =this.lineWidth/2
this.ctx.beginPath()
// 向線條的每個(gè)末端添加圓形線帽。
this.ctx.lineCap = 'square'
// 每次將數(shù)組中最后一個(gè)值取出,作為起始點(diǎn)
this.ctx.moveTo(this.ArrX[this.ArrX.length-1],this.ArrY[this.ArrY.length-1])
this.ctx.lineTo(e.touches[0].localX,e.touches[0].localY)
this.ctx.stroke()
this.ArrX.push(e.touches[0].localX)
this.ArrY.push(e.touches[0].localY)

2、線條粗細(xì)

想要通過(guò)速度來(lái)計(jì)算線條粗細(xì),那么可以是需要獲取兩點(diǎn)之間的時(shí)間,通過(guò)時(shí)間和距離得到速度。

當(dāng)觸發(fā)touchmove事件,將當(dāng)前時(shí)間戳存儲(chǔ)起來(lái),通過(guò)上一次觸發(fā)事件獲得的時(shí)間-當(dāng)前觸發(fā)事件獲得的時(shí)間,就可以得到兩次觸發(fā)事件的事件間隔,此時(shí)我們就獲得了兩點(diǎn)之間的時(shí)間。

再計(jì)算兩點(diǎn)之間的距離(平方和再開(kāi)根號(hào)),通過(guò) 路程/時(shí)間 = 速度計(jì)算出兩點(diǎn)之間的速度,從而可以動(dòng)態(tài)生成線條粗細(xì)。

// 計(jì)算線條粗細(xì)
const currTime = Date.now()
if(this.startTime !== 0){
const duration = currTime - this.startTime
// 傳入倒數(shù)第二個(gè)點(diǎn)和最后一個(gè)點(diǎn),和持續(xù)時(shí)間,會(huì)返回加速度
const v = this.speed(this.ArrX[this.ArrX.length-2],this.ArrY[this.ArrY.length-2],this.ArrX[this.ArrX.length-1],this.ArrY[this.ArrY.length-1],duration)
this.lineWidth = this.lineWidth/v
if(this.lineWidth>25){
this.lineWidth = 25
}
if(this.lineWidth<1){
this.lineWidth = 1
}
}
this.startTime = currTime

完整代碼

index.js:

// @ts-nocheck
export default {
data: {
ctx:'',
ArrX:[],
ArrY:[],
// 開(kāi)始時(shí)間
startTime:0,
lineWidth:14
},
// 偏移很多
touchstart(e){
// 開(kāi)始時(shí)間清空
this.startTime = 0
this.ArrX.push(e.touches[0].localX)
this.ArrY.push(e.touches[0].localY)
},
// 計(jì)算最后兩點(diǎn)的速度
speed(x1,y1,x2,y2,s){
const x = Math.abs(x1-x2)*Math.abs(x1-x2)
const y = Math.abs(y1-y2)*Math.abs(y1-y2)
return Math.sqrt(x+y)/s
},
touchmove(e){
// 計(jì)算線條粗細(xì)
const currTime = Date.now()
if(this.startTime !== 0){
const duration = currTime - this.startTime
// 傳入倒數(shù)第二個(gè)點(diǎn)和最后一個(gè)點(diǎn),和持續(xù)時(shí)間,會(huì)返回加速度
const v = this.speed(this.ArrX[this.ArrX.length-2],this.ArrY[this.ArrY.length-2],this.ArrX[this.ArrX.length-1],this.ArrY[this.ArrY.length-1],duration)
this.lineWidth = this.lineWidth/v
if(this.lineWidth>25){
this.lineWidth = 25
}
if(this.lineWidth<1){
this.lineWidth = 1
}
}
this.startTime = currTime
const el = this.$refs.canvas;
this.ctx = el.getContext('2d')
this.ctx.lineWidth =this.lineWidth/2
this.ctx.beginPath()
// 向線條的每個(gè)末端添加圓形線帽。
this.ctx.lineCap = 'square'
// 每次將數(shù)組中最后一個(gè)值取出,作為起始點(diǎn)
this.ctx.moveTo(this.ArrX[this.ArrX.length-1],this.ArrY[this.ArrY.length-1])
this.ctx.lineTo(e.touches[0].localX,e.touches[0].localY)
this.ctx.stroke()
this.ArrX.push(e.touches[0].localX)
this.ArrY.push(e.touches[0].localY)
},
touchend(e){
this.startTime = 0
}
}

index.hml:

<div class="container">
<canvas ref="canvas" class="canvas" @touchstart="touchstart"
@touchmove="touchmove" @touchend="touchend"/>
</div>

index.css:

.container{
margin: 50px;
}
.canvas{
height: 100%;
width: 100%;
background-color: #eee;
border: 1px solid #ffc300;
}

總結(jié)

不足點(diǎn):使用體驗(yàn)不是很好,后續(xù)還需要優(yōu)化。

最后,通過(guò)自定義組件,加深對(duì)HarmonyOS的開(kāi)發(fā),共建鴻蒙生態(tài)!

??想了解更多關(guān)于開(kāi)源的內(nèi)容,請(qǐng)?jiān)L問(wèn):??

??51CTO 開(kāi)源基礎(chǔ)軟件社區(qū)??

??https://ost.51cto.com??。

責(zé)任編輯:jianghua 來(lái)源: 鴻蒙社區(qū)
相關(guān)推薦

2024-10-12 16:38:09

2022-07-28 14:20:44

懸浮球鴻蒙

2022-08-02 14:21:20

滑動(dòng)驗(yàn)證碼鴻蒙

2022-08-01 14:07:26

canvas繪畫板

2009-07-02 10:02:40

JSP程序

2020-10-27 10:00:26

鴻蒙開(kāi)發(fā)板物聯(lián)網(wǎng)

2020-11-19 10:25:24

MQTT

2021-09-07 07:34:42

CSS 技巧代碼重構(gòu)

2017-12-12 15:24:32

Web Server單線程實(shí)現(xiàn)

2014-04-14 15:54:00

print()Web服務(wù)器

2020-09-24 11:46:03

Promise

2022-01-04 11:08:02

實(shí)現(xiàn)Localcache存儲(chǔ)

2021-06-02 16:32:23

鴻蒙HarmonyOS應(yīng)用

2020-12-17 12:06:49

鴻蒙應(yīng)用鴻蒙開(kāi)發(fā)

2020-08-17 08:20:16

iOSAOP框架

2022-05-13 07:42:25

JS編程題LazyMan

2022-11-29 17:34:43

虛擬形象系統(tǒng)

2025-05-20 08:00:00

鏈?zhǔn)?/a>調(diào)用異步

2024-12-06 09:58:09

2011-03-28 09:56:03

存儲(chǔ)增刪操作
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 在线观看午夜视频 | 欧美国产日韩在线观看成人 | 一区二区三区在线免费观看视频 | 国产精品久久久久久久毛片 | 精品国产免费一区二区三区五区 | 欧美a在线 | 亚洲国产一| 日本a级大片 | 欧美视频精品 | 男人久久天堂 | 日本一区不卡 | 久久久久久黄 | 日韩欧美精品一区 | 欧美国产在线一区 | 国产精品国产成人国产三级 | 婷婷久久久久 | 精品三区| 日韩视频在线免费观看 | 成人免费视频网站在线看 | 天天干天天干 | 日本三级在线网站 | 激情的网站 | 国产高清视频 | 一区二区三区亚洲 | 嫩草视频网| 久久国产一区 | www.久草.com | 一区二区高清不卡 | 精品国产一区二区国模嫣然 | 久久视频免费看 | 91麻豆精品国产91久久久久久 | 超碰最新在线 | 久久精品欧美一区二区三区不卡 | 久久久国产一区二区三区 | 992tv人人草 久久精品超碰 | 中文字幕免费在线 | 天天操操| 久草在线 | 日日夜夜精品视频 | 毛片在线视频 | 夜夜艹天天干 |