/*
* Copyright (C) 2012 www.amsoft.cn
*
* 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.
*/
package com.ab.view.app;
import java.util.List;
import java.util.TimeZone;
import com.ab.util.AbStrUtil;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.graphics.Canvas;
import android.graphics.drawable.Drawable;
import android.os.Handler;
import android.text.format.DateUtils;
import android.text.format.Time;
import android.view.View;
// TODO: Auto-generated Javadoc
/**
* © 2012 amsoft.cn
* 名称:AbNumberClock.java
* 描述:自定义数字时钟的view
*
* @author 还如一梦中
* @version v1.0
* @date:2013-11-6 上午10:19:42
*/
public class AbNumberClock extends View {
/** The m calendar. */
private Time mCalendar;
/** The m time bg. */
private Drawable mTimeBg;
/** The m time colon. */
private Drawable mTimeColon;
/** The d time bmp. */
private List<Drawable> dTimeBmp;
/** The d apm bmp. */
private List<Drawable> dApmBmp;
/** The m time bg width. */
private int mTimeBgWidth;
/** The m time bg height. */
private int mTimeBgHeight;
/** The m attached. */
private boolean mAttached;
/** The m handler. */
private final Handler mHandler = new Handler();
/** The m minutes. */
private String mMinutes;
/** The m hour. */
private String mHour;
/** The m second. */
private String mSecond;
/**
* Instantiates a new ab number clock.
*
* @param context the context
* @param timeBg the time bg
* @param timeColon the time colon
* @param timeBmp the time bmp
* @param apmBmp the apm bmp
*/
public AbNumberClock(Context context, Drawable timeBg, Drawable timeColon,
List<Drawable> timeBmp, List<Drawable> apmBmp) {
super(context);
mTimeBg = timeBg;
mTimeColon = timeColon;
dTimeBmp = timeBmp;
dApmBmp = apmBmp;
mCalendar = new Time();
if (!dApmBmp.isEmpty() && dApmBmp.size() > 0) {
mTimeBgWidth = 2 * mTimeColon.getIntrinsicWidth() + 6
* dTimeBmp.get(0).getIntrinsicWidth()
+ dApmBmp.get(0).getIntrinsicWidth();
mTimeBgHeight = dTimeBmp.get(0).getIntrinsicHeight();
} else {
mTimeBgWidth = 2 * mTimeColon.getIntrinsicWidth() + 8
* dTimeBmp.get(0).getIntrinsicWidth();
mTimeBgHeight = dTimeBmp.get(0).getIntrinsicHeight();
}
}
/* (non-Javadoc)
* @see android.view.View#onAttachedToWindow()
*/
@Override
protected void onAttachedToWindow() {
super.onAttachedToWindow();
if (!mAttached) {
mAttached = true;
IntentFilter filter = new IntentFilter();
filter.addAction(Intent.ACTION_TIME_TICK);
filter.addAction(Intent.ACTION_TIME_CHANGED);
filter.addAction(Intent.ACTION_TIMEZONE_CHANGED);
getContext().registerReceiver(mIntentReceiver, filter, null,mHandler);
}
// NOTE: It's safe to do these after registering the receiver since the
// receiver always runs
// in the main thread, therefore the receiver can't run before this
// method returns.
// The time zone may have changed while the receiver wasn't registered,
// so update the Time
mCalendar = new Time();
// Make sure we update to the current time
onTimeChanged();
}
/* (non-Javadoc)
* @see android.view.View#onDetachedFromWindow()
*/
@Override
protected void onDetachedFromWindow() {
super.onDetachedFromWindow();
if (mAttached) {
getContext().unregisterReceiver(mIntentReceiver);
mAttached = false;
}
}
/* (non-Javadoc)
* @see android.view.View#onMeasure(int, int)
*/
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
int widthMode = MeasureSpec.getMode(widthMeasureSpec);
int widthSize = MeasureSpec.getSize(widthMeasureSpec);
int heightMode = MeasureSpec.getMode(heightMeasureSpec);
int heightSize = MeasureSpec.getSize(heightMeasureSpec);
float hScale = 1.0f;
float vScale = 1.0f;
if (widthMode != MeasureSpec.UNSPECIFIED && widthSize < mTimeBgWidth) {
hScale = (float) widthSize / (float) mTimeBgWidth;
}
if (heightMode != MeasureSpec.UNSPECIFIED && heightSize < mTimeBgHeight) {
vScale = (float) heightSize / (float) mTimeBgHeight;
}
float scale = Math.min(hScale, vScale);
setMeasuredDimension(mTimeBgWidth * (int)scale,mTimeBgHeight * (int)scale);
/*setMeasuredDimension(
resolveSizeAndState((int) (mTimeBgWidth * scale),widthMeasureSpec, 0),
resolveSizeAndState((int) (mTimeBgHeight * scale),heightMeasureSpec, 0));*/
}
/* (non-Javadoc)
* @see android.view.View#onSizeChanged(int, int, int, int)
*/
@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
super.onSizeChanged(w, h, oldw, oldh);
}
/* (non-Javadoc)
* @see android.view.View#onDraw(android.graphics.Canvas)
*/
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
int availableWidth = getRight() - getLeft();
int availableHeight = getBottom() - getTop();
int x = availableWidth / 2;
int y = availableHeight / 2;
int w = 0;
int h = 0;
if (!dApmBmp.isEmpty() && dApmBmp.size() > 0) {
w = 2*mTimeColon.getIntrinsicWidth() + 8
* dTimeBmp.get(0).getIntrinsicWidth()
+ dApmBmp.get(0).getIntrinsicWidth();
h = dTimeBmp.get(0).getIntrinsicHeight();
} else {
w = 2*mTimeColon.getIntrinsicWidth() + 6
* dTimeBmp.get(0).getIntrinsicWidth();
h = dTimeBmp.get(0).getIntrinsicHeight();
}
boolean scaled = false;
if (availableWidth < w || availableHeight < h) {
scaled = true;
float scale = Math.min((float) availableWidth / (float) w,
(float) availableHeight / (float) h);
canvas.save();
canvas.scale(scale, scale, x, y);
}
int dis_x = x - (w / 2);
int dis_y = y - (h / 2);
if (mTimeBg != null) {
final Drawable timeBg = mTimeBg;
timeBg.setBounds(dis_x, dis_y, dis_x + w, dis_y + h);
timeBg.draw(canvas);
canvas.save();
}
int pos = Integer.parseInt(mHour.substring(0, 1));
Drawable timeBmp = dTimeBmp.get(pos);
int numW = timeBmp.getIntrinsicWidth();
int numH = timeBmp.getIntrinsicHeight();
timeBmp.setBounds(dis_x, dis_y, dis_x + numW, dis_y + numH);
timeBmp.draw(canvas);
pos = Integer.parseInt(mHour.substring(1, 2));
timeBmp = dTimeBmp.get(pos);
timeBmp.setBounds(dis_x + numW, dis_y, dis_x + 2 * numW, dis_y + numH);
timeBmp.draw(canvas);
final Drawable timeColon = mTimeColon;
int colonW = timeColon.getIntrinsicWidth();
int colonH = timeColon.getIntrinsicHeight();
if (colonH < numH) {
timeColon.setBounds(dis_x + 2 * numW, dis_y + (numH - colonH) / 2,
dis_x + 2 * numW + colonW, dis_y + (numH - colonH) / 2
+ colonH);
} else {
timeColon.setBounds(dis_x + 2 * numW, dis_y, dis_x + 2 * numW
+ colonW, dis_y + colonH);
}
timeColon.draw(canvas);
pos = Integer.parseInt(mMinutes.substring(0, 1));
timeBmp = dTimeBmp.get(pos);
timeBmp.setBounds(dis_x + 2 * numW + colonW, dis_y, dis_x + 3 * numW
+ colonW, dis_y + numH);
timeBmp.draw(canvas);
pos = Integer.parseInt(mMinutes.substring(1, 2));
timeBmp = dTimeBmp.get(pos);
timeBmp.setBounds(dis_x + 3 * numW + colonW, dis_y, dis_x + 4 * numW
+ colonW, dis_y + numH);
timeBmp.draw(canvas);
if (colonH < numH) {
timeColon.setBounds(dis_x + 4 * numW+colonW, dis_y + (numH - colonH) / 2,
dis_x + 4 * numW + 2*colonW, dis_y + (numH - colonH) / 2
+ colonH);
} else {
timeColon.setBounds(dis_x + 4 * numW+colonW, dis_y, dis_x + 4 * numW
+ 2*colonW, dis_y + colonH);
}
timeColon.draw(canvas);
pos = Integer.parseInt(mSecond.substring(0, 1));
timeBmp = dTimeBmp.get(pos);
timeBmp.setBounds(dis_x + 4 * numW + 2*colonW, dis_y, dis_x + 5 * numW
+ 2*colonW, dis_y + numH);
timeBmp.draw(canvas);
pos = Integer.parseInt(mSecond.substring(1, 2));
timeBmp = dTimeBmp.get(pos);
timeBmp.setBounds(dis_x + 5 * numW + 2*colonW, dis_y, dis_x + 6 * numW
+ 2*colonW, dis_y + numH);
timeBmp.draw(canvas);
if (!dApmBmp.isEmpty() && dApmBmp.size() > 0) {
if (!get24HourMode()) {
if (mCalendar.hour > 12) {
pos = 1;
} else {
pos = 0;
}
timeBmp = dApmBmp.get(pos);
int apmW = timeBmp.getIntrinsicWidth();
int apmH = timeBmp.getIntrinsicHeight();
if (apmH < numH) {
timeBmp.setBounds(dis_x + 4 * numW + colonW, dis_y
+ (numH - apmH) / 2, dis_x + 4 * numW + colonW
+ apmW, dis_y + (numH - apmH) / 2 + apmH);
} else {
timeBmp.setBounds(dis_x + 4 * numW + colonW, dis_y, dis_x
+ 4 * numW + colonW + apmW, dis_y + apmH);
}
timeBmp.draw(canvas);
}
}
if (scaled) {
canvas.restore();
}
}
/**
* Gets the 24 hour mode.
*
* @return the 24 hour mode
*/
private boolean get24HourMode() {
return android.text.format.DateFormat.is24HourFormat(getContext());
}
/**
* On time changed.
*/
private void onTimeChanged() {
mCalendar.setToNow();
if (!get24HourMode() && !dApmBmp.isEmpty() && dApmBmp.size() > 0) {
if (mCalendar.hour > 12) {
mHour = String.format("%02d", mCalendar.hour - 12);
} else {
mHour = String.format("%02d", mCalendar.hour);
}
} else {
mHour = String.format("%02d", mCalendar.hour);
}
mSecond = AbStrUtil.strFormat2(String.valueOf(mCalendar.second));
mMinutes = String.format("%02d", mCalendar.minute);
updateContentDescription(mCalendar);
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
onTimeChanged();
}
}, 1000);
invalidate();
}
/** The m intent receiver. */
private final BroadcastReceiver mIntentReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
if (intent.getAction().equals(Intent.ACTION_TIMEZONE_CHANGED)) {
String tz = intent.getStringExtra("time-zone");
mCalendar = new Time(TimeZone.getTimeZone(tz).getID());
}
}
};
/**
* Update content description.
*
* @param time the time
*/
private void updateContentDescription(Time time) {
int flags = DateUtils.FORMAT_SHOW_TIME | DateUtils.FORMAT_24HOUR;
String contentDescription = DateUtils.formatDateTime(this.getContext(),
time.toMillis(false), flags);
setContentDescription(contentDescription);
}
}