Native Mac OS X OtaClock replica
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

MainWindow.m 15KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439
  1. //
  2. // MainWindow.m
  3. // OtaClock
  4. //
  5. // Created by Thomas Buck on 16.08.15.
  6. // Copyright (c) 2015 xythobuz. All rights reserved.
  7. //
  8. #import "MainView.h"
  9. #import "MainWindow.h"
  10. #define RESIZE_START 1
  11. #define RESIZE_END 10
  12. #define RESIZE_STEP 1
  13. #define CONFIG_START_SCALE @"window_scale"
  14. #define CONFIG_KEEP_POSITION @"keep_position"
  15. #define CONFIG_KEEP_ON_TOP @"keep_on_top"
  16. #define CONFIG_MILITARY_TIME @"military_time"
  17. #define CONFIG_SHOW_DATE @"show_date"
  18. #define CONFIG_ALARM_TIME @"alarm_time"
  19. #define CONFIG_ALARM_MODE @"alarm_mode"
  20. #define MOUSE_CENTER_X 67
  21. #define MOUSE_CENTER_Y 47
  22. #define EYE_BLINK 0
  23. #define EYE_TOP_RIGHT 4
  24. #define EYE_TOP_LEFT 2
  25. #define EYE_BOTTOM_RIGHT 3
  26. #define EYE_BOTTOM_LEFT 1
  27. #define MAX_BLINK_DELAY 5.0
  28. #define UNBLINK_DELAY 0.1
  29. #define TIME_KEEPING_DELAY 0.25
  30. // The shadow is not properly recomputed if we don't do it multiple times
  31. #define ANIMATION_SHADOW_HACK 3
  32. #define ALARM_ANIMATION_DELAY (0.25 / ANIMATION_SHADOW_HACK)
  33. #define ALARM_ANIMATION_DELAY_END 3.0
  34. @interface MainWindow ()
  35. @property (assign) NSSize defaultSize;
  36. @property (assign) NSInteger startScale;
  37. @property (assign) NSInteger lastEyeState;
  38. @property (assign) BOOL currentlyBlinking, showDate;
  39. @property (weak) IBOutlet MainView *mainView;
  40. @property (weak) IBOutlet NSDatePicker *alarmDatePicker;
  41. @property (weak) IBOutlet NSMenuItem *lockPositionItem;
  42. @property (weak) IBOutlet NSMenuItem *keepOnTopItem;
  43. @property (weak) IBOutlet NSMenuItem *setAlarmItem;
  44. @property (weak) IBOutlet NSMenuItem *showDateItem;
  45. @property (weak) IBOutlet NSMenuItem *alarmModeItem;
  46. @property (weak) IBOutlet NSMenuItem *alarmTextItem;
  47. @property (weak) IBOutlet NSMenuItem *changeSize1;
  48. @property (weak) IBOutlet NSMenuItem *changeSize2;
  49. @property (weak) IBOutlet NSMenuItem *changeSize3;
  50. @property (weak) IBOutlet NSMenuItem *changeSize4;
  51. @property (weak) IBOutlet NSMenuItem *changeSize5;
  52. @property (weak) IBOutlet NSMenuItem *changeSize6;
  53. @property (weak) IBOutlet NSMenuItem *changeSize7;
  54. @property (weak) IBOutlet NSMenuItem *changeSize8;
  55. @property (weak) IBOutlet NSMenuItem *changeSize9;
  56. @property (weak) IBOutlet NSMenuItem *changeSize10;
  57. @end
  58. @implementation MainWindow
  59. @synthesize dragStart;
  60. @synthesize keepPosition;
  61. @synthesize defaultSize;
  62. @synthesize startScale;
  63. @synthesize lastEyeState;
  64. @synthesize currentlyBlinking, showDate;
  65. - (id)initWithContentRect:(NSRect)contentRect
  66. styleMask:(NSUInteger)aStyle
  67. backing:(NSBackingStoreType)bufferingType
  68. defer:(BOOL)flag {
  69. self = [super initWithContentRect:contentRect styleMask:NSBorderlessWindowMask backing:NSBackingStoreBuffered defer:NO];
  70. if (self != nil) {
  71. [self setAlphaValue:1.0];
  72. [self setOpaque:NO];
  73. lastEyeState = EYE_TOP_LEFT;
  74. currentlyBlinking = NO;
  75. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  76. // load and see if window should be kept on top
  77. if ([defaults objectForKey:CONFIG_KEEP_ON_TOP] != nil) {
  78. if ([defaults boolForKey:CONFIG_KEEP_ON_TOP]) {
  79. [self setLevel:NSFloatingWindowLevel];
  80. }
  81. }
  82. // load and see if window pos should be fixed
  83. if ([defaults objectForKey:CONFIG_KEEP_POSITION] == nil) {
  84. keepPosition = NO;
  85. } else {
  86. keepPosition = [defaults boolForKey:CONFIG_KEEP_POSITION];
  87. }
  88. // load window scale factor
  89. if ([defaults objectForKey:CONFIG_START_SCALE] == nil) {
  90. startScale = 1;
  91. } else {
  92. startScale = [defaults integerForKey:CONFIG_START_SCALE];
  93. }
  94. }
  95. [self setAcceptsMouseMovedEvents:YES];
  96. [NSEvent addGlobalMonitorForEventsMatchingMask:NSMouseMovedMask handler:^(NSEvent *mouseMovedEvent) { [self mouseMoved:mouseMovedEvent]; }];
  97. return self;
  98. }
  99. - (void)setDefaultBackgroundSize:(NSSize)size {
  100. defaultSize = size;
  101. NSRect frame = [self frame];
  102. frame.size = defaultSize;
  103. frame.size.width *= startScale;
  104. frame.size.height *= startScale;
  105. // We need to do all initialization of view state in here, because they are not ready in init
  106. if (keepPosition) [self.lockPositionItem setState:NSOnState];
  107. if ([self level] == NSFloatingWindowLevel) [self.keepOnTopItem setState:NSOnState];
  108. if (startScale == 1) [self.changeSize1 setState:NSOnState];
  109. if (startScale == 2) [self.changeSize2 setState:NSOnState];
  110. if (startScale == 3) [self.changeSize3 setState:NSOnState];
  111. if (startScale == 4) [self.changeSize4 setState:NSOnState];
  112. if (startScale == 5) [self.changeSize5 setState:NSOnState];
  113. if (startScale == 6) [self.changeSize6 setState:NSOnState];
  114. if (startScale == 7) [self.changeSize7 setState:NSOnState];
  115. if (startScale == 8) [self.changeSize8 setState:NSOnState];
  116. if (startScale == 9) [self.changeSize9 setState:NSOnState];
  117. if (startScale == 10) [self.changeSize10 setState:NSOnState];
  118. [[self.mainView render] drawWithEye:lastEyeState]; // Initialize render image
  119. [self unblink]; // Schedule next blinking
  120. // Start time keeping
  121. [self updateTime:nil];
  122. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  123. // load alarm date
  124. if ([defaults objectForKey:CONFIG_ALARM_TIME] == nil) {
  125. self.alarmDatePicker.dateValue = [NSDate date];
  126. } else {
  127. self.alarmDatePicker.dateValue = [defaults objectForKey:CONFIG_ALARM_TIME];
  128. }
  129. self.setAlarmItem.view = self.alarmDatePicker;
  130. NSDateComponents *components = [[NSCalendar currentCalendar] components:(NSHourCalendarUnit | NSMinuteCalendarUnit) fromDate:self.alarmDatePicker.dateValue];
  131. [self.alarmTextItem setTitle:[NSString stringWithFormat:@"%02ld:%02ld", (long)[components hour], (long)[components minute]]];
  132. // load show date state
  133. if ([defaults objectForKey:CONFIG_SHOW_DATE] == nil) {
  134. showDate = YES;
  135. } else {
  136. showDate = [defaults boolForKey:CONFIG_SHOW_DATE];
  137. }
  138. if (showDate == YES) {
  139. self.showDateItem.state = NSOnState;
  140. } else {
  141. self.showDateItem.state = NSOffState;
  142. }
  143. if ([defaults objectForKey:CONFIG_ALARM_MODE] != nil) {
  144. if ([defaults boolForKey:CONFIG_ALARM_MODE] == YES) {
  145. [self.alarmModeItem setState:NSOnState];
  146. [[self.mainView render] drawAlarmDate:self.alarmDatePicker.dateValue];
  147. }
  148. }
  149. [[self.mainView render] drawDate:showDate];
  150. [self setFrame:frame display:YES];
  151. }
  152. - (IBAction)toggleAlarm:(id)sender {
  153. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  154. if (self.alarmModeItem.state == NSOnState) {
  155. // Turn off alarm
  156. [self.alarmModeItem setState:NSOffState];
  157. [[self.mainView render] drawAlarmDate:nil];
  158. [defaults setBool:NO forKey:CONFIG_ALARM_MODE];
  159. } else {
  160. // Turn on alarm
  161. [self.alarmModeItem setState:NSOnState];
  162. [[self.mainView render] drawAlarmDate:self.alarmDatePicker.dateValue];
  163. [defaults setBool:YES forKey:CONFIG_ALARM_MODE];
  164. }
  165. [defaults synchronize];
  166. self.mainView.needsDisplay = YES;
  167. }
  168. - (void)drawAnimation:(NSNumber *)anim {
  169. if ([anim integerValue] > (3 * ANIMATION_SHADOW_HACK)) {
  170. [[self.mainView render] drawAnimation:0]; // reset
  171. } else {
  172. [[self.mainView render] drawAnimation:([anim integerValue] / ANIMATION_SHADOW_HACK)];
  173. }
  174. if ([anim integerValue] < (3 * ANIMATION_SHADOW_HACK)) {
  175. [self performSelector:@selector(drawAnimation:) withObject:[NSNumber numberWithInteger:([anim integerValue] + 1)] afterDelay:ALARM_ANIMATION_DELAY];
  176. } else if ([anim integerValue] < (4 * ANIMATION_SHADOW_HACK)) {
  177. [self performSelector:@selector(drawAnimation:) withObject:[NSNumber numberWithInteger:([anim integerValue] + 1)] afterDelay:ALARM_ANIMATION_DELAY_END];
  178. }
  179. [self invalidateShadow];
  180. self.mainView.needsDisplay = YES;
  181. }
  182. - (void)updateTime:(id)sender {
  183. [self performSelector:@selector(updateTime:) withObject:nil afterDelay:TIME_KEEPING_DELAY];
  184. // Check if a second went by
  185. NSDate *now = [NSDate date];
  186. NSDateComponents *components = [[NSCalendar currentCalendar] components:(NSHourCalendarUnit | NSMinuteCalendarUnit | NSSecondCalendarUnit) fromDate:now];
  187. static NSInteger lastSec = -1;
  188. if (lastSec == [components second]) {
  189. return;
  190. }
  191. lastSec = [components second];
  192. [[self.mainView render] drawWithDate:now];
  193. [[self.mainView render] blinkDots];
  194. if (self.alarmModeItem.state == NSOnState) {
  195. NSDateComponents *alarmComponents = [[NSCalendar currentCalendar] components:(NSHourCalendarUnit | NSMinuteCalendarUnit) fromDate:self.alarmDatePicker.dateValue];
  196. if (([components minute] == [alarmComponents minute]) && ([components hour] == [alarmComponents hour]) && ([components second] == 0)) {
  197. // We have reached the alarm time
  198. NSLog(@"Alarm time reached!");
  199. [self drawAnimation:[NSNumber numberWithInteger:1]];
  200. }
  201. }
  202. self.mainView.needsDisplay = YES;
  203. }
  204. - (void)blink {
  205. if (currentlyBlinking == NO) {
  206. currentlyBlinking = YES;
  207. [[self.mainView render] drawWithEye:EYE_BLINK];
  208. self.mainView.needsDisplay = YES;
  209. }
  210. [self performSelector:@selector(unblink) withObject:nil afterDelay:UNBLINK_DELAY];
  211. }
  212. - (void)unblink {
  213. if (currentlyBlinking == YES) {
  214. currentlyBlinking = NO;
  215. [[self.mainView render] drawWithEye:lastEyeState];
  216. self.mainView.needsDisplay = YES;
  217. }
  218. [self performSelector:@selector(blink) withObject:nil afterDelay:(((float)rand() / RAND_MAX) * MAX_BLINK_DELAY)];
  219. }
  220. - (IBAction)alarmDateSelected:(id)sender {
  221. // Update text representation
  222. NSDateComponents *components = [[NSCalendar currentCalendar] components:(NSHourCalendarUnit | NSMinuteCalendarUnit) fromDate:self.alarmDatePicker.dateValue];
  223. [self.alarmTextItem setTitle:[NSString stringWithFormat:@"%02ld:%02ld", (long)[components hour], (long)[components minute]]];
  224. if (self.alarmModeItem.state == NSOnState) {
  225. [[self.mainView render] drawAlarmDate:self.alarmDatePicker.dateValue];
  226. self.mainView.needsDisplay = YES;
  227. }
  228. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  229. [defaults setObject:self.alarmDatePicker.dateValue forKey:CONFIG_ALARM_TIME];
  230. [defaults synchronize];
  231. }
  232. - (IBAction)showDate:(id)sender {
  233. if (showDate == YES) {
  234. showDate = NO;
  235. self.showDateItem.state = NSOffState;
  236. } else {
  237. showDate = YES;
  238. self.showDateItem.state = NSOnState;
  239. }
  240. [[self.mainView render] drawDate:showDate];
  241. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  242. [defaults setBool:showDate forKey:CONFIG_SHOW_DATE];
  243. [defaults synchronize];
  244. }
  245. - (IBAction)changeSize:(NSMenuItem *)sender {
  246. NSRect frame = [self frame];
  247. [self.changeSize1 setState:NSOffState];
  248. [self.changeSize2 setState:NSOffState];
  249. [self.changeSize3 setState:NSOffState];
  250. [self.changeSize4 setState:NSOffState];
  251. [self.changeSize5 setState:NSOffState];
  252. [self.changeSize6 setState:NSOffState];
  253. [self.changeSize7 setState:NSOffState];
  254. [self.changeSize8 setState:NSOffState];
  255. [self.changeSize9 setState:NSOffState];
  256. [self.changeSize10 setState:NSOffState];
  257. BOOL found = NO;
  258. for (int i = RESIZE_START; i <= RESIZE_END; i += RESIZE_STEP) {
  259. NSString *title = [NSString stringWithFormat:@"x%d", i];
  260. if ([[sender title] isEqualToString:title]) {
  261. [sender setState:NSOnState];
  262. NSSize newSize = defaultSize;
  263. newSize.height *= i;
  264. newSize.width *= i;
  265. frame.size = newSize;
  266. found = YES;
  267. startScale = i;
  268. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  269. [defaults setInteger:i forKey:CONFIG_START_SCALE];
  270. [defaults synchronize];
  271. break;
  272. }
  273. }
  274. if (found == NO) {
  275. NSLog(@"Unknown changeSize sender: %@", sender);
  276. } else {
  277. [self setFrame:frame display:YES];
  278. }
  279. }
  280. - (IBAction)lockPosition:(NSMenuItem *)sender {
  281. BOOL state = [sender state];
  282. if (state == NSOffState) {
  283. // Lock position
  284. state = NSOnState;
  285. [sender setState:state];
  286. self.keepPosition = YES;
  287. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  288. [defaults setBool:YES forKey:CONFIG_KEEP_POSITION];
  289. [defaults synchronize];
  290. } else {
  291. // Unlock position
  292. state = NSOffState;
  293. [sender setState:state];
  294. self.keepPosition = NO;
  295. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  296. [defaults setBool:NO forKey:CONFIG_KEEP_POSITION];
  297. [defaults synchronize];
  298. }
  299. }
  300. - (IBAction)keepOnTop:(NSMenuItem *)sender {
  301. BOOL state = [sender state];
  302. if (state == NSOffState) {
  303. // Keep window on top
  304. state = NSOnState;
  305. [sender setState:state];
  306. [self setLevel:NSFloatingWindowLevel];
  307. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  308. [defaults setBool:YES forKey:CONFIG_KEEP_ON_TOP];
  309. [defaults synchronize];
  310. } else {
  311. // Don't keep window on top
  312. state = NSOffState;
  313. [sender setState:state];
  314. [self setLevel:NSNormalWindowLevel];
  315. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  316. [defaults setBool:NO forKey:CONFIG_KEEP_ON_TOP];
  317. [defaults synchronize];
  318. }
  319. }
  320. - (BOOL)canBecomeKeyWindow {
  321. return YES;
  322. }
  323. - (void)mouseDown:(NSEvent *)theEvent {
  324. dragStart = [theEvent locationInWindow];
  325. }
  326. - (void)mouseDragged:(NSEvent *)theEvent {
  327. NSRect screenVisibleFrame = [[NSScreen mainScreen] visibleFrame];
  328. NSRect windowFrame = [self frame];
  329. NSPoint newOrigin = windowFrame.origin;
  330. if (self.keepPosition == NO) {
  331. NSPoint currentLocation = [theEvent locationInWindow];
  332. newOrigin.x += (currentLocation.x - dragStart.x);
  333. newOrigin.y += (currentLocation.y - dragStart.y);
  334. if ((newOrigin.y + windowFrame.size.height) > (screenVisibleFrame.origin.y + screenVisibleFrame.size.height)) {
  335. newOrigin.y = screenVisibleFrame.origin.y + (screenVisibleFrame.size.height - windowFrame.size.height);
  336. }
  337. [self setFrameOrigin:newOrigin];
  338. }
  339. }
  340. - (void)mouseMoved:(NSEvent *)theEvent {
  341. NSPoint mousePoint = [NSEvent mouseLocation];
  342. mousePoint.x -= [self frame].origin.x;
  343. mousePoint.y -= [self frame].origin.y;
  344. BOOL top = (mousePoint.y > (MOUSE_CENTER_Y * startScale));
  345. BOOL right = (mousePoint.x > (MOUSE_CENTER_X * startScale));
  346. NSInteger eyeState = EYE_BOTTOM_LEFT;
  347. if (top && right) {
  348. eyeState = EYE_TOP_RIGHT;
  349. } else if (top && (!right)) {
  350. eyeState = EYE_TOP_LEFT;
  351. } else if ((!top) && right) {
  352. eyeState = EYE_BOTTOM_RIGHT;
  353. }
  354. if (eyeState != lastEyeState) {
  355. lastEyeState = eyeState;
  356. if (currentlyBlinking == NO) {
  357. [[self.mainView render] drawWithEye:lastEyeState];
  358. self.mainView.needsDisplay = YES;
  359. }
  360. }
  361. }
  362. @end