2021-09-28 13:06:37 +08:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
2021-10-04 08:37:15 +08:00
|
|
|
#include <limits.h>
|
2021-09-28 13:06:37 +08:00
|
|
|
#include "watch.h"
|
2021-10-17 00:58:14 +08:00
|
|
|
#include "movement.h"
|
|
|
|
#include "movement_config.h"
|
2021-09-28 13:06:37 +08:00
|
|
|
|
2021-10-17 01:23:23 +08:00
|
|
|
movement_state_t movement_state;
|
2021-10-17 01:14:52 +08:00
|
|
|
void * watch_face_contexts[MOVEMENT_NUM_FACES];
|
2021-10-19 00:15:57 +08:00
|
|
|
const int32_t movement_le_inactivity_deadlines[8] = {INT_MAX, 3600, 7200, 21600, 43200, 86400, 172800, 604800};
|
|
|
|
const int32_t movement_timeout_inactivity_deadlines[4] = {60, 120, 300, 1800};
|
2021-10-17 01:23:23 +08:00
|
|
|
movement_event_t event;
|
2021-09-28 13:06:37 +08:00
|
|
|
|
2021-11-25 23:06:49 +08:00
|
|
|
const int16_t movement_timezone_offsets[] = {
|
|
|
|
-720, // 0 : -12:00:00 (Baker Island Time)
|
|
|
|
-660, // 1 : -11:00:00 (Niue Time)
|
|
|
|
-600, // 2 : -10:00:00 (Hawaii-Aleutian Standard Time)
|
|
|
|
-570, // 3 : -9:30:00 (Marquesas Islands Time)
|
|
|
|
-540, // 4 : -9:00:00 (Alaska Standard Time)
|
|
|
|
-480, // 5 : -8:00:00 (Pacific Standard Time)
|
|
|
|
-420, // 6 : -7:00:00 (Mountain Standard Time)
|
|
|
|
-360, // 7 : -6:00:00 (Central Standard Time)
|
|
|
|
-300, // 8 : -5:00:00 (Eastern Standard Time)
|
|
|
|
-270, // 9 : -4:30:00 (Venezuelan Standard Time)
|
|
|
|
-240, // 10 : -4:00:00 (Atlantic Standard Time)
|
|
|
|
-210, // 11 : -3:30:00 (Newfoundland Standard Time)
|
|
|
|
-180, // 12 : -3:00:00 (Brasilia Time)
|
|
|
|
-150, // 13 : -2:30:00 (Newfoundland Daylight Time)
|
|
|
|
-120, // 14 : -2:00:00 (Fernando de Noronha Time)
|
|
|
|
-60, // 15 : -1:00:00 (Azores Standard Time)
|
|
|
|
0, // 16 : 0:00:00 (UTC)
|
|
|
|
60, // 17 : 1:00:00 (Central European Time)
|
|
|
|
120, // 18 : 2:00:00 (South African Standard Time)
|
|
|
|
180, // 19 : 3:00:00 (Arabia Standard Time)
|
|
|
|
210, // 20 : 3:30:00 (Iran Standard Time)
|
|
|
|
240, // 21 : 4:00:00 (Georgia Standard Time)
|
|
|
|
270, // 22 : 4:30:00 (Afghanistan Time)
|
|
|
|
300, // 23 : 5:00:00 (Pakistan Standard Time)
|
|
|
|
330, // 24 : 5:30:00 (Indian Standard Time)
|
|
|
|
345, // 25 : 5:45:00 (Nepal Time)
|
|
|
|
360, // 26 : 6:00:00 (Kyrgyzstan time)
|
|
|
|
390, // 27 : 6:30:00 (Myanmar Time)
|
|
|
|
420, // 28 : 7:00:00 (Thailand Standard Time)
|
|
|
|
480, // 29 : 8:00:00 (China Standard Time, Australian Western Standard Time)
|
|
|
|
525, // 30 : 8:45:00 (Australian Central Western Standard Time)
|
|
|
|
540, // 31 : 9:00:00 (Japan Standard Time, Korea Standard Time)
|
|
|
|
570, // 32 : 9:30:00 (Australian Central Standard Time)
|
|
|
|
600, // 33 : 10:00:00 (Australian Eastern Standard Time)
|
|
|
|
630, // 34 : 10:30:00 (Lord Howe Standard Time)
|
|
|
|
660, // 35 : 11:00:00 (Solomon Islands Time)
|
|
|
|
720, // 36 : 12:00:00 (New Zealand Standard Time)
|
|
|
|
765, // 37 : 12:45:00 (Chatham Standard Time)
|
|
|
|
780, // 38 : 13:00:00 (Tonga Time)
|
|
|
|
825, // 39 : 13:45:00 (Chatham Daylight Time)
|
|
|
|
840, // 40 : 14:00:00 (Line Islands Time)
|
|
|
|
};
|
|
|
|
|
2021-10-04 08:37:15 +08:00
|
|
|
void cb_mode_btn_interrupt();
|
|
|
|
void cb_light_btn_interrupt();
|
|
|
|
void cb_alarm_btn_interrupt();
|
|
|
|
void cb_alarm_btn_extwake();
|
|
|
|
void cb_alarm_fired();
|
2021-10-04 02:26:17 +08:00
|
|
|
void cb_tick();
|
|
|
|
|
2021-10-17 01:40:17 +08:00
|
|
|
static inline void _movement_reset_inactivity_countdown() {
|
2021-10-20 01:33:11 +08:00
|
|
|
movement_state.le_mode_ticks = movement_le_inactivity_deadlines[movement_state.settings.bit.le_interval];
|
|
|
|
movement_state.timeout_ticks = movement_timeout_inactivity_deadlines[movement_state.settings.bit.to_interval];
|
2021-10-04 08:37:15 +08:00
|
|
|
}
|
|
|
|
|
2021-10-24 05:55:19 +08:00
|
|
|
void _movement_handle_background_tasks() {
|
|
|
|
for(uint8_t i = 0; i < MOVEMENT_NUM_FACES; i++) {
|
|
|
|
// For each face, if the watch face wants a background task...
|
|
|
|
if (watch_faces[i].wants_background_task != NULL && watch_faces[i].wants_background_task(&movement_state.settings, watch_face_contexts[i])) {
|
|
|
|
// ...we give it one. pretty straightforward!
|
|
|
|
movement_event_t background_event = { EVENT_BACKGROUND_TASK, 0 };
|
|
|
|
watch_faces[i].loop(background_event, &movement_state.settings, watch_face_contexts[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
movement_state.needs_background_tasks_handled = false;
|
|
|
|
}
|
|
|
|
|
2021-10-17 00:58:14 +08:00
|
|
|
void movement_request_tick_frequency(uint8_t freq) {
|
2021-10-04 00:31:51 +08:00
|
|
|
watch_rtc_disable_all_periodic_callbacks();
|
2021-10-17 00:58:14 +08:00
|
|
|
movement_state.subsecond = 0;
|
|
|
|
movement_state.tick_frequency = freq;
|
2021-10-04 00:31:51 +08:00
|
|
|
watch_rtc_register_periodic_callback(cb_tick, freq);
|
|
|
|
}
|
|
|
|
|
2021-10-17 00:58:14 +08:00
|
|
|
void movement_illuminate_led() {
|
2021-10-20 01:33:11 +08:00
|
|
|
watch_set_led_color(movement_state.settings.bit.led_red_color ? (0xF | movement_state.settings.bit.led_red_color << 4) : 0,
|
|
|
|
movement_state.settings.bit.led_green_color ? (0xF | movement_state.settings.bit.led_green_color << 4) : 0);
|
|
|
|
movement_state.led_on = true;
|
|
|
|
movement_state.light_ticks = movement_state.settings.bit.led_duration * 2;
|
2021-10-04 00:31:51 +08:00
|
|
|
}
|
|
|
|
|
2021-10-17 01:14:52 +08:00
|
|
|
void movement_move_to_face(uint8_t watch_face_index) {
|
|
|
|
movement_state.watch_face_changed = true;
|
2021-10-17 01:28:52 +08:00
|
|
|
movement_state.next_watch_face = watch_face_index;
|
2021-10-04 02:26:17 +08:00
|
|
|
}
|
|
|
|
|
2021-10-17 01:14:52 +08:00
|
|
|
void movement_move_to_next_face() {
|
|
|
|
movement_move_to_face((movement_state.current_watch_face + 1) % MOVEMENT_NUM_FACES);
|
2021-10-04 02:26:17 +08:00
|
|
|
}
|
2021-10-04 00:31:51 +08:00
|
|
|
|
2021-09-28 13:06:37 +08:00
|
|
|
void app_init() {
|
2021-10-17 00:58:14 +08:00
|
|
|
memset(&movement_state, 0, sizeof(movement_state));
|
2021-10-04 08:37:15 +08:00
|
|
|
|
2021-10-17 01:28:52 +08:00
|
|
|
movement_state.settings.bit.led_green_color = 0xF;
|
|
|
|
movement_state.settings.bit.button_should_sound = true;
|
2021-10-20 01:33:11 +08:00
|
|
|
movement_state.settings.bit.le_interval = 1;
|
|
|
|
movement_state.settings.bit.led_duration = 1;
|
2021-11-25 23:06:49 +08:00
|
|
|
movement_state.settings.bit.time_zone = 16; // default to GMT
|
2021-10-17 01:40:17 +08:00
|
|
|
_movement_reset_inactivity_countdown();
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
2021-10-21 01:45:22 +08:00
|
|
|
void app_wake_from_backup() {
|
|
|
|
// This app does not support BACKUP mode.
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void app_setup() {
|
2021-10-06 03:55:34 +08:00
|
|
|
static bool is_first_launch = true;
|
|
|
|
|
|
|
|
if (is_first_launch) {
|
2021-10-17 01:14:52 +08:00
|
|
|
for(uint8_t i = 0; i < MOVEMENT_NUM_FACES; i++) {
|
|
|
|
watch_face_contexts[i] = NULL;
|
2021-10-06 03:55:34 +08:00
|
|
|
is_first_launch = false;
|
|
|
|
}
|
2021-10-24 05:55:19 +08:00
|
|
|
|
|
|
|
// set up the 1 minute alarm (for background tasks and low power updates)
|
|
|
|
watch_date_time alarm_time;
|
|
|
|
alarm_time.reg = 0;
|
|
|
|
alarm_time.unit.second = 59; // after a match, the alarm fires at the next rising edge of CLK_RTC_CNT, so 59 seconds lets us update at :00
|
|
|
|
watch_rtc_register_alarm_callback(cb_alarm_fired, alarm_time, ALARM_MATCH_SS);
|
2021-10-06 03:55:34 +08:00
|
|
|
}
|
2021-10-17 01:40:17 +08:00
|
|
|
if (movement_state.le_mode_ticks != -1) {
|
2021-10-04 08:37:15 +08:00
|
|
|
watch_disable_extwake_interrupt(BTN_ALARM);
|
2021-09-28 13:06:37 +08:00
|
|
|
|
2021-10-04 08:37:15 +08:00
|
|
|
watch_enable_external_interrupts();
|
|
|
|
watch_register_interrupt_callback(BTN_MODE, cb_mode_btn_interrupt, INTERRUPT_TRIGGER_BOTH);
|
|
|
|
watch_register_interrupt_callback(BTN_LIGHT, cb_light_btn_interrupt, INTERRUPT_TRIGGER_BOTH);
|
|
|
|
watch_register_interrupt_callback(BTN_ALARM, cb_alarm_btn_interrupt, INTERRUPT_TRIGGER_BOTH);
|
2021-09-28 13:06:37 +08:00
|
|
|
|
2021-10-04 08:37:15 +08:00
|
|
|
watch_enable_buzzer();
|
|
|
|
watch_enable_leds();
|
|
|
|
watch_enable_display();
|
2021-09-28 13:06:37 +08:00
|
|
|
|
2021-10-17 00:58:14 +08:00
|
|
|
movement_request_tick_frequency(1);
|
2021-10-04 08:37:15 +08:00
|
|
|
|
2021-10-17 01:14:52 +08:00
|
|
|
for(uint8_t i = 0; i < MOVEMENT_NUM_FACES; i++) {
|
2021-10-17 01:28:52 +08:00
|
|
|
watch_faces[i].setup(&movement_state.settings, &watch_face_contexts[i]);
|
2021-10-04 08:37:15 +08:00
|
|
|
}
|
2021-10-04 02:26:17 +08:00
|
|
|
|
2021-11-08 21:11:59 +08:00
|
|
|
watch_faces[movement_state.current_watch_face].activate(&movement_state.settings, watch_face_contexts[movement_state.current_watch_face]);
|
2021-10-06 22:25:28 +08:00
|
|
|
event.subsecond = 0;
|
|
|
|
event.event_type = EVENT_ACTIVATE;
|
2021-10-04 08:37:15 +08:00
|
|
|
}
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
2021-10-21 01:45:22 +08:00
|
|
|
void app_prepare_for_standby() {
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
2021-10-21 01:45:22 +08:00
|
|
|
void app_wake_from_standby() {
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool app_loop() {
|
2021-10-17 01:14:52 +08:00
|
|
|
if (movement_state.watch_face_changed) {
|
2021-10-17 01:28:52 +08:00
|
|
|
if (movement_state.settings.bit.button_should_sound) {
|
2021-10-17 01:14:52 +08:00
|
|
|
// low note for nonzero case, high note for return to watch_face 0
|
2021-10-17 01:28:52 +08:00
|
|
|
watch_buzzer_play_note(movement_state.next_watch_face ? BUZZER_NOTE_C7 : BUZZER_NOTE_C8, 50);
|
2021-10-04 07:16:14 +08:00
|
|
|
}
|
2021-10-17 01:28:52 +08:00
|
|
|
watch_faces[movement_state.current_watch_face].resign(&movement_state.settings, watch_face_contexts[movement_state.current_watch_face]);
|
|
|
|
movement_state.current_watch_face = movement_state.next_watch_face;
|
2021-10-04 21:51:49 +08:00
|
|
|
watch_clear_display();
|
2021-10-17 01:28:52 +08:00
|
|
|
watch_faces[movement_state.current_watch_face].activate(&movement_state.settings, watch_face_contexts[movement_state.current_watch_face]);
|
2021-10-06 22:25:28 +08:00
|
|
|
event.subsecond = 0;
|
|
|
|
event.event_type = EVENT_ACTIVATE;
|
2021-10-17 01:14:52 +08:00
|
|
|
movement_state.watch_face_changed = false;
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// if the LED is on and should be off, turn it off
|
2021-10-17 00:58:14 +08:00
|
|
|
if (movement_state.led_on && movement_state.light_ticks == 0) {
|
2021-09-28 13:06:37 +08:00
|
|
|
// unless the user is holding down the LIGHT button, in which case, give them more time.
|
|
|
|
if (watch_get_pin_level(BTN_LIGHT)) {
|
2021-10-17 00:58:14 +08:00
|
|
|
movement_state.light_ticks = 3;
|
2021-09-28 13:06:37 +08:00
|
|
|
} else {
|
|
|
|
watch_set_led_off();
|
2021-10-17 00:58:14 +08:00
|
|
|
movement_state.led_on = false;
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-24 05:55:19 +08:00
|
|
|
// handle background tasks, if the alarm handler told us we need to
|
|
|
|
if (movement_state.needs_background_tasks_handled) _movement_handle_background_tasks();
|
|
|
|
|
2021-10-17 01:40:17 +08:00
|
|
|
// if we have timed out of our low energy mode countdown, enter low energy mode.
|
|
|
|
if (movement_state.le_mode_ticks == 0) {
|
|
|
|
movement_state.le_mode_ticks = -1;
|
2021-10-04 08:37:15 +08:00
|
|
|
watch_register_extwake_callback(BTN_ALARM, cb_alarm_btn_extwake, true);
|
2021-10-06 22:25:28 +08:00
|
|
|
event.event_type = EVENT_NONE;
|
|
|
|
event.subsecond = 0;
|
2021-10-04 23:09:11 +08:00
|
|
|
|
|
|
|
// this is a little mini-runloop.
|
2021-10-17 01:40:17 +08:00
|
|
|
// as long as le_mode_ticks is -1 (i.e. we are in low energy mode), we wake up here, update the screen, and go right back to sleep.
|
|
|
|
while (movement_state.le_mode_ticks == -1) {
|
2021-10-24 05:55:19 +08:00
|
|
|
// we also have to handle background tasks here in the mini-runloop
|
|
|
|
if (movement_state.needs_background_tasks_handled) _movement_handle_background_tasks();
|
|
|
|
|
2021-10-17 04:03:27 +08:00
|
|
|
event.event_type = EVENT_LOW_ENERGY_UPDATE;
|
2021-10-17 01:28:52 +08:00
|
|
|
watch_faces[movement_state.current_watch_face].loop(event, &movement_state.settings, watch_face_contexts[movement_state.current_watch_face]);
|
2021-10-21 01:45:22 +08:00
|
|
|
watch_enter_sleep_mode();
|
2021-10-04 23:09:11 +08:00
|
|
|
}
|
2021-10-17 01:40:17 +08:00
|
|
|
// as soon as le_mode_ticks is reset by the extwake handler, we bail out of the loop and reactivate ourselves.
|
2021-10-06 22:25:28 +08:00
|
|
|
event.event_type = EVENT_ACTIVATE;
|
2021-10-21 01:45:22 +08:00
|
|
|
// this is a hack tho: waking from sleep mode, app_setup does get called, but it happens before we have reset our ticks.
|
2021-10-04 23:09:11 +08:00
|
|
|
// need to figure out if there's a better heuristic for determining how we woke up.
|
|
|
|
app_setup();
|
2021-10-04 08:37:15 +08:00
|
|
|
}
|
|
|
|
|
2021-10-04 23:57:57 +08:00
|
|
|
static bool can_sleep = true;
|
2021-10-04 21:51:49 +08:00
|
|
|
|
2021-10-06 22:25:28 +08:00
|
|
|
if (event.event_type) {
|
2021-10-17 00:58:14 +08:00
|
|
|
event.subsecond = movement_state.subsecond;
|
2021-10-17 01:28:52 +08:00
|
|
|
can_sleep = watch_faces[movement_state.current_watch_face].loop(event, &movement_state.settings, watch_face_contexts[movement_state.current_watch_face]);
|
2021-10-06 22:25:28 +08:00
|
|
|
event.event_type = EVENT_NONE;
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
2021-11-07 12:29:27 +08:00
|
|
|
// if we have timed out of our timeout countdown, give the app a hint that they can resign.
|
|
|
|
if (movement_state.timeout_ticks == 0) {
|
|
|
|
event.event_type = EVENT_TIMEOUT;
|
|
|
|
event.subsecond = movement_state.subsecond;
|
|
|
|
watch_faces[movement_state.current_watch_face].loop(event, &movement_state.settings, watch_face_contexts[movement_state.current_watch_face]);
|
|
|
|
event.event_type = EVENT_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
event.subsecond = 0;
|
|
|
|
|
2021-10-17 00:58:14 +08:00
|
|
|
return can_sleep && !movement_state.led_on;
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
2021-10-17 01:23:23 +08:00
|
|
|
movement_event_type_t _figure_out_button_event(movement_event_type_t button_down_event_type, uint8_t *down_timestamp) {
|
2021-10-04 02:26:17 +08:00
|
|
|
watch_date_time date_time = watch_rtc_get_date_time();
|
|
|
|
if (*down_timestamp) {
|
|
|
|
uint8_t diff = ((61 + date_time.unit.second) - *down_timestamp) % 60;
|
|
|
|
*down_timestamp = 0;
|
2021-10-17 01:23:23 +08:00
|
|
|
if (diff > 1) return button_down_event_type + 2;
|
|
|
|
else return button_down_event_type + 1;
|
2021-10-04 02:26:17 +08:00
|
|
|
} else {
|
|
|
|
*down_timestamp = date_time.unit.second + 1;
|
2021-10-17 01:23:23 +08:00
|
|
|
return button_down_event_type;
|
2021-10-04 02:26:17 +08:00
|
|
|
}
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
2021-10-04 08:37:15 +08:00
|
|
|
void cb_light_btn_interrupt() {
|
2021-10-17 01:40:17 +08:00
|
|
|
_movement_reset_inactivity_countdown();
|
2021-10-17 00:58:14 +08:00
|
|
|
event.event_type = _figure_out_button_event(EVENT_LIGHT_BUTTON_DOWN, &movement_state.light_down_timestamp);
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
2021-10-04 08:37:15 +08:00
|
|
|
void cb_mode_btn_interrupt() {
|
2021-10-17 01:40:17 +08:00
|
|
|
_movement_reset_inactivity_countdown();
|
2021-10-17 00:58:14 +08:00
|
|
|
event.event_type = _figure_out_button_event(EVENT_MODE_BUTTON_DOWN, &movement_state.mode_down_timestamp);
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
2021-10-04 08:37:15 +08:00
|
|
|
void cb_alarm_btn_interrupt() {
|
2021-10-17 01:40:17 +08:00
|
|
|
_movement_reset_inactivity_countdown();
|
2021-10-17 00:58:14 +08:00
|
|
|
event.event_type = _figure_out_button_event(EVENT_ALARM_BUTTON_DOWN, &movement_state.alarm_down_timestamp);
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
2021-10-04 08:37:15 +08:00
|
|
|
void cb_alarm_btn_extwake() {
|
2021-10-04 23:09:11 +08:00
|
|
|
// wake up!
|
2021-10-17 01:40:17 +08:00
|
|
|
_movement_reset_inactivity_countdown();
|
2021-10-04 08:37:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void cb_alarm_fired() {
|
2021-10-24 05:55:19 +08:00
|
|
|
movement_state.needs_background_tasks_handled = true;
|
2021-10-04 08:37:15 +08:00
|
|
|
}
|
|
|
|
|
2021-09-28 13:06:37 +08:00
|
|
|
void cb_tick() {
|
2021-10-06 22:25:28 +08:00
|
|
|
event.event_type = EVENT_TICK;
|
2021-10-04 06:49:21 +08:00
|
|
|
watch_date_time date_time = watch_rtc_get_date_time();
|
2021-10-17 00:58:14 +08:00
|
|
|
if (date_time.unit.second != movement_state.last_second) {
|
2021-10-20 22:36:55 +08:00
|
|
|
// TODO: since we time the LED with the 1 Hz tick, the actual time lit can vary depending on whether the
|
|
|
|
// user hit it just before or just after a tick. If we time this with the system tick we can do better.
|
2021-10-17 00:58:14 +08:00
|
|
|
if (movement_state.light_ticks) movement_state.light_ticks--;
|
2021-10-20 22:36:55 +08:00
|
|
|
|
|
|
|
// TODO: can we consolidate these two ticks?
|
2021-10-20 01:33:11 +08:00
|
|
|
if (movement_state.settings.bit.le_interval && movement_state.le_mode_ticks > 0) movement_state.le_mode_ticks--;
|
2021-10-19 00:15:57 +08:00
|
|
|
if (movement_state.timeout_ticks > 0) movement_state.timeout_ticks--;
|
2021-10-04 06:49:21 +08:00
|
|
|
|
2021-10-17 00:58:14 +08:00
|
|
|
movement_state.last_second = date_time.unit.second;
|
|
|
|
movement_state.subsecond = 0;
|
2021-10-04 06:49:21 +08:00
|
|
|
} else {
|
2021-10-17 00:58:14 +08:00
|
|
|
movement_state.subsecond++;
|
2021-10-04 06:49:21 +08:00
|
|
|
}
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|