-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathAugmentedMessage.swift
304 lines (245 loc) · 10.3 KB
/
AugmentedMessage.swift
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
//
// Copyright (c) 2018 ANONYMISED
//
// Permission is hereby granted, free of charge, to any person
// obtaining a copy of this software and associated documentation
// files (the "Software"), to deal in the Software without
// restriction, including without limitation the rights to use,
// copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the
// Software is furnished to do so, subject to the following
// conditions:
//
// The above copyright notice and this permission notice shall be
// included in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
// OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
// HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
// WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
// OTHER DEALINGS IN THE SOFTWARE.
import Cocoa
/// Contains everything that we need to represent a
/// message augmented by behavioural and gaze data.
/// Start and end times of events allow us to time-window
/// addionional physiological data to this message.
struct AugmentedMessage: DiMeData {
// MARK: - Constant fields
/// Type of this structure (useful when exporting)
var _type: String = "AugmentedMessage"
/// URL representation of type
/// It doesn't point to anything but can be used for hierarchical representation of types
var type: String = "http://www.hiit.fi/ontologies/dime/#AugmentedMessage"
// MARK: - User tags
/// 1 to 4, if set by user
private(set) var priority: Int = 0
/// 1 to 4, if set by user
private(set) var pleasure: Int = 0
/// 1 to 4, if set by user
private(set) var workload: Int = 0
/// If this message was maked as spam by user
private(set) var spam: Bool = false
/// If this message was marked as corrupted by the user
private(set) var corrupted: Bool = false
/// If this message was marked as eagerly expected by user
private(set) var eagerlyExpected = false
// MARK: - Mailbox-related fields
/// If the message was downloaded as unread from the mailbox
var wasUnread: Bool
/// String uniquely identifying this message
var appId: String
/// Content, or hash of content of message's body
var plainTextContent: String?
/// Subject, or hash of message's subject
var subject: String
/// Name (if any) or e-mail of sender, or hash of name or e-mail of sender
let fromString: String
/// Type-representation of sender
let from: Person?
/// Size of whole body, in view coordinates
let bodySize: NSSize
/// True if an attachment was present in the mailbox
let containsAttachment: Bool
/// This corresponds to Message.id
/// It may be useful to re-reference the original message later.
/// Commercial applications should not expose this for privacy reasons.
var id: String
// MARK: - Data
/// Timestamp when user started working on this message
let startUnixtime: Int
/// Timestap when user has done working on this message
var endUnixtime = 0
/// Fixations on body
var gazes: EyeData = EyeData.empty
/// Fixations on body, before the user started working on this message
/// (i.e. the user was working on another message, then read this one
/// before completing the previous)
var pre_gazes: EyeData?
/// Fixations on body, after the user completed this message.
var post_gazes: EyeData?
/// Array of start and end times of when the message was displayed on screen
/// while the user was working on this message.
var visits: [Event] = [Event]()
/// Then the message was displayed on screen, but before the user
/// started working on this message.
var pre_visits: [Event]?
/// Then the message was displayed on screen, after the user
/// completed work on this message.
var post_visits: [Event]?
/// Keywords extracted from fixations detected in body, when
/// the user was working on this message.
var keywords: [Keyword] = [Keyword]()
/// Keywords extracted from fixations detected in body, before
/// the user started working on this message.
var pre_keywords: [Keyword]?
/// Keywords extracted from fixations detected in body, after
/// the user completed work on this message.
var post_keywords: [Keyword]?
/// Selections made by user on message's body, when
/// the user was working on this message.
var selections: [Selection] = [Selection]()
/// Selections made by user on message's body, before
/// the user started working on this message.
var pre_selections: [Selection]?
/// Selections made by user on message's body, after
/// the user completed work on this message.
var post_selections: [Selection]?
/// Array of chunks of keyboard activity detected when
/// the user was working on this message (regardless of whether the window was front or not).
/// See `KeyboardEventTracker` for more information.
var keyboardActivity: [Double] = []
/// Array of chunks of mouse movement activity detected when
/// the user was working on this message (regardless of whether the window was front or not).
/// See `PointerMovementEventTracker` for more information.
var pointerActivity: [Double] = []
/// Array of chunks of mouse click activity detected when
/// the user was working on this message (regardless of whether the window was front or not).
/// See `PointerClickEventTracker` for more information.
var clickActivity: [Double] = []
/// Paired to keyboardActivity, references each chunk to timestamps
var keyboardTimes: [Event] = []
/// Paired to pointerActivity, references each chunk to timestamps
var pointerTimes: [Event] = []
/// Paired to clickActivity, references each chunk to timestamps
var clickTimes: [Event] = []
/// Unixtime of when the user first started typing a response in the reply box
var replyTime = -1
// MARK: - Convenience
/// Every time HistoryManager.completeAllAugmentedMessages()
/// is called and this message was updated, a Date() is appended here.
var dataUpdates: [Date] = []
// MARK: - Coding Keys
// we explicitly need these since we cannot
// have a field named `@type` in Swift
enum CodingKeys: String, CodingKey {
case _type = "@type"
case appId
case type
case plainTextContent
case subject
case fromString
case from
case priority
case pleasure
case id
case workload
case spam
case corrupted
case eagerlyExpected
case containsAttachment
case bodySize
case gazes
case wasUnread
case pre_gazes
case post_gazes
case startUnixtime
case endUnixtime
case dataUpdates
case visits
case pre_visits
case post_visits
case keywords
case pre_keywords
case post_keywords
case keyboardActivity
case pointerActivity
case clickActivity
case selections
case pre_selections
case post_selections
case replyTime
case keyboardTimes
case pointerTimes
case clickTimes
}
// MARK: - Init
init(fromMessage message: Message, bodySize: NSSize, preVisits: [Event]?, preGazes: EyeData?, preKeywords: [Keyword]?, preSelections: [Selection]?) {
let pte: String?
let subject: String
let fromString: String
if AppSingleton.hashEverything {
pte = nil
subject = message.subject.md5
fromString = message.senderName?.md5 ?? message.sender.md5
self.from = nil
} else {
pte = message.body
subject = message.subject
fromString = message.senderName ?? message.sender
if let sn = message.senderName, let p = Person(fromString: sn, email: message.sender) {
self.from = p
} else {
self.from = nil
}
}
self.id = message.id
self.appId = AugmentedMessage.makeAppId(id)
self.plainTextContent = pte
self.subject = subject
self.containsAttachment = message.containsAttachments
self.fromString = fromString
self.bodySize = bodySize
self.pre_visits = preVisits
self.pre_gazes = preGazes
self.pre_keywords = preKeywords
self.pre_selections = preSelections
self.wasUnread = HistoryManager.originallyUnreadIDs.contains(id)
startUnixtime = Date().unixTime
}
mutating func done(_ when: Date) {
endUnixtime = when.unixTime
}
mutating func addKeyword(_ kw: Keyword) {
guard kw.gazeDurations.count == 1 else {
fatalError("Keyword to add must have only one gaze")
}
if let kwi = keywords.index(of: kw) {
var newkw = keywords[kwi]
newkw.add(gazeDuration: kw.gazeDurations[0])
keywords[kwi] = newkw
} else {
keywords.append(kw)
}
}
/// Sets all user defined tags (fields) and reply time
mutating func setTags(priority: Int, pleasure: Int, workload: Int, spam: Bool, eager: Bool, corrupted: Bool) {
self.priority = priority
self.pleasure = pleasure
self.workload = workload
self.spam = spam
self.eagerlyExpected = eager
self.corrupted = corrupted
if let rtime = HistoryManager.replyTimes[id] {
self.replyTime = rtime
}
}
/// Fetches and resets behavioural data. Must be called only once, when definitely done working on this message.
mutating func fetchBehavioural() {
(keyboardActivity, keyboardTimes) = HistoryManager.keyboardTracker.reset()
(pointerActivity, pointerTimes) = HistoryManager.pointerMovementTracker.reset()
(clickActivity, clickTimes) = HistoryManager.pointerClickTracker.reset()
}
}