Android 数独小游戏

原文https://www.zhangman523.cn/401.html

先看看效果图

sudoku-o3.gif

数独设计思路

先看布局,我们可以看到数独由9x9的格子组成,每个格子中间有一个数字。

  • Cell (单个格子、android 中我们可以先用TextView代替)

  • Grid (由3x3Cell组成)

  • Borad (由3x3Grid组成)

数独是由9x9 的格子组成,我们可以分为3个3x3Grid 组成
布局方式用RelativeLayout来布局。

数独检查游戏结束

数独游戏在每一行,每一列 和每个Grid1-9数字不能重复

每次输入时检查是否游戏结束和错误。

代码实现

首先我们使用自定义RelativeLayout来实现 Grid

public class Grid extends RelativeLayout {
    ...
}

我们定义3x3Cell数组

private List<List<TextView>> mTextArrays;

TextView来表示单个格子

初始化Cell

mTextArrays = new ArrayList<>();
for (int i = 0; i < 3; i++) {
    List<TextView> viewList = new ArrayList<>();
    for (int j = 0; j < 3; j++) {
        TextView textView = new TextView(context);
        textView.setWidth(TEXT_SIZE);
        textView.setHeight(TEXT_SIZE);
        textView.setBackgroundColor(Color.WHITE);
        textView.setId(View.generateViewId());
        textView.setGravity(Gravity.CENTER);
        addView(textView);
        viewList.add(textView);
        LayoutParams params = (LayoutParams) textView.getLayoutParams();
        if (j == 0) {
            if (i == 0) {
                params.addRule(RelativeLayout.ALIGN_PARENT_START);
                params.addRule(RelativeLayout.ALIGN_PARENT_LEFT);
            } else if (i == 1) {
                params.addRule(RelativeLayout.BELOW, mTextArrays.get(0).get(0).getId());
                params.topMargin = DensityUtils.dp2px(context,1);
            } else {
                params.addRule(RelativeLayout.BELOW, mTextArrays.get(1).get(0).getId());
                params.topMargin = DensityUtils.dp2px(context,1);
            }
        } else if (j == 1) {
            params.addRule(RelativeLayout.RIGHT_OF, viewList.get(j - 1).getId());
            params.addRule(ALIGN_TOP, viewList.get(j - 1).getId());
            params.leftMargin = DensityUtils.dp2px(context,1);
        } else {
            params.addRule(RelativeLayout.RIGHT_OF, viewList.get(j - 1).getId());
            params.addRule(ALIGN_TOP, viewList.get(j - 1).getId());
            params.leftMargin = DensityUtils.dp2px(context,1);
        }
    }
    mTextArrays.add(viewList);
}

使用自定义RelativeLayout来实现 Board

public class Board extends RelativeLayout{
    ...
}

实现也是跟Grid实现一样,只是Grid作为子View

private List<List<Grid>> mGridArray = new ArrayList<>(); //3x3 的Grid
private List<List<TextView>> mCellArray; //9x9的Cell
private TextView mCurrentCell; //当前选中的Cell

private String mErrorTextColor = "#ff0000";//错误时候的文字颜色
private String mLightTextColor = "#ffffff";//选中时候的文字颜色
private String mDefaultTextColor = "#000000";//默认文字颜色
private String mLightBgColor = "#4fe8fc";//选中的背景颜色
private String mDefaultBgColor = "#ffffff";//默认背景颜色

private String mDisableTextColor = "#e2e2e2";//不可编辑的文字颜色

初始化 mGridArray 的逻辑和Grid 一样就不贴代码了。自己去看>_<!

mCellArray = new ArrayList<>();
for (int i = 0; i < 9; i++) {//初始化Cell Array
    List<TextView> cellArray = new ArrayList<>();
    for (int j = 0; j < 9; j++) {
        int x = i < 3 ? 0 : i < 6 ? 1 : 2; //3x3 的格子
        int y = j < 3 ? 0 : j < 6 ? 1 : 2;
        Grid grid = mGridArray.get(x).get(y);
        List<List<TextView>> gridTextArrays = grid.getTextArrays();
        TextView cell = gridTextArrays.get(i - x * 3).get(j - y * 3);
        cell.setTag(R.id.row, i);//设置tag 信息
        cell.setTag(R.id.column, j);
        cell.setTag(R.id.isLoad, false);
        cell.setTextColor(Color.parseColor(mDefaultTextColor));
        cell.setBackgroundColor(Color.parseColor(mDefaultBgColor));
        cell.setOnClickListener(this);
        cellArray.add(j, cell);
    }
    mCellArray.add(i, cellArray);
}

加载map

数独的map 我们用81位长度的字符串来表示 0表示需要补全的,1-9 为默认的数字

找一个默认的地图

005406000000000201007380000062700090050023804704109060823590010490867020576031948

加载方法

/**
 * load sudoku map
 *
 * @param map map
 */
public void loadMap(String map) {
    if (TextUtils.isEmpty(map)) return;
    for (int i = 0; i < mCellArray.size(); i++) {
        List<TextView> array = mCellArray.get(i);
        for (int j = 0; j < array.size(); j++) {
            TextView cell = array.get(j);//将81位的字符串 转为 9x9 的数组
            String s = map.substring(9 * i + j, 9 * i + j + 1);
            if (!"0".equals(s)) {
                cell.setText(s);
                cell.setTag(R.id.isLoad, true);
                cell.setTextColor(Color.parseColor(mDisableTextColor));
            }
        }
    }
}

点击需要改变背景 高亮行和列。

/**
* Highlight the  grid by row and column
*
* @param row    row
* @param column column
*/
private void lightUpCellByRowAndColumn(int row, int column) {
    boolean lightText = !TextUtils.isEmpty(mCellArray.get(row).get(column).getText().toString());
    //这里做了个判断 如果是有数字 则高亮所有一样的数字 否则高亮选中行和列
    if (lightText) {
        lightSameNumber(row, column, false);
    } else {
        lightRowAndColumn(row, column);
    }
}

遍历Cell数组选中相同的数字

 /**
  * Highlight the same number
  *
  * @param row     row
  * @param column  column
  * @param isError error
  */
private void lightSameNumber(int row, int column, boolean isError) {
    String value = mCellArray.get(row).get(column).getText().toString();
    for (int i = 0; i < 9; i++) {
        for (int j = 0; j < 9; j++) {
            if (value.equals(mCellArray.get(i).get(j).getText().toString())) {
                //change number color
                if (i == row && column == j) {
                    //If it is wrong, change the text color without changing the background.
                    if (isError || mCellArray.get(i).get(j).getCurrentTextColor() == Color.parseColor(mErrorTextColor)) {
                        mCellArray.get(i).get(j).setBackgroundColor(Color.parseColor(mErrorTextColor));
                        mCellArray.get(i).get(j).setTextColor(Color.parseColor(mLightTextColor));
                    } else {
                        mCellArray.get(i).get(j).setBackgroundColor(Color.parseColor(mLightBgColor));
                        mCellArray.get(i).get(j).setTextColor(Color.parseColor(mLightTextColor));
                    }
                } else {
                    if (mCellArray.get(i).get(j).getCurrentTextColor() == Color.parseColor(mErrorTextColor)) {
                        mCellArray.get(i).get(j).setTextColor(Color.parseColor(mErrorTextColor));
                    } else {
                        mCellArray.get(i).get(j).setTextColor(Color.parseColor(mLightTextColor));
                    }
                    mCellArray.get(i).get(j).setBackgroundColor(Color.parseColor(mLightBgColor));
                }
            } else {
                if ((Boolean) mCellArray.get(i).get(j).getTag(R.id.isLoad)) {
                    mCellArray.get(i).get(j).setTextColor(Color.parseColor(mDisableTextColor));
                } else {
                    mCellArray.get(i).get(j).setTextColor(Color.parseColor(mDefaultTextColor));
                }
                mCellArray.get(i).get(j).setBackgroundColor(Color.parseColor(mDefaultBgColor));
            }
        }
    }
}

遍历Cell数组高亮行和列

/**
 * Highlight row and column
 *
 * @param row    row
 * @param column column
 */
private void lightRowAndColumn(int row, int column) {
    for (int i = 0; i < 9; i++) {
        for (int j = 0; j < 9; j++) {
            if (i == row || j == column) {
                if (mCellArray.get(i).get(j).getCurrentTextColor() == Color.parseColor(mErrorTextColor)) {
                    mCellArray.get(i).get(j).setTextColor(Color.parseColor(mErrorTextColor));
                }
                mCellArray.get(i).get(j).setBackgroundColor(Color.parseColor(mLightBgColor));
            } else {
                if ((Boolean) mCellArray.get(i).get(j).getTag(R.id.isLoad)) {
                    mCellArray.get(i).get(j).setTextColor(Color.parseColor(mDisableTextColor));
                } else {
                    if (mCellArray.get(i).get(j).getCurrentTextColor() == Color.parseColor(mErrorTextColor)) {
                        mCellArray.get(i).get(j).setTextColor(Color.parseColor(mErrorTextColor));
                    } else {
                        mCellArray.get(i).get(j).setTextColor(Color.parseColor(mDefaultTextColor));
                    }
                }
                mCellArray.get(i).get(j).setBackgroundColor(Color.parseColor(mLightTextColor));
            }
        }
    }
    mCellArray.get(row).get(column).setBackgroundColor(Color.parseColor(mLightBgColor));
}

输入数字

每次输入我们需要判断游戏是否结束(数字重复或者完成数独)

/**
 * Enter a number from 1-9
 *
 * @param number number
 */
public void inputText(String number) {
    if (mCurrentCell == null) return;
    if (!(Boolean) mCurrentCell.getTag(R.id.isLoad)) {
        mCurrentCell.setText(number);
        boolean gameOver = checkFinish();
        if (gameOver) {
            if (mGameOverCallBack != null) mGameOverCallBack.gameOver();
        }
    }
}

checkFinish()方法中包括检查错误的方法,重复的数字需要高亮。需要检查行、列和宫(3x3的Grid)

/**
 * check game error
 *
 * @param row    row
 * @param column column
 * @return boolean
 */
private boolean checkGameError(int row, int column) {
    boolean result = false;
    result = checkSection(row, column);
    if (result) return result;
    //check row
    for (int i = 0; i < 9; i++) {
        String value = mCellArray.get(i).get(column).getText().toString();
        if (TextUtils.isEmpty(value)) continue;
        for (int j = i; j < 9; j++) {
            if (i == j) continue;
            if (value.equals(mCellArray.get(j).get(column).getText().toString())) {
                Log.d(TAG, String.format("row error,value:%1$s in row:%2$d and column:%3$d", value, row, column));
                result = true;
                break;
            }
        }
    }

    if (result) return result;

    //check column
    for (int i = 0; i < 9; i++) {
        String value = mCellArray.get(row).get(i).getText().toString();
        if (TextUtils.isEmpty(value)) continue;
        for (int j = i; j < 9; j++) {
            if (i == j) continue;
            if (value.equals(mCellArray.get(row).get(j).getText().toString())) {
                Log.d(TAG, String.format("column error,value:%1$s in row:%2$d and column:%3$d", value, row, column));
                result = true;
                break;
            }
        }
    }
    return result;
}

检查3x3 格子中是否有重复的数字

/**
 * check duplicate numbers in the 3x3 grid
 *
 * @param row    row
 * @param column column
 * @return true or false
 */
private boolean checkSection(int row, int column) {
    boolean result = false;
    String value = mCellArray.get(row).get(column).getText().toString();
    if (TextUtils.isEmpty(value)) {
        return result;
    }
    int start_i = row < 3 ? 0 : (row < 6 ? 3 : 6);//3x3 格子的边界
    int start_j = column < 3 ? 0 : (column < 6 ? 3 : 6);
    int end_i = start_i + 3;
    int end_j = start_j + 3;

    for (int i = start_i; i < end_i; i++) {
        for (int j = start_j; j < end_j; j++) {
            if (i == row && j == column) continue;
            if (value.equals(mCellArray.get(i).get(j).getText().toString())) {//如果3x3格子的内容有重复的数字则返回错误
                Log.d(TAG, String.format("section error,value:%1$s in row:%2$d and column:%3$d", value, row, column));
                result = true;
                break;
            }
        }
    }
    return result;
}

游戏的逻辑已经完成了。大家可以去下载代码运行玩玩!

工程已经放在GITHUB

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,254评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,875评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,682评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,896评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,015评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,152评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,208评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,962评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,388评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,700评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,867评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,551评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,186评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,901评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,689评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,757评论 2 351

推荐阅读更多精彩内容

  • 周末公司活动安排在一旅游景点,凑巧也是婚纱摄影基地。 暴晒闷热,本以为是拍婚纱照的淡季,路过一草坪时,竟然数了有二...
    依布茶卡阅读 163评论 0 0
  • 下班时厂车没有了。走到公交站台,7点43分。坐3路车,到鸠江饭店下,用了35分钟,再走回家,又用了25分钟。大宝对...
    一个平凡的人阅读 167评论 0 0