HTML5的Canvas画图模拟太阳系运转

有一个需求是:在一个图片按钮上点击,在按钮的上方弹出一个弹框,根据弹框的内容页面做不同的显示。这个其实没什么难的,主要是要控制好弹框的显示位置,让弹框显示在图片的正上方的中间。

一开始是用的Popupwindow,但是Popupwindow不能给弹窗之外的页面加一个半透明的蒙层,当然可以在页面上加一个专门的作为蒙层的View,但是很显然,这么做会代码变得很恶心,于是又换成了Dialog,因为Dialog弹出的时候会自动加一个蒙层的,但是这个时候,弹框显示位置的Y坐标不对了,后来一顿查,原来Dialog默认是带有title的,只要把title去掉就可以了,看代码:

activity_main.xml:

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <ImageView
        android:id="@+id/img1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerInParent="true"
        android:src="@drawable/img" />

    <ImageView
        android:id="@+id/img2"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_toRightOf="@id/img1"
        android:layout_marginLeft="50dp"
        android:layout_alignTop="@id/img1"
        android:src="@drawable/img" />

</RelativeLayout>

MainActivity.java:

public class MainActivity extends Activity {

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);

		final ImageView img1 = (ImageView)this.findViewById(R.id.img1);
		img1.setOnClickListener(new View.OnClickListener() {
			@Override
			public void onClick(View v) {
				usePopup(img1);
			}
		});

		final ImageView img2 = (ImageView)this.findViewById(R.id.img2);
		img2.setOnClickListener(new View.OnClickListener() {
			@Override
			public void onClick(View v) {
				useDialog(img2);
			}
		});
	}
}

menu.xml

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

    <RelativeLayout
        android:id="@+id/menu_layout"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerHorizontal="true">

	    <TextView
	        android:id="@+id/menu_camera"
	        android:layout_width="100dp"
	        android:layout_height="50dp"
	        android:gravity="center"
	        android:clickable="true"
	        android:text="相机拍照"
	        android:textSize="16sp"
	        android:textColor="@android:color/black"
	        android:background="@android:color/white"/>

	    <View
	        android:id="@+id/menu_sep"
	        android:layout_width="100dp"
	        android:layout_height="1dp"
	        android:background="@android:color/black"
	        android:layout_below="@id/menu_camera"/>

	    <TextView
	        android:id="@+id/menu_album"
	        android:layout_width="100dp"
	        android:layout_height="50dp"
	        android:layout_below="@id/menu_sep"
	        android:gravity="center"
	        android:clickable="true"
	        android:text="选取图片"
	        android:textSize="16sp"
	        android:textColor="@android:color/black"
	        android:background="@android:color/white"/>
	</RelativeLayout>

    <ImageView
            android:id="@+id/arrow_up"
            android:layout_width="wrap_content"
        	android:layout_height="wrap_content"
        	android:layout_below="@id/menu_layout"
        	android:layout_marginTop="0dp"
        	android:layout_centerHorizontal="true"
        	android:src="@drawable/arrow_up"/>

</RelativeLayout>
MainActivity.java:
private void usePopup(final ImageView anchor){
		//参考: http://www.cnblogs.com/sw926/p/3230659.html
		LayoutInflater mInflater = LayoutInflater.from(this);
	 	ViewGroup rootView = (ViewGroup)mInflater.inflate(R.layout.menu, null);
	 	rootView.setLayoutParams(new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT));
		final PopupWindow popup = new PopupWindow(this);
		//setContentView之前一定要设置宽高,否则不显示
		popup.setWidth(WindowManager.LayoutParams.WRAP_CONTENT);
		popup.setHeight(WindowManager.LayoutParams.WRAP_CONTENT);
		//去掉默认的背景
		popup.setBackgroundDrawable(new ColorDrawable(android.R.color.transparent));
		popup.setContentView(rootView);
		//点击空白处的时候PopupWindow会消失
		popup.setTouchable(true);
		popup.setOutsideTouchable(true);
		//如果focusable为false,在一个Activity弹出一个PopupWindow,按返回键,由于PopupWindow没有焦点,会直接退出Activity。如果focusable为true,PopupWindow弹出后,所有的触屏和物理按键都有PopupWindows处理。
		popup.setFocusable(true);
		//计算弹框位置
		int[] xy = calcPopupXY(rootView,anchor);
		//不用任何gravity,使用绝对的(x,y)坐标
		popup.showAtLocation((View)anchor.getParent(),Gravity.NO_GRAVITY, xy[0], xy[1]);
	}

	private void useDialog(final ImageView anchor){
		LayoutInflater mInflater = LayoutInflater.from(this);
	 	ViewGroup rootView = (ViewGroup)mInflater.inflate(R.layout.menu, null);
	 	rootView.setLayoutParams(new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT));
		Dialog dialog = new Dialog(this);
		WindowManager.LayoutParams params = dialog.getWindow().getAttributes();
		params.width = WindowManager.LayoutParams.WRAP_CONTENT;
		params.height = WindowManager.LayoutParams.WRAP_CONTENT;
		//去掉默认的背景,下面两个都可以
		dialog.getWindow().setBackgroundDrawable(new ColorDrawable(android.R.color.transparent));
		//dialog.getWindow().setBackgroundDrawableResource(android.R.color.transparent);
		//http://stackoverflow.com/questions/12348405/dialog-is-bigger-than-expected-when-using-relativelayout
		//dialog默认都是有title的
		dialog.requestWindowFeature(Window.FEATURE_NO_TITLE);//去掉标题,否则会影响高度计算,一定要在setContentView之前调用,终于明白有一个设置theme的构造函数的目的了
		dialog.setContentView(rootView);

		//计算弹框位置
		int[] xy = calcPopupXY(rootView,anchor);
		//gravity的默认值为Gravity.CENTER,即Gravity.CENTER_HORIZONTAL | Gravity.CENTER_VERTICAL.
		//参考: http://www.cnblogs.com/angeldevil/archive/2012/03/31/2426242.html
		dialog.getWindow().setGravity(Gravity.LEFT | Gravity.TOP);
		params.x = xy[0];
		params.y = xy[1];

		dialog.show();
	}
	//参考:http://blog.csdn.net/johnny901114/article/details/7839512
	private int[] calcPopupXY(View rootView, View anchor){
		int w = View.MeasureSpec.makeMeasureSpec(0,View.MeasureSpec.UNSPECIFIED);
    	int h = View.MeasureSpec.makeMeasureSpec(0,View.MeasureSpec.UNSPECIFIED);
    	rootView.measure(w, h);
    	int popupWidth = rootView.getMeasuredWidth();
    	int popupHeight = rootView.getMeasuredHeight();
    	Rect anchorRect = getViewAbsoluteLocation(anchor);
		int x = anchorRect.left + (anchorRect.right - anchorRect.left)/2 - popupWidth / 2;
		int y = anchorRect.top - popupHeight;
		return new int[]{x,y};
	}

    public static Rect getViewAbsoluteLocation(View view){
    	if(view == null){
    		return new Rect();
    	}
		// 获取View相对于屏幕的坐标
		int[] location = new int[2] ;
		view.getLocationOnScreen(location);//这是获取相对于屏幕的绝对坐标,而view.getLocationInWindow(location); 是获取window上的相对坐标,本例中只有一个window,二者等价
		// 获取View的宽高
		int width = view.getMeasuredWidth();
		int height = view.getMeasuredHeight();
		// 获取View的Rect
		Rect rect = new Rect();
		rect.left = location[0];
		rect.top = location[1];
		rect.right = rect.left + width;
		rect.bottom = rect.top + height;
		return rect;
	}

源码在:http://download.csdn.net/download/goldenfish1919/7291951

总结一下:

(1)Popupwindow在显示之前一定要设置宽高,Dialog无此限制。

(2)Popupwindow默认不会响应物理键盘的back,除非显示设置了popup.setFocusable(true);而在点击back的时候,Dialog会消失。

(3)Popupwindow不会给页面其他的部分添加蒙层,而Dialog会。

(4)Popupwindow没有标题,Dialog默认有标题,可以通过dialog.requestWindowFeature(Window.FEATURE_NO_TITLE);取消标题

(5)二者显示的时候都要设置Gravity。如果不设置,Dialog默认是Gravity.CENTER。

(6)二者都有默认的背景,都可以通过setBackgroundDrawable(new ColorDrawable(android.R.color.transparent));去掉。

HTML5的Canvas画图模拟太阳系运转

时间: 2024-08-02 23:08:29

HTML5的Canvas画图模拟太阳系运转的相关文章

html5之canvas画图

导航 前言 基本知识 绘制矩形 清除矩形区域 圆弧 路径 绘制线段 绘制贝塞尔曲线 线性渐变 径向渐变(发散) 图形变形(平移.旋转.缩放) 矩阵变换(图形变形的机制) 图形组合 给图形绘制阴影 绘制图像(图片平铺.裁剪.像素处理[不只图像.包括其他绘制图形]) 绘制文字 保存和恢复状态(context) 保存文件 结合setInterval制作动画 结语.demo下载   前言 <canvas></canvas>是html5出现的新标签,像所有的dom对象一样它有自己本身的属性.

HTML5使用canvas画图时,图片被自动放大模糊的问题

最近在研究canvas技术,发现一个问题,就是所画图像会随着画布大小自动变换大小.原因如下 <canvas id="cxt" style="width: 500px; height: 300px; background: #abcdef;"></canvas> 使用了style属性 或者在页面中引入内嵌样式 <style> #cxt{ width: 500px; height: 300px; } </style> 后来

html5之canvas画图 1.写字板功能

 写字板事例:       写字板分析:1.点击鼠标開始写字(onmosedown)2.按下鼠标写字(onmousemove)3.松开鼠标,停下写字(撤销onmousemove事件):       代码: <strong><!doctype html> </strong><html> <head> <meta charset="utf-8"> <title>Canvas</title> &l

html5学习 - canvas画图和清除图片

在canvas上画一张图其实很简单,就是用drawImgage函数. 定义 这里先贴上w3c里的定义和用法: JavaScript 语法 1 (在画布上定位图像:) context.drawImage(img,x,y); JavaScript 语法 2 (在画布上定位图像,并规定图像的宽度和高度:) context.drawImage(img,x,y,width,height); JavaScript 语法 3 (剪切图像,并在画布上定位被剪切的部分:) context.drawImage(im

html5 Canvas画图3:1px线条模糊问题

点击查看原文地址: html5 Canvas画图3:1px线条模糊问题 本文属于<html5 Canvas画图系列教程> 接上一篇canvas画线条教程 上次我们讲到,canvas有时候会出现1像素的线条模糊不清且好像更宽的情况,如下图: 这样的线条显然不是我们想要的. 这篇文章的目的就是弄清楚里面的原理,以及解决它. 大家都知道屏幕上最小的显示尺寸就是1像素,虽然小于1像素的东西可能显示不出来,但计算机可不管,他会试着画一下. 其实像素终究来说也是一个单位,假如我们把画布放大到足够大,足以看

[转]html5 Canvas画图教程(1)—画图的基本常识

今天看到一个讲Canvas的教程,很通俗移动,所以转载了下. 虽然大家都称Canvas为html5的新标签,看起来好像Canvas属于html语言的新知识,但其实Canvas画图是通过javascript来做的.所以,如果你想学习Canvas画图,你必须要有Javascript基础. 另外,画图嘛,总有一些图像方面的术语和知识点,所以如果你有过做图或美工经验,学习Canvas会更容易. Canvas,意为画布也.而Html5中的Canvas也真的跟现实生活中的画布非常相似.所以,把他看成一块实实

[转]html5 Canvas画图教程(6)—canvas里画曲线之arcTo方法

arc与arcTo,从名字都能看出来相似.arcTo也是画曲线的方法,而且他画出的曲线也是正圆的一段弧线.但他的参数和arc简直是不共戴天~ ctx.arcTo(x1,y1,x2,y2,radius);arcTo的参数中包括两个点,而且这两个点中并没有表示圆心的点,仅仅最后的参数是圆的半径,表示arcTo和圆有那么点关系. 网上关于arcTo的文章很少,好不容易找到一篇还是外国的:而且canvas画图木有直观工具,只能靠猜,arcTo害我猜了半天.. 为了直观的描述,我采取了一种辅助办法:arc

[转]html5 Canvas画图教程(8)—canvas里画曲线之bezierCurveTo方法

前面的文章我已经讲了3种在canvas中画曲线的方法:arc,arcTo以及quadraticCurveTo.他们都有一个共同点,就是他们画的曲线都只能偏向一边,今天讲的bezierCurveTo与他们最大的不同点就是有两个控制点,即可以画出S形的曲线了. bezierCurveTo,也就是所谓的贝赛尔曲线了,如果你学过某些画图工具,就能马上理解. bezierCurveTo的语法如下: ctx.bezierCurveTo(x1,y1,x2,y2,x,y);他的参数我照例解释一下,其中的(x1,

毕业设计总结(1)-canvas画图

去年6月底完成的毕业设计,到现在也才开始给它做个总结,里面有很多可以学习和借鉴的东西. 我的毕业设计的题目是“一种路径规划算法的改进与设计”,具体的要求可参见下面的表格: 题目 一种路径规划算法的改进与设计 类型 毕业设计 性质 工程设计 来源 结合科研 限选人数 1 已选人数 1 毕业设计的任务和要求 在一个布满障碍物的地图上,过凸极值点划分区域:在相应的区域中抽象出一个点来对应各区域,画出连通无向图:根据对应的权值找出最优路径:写出相应的算法. 毕业设计的具体工作内容 1.学习课题相关理论知