@Entry
@Component
struct Page2 {
//Two-dimensional array represents the motion area
@State area:number[][] = []
//Current direction of movement
@State directions: 'up'| 'down'| 'left'| 'right' = 'right'
//Control direction
@State tryDirections: 'up'| 'down'| 'left'| 'right' = 'right'
//Historical location Used to update the snake's head and tail when it moves
@State historyPosition:number[][]=[]
//Head coordinates
@State headX:number = 0
@State headY:number = 0
//The length of the snake
@State length:number = 0
//Color Different representations are different, etc.
@State colors:ResourceColor[]=[, , ]
//End Tips
@State end:string=''
//initialization
initArea(){
// 25 * 25Initialize the map
this.area = []
for(let i=0;i<25;i++){
this.area.push([])
for(let j=0;j<25;j++){
this.area[i].push(0)
}
}
//Initialize snakes
this.area[4][6]=1
this.area[4][7]=1
this.area[4][8]=1
this.area[4][9]=1
this.area[4][10]=1
//Initialize the historical location
=[]
([6,4])
([7,4])
([8,4])
([9,4])
([10,4])
//Initialize the head position
= 10
= 4
//Initialize the length of the snake
this.length = 5
//Initialization End prompt
this.end=''
//The initialization direction and the direction you want to try are right
= 'right'
= 'right'
}
aboutToAppear(): void {
//initialization
a()
//Randomly generated energy location
let foodX:number = (Math.random()*this.area[0].length)
let foodY:number = (Math.random()*this.area[0].length)
//Control the movement of the snake (update the coordinates)
setInterval(()=>{
//No operation after the game is over
if(!this.end){
//The previous one has not been eaten yet or the snake's length covers the full screen without generating it
if(this.area[foodY][foodX]!=2&&this.length<25*25){
while(true){
//The generated random coordinates are0When it is empty, energy will be generated and exit the cycle
if(this.area[foodY][foodX]==0){
this.area[foodY][foodX]=2
break
}
//Continue to generate
foodX = (Math.random()*this.area[0].length)
foodY = (Math.random()*this.area[0].length)
}
}
//New position of the head after moving
let newHeadY:number
let newHeadX:number
//Change the current position only when the control direction and the current moving direction meet the conditions. Avoid the snake's head moving behind him.
if( == 'right' && != 'left') = 'right'
else if( == 'left' && != 'right') = 'left'
else if( == 'up' && != 'down') = 'up'
else if( == 'down' && != 'up') = 'down'
//Update different headers in different directions
if( == 'right'){
newHeadY =
newHeadX = +1
}else if( == 'up'){
newHeadY = -1
newHeadX =
}else if( == 'down'){
newHeadY = +1
newHeadX =
}else{
newHeadY =
newHeadX = -1
}
//Determine whether it hits a wall
if(newHeadY>=0 && newHeadY<this.area.length && newHeadX>=0 && newHeadX<this.area[0].length){
//Not hit the wall
//Hit yourself with your head
if(this.area[newHeadY][newHeadX]==1){
this.end = 'Eat until you're alone The game is over! '
} //Not hit
else{
//Do you eat energy
let eat:boolean = false
//If the head touches2It is considered to eat energy
if(this.area[newHeadY][newHeadX] == 2) eat = true
//Put the head position into the historical position
([newHeadX,newHeadY])
//Get the line of header
let newRowHead = this.area[newHeadY]
//Set the head position to1It becomes part of a snake
newRowHead[newHeadX] = 1
//Update the location of the snake head
= newHeadY
= newHeadX
//The line that updates the snake head
this.area.splice(newHeadY,1,newRowHead)
//Eat energy without renewing the tail. Add one in length
if(!eat){
//The line that updates the tail
//Get the tail position
let tailX = [0][0]
let tailY = [0][1]
//Remove the tail
()
//The line that gets the old tail
let pastRowTail = this.area[tailY]
//Set the position of the old tail to0Indicates tail movement
pastRowTail[tailX] = 0
//The line that updates the old tail
this.area.splice(tailY,1,pastRowTail)
}else {
//Eat it, add one length
this.length++
}
}
} //Running out of the border is hitting the wall
else this.end='It hit the wall and the game ends! '
}
},200) //200Move once
}
build() {
Column() {
Stack(){
//area
Column(){
//Iterate through each line
ForEach(this.area,(itemR:number[], indexR:number)=>{
Row(){
//Iterate through each position of each line
ForEach(itemR,(itemC:number, indexC:number)=>{
//Render each location0:Empty location1:Snake part2:Energy and whether it is a snake head
Text(indexR == && indexC == ?'·' : '')
.textAlign()
.fontSize(12)
.fontWeight(900)
.fontColor()
.width(12)
.aspectRatio(1)
.backgroundColor([.length-1][0] == indexC &&
[.length-1][1] == indexR?
: [this.area[indexR][indexC]])
})
}
})
}
.borderWidth(2)
.borderColor()
//Show prompts at the end
Column({space:10}){
Text(this.end)
.fontSize(20)
.fontColor()
Text(this.end?'restart':'')
.fontSize(22)
.fontColor()
.onClick(()=>{
a()
})
}
.zIndex(this.end?1:-1) //Set the z-axis at the end1Show it
}
//Control keys
Stack(){
//Up, down, left, right logic is the same as"superior"Operation as an example
Button('superior')
.fontSize(20)
.height(50)
.width(80)
.position({top:0, left:'50%'})
.translate({x:'-50%'})
.onClick(()=>{
= 'up'
})
Button('Down')
.fontSize(20)
.height(50)
.width(80)
.position({bottom:0, left:'50%'})
.translate({x:'-50%'})
.onClick(()=>{
= 'down'
})
Button('Left')
.fontSize(20)
.height(50)
.width(70)
.position({left:0, top:'50%'})
.translate({y:'-50%'})
.offset({left:0})
.onClick(()=>{
= 'left'
})
Button('right')
.fontSize(20)
.height(50)
.width(70)
.position({right:0, top:'50%'})
.translate({y:'-50%'})
.onClick(()=>{
= 'right'
})
}
.enabled(this.end?false:true) //The game is disabled when the button expires
.width('45%')
.aspectRatio(1)
}
.height('100%')
.width('100%')
}
}