source: trunk/hiptop/pester/net/sabi/pester/Alarm.java@ 249

Last change on this file since 249 was 249, checked in by Nicholas Riley, 18 years ago

Better (informative, not yet pretty) date display.

File size: 6.3 KB
Line 
1package net.sabi.pester;
2
3import java.io.ByteArrayInputStream;
4import java.io.ByteArrayOutputStream;
5import java.io.DataInputStream;
6import java.io.DataOutputStream;
7import danger.app.Application;
8import danger.audio.RingToneObject;
9import danger.internal.Date;
10import danger.util.LocaleUtils;
11import danger.util.StdActiveList;
12import danger.util.StdActiveObject;
13import danger.util.DEBUG;
14import danger.util.format.DateFormat;
15import danger.util.format.StringFormat;
16
17public class Alarm extends StdActiveObject {
18 private static final int VERSION_1 = 1;
19
20 // persisted
21 private String mMessage;
22 private int mType;
23 private int mInterval; // XXX rename to 'interval'
24 private Date mDate;
25 private RingToneObject mAlert;
26
27 // transient
28 private int mState;
29 private danger.app.Alarm mAlarm;
30
31 public Alarm() {
32 mState = STATE_INVALID;
33 mAlarm = new danger.app.Alarm(0, Application.getCurrentApp(), this);
34 }
35
36 public String getMessage() {
37 return mMessage;
38 }
39 public int getPeriod() {
40 return mInterval;
41 }
42 public boolean getUsesPeriod() {
43 return mType != TYPE_DATE;
44 }
45 public boolean getRepeating() {
46 return mType == TYPE_PERIODIC_REPEATING;
47 }
48 public Date getDate() {
49 return mDate;
50 }
51 public RingToneObject getAlert() {
52 return mAlert;
53 }
54
55 public void setMessage(String message) {
56 mMessage = message;
57 }
58 public void setPeriod(int period, boolean repeating) {
59 mType = repeating ? TYPE_PERIODIC_REPEATING : TYPE_PERIODIC;
60 mInterval = period;
61 }
62 public void setDate(Date date) {
63 mType = TYPE_DATE;
64 mDate = date;
65 }
66 public void setAlert(RingToneObject alert) {
67 mAlert = alert.isValid() ? alert : null;
68 }
69
70 public byte[] toByteArray() {
71 try {
72 ByteArrayOutputStream byteStream = new ByteArrayOutputStream();
73 DataOutputStream dataStream = new DataOutputStream(byteStream);
74
75 dataStream.writeByte(VERSION_1);
76 dataStream.writeUTF(mMessage);
77 dataStream.writeInt(mInterval);
78 dataStream.writeInt(mDate.getUnixTimeGMT());
79 dataStream.writeInt(mType);
80 dataStream.flush();
81 return byteStream.toByteArray();
82 } catch (Exception e) {
83 // XXX do something
84 }
85 return null;
86 }
87
88 public void fromByteArray(byte[] data) {
89 try {
90 ByteArrayInputStream byteStream = new ByteArrayInputStream(data);
91 DataInputStream dataStream = new DataInputStream(byteStream);
92
93 byte version = dataStream.readByte();
94 if (version != VERSION_1) {
95 // XXX barf
96 }
97 mMessage = dataStream.readUTF();
98 mInterval = dataStream.readInt();
99 mType = dataStream.readInt();
100 } catch (Exception e) {
101 // XXX do something
102 }
103 }
104
105 void beginEditing() {
106 mState = STATE_EDITING;
107 mAlarm.deactivate();
108 }
109
110 public static int secondsFromNow(Date date) {
111 return date.getDangerTimeGMT() - new Date().getDangerTimeGMT();
112 }
113
114 protected int remainingInterval() {
115 return secondsFromNow(mDate);
116 }
117
118 void resume() {
119 mState = STATE_SCHEDULED;
120 int interval = remainingInterval();
121 if (interval < 0) interval = 0;
122 mAlarm.resetWake(interval);
123 DEBUG.p("resetWake: " + description() + " - " + interval + "s left");
124 update();
125 }
126
127 void schedule() {
128 if (getUsesPeriod()) {
129 mDate = new Date();
130 mDate.addSeconds(mInterval);
131 }
132 resume();
133 }
134
135 void dismiss() {
136 if (getRepeating())
137 schedule();
138 else
139 ((StdActiveList)getDelegate()).removeItem(this);
140 }
141
142 void cancel() {
143 mState = STATE_INVALID;
144 mAlarm.deactivate();
145 }
146
147 public String description() {
148 StringBuffer sb = new StringBuffer();
149 switch (mState) {
150 case STATE_INVALID: sb.append("invalid "); break;
151 case STATE_EDITING: sb.append("editing "); break;
152 case STATE_SCHEDULED: sb.append("scheduled "); break;
153 }
154 switch (mType) {
155 case TYPE_PERIODIC:
156 sb.append("periodic (").append(mInterval).append("s) "); break;
157 case TYPE_PERIODIC_REPEATING:
158 sb.append("repeating periodic (").append(mInterval).append("s) ");
159 break;
160 case TYPE_DATE:
161 sb.append("date (").append(getDateTimeString()).append(") "); break;
162 }
163 sb.append("alarm: ");
164 sb.append('"').append(mMessage).append('"');
165 return sb.toString();
166 }
167
168 public static String dateTimeString(Date date, boolean relative) {
169 if (date == null)
170 return null;
171 String layout = LocaleUtils.getDateTimePattern();
172 String timeFormat = LocaleUtils.getMediumTimeFormat();
173 String dateFormat = LocaleUtils.getMediumDateFormat();
174 String timeString = DateFormat.withFormat(timeFormat, date);
175 String dateString = DateFormat.withFormat(dateFormat, date);
176 if (relative) {
177 int daysFromToday = date.getDaysBetween(new Date());
178 if (daysFromToday == 0)
179 dateString = "Today"; // XXX localize
180 else if (daysFromToday == 1)
181 dateString = "Tomorrow"; // XXX localize
182 else if (daysFromToday < 7)
183 dateString = date.getDayString();
184 }
185 return StringFormat.withFormat(layout, timeString, dateString);
186 }
187
188 protected static void addUnit(StringBuffer sb, int n, String unit) {
189 if (n == 0) return;
190 if (sb.length() > 0) sb.append(", ");
191 if (n == 1) sb.append("one ").append(unit);
192 else sb.append(n).append(' ').append(unit).append("s");
193 }
194
195 public static String intervalString(int i) { // XXX localize
196 if (i < 0)
197 return "expired";
198 StringBuffer sb = new StringBuffer();
199 if (i < 60) {
200 addUnit(sb, i, "second");
201 } else {
202 i /= 60; // minutes
203 if (i < 24 * 60) {
204 addUnit(sb, i / 60, "hour");
205 addUnit(sb, i % 60, "minute");
206 } else {
207 i /= 60; // hours
208 if (i < 365) {
209 addUnit(sb, i / 24, "day");
210 addUnit(sb, i % 24, "hour");
211 } else {
212 i /= 24; // days
213 addUnit(sb, i / 365, "year"); // XXX actually 365.242199
214 addUnit(sb, i % 365, "day");
215 }
216 }
217 }
218 return sb.toString();
219 }
220
221 public String getDateTimeString(boolean relative) {
222 return dateTimeString(mDate, relative);
223 }
224
225 public String getDateTimeString() {
226 return getDateTimeString(false);
227 }
228
229 public String getIntervalString() {
230 return intervalString(remainingInterval());
231 }
232
233 public String toString() {
234 return getDateTimeString(true) + " - " + mMessage;
235 }
236
237 public static final int TYPE_PERIODIC = 0;
238 public static final int TYPE_PERIODIC_REPEATING = 1;
239 public static final int TYPE_DATE = 2;
240
241 public static final int STATE_INVALID = 0;
242 public static final int STATE_EDITING = 1;
243 public static final int STATE_SCHEDULED = 2;
244}
Note: See TracBrowser for help on using the repository browser.