codeapp-js 0.2.2 → 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (176) hide show
  1. package/AI/codeapp.agent.md +105 -0
  2. package/AI/skills/connections/SKILL.md +47 -0
  3. package/AI/skills/dataverse/SKILL.md +99 -0
  4. package/AI/skills/environment-variables/SKILL.md +89 -0
  5. package/AI/skills/frontend-design/SKILL.md +34 -0
  6. package/AI/skills/jira/SKILL.md +81 -0
  7. package/AI/skills/office365-groups/SKILL.md +61 -0
  8. package/AI/skills/office365-outlook/SKILL.md +52 -0
  9. package/AI/skills/office365-users/SKILL.md +78 -0
  10. package/AI/skills/sharepoint/SKILL.md +77 -0
  11. package/AI/skills/sql/SKILL.md +85 -0
  12. package/AI/skills/start/SKILL.md +46 -0
  13. package/AI/skills/teams/SKILL.md +55 -0
  14. package/codeApp/.power/schemas/appschemas/dataSourcesInfo.ts +6275 -0
  15. package/codeApp/.power/schemas/jira/jira.Schema.json +6903 -0
  16. package/codeApp/.power/schemas/keyvault/keyvault.Schema.json +1600 -0
  17. package/{examples/combined demo/.power/schemas/office365groups/office365groups.Schema.json → codeApp/.power/schemas/office365groups/office365groups.Schema.json} +2203 -2203
  18. package/codeApp/.power/schemas/teams/teams.Schema.json +11112 -0
  19. package/codeApp/dist/codeapp.js +103 -1043
  20. package/codeApp/dist/connectors/azureKeyvault.js +459 -0
  21. package/codeApp/dist/connectors/jira.js +1247 -0
  22. package/codeApp/dist/connectors/office365groups.js +642 -0
  23. package/codeApp/dist/connectors/office365users.js +513 -0
  24. package/codeApp/dist/connectors/outlook.js +1393 -0
  25. package/{examples/kanban/dist → codeApp/dist/connectors}/sharepoint.js +466 -339
  26. package/codeApp/dist/connectors/sql.js +149 -0
  27. package/codeApp/dist/connectors/teams.js +280 -0
  28. package/codeApp/dist/index.js +1 -1
  29. package/codeApp/dist/power-apps-data.js +725 -176
  30. package/codeApp/src/generated/index.ts +12 -0
  31. package/codeApp/src/generated/models/AzureKeyVaultModel.ts +107 -0
  32. package/codeApp/src/generated/models/JiraModel.ts +501 -0
  33. package/codeApp/src/generated/services/AzureKeyVaultService.ts +257 -0
  34. package/codeApp/src/generated/services/JiraService.ts +1124 -0
  35. package/examples/{kanban → apps/kanban}/dist/dataverse.js +94 -94
  36. package/examples/{kanban → apps/kanban}/dist/index.css +605 -605
  37. package/examples/{kanban → apps/kanban}/dist/index.html +21 -21
  38. package/examples/{kanban → apps/kanban}/dist/index.js +860 -860
  39. package/examples/{kanban → apps/kanban}/dist/office365groups.js +97 -97
  40. package/examples/apps/kanban/dist/office365users.js +451 -0
  41. package/examples/{kanban → apps/kanban}/dist/outlook.js +162 -162
  42. package/examples/{planning Poker/dist/power-apps-data.js → apps/kanban/dist/power-apps-data.js} +2953 -2953
  43. package/{dev files/sharepoint.js → examples/apps/kanban/dist/sharepoint.js} +195 -99
  44. package/examples/{kanban → apps/kanban}/power.config.json +35 -35
  45. package/examples/{planning Poker → apps/planning Poker}/additional files/customizations (tables).xml +6428 -6428
  46. package/examples/{planning Poker → apps/planning Poker}/additional files/dataverse-tables.json +165 -165
  47. package/examples/{planning Poker → apps/planning Poker}/additional files/readme.md +122 -122
  48. package/examples/{planning Poker → apps/planning Poker}/dist/dataverse.js +78 -78
  49. package/examples/{planning Poker → apps/planning Poker}/dist/index.html +198 -198
  50. package/examples/{planning Poker → apps/planning Poker}/dist/index.js +954 -954
  51. package/examples/{kanban/dist/power-apps-data.js → apps/planning Poker/dist/power-apps-data.js } +2953 -2953
  52. package/examples/{planning Poker → apps/planning Poker}/dist/styles.css +815 -815
  53. package/examples/{planning Poker → apps/planning Poker}/power.config.json +50 -50
  54. package/examples/{solution explorer → apps/solution explorer}/dist/codeapp.js +1098 -1098
  55. package/examples/{solution explorer → apps/solution explorer}/dist/index.html +80 -80
  56. package/examples/{solution explorer → apps/solution explorer}/dist/index.js +735 -735
  57. package/examples/{solution explorer → apps/solution explorer}/dist/styles.css +571 -571
  58. package/examples/{solution explorer → apps/solution explorer}/power.config.json +150 -150
  59. package/examples/{todo → apps/todo}/dist/dataverse.js +64 -64
  60. package/examples/{todo → apps/todo}/dist/index.html +75 -75
  61. package/examples/{todo → apps/todo}/dist/index.js +8 -8
  62. package/examples/{todo → apps/todo}/dist/power-apps-data.js +2953 -2953
  63. package/examples/{todo → apps/todo}/dist/renderer.js +375 -375
  64. package/examples/{todo → apps/todo}/dist/styles.css +691 -691
  65. package/examples/{todo → apps/todo}/power.config.json +34 -34
  66. package/examples/combined demo/.power/schemas/appschemas/dataSourcesInfo.ts +6275 -7830
  67. package/examples/combined demo/.power/schemas/jira/jira.Schema.json +6903 -0
  68. package/examples/combined demo/.power/schemas/keyvault/keyvault.Schema.json +1600 -0
  69. package/examples/combined demo/.power/schemas/teams/teams.Schema.json +11112 -0
  70. package/examples/combined demo/dist/codeapp.js +394 -1098
  71. package/examples/combined demo/dist/index.html +29 -511
  72. package/examples/combined demo/dist/index.js +490 -470
  73. package/examples/combined demo/dist/office365users.js +513 -0
  74. package/examples/combined demo/dist/outlook.js +1393 -0
  75. package/examples/combined demo/dist/power-apps-data.js +3079 -3006
  76. package/examples/combined demo/dist/styles.css +483 -0
  77. package/examples/combined demo/power.config.json +33 -42
  78. package/examples/combined demo/src/generated/index.ts +12 -14
  79. package/examples/combined demo/src/generated/models/AzureKeyVaultModel.ts +107 -0
  80. package/examples/combined demo/src/generated/models/JiraModel.ts +501 -0
  81. package/examples/combined demo/src/generated/services/AzureKeyVaultService.ts +257 -0
  82. package/examples/combined demo/src/generated/services/JiraService.ts +1124 -0
  83. package/examples/dataverse Demo/dist/codeapp.js +394 -1085
  84. package/examples/{outlook Demo2/OutlookDemo_1_0_0_1.zip → dataverse Demo/dist/icon-512.png} +0 -0
  85. package/examples/dataverse Demo/dist/index.html +146 -54
  86. package/examples/dataverse Demo/dist/index.js +693 -83
  87. package/examples/dataverse Demo/dist/power-apps-data.js +3079 -2911
  88. package/examples/dataverse Demo/dist/styles.css +528 -0
  89. package/examples/dataverse Demo/power.config.json +41 -35
  90. package/examples/dataverse Demo/readme.md +79 -79
  91. package/examples/groups Demo/dist/codeapp.js +394 -1085
  92. package/examples/groups Demo/dist/icon-512.png +0 -0
  93. package/examples/groups Demo/dist/index.html +21 -25
  94. package/examples/groups Demo/dist/index.js +304 -113
  95. package/examples/groups Demo/dist/office365groups.js +642 -0
  96. package/examples/groups Demo/dist/power-apps-data.js +3079 -2911
  97. package/examples/groups Demo/dist/styles.css +509 -0
  98. package/examples/groups Demo/power.config.json +25 -25
  99. package/examples/myProfile/dist/codeapp.js +398 -0
  100. package/examples/myProfile/dist/index.html +21 -184
  101. package/examples/myProfile/dist/index.js +324 -141
  102. package/examples/myProfile/dist/office365users.js +517 -169
  103. package/examples/myProfile/dist/power-apps-data.js +3080 -2953
  104. package/examples/myProfile/dist/styles.css +458 -0
  105. package/examples/myProfile/power.config.json +24 -23
  106. package/examples/outlook Demo/dist/codeapp.js +394 -1085
  107. package/examples/outlook Demo/dist/index.html +150 -35
  108. package/examples/outlook Demo/dist/index.js +516 -170
  109. package/examples/outlook Demo/dist/outlook.js +1393 -121
  110. package/examples/outlook Demo/dist/power-apps-data.js +3079 -2911
  111. package/examples/outlook Demo/dist/styles.css +408 -84
  112. package/examples/outlook Demo/power.config.json +24 -23
  113. package/examples/outlook Demo/readme.md +92 -82
  114. package/examples/sharePoint Demo/dist/codeapp.js +394 -1085
  115. package/examples/sharePoint Demo/dist/icon-512.png +0 -0
  116. package/examples/sharePoint Demo/dist/index.html +22 -255
  117. package/examples/sharePoint Demo/dist/index.js +899 -262
  118. package/examples/sharePoint Demo/dist/power-apps-data.js +3079 -2911
  119. package/examples/sharePoint Demo/dist/sharepoint.js +466 -0
  120. package/examples/sharePoint Demo/dist/styles.css +587 -0
  121. package/examples/sharePoint Demo/power.config.json +23 -22
  122. package/package.json +1 -1
  123. package/readme.md +479 -61
  124. package/.github/instructions/wyattdave.instructions.md +0 -39
  125. package/.vscode/settings.json +0 -6
  126. package/dev files/dataverse.js +0 -105
  127. package/dev files/office365groups.js +0 -65
  128. package/dev files/office365users.js +0 -169
  129. package/dev files/outlook.js +0 -330
  130. package/examples/combined demo/.power/schemas/office365/office365.Schema.json +0 -21098
  131. package/examples/combined demo/.power/schemas/office365users/office365users.Schema.json +0 -2094
  132. package/examples/kanban/agent/decision-log.md +0 -9
  133. package/examples/kanban/agent/mockup-01-editorial-glass.html +0 -159
  134. package/examples/kanban/agent/mockup-02-dark-rail.html +0 -147
  135. package/examples/kanban/agent/mockup-03-paper-grid.html +0 -114
  136. package/examples/kanban/agent/mockup-04-neon-minimal.html +0 -141
  137. package/examples/kanban/agent/mockup-05-mono-architect.html +0 -119
  138. package/examples/kanban/dist/environmentVar.js +0 -55
  139. package/examples/kanban/dist/office365users.js +0 -169
  140. package/examples/kanban/src/generated/index.ts +0 -14
  141. package/examples/outlook Demo2/agent/decision-log.md +0 -7
  142. package/examples/outlook Demo2/dist/codeapp.js +0 -1334
  143. package/examples/outlook Demo2/dist/index.html +0 -98
  144. package/examples/outlook Demo2/dist/index.js +0 -346
  145. package/examples/outlook Demo2/dist/styles.css +0 -639
  146. package/examples/outlook Demo2/power.config.json +0 -23
  147. package/examples/planning Poker/.vscode/settings.json +0 -5
  148. package/examples/sharePoint Demo/agent/decision-log.md +0 -17
  149. package/examples/solution explorer/agent/decision-log.md +0 -27
  150. package/examples/solution explorer/agent/mockup-01-swiss-grid.html +0 -452
  151. package/examples/solution explorer/agent/mockup-02-dark-glass.html +0 -496
  152. package/examples/solution explorer/agent/mockup-03-paper-console.html +0 -510
  153. package/examples/solution explorer/agent/mockup-04-neon-noir.html +0 -546
  154. package/examples/solution explorer/agent/mockup-05-zen-garden.html +0 -534
  155. package/examples/solution explorer/dist/power-apps-data.js +0 -3007
  156. package/scripts/build-power-sdk.mjs +0 -69
  157. /package/{examples/kanban → codeApp}/src/generated/models/Office365GroupsModel.ts +0 -0
  158. /package/{examples/kanban → codeApp}/src/generated/models/Office365OutlookModel.ts +0 -0
  159. /package/{examples/kanban → codeApp}/src/generated/models/Office365UsersModel.ts +0 -0
  160. /package/{examples/kanban → codeApp}/src/generated/services/Office365GroupsService.ts +0 -0
  161. /package/{examples/kanban → codeApp}/src/generated/services/Office365OutlookService.ts +0 -0
  162. /package/{examples/kanban → codeApp}/src/generated/services/Office365UsersService.ts +0 -0
  163. /package/{dev files → examples/apps/kanban/dist}/environmentVar.js +0 -0
  164. /package/examples/{outlook Demo2 → apps/kanban}/src/generated/index.ts +0 -0
  165. /package/examples/{outlook Demo2 → apps/kanban}/src/generated/models/Office365GroupsModel.ts +0 -0
  166. /package/examples/{outlook Demo2 → apps/kanban}/src/generated/models/Office365OutlookModel.ts +0 -0
  167. /package/examples/{outlook Demo2 → apps/kanban}/src/generated/models/Office365UsersModel.ts +0 -0
  168. /package/examples/{outlook Demo2 → apps/kanban}/src/generated/services/Office365GroupsService.ts +0 -0
  169. /package/examples/{outlook Demo2 → apps/kanban}/src/generated/services/Office365OutlookService.ts +0 -0
  170. /package/examples/{outlook Demo2 → apps/kanban}/src/generated/services/Office365UsersService.ts +0 -0
  171. /package/examples/{planning Poker → apps/planning Poker}/additional files/AgilePoker_1_0_0_1.zip +0 -0
  172. /package/examples/{planning Poker → apps/planning Poker}/additional files/PokerTables_1_0_0_1.zip +0 -0
  173. /package/examples/{outlook Demo2 → apps/solution explorer}/dist/icon-512.png +0 -0
  174. /package/examples/{outlook Demo2 → apps/solution explorer}/dist/power-apps-data.js +0 -0
  175. /package/examples/{todo → apps/todo}/dist/icon192.png +0 -0
  176. /package/examples/{solution explorer → combined demo}/dist/icon-512.png +0 -0
@@ -1,470 +1,490 @@
1
-
2
- import { getMyProfile, listEmails, listMyGroups } from './codeapp.js';
3
- import { getClient } from './power-apps-data.js';
4
-
5
- // ── SendEmailV2 connector (not yet in codeapp.js) ─────────────
6
- const SEND_EMAIL_CANDIDATES = ['office365outlook', 'Office365Outlook', 'office365'];
7
- const SEND_EMAIL_APIS = {
8
- SendEmailV2: {
9
- path: '/{connectionId}/v2/Mail',
10
- method: 'POST',
11
- parameters: [
12
- { name: 'connectionId', in: 'path', required: true },
13
- { name: 'emailMessage', in: 'body', required: true }
14
- ]
15
- }
16
- };
17
-
18
- let oSendClient = null;
19
-
20
- function getSendClient() {
21
- if (!oSendClient) {
22
- let oSources = {};
23
- SEND_EMAIL_CANDIDATES.forEach((sName) => {
24
- oSources[sName] = {
25
- tableId: '',
26
- version: '',
27
- primaryKey: '',
28
- dataSourceType: 'Connector',
29
- apis: SEND_EMAIL_APIS
30
- };
31
- });
32
- oSendClient = getClient(oSources);
33
- }
34
- return oSendClient;
35
- }
36
-
37
- let eStatusMessage = null;
38
- let eProfileCard = null;
39
- let eProfileStatus = null;
40
- let eEmailList = null;
41
- let eGroupList = null;
42
- let eEmailCount = null;
43
- let eGroupCount = null;
44
- let eRefreshButton = null;
45
- let eScrollComposeButton = null;
46
- let eComposeForm = null;
47
- let eComposeTo = null;
48
- let eComposeSubject = null;
49
- let eComposeBody = null;
50
- let eSendButton = null;
51
-
52
- let oProfile = null;
53
- let aEmails = [];
54
- let aGroups = [];
55
-
56
- function getElement(sId) {
57
- return document.getElementById(sId);
58
- }
59
-
60
- function cacheDomElements() {
61
- eStatusMessage = getElement('statusMessage');
62
- eProfileCard = getElement('profileCard');
63
- eProfileStatus = getElement('profileStatus');
64
- eEmailList = getElement('emailList');
65
- eGroupList = getElement('groupList');
66
- eEmailCount = getElement('emailCount');
67
- eGroupCount = getElement('groupCount');
68
- eRefreshButton = getElement('refreshButton');
69
- eScrollComposeButton = getElement('scrollComposeButton');
70
- eComposeForm = getElement('composeForm');
71
- eComposeTo = getElement('composeTo');
72
- eComposeSubject = getElement('composeSubject');
73
- eComposeBody = getElement('composeBody');
74
- eSendButton = getElement('sendButton');
75
- }
76
-
77
- function escapeHtml(sValue) {
78
- let sText = String(sValue == null ? '' : sValue);
79
- return sText
80
- .replace(new RegExp('&', 'g'), '&')
81
- .replace(new RegExp('<', 'g'), '&lt;')
82
- .replace(new RegExp('>', 'g'), '&gt;')
83
- .replace(new RegExp('"', 'g'), '&quot;')
84
- .replace(new RegExp("'", 'g'), '&#39;');
85
- }
86
-
87
- function setStatus(sMessage, sTone = 'info') {
88
- if (!eStatusMessage) {
89
- return;
90
- }
91
- eStatusMessage.textContent = sMessage;
92
- eStatusMessage.className = 'statusBar ' + sTone;
93
- }
94
-
95
- function getInitials(sName) {
96
- let aParts = String(sName || 'U').trim().split(new RegExp('\\s+', 'g')).filter(Boolean);
97
- return aParts.slice(0, 2).map((sPart) => sPart.charAt(0).toUpperCase()).join('');
98
- }
99
-
100
- function normalizePayload(oValue) {
101
- if (oValue == null) {
102
- return null;
103
- }
104
-
105
- if (typeof oValue === 'string') {
106
- try {
107
- return JSON.parse(oValue);
108
- } catch (oErr) {
109
- return oValue;
110
- }
111
- }
112
-
113
- if (typeof oValue === 'object' && Object.prototype.hasOwnProperty.call(oValue, 'body')) {
114
- return normalizePayload(oValue.body);
115
- }
116
-
117
- if (typeof oValue === 'object' && Object.prototype.hasOwnProperty.call(oValue, 'value') && Array.isArray(oValue.value)) {
118
- return oValue;
119
- }
120
-
121
- return oValue;
122
- }
123
-
124
- function getArrayFromPayload(oValue) {
125
- let oPayload = normalizePayload(oValue);
126
-
127
- if (Array.isArray(oPayload)) {
128
- return oPayload;
129
- }
130
-
131
- if (oPayload && Array.isArray(oPayload.value)) {
132
- return oPayload.value;
133
- }
134
-
135
- if (oPayload && Array.isArray(oPayload.messages)) {
136
- return oPayload.messages;
137
- }
138
-
139
- if (oPayload && Array.isArray(oPayload.items)) {
140
- return oPayload.items;
141
- }
142
-
143
- return [];
144
- }
145
-
146
- function getProfileFromPayload(oValue) {
147
- let oPayload = normalizePayload(oValue);
148
- if (!oPayload || typeof oPayload !== 'object') {
149
- return {};
150
- }
151
- return oPayload.user && typeof oPayload.user === 'object' ? oPayload.user : oPayload;
152
- }
153
-
154
- function formatEmailDate(sValue) {
155
- if (!sValue) {
156
- return 'No date';
157
- }
158
-
159
- let oDate = new Date(sValue);
160
- if (Number.isNaN(oDate.getTime())) {
161
- return String(sValue);
162
- }
163
-
164
- return oDate.toLocaleString();
165
- }
166
-
167
- function getEmailFromValue(oValue) {
168
- if (!oValue || typeof oValue !== 'object') {
169
- return '';
170
- }
171
-
172
- if (typeof oValue.address === 'string') {
173
- return oValue.address;
174
- }
175
-
176
- if (typeof oValue.email === 'string') {
177
- return oValue.email;
178
- }
179
-
180
- if (oValue.emailAddress && typeof oValue.emailAddress.address === 'string') {
181
- return oValue.emailAddress.address;
182
- }
183
-
184
- if (oValue.EmailAddress && typeof oValue.EmailAddress.Address === 'string') {
185
- return oValue.EmailAddress.Address;
186
- }
187
-
188
- return '';
189
- }
190
-
191
- function getSenderLabel(oEmail) {
192
- let oFrom = oEmail.from || oEmail.From || oEmail.sender || oEmail.Sender || {};
193
- if (oFrom.emailAddress && typeof oFrom.emailAddress === 'object') {
194
- return oFrom.emailAddress.name || oFrom.emailAddress.address || 'Unknown sender';
195
- }
196
- if (oFrom.EmailAddress && typeof oFrom.EmailAddress === 'object') {
197
- return oFrom.EmailAddress.Name || oFrom.EmailAddress.Address || 'Unknown sender';
198
- }
199
- if (typeof oFrom.displayName === 'string') {
200
- return oFrom.displayName;
201
- }
202
- return getEmailFromValue(oFrom) || 'Unknown sender';
203
- }
204
-
205
- function getGroupTagsMarkup(oGroup) {
206
- let aTags = Array.isArray(oGroup.groupTypes) ? oGroup.groupTypes : [];
207
- return aTags.map((sTag) => '<span class="pill">' + escapeHtml(sTag) + '</span>').join('');
208
- }
209
-
210
- function renderProfile() {
211
- if (!eProfileCard) {
212
- return;
213
- }
214
-
215
- if (!oProfile || Object.keys(oProfile).length === 0) {
216
- eProfileCard.innerHTML = '<div class="emptyState">Profile information was not returned.</div>';
217
- if (eProfileStatus) {
218
- eProfileStatus.textContent = 'Unavailable';
219
- }
220
- return;
221
- }
222
-
223
- let sName = oProfile.displayName || oProfile.DisplayName || oProfile.name || 'Unknown user';
224
- let sEmail = oProfile.mail || oProfile.Mail || oProfile.userPrincipalName || oProfile.UserPrincipalName || '';
225
- let sJobTitle = oProfile.jobTitle || oProfile.JobTitle || 'No title';
226
- let sDepartment = oProfile.department || oProfile.Department || 'No department';
227
- let sPhone = oProfile.mobilePhone || oProfile.MobilePhone || '';
228
-
229
- eProfileCard.innerHTML = `
230
- <div class="profileBadge">${escapeHtml(getInitials(sName))}</div>
231
- <div>
232
- <h3 class="profileName">${escapeHtml(sName)}</h3>
233
- <p class="profileMeta">${escapeHtml(sJobTitle)} · ${escapeHtml(sDepartment)}</p>
234
- <p class="profileSubMeta">${escapeHtml(sEmail || 'No email')}</p>
235
- <p class="profileSubMeta">${escapeHtml(sPhone || 'No phone')}</p>
236
- </div>
237
- `;
238
-
239
- if (eProfileStatus) {
240
- eProfileStatus.textContent = 'Ready';
241
- }
242
- }
243
-
244
- function renderEmails() {
245
- if (!eEmailList) {
246
- return;
247
- }
248
-
249
- if (eEmailCount) {
250
- eEmailCount.textContent = String(aEmails.length);
251
- }
252
-
253
- if (!Array.isArray(aEmails) || aEmails.length === 0) {
254
- eEmailList.innerHTML = '<div class="emptyState">No emails were returned.</div>';
255
- return;
256
- }
257
-
258
- eEmailList.innerHTML = aEmails.map((oEmail) => {
259
- let sSubject = oEmail.subject || oEmail.Subject || '(No subject)';
260
- let sPreview = oEmail.bodyPreview || oEmail.BodyPreview || oEmail.body || oEmail.Body || '';
261
- let sReceived = oEmail.receivedDateTime || oEmail.DateTimeReceived || oEmail.createdDateTime || '';
262
- let sSender = getSenderLabel(oEmail);
263
-
264
- return `
265
- <article class="listCard">
266
- <h3 class="listTitle">${escapeHtml(sSubject)}</h3>
267
- <p class="listMeta">From: ${escapeHtml(sSender)} · ${escapeHtml(formatEmailDate(sReceived))}</p>
268
- <p class="listBody">${escapeHtml(String(sPreview).slice(0, 220) || 'No preview available.')}</p>
269
- </article>
270
- `;
271
- }).join('');
272
- }
273
-
274
- function renderGroups() {
275
- if (!eGroupList) {
276
- return;
277
- }
278
-
279
- if (eGroupCount) {
280
- eGroupCount.textContent = String(aGroups.length);
281
- }
282
-
283
- if (!Array.isArray(aGroups) || aGroups.length === 0) {
284
- eGroupList.innerHTML = '<div class="emptyState">No group memberships were returned.</div>';
285
- return;
286
- }
287
-
288
- eGroupList.innerHTML = aGroups.map((oGroup) => {
289
- let sName = oGroup.displayName || oGroup.DisplayName || 'Unnamed group';
290
- let sDescription = oGroup.description || oGroup.Description || 'No description';
291
- let sMail = oGroup.mail || oGroup.Mail || 'No group mailbox';
292
-
293
- return `
294
- <article class="listCard">
295
- <h3 class="listTitle">${escapeHtml(sName)}</h3>
296
- <p class="listMeta">${escapeHtml(sMail)}</p>
297
- <p class="listBody">${escapeHtml(sDescription)}</p>
298
- ${getGroupTagsMarkup(oGroup)}
299
- </article>
300
- `;
301
- }).join('');
302
- }
303
-
304
- async function loadProfile() {
305
- oProfile = getProfileFromPayload(await getMyProfile());
306
- renderProfile();
307
- }
308
-
309
- async function loadEmails() {
310
- aEmails = getArrayFromPayload(await listEmails({ folderId: 'Inbox', top: 15 }));
311
- renderEmails();
312
- }
313
-
314
- async function loadGroups() {
315
- aGroups = getArrayFromPayload(await listMyGroups());
316
- renderGroups();
317
- }
318
-
319
- async function sendEmailMessage(sTo, sSubject, sBody) {
320
- let client = getSendClient();
321
- let aErrors = [];
322
-
323
- for (let iIndex = 0; iIndex < SEND_EMAIL_CANDIDATES.length; iIndex += 1) {
324
- let sName = SEND_EMAIL_CANDIDATES[iIndex];
325
- try {
326
- let oResult = await client.executeAsync({
327
- connectorOperation: {
328
- tableName: sName,
329
- operationName: 'SendEmailV2',
330
- parameters: {
331
- emailMessage: {
332
- To: sTo,
333
- Subject: sSubject,
334
- Body: sBody,
335
- Importance: 'Normal',
336
- IsHtml: true
337
- }
338
- }
339
- }
340
- });
341
- if (oResult && oResult.success === false) {
342
- throw new Error(oResult.error ? (oResult.error.message || JSON.stringify(oResult.error)) : 'Send failed');
343
- }
344
- return oResult && Object.prototype.hasOwnProperty.call(oResult, 'data') ? oResult.data : oResult;
345
- } catch (oErr) {
346
- let sMessage = oErr.message || String(oErr);
347
- aErrors.push(sName + ': ' + sMessage);
348
- if (sMessage.indexOf('Connection reference not found') === -1) {
349
- throw oErr;
350
- }
351
- }
352
- }
353
-
354
- throw new Error('No Outlook connection reference matched. Tried: ' + aErrors.join(' || '));
355
- }
356
-
357
- async function loadDashboard() {
358
- setStatus('Loading your dashboard…', 'info');
359
-
360
- let aResults = await Promise.allSettled([
361
- loadProfile(),
362
- loadEmails(),
363
- loadGroups()
364
- ]);
365
-
366
- let aErrors = aResults
367
- .filter((oResult) => oResult.status === 'rejected')
368
- .map((oResult) => oResult.reason?.message || String(oResult.reason));
369
-
370
- renderProfile();
371
- renderEmails();
372
- renderGroups();
373
-
374
- if (aErrors.length > 0) {
375
- setStatus('Loaded with connector issues: ' + aErrors.join(' | '), 'error');
376
- return;
377
- }
378
-
379
- setStatus('Dashboard loaded successfully.', 'success');
380
- }
381
-
382
- async function handleRefreshClick() {
383
- await loadDashboard();
384
- }
385
-
386
- function handleScrollComposeClick() {
387
- let eComposeSection = getElement('composeSection');
388
- if (eComposeSection) {
389
- eComposeSection.scrollIntoView({ behavior: 'smooth', block: 'start' });
390
- }
391
- }
392
-
393
- async function handleComposeSubmit(oEvent) {
394
- oEvent.preventDefault();
395
-
396
- let sTo = String(eComposeTo?.value || '').trim();
397
- let sSubject = String(eComposeSubject?.value || '').trim();
398
- let sBody = String(eComposeBody?.value || '').trim();
399
-
400
- if (!sTo || !sSubject || !sBody) {
401
- setStatus('Complete the To, Subject, and Message fields before sending.', 'error');
402
- return;
403
- }
404
-
405
- try {
406
- if (eSendButton) {
407
- eSendButton.disabled = true;
408
- eSendButton.textContent = 'Sending...';
409
- }
410
-
411
- setStatus('Sending email…', 'info');
412
- await sendEmailMessage(sTo, sSubject, sBody);
413
-
414
- if (eComposeForm) {
415
- eComposeForm.reset();
416
- }
417
-
418
- setStatus('Email sent successfully.', 'success');
419
- await loadEmails();
420
- } catch (oErr) {
421
- setStatus('Send email failed: ' + (oErr.message || oErr), 'error');
422
- } finally {
423
- if (eSendButton) {
424
- eSendButton.disabled = false;
425
- eSendButton.textContent = 'Send email';
426
- }
427
- }
428
- }
429
-
430
- function attachEvents() {
431
- if (eRefreshButton) {
432
- eRefreshButton.addEventListener('click', () => {
433
- handleRefreshClick().catch((oErr) => {
434
- setStatus('Refresh failed: ' + (oErr.message || oErr), 'error');
435
- });
436
- });
437
- }
438
-
439
- if (eScrollComposeButton) {
440
- eScrollComposeButton.addEventListener('click', handleScrollComposeClick);
441
- }
442
-
443
- if (eComposeForm) {
444
- eComposeForm.addEventListener('submit', (oEvent) => {
445
- handleComposeSubmit(oEvent).catch((oErr) => {
446
- setStatus('Send email failed: ' + (oErr.message || oErr), 'error');
447
- });
448
- });
449
- }
450
- }
451
-
452
- async function boot() {
453
- cacheDomElements();
454
- attachEvents();
455
- renderProfile();
456
- renderEmails();
457
- renderGroups();
458
-
459
- try {
460
- await loadDashboard();
461
- } catch (oErr) {
462
- setStatus('App failed to start: ' + (oErr.message || oErr), 'error');
463
- }
464
- }
465
-
466
- window.addEventListener('DOMContentLoaded', () => {
467
- boot().catch((oErr) => {
468
- setStatus('App failed to start: ' + (oErr.message || oErr), 'error');
469
- });
470
- });
1
+ import { getMyProfile, getUserPhoto } from './office365users.js';
2
+ import { getCalendarView, listEmails } from './outlook.js';
3
+ import { enableDebugger } from "./codeapp.js";
4
+
5
+ enableDebugger();
6
+
7
+ const eRoot = document.getElementById('root');
8
+
9
+ const oState = {
10
+ bLoading: true,
11
+ bRefreshing: false,
12
+ oProfile: null,
13
+ sPhoto: '',
14
+ aEmails: [],
15
+ aMeetings: [],
16
+ aErrors: [],
17
+ sUpdatedAt: '',
18
+ };
19
+
20
+ function getDayWindow() {
21
+ const oNow = new Date();
22
+ const oStart = new Date(oNow.getFullYear(), oNow.getMonth(), oNow.getDate());
23
+ const oEnd = new Date(oNow.getFullYear(), oNow.getMonth(), oNow.getDate() + 1);
24
+
25
+ return {
26
+ oNow,
27
+ oStart,
28
+ oEnd,
29
+ sStartIso: oStart.toISOString(),
30
+ sEndIso: oEnd.toISOString(),
31
+ sLabel: oStart.toLocaleDateString([], {
32
+ weekday: 'long',
33
+ month: 'long',
34
+ day: 'numeric',
35
+ }),
36
+ };
37
+ }
38
+
39
+ function normalizeItems(oResult) {
40
+ if (Array.isArray(oResult)) {
41
+ return oResult;
42
+ }
43
+
44
+ if (oResult && Array.isArray(oResult.value)) {
45
+ return oResult.value;
46
+ }
47
+
48
+ return [];
49
+ }
50
+
51
+ function getErrorMessage(oError) {
52
+ if (!oError) {
53
+ return 'Unknown error';
54
+ }
55
+
56
+ if (typeof oError === 'string') {
57
+ return oError;
58
+ }
59
+
60
+ if (typeof oError.message === 'string' && oError.message) {
61
+ return oError.message;
62
+ }
63
+
64
+ try {
65
+ return JSON.stringify(oError);
66
+ } catch (oInnerError) {
67
+ return String(oError);
68
+ }
69
+ }
70
+
71
+ function escapeHtml(sValue) {
72
+ const eDiv = document.createElement('div');
73
+ eDiv.textContent = sValue || '';
74
+ return eDiv.innerHTML;
75
+ }
76
+
77
+ function stripHtml(sValue) {
78
+ const eDiv = document.createElement('div');
79
+ eDiv.innerHTML = sValue || '';
80
+ return eDiv.textContent || eDiv.innerText || '';
81
+ }
82
+
83
+ function getInitials(sName) {
84
+ if (!sName) {
85
+ return '?';
86
+ }
87
+
88
+ const aParts = String(sName)
89
+ .split(' ')
90
+ .map((sPart) => sPart.trim())
91
+ .filter(Boolean);
92
+
93
+ if (aParts.length === 0) {
94
+ return '?';
95
+ }
96
+
97
+ if (aParts.length === 1) {
98
+ return aParts[0].slice(0, 2).toUpperCase();
99
+ }
100
+
101
+ return (aParts[0].charAt(0) + aParts[aParts.length - 1].charAt(0)).toUpperCase();
102
+ }
103
+
104
+ function getEmailDate(oEmail) {
105
+ return oEmail.DateTimeReceived || oEmail.receivedDateTime || oEmail.ReceivedDateTime || '';
106
+ }
107
+
108
+ function getMeetingStart(oMeeting) {
109
+ return oMeeting.Start || oMeeting.start || oMeeting.startDateTime || '';
110
+ }
111
+
112
+ function getMeetingEnd(oMeeting) {
113
+ return oMeeting.End || oMeeting.end || oMeeting.endDateTime || '';
114
+ }
115
+
116
+ function toDate(oValue) {
117
+ if (!oValue) {
118
+ return null;
119
+ }
120
+
121
+ const oDate = new Date(oValue);
122
+ if (Number.isNaN(oDate.getTime())) {
123
+ return null;
124
+ }
125
+
126
+ return oDate;
127
+ }
128
+
129
+ function isInDay(oValue, oWindow) {
130
+ const oDate = toDate(oValue);
131
+ if (!oDate) {
132
+ return false;
133
+ }
134
+
135
+ return oDate >= oWindow.oStart && oDate < oWindow.oEnd;
136
+ }
137
+
138
+ function formatTime(oValue) {
139
+ const oDate = toDate(oValue);
140
+ if (!oDate) {
141
+ return '';
142
+ }
143
+
144
+ return oDate.toLocaleTimeString([], {
145
+ hour: 'numeric',
146
+ minute: '2-digit',
147
+ });
148
+ }
149
+
150
+ function formatUpdatedAt() {
151
+ return new Date().toLocaleTimeString([], {
152
+ hour: 'numeric',
153
+ minute: '2-digit',
154
+ });
155
+ }
156
+
157
+ function formatMeetingRange(oMeeting) {
158
+ if (oMeeting.IsAllDay || oMeeting.isAllDay) {
159
+ return 'All day';
160
+ }
161
+
162
+ const sStart = formatTime(getMeetingStart(oMeeting));
163
+ const sEnd = formatTime(getMeetingEnd(oMeeting));
164
+ if (!sStart && !sEnd) {
165
+ return 'Time unavailable';
166
+ }
167
+
168
+ return sStart + ' - ' + sEnd;
169
+ }
170
+
171
+ function renderMetaChip(sLabel, sValue) {
172
+ if (!sValue) {
173
+ return '';
174
+ }
175
+
176
+ return '<span class="meta-chip">' + escapeHtml(sLabel + ': ' + sValue) + '</span>';
177
+ }
178
+
179
+ function renderProfileCard(oWindow) {
180
+ if (!oState.oProfile) {
181
+ return '<div class="profile-card skeleton" style="min-height:130px"></div>';
182
+ }
183
+
184
+ const oProfile = oState.oProfile;
185
+ const sImageHtml = oState.sPhoto
186
+ ? '<img src="data:image/jpeg;base64,' + oState.sPhoto + '" alt="Profile photo" />'
187
+ : '<span>' + escapeHtml(getInitials(oProfile.displayName || oProfile.mail || 'Me')) + '</span>';
188
+
189
+ const aMeta = [
190
+ renderMetaChip('Mail', oProfile.mail || oProfile.userPrincipalName),
191
+ renderMetaChip('Phone', (oProfile.businessPhones && oProfile.businessPhones[0]) || oProfile.mobilePhone),
192
+ renderMetaChip('Department', oProfile.department),
193
+ renderMetaChip('Office', oProfile.officeLocation),
194
+ ].filter(Boolean);
195
+
196
+ return '<div class="profile-card">'
197
+ + '<div class="avatar-frame">' + sImageHtml + '</div>'
198
+ + '<div>'
199
+ + '<h2 class="profile-name">' + escapeHtml(oProfile.displayName || 'Signed-in user') + '</h2>'
200
+ + '<p class="profile-role">' + escapeHtml(oProfile.jobTitle || 'Role unavailable') + '</p>'
201
+ + '<p class="profile-line">' + escapeHtml(oWindow.sLabel) + '</p>'
202
+ + '<p class="profile-line">'
203
+ + escapeHtml((oProfile.mail || oProfile.userPrincipalName || '').toString())
204
+ + '</p>'
205
+ + '</div>'
206
+ + '</div>'
207
+ + '<div class="hero-meta">' + aMeta.join('') + '</div>';
208
+ }
209
+
210
+ function renderStatsCard() {
211
+ const sUpdatedCopy = oState.sUpdatedAt
212
+ ? 'Updated at ' + oState.sUpdatedAt
213
+ : 'Waiting for data';
214
+
215
+ return '<div class="stats-card">'
216
+ + '<div class="stats-row">'
217
+ + '<div class="stat-tile' + (oState.bLoading ? ' skeleton' : '') + '">'
218
+ + '<span class="stat-label">Emails Today</span>'
219
+ + '<span class="stat-value">' + String(oState.aEmails.length) + '</span>'
220
+ + '</div>'
221
+ + '<div class="stat-tile' + (oState.bLoading ? ' skeleton' : '') + '">'
222
+ + '<span class="stat-label">Meetings Today</span>'
223
+ + '<span class="stat-value">' + String(oState.aMeetings.length) + '</span>'
224
+ + '</div>'
225
+ + '</div>'
226
+ + '<div class="toolbar">'
227
+ + '<span class="status-copy">' + escapeHtml(sUpdatedCopy) + '</span>'
228
+ + '<button class="action-button" data-action="refresh"' + (oState.bRefreshing ? ' disabled' : '') + '>'
229
+ + (oState.bRefreshing ? 'Refreshing...' : 'Refresh')
230
+ + '</button>'
231
+ + '</div>'
232
+ + '</div>';
233
+ }
234
+
235
+ function renderEmailList() {
236
+ if (oState.bLoading && oState.aEmails.length === 0) {
237
+ return '<div class="item-list">'
238
+ + '<div class="list-item skeleton" style="min-height:110px"></div>'
239
+ + '<div class="list-item skeleton" style="min-height:110px"></div>'
240
+ + '<div class="list-item skeleton" style="min-height:110px"></div>'
241
+ + '</div>';
242
+ }
243
+
244
+ if (oState.aEmails.length === 0) {
245
+ return '<div class="empty-state">'
246
+ + '<h3 class="empty-title">No emails yet today</h3>'
247
+ + '<p class="empty-copy">Your inbox is clear so far, or newer mail has not arrived in the first inbox batch yet.</p>'
248
+ + '</div>';
249
+ }
250
+
251
+ return '<div class="item-list">'
252
+ + oState.aEmails.map((oEmail) => {
253
+ const oFrom = oEmail.From || oEmail.from || {};
254
+ const oEmailAddress = oFrom.EmailAddress || oFrom.emailAddress || {};
255
+ const sSender = oEmailAddress.Name || oEmailAddress.name || oFrom.Name || 'Unknown sender';
256
+ const sAddress = oEmailAddress.Address || oEmailAddress.address || oFrom.Address || '';
257
+ const sSubject = oEmail.Subject || oEmail.subject || '(No subject)';
258
+ const sPreview = stripHtml(oEmail.BodyPreview || oEmail.bodyPreview || oEmail.Body || oEmail.body || '');
259
+ const sReceived = formatTime(getEmailDate(oEmail));
260
+ const bUnread = oEmail.IsRead === false || oEmail.isRead === false;
261
+
262
+ return '<article class="list-item' + (bUnread ? ' list-item--unread' : '') + '">'
263
+ + '<div class="item-topline">'
264
+ + '<div>'
265
+ + '<h3 class="item-title">' + escapeHtml(sSubject) + '</h3>'
266
+ + '<p class="item-meta">' + escapeHtml(sSender + (sAddress ? ' • ' + sAddress : '')) + '</p>'
267
+ + '</div>'
268
+ + '<span class="item-time">' + escapeHtml(sReceived || 'Today') + '</span>'
269
+ + '</div>'
270
+ + '<p class="item-preview">' + escapeHtml(sPreview.slice(0, 180) || 'No preview available.') + '</p>'
271
+ + '</article>';
272
+ }).join('')
273
+ + '</div>';
274
+ }
275
+
276
+ function renderMeetingList() {
277
+ if (oState.bLoading && oState.aMeetings.length === 0) {
278
+ return '<div class="item-list">'
279
+ + '<div class="list-item skeleton" style="min-height:104px"></div>'
280
+ + '<div class="list-item skeleton" style="min-height:104px"></div>'
281
+ + '<div class="list-item skeleton" style="min-height:104px"></div>'
282
+ + '</div>';
283
+ }
284
+
285
+ if (oState.aMeetings.length === 0) {
286
+ return '<div class="empty-state">'
287
+ + '<h3 class="empty-title">No meetings on the calendar</h3>'
288
+ + '<p class="empty-copy">Today looks open. When events appear on your primary calendar, they will show up here.</p>'
289
+ + '</div>';
290
+ }
291
+
292
+ return '<div class="item-list">'
293
+ + oState.aMeetings.map((oMeeting) => {
294
+ const sSubject = oMeeting.Subject || oMeeting.subject || '(Untitled meeting)';
295
+ const sLocation = oMeeting.Location || oMeeting.location || 'Location not set';
296
+ const sOrganizer = oMeeting.Organizer || oMeeting.organizer || '';
297
+ const sRange = formatMeetingRange(oMeeting);
298
+ const sWebLink = oMeeting.WebLink || oMeeting.webLink || '';
299
+
300
+ return '<article class="list-item">'
301
+ + '<div class="item-topline">'
302
+ + '<div>'
303
+ + '<h3 class="item-title">' + escapeHtml(sSubject) + '</h3>'
304
+ + '<p class="item-meta">' + escapeHtml(sLocation) + '</p>'
305
+ + '</div>'
306
+ + '<span class="item-time">' + escapeHtml(sRange) + '</span>'
307
+ + '</div>'
308
+ + (sOrganizer ? '<p class="item-preview">Organizer: ' + escapeHtml(sOrganizer) + '</p>' : '')
309
+ + (sWebLink ? '<a class="meeting-link" href="' + escapeHtml(sWebLink) + '" target="_blank" rel="noreferrer">Open in Outlook</a>' : '')
310
+ + '</article>';
311
+ }).join('')
312
+ + '</div>';
313
+ }
314
+
315
+ function renderNotice() {
316
+ if (oState.aErrors.length === 0) {
317
+ return '';
318
+ }
319
+
320
+ return '<section class="notice">'
321
+ + '<h3 class="notice-title">Some data could not be loaded</h3>'
322
+ + '<p class="notice-copy">' + escapeHtml(oState.aErrors.join(' ')) + ' If this is a fresh app, use Sync Connections so Outlook and Office 365 Users are available.</p>'
323
+ + '</section>';
324
+ }
325
+
326
+ function renderApp() {
327
+ const oWindow = getDayWindow();
328
+
329
+ eRoot.innerHTML = '<div class="app-shell">'
330
+ + '<section class="hero-card">'
331
+ + '<div class="hero-layout">'
332
+ + '<div>'
333
+ + '<p class="eyebrow">Daily snapshot</p>'
334
+ + '<h1 class="hero-heading">Profile, mail, and meetings in one glance.</h1>'
335
+ + '<p class="hero-subtitle">A compact view of the signed-in user, the messages received today, and the meetings scheduled for the rest of the day.</p>'
336
+ + renderProfileCard(oWindow)
337
+ + '</div>'
338
+ + '<div class="hero-side">'
339
+ + renderStatsCard()
340
+ + '</div>'
341
+ + '</div>'
342
+ + '</section>'
343
+ + '<div class="content-grid">'
344
+ + '<section class="panel">'
345
+ + '<header class="panel-header">'
346
+ + '<div>'
347
+ + '<h2 class="panel-title">Today\'s Emails</h2>'
348
+ + '<p class="panel-subtitle">Latest inbox items received since midnight.</p>'
349
+ + '</div>'
350
+ + '<span class="count-pill">' + String(oState.aEmails.length) + '</span>'
351
+ + '</header>'
352
+ + renderEmailList()
353
+ + '</section>'
354
+ + '<section class="panel">'
355
+ + '<header class="panel-header">'
356
+ + '<div>'
357
+ + '<h2 class="panel-title">Today\'s Meetings</h2>'
358
+ + '<p class="panel-subtitle">Events from your primary calendar for ' + escapeHtml(oWindow.sLabel) + '.</p>'
359
+ + '</div>'
360
+ + '<span class="count-pill">' + String(oState.aMeetings.length) + '</span>'
361
+ + '</header>'
362
+ + renderMeetingList()
363
+ + '</section>'
364
+ + '</div>'
365
+ + renderNotice()
366
+ + '</div>';
367
+ }
368
+
369
+ async function loadProfileBundle() {
370
+ const oProfile = await getMyProfile({
371
+ select: ['displayName', 'mail', 'userPrincipalName', 'jobTitle', 'department', 'officeLocation', 'businessPhones', 'mobilePhone', 'id'],
372
+ });
373
+ const sUserId = oProfile.id || oProfile.mail || oProfile.userPrincipalName;
374
+ let sPhoto = '';
375
+
376
+ if (sUserId) {
377
+ try {
378
+ const oPhotoResult = await getUserPhoto(sUserId);
379
+ sPhoto = (oPhotoResult && oPhotoResult.value) || oPhotoResult || '';
380
+ } catch (oPhotoError) {
381
+ sPhoto = '';
382
+ }
383
+ }
384
+
385
+ return {
386
+ oProfile,
387
+ sPhoto,
388
+ };
389
+ }
390
+
391
+ async function loadTodayEmails() {
392
+ const oWindow = getDayWindow();
393
+ const oResult = await listEmails({ folderId: 'Inbox', top: 50 });
394
+
395
+ return normalizeItems(oResult)
396
+ .filter((oEmail) => isInDay(getEmailDate(oEmail), oWindow))
397
+ .sort((oLeft, oRight) => {
398
+ const iLeft = (toDate(getEmailDate(oLeft)) || new Date(0)).getTime();
399
+ const iRight = (toDate(getEmailDate(oRight)) || new Date(0)).getTime();
400
+ return iRight - iLeft;
401
+ })
402
+ .slice(0, 10);
403
+ }
404
+
405
+ async function loadTodayMeetings() {
406
+ const oWindow = getDayWindow();
407
+ const oResult = await getCalendarView({
408
+ calendarId: 'Calendar',
409
+ startDateTimeUtc: oWindow.sStartIso,
410
+ endDateTimeUtc: oWindow.sEndIso,
411
+ top: 20,
412
+ });
413
+
414
+ return normalizeItems(oResult)
415
+ .filter((oMeeting) => isInDay(getMeetingStart(oMeeting), oWindow))
416
+ .sort((oLeft, oRight) => {
417
+ const iLeft = (toDate(getMeetingStart(oLeft)) || new Date(0)).getTime();
418
+ const iRight = (toDate(getMeetingStart(oRight)) || new Date(0)).getTime();
419
+ return iLeft - iRight;
420
+ })
421
+ .slice(0, 20);
422
+ }
423
+
424
+ async function refreshDashboard() {
425
+ oState.bLoading = !oState.sUpdatedAt;
426
+ oState.bRefreshing = true;
427
+ oState.aErrors = [];
428
+ renderApp();
429
+
430
+ const aResults = await Promise.allSettled([
431
+ loadProfileBundle(),
432
+ loadTodayEmails(),
433
+ loadTodayMeetings(),
434
+ ]);
435
+
436
+ const [oProfileResult, oEmailResult, oMeetingResult] = aResults;
437
+ const aErrors = [];
438
+
439
+ if (oProfileResult.status === 'fulfilled') {
440
+ oState.oProfile = oProfileResult.value.oProfile;
441
+ oState.sPhoto = oProfileResult.value.sPhoto;
442
+ } else {
443
+ oState.oProfile = null;
444
+ oState.sPhoto = '';
445
+ aErrors.push('Profile: ' + getErrorMessage(oProfileResult.reason) + '.');
446
+ }
447
+
448
+ if (oEmailResult.status === 'fulfilled') {
449
+ oState.aEmails = oEmailResult.value;
450
+ } else {
451
+ oState.aEmails = [];
452
+ aErrors.push('Emails: ' + getErrorMessage(oEmailResult.reason) + '.');
453
+ }
454
+
455
+ if (oMeetingResult.status === 'fulfilled') {
456
+ oState.aMeetings = oMeetingResult.value;
457
+ } else {
458
+ oState.aMeetings = [];
459
+ aErrors.push('Meetings: ' + getErrorMessage(oMeetingResult.reason) + '.');
460
+ }
461
+
462
+ oState.aErrors = aErrors;
463
+ oState.bLoading = false;
464
+ oState.bRefreshing = false;
465
+ oState.sUpdatedAt = formatUpdatedAt();
466
+ renderApp();
467
+ }
468
+
469
+ function handleRootClick(oEvent) {
470
+ const eTarget = oEvent.target.closest('[data-action]');
471
+ if (!eTarget) {
472
+ return;
473
+ }
474
+
475
+ if (eTarget.dataset.action === 'refresh' && !oState.bRefreshing) {
476
+ refreshDashboard();
477
+ }
478
+ }
479
+
480
+ async function boot() {
481
+ if (!eRoot) {
482
+ return;
483
+ }
484
+
485
+ eRoot.addEventListener('click', handleRootClick);
486
+ renderApp();
487
+ await refreshDashboard();
488
+ }
489
+
490
+ boot();