Android游戏之数独游戏开发

 更新时间:2022年03月24日 12:10:26   作者:哈丨吅  
这篇文章主要为大家详细介绍了Android游戏之数独游戏开发的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

 数独游戏是一种源自18世纪末的瑞士的游戏,后在美国发展、并在日本得以发扬光大的数学智力拼图游戏。在每一个小九宫格中,分别填上1至9的数字,让整个大九宫格每一列、每一行的数字都不重复。 数独的玩法逻辑简单,数字排列方式千变万化,是锻炼脑筋的好方法。

本文实现简单的数独游戏,通过mars的视频教程完成编程

1.自定义View:

package com.example.administrator.shudugame;

/**
 * Created by Administrator on 2016/9/19.
 */

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Paint.Align;
import android.graphics.Paint.FontMetrics;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;

public class MyView extends View{

  public MyView(Context context, AttributeSet attrs) {
    super(context, attrs);
  }
  //方格长宽
  float width;
  float height;
  //选定的坐标
  int selectX;
  int selectY;
  MyGame mGame=new MyGame();

  //获得屏幕尺寸
  @Override
  protected void onSizeChanged(int w, int h, int oldw, int oldh) {
    super.onSizeChanged(w, h, oldw, oldh);
    //每一个小格的长宽
    this.width=w/9f;
    this.height=h/9f*2/3;
  }


  //绘图函数
  @Override
  protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    //画背景
    Paint bgPaint=new Paint();
    bgPaint.setColor(getResources().getColor(R.color.bg));
    canvas.drawRect(0,0,getWidth(),getHeight()*2/3f, bgPaint);
    //线条画笔
    Paint darkPaint =new Paint();
    darkPaint.setColor(getResources().getColor(R.color.dark));
    Paint hilitePaint =new Paint();
    hilitePaint.setColor(getResources().getColor(R.color.hilite));
    Paint lightPaint =new Paint();
    lightPaint.setColor(getResources().getColor(R.color.light));
    //绘制线条
    for(int i=0;i<9;i++){
      canvas.drawLine(0, i*height,getWidth(),i*height,lightPaint);
      canvas.drawLine(0, i*height+1,getWidth(),i*height+1,hilitePaint);
      canvas.drawLine(i*width, 0,i*width,getWidth()+width*0.4f,lightPaint);
      canvas.drawLine(i*width+1,0,i*width+1,getWidth()+width*0.4f,hilitePaint);
      if(i%3==0){
        canvas.drawLine(0, i*height,getWidth(),i*height,darkPaint);
        canvas.drawLine(0, i*height+3,getWidth(),i*height+3,hilitePaint);
        canvas.drawLine(i*width, 0,i*width,getWidth()+width*0.4f,darkPaint);
        canvas.drawLine(i*width+3, 0,i*width+3,getWidth()+width*0.4f,hilitePaint);
      }
    }
    //绘制数字
    Paint numberPaint =new Paint();
    numberPaint.setColor(Color.BLACK);
    numberPaint.setStyle(Paint.Style.STROKE);
    numberPaint.setTextSize(height*0.75f);
    numberPaint.setTextAlign(Align.CENTER);

    //调节文字居中
    FontMetrics fMetrics=numberPaint.getFontMetrics();
    float x=width/2;
    float y=height/2-(fMetrics.ascent+fMetrics.descent)/2;
    for(int i=0;i<9;i++)
    {
      for(int j=0;j<9;j++)
      {
        canvas.drawText(mGame.getNumber(i+1, j+1), i*width+x, y+j*height, numberPaint);
      }
    }

  }
  //触摸事件
  @Override
  public boolean onTouchEvent(MotionEvent event) {
    int x=(int)(event.getX()/width);
    int y=(int)(event.getY()/height);
    if(x<=8 && y<=8){ //判断点击的是否是游戏界面
      int []t=mGame.getUsed(x, y);
      selectX=x;
      selectY=y;
      MyDialog mDialog=new MyDialog(getContext(), t, this); //调用自定义Dialog
      mDialog.show();
    }
    return super.onTouchEvent(event);
  }
  public void setTitle(int i){
    mGame.setTitle(i,selectX,selectY);
    invalidate(); //每次填写一个数 都要重新进行绘制
    mGame.youWin();
  }
}

2.新建MyGame类:

package com.example.administrator.shudugame;

import android.content.Context;
import android.content.DialogInterface;
import android.support.v7.app.AlertDialog;

/**
 * Created by Administrator on 2016/9/19.
 */
public class MyGame {

  Context context;
  String data="008309100" +
      "900060004" +
      "007504800" +
      "036000540" +
      "001000600" +
      "042000970" +
      "005907300" +
      "600010008" +
      "004608200";
  int numbers[][] =new int[9][9];
  public MyGame(){
    //初始化data
    for(int i=0;i<9;i++)
      for(int j=0;j<9;j++)
      {
        numbers[i][j]=data.charAt(i+j*9)-'0'; //字符转化为整形
      }
  }
  //得到值
  public String getNumber(int x,int y){
    if(numbers[x-1][y-1]==0)
      return "";
    else
      return ""+numbers[x-1][y-1];
  }
  //算出已经被用的数字
  public int[] getUsed(int x,int y){
    int c[]=new int[9];
    //x列
    for(int i=0;i<9;i++)
    {
      if(numbers[x][i]!=0)
      {
        c[numbers[x][i]-1]=numbers[x][i];
      }
    }
    //y排
    for(int i=0;i<9;i++)
    {
      if(numbers[i][y]!=0)
      {
        c[numbers[i][y]-1]=numbers[i][y];
      }
    }
    //小九宫格
    x=(x/3)*3;
    y=(y/3)*3;
    for(int i=0;i<9;i++)
    {
      if(numbers[x+i%3][y+i/3]!=0)
      {
        c[numbers[x+i%3][y+i/3]-1]=numbers[x+i%3][y+i/3];
      }
    }
    return c;
  }
  //设置选定的数字
  public void setTitle(int i,int x,int y){
    numbers[x][y]=i;
  }
  //判断游戏结束
  public void youWin(){
    int t=0;
    for (int i=0;i<9;i++){
      for (int j=0;j<9;j++){
        if (numbers[i][j]!=0){
           t++;
        }
      }
    }
    if (t==81){
      AlertDialog.Builder builder=new AlertDialog.Builder(context);
      builder.setTitle("Tips")
          .setMessage("You Win !")
          .setPositiveButton("Restart", new DialogInterface.OnClickListener() {
            @Override
            public void onClick(DialogInterface dialog, int which) {

            }
          })
          .setNegativeButton("Exit", new DialogInterface.OnClickListener() {
            @Override
            public void onClick(DialogInterface dialog, int which) {
              System.exit(0);
            }
          }).show();
    }
  }
}

3.自定义Dialog类:

package com.example.administrator.shudugame;

/**
 * Created by Administrator on 2016/9/19.
 */
import android.app.Dialog;
import android.content.Context;
import android.os.Bundle;
import android.view.View;

public class MyDialog extends Dialog{

  MyView mView;
  View keys[]=new View[9];
  int []used=new int[9];

  //传入已经使用过的数字
  public MyDialog(Context context, int[] useed, MyView m) {
    super(context);
    this.mView=m;
    this.used=useed;

  }
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    //设置标题
    setTitle("可选数字");
    //设置布局文件
    setContentView(R.layout.table);
    int id[]=new int[]{R.id.bt1,R.id.bt2,R.id.bt3,R.id.bt4,
        R.id.bt5,R.id.bt6,R.id.bt7,R.id.bt8,R.id.bt9};
    for(int i=0;i<9;i++)
    {
      final int t=i+1;
      keys[i]=findViewById(id[i]);
      keys[i].setOnClickListener(new View.OnClickListener() {

        public void onClick(View v) {
          mView.setTitle(t);
          dismiss(); //取消对话框
        }
      });
    }
    for(int i=0;i<9;i++)
    {
      if(used[i]!=0)
        keys[used[i]-1].setVisibility(View.INVISIBLE); //将不可使用的数字对应的Button设置为隐藏
    }
  }
}

4.在MainActivity中,在onCreate()方法: 更改:setContentView(R.layout.activity_main); 为 setContentView(new MyView(this,attrs));

5.新建table.xml:

<?xml version="1.0" encoding="utf-8"?>
<TableLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:orientation="vertical"
  android:layout_width="wrap_content"
  android:layout_height="wrap_content"
  android:stretchColumns="*" >

  <TableRow
    android:id="@+id/row1"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content" >



    <Button
      android:id="@+id/bt1"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:text="1" />


    <Button
      android:id="@+id/bt2"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:text="2" />


    <Button
      android:id="@+id/bt3"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:text="3" />

  </TableRow>


  <TableRow
    android:id="@+id/row2"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content" >


    <Button
      android:id="@+id/bt4"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:text="4" />


    <Button
      android:id="@+id/bt5"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:text="5" />


    <Button
      android:id="@+id/bt6"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:text="6" />

  </TableRow>


  <TableRow
    android:id="@+id/row3"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content" >


    <Button
      android:id="@+id/bt7"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:text="7" />


    <Button
      android:id="@+id/bt8"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:text="8" />


    <Button
      android:id="@+id/bt9"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:text="9" />

  </TableRow>

</TableLayout>

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • Flutter Future异步操作详细讲解

    Flutter Future异步操作详细讲解

    这篇文章主要介绍了Flutter Future异步操作,future是Future类的对象,其表示一个T类型的异步操作结果。如果异步操作不需要结果,则future的类型可为Future
    2023-03-03
  • 简单实现Android闹钟功能

    简单实现Android闹钟功能

    这篇文章主要教大家如何简单实现Android闹钟功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-01-01
  • 一文讲解Kotlin中的contract到底有什么用

    一文讲解Kotlin中的contract到底有什么用

    我们在开发中肯定会经常用Kotlin提供的一些通用拓展函数,当我们进去看源码的时候会发现许多函数里面有contract{}包裹的代码块,那么这些代码块到底有什么作用呢?下面这篇文章主要给大家介绍了关于Kotlin中contract到底有什么用的相关资料,需要的朋友可以参考下
    2022-01-01
  • Flutter状态管理Bloc之登录示例

    Flutter状态管理Bloc之登录示例

    这篇文章主要为大家详细介绍了Flutter状态管理Bloc之登录示例,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-03-03
  • Android RxJava创建操作符Interval

    Android RxJava创建操作符Interval

    这篇文章主要为大家详细介绍了Android RxJava创建操作符Interval的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-12-12
  • Android自定义控件仿QQ抽屉效果

    Android自定义控件仿QQ抽屉效果

    这篇文章主要为大家详细介绍了Android自定义控件仿QQ抽屉效果的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-03-03
  • 基于android中权限的集合汇总

    基于android中权限的集合汇总

    本篇文章把android中的权限进行了集合汇总。需要的朋友参考下
    2013-05-05
  • 浅析Android手机卫士之手机实现短信指令获取位置

    浅析Android手机卫士之手机实现短信指令获取位置

    这篇文章主要介绍了浅析Android手机卫士之手机实现短信指令获取位置的相关资料,需要的朋友可以参考下
    2016-04-04
  • Android实现自适应屏幕的弹窗广告

    Android实现自适应屏幕的弹窗广告

    这篇文章主要为大家详细介绍了Android实现自适应屏幕的弹窗广告,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-07-07
  • Android通过aapt命令获取apk详细信息(包括:文件包名,版本号,SDK等信息)

    Android通过aapt命令获取apk详细信息(包括:文件包名,版本号,SDK等信息)

    本文给大家分享android通过aapt命令获取apk详细信息(包括:文件包名,版本号,SDK等信息),非常不错,简单实用,对android sdk aapt知识感兴趣的朋友一起通过本文学习吧
    2016-11-11

最新评论