@mx-cartographer/experiences 6.26.0 → 6.26.1-alpha.mega1

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 (84) hide show
  1. package/dist/AccountStore-DkL93tXL.mjs +317 -0
  2. package/dist/AccountStore-DkL93tXL.mjs.map +1 -0
  3. package/dist/BudgetUtil-CVqAdv-j.mjs +187 -0
  4. package/dist/BudgetUtil-CVqAdv-j.mjs.map +1 -0
  5. package/dist/CashflowStore-gPEEBiXo.mjs +53 -0
  6. package/dist/CashflowStore-gPEEBiXo.mjs.map +1 -0
  7. package/dist/CategoryStore-m7oieLzP.mjs +260 -0
  8. package/dist/CategoryStore-m7oieLzP.mjs.map +1 -0
  9. package/dist/ConnectMiniWidget-CQWafYtQ.mjs +22966 -0
  10. package/dist/ConnectMiniWidget-CQWafYtQ.mjs.map +1 -0
  11. package/dist/DebtsStore-cUhIgJx6.mjs +145 -0
  12. package/dist/DebtsStore-cUhIgJx6.mjs.map +1 -0
  13. package/dist/Donut-CGNFni9y.mjs +58 -0
  14. package/dist/Donut-CGNFni9y.mjs.map +1 -0
  15. package/dist/FinstrongStore-BDG7l0nI.mjs +183 -0
  16. package/dist/FinstrongStore-BDG7l0nI.mjs.map +1 -0
  17. package/dist/GoalStore-o2waGeyf.mjs +293 -0
  18. package/dist/GoalStore-o2waGeyf.mjs.map +1 -0
  19. package/dist/Loader-CwdK8lfx.mjs +147 -0
  20. package/dist/Loader-CwdK8lfx.mjs.map +1 -0
  21. package/dist/MerchantStore-Cck9IGIH.mjs +125 -0
  22. package/dist/MerchantStore-Cck9IGIH.mjs.map +1 -0
  23. package/dist/NetWorthStore-D5eUfGQt.mjs +66 -0
  24. package/dist/NetWorthStore-D5eUfGQt.mjs.map +1 -0
  25. package/dist/NotificationStore-DCxw8W7u.mjs +141 -0
  26. package/dist/NotificationStore-DCxw8W7u.mjs.map +1 -0
  27. package/dist/RecurringTransactionsStore-Bijrgllq.mjs +337 -0
  28. package/dist/RecurringTransactionsStore-Bijrgllq.mjs.map +1 -0
  29. package/dist/SettingsStore-BhOYgUOa.mjs +352 -0
  30. package/dist/SettingsStore-BhOYgUOa.mjs.map +1 -0
  31. package/dist/TransactionApi-Dnbyi1JL.mjs +175 -0
  32. package/dist/TransactionApi-Dnbyi1JL.mjs.map +1 -0
  33. package/dist/TransactionStore-CiEtt5XJ.mjs +417 -0
  34. package/dist/TransactionStore-CiEtt5XJ.mjs.map +1 -0
  35. package/dist/TrendsStore-Idq3QfoA.mjs +131 -0
  36. package/dist/TrendsStore-Idq3QfoA.mjs.map +1 -0
  37. package/dist/accounts/index.es.js +18 -0
  38. package/dist/accounts/index.es.js.map +1 -0
  39. package/dist/analytics/index.es.js +170 -0
  40. package/dist/analytics/index.es.js.map +1 -0
  41. package/dist/budgets/index.es.js +1116 -0
  42. package/dist/budgets/index.es.js.map +1 -0
  43. package/dist/cashflow/index.es.js +118 -0
  44. package/dist/cashflow/index.es.js.map +1 -0
  45. package/dist/categories/index.es.js +10 -0
  46. package/dist/categories/index.es.js.map +1 -0
  47. package/dist/common/context/hooks.d.ts +13 -13
  48. package/dist/common/index.es.js +1549 -0
  49. package/dist/common/index.es.js.map +1 -0
  50. package/dist/dashboard/index.es.js +257 -0
  51. package/dist/dashboard/index.es.js.map +1 -0
  52. package/dist/debts/index.es.js +9 -0
  53. package/dist/debts/index.es.js.map +1 -0
  54. package/dist/finstrong/index.es.js +11 -0
  55. package/dist/finstrong/index.es.js.map +1 -0
  56. package/dist/goals/index.es.js +9 -0
  57. package/dist/goals/index.es.js.map +1 -0
  58. package/dist/help/components/content/section/index.d.ts +15 -15
  59. package/dist/help/index.es.js +6 -0
  60. package/dist/help/index.es.js.map +1 -0
  61. package/dist/investments/index.es.js +1694 -0
  62. package/dist/investments/index.es.js.map +1 -0
  63. package/dist/merchants/index.es.js +7 -0
  64. package/dist/merchants/index.es.js.map +1 -0
  65. package/dist/microinsights/index.es.js +21 -0
  66. package/dist/microinsights/index.es.js.map +1 -0
  67. package/dist/networth/index.es.js +11 -0
  68. package/dist/networth/index.es.js.map +1 -0
  69. package/dist/notifications/index.es.js +116 -0
  70. package/dist/notifications/index.es.js.map +1 -0
  71. package/dist/recurringtransactions/index.es.js +14 -0
  72. package/dist/recurringtransactions/index.es.js.map +1 -0
  73. package/dist/settings/index.es.js +112 -0
  74. package/dist/settings/index.es.js.map +1 -0
  75. package/dist/spending/index.es.js +632 -0
  76. package/dist/spending/index.es.js.map +1 -0
  77. package/dist/transactions/index.es.js +19 -0
  78. package/dist/transactions/index.es.js.map +1 -0
  79. package/dist/trends/index.es.js +10 -0
  80. package/dist/trends/index.es.js.map +1 -0
  81. package/package.json +91 -3
  82. package/dist/index.d.ts +0 -22
  83. package/dist/index.es.js +0 -29907
  84. package/dist/index.es.js.map +0 -1
@@ -0,0 +1,417 @@
1
+ import { makeAutoObservable as I, runInAction as g } from "mobx";
2
+ import { endOfToday as T } from "date-fns/endOfToday";
3
+ import { formatISO as x } from "date-fns/formatISO";
4
+ import { fromUnixTime as w } from "date-fns/fromUnixTime";
5
+ import { getUnixTime as f } from "date-fns/getUnixTime";
6
+ import { isBefore as b } from "date-fns/isBefore";
7
+ import { startOfMonth as A } from "date-fns/startOfMonth";
8
+ import { startOfToday as E } from "date-fns/startOfToday";
9
+ import { subDays as O } from "date-fns/subDays";
10
+ import "bowser";
11
+ import "date-fns/intlFormat";
12
+ import { b1 as D, aK as N, b2 as L, aI as u, aJ as m, b3 as M, b4 as v, b5 as _, b6 as B } from "./ConnectMiniWidget-CQWafYtQ.mjs";
13
+ import "date-fns/addMonths";
14
+ import "date-fns/addQuarters";
15
+ import "date-fns/addWeeks";
16
+ import "date-fns/addYears";
17
+ import "date-fns/getDay";
18
+ import "date-fns/nextDay";
19
+ import "date-fns/setDate";
20
+ import "react";
21
+ import "numeral";
22
+ import "@mui/material";
23
+ import "react/jsx-runtime";
24
+ import "@mui/material/Button";
25
+ import "@mxenabled/mxui";
26
+ import "@mui/material/Tooltip";
27
+ import "@mui/material/Box";
28
+ import "@mui/material/IconButton";
29
+ import "./Loader-CwdK8lfx.mjs";
30
+ import "@mui/material/styles/useTheme";
31
+ import "@mui/material/useMediaQuery";
32
+ import "@mxenabled/cssinjs";
33
+ import "date-fns/addDays";
34
+ import "date-fns/eachDayOfInterval";
35
+ import "date-fns/eachWeekOfInterval";
36
+ import { endOfMonth as U } from "date-fns/endOfMonth";
37
+ import { format as F } from "date-fns/format";
38
+ import "date-fns/isSameMonth";
39
+ import "date-fns/subMonths";
40
+ import "@mui/material/Paper";
41
+ import "@mxenabled/mx-icons";
42
+ import "date-fns/isSameDay";
43
+ import "date-fns/isWithinInterval";
44
+ import "d3";
45
+ import "@mui/material/styles";
46
+ import "@mui/x-charts";
47
+ import "@mui/x-charts/ChartsTooltip";
48
+ import "@mui/material/Stack";
49
+ import "date-fns/differenceInCalendarDays";
50
+ import "date-fns/endOfYear";
51
+ import "date-fns/startOfYear";
52
+ import "date-fns/subYears";
53
+ import "@mui/material/ButtonGroup";
54
+ import "@mui/material/Tab";
55
+ import "@mui/material/Tabs";
56
+ import "@mui/material/ToggleButton";
57
+ import "@mui/material/ToggleButtonGroup";
58
+ import "@mui/material/Dialog";
59
+ import "@mui/material/DialogContent";
60
+ import "@mui/material/Card";
61
+ import "@mui/material/CardContent";
62
+ import "@mui/material/CardHeader";
63
+ import "uuid";
64
+ import "@mui/material/CircularProgress";
65
+ import "@mui/material/FormControlLabel";
66
+ import "@mui/material/Radio";
67
+ import "@mui/material/Avatar";
68
+ import "@mui/material/ListItem";
69
+ import "@mui/material/ListItemAvatar";
70
+ import "@mui/material/ListItemButton";
71
+ import "@mui/material/ListItemText";
72
+ import "@mui/material/ListItemIcon";
73
+ import "@mui/material/MenuItem";
74
+ import "@mui/material/TextField";
75
+ import "@mui/material/Switch";
76
+ import "posthog-js";
77
+ import "date-fns/differenceInDays";
78
+ import "@mui/x-data-grid-pro";
79
+ import "date-fns";
80
+ import "@mui/material/Divider";
81
+ import "@mui/material/List";
82
+ import { T as y } from "./TransactionApi-Dnbyi1JL.mjs";
83
+ import "date-fns/isAfter";
84
+ import "date-fns/parseISO";
85
+ import "date-fns/subQuarters";
86
+ import "date-fns/lastDayOfMonth";
87
+ import "date-fns/setDayOfYear";
88
+ import "date-fns/addSeconds";
89
+ import "date-fns/differenceInCalendarMonths";
90
+ import "date-fns/getMonth";
91
+ const R = {
92
+ ASC: "ASC",
93
+ DESC: "DESC",
94
+ DIRECTION: {
95
+ ASC: "ascending",
96
+ DESC: "descending"
97
+ },
98
+ ALIGN: {
99
+ center: "center",
100
+ left: "flex-start",
101
+ right: "flex-end"
102
+ }
103
+ }, G = async (c, t) => !t || !D[c] ? null : t.sendAnalyticEvent(D[c]), S = {
104
+ FLAG: "flag",
105
+ DATE: "date",
106
+ PAYEE: "payee",
107
+ CATEGORY: "category",
108
+ ACCOUNT: "account",
109
+ AMOUNT: "amount"
110
+ }, V = (c, t, i, s = []) => {
111
+ const e = t.reduce((a, o) => ({ ...a, [o.guid]: o }), {}), r = i.reduce((a, o) => ({ ...a, [o.guid]: o }), {}), n = s.reduce((a, o) => {
112
+ const { transaction_guid: l, tag_guid: d } = o;
113
+ if (!a[l])
114
+ return {
115
+ ...a,
116
+ [l]: [d]
117
+ };
118
+ const h = [...a[l], d];
119
+ return {
120
+ ...a,
121
+ [l]: h
122
+ };
123
+ }, {});
124
+ return c.map((a) => {
125
+ const o = e[a.account_guid], l = a.category_guid ? r[a.category_guid] : null, d = n[a.guid] ?? [];
126
+ return {
127
+ ...a,
128
+ account: o?.user_name || "Unknown",
129
+ accountIsClosed: o?.is_closed,
130
+ accountIsHidden: o?.is_hidden,
131
+ category: l?.name || "Uncategorized",
132
+ isIncome: a.top_level_category_guid === N.INCOME,
133
+ isPending: a.feed_status === L.PENDING,
134
+ number: o?.account_number || "",
135
+ payee: a.description,
136
+ tags: d
137
+ };
138
+ });
139
+ }, z = "uncategorized", P = (c, t, i, s, e, r) => {
140
+ const n = Y(c, e, r), a = k(n, i, s);
141
+ return t.forEach((o) => {
142
+ const l = a.findIndex((h) => h.guid === o), d = r.filter((h) => h.parent_guid === o);
143
+ a.splice(l + 1, 0, ...d);
144
+ }), a;
145
+ }, Y = (c, t, i) => {
146
+ const {
147
+ accounts: s = [],
148
+ dateRange: e = { start: A(E()), end: T() },
149
+ custom: r = () => !0,
150
+ searchTerm: n = ""
151
+ } = c;
152
+ return i.filter((a) => !a.parent_guid).filter((a) => !!s.find((o) => o === a.account_guid)).filter((a) => a.date >= f(e.start) && a.date <= f(e.end)).filter(r).filter((a) => {
153
+ if (n === z && a.has_been_split)
154
+ return !1;
155
+ if (/^spl/i.test(n) && (a.has_been_split || a.parent_guid))
156
+ return !0;
157
+ const l = a.tags.some(
158
+ (h) => p(n, t.find((C) => C.guid === h)?.name)
159
+ ), d = w(a.date);
160
+ return p(n, a.payee) || p(n, a.feed_description) || p(n, a.category) || p(n, u(d, m.MONTH)) || p(n, u(d, m.DAY)) || p(n, u(d, m.FULL_MONTH_DAY)) || p(n, u(d, m.MONTH_DAY_YEAR)) || p(n, F(d, M)) || p(n, a.account) || p(n, `${a.amount}`) || p(n, a.memo) || l;
161
+ });
162
+ }, k = (c, t, i) => [...c].sort((s, e) => {
163
+ const r = t, n = (s[r] || "") > (e[r] || "");
164
+ let a = n ? -1 : 1, o = n ? 1 : -1;
165
+ return (t === S.DATE || t === S.AMOUNT) && (a = n ? 1 : -1, o = n ? -1 : 1), i === R.ASC ? a : o;
166
+ }), p = (c, t) => {
167
+ const i = c.toLowerCase().split(" "), s = t?.toString().toLowerCase() ?? "";
168
+ return i.every((e) => s.indexOf(e) > -1);
169
+ };
170
+ class H {
171
+ beginDate = A(/* @__PURE__ */ new Date());
172
+ endDate = U(/* @__PURE__ */ new Date());
173
+ expandedSplits = [];
174
+ isInitialized = !1;
175
+ height = 550;
176
+ width = 1e3;
177
+ shouldDisableDrawerScroll = !1;
178
+ searchValue = "";
179
+ selectedAccountGuids = [];
180
+ manualTransaction = v;
181
+ isAmountValid = !1;
182
+ showInsights = !1;
183
+ constructor() {
184
+ I(this);
185
+ }
186
+ setDateRange = (t, i) => {
187
+ this.beginDate = t, this.endDate = i;
188
+ };
189
+ setIsInitialized = (t) => this.isInitialized = t;
190
+ setIsAmountValid = (t) => this.isAmountValid = t;
191
+ setSearchValue = (t) => this.searchValue = t;
192
+ setSelectedAccountGuids = (t) => this.selectedAccountGuids = t;
193
+ setShouldDisableDrawerScroll = (t) => this.shouldDisableDrawerScroll = t;
194
+ setShowInsights = (t) => {
195
+ this.showInsights = t;
196
+ };
197
+ setSize = (t, i) => {
198
+ this.height = t, this.width = i;
199
+ };
200
+ toggleSplit = (t) => {
201
+ const i = [...this.expandedSplits], s = this.expandedSplits.findIndex((e) => e === t);
202
+ s >= 0 ? i.splice(s, 1) : i.push(t), this.expandedSplits = i;
203
+ };
204
+ clearExpandedSplits = () => {
205
+ this.expandedSplits.length !== 0 && (this.expandedSplits = []);
206
+ };
207
+ updateManualTransaction = (t) => {
208
+ this.manualTransaction = t;
209
+ };
210
+ }
211
+ class Ci {
212
+ globalStore;
213
+ uiStore;
214
+ api = new y("/", "");
215
+ beatApi = new _("/", "");
216
+ associatedBeats = [];
217
+ cachedStartDate = O(E(), 90);
218
+ filter = {};
219
+ isLoading = !0;
220
+ rawTransactions = [];
221
+ sortColumn = S.DATE;
222
+ sortDirection = R.DESC;
223
+ taggings = [];
224
+ tags = [];
225
+ transactionRules = [];
226
+ onAnalyticEvent = G;
227
+ constructor(t) {
228
+ this.globalStore = t, this.uiStore = new H(), this.api = new y(t.endpoint, t.sessionToken), this.beatApi = new _(t.endpoint, t.sessionToken), I(this);
229
+ }
230
+ loadTransactionData = async ({
231
+ associatedBeats: t,
232
+ transactionRules: i,
233
+ transactions: s
234
+ } = {}) => {
235
+ await Promise.all([
236
+ this.loadAssociatedBeats(A(/* @__PURE__ */ new Date()), t),
237
+ this.loadTags(),
238
+ this.loadTaggings(),
239
+ this.loadTransactionRules(i),
240
+ // By default loads 90 days of transactions
241
+ this.loadTransactions(this.cachedStartDate, T(), s)
242
+ ]), g(() => {
243
+ this.uiStore.setIsInitialized(!0);
244
+ });
245
+ };
246
+ get sortedTransactions() {
247
+ return P(
248
+ this.filter,
249
+ this.uiStore.expandedSplits,
250
+ this.sortColumn,
251
+ this.sortDirection,
252
+ this.tags,
253
+ this.transactions
254
+ );
255
+ }
256
+ get sortedTransactionsWithSplits() {
257
+ return B(this.sortedTransactions, this.transactions);
258
+ }
259
+ get transactions() {
260
+ return this.augmentTransactions(this.rawTransactions);
261
+ }
262
+ addTag = async (t) => {
263
+ await this.api.addTag({ name: t }), await this.loadTags();
264
+ };
265
+ addTagging = async (t, i) => {
266
+ await this.api.addTagging({
267
+ tag_guid: t,
268
+ transaction_guid: i
269
+ });
270
+ };
271
+ addOrUpdateTransactionRule = async (t, i) => {
272
+ const s = this.transactionRules.find(
273
+ (e) => e.guid === i.user_transaction_rule_guid
274
+ );
275
+ s ? await this.api.updateTransactionRule({
276
+ ...s,
277
+ category_guid: t,
278
+ description: i.description
279
+ }) : await this.api.addTransactionRule({
280
+ category_guid: t,
281
+ description: i.description,
282
+ match_description: i.description
283
+ }), await this.loadTransactionRules();
284
+ };
285
+ addTransaction = async (t) => {
286
+ const i = await this.api.addTransaction(t);
287
+ i && g(() => this.rawTransactions = [i, ...this.transactions]);
288
+ };
289
+ augmentTransactions = (t) => V(
290
+ t,
291
+ this.globalStore.accountStore.visibleAccounts,
292
+ this.globalStore.categoryStore.categories,
293
+ this.taggings
294
+ );
295
+ getTransactionData = () => ({});
296
+ loadAssociatedBeats = async (t, i) => {
297
+ const s = t ? x(t, { representation: "date" }) : void 0, e = i ?? await this.beatApi.getBeatsWithAssociatedTransaction(s);
298
+ g(() => this.associatedBeats = e);
299
+ };
300
+ loadTags = async () => {
301
+ const t = await this.api.getTags();
302
+ g(() => this.tags = t);
303
+ };
304
+ loadTaggings = async () => {
305
+ const t = await this.api.getTaggings();
306
+ g(() => this.taggings = t);
307
+ };
308
+ loadTransactionRules = async (t) => {
309
+ const i = t ?? await this.api.getTransactionRules();
310
+ g(() => this.transactionRules = i);
311
+ };
312
+ loadTransactions = async (t = this.cachedStartDate, i = T(), s) => {
313
+ this.setIsLoading(!0), this.rawTransactions = [];
314
+ try {
315
+ const e = s ?? await this.api.getTransactionsByDateRange(f(t), f(i));
316
+ g(() => {
317
+ this.rawTransactions = e;
318
+ });
319
+ } catch (e) {
320
+ console.error(`Error occurred while loading transactions: ${e}`);
321
+ }
322
+ this.setIsLoading(!1);
323
+ };
324
+ // DEPRECATED
325
+ loadTransactionsByDateRange = async (t, i) => {
326
+ console.warn("DEPRECATED: loadTransactionsByDateRange will be removed"), this.setFilter({
327
+ ...this.filter,
328
+ dateRange: { start: w(i), end: w(t) }
329
+ });
330
+ };
331
+ modifyTransactionByAccountGuid = (t, i) => {
332
+ this.rawTransactions = this.transactions.map(
333
+ (s) => s.account_guid === t ? { ...s, ...i } : s
334
+ );
335
+ };
336
+ removeTag = async (t) => {
337
+ await this.api.removeTag(t), await this.loadTags();
338
+ };
339
+ removeTagging = async (t) => {
340
+ await this.api.removeTagging(t);
341
+ };
342
+ removeTransaction = async (t) => {
343
+ await this.api.removeTransaction(t);
344
+ const i = this.transactions.findIndex((s) => s.guid === t);
345
+ if (i >= 0) {
346
+ const s = [...this.transactions];
347
+ s.splice(i, 1), g(() => this.rawTransactions = s);
348
+ }
349
+ };
350
+ setFilter = (t) => {
351
+ this.filter = t, t.dateRange && b(t.dateRange.start, this.cachedStartDate) && this.loadTransactions(t.dateRange.start, T()).then(() => {
352
+ t.dateRange && (this.cachedStartDate = t.dateRange.start);
353
+ });
354
+ };
355
+ setIsLoading = (t) => this.isLoading = t;
356
+ setSortData = (t, i) => {
357
+ this.sortColumn = t, this.sortDirection = i;
358
+ };
359
+ splitTransaction = async (t, i) => {
360
+ const s = await this.api.updateTransaction(t), e = [];
361
+ for (let n = 0; n < i.length; n++) {
362
+ const a = await this.api.addTransaction(i[n]);
363
+ e.push(a);
364
+ }
365
+ const r = this.transactions.findIndex((n) => n.guid === t.guid);
366
+ r >= 0 && g(
367
+ () => this.rawTransactions = [
368
+ ...this.transactions.slice(0, r),
369
+ s,
370
+ ...this.transactions.slice(r + 1),
371
+ ...e
372
+ ]
373
+ );
374
+ };
375
+ unSplitTransaction = async (t) => {
376
+ const i = this.transactions.filter((n) => n.parent_guid === t);
377
+ await this.api.removeTransactionSplit(t);
378
+ const s = [...this.transactions];
379
+ i.forEach((n) => {
380
+ const a = s.findIndex((o) => o.guid === n.guid);
381
+ a >= 0 && s.splice(a, 1);
382
+ });
383
+ const e = s.findIndex((n) => n.guid === t), r = s[e];
384
+ r.has_been_split = !1, e >= 0 && g(
385
+ () => this.rawTransactions = [
386
+ ...s.slice(0, e),
387
+ r,
388
+ ...s.slice(e + 1)
389
+ ]
390
+ );
391
+ };
392
+ updateTaggings = async (t, i) => {
393
+ const s = this.taggings.filter((n) => n.transaction_guid === i), e = new Set(s.map((n) => n.tag_guid)), r = new Set(t);
394
+ for (const n of t)
395
+ e.has(n) || await this.addTagging(n, i);
396
+ for (const n of s)
397
+ r.has(n.tag_guid) || await this.removeTagging(n.guid);
398
+ await this.loadTaggings();
399
+ };
400
+ updateTag = async (t) => {
401
+ await this.api.updateTag(t), await this.loadTags();
402
+ };
403
+ updateTransaction = async (t) => {
404
+ const i = await this.api.updateTransaction(t), s = this.transactions.findIndex((e) => e.guid === t.guid);
405
+ s >= 0 && g(
406
+ () => this.rawTransactions = [
407
+ ...this.transactions.slice(0, s),
408
+ i,
409
+ ...this.transactions.slice(s + 1)
410
+ ]
411
+ );
412
+ };
413
+ }
414
+ export {
415
+ Ci as T
416
+ };
417
+ //# sourceMappingURL=TransactionStore-CiEtt5XJ.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TransactionStore-CiEtt5XJ.mjs","sources":["../src/common/constants/Table.ts","../src/transactions/analytics.ts","../src/transactions/constants/TransactionsTable.ts","../src/transactions/utils/buildDetailedTransactions.ts","../src/transactions/utils/sortAndFilterTransactions.ts","../src/transactions/stores/UiStore.ts","../src/transactions/stores/TransactionStore.ts"],"sourcesContent":["export const TABLE_CONST = {\n ASC: 'ASC',\n DESC: 'DESC',\n DIRECTION: {\n ASC: 'ascending',\n DESC: 'descending',\n },\n ALIGN: {\n center: 'center',\n left: 'flex-start',\n right: 'flex-end',\n },\n}\n","import { analyticsSession } from '../common'\nimport { AnalyticsEventsObjectType } from '../common/types/Analytics'\n\nimport { ANALYTICS_EVENTS } from './constants/Analytics'\n\nexport type AnalyticSession = Awaited<ReturnType<typeof analyticsSession>>\n\nexport const onAnalyticEvent = async (event: string, session: AnalyticSession | null) => {\n if (!session || !ANALYTICS_EVENTS[event]) return null\n\n return session.sendAnalyticEvent(ANALYTICS_EVENTS[event] as AnalyticsEventsObjectType)\n}\n","export const ROW_HEIGHT = 48\n\nconst FLAG_COLUMN_WIDTH = 24\nconst DATE_COLUMN_WIDTH = 64\nconst AMOUNT_COLUMN_WIDTH = 112\nconst VARIABLE_COLUMN_WIDTH = `calc((100% - ${\n FLAG_COLUMN_WIDTH + DATE_COLUMN_WIDTH + AMOUNT_COLUMN_WIDTH\n}px) / 3)`\n\nexport const COLUMN_WIDTH = {\n FLAG: FLAG_COLUMN_WIDTH, // Fixed width\n DATE: DATE_COLUMN_WIDTH, // Fixed width\n PAYEE: VARIABLE_COLUMN_WIDTH, // 1/3 of remaining width\n CATEGORY: VARIABLE_COLUMN_WIDTH, // 1/3 of remaining width\n ACCOUNT: VARIABLE_COLUMN_WIDTH, // 1/3 of remaining width\n AMOUNT: AMOUNT_COLUMN_WIDTH, // Fixed width\n}\n\nexport const COLUMN_NAMES = {\n FLAG: 'flag',\n DATE: 'date',\n PAYEE: 'payee',\n CATEGORY: 'category',\n ACCOUNT: 'account',\n AMOUNT: 'amount',\n}\n","import { CategoryGuids, TransactionStatus } from '../../common'\n\nimport type { Account, Category, Tagging, Transaction } from '../../common'\n\ninterface AccountsMap {\n [key: string]: Account\n}\n\ninterface CategoriesMap {\n [key: string]: Category\n}\n\ninterface TaggingsMap {\n [key: string]: string[]\n}\n\nexport const buildDetailedTransactions = (\n transactions: Transaction[],\n accounts: Account[],\n categories: Category[],\n taggings: Tagging[] = [],\n) => {\n const accountsMap = accounts.reduce((acc: AccountsMap, account: Account) => {\n return { ...acc, [account.guid]: account }\n }, {})\n\n const categoriesMap = categories.reduce((acc: CategoriesMap, category: Category) => {\n return { ...acc, [category.guid]: category }\n }, {})\n\n const taggingsMap = taggings.reduce((acc: TaggingsMap, tagging: Tagging) => {\n const { transaction_guid, tag_guid } = tagging\n\n if (!acc[transaction_guid]) {\n return {\n ...acc,\n [transaction_guid]: [tag_guid],\n }\n }\n const tags = [...acc[transaction_guid], tag_guid] as string[]\n\n return {\n ...acc,\n [transaction_guid]: tags,\n }\n }, {})\n\n return transactions.map((transaction: Transaction) => {\n const account = accountsMap[transaction.account_guid]\n const category = transaction.category_guid ? categoriesMap[transaction.category_guid] : null\n const tags = taggingsMap[transaction.guid] ?? []\n\n return {\n ...transaction,\n account: account?.user_name || 'Unknown',\n accountIsClosed: account?.is_closed,\n accountIsHidden: account?.is_hidden,\n category: category?.name || 'Uncategorized',\n isIncome: transaction.top_level_category_guid === CategoryGuids.INCOME,\n isPending: transaction.feed_status === TransactionStatus.PENDING,\n number: account?.account_number || '',\n payee: transaction.description,\n tags,\n }\n })\n}\n","import { endOfToday } from 'date-fns/endOfToday'\nimport { format } from 'date-fns/format'\nimport { fromUnixTime } from 'date-fns/fromUnixTime'\nimport { getUnixTime } from 'date-fns/getUnixTime'\nimport { startOfMonth } from 'date-fns/startOfMonth'\nimport { startOfToday } from 'date-fns/startOfToday'\n\nimport { TABLE_CONST } from '../../common/constants/Table'\nimport { DATE_FORMATS_INTL, formatDate, ORDINAL_DATE_FORMAT, Tag, Transaction } from '../../common'\n\nimport { COLUMN_NAMES } from '../constants/TransactionsTable'\nimport { type TransactionFilter } from '../stores/TransactionStore'\n\nconst UNCATEGORIZED_TERM = 'uncategorized'\n\nexport const sortAndFilterTransactions = (\n filter: TransactionFilter,\n expandedSplits: string[],\n sortColumn: string,\n sortDirection: string,\n tags: Tag[],\n transactions: Transaction[],\n) => {\n const filteredTransactions = filterTransactions(filter, tags, transactions)\n\n const sortedTransactions = sortTransactions(filteredTransactions, sortColumn, sortDirection)\n\n expandedSplits.forEach((guid) => {\n const index = sortedTransactions.findIndex((t) => t.guid === guid)\n const children = transactions.filter((t) => t.parent_guid === guid)\n sortedTransactions.splice(index + 1, 0, ...children)\n })\n return sortedTransactions\n}\n\nconst filterTransactions = (\n filter: TransactionFilter,\n tags: Tag[],\n transactions: Transaction[],\n) => {\n const {\n accounts = [],\n dateRange = { start: startOfMonth(startOfToday()), end: endOfToday() },\n custom = () => true,\n searchTerm: searchFilter = '',\n } = filter\n\n return (\n transactions\n // Filters out Split transaction children\n .filter((t) => !t.parent_guid)\n // Filter for selected accounts\n .filter((t) => Boolean(accounts.find((guid) => guid === t.account_guid)))\n // Filter for Date Range\n .filter((t) => t.date >= getUnixTime(dateRange.start) && t.date <= getUnixTime(dateRange.end))\n // Custom filter\n .filter(custom)\n // Search filter\n .filter((transaction) => {\n // Don't return uncategorized split transactions\n if (searchFilter === UNCATEGORIZED_TERM && transaction.has_been_split) {\n return false\n }\n\n // Split check - When searching for \"Splits\" and transaction is a split or is a parent of a split(s)\n const splitPattern = /^spl/i\n if (\n splitPattern.test(searchFilter) &&\n (transaction.has_been_split || Boolean(transaction.parent_guid))\n ) {\n return true\n }\n\n // Check transactions tags\n const isTagIncludedInSearch = transaction.tags.some((guid) =>\n isIncludedInSearch(searchFilter, tags.find((tag) => tag.guid === guid)?.name),\n )\n\n const date = fromUnixTime(transaction.date)\n return (\n isIncludedInSearch(searchFilter, transaction.payee) ||\n isIncludedInSearch(searchFilter, transaction.feed_description) ||\n isIncludedInSearch(searchFilter, transaction.category) ||\n isIncludedInSearch(searchFilter, formatDate(date, DATE_FORMATS_INTL.MONTH)) ||\n isIncludedInSearch(searchFilter, formatDate(date, DATE_FORMATS_INTL.DAY)) ||\n isIncludedInSearch(searchFilter, formatDate(date, DATE_FORMATS_INTL.FULL_MONTH_DAY)) ||\n isIncludedInSearch(searchFilter, formatDate(date, DATE_FORMATS_INTL.MONTH_DAY_YEAR)) ||\n isIncludedInSearch(searchFilter, format(date, ORDINAL_DATE_FORMAT)) ||\n isIncludedInSearch(searchFilter, transaction.account) ||\n isIncludedInSearch(searchFilter, `${transaction.amount}`) ||\n isIncludedInSearch(searchFilter, transaction.memo) ||\n isTagIncludedInSearch\n )\n })\n )\n}\n\nconst sortTransactions = (\n transactions: Transaction[],\n sortColumn: string,\n sortDirection: string,\n) => {\n return [...transactions].sort((a: Transaction, b: Transaction) => {\n const column = sortColumn as keyof Transaction\n const sortCompare = (a[column] || '') > (b[column] || '')\n\n let sortASC = sortCompare ? -1 : 1\n let sortDESC = sortCompare ? 1 : -1\n\n if (sortColumn === COLUMN_NAMES.DATE || sortColumn === COLUMN_NAMES.AMOUNT) {\n sortASC = sortCompare ? 1 : -1\n sortDESC = sortCompare ? -1 : 1\n }\n\n return sortDirection === TABLE_CONST.ASC ? sortASC : sortDESC\n })\n}\n\n/**\n * Determines if a value is included within the terms set in the search bar\n */\nconst isIncludedInSearch = (searchFilter: string, value: string | null | undefined) => {\n const lowerCasedTerms = searchFilter.toLowerCase().split(' ')\n const lowerCasedValue = value?.toString().toLowerCase() ?? ''\n\n return lowerCasedTerms.every((filter) => lowerCasedValue.indexOf(filter) > -1)\n}\n","import { makeAutoObservable } from 'mobx'\nimport { startOfMonth } from 'date-fns/startOfMonth'\nimport { endOfMonth } from 'date-fns/endOfMonth'\n\nimport type { Transaction } from '../../common'\n\nimport { DEFAULT_TRANSACTION } from '../constants/Transaction'\n\nexport class UiStore {\n beginDate = startOfMonth(new Date())\n endDate = endOfMonth(new Date())\n\n expandedSplits = [] as string[]\n isInitialized = false\n height = 550\n width = 1000\n shouldDisableDrawerScroll = false\n\n searchValue = ''\n selectedAccountGuids = [] as string[]\n\n manualTransaction: Transaction = DEFAULT_TRANSACTION\n isAmountValid = false\n\n showInsights = false\n\n constructor() {\n makeAutoObservable(this)\n }\n\n setDateRange = (beginDate: Date, endDate: Date) => {\n this.beginDate = beginDate\n this.endDate = endDate\n }\n\n setIsInitialized = (isInitialized: boolean) => (this.isInitialized = isInitialized)\n\n setIsAmountValid = (valid: boolean) => (this.isAmountValid = valid)\n\n setSearchValue = (searchValue: string) => (this.searchValue = searchValue)\n\n setSelectedAccountGuids = (accountGuids: string[]) => (this.selectedAccountGuids = accountGuids)\n\n setShouldDisableDrawerScroll = (shouldDisable: boolean) =>\n (this.shouldDisableDrawerScroll = shouldDisable)\n\n setShowInsights = (showInsights: boolean) => {\n this.showInsights = showInsights\n }\n\n setSize = (height: number, width: number) => {\n this.height = height\n this.width = width\n }\n\n toggleSplit = (guid: string) => {\n const expandedSplits = [...this.expandedSplits]\n const index = this.expandedSplits.findIndex((splitGuid) => splitGuid === guid)\n\n if (index >= 0) {\n expandedSplits.splice(index, 1)\n } else {\n expandedSplits.push(guid)\n }\n\n this.expandedSplits = expandedSplits\n }\n\n clearExpandedSplits = () => {\n if (this.expandedSplits.length === 0) return\n this.expandedSplits = []\n }\n\n updateManualTransaction = (manualTransaction: Transaction) => {\n this.manualTransaction = manualTransaction\n }\n}\n","import { makeAutoObservable, runInAction } from 'mobx'\nimport { endOfToday } from 'date-fns/endOfToday'\nimport { formatISO } from 'date-fns/formatISO'\nimport { fromUnixTime } from 'date-fns/fromUnixTime'\nimport { getUnixTime } from 'date-fns/getUnixTime'\nimport { isBefore } from 'date-fns/isBefore'\nimport { startOfMonth } from 'date-fns/startOfMonth'\nimport { startOfToday } from 'date-fns/startOfToday'\nimport { subDays } from 'date-fns/subDays'\n\nimport { BeatApi } from '../../common'\nimport type { Beat, GlobalStore, Tag, Tagging, Transaction, TransactionRule } from '../../common'\nimport { TABLE_CONST } from '../../common/constants/Table'\n\nimport { onAnalyticEvent } from '../analytics'\nimport { TransactionApi } from '../api/TransactionApi'\nimport { COLUMN_NAMES } from '../constants/TransactionsTable'\nimport { buildDetailedTransactions } from '../utils/buildDetailedTransactions'\nimport { flattenTransactionsWithSplits } from '../utils/exportTransactionsToCSV'\nimport { sortAndFilterTransactions } from '../utils/sortAndFilterTransactions'\n\nimport { UiStore } from './UiStore'\n\nexport type TransactionFilter = {\n accounts?: string[]\n dateRange?: { start: Date; end: Date }\n custom?: (value: Transaction) => boolean\n searchTerm?: string\n}\n\nexport class TransactionStore {\n globalStore: GlobalStore\n uiStore: UiStore\n api = new TransactionApi('/', '')\n beatApi = new BeatApi('/', '')\n\n associatedBeats: Beat[] = []\n cachedStartDate: Date = subDays(startOfToday(), 90)\n filter: TransactionFilter = {}\n isLoading = true\n rawTransactions: Transaction[] = []\n sortColumn = COLUMN_NAMES.DATE\n sortDirection = TABLE_CONST.DESC\n taggings: Tagging[] = []\n tags: Tag[] = []\n transactionRules: TransactionRule[] = []\n\n onAnalyticEvent = onAnalyticEvent\n\n constructor(globalStore: GlobalStore) {\n this.globalStore = globalStore\n this.uiStore = new UiStore()\n this.api = new TransactionApi(globalStore.endpoint, globalStore.sessionToken)\n this.beatApi = new BeatApi(globalStore.endpoint, globalStore.sessionToken)\n\n makeAutoObservable(this)\n }\n\n loadTransactionData = async ({\n associatedBeats,\n transactionRules,\n transactions,\n }: {\n associatedBeats?: Beat[]\n taggings?: Tagging[]\n tags?: Tag[]\n transactionRules?: TransactionRule[]\n transactions?: Transaction[]\n } = {}) => {\n await Promise.all([\n this.loadAssociatedBeats(startOfMonth(new Date()), associatedBeats),\n this.loadTags(),\n this.loadTaggings(),\n this.loadTransactionRules(transactionRules),\n // By default loads 90 days of transactions\n this.loadTransactions(this.cachedStartDate, endOfToday(), transactions),\n ])\n\n runInAction(() => {\n this.uiStore.setIsInitialized(true)\n })\n }\n\n get sortedTransactions(): Transaction[] {\n return sortAndFilterTransactions(\n this.filter,\n this.uiStore.expandedSplits,\n this.sortColumn,\n this.sortDirection,\n this.tags,\n this.transactions,\n )\n }\n\n get sortedTransactionsWithSplits(): Transaction[] {\n return flattenTransactionsWithSplits(this.sortedTransactions, this.transactions)\n }\n\n get transactions(): Transaction[] {\n return this.augmentTransactions(this.rawTransactions)\n }\n\n addTag = async (name: string) => {\n await this.api.addTag({ name } as Tag)\n await this.loadTags()\n }\n\n addTagging = async (tagGuid: string, transactionGuid: string) => {\n await this.api.addTagging({\n tag_guid: tagGuid,\n transaction_guid: transactionGuid,\n } as Tagging)\n }\n\n addOrUpdateTransactionRule = async (categoryGuid: string, transaction: Transaction) => {\n const existingRule = this.transactionRules.find(\n (rule: TransactionRule) => rule.guid === transaction.user_transaction_rule_guid,\n ) as TransactionRule | undefined\n\n if (existingRule) {\n await this.api.updateTransactionRule({\n ...existingRule,\n category_guid: categoryGuid,\n description: transaction.description,\n } as TransactionRule)\n } else {\n await this.api.addTransactionRule({\n category_guid: categoryGuid,\n description: transaction.description,\n match_description: transaction.description,\n } as TransactionRule)\n }\n await this.loadTransactionRules()\n }\n\n addTransaction = async (transaction: Transaction) => {\n const newTransaction = await this.api.addTransaction(transaction)\n\n if (newTransaction) {\n runInAction(() => (this.rawTransactions = [newTransaction, ...this.transactions]))\n }\n }\n\n augmentTransactions = (transactions: Transaction[]) => {\n return buildDetailedTransactions(\n transactions,\n this.globalStore.accountStore.visibleAccounts,\n this.globalStore.categoryStore.categories,\n this.taggings,\n )\n }\n\n getTransactionData = () => ({})\n\n loadAssociatedBeats = async (fromDate?: Date, beatsOverride?: Beat[]) => {\n const fromDateISO = fromDate ? formatISO(fromDate, { representation: 'date' }) : undefined\n const beats =\n beatsOverride ?? (await this.beatApi.getBeatsWithAssociatedTransaction(fromDateISO))\n runInAction(() => (this.associatedBeats = beats))\n }\n\n loadTags = async () => {\n const tags = await this.api.getTags()\n runInAction(() => (this.tags = tags))\n }\n\n loadTaggings = async () => {\n const taggings = await this.api.getTaggings()\n runInAction(() => (this.taggings = taggings))\n }\n\n loadTransactionRules = async (transactionRulesOverrides?: TransactionRule[]) => {\n const transactionRules = transactionRulesOverrides ?? (await this.api.getTransactionRules())\n runInAction(() => (this.transactionRules = transactionRules))\n }\n\n loadTransactions = async (\n startDate: Date = this.cachedStartDate,\n endDate: Date = endOfToday(),\n transactionsOverrides?: Transaction[],\n ) => {\n this.setIsLoading(true)\n this.rawTransactions = []\n\n try {\n const transactions =\n transactionsOverrides ??\n (await this.api.getTransactionsByDateRange(getUnixTime(startDate), getUnixTime(endDate)))\n runInAction(() => {\n this.rawTransactions = transactions\n })\n } catch (error) {\n console.error(`Error occurred while loading transactions: ${error}`)\n }\n this.setIsLoading(false)\n }\n\n // DEPRECATED\n loadTransactionsByDateRange = async (endDate: number, startDate: number) => {\n console.warn('DEPRECATED: loadTransactionsByDateRange will be removed')\n this.setFilter({\n ...this.filter,\n dateRange: { start: fromUnixTime(startDate), end: fromUnixTime(endDate) },\n })\n }\n\n modifyTransactionByAccountGuid = (guid: string, updates: Partial<Transaction>) => {\n this.rawTransactions = this.transactions.map((txn) =>\n txn.account_guid === guid ? { ...txn, ...updates } : txn,\n )\n }\n\n removeTag = async (tagGuid: string) => {\n await this.api.removeTag(tagGuid)\n await this.loadTags()\n }\n\n removeTagging = async (taggingGuid: string) => {\n await this.api.removeTagging(taggingGuid)\n }\n\n removeTransaction = async (transactionGuid: string) => {\n await this.api.removeTransaction(transactionGuid)\n\n const index = this.transactions.findIndex((t: Transaction) => t.guid === transactionGuid)\n\n if (index >= 0) {\n const updatedTransactions = [...this.transactions]\n updatedTransactions.splice(index, 1)\n runInAction(() => (this.rawTransactions = updatedTransactions))\n }\n }\n\n setFilter = (filter: TransactionFilter) => {\n this.filter = filter\n\n // Only reload transactions if new start date is prior to the current cached start date\n // Cached date range end date is always today\n if (filter.dateRange && isBefore(filter.dateRange.start, this.cachedStartDate)) {\n this.loadTransactions(filter.dateRange.start, endOfToday()).then(() => {\n if (filter.dateRange) {\n this.cachedStartDate = filter.dateRange.start\n }\n })\n }\n }\n\n setIsLoading = (isLoading: boolean) => (this.isLoading = isLoading)\n\n setSortData = (sortColumn: string, sortDirection: string) => {\n this.sortColumn = sortColumn\n this.sortDirection = sortDirection\n }\n\n splitTransaction = async (transaction: Transaction, splits: Transaction[]) => {\n const updatedTransaction = await this.api.updateTransaction(transaction)\n const children = [] as Transaction[]\n\n for (let i = 0; i < splits.length; i++) {\n const child = await this.api.addTransaction(splits[i])\n children.push(child)\n }\n\n const index = this.transactions.findIndex((t: Transaction) => t.guid === transaction.guid)\n\n if (index >= 0) {\n runInAction(\n () =>\n (this.rawTransactions = [\n ...this.transactions.slice(0, index),\n updatedTransaction,\n ...this.transactions.slice(index + 1),\n ...children,\n ]),\n )\n }\n }\n\n unSplitTransaction = async (guid: string) => {\n const children = this.transactions.filter((t) => t.parent_guid === guid)\n\n await this.api.removeTransactionSplit(guid)\n\n // Remove children\n const transactions = [...this.transactions]\n children.forEach((child) => {\n const index = transactions.findIndex((t) => t.guid === child.guid)\n\n if (index >= 0) {\n transactions.splice(index, 1)\n }\n })\n\n // Update parent\n const index = transactions.findIndex((t) => t.guid === guid)\n const transaction = transactions[index]\n transaction.has_been_split = false\n if (index >= 0) {\n runInAction(\n () =>\n (this.rawTransactions = [\n ...transactions.slice(0, index),\n transaction,\n ...transactions.slice(index + 1),\n ]),\n )\n }\n }\n\n updateTaggings = async (tagGuids: string[], transactionGuid: string) => {\n const currentTaggings = this.taggings.filter((t) => t.transaction_guid === transactionGuid)\n\n const currentTagGuids = new Set(currentTaggings.map((t) => t.tag_guid))\n const incomingTagGuids = new Set(tagGuids)\n\n // Add new taggings that don't already exist\n for (const tagGuid of tagGuids) {\n if (!currentTagGuids.has(tagGuid)) {\n await this.addTagging(tagGuid, transactionGuid)\n }\n }\n\n // Remove taggings that are no longer needed\n for (const tagging of currentTaggings) {\n if (!incomingTagGuids.has(tagging.tag_guid)) {\n await this.removeTagging(tagging.guid)\n }\n }\n\n await this.loadTaggings()\n }\n\n updateTag = async (tag: Tag) => {\n await this.api.updateTag(tag)\n await this.loadTags()\n }\n\n updateTransaction = async (transaction: Transaction) => {\n const updatedTransaction = await this.api.updateTransaction(transaction)\n\n const index = this.transactions.findIndex((t: Transaction) => t.guid === transaction.guid)\n\n if (index >= 0) {\n runInAction(\n () =>\n (this.rawTransactions = [\n ...this.transactions.slice(0, index),\n updatedTransaction,\n ...this.transactions.slice(index + 1),\n ]),\n )\n }\n }\n}\n"],"names":["TABLE_CONST","onAnalyticEvent","event","session","ANALYTICS_EVENTS","COLUMN_NAMES","buildDetailedTransactions","transactions","accounts","categories","taggings","accountsMap","acc","account","categoriesMap","category","taggingsMap","tagging","transaction_guid","tag_guid","tags","transaction","CategoryGuids","TransactionStatus","UNCATEGORIZED_TERM","sortAndFilterTransactions","filter","expandedSplits","sortColumn","sortDirection","filteredTransactions","filterTransactions","sortedTransactions","sortTransactions","guid","index","t","children","dateRange","startOfMonth","startOfToday","endOfToday","custom","searchFilter","getUnixTime","isTagIncludedInSearch","isIncludedInSearch","tag","date","fromUnixTime","formatDate","DATE_FORMATS_INTL","format","ORDINAL_DATE_FORMAT","a","b","column","sortCompare","sortASC","sortDESC","value","lowerCasedTerms","lowerCasedValue","UiStore","endOfMonth","DEFAULT_TRANSACTION","makeAutoObservable","beginDate","endDate","isInitialized","valid","searchValue","accountGuids","shouldDisable","showInsights","height","width","splitGuid","manualTransaction","TransactionStore","TransactionApi","BeatApi","subDays","globalStore","associatedBeats","transactionRules","runInAction","flattenTransactionsWithSplits","name","tagGuid","transactionGuid","categoryGuid","existingRule","rule","newTransaction","fromDate","beatsOverride","fromDateISO","formatISO","beats","transactionRulesOverrides","startDate","transactionsOverrides","error","updates","txn","taggingGuid","updatedTransactions","isBefore","isLoading","splits","updatedTransaction","i","child","tagGuids","currentTaggings","currentTagGuids","incomingTagGuids"],"mappings":"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"}
@@ -0,0 +1,131 @@
1
+ import { makeAutoObservable as m } from "mobx";
2
+ import { addSeconds as p } from "date-fns/addSeconds";
3
+ import { differenceInCalendarMonths as a } from "date-fns/differenceInCalendarMonths";
4
+ import { endOfMonth as e } from "date-fns/endOfMonth";
5
+ import { subMonths as s } from "date-fns/subMonths";
6
+ import "bowser";
7
+ import "date-fns/getUnixTime";
8
+ import { aI as r, aJ as i } from "./ConnectMiniWidget-CQWafYtQ.mjs";
9
+ import "date-fns/addMonths";
10
+ import "date-fns/addQuarters";
11
+ import "date-fns/addWeeks";
12
+ import "date-fns/addYears";
13
+ import "date-fns/getDay";
14
+ import "date-fns/isBefore";
15
+ import "date-fns/nextDay";
16
+ import "date-fns/setDate";
17
+ import "date-fns/startOfToday";
18
+ import "date-fns/fromUnixTime";
19
+ import "react";
20
+ import "numeral";
21
+ import "@mui/material";
22
+ import "react/jsx-runtime";
23
+ import "@mui/material/Button";
24
+ import "@mxenabled/mxui";
25
+ import "@mui/material/Tooltip";
26
+ import "@mui/material/Box";
27
+ import "@mui/material/IconButton";
28
+ import "./Loader-CwdK8lfx.mjs";
29
+ import "@mui/material/styles/useTheme";
30
+ import "@mui/material/useMediaQuery";
31
+ import "@mxenabled/cssinjs";
32
+ import "date-fns/addDays";
33
+ import "date-fns/eachDayOfInterval";
34
+ import "date-fns/eachWeekOfInterval";
35
+ import "date-fns/format";
36
+ import "date-fns/isSameMonth";
37
+ import "date-fns/startOfMonth";
38
+ import "@mui/material/Paper";
39
+ import "@mxenabled/mx-icons";
40
+ import "date-fns/isSameDay";
41
+ import "date-fns/isWithinInterval";
42
+ import "d3";
43
+ import "@mui/material/styles";
44
+ import "@mui/x-charts";
45
+ import "@mui/x-charts/ChartsTooltip";
46
+ import "@mui/material/Stack";
47
+ import "date-fns/differenceInCalendarDays";
48
+ import "date-fns/endOfToday";
49
+ import "date-fns/endOfYear";
50
+ import "date-fns/startOfYear";
51
+ import "date-fns/subDays";
52
+ import "date-fns/subYears";
53
+ import "@mui/material/ButtonGroup";
54
+ import "@mui/material/Tab";
55
+ import "@mui/material/Tabs";
56
+ import "@mui/material/ToggleButton";
57
+ import "@mui/material/ToggleButtonGroup";
58
+ import "@mui/material/Dialog";
59
+ import "@mui/material/DialogContent";
60
+ import "@mui/material/Card";
61
+ import "@mui/material/CardContent";
62
+ import "@mui/material/CardHeader";
63
+ import "uuid";
64
+ import "@mui/material/CircularProgress";
65
+ import "@mui/material/FormControlLabel";
66
+ import "@mui/material/Radio";
67
+ import "@mui/material/Avatar";
68
+ import "@mui/material/ListItem";
69
+ import "@mui/material/ListItemAvatar";
70
+ import "@mui/material/ListItemButton";
71
+ import "@mui/material/ListItemText";
72
+ import "@mui/material/ListItemIcon";
73
+ import "@mui/material/MenuItem";
74
+ import "@mui/material/TextField";
75
+ import "@mui/material/Switch";
76
+ import "posthog-js";
77
+ import "date-fns/differenceInDays";
78
+ import "@mui/x-data-grid-pro";
79
+ import "date-fns";
80
+ import "@mui/material/Divider";
81
+ import "@mui/material/List";
82
+ import "date-fns/formatISO";
83
+ import "date-fns/isAfter";
84
+ import "date-fns/parseISO";
85
+ import "date-fns/subQuarters";
86
+ import "date-fns/lastDayOfMonth";
87
+ import "date-fns/setDayOfYear";
88
+ import "date-fns/getMonth";
89
+ class Bt {
90
+ globalStore;
91
+ selectedCategoryData = null;
92
+ selectedDateRange;
93
+ constructor(t) {
94
+ this.globalStore = t, this.selectedDateRange = {
95
+ start: p(e(s(/* @__PURE__ */ new Date(), 6)), 1),
96
+ end: e(/* @__PURE__ */ new Date())
97
+ }, m(this);
98
+ }
99
+ setSelectedCategoryData = (t) => {
100
+ this.selectedCategoryData = t;
101
+ };
102
+ setSelectedDateRange = (t) => {
103
+ this.selectedDateRange = t;
104
+ };
105
+ get selectedDateRangeMonthRange() {
106
+ const t = r(this.selectedDateRange.start, i.MONTH_LONG), o = r(this.selectedDateRange.end, i.MONTH_LONG);
107
+ return `${t} - ${o}`;
108
+ }
109
+ get selectedDateRangeMonthCount() {
110
+ return a(this.selectedDateRange.end, this.selectedDateRange.start) + 1;
111
+ }
112
+ get sortedDetailedCategories() {
113
+ return this.globalStore.categoryStore.detailedCategories.slice().filter((t) => t.totalAmount !== 0).sort((t, o) => Math.abs(o.totalAmount) - Math.abs(t.totalAmount));
114
+ }
115
+ get visibleCategories() {
116
+ return this.sortedDetailedCategories.slice(0, 5);
117
+ }
118
+ get collapsedCategories() {
119
+ return this.sortedDetailedCategories.slice(5);
120
+ }
121
+ get categoryDetailsChartData() {
122
+ return { monthlyAmounts: this.selectedCategoryData ? this.selectedCategoryData.totalMonthlyAmounts.slice(-this.selectedDateRangeMonthCount).map((o) => ({
123
+ x: o.label.substring(0, 3),
124
+ y: o.amount
125
+ })) : [] };
126
+ }
127
+ }
128
+ export {
129
+ Bt as T
130
+ };
131
+ //# sourceMappingURL=TrendsStore-Idq3QfoA.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TrendsStore-Idq3QfoA.mjs","sources":["../src/trends/stores/TrendsStore.ts"],"sourcesContent":["import { makeAutoObservable } from 'mobx'\nimport { addSeconds } from 'date-fns/addSeconds'\nimport { differenceInCalendarMonths } from 'date-fns/differenceInCalendarMonths'\nimport { endOfMonth } from 'date-fns/endOfMonth'\nimport { subMonths } from 'date-fns/subMonths'\n\nimport type { DateRange, DetailedCategory, GlobalStore } from '../../common'\nimport { DATE_FORMATS_INTL, formatDate } from '../../common'\nimport { LineChartDataset } from '../../common/components/charts/LineChart'\n\nexport class TrendsStore {\n globalStore: GlobalStore\n selectedCategoryData: DetailedCategory | null = null\n selectedDateRange: DateRange\n\n constructor(globalStore: GlobalStore) {\n this.globalStore = globalStore\n this.selectedDateRange = {\n start: addSeconds(endOfMonth(subMonths(new Date(), 6)), 1),\n end: endOfMonth(new Date()),\n }\n makeAutoObservable(this)\n }\n\n setSelectedCategoryData = (category: DetailedCategory | null) => {\n this.selectedCategoryData = category\n }\n\n setSelectedDateRange = (dateRange: DateRange) => {\n this.selectedDateRange = dateRange\n }\n\n get selectedDateRangeMonthRange(): string {\n const startMonth = formatDate(this.selectedDateRange.start, DATE_FORMATS_INTL.MONTH_LONG)\n const endMonth = formatDate(this.selectedDateRange.end, DATE_FORMATS_INTL.MONTH_LONG)\n return `${startMonth} - ${endMonth}`\n }\n\n get selectedDateRangeMonthCount(): number {\n return differenceInCalendarMonths(this.selectedDateRange.end, this.selectedDateRange.start) + 1\n }\n\n get sortedDetailedCategories(): DetailedCategory[] {\n return this.globalStore.categoryStore.detailedCategories\n .slice()\n .filter((category) => category.totalAmount !== 0)\n .sort((a, b) => Math.abs(b.totalAmount) - Math.abs(a.totalAmount))\n }\n\n get visibleCategories(): DetailedCategory[] {\n return this.sortedDetailedCategories.slice(0, 5)\n }\n\n get collapsedCategories(): DetailedCategory[] {\n return this.sortedDetailedCategories.slice(5)\n }\n\n get categoryDetailsChartData(): {\n monthlyAmounts: LineChartDataset\n } {\n const monthlyAmounts = this.selectedCategoryData\n ? this.selectedCategoryData.totalMonthlyAmounts\n .slice(-this.selectedDateRangeMonthCount) // get the last x items in the totalMonthlyAmounts array and map to chart dataset\n .map((monthData) => ({\n x: monthData.label.substring(0, 3),\n y: monthData.amount,\n }))\n : []\n\n return { monthlyAmounts }\n }\n}\n"],"names":["TrendsStore","globalStore","addSeconds","endOfMonth","subMonths","makeAutoObservable","category","dateRange","startMonth","formatDate","DATE_FORMATS_INTL","endMonth","differenceInCalendarMonths","a","b","monthData"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAUO,MAAMA,GAAY;AAAA,EACvB;AAAA,EACA,uBAAgD;AAAA,EAChD;AAAA,EAEA,YAAYC,GAA0B;AACpC,SAAK,cAAcA,GACnB,KAAK,oBAAoB;AAAA,MACvB,OAAOC,EAAWC,EAAWC,EAAU,oBAAI,QAAQ,CAAC,CAAC,GAAG,CAAC;AAAA,MACzD,KAAKD,EAAe,oBAAA,MAAM;AAAA,IAAA,GAE5BE,EAAmB,IAAI;AAAA,EACzB;AAAA,EAEA,0BAA0B,CAACC,MAAsC;AAC/D,SAAK,uBAAuBA;AAAA,EAAA;AAAA,EAG9B,uBAAuB,CAACC,MAAyB;AAC/C,SAAK,oBAAoBA;AAAA,EAAA;AAAA,EAG3B,IAAI,8BAAsC;AACxC,UAAMC,IAAaC,EAAW,KAAK,kBAAkB,OAAOC,EAAkB,UAAU,GAClFC,IAAWF,EAAW,KAAK,kBAAkB,KAAKC,EAAkB,UAAU;AAC7E,WAAA,GAAGF,CAAU,MAAMG,CAAQ;AAAA,EACpC;AAAA,EAEA,IAAI,8BAAsC;AACxC,WAAOC,EAA2B,KAAK,kBAAkB,KAAK,KAAK,kBAAkB,KAAK,IAAI;AAAA,EAChG;AAAA,EAEA,IAAI,2BAA+C;AAC1C,WAAA,KAAK,YAAY,cAAc,mBACnC,MAAA,EACA,OAAO,CAACN,MAAaA,EAAS,gBAAgB,CAAC,EAC/C,KAAK,CAACO,GAAGC,MAAM,KAAK,IAAIA,EAAE,WAAW,IAAI,KAAK,IAAID,EAAE,WAAW,CAAC;AAAA,EACrE;AAAA,EAEA,IAAI,oBAAwC;AAC1C,WAAO,KAAK,yBAAyB,MAAM,GAAG,CAAC;AAAA,EACjD;AAAA,EAEA,IAAI,sBAA0C;AACrC,WAAA,KAAK,yBAAyB,MAAM,CAAC;AAAA,EAC9C;AAAA,EAEA,IAAI,2BAEF;AAUA,WAAO,EAAE,gBATc,KAAK,uBACxB,KAAK,qBAAqB,oBACvB,MAAM,CAAC,KAAK,2BAA2B,EACvC,IAAI,CAACE,OAAe;AAAA,MACnB,GAAGA,EAAU,MAAM,UAAU,GAAG,CAAC;AAAA,MACjC,GAAGA,EAAU;AAAA,IAAA,EACb,IACJ,CAAA,EAEoB;AAAA,EAC1B;AACF;"}
@@ -0,0 +1,18 @@
1
+ import { c as a, b as c, a as s, A as n, C as o, F as i, M as u, g as A, d as r, e as l, u as d } from "../ConnectMiniWidget-CQWafYtQ.mjs";
2
+ import { A as M, a as p } from "../AccountStore-DkL93tXL.mjs";
3
+ export {
4
+ M as AccountApi,
5
+ a as AccountDetailsContent,
6
+ c as AccountDetailsHeader,
7
+ p as AccountStore,
8
+ s as AccountsMiniWidget,
9
+ n as AccountsWidget,
10
+ o as ConnectMiniWidget,
11
+ i as FieldType,
12
+ u as MemberConnectionStatus,
13
+ A as getAccountBalanceDetails,
14
+ r as getAccountDetailFields,
15
+ l as useAccountUiStore,
16
+ d as useGetMergeableAccounts
17
+ };
18
+ //# sourceMappingURL=index.es.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.es.js","sources":[],"sourcesContent":[],"names":[],"mappings":";;"}