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

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

Better alarm debug description, datastore debugging; use absolute fire time for snoozed alarms too; replace the buggy alarm stack and misaligned sleep message with a painfully constructed alert.

File size: 8.8 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.app.IPCMessage;
9import danger.audio.RingToneObject;
10import danger.internal.Date;
11import danger.system.Hardware;
12import danger.text.Collator;
13import danger.util.LocaleUtils;
14import danger.util.StdActiveList;
15import danger.util.StdActiveObject;
16import danger.util.DEBUG;
17import danger.util.format.DateFormat;
18import danger.util.format.StringFormat;
19import java.util.Comparator;
20
21public class Alarm extends StdActiveObject implements Comparator {
22 private static final int VERSION_1 = 1;
23
24 // persisted
25 private String mMessage;
26 private int mType;
27 private int mPeriod;
28 private Date mDate;
29 private RingToneObject mAlert;
30
31 // transient
32 private int mState;
33 private int mAbsoluteFireTime; // valid if periodic or snoozed,
34 // and modified since reboot
35 private danger.app.Alarm mAlarm;
36 private int mUID;
37
38 public Alarm() {
39 mState = STATE_INVALID;
40 mAlarm = new danger.app.Alarm(0, Application.getCurrentApp(), this);
41 mUID = 0;
42 mAbsoluteFireTime = -1;
43 }
44
45 public String getMessage() {
46 return mMessage;
47 }
48 public int getPeriod() {
49 return mPeriod;
50 }
51 public boolean getUsesPeriod() {
52 return mType != TYPE_DATE;
53 }
54 public boolean getRepeating() {
55 return mType == TYPE_PERIODIC_REPEATING;
56 }
57 public Date getDate() {
58 return mDate;
59 }
60 public RingToneObject getAlert() {
61 return mAlert;
62 }
63 public int getUID() {
64 return mUID;
65 }
66
67 public void setMessage(String message) {
68 mMessage = message;
69 }
70 public void setPeriod(int period, boolean repeating) {
71 mType = repeating ? TYPE_PERIODIC_REPEATING : TYPE_PERIODIC;
72 mPeriod = period;
73 }
74 public void setDate(Date date) {
75 mType = TYPE_DATE;
76 mDate = date;
77 }
78 public void setAlert(RingToneObject alert) {
79 mAlert = alert.isValid() ? alert : null;
80 }
81 public void setUID(int uid) {
82 mUID = uid;
83 }
84 public void snoozeForMinutes(int minutes) {
85 mDate = new Date();
86 mDate.addMinutes(minutes);
87 mAbsoluteFireTime = Hardware.getAbsoluteTime() + (minutes * 60);
88 resume();
89 }
90
91 public byte[] toByteArray() {
92 try {
93 ByteArrayOutputStream byteStream = new ByteArrayOutputStream();
94 DataOutputStream dataStream = new DataOutputStream(byteStream);
95
96 dataStream.writeByte(VERSION_1);
97 dataStream.writeUTF(mMessage);
98 dataStream.writeInt(mPeriod);
99 dataStream.writeInt(mDate.getUnixTimeGMT());
100 dataStream.writeInt(mType);
101 dataStream.writeInt(mAlert == null ? 0 : mAlert.getID());
102 dataStream.flush();
103 return byteStream.toByteArray();
104 } catch (Exception e) {
105 // XXX do something
106 DEBUG.p("failed to write alarm:" + e);
107 }
108 return null;
109 }
110
111 public void fromByteArray(byte[] data) {
112 try {
113 ByteArrayInputStream byteStream = new ByteArrayInputStream(data);
114 DataInputStream dataStream = new DataInputStream(byteStream);
115
116 byte version = dataStream.readByte();
117 if (version != VERSION_1) {
118 // XXX barf
119 }
120 mMessage = dataStream.readUTF();
121 mPeriod = dataStream.readInt();
122 mDate = new Date(dataStream.readInt());
123 mType = dataStream.readInt();
124 int alertID = dataStream.readInt();
125 mAlert = (alertID == 0 ? null : new RingToneObject(alertID));
126 mAbsoluteFireTime = -1; // just in case
127 } catch (Exception e) {
128 // XXX do something
129 DEBUG.p("failed to read alarm:" + e);
130 }
131 }
132
133 void beginEditing() {
134 mState = STATE_EDITING;
135 mAlarm.deactivate();
136 }
137
138 public static int secondsFromNow(Date date) {
139 return date.getDangerTimeGMT() - new Date().getDangerTimeGMT();
140 }
141
142 protected int remainingInterval() {
143 return secondsFromNow(mDate);
144 }
145
146 void resume() {
147 mState = STATE_SCHEDULED;
148 int interval = remainingInterval();
149 if (interval < 0) interval = 0;
150 mAlarm.resetWake(interval);
151 DEBUG.p("resetWake: " + description() + " - " + interval + "s left");
152 update();
153 }
154
155 void timeChanged() {
156 if (mState != STATE_SCHEDULED)
157 return;
158 if (mAbsoluteFireTime != -1) {
159 int realSecondsLeft = mAbsoluteFireTime - Hardware.getAbsoluteTime();
160 mDate = new Date();
161 mDate.addSeconds(realSecondsLeft < 0 ? 0 : realSecondsLeft);
162 // no need to call resume(): alarm uses absolute time
163 update();
164 } else {
165 resume();
166 }
167 }
168
169 void schedule() {
170 if (getUsesPeriod()) {
171 mDate = new Date();
172 mDate.addSeconds(mPeriod);
173 mAbsoluteFireTime = Hardware.getAbsoluteTime() + mPeriod;
174 }
175 resume();
176 }
177
178 void dismiss() {
179 if (getRepeating())
180 schedule();
181 else
182 ((StdActiveList)getDelegate()).removeItem(this);
183 }
184
185 void cancel() {
186 mState = STATE_INVALID;
187 mAlarm.deactivate();
188 }
189
190 // XXX not sure if this is featureful enough to be worthwhile
191 public IPCMessage getCalendarIPCMessage() {
192 if (mState != STATE_SCHEDULED)
193 return null;
194 IPCMessage message = new IPCMessage();
195 message.addItem("action", "new");
196 message.addItem("title", getMessage());
197 message.addItem("start", mDate.getUnixTimeGMT());
198 return message;
199 }
200
201 public String description() {
202 StringBuffer sb = new StringBuffer();
203 if (mUID != 0) sb.append('(').append(mUID).append(") ");
204 switch (mState) {
205 case STATE_INVALID: sb.append("inv "); break;
206 case STATE_EDITING: sb.append("edi "); break;
207 case STATE_SCHEDULED: sb.append("sch "); break;
208 }
209 switch (mType) {
210 case TYPE_PERIODIC_REPEATING:
211 sb.append("r-");
212 case TYPE_PERIODIC:
213 sb.append("per(").append(mPeriod).append("s)");
214 if (mState != STATE_SCHEDULED)
215 break;
216 case TYPE_DATE:
217 sb.append("(").append(getDateTimeString()).append(")"); break;
218 }
219 sb.append(": ");
220 sb.append('"').append(mMessage).append('"');
221 return sb.toString();
222 }
223
224 public static String dateTimeString(Date date, boolean relative) {
225 if (date == null)
226 return null;
227 String layout = LocaleUtils.getDateTimePattern();
228 String dateFormat = LocaleUtils.getMediumDateFormat();
229 String dateString;
230 String timeFormat;
231 if (relative) {
232 timeFormat = LocaleUtils.getShortTimeFormat();
233 int daysFromToday = date.getDaysBetween(new Date());
234 if (daysFromToday == 0)
235 dateString = "Today"; // XXX localize
236 else if (daysFromToday == 1)
237 dateString = "Tomorrow"; // XXX localize
238 else if (daysFromToday > 1 && daysFromToday < 7)
239 dateString = date.getDayString();
240 else
241 dateString = DateFormat.withFormat(dateFormat, date);
242 } else {
243 dateString = DateFormat.withFormat(dateFormat, date);
244 timeFormat = LocaleUtils.getMediumTimeFormat();
245 }
246 String timeString = DateFormat.withFormat(timeFormat, date);
247 return StringFormat.withFormat(layout, timeString, dateString);
248 }
249
250 protected static void addUnit(StringBuffer sb, int n, String unit) {
251 if (n == 0) return;
252 if (sb.length() > 0) sb.append(", ");
253 if (n == 1) sb.append("one ").append(unit);
254 else sb.append(n).append(' ').append(unit).append("s");
255 }
256
257 public static String intervalString(int i) { // XXX localize
258 if (i < 0)
259 return "expired";
260 StringBuffer sb = new StringBuffer();
261 if (i < 60) {
262 addUnit(sb, i, "second");
263 } else {
264 i /= 60; // minutes
265 if (i < 24 * 60) {
266 addUnit(sb, i / 60, "hour");
267 addUnit(sb, i % 60, "minute");
268 } else {
269 i /= 60; // hours
270 if (i < 365) {
271 addUnit(sb, i / 24, "day");
272 addUnit(sb, i % 24, "hour");
273 } else {
274 i /= 24; // days
275 addUnit(sb, i / 365, "year"); // XXX actually 365.242199
276 addUnit(sb, i % 365, "day");
277 }
278 }
279 }
280 return sb.toString();
281 }
282
283 public int getSecondsUntilNextIntervalStringUpdate() {
284 int i = remainingInterval();
285 if (i <= 1)
286 return 0;
287 if (i <= 60)
288 return 1;
289 if (i <= 24 * 60 * 60) {
290 i %= 60;
291 return (i == 0 ? 60 : i + 1);
292 }
293 i %= 3600;
294 return (i == 0 ? 3600 : i + 1);
295 }
296
297 public String getDateTimeString(boolean relative) {
298 return dateTimeString(mDate, relative);
299 }
300
301 public String getDateTimeString() {
302 return getDateTimeString(false);
303 }
304
305 public String getIntervalString() {
306 return intervalString(remainingInterval());
307 }
308
309 public String toString() {
310 return getDateTimeString(true) + " - " + mMessage;
311 }
312
313 public int compare(Object arg0, Object arg1) {
314 Alarm alarm0 = (Alarm)arg0, alarm1 = (Alarm)arg1;
315 int result = alarm0.getDate().compareTo(alarm1.getDate());
316 if (result != 0) return result;
317 result = Collator.getInstance().compare(alarm0.getMessage(),
318 alarm1.getMessage());
319 if (result != 0) return result;
320 return alarm0.hashCode() - alarm1.hashCode();
321 }
322
323 public static final int TYPE_PERIODIC = 0;
324 public static final int TYPE_PERIODIC_REPEATING = 1;
325 public static final int TYPE_DATE = 2;
326
327 public static final int STATE_INVALID = 0;
328 public static final int STATE_EDITING = 1;
329 public static final int STATE_SCHEDULED = 2;
330}
Note: See TracBrowser for help on using the repository browser.