-
Notifications
You must be signed in to change notification settings - Fork 11
/
DateTime.ino
309 lines (247 loc) · 9.7 KB
/
DateTime.ino
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
/*
* DateTime.ino -- example usage for DateTime and Span objects.
*
* To use:
* - install Arduino Time library, if not built-in https://www.pjrc.com/teensy/td_libs_Time.html
* - optionally, edit the configuration #defines below
* - compile and install on Arduino-compatible device
* - open serial monitor/terminal and check it out
*
*
* Synopsis:
* This program demonstrates some of the functions of the basic DateTime object. In addition
* to the obvious, it shows how to interact with time spans and find specific time marks relative
* to a given DateTime.
*
* For more info check out the documentation at http://flyingcarsandstuff.com/projects/chronos/
* or the examples for:
* - Point event (basically repeating marks on a timeline, like "every Sunday at 07h00")
* handling in the PointEvents.ino example.
* - Calendar handling, for defining and querying schedule of Chronos::Events (1D occurrences,
* with start times and duration).
*
*
* Created on: Jan 2, 2016
* Author: Pat Deegan
* Part of the Chronos library project
* Copyright (C) 2016 Pat Deegan, http://psychogenic.com
*
* This file is part of the Chronos embedded datetime/calendar library.
*
* Chronos is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Chronos is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser Public License for more details.
*
* You should have received a copy of the GNU Lesser Public License
* along with Chronos. If not, see <http://www.gnu.org/licenses/>.
*/
#include <Time.h>
#include <Chronos.h>
#define DELAY_BETWEEN_TESTS_MS 10000
#define SERIAL_BAUDRATE 9600
#define SERIAL_DEVICE Serial
// a few defines to abstract away the Serial-specific stuff
#define PRINT(...) SERIAL_DEVICE.print(__VA_ARGS__)
#define PRINTLN(...) SERIAL_DEVICE.println(__VA_ARGS__)
#define LINE() PRINTLN(' ')
#define LINES(n) for (uint8_t _bl=0; _bl<n; _bl++) { PRINTLN(' '); }
void setup() {
SERIAL_DEVICE.begin(SERIAL_BAUDRATE);
PRINTLN(F("Starting up DateTime test"));
// setting for monday, dec 21st 2015 @ 17:30:00
// same as setTime(17, 30, 0, 21, 12, 2015);
// call setTime once with
// setTime(YYYY, MM, DD, HH, mm, SS)
// or setup the source directly and don't call this at all...
Chronos::DateTime::setTime(2015, 12, 21, 17, 30, 0);
PRINTLN(" *** No real time source set, so pretending it is currently :");
// Chronos::DateTime objects have a printTo() function that takes
// any Print-complatible object and dumps info through it. This is mostly
// just for debugging, and you can use the various attribute accessor
// methods to pretty-print however you like, in real life.
Chronos::DateTime::now().printTo(SERIAL_DEVICE);
LINES(3);
delay(2000);
}
void basicDateTimeManips() {
LINE();
PRINTLN("**** Basic DateTime Manips Test ****");
/*
* Chronos::DateTime construction.
*
* You can create DateTime objects using:
* - specific date-time info: DateTime(YYYY, MM, DD, HH, mm, SS)
* - specific epoch (unix timestamp seconds): DateTime(EEEEEEEEE);
* - another DateTime object: DateTime(someotherDateTime)
* - by assignment: DateTime a = someotherDateTime
* - nothing at all: DateTime() -- sets to DateTime(Chronos::DateTime::now())
*
*/
Chronos::DateTime nowTime(Chronos::DateTime::now());
PRINT(F("Right 'now' it's: "));
nowTime.printTo(SERIAL_DEVICE); // dump to serial
LINES(2);
// you can add or subtract "deltas" to/from DateTimes to get other dateTimes
Chronos::DateTime inTwoWeeks = nowTime + Chronos::Span::Weeks(2);
PRINT("In two weeks exactly it will be ");
inTwoWeeks.printTo(SERIAL_DEVICE);
// and subtracting datetimes between each other gives the absolute difference
Chronos::Span::Absolute diff = inTwoWeeks - nowTime;
PRINT(F(" which is "));
PRINT(diff.days());
PRINTLN(F(" days away (of course)"));
// DateTimes and Deltas can be compared using all the standard operators
if ((inTwoWeeks > nowTime) && (diff == Chronos::Span::Days(14))) {
PRINTLN(F("All is well, the future date > the now date..."));
} else {
// should never happen
PRINTLN("Something is going horribly wrong????");
}
Chronos::DateTime evenLater = inTwoWeeks
+ (Chronos::Span::Days(2) - Chronos::Span::Hours(1));
// some overly complicate mathemagic with spans, just to show it...
if (nowTime + (evenLater - inTwoWeeks) == (nowTime + Chronos::Span::Hours(47))) {
(nowTime + (evenLater - inTwoWeeks)).printTo(SERIAL_DEVICE);
PRINTLN(F(" is indeed in 47 hours"));
} else {
// should never happen
PRINTLN(F("Something is going horribly wrong????"));
}
LINES(2);
}
void boundsAndNextPrev() {
LINE();
PRINTLN(F("**** Bounds and prev/next Test ****"));
Chronos::DateTime curDateTime = Chronos::DateTime::now();
PRINT(F("Right 'now' it's: "));
curDateTime.printTo(SERIAL_DEVICE); // dump to serial
LINES(2);
// you can get the next/previous day, month rather easily
Chronos::DateTime nextSaturday = curDateTime.next(Chronos::Weekday::Saturday);
PRINT(F("Next saturday is in "));
(nextSaturday - curDateTime).printTo(SERIAL_DEVICE);
LINE();
Chronos::DateTime previousJune = curDateTime.previous(Chronos::Named::Month::June);
PRINT(F("And the preceding June started on "));
previousJune.printTo(SERIAL_DEVICE);
LINE();
// or get fancy and use Chronos::Marks (see the PointEvents example for more on that)
PRINT(F("The next time we see december 3rd is "));
PRINT((curDateTime.next(Chronos::Mark::Yearly(12, 3)) - curDateTime).days());
PRINT(F(" days away, whereas the last time we were dec. 3rd was "));
PRINT((curDateTime - curDateTime.previous(Chronos::Mark::Yearly(12, 3))).days());
PRINTLN(F(" days ago."));
// a Chronos::DateTime::Bounds is a structure containing the
// (inclusive) datetime boundaries for a period. you can query
// dayBounds
// monthBounds
// yearBounds
//
// The important thing to remember is that the boundaries contain the first
// and last DateTimes that are *part* of the period (with a granularity of 1 second)
// so both
// bounds.start
// bounds.finish
// are 'within' the period. Thus if you subtract one from the other, you'll be off
// by one second of what you might expect. e.g. say you have the boundary for a
// certain day, then
// (bounds.finish - bounds.start) will = 23 hours, 59 minutes and 59 seconds.
// therefore, the total length delta may be accessed using the length() method which
// in this case would give 24 hours (and 0 minutes, 0 seconds).
Chronos::DateTime::Bounds todayBounds = Chronos::DateTime::dayBounds(curDateTime);
if (todayBounds.start <= curDateTime && curDateTime <= todayBounds.finish)
{
PRINT(F("Yes indeed, "));
todayBounds.start.printTo(SERIAL_DEVICE);
PRINT(F(" <= "));
curDateTime.printTo(SERIAL_DEVICE);
PRINT(F(" <= "));
todayBounds.finish.printTo(SERIAL_DEVICE);
LINE();
PRINT(F("And day-bounds length is "));
todayBounds.spanRounded().printTo(SERIAL_DEVICE);
LINE();
} else {
// should never happen
PRINTLN(F("Something is going horribly wrong????"));
}
PRINTLN("More bounds... Feb FYI...");
// check years 2015-2022
for (int i = 2015; i < 2022; i++) {
Chronos::DateTime::Bounds febBounds = Chronos::DateTime::monthBounds(i, 2);
PRINT(F("Feb "));
PRINT(i);
PRINT(F(" has "));
// could do febBounds.spanRounded() == Chronos::Span::Days(29))
// but we'll do:
if (febBounds.spanRounded().days() == 29) {
PRINT(F(" 29! "));
} else {
PRINT(F("only 28 "));
}
PRINTLN(F(" days"));
}
}
void countDownToBladeRunner() {
LINE();
PRINTLN(F("**** 'BladeRunner' Test ****"));
// now time
Chronos::DateTime curDateTime = Chronos::DateTime::now();
// Blade runner: 2019-11-01 18:05:00
Chronos::DateTime targetDateTime(2019, 11, 1, 18, 05, 00);
// is that still in the glorious future?
if (curDateTime > targetDateTime) {
PRINTLN(F("Time sure flies... replicants are already here!"));
return;
}
// how far away is that?
Chronos::Span::Absolute timeDiff = targetDateTime - curDateTime;
PRINT(F("There are "));
PRINT(timeDiff.days());
PRINT(F(" days, "));
PRINT(timeDiff.hours());
PRINT(F(" hours, "));
PRINT(timeDiff.minutes());
PRINT(F(" minutes and "));
PRINT(timeDiff.seconds());
PRINT(F(" seconds between 'now' ("));
curDateTime.printTo(SERIAL_DEVICE);
PRINTLN(F(") and when Deckard chases after rogue Nexus-6 models on earth, on "));
targetDateTime.printTo(SERIAL_DEVICE);
PRINTLN(F("... moving forward, 10 days at a time: "));
while (curDateTime < targetDateTime) {
// increment using a convoluted 1 week + 3 days, just to
// show the operators in action
curDateTime += Chronos::Span::Weeks(1) + Chronos::Span::Days(3);
PRINT('.');
}
PRINTLN('!');
PRINT(F("Ooof, we have reached "));
PRINT(curDateTime.year());
PRINT('-');
PRINT(curDateTime.month());
PRINT('-');
PRINT(curDateTime.day());
PRINT(' ');
PRINT(curDateTime.hour());
PRINT(':');
PRINT(curDateTime.minute());
PRINT(':');
PRINT(curDateTime.second());
PRINTLN('!');
LINES(2);
}
void loop() {
basicDateTimeManips();
delay(DELAY_BETWEEN_TESTS_MS);
boundsAndNextPrev();
delay(DELAY_BETWEEN_TESTS_MS);
countDownToBladeRunner();
delay(DELAY_BETWEEN_TESTS_MS);
}