Android开心消消乐代码实例详解

突然想要在android上写一个消消乐的代码,在此之前没有系统地学过java的面向对象,也没有任何android相关知识,不过还是会一点C++。8月初开始搭建环境,在这上面花了相当多的时间,然后看了一些视频和电子书,对android有了一个大概的了解,感觉差不多了的时候就开始写了。

疯狂地查阅各种资料,反反复复了好几天后,也算是写出了个成品。原计划有很多地方还是可以继续写下去的,比如UI设计,比如动画特效,时间设计,关卡设计,以及与数据库的连接,如果可以的话还能写个联网功能,当然因为写到后期内心感到格外的疲倦,所以以上内容全部被砍掉了。

所以最后的成果就只有这样了……因为完全没有设计过所以非常难看,功能也只有消方块而已,图片还是从之前写的连连看里搬过来的。反正就算一个小的demo好了。

布局

一开始设计的布局是xml里的,但是后来想了想有64个按钮这样复制下去实在是太不好了,所以就把按钮那一部分用代码布局了。

private void initBtn(int i,int j)
  {
		btn[8*i+j].setBackgroundDrawable
		(this.getResources().getDrawable(getStyle()));
		point p = new point(i,j,btn[8*i+j],id);
		btn[8*i+j].setTag(p);
		map[i][j] = num;
  }

//……
    LinearLayout vlayout = (LinearLayout)findViewById(R.id.vlayout);
    TableLayout tlayout = new TableLayout(this);
    TableRow row[] = new TableRow[8];
    for(int i=0;i<8;i++){
    	row[i] = new TableRow(this);
    	row[i].setGravity(Gravity.CENTER);
    	for(int j=0;j<8;j++){
    		btn[8*i+j] = new ImageButton(this);
    		btn[8*i+j].setLayoutParams(new TableRow.LayoutParams(38,38));
    		initBtn(i,j);
    		btn[8*i+j].setOnClickListener(listener);
    		row[i].addView(btn[8*i+j]);
    	}
    	tlayout.addView(row[i]);
    }
//……

其中getStyle()函数返回了按钮的样式id,该id是随机生成的。

point p存储了关于按钮的信息,它在按钮点击事件中会被使用。

android中的按钮有三种状态:点击态、普通态、焦点态。最后一个的意思是用户按方向键盘(或类似功能键)来控制哪个按钮处于焦点,这样就可以不通过鼠标对按钮进行操作。当然在这里并没有用到这个状态,只用到了前面两种,但是因为已经有现成图片了,就把三种状态都写入了btn?.xml(放在drawable文件夹下),使用的时候直接引用这个xml文件就可以了。

btn1.xml:

<?xml version="1.0" encoding="UTF-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android"> 

  <item android:drawable="@drawable/a1_2" android:state_pressed="true"/>
  <item android:drawable="@drawable/a1" android:state_focused="false" android:state_pressed="false"/>
  <item android:drawable="@drawable/a1_1" android:state_focused="true"/>
  <item android:drawable="@drawable/a1" android:state_focused="false"/> 

</selector>

判断是否可消去

在点击事件中,用两个变量记录先后点击的按钮信息,这两个变量是滚动使用的,也就是说,第一次点击存入变量1,第二次点击存入变量2,第三次点击存入变量1,第四次点击存入变量2……额外有一个布尔变来控制信息存入哪个变量。

每次点击后,都计算先后两次点击的按钮是否相邻。如果相邻,那么交换它们的位置。

我们用二维数组mark来记录每个方块是否可以消去,初始化为0。交换后,先一行行地扫描一遍,检查是否有相连的三个以上图案相同的按钮,如果有的话,把它们都做上标记。然后再一列列扫描一遍,同样检查是否有相连的三个以上图案相同的按钮。总体复杂度为O(2*n^2)

当然可以有更快的算法,但是在这个代码里,我们对计算的速度没有要求,甚至希望它慢一些(因为动画效果需要在这里停顿一下),所以就没有必要进行优化了。

那么mark里存储些什么呢?一开始我把mark设计为布尔量,可消去的设为true。后来写到消去后更新按钮后,我突然想到mark里不仅仅可以记录是否可消去,还可以记录消去后应该更新为什么。

我们知道,消去一行按钮后,上面的按钮会掉下来补充空位,也就是说消去的这一行会被上面一行取代。所以我们把这些按钮的mark赋值为1。

同样,消去一列n个按钮后,每个按钮会被它上面第n个按钮取代,所以我们把这些按钮的mark赋值为n。
这样,在更新的时候,我们只需要遍历二维数组mark,根据它对应的值,来采取相应的操作就可以了。

在这里,要注意顺序性的问题,首先,在赋值mark的时候,应该先扫描横行,再扫描竖列,因为可能会出现十字架或T字型的消去,这里横纵有重叠,而更新的时候以纵为准,所以后扫描纵列来覆盖横行的值。

另外一个要注意的顺序,就是在更新的时候,一定要从上往下扫描。如果从下往上扫描的话,下面图案的更新可能会破坏到上面的图,那么,上面原来存在的可以被消去的方块就已经被破坏了,但是mark还记录着消去方块的索引,这样就会引起错的消去。反之,上面图案的消去是不会破坏下面的图案的。

判断地图是否还存在解

每一轮消去后,我们都需要判断地图上是否还存在解,如果不存在,就要进行更新

因为仅仅是判断存在性,算法略有变化。

我们需要考虑到所有的交换情况,以及它们能否产生可行解,一旦找到一个,我们就可以返回true。所以外循环包含两个,一个扫描横向相邻的所有方块,一个扫描纵向相邻的所有方块。

对于特定的两个方块,我们先交换它们,然后对于两个方块,都计算它们的十字架区域是否存在可行解,之后再把它们交换回来:

(十字架区域)
总体的最坏时间复杂度是2*(n^2)*2*8。

多线程

在主线程里,按理来说应该有消去 — 更新这样的画面,但是我发现android是直接把所有东西都计算了出来,然后再去显示UI的,而不是边计算边显示,所以我之前设置的那些一步步更新画面的代码一点儿用也没有,然后我想了想估计是要用多线程来写,在此之前我没有写过多线程的代码,所以花了一天时间看了多线程并把这部分修正了。

大概的想法是这样的:每次点击了两个相邻按钮后,先交换两个按钮,然后调用线程的start()方法。

在线程重写的run()方法里,先计算是否存在解(find函数),如果可以消去,每隔0.03s发送一个消息,通知主线程,主线程设置按钮的alpha值减小(增加透明度),反复10次,这样就做出了按钮慢慢消失的效果。

之后,再停顿0.1s,向主线程发送一个消息,主线程开始进行更新按钮操作。更新之后,因为掉落下来的按钮还可能组成新的可消去的部分,所以继续调用线程的start方法,直到不存在可消去的按钮。

如果不存在解,那么,先判断这是因为用户点击的两个按钮无法产生解,还是之前消去后掉落下来的按钮不会产生解。我们用flag来记录这一状态。如果是前者,先停顿0.3s,再把两个按钮的位置交换回来;如果是后者,说明已经消去所有按钮,这时我们检查是否还存在可行解,如果不存在,向主线程发送一个消息,通知它更新地图。

总之就是重写run,以及消息机制的使用。特别注意的就是run中绝对不能写UI的操作,UI的事情只能交给主线程做,所以才需要不断通知。

还有一个要注意线程之间的执行顺序,调试了很久的一个地方,后来发现是因为主线程那边还没计算出结果,但是次线程已经开始新的计算了,所以此线程用到的是主线程没有更新过的旧数据。后来想到的方法就是等主线程算完了,再回来调用次线程的函数。

@Override
  public void run()
  {
		if(find()){
			flag = false;
			int n = 10;
			alpha = 255;
			while(n--!=0){
  		  wait(30);
  		  mHandler.sendEmptyMessage(0);
			}
  		wait(100);
  		mHandler.sendEmptyMessage(1);

	  }
		else if(flag==true){
			swapMap(p1.x,p1.y,p2.x,p2.y);
  		wait(300);
  		mHandler.sendEmptyMessage(2);
	  	}
		else if(flag==false){
	  	p1 = new point(-2,-2);
  	  p2 = new point(-2,-2);
			if(check()==false){
				mHandler.sendEmptyMessage(3);
			}
		}
  }

代码

main.xml

<?xml version="1.0" encoding="UTF-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:background="@drawable/background" 
        android:orientation="vertical"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"
        android:id="@+id/vlayout">
        <TextView
            android:id="@+id/text"
            android:layout_width="fill_parent"
            android:layout_height="40dip"
            android:textSize="18sp"
            android:autoText="true"
            android:textColor="#000000"
            android:capitalize="sentences"
            android:text="开心消消乐" />
</LinearLayout>

MainActivity.java

package com.example.android.market.licensing; 

import android.annotation.SuppressLint;
import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.widget.ImageButton;
import android.widget.LinearLayout;
import android.widget.TableLayout;
import android.widget.TableRow;
import android.widget.TextView;
import android.widget.Toast;
import android.view.Gravity;
import android.view.View; 

@SuppressLint("NewApi")
public class MainActivity extends Activity implements Runnable {
  private static final String TAG = "App";
  private point p1 = new point(-2,-2);
  private point p2 = new point(-2,-2);;
  private boolean isPoint1 = true;
  private int map[][] = new int[8][8];
  private int id;
  private int mark[][] = new int[8][8];
  private Thread thread;
  private int num;
  private int alpha = 255;
  boolean flag = false;
  private int score = 0;
  private TextView text;
  ImageButton[] btn = new ImageButton[64]; 

  private int getStyle(){
    num = (int)(1+Math.random()*(7-1+1));
    switch(num){
    case 1:return id = R.drawable.btn1;
    case 2:return id = R.drawable.btn2;
    case 3:return id = R.drawable.btn3;
    case 4:return id = R.drawable.btn4;
    case 5:return id = R.drawable.btn5;
    case 6:return id = R.drawable.btn6;
    case 7:return id = R.drawable.btn7;
    default:return id = 0;
    }
  } 

  public MainActivity() { } 

  private void initBtn(int i,int j)
  {
    btn[8*i+j].setBackgroundDrawable
    (this.getResources().getDrawable(getStyle()));
    point p = new point(i,j,btn[8*i+j],id);
    btn[8*i+j].setTag(p);
    map[i][j] = num;
  } 

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);
    LinearLayout vlayout = (LinearLayout)findViewById(R.id.vlayout);
    TableLayout tlayout = new TableLayout(this);
    TableRow row[] = new TableRow[8];
    for(int i=0;i<8;i++){
      row[i] = new TableRow(this);
      row[i].setGravity(Gravity.CENTER);
      for(int j=0;j<8;j++){
        btn[8*i+j] = new ImageButton(this);
        btn[8*i+j].setLayoutParams(new TableRow.LayoutParams(38,38));
        initBtn(i,j);
        btn[8*i+j].setOnClickListener(listener);
        row[i].addView(btn[8*i+j]);
      }
      tlayout.addView(row[i]);
    }
    text = new TextView(this);
    text.setText("分数 : 0");
    vlayout.addView(tlayout);
    vlayout.addView(text);
    while(find()){
      updateState();
    }
    thread = new Thread(this);
  } 

  private void wait(int time)
  {
    try {
      Thread.sleep(time);
    } catch (InterruptedException e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
    }
  } 

  @Override
  public void run()
  {
    if(find()){
      flag = false;
      int n = 10;
      alpha = 255;
      while(n--!=0){
        wait(30);
        mHandler.sendEmptyMessage(0);
      }
      wait(100);
      mHandler.sendEmptyMessage(1); 

    }
    else if(flag==true){
      swapMap(p1.x,p1.y,p2.x,p2.y);
      wait(300);
      mHandler.sendEmptyMessage(2);
    }
    else if(flag==false){
      p1 = new point(-2,-2);
      p2 = new point(-2,-2);
      if(check()==false){
        mHandler.sendEmptyMessage(3);
      }
    }
  }  

   void swapImage()
  {
    p1.v.setBackgroundDrawable
    (MainActivity.this.getResources().getDrawable(p2.id));
    p2.v.setBackgroundDrawable
    (MainActivity.this.getResources().getDrawable(p1.id));
    int tmp =p1.id;
    p1.id = p2.id;
    p2.id = tmp;
    p1.v.setTag(p1);
    p2.v.setTag(p2);
  } 

  private void updateBtn(int x1,int y1,int x2,int y2)
  {
    map[x1][y1] = map[x2][y2];
    point p = (point)(btn[8*x1+y1].getTag());
    p.id = ((point)(btn[8*x2+y2].getTag())).id;
    btn[8*x1+y1].setTag(p);
    btn[8*x1+y1].setBackgroundDrawable
    (MainActivity.this.getResources().getDrawable(p.id));
  } 

  private void updateBtn(int i,int j)
  {
    btn[8*i+j].setBackgroundDrawable
    (MainActivity.this.getResources().getDrawable(getStyle()));
    map[i][j] = num;
    point p = (point)(btn[8*i+j].getTag());
    p.id = id;
    btn[8*i+j].setTag(p);
  } 

  private void hideBtn(){
    alpha -= 25;
    for(int i=0;i<8;i++){
      for(int j=0;j<8;j++){
        if(mark[i][j]!=0&&mark[i][j]!=2){
          btn[8*i+j].getBackground().setAlpha(alpha);
        }
      }
    }
  } 

  private void updateState()
  { 

    for(int i=0;i<8;i++){
      for(int j=0;j<8;j++){
        btn[8*i+j].getBackground().setAlpha(255);
      }
    }
    for(int i=0;i<8;i++){
      for(int j=0;j<8;j++){
        if(mark[i][j]==1){
          for(int k=i;k>0;k--){
            updateBtn(k,j,k-1,j);
          }
           updateBtn(0,j);
        }
        else if(mark[i][j]>=3){
          if(i-mark[i][j]>=0) {
            updateBtn(i,j,i-mark[i][j],j);
            updateBtn(i-mark[i][j],j);
          }
          else{
            updateBtn(i,j);
          }
        }
        else if(mark[i][j]==2){
        	updateBtn(i,j);
        }
      }
    }
  } 

  public Handler mHandler=new Handler()
  {
    public void handleMessage(Message msg)
    {
      switch(msg.what){
        case 0:{
          hideBtn();
          break;
        }
        case 1:{
          updateState();
          text.setText("分数 " + ((Integer)score).toString());
          thread.start();
          break;
        }
        case 2:{
          swapImage();
          p1 = new point(-2,-2);
          p2 = new point(-2,-2);
          break;
        }
        case 3:{
          Toast.makeText(MainActivity.this, "已自动生成新地图", Toast.LENGTH_SHORT).show();
          for(int i=0;i<8;i++){
            for(int j=0;j<8;j++){
              initBtn(i,j);
            }
          }
          while(find()){
            updateState();
          }
          break;
        }
      }
      super.handleMessage(msg);
    }
  };    

  private boolean find()
  {
    for(int i=0;i<8;i++){
      for(int j=0;j<8;j++){
        mark[i][j] = 0;
      }
    }
    boolean flag = false;
    // heng
    for(int i=0;i<8;i++){
      int count = 1;
      for(int j=0;j<7;j++){
        if(map[i][j]==map[i][j+1]){
          count++;
          if(count==3){
            flag = true;
            mark[i][j-1] = 1;
            mark[i][j] = 1;
            mark[i][j+1] = 1;
            score += 15;
          }
          else if(count>3){
            mark[i][j+1] = 1;
            score += 5;
          }
        }
        else count = 1;
      }
    }
    //shu
    for(int j=0;j<8;j++){
      int count = 1;
      for(int i=0;i<7;i++){
        if(map[i][j]==map[i+1][j]){
          count++;
          if(count==3){
            flag = true;
            if(mark[i][j]==1)score+=10;
            else score +=15;
            mark[i-1][j] = 3;
            mark[i][j] = 3;
            mark[i+1][j] = 3;
            for(int k=i-5;k>=0;k--){
            	mark[k][j] = 2;
            }
          }
          else if(count>3){
            mark[i+1][j] = count;
            for(int k=1;k<count;k++){
              mark[i-k+1][j]++;
            }
            if(i-2*count+2>=0)mark[i-2*count+2][j] = 0;
            score += 5;
          }
        }
        else count = 1;
      }
    }
    return flag;
  } 

  private boolean check(int i,int j)
  {
    //shu
    int count = 1;
    if(i>=1&&map[i-1][j]==map[i][j]){
      count++;
      if(i>=2&&map[i-2][j]==map[i][j]){
        count++;
      }
    }
    if(count>=3)return true;
    if(i+1<8&&map[i+1][j]==map[i][j]){
      count++;
      if(i+2<8&&map[i+2][j]==map[i][j]){
        count++;
      }
    }
    if(count>=3)return true; 

    //heng
    count = 1;
    if(j>=1&&map[i][j-1]==map[i][j]){
      count++;
      if(j>=2&&map[i][j-2]==map[i][j]){
        count++;
      }
    }
    if(count>=3)return true; 

    if(j+1<8&&map[i][j+1]==map[i][j]){
      count++;
      if(j+2<8&&map[i][j+2]==map[i][j]){
        count++;
      }
    }
    if(count>=3)return true; 

    return false;
  } 

  private void swapMap(int x1,int y1,int x2,int y2)
  {
    int tmp = map[x1][y1];
    map[x1][y1] = map[x2][y2];
    map[x2][y2] = tmp;
  } 

  private boolean check()
  { 

    //heng
    for(int i=0;i<8;i++){
      for(int j=0;j<7;j++){
        swapMap(i,j,i,j+1);
        if(check(i,j)){
          swapMap(i,j,i,j+1);
          return true;
        }
        if(check(i,j+1)){
          swapMap(i,j,i,j+1);
          return true;
        }
        swapMap(i,j,i,j+1);
      }
    }
    //shu
    for(int j=0;j<8;j++){
      for(int i=0;i<7;i++){
        swapMap(i,j,i+1,j);
        if(check(i,j)){
          swapMap(i,j,i+1,j);
          return true;
        }
        if(check(i+1,j)){
          swapMap(i,j,i+1,j);
          return true;
        }
        swapMap(i,j,i+1,j);
      }
    }
    return false;
  } 

  ImageButton.OnClickListener listener = new ImageButton.OnClickListener(){//创建监听对象
    @SuppressLint("NewApi") 

    public void onClick(View v) {
      if(isPoint1){
        p1 = (point)v.getTag();
        isPoint1 = false;
      }
      else {
        p2 = (point)v.getTag();
        isPoint1 = true;
      } 

      if(((p1.x-p2.x==1||p1.x-p2.x==-1)&&p1.y==p2.y)||
        (p1.y-p2.y==1||p1.y-p2.y==-1)&&p1.x==p2.x){
        flag = true;
        swapMap(p1.x,p1.y,p2.x,p2.y);
        swapImage();
        thread.start();
      }
    } 

  };
}

point.java

package com.example.android.market.licensing;
import android.view.View;
public class point {
	public int x;
	public int y;
	View v;
	int id;
	point(int _x,int _y,View _v,int _id){
		x = _x;
		y = _y;
		v = _v;
		id = _id;
	}
	point(int _x,int _y){
		x = _x;
		y = _y;
	}
}

AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>
<!-- Copyright (C) 2010 The Android Open Source Project
   Licensed under the Apache License, Version 2.0 (the "License");
   you may not use this file except in compliance with the License.
   You may obtain a copy of the License at
     http://www.apache.org/licenses/LICENSE-2.0
   Unless required by applicable law or agreed to in writing, software
   distributed under the License is distributed on an "AS IS" BASIS,
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
   See the License for the specific language governing permissions and
   limitations under the License.
-->
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
  package="com.example.android.market.licensing"
  android:versionCode="2"
  android:versionName="1.1">
  <application android:icon="@drawable/icon" android:label="@string/app_name">
    <activity android:name=".MainActivity"
      android:label="@string/app_name"
      android:configChanges="orientation|keyboardHidden">
      <intent-filter>
        <action android:name="android.intent.action.MAIN" />
        <category android:name="android.intent.category.LAUNCHER" />
      </intent-filter>
    </activity>
  </application>
  <!-- Devices >= 3 have version of Android Market that supports licensing. -->
  <uses-sdk android:minSdkVersion="3" />
  <!-- Required permission to check licensing. -->
  <uses-permission android:name="com.android.vending.CHECK_LICENSE" />
</manifest>

以上所述是小编给大家介绍的Android开心消消乐代码实例详解整合,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对我们网站的支持!

(0)

相关推荐

  • Android游戏源码分享之2048

    引言 程序猿们,是否还在为你的老板辛辛苦苦的打工而拿着微薄的薪水呢,还是不知道如何用自己的应用或游戏来赚钱呢! 在这里IQuick将教您如何同过自己的应用来赚取自己的第一桶金! 你是说自己的应用还没有做出来? 不,在這里已经为你提供好了一个完整的游戏应用了,在文章的下面有源码的地址哦.你只要稍做修改就可以变成一个完全属于自己的应用了,比如将4*4换成5*5,甚至是其它的.如果你实在是慵懒至极的话,你只要将本应用的包名及广告换成自己的,就可以上传到市场上轻轻松松赚取自己的第一桶金了. 如果你觉得本

  • Android开发之经典游戏贪吃蛇

    前言 这款游戏实现的思路和源码参考了Google自带的Snake的例子,其中修改了一些个人认为还不够完善的地方,加入了一些新的功能,比如屏幕上的方向操作盘,暂停按钮,开始按钮,退出按钮.另外,为了稍微增加些用户体验,除了游戏的主界面,本人自己新增了5个界面,分别是登陆界面,菜单界面,背景音乐设置界面,难度设置界面,还有个关于游戏的介绍界面.个人觉得在新手阶段,参考现成的思路和实现方式是难以避免的.重要的是我们需要有自己的理解,读懂代码之后,需要思考代码背后的实现逻辑,形成自己的思维.这样在下次开

  • Android游戏开发之碰撞检测(矩形碰撞、圆形碰撞、像素碰撞)

    本文为大家分享了Android游戏开发之碰撞检测,供大家参考,具体内容如下 矩形碰撞 原理: 两个矩形位置 的四种情况 不是这四中情况 则碰撞 圆形碰撞 原理: 利用两个圆心之间的距离进行判定.当两个圆心的距离小于半径之和则碰撞. 像素碰撞 原理:不适用 遍历所有像素 检测 太多了 多矩形碰撞 原理:设置多个矩形碰撞检测区域 检测碰撞矩形数组 与另一碰撞矩形数组之间的位置关系. 矩形碰撞 代码: public class MySurfaceView extends SurfaceView imp

  • Android 游戏开发之Canvas画布的介绍及方法

    Canvas,在英语中,这个单词的意思是帆布.在Android中,则把Canvas当做画布,只要我们借助设置好的画笔(Paint类)就可以在画布上绘制我们想要的任何东西:另外它也是显示位图(Bitmap类)的核心类.随用户的喜好,Canvas还可设置一些关于画布的属性,比如,画布的颜色.尺寸等.Canvas提供了如下一些方法:    Canvas(): 创建一个空的画布,可以使用setBitmap()方法来设置绘制具体的画布.    Canvas(Bitmap bitmap): 以bitmap对

  • Android仿开心消消乐大树星星无限循环效果

    啥都不说先上效果图,这个是我项目里的效果: 下面的是我抽取出来的 demo 适配啥的我基本上都做好了没做其他的 ok 下面 说一下思路把 首先 说一下原理 我是使用bitmap 创建两张图 一开始的时候 一张在下面记为1号 一张在上面 记为2号 当手指向下滑动时 判断 1号 的起点位置 是否完全超出屏幕 如果超出屏幕的话 那么将2号变为下面 1号变为上面 (移动1号的Y坐标) 大体思路 是这样. 里面还有许多判断 比如 是否能向下滑动 起点位置, 星星的判定 哎呀 好烦 说的东西太多啦 来张我的

  • Android五子棋游戏程序完整实例分析

    最近学习了五子棋的课程,感觉挺不错.然后自己写了个关于五子棋的android程序,从中还是能够学习到很多东西的.现在我们开始今天五子棋程序的编写历程.程序的源码请参见友情链接: 好了,我们现在开始一步步的构建出项目来,首先是如下的项目结构图: 运行的效果图: 一些前期做准备的代码 1. 主活动类MainActivity,在菜单中加入了再来一局的功能: public class MainActivity extends AppCompatActivity { private ChessBoardV

  • Android游戏开发实践之人物移动地图的平滑滚动处理

    如图所示为程序效果动画图 地图滚动的原理 在本人之前博客的文章中介绍过人物在屏幕中的移动方式,因为之前拼的游戏地图是完全填充整个手机屏幕的,所以无需处理地图的平滑滚动.这篇文章我着重的向 大家介绍一下控制人物移动后地图滚动的处理方式.举个例子 如上图所示 比如人物向右移动,如果地图贴在屏幕左边边界 将先移动人物在地图的坐标,当人物在屏幕中超过三分之二后 则将地图向人物行走的反方向移动给玩家一种人物还在向右移动的假象,其实这时候人物只是播放向右行走的动画 在屏幕中的坐标不变 ,当地图向人物行走反方

  • 以一个着色游戏展开讲解Android中区域图像填色的方法

    一.着色游戏概述 近期群里偶然看到一哥们在群里聊不规则图像填充什么四联通.八联通什么的,就本身好学务实的态度去查阅了相关资料.对于这类着色的资料,最好的就是去搜索些相关app,根据我的观察呢,不规则图像填充在着色游戏里面应用居多,不过大致可以分为两种: 基于层的的填充 基于边界的填充 那么针对上述两种,我们会通过两篇博文来讲解,本篇就是叙述基于层的填充方式,那么什么基于层的填充方式呢?其实就是一张图实际上是由多个层组成的,每个层显示部分图像(无图像部分为透明),多层叠加后形成一张完整的图案,图层

  • Unity3D游戏引擎实现在Android中打开WebView的实例

    本文讲述了如何在Unity中调用Android中的WebView组件,实现内部浏览器样式的页面切换.首先打开Eclipse创建一个Android的工程: UnityTestActivity.java 入口Activity ,Unity中会调用这个Activity中的方法从而打开网页. package com.xys; import android.content.Context; import android.content.Intent; import android.os.Bundle; i

  • Android高仿2048小游戏实现代码

    刚开始进入Splash界面: 1.SplashActivity.Java(两秒后进入开始界面,Splash界面的布局只有一个图片,在博客后,会展示给大家看) public class SplashActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.a

随机推荐