Android图形编程基本概念

  1. 颜色对象

    Color 类

    int color = Color.bule    //蓝色

    int color = Color.argb(255,255,255,255);        //透明度,红,绿,蓝

    在XML文件中定义颜色

  2. 画笔对象

    Paint 类

    Paint.setColor(Color.blue);

  3. 画布对象

    Canvas 类

    Canvas.drawCircle(300, 400, 100, paint);

自定义View的基本实现方法

  1. 定义一个类,继承View
  2. 复写View中的onDraw()方法
  3. 在onDraw()中使用Canvas和Paint绘制图形

    矩形。圆形、空心、实心、线、文字、……

@Override

protected void onDraw(Canvas canvas) {

Paint paint = new Paint();

paint.setARGB(255, 0, 255, 0);

paint.setStyle(Paint.Style.STROKE);// 空心

paint.setStrokeWidth(10);// 边的宽度

canvas.drawRect(100, 100, 500, 500, paint);// 矩形

paint.setStyle(Paint.Style.FILL);// 实心

paint.setColor(Color.CYAN);

canvas.drawCircle(500, 500, 100, paint);// 圆形

paint.setTextSize(100);

paint.setColor(Color.BLUE);

canvas.drawText("Apple ggGG", 50, 800, paint);

paint.setColor(Color.RED);

paint.setStrokeWidth(5);// 边的宽度

canvas.drawLine(0, 800, 720, 800, paint);

canvas.drawBitmap(BitmapFactory.decodeResource(getResources(),

R.drawable.ic_launcher), 200, 200, paint);

super.onDraw(canvas);

}

绘制九宫格

@Override

protected void onDraw(Canvas canvas) {

// 首先取得屏幕的宽度和高度

width = this.getWidth();

cell_width = width / 9f;

height = this.getHeight();

cell_height = height / 9f;

// 首先画出背景

Paint bgPaint = new Paint();// 用于绘制背景

bgPaint.setColor(getResources().getColor(R.color.shudu_background));

canvas.drawRect(0, 0, width, height, bgPaint);

// 画出九宫格

Paint darkPaint = new Paint();// 暗色

darkPaint.setColor(getResources().getColor(R.color.shudu_dark));

Paint lightPaint = new Paint();// 亮色

darkPaint.setColor(getResources().getColor(R.color.shudu_light));

Paint hilitePaint = new Paint();// 线条

darkPaint.setColor(getResources().getColor(R.color.shudu_hilite));

for (int i = 0; i < 9; i++) {

if (i % 3 == 0) {

canvas.drawLine(0, i * cell_height, width, i * cell_height,

lightPaint);

canvas.drawLine(i * cell_width, 0, i * cell_width, height,

lightPaint);

} else {

canvas.drawLine(0, i * cell_height, width, i * cell_height,

darkPaint);

canvas.drawLine(i * cell_width, 0, i * cell_width, height,

darkPaint);

}

canvas.drawLine(0, i * cell_height + 1, width, i * cell_height + 1,

hilitePaint);

canvas.drawLine(i * cell_width + 1, 0, i * cell_width + 1, height,

hilitePaint);

}

// 绘制文字

Paint numPaint = new Paint();

numPaint.setColor(Color.BLUE);

numPaint.setStyle(Paint.Style.STROKE);

numPaint.setTextSize(cell_height * 0.25f);

numPaint.setTextAlign(Paint.Align.CENTER);

float x = cell_width / 2;

float y = cell_height / 2;

for (int i = 0; i < 9; i++) {

for (int j = 0; j < 9; j++) {

canvas.drawText("" + i + "," + j, i * cell_width

+ x, j * cell_height + y, numPaint);

}

}

super.onDraw(canvas);

}

文字居中显示

FontMtrics类

FontMetrics fontMetrics = paint.get FontMetrics();

FontMetrics fm = numPaint.getFontMetrics();

float x = cell_width / 2;

float y = cell_height / 2 - (fm.ascent + fm.descent) / 2;

显示数据

把逻辑处理放在Activity外部,新建一个类

package com.arlen.android.game.shudu03;

public class Game {

private final String str = "450890000000000000008700090607005030090020040040900102070006300000000000000048016";

private int shuduku[] = new int[81];

public Game() {

shuduku = fromPuzzleString(str);

}

private int getTitle(int x, int y) {

return shuduku[y * 9 + x];

}

public String getTitleString(int x, int y) {

int v = getTitle(x, y);

if (v == 0) {

return "";

} else {

return String.valueOf(v);

}

}

protected int[] fromPuzzleString(String src) {

int sudu[] = new int[src.length()];

for (int i = 0; i < sudu.length; i++) {

sudu[i] = src.charAt(i) - '0';

}

return sudu;

}

}

在View中调用

for (int i = 0; i < 9; i++) {

for (int j = 0; j < 9; j++) {

canvas.drawText(game.getTitleString(i, j), i * cell_width + x,

j * cell_height + y, numPaint);

}

}

单点触摸事件

在为初始化数字的空格中点击,然后输入数字

public Boolean onTouchEvent(MotionEvent event){

//获取事件的类型

event.getAction();

//获取点击坐标

Event.getX();

Event.getY();

}

// 生成一个layoutInflater对象

LayoutInflater layoutInflater = LayoutInflater.from(this.getContext());

// 使用layoutInflater对象更具一个布局文件生成一个view对象

View layoutView = layoutInflater.inflate(R.layout.dialog, null);

// 从生成好的textView中取出相应的控件

TextView textView = (TextView) layoutView

.findViewById(R.id.textViewUsedId);

// 设置textView的内容

textView.setText(sb.toString());

// 生成一个对话框的builder对象

AlertDialog.Builder builder = new AlertDialog.Builder(this.getContext());

// 设置地对话框所要显示的内容

builder.setView(layoutView);

// 生成对话框对象并将其显示出来

AlertDialog dialog = builder.create();

dialog.show();

碰撞检测

用户点击屏幕之后,确定用户点击的位置属于哪一个格子

取得用户点击的坐标,判断是否在startX、startY与stopX、stopY之间

@Override

public boolean onTouchEvent(MotionEvent event) {

if (event.getAction() != MotionEvent.ACTION_DOWN) {

return super.onTouchEvent(event);

}

int selectX = (int) (event.getX() / cell_width);

int selectY = (int) (event.getY() / cell_height);

int used[] = game.getUsedTileByCoor(selectY, selectY);

for (int i = 0; i < used.length; i++) {

System.out.println(used[i]);

}

return true;

}

可用数据计算

在每一行、每一列、每一个小的九宫格中已有的数据禁止重复输入,

//用于计算所有单元格对应的不可用的数据

public void calculateAllUsedTitles() {

for (int x = 0; x < 9; x++) {

for (int y = 0; y < 9; y++) {

used[x][y] = calculateUsedTiles(x, y);

}

}

}

//取出某一单元格已经不可用的数据

public int[] getUsedTileByCoor(int x, int y) {

return used[x][y];

}

// 计算某一单元格中已经用过的数据

public int[] calculateUsedTiles(int x, int y) {

int c[] = new int[9];

for (int i = 0; i < 9; i++) {

if (i == y) {

continue;

}

int t = getTitle(x, i);

if (t != 0) {

c[t - 1] = t;

}

}

for (int i = 0; i < 9; i++) {

if (i == x) {

continue;

}

int t = getTitle(i, y);

if (t != 0) {

c[t - 1] = t;

}

}

int startX = (x / 3) * 3;

int startY = (y / 3) * 3;

for (int i = startX; i < startX + 3; i++) {

for (int j = startY; j < startY + 3; j++) {

if (i == x && j == y) {

continue;

}

int t = getTitle(i, y);

if (t != 0) {

c[t - 1] = t;

}

}

}

// compress

int nused = 0;

for (int t : c) {

if (t != 0) {

nused++;

}

}

int c1[] = new int[nused];

nused = 0;

for (int t : c) {

if (t != 0) {

c1[nused++] = t;

}

}

return c1;

}

弹出交互对话框

布局文件

<?xml version="1.0" encoding="utf-8"?>

<TableLayout xmlns:android="http://schemas.android.com/apk/res/android"

android:id="@+id/keypad"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:orientation="vertical" >

<TableRow>

<Button

android:id="@+id/button1"

android:text="1" />

<Button

android:id="@+id/button2"

android:text="2" />

<Button

android:id="@+id/button3"

android:text="3" />

</TableRow>

<TableRow>

<Button

android:id="@+id/button4"

android:text="4" />

<Button

android:id="@+id/button5"

android:text="5" />

<Button

android:id="@+id/button6"

android:text="6" />

</TableRow>

<TableRow>

<Button

android:id="@+id/button7"

android:text="7" />

<Button

android:id="@+id/button8"

android:text="8" />

<Button

android:id="@+id/button9"

android:text="9" />

</TableRow>

</TableLayout>

根据点击的位置不同,弹出的窗口不同

package com.arlen.android.game.shudu03;

import android.app.Dialog;

import android.content.Context;

import android.os.Bundle;

import android.view.View;

public class KeyDialog extends Dialog {

// 用来存放代表对话框中按钮的对象

private final View keys[] = new View[9];

private final int used[];

// 构造函数的第二个参数保存当前单元格已经用过的数据

public KeyDialog(Context context, int[] used) {

super(context);

this.used = used;

}

// 当对话框第一次被调用的时候,会调用其onCreate方法

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

// 标题

this.setTitle("Key Dialog");

// 为该对话框设置布局文件

this.setContentView(R.layout.keypad);

findViews();

// 便利整个used数组

for (int i = 0; i < used.length; i++) {

if (used[i] != 0) {

keys[used[i] - 1].setVisibility(View.INVISIBLE);

}

}

}

private void findViews() {

keys[0] = findViewById(R.id.button1);

keys[1] = findViewById(R.id.button2);

keys[2] = findViewById(R.id.button3);

keys[3] = findViewById(R.id.button4);

keys[4] = findViewById(R.id.button5);

keys[5] = findViewById(R.id.button6);

keys[6] = findViewById(R.id.button7);

keys[7] = findViewById(R.id.button8);

keys[8] = findViewById(R.id.button9);

}

}

Android 数独游戏 记录的更多相关文章

  1. android数独游戏

    最近没事干,照着视频教程写了一个数独游戏,很粗糙还有很多要修改的地方.下面就来说说这个游戏吧 1.自定义一个View控件,用来在屏幕上显示一个9*9的格子,其实就是横着画8条线,竖着画8跳线,然后将其 ...

  2. android开发——数独游戏

    最近研究了一下android,写了一个数独游戏,具体如下: 游戏界面需要重写一个ShuduView继承View, 然后自定义一个Dialog: 1.需要继承 Dialog 类, 2.并要定义一个有参构 ...

  3. iOS开发 Swift开发数独游戏(一)

    一.前言 我姥姥是一名退休数学老师,一直很喜欢玩数独游戏.我以前答应过她要给她写一个数独游戏.本来计划是写一个Android应用的,但恰好我学了好长时间iOS开发一直没做什么"大项目&quo ...

  4. 用html5 canvas和JS写个数独游戏

    为啥要写这个游戏? 因为我儿子二年级数字下册最后一章讲到了数独.他想玩儿. 因为我也想玩有提示功能的数独. 因为我也正想决定要把HTML5和JS搞搞熟.熟悉一个编程平台,最好的办法,就是了解其原理与思 ...

  5. Scrum4.0+5.0 数独游戏

    1.题目: 1.准备看板. 形式参考图4. 2.任务认领,并把认领人标注在看板上的任务标签上. 先由个人主动领任务,PM根据具体情况进行任务的平衡. 然后每个人都着手实现自己的任务. 3.为了团队合作 ...

  6. C语言学习 数独游戏

    摘要:花了1周多时间学习了C语言,开始练手写解数独游戏的程序. C语言学习 数独游戏 作者:乌龙哈里 时间:2015-11-22 平台:Window7 64bit,TCC 0.9.26(x86-64 ...

  7. Android拼图游戏的设计逻辑,从切图到交互动画,从关卡到倒计时,实例提高!

    Android拼图游戏的设计逻辑,从切图到交互动画,从关卡到倒计时,实例提高! 群英传的最后一章,我大致的看了一下这个例子,发现鸿洋大神也做过,就参考两个人的设计逻辑,感觉都差不多,就这样实现起来了 ...

  8. 【DFS】数独游戏

    DFS(深度优先搜索): 深度优先搜索算法(英语:Depth-First-Search,简称DFS)是一种用于遍历或搜索树或图的算法. 沿着树的深度遍历树的节点,尽可能深的搜索树的分支.当节点v的所在 ...

  9. 创建随机的9x9数独游戏终盘并打印

    创建随机的9x9数独游戏终盘并打印 项目github地址 1. 项目相关要求 1.1 要求 利用程序随机构造出N个已解答的9x9数独棋盘 . 输入 数独棋盘题目个数N(0<N<=10000 ...

随机推荐

  1. Python字符串的encode与decode研究心得 乱码问题解决方法

    以下摘自:http://www.jb51.net/article/17560.htm 为什么Python使用过程中会出现各式各样的乱码问题,明明是中文字符却显示成“\xe4\xb8\xad\xe6\x ...

  2. centos 安装jdk

    不要使用yum 安装openjdk,他妈的就是一坑货 首先到官网下载jdk,http://www.oracle.com/technetwork/java/javase/downloads/jdk7-d ...

  3. 常用Linux命令小结

    常用Linux命令小结 Linux下有很多常用的很有用的命令,这种命令用的多了就熟了,对于我来说,如果长时间没有用的话,就容易忘记.当然,可以到时候用man命令查看帮助,但是,到时候查找的话未免有些临 ...

  4. Html 全屏切换效果

    来源 http://www.imooc.com/learn/374 pageswitch.js (function ($) { var defaults = { 'container': '#cont ...

  5. Android内存管理(5)*官方教程:Logcat内存日志各字段含义,查看当前内存快照,跟踪记录内存分配,用adb查看内存情况时各行列的含义,捕获内存快照的3种方法,如何让程序暴漏内存泄漏的方法

    Investigating Your RAM Usage In this document Interpreting Log Messages                 内存分析日志中各消息的含 ...

  6. shell 块注释

    方法一 : ' 被注释的多行内容 ' 方法二 :<<eof 被注释的多行内容 eof 方法三 :<<! 被注释的多行内容 ! 方法四 if false ; then 被注释的多 ...

  7. [HDOJ4325]Flowers(树状数组 离散化)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4325 关于离散化的简介:http://blog.csdn.net/gokou_ruri/article ...

  8. leetcode:Isomorphic Strings

    Isomorphic Strings Given two strings s and t, determine if they are isomorphic. Two strings are isom ...

  9. jpa+spring配置多数据源

    property name="hibernate.connection.url" value="jdbc:mysql://localhost:3306/test?useU ...

  10. HDU 5360 Hiking 登山 (优先队列,排序)

    题意: 有n个人可供邀请去hiking,但是他们很有个性,每个人都有个预期的人数上下限[Li,Ri],只有当前确定会去的人数在这个区间内他才肯去.一旦他答应了,无论人数怎样变更,他都不会反悔.问最多能 ...