|
4 | 4 | #ifdef __APPLE__ |
5 | 5 |
|
6 | 6 | #import <Foundation/Foundation.h> |
| 7 | +#include <dispatch/dispatch.h> |
7 | 8 | #include <objc/runtime.h> |
| 9 | +#include <cmath> |
8 | 10 | #include "Timers.h" |
9 | 11 |
|
| 12 | +@interface NSTimerHandle : NSObject { |
| 13 | + @public |
| 14 | + NSTimer* timer; |
| 15 | + napi_env env; |
| 16 | + napi_ref callback; |
| 17 | +} |
| 18 | +@end |
| 19 | + |
| 20 | +@implementation NSTimerHandle |
| 21 | +- (void)dealloc { |
| 22 | + if (timer != nil) { |
| 23 | + [timer release]; |
| 24 | + timer = nil; |
| 25 | + } |
| 26 | + [super dealloc]; |
| 27 | +} |
| 28 | +@end |
| 29 | + |
| 30 | +namespace { |
| 31 | +const void* kTimerHandleAssociationKey = &kTimerHandleAssociationKey; |
| 32 | + |
| 33 | +void AddTimerToMainRunLoop(NSTimer* timer) { |
| 34 | + if (timer == nil) { |
| 35 | + return; |
| 36 | + } |
| 37 | + |
| 38 | + auto addTimer = ^{ |
| 39 | + if ([timer isValid]) { |
| 40 | + [[NSRunLoop mainRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode]; |
| 41 | + } |
| 42 | + }; |
| 43 | + |
| 44 | + if ([NSThread isMainThread]) { |
| 45 | + addTimer(); |
| 46 | + return; |
| 47 | + } |
| 48 | + |
| 49 | + dispatch_sync(dispatch_get_main_queue(), addTimer); |
| 50 | +} |
| 51 | + |
| 52 | +void DisposeTimerHandle(napi_env callEnv, NSTimerHandle* handle) { |
| 53 | + if (handle == nil) { |
| 54 | + return; |
| 55 | + } |
| 56 | + |
| 57 | + if (handle->timer != nil) { |
| 58 | + NSTimer* rawTimer = handle->timer; |
| 59 | + objc_setAssociatedObject(rawTimer, kTimerHandleAssociationKey, nil, OBJC_ASSOCIATION_ASSIGN); |
| 60 | + [rawTimer invalidate]; |
| 61 | + [rawTimer release]; |
| 62 | + handle->timer = nil; |
| 63 | + } |
| 64 | + |
| 65 | + napi_env cleanupEnv = callEnv != nullptr ? callEnv : handle->env; |
| 66 | + if (cleanupEnv != nullptr && handle->callback != nullptr) { |
| 67 | + napi_delete_reference(cleanupEnv, handle->callback); |
| 68 | + handle->callback = nullptr; |
| 69 | + } |
| 70 | +} |
| 71 | +} // namespace |
| 72 | + |
10 | 73 | namespace nativescript { |
11 | 74 |
|
12 | 75 | JS_CLASS_INIT(Timers::Init) { |
|
29 | 92 | napi_value argv[2]; |
30 | 93 | napi_get_cb_info(env, cbinfo, &argc, argv, nullptr, nullptr); |
31 | 94 |
|
32 | | - double ms; |
33 | | - napi_get_value_double(env, argv[1], &ms); |
| 95 | + double ms = 0; |
| 96 | + if (argc > 1) { |
| 97 | + napi_get_value_double(env, argv[1], &ms); |
| 98 | + } |
| 99 | + if (ms < 0 || !std::isfinite(ms)) { |
| 100 | + ms = 0; |
| 101 | + } |
34 | 102 |
|
35 | 103 | NSTimeInterval interval = ms / 1000; |
36 | 104 |
|
37 | 105 | napi_ref callback; |
38 | 106 | napi_create_reference(env, argv[0], 1, &callback); |
39 | 107 |
|
| 108 | + NSTimerHandle* handle = [[NSTimerHandle alloc] init]; |
| 109 | + handle->env = env; |
| 110 | + handle->callback = callback; |
| 111 | + |
40 | 112 | NSTimer* timer = [NSTimer |
41 | 113 | timerWithTimeInterval:interval |
42 | 114 | repeats:NO |
43 | 115 | block:^(NSTimer* timer) { |
44 | | - NapiScope scope(env); |
| 116 | + NSTimerHandle* handle = (NSTimerHandle*)objc_getAssociatedObject( |
| 117 | + timer, kTimerHandleAssociationKey); |
| 118 | + if (handle == nil || handle->callback == nullptr) { |
| 119 | + return; |
| 120 | + } |
| 121 | + [handle retain]; |
| 122 | + |
| 123 | + NapiScope scope(handle->env); |
45 | 124 | napi_value global, callbackValue; |
46 | | - napi_get_global(env, &global); |
47 | | - napi_get_reference_value(env, callback, &callbackValue); |
48 | | - napi_call_function(env, global, callbackValue, 0, nullptr, nullptr); |
49 | | - napi_reference_unref(env, callback, nullptr); |
50 | | - objc_setAssociatedObject(timer, "callback", nil, OBJC_ASSOCIATION_ASSIGN); |
| 125 | + napi_get_global(handle->env, &global); |
| 126 | + napi_get_reference_value(handle->env, handle->callback, &callbackValue); |
| 127 | + napi_call_function(handle->env, global, callbackValue, 0, nullptr, nullptr); |
| 128 | + |
| 129 | + napi_delete_reference(handle->env, handle->callback); |
| 130 | + handle->callback = nullptr; |
| 131 | + objc_setAssociatedObject(timer, kTimerHandleAssociationKey, nil, |
| 132 | + OBJC_ASSOCIATION_ASSIGN); |
| 133 | + if (handle->timer != nil) { |
| 134 | + [handle->timer release]; |
| 135 | + handle->timer = nil; |
| 136 | + } |
| 137 | + [handle release]; |
51 | 138 | }]; |
52 | 139 |
|
53 | | - objc_setAssociatedObject(timer, "callback", (id)callback, OBJC_ASSOCIATION_ASSIGN); |
| 140 | + handle->timer = [timer retain]; |
| 141 | + objc_setAssociatedObject(timer, kTimerHandleAssociationKey, handle, |
| 142 | + OBJC_ASSOCIATION_RETAIN_NONATOMIC); |
54 | 143 |
|
55 | 144 | napi_value result; |
56 | | - napi_create_external(env, timer, nullptr, nullptr, &result); |
| 145 | + napi_create_external( |
| 146 | + env, handle, |
| 147 | + [](napi_env env, void* data, void* /*hint*/) { |
| 148 | + NSTimerHandle* handle = (NSTimerHandle*)data; |
| 149 | + [handle release]; |
| 150 | + }, |
| 151 | + nullptr, &result); |
57 | 152 |
|
58 | | - [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode]; |
| 153 | + AddTimerToMainRunLoop(timer); |
59 | 154 |
|
60 | 155 | return result; |
61 | 156 | } |
|
65 | 160 | napi_value argv[2]; |
66 | 161 | napi_get_cb_info(env, cbinfo, &argc, argv, nullptr, nullptr); |
67 | 162 |
|
68 | | - double ms; |
69 | | - napi_get_value_double(env, argv[1], &ms); |
| 163 | + double ms = 0; |
| 164 | + if (argc > 1) { |
| 165 | + napi_get_value_double(env, argv[1], &ms); |
| 166 | + } |
| 167 | + if (ms < 0 || !std::isfinite(ms)) { |
| 168 | + ms = 0; |
| 169 | + } |
70 | 170 |
|
71 | 171 | NSTimeInterval interval = ms / 1000; |
72 | 172 |
|
73 | 173 | napi_ref callback; |
74 | 174 | napi_create_reference(env, argv[0], 1, &callback); |
75 | 175 |
|
| 176 | + NSTimerHandle* handle = [[NSTimerHandle alloc] init]; |
| 177 | + handle->env = env; |
| 178 | + handle->callback = callback; |
| 179 | + |
76 | 180 | NSTimer* timer = [NSTimer |
77 | 181 | timerWithTimeInterval:interval |
78 | 182 | repeats:YES |
79 | 183 | block:^(NSTimer* timer) { |
80 | | - NapiScope scope(env); |
| 184 | + NSTimerHandle* handle = (NSTimerHandle*)objc_getAssociatedObject( |
| 185 | + timer, kTimerHandleAssociationKey); |
| 186 | + if (handle == nil || handle->callback == nullptr) { |
| 187 | + return; |
| 188 | + } |
| 189 | + [handle retain]; |
| 190 | + |
| 191 | + NapiScope scope(handle->env); |
81 | 192 | napi_value global, callbackValue; |
82 | | - napi_get_global(env, &global); |
83 | | - napi_get_reference_value(env, callback, &callbackValue); |
84 | | - napi_call_function(env, global, callbackValue, 0, nullptr, nullptr); |
| 193 | + napi_get_global(handle->env, &global); |
| 194 | + napi_get_reference_value(handle->env, handle->callback, &callbackValue); |
| 195 | + napi_call_function(handle->env, global, callbackValue, 0, nullptr, nullptr); |
| 196 | + [handle release]; |
85 | 197 | }]; |
86 | 198 |
|
87 | | - objc_setAssociatedObject(timer, "callback", (id)callback, OBJC_ASSOCIATION_ASSIGN); |
| 199 | + handle->timer = [timer retain]; |
| 200 | + objc_setAssociatedObject(timer, kTimerHandleAssociationKey, handle, |
| 201 | + OBJC_ASSOCIATION_RETAIN_NONATOMIC); |
88 | 202 |
|
89 | 203 | napi_value result; |
90 | | - napi_create_external(env, timer, nullptr, nullptr, &result); |
| 204 | + napi_create_external( |
| 205 | + env, handle, |
| 206 | + [](napi_env env, void* data, void* /*hint*/) { |
| 207 | + NSTimerHandle* handle = (NSTimerHandle*)data; |
| 208 | + [handle release]; |
| 209 | + }, |
| 210 | + nullptr, &result); |
91 | 211 |
|
92 | | - [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode]; |
| 212 | + AddTimerToMainRunLoop(timer); |
93 | 213 |
|
94 | 214 | return result; |
95 | 215 | } |
|
105 | 225 | return nullptr; |
106 | 226 | } |
107 | 227 |
|
108 | | - NSTimer* t = nullptr; |
109 | | - napi_get_value_external(env, argv[0], (void**)&t); |
110 | | - |
111 | | - if (t != nullptr) { |
112 | | - [t invalidate]; |
113 | | - } |
114 | | - |
115 | | - napi_ref callback = (napi_ref)objc_getAssociatedObject(t, "callback"); |
116 | | - if (callback != nil) { |
117 | | - napi_delete_reference(env, callback); |
118 | | - } |
| 228 | + void* rawHandle = nullptr; |
| 229 | + napi_get_value_external(env, argv[0], &rawHandle); |
| 230 | + NSTimerHandle* handle = (NSTimerHandle*)rawHandle; |
| 231 | + DisposeTimerHandle(env, handle); |
119 | 232 |
|
120 | 233 | return nullptr; |
121 | 234 | } |
|
0 commit comments