1029 Commits
1.0.0 ... 1.6.0

Author SHA1 Message Date
MartinBraquet
62821ed803 Release 2025-11-08 20:52:41 +01:00
MartinBraquet
903d62ed57 Add error comment 2025-11-08 19:55:13 +01:00
MartinBraquet
adef626b34 Cache build ID 2025-11-08 19:46:42 +01:00
MartinBraquet
11a933cc04 Comment 2025-11-08 19:46:21 +01:00
MartinBraquet
4b5ce99bb1 Fix avatar link not working on mobile 2025-11-08 19:46:13 +01:00
MartinBraquet
f8dff77cee Do not use ID filtering anymore as loaded messages may be deleted or edited 2025-11-08 18:50:23 +01:00
MartinBraquet
0a9b08803e Add comment 2025-11-08 18:48:47 +01:00
MartinBraquet
39a8568663 Fix orderBy that must be filtered out 2025-11-08 18:12:13 +01:00
MartinBraquet
e1502440eb Increase debounce since moved back to free plan (1 / sec) 2025-11-07 23:31:18 +01:00
MartinBraquet
5834d032c3 Android release 2025-11-07 23:26:46 +01:00
MartinBraquet
0a28a2af61 Allow editing old messages 2025-11-07 23:26:38 +01:00
MartinBraquet
259f56bd26 Fix typescript / linting warnings 2025-11-07 23:17:16 +01:00
MartinBraquet
011ad66a3f Fix emoji location (2) 2025-11-07 23:13:32 +01:00
MartinBraquet
281c72f88d Merge remote-tracking branch 'origin/main' 2025-11-07 22:53:07 +01:00
Okechi Jones-Williams
1293523ebf Added further playwright structure (#17)
* setting up test structure

* .

* added playwright config file, deleted original playwright folder and moved "some.test" file

* continued test structure setup

* Updating test folder structure
2025-11-07 22:52:19 +01:00
MartinBraquet
67e5d34f39 Fix emoji location 2025-11-07 22:48:38 +01:00
MartinBraquet
8b3dec6116 Merge remote-tracking branch 'origin/main' 2025-11-07 21:52:25 +01:00
MartinBraquet
3359f49f0a Add edit, delete and emojis on messages 2025-11-07 21:51:56 +01:00
Okechi Jones-Williams
9715cae587 Setting up test file structure (#16)
* setting up test structure

* .
2025-11-07 14:28:07 +01:00
MartinBraquet
4b122bd907 Increase version 2025-11-05 20:52:19 +01:00
MartinBraquet
edefa36c8a Comments 2025-11-05 20:46:56 +01:00
MartinBraquet
e40a352aed When clicking on the DM notif, go to the DM chat 2025-11-05 20:46:41 +01:00
MartinBraquet
70267f0623 Add demo fix
Hope it works finally
2025-11-04 17:51:38 +01:00
MartinBraquet
c19ed7f6d5 Add demo
Hope it works finally
2025-11-04 17:37:49 +01:00
MartinBraquet
e2fb0f2ee8 Line clamp notif to 5 2025-11-04 13:07:57 +01:00
MartinBraquet
f9e208a7e0 Add createAndroidTestNotifications 2025-11-04 13:04:06 +01:00
MartinBraquet
ee2b8a60a2 Increase notif lines 2025-11-04 13:03:43 +01:00
MartinBraquet
5651f17e96 Increase android version 2025-11-04 12:46:17 +01:00
MartinBraquet
9679b3722a Show error 2025-11-04 01:25:19 +01:00
MartinBraquet
6cb736a10d Upgrade 2025-11-03 23:55:37 +01:00
MartinBraquet
ac5c3b421d Upgrade 2025-11-03 23:45:53 +01:00
MartinBraquet
f01fad5fa6 Improve message rendering on webview (fix 2) 2025-11-03 23:45:03 +01:00
MartinBraquet
110b727cbc Improve message rendering on webview (fix) 2025-11-03 23:13:56 +01:00
MartinBraquet
1dba2debc5 Clean 2025-11-03 20:27:32 +01:00
MartinBraquet
8fec73341f Comment 2025-11-03 18:39:56 +01:00
MartinBraquet
33c76de397 Improve bio tips 2025-11-03 17:19:37 +01:00
MartinBraquet
d409eb1126 Refactor news 2025-11-03 16:53:50 +01:00
MartinBraquet
69d53591c8 Load cached remote page data from webview 2025-11-03 16:44:41 +01:00
MartinBraquet
4cf783f07f Add proxy 2025-11-03 16:19:10 +01:00
MartinBraquet
a971c121cc Allow any origin 2025-11-03 16:04:50 +01:00
MartinBraquet
b4abfb6aa4 Simplify build-id 2025-11-03 15:53:44 +01:00
MartinBraquet
8f760081a5 Add build-id/route.ts 2025-11-03 15:25:11 +01:00
MartinBraquet
c15934f210 Render home icon if logged out 2025-11-03 15:17:14 +01:00
MartinBraquet
c7e1bb4463 People is more human than profiles 2025-11-03 12:15:18 +01:00
MartinBraquet
4ac97fc476 Make webview go back one page instead of closing the app 2025-11-02 22:26:22 +01:00
MartinBraquet
a5dda01ffd Clean 2025-11-02 22:20:49 +01:00
MartinBraquet
9db4937e54 Fix layout 2025-11-02 22:07:56 +01:00
MartinBraquet
3dc50a384e Add title when loading 2025-11-02 22:05:36 +01:00
MartinBraquet
ccc48620b8 Fix profile page not switching between users 2025-11-02 21:41:23 +01:00
MartinBraquet
943c3960e1 Comment logs 2025-11-02 21:39:17 +01:00
MartinBraquet
9a84a67555 Improve margins for mobile 2025-11-02 21:29:51 +01:00
MartinBraquet
2241e3c7d3 Log pixel top and bottom 2025-11-02 21:27:07 +01:00
MartinBraquet
03042dae96 Do not disclose exact last online time for privacy reasons 2025-11-02 21:26:17 +01:00
MartinBraquet
3df9d067d6 Clean posthog 2025-11-02 21:15:09 +01:00
MartinBraquet
c5f3e8b3c2 Clean css 2025-11-02 20:26:39 +01:00
MartinBraquet
3a3e8c0e10 Add get pixel height 2025-11-02 20:26:29 +01:00
MartinBraquet
f010bf7eed Fix 2025-11-02 13:14:20 +01:00
MartinBraquet
b22d6a77b0 Update readme 2025-11-02 13:09:14 +01:00
MartinBraquet
f91d9125e4 Ignore getStaticProps and getStaticPaths for mobile webview build 2025-11-02 13:09:00 +01:00
MartinBraquet
625eedef1e Add middleware template 2025-11-02 13:04:10 +01:00
MartinBraquet
61dd33187a Use SSR/ISR only for web app (load client side on mobile app) 2025-11-02 13:03:22 +01:00
MartinBraquet
817640c7d0 Update reserved path 2025-11-02 00:21:25 +01:00
MartinBraquet
aa15d5a6df Comment 2025-11-02 00:21:15 +01:00
MartinBraquet
60857007bd Fix 2025-11-01 22:11:50 +01:00
MartinBraquet
cfd0c5d846 versionCode 4 2025-11-01 22:05:43 +01:00
MartinBraquet
71ec88d235 Add mobile icons 2025-11-01 22:05:28 +01:00
MartinBraquet
902ea583bd Remove AD_ID 2025-11-01 21:27:36 +01:00
MartinBraquet
fb6b54fba0 Delete mobile subscriptions if not set up 2025-11-01 21:26:43 +01:00
MartinBraquet
9dfc56987c Add Child Sexual Abuse and Exploitation (CSAE) 2025-11-01 20:30:42 +01:00
MartinBraquet
b982a02717 V2 2025-11-01 18:47:55 +01:00
MartinBraquet
15a0d8ee16 Add /delete-account 2025-11-01 17:20:31 +01:00
MartinBraquet
9ca03132db Rename to com.compassconnections.com 2025-11-01 17:14:08 +01:00
MartinBraquet
f908ba3ea3 Update terms 2025-11-01 17:09:15 +01:00
MartinBraquet
79ea09bd91 Skip web service-worker.js on native mobile 2025-11-01 16:25:55 +01:00
MartinBraquet
c743a4f1fe Comment unused google auth 2025-11-01 16:22:09 +01:00
MartinBraquet
ccd1fc5c10 Fix data not in notif 2025-11-01 16:06:22 +01:00
MartinBraquet
2f82a64dbe Add isNativeMobile 2025-11-01 16:06:07 +01:00
MartinBraquet
71c6eae9c4 Fix googleNativeLogin 2025-11-01 16:05:58 +01:00
MartinBraquet
28894a08cc Add capgo-capacitor-social-login to gradle 2025-11-01 16:05:42 +01:00
MartinBraquet
41e366dcc4 Output export for webview 2025-11-01 16:05:26 +01:00
MartinBraquet
6ee11a6b2d Skip getStaticProps in webview 2025-11-01 16:05:03 +01:00
MartinBraquet
ab82c66f83 Add bridge for social 2025-11-01 16:04:36 +01:00
MartinBraquet
f94820f45e Fix favicon not showing in webview 2025-11-01 16:03:39 +01:00
MartinBraquet
fe03e1ca68 Use local assets for prod 2025-11-01 16:03:26 +01:00
MartinBraquet
44ee6951c9 Comment usused deep link 2025-11-01 16:02:52 +01:00
MartinBraquet
40194f7204 Update readme 2025-11-01 16:02:37 +01:00
MartinBraquet
40eefef9a2 Use full ico path for webview 2025-11-01 16:02:25 +01:00
MartinBraquet
d8ab44ebb5 Install @capgo/capacitor-social-login 2025-11-01 16:01:47 +01:00
MartinBraquet
f875539f2e Move to unused 2025-11-01 16:01:36 +01:00
MartinBraquet
d9c2d142cb Format 2025-11-01 16:00:11 +01:00
MartinBraquet
51d5715f04 Clean 404 2025-11-01 12:38:01 +01:00
MartinBraquet
8ad81b1d50 Add bottom margin 2025-11-01 12:37:55 +01:00
MartinBraquet
6d4083d8a7 Move hosting constants to new file 2025-11-01 11:19:36 +01:00
MartinBraquet
bb120afea2 Clean web build 2025-11-01 11:17:20 +01:00
MartinBraquet
cebcc20c26 Add email verif link in welcome email 2025-11-01 10:53:10 +01:00
MartinBraquet
0522e787cd Add Forgot Password? 2025-10-31 20:41:57 +01:00
MartinBraquet
81e01f1485 Change tag 2025-10-31 20:32:09 +01:00
MartinBraquet
9b2b93d56f Update faq 2025-10-31 20:31:40 +01:00
MartinBraquet
133b402e2b Add IS_WEBVIEW_BUILD 2025-10-31 13:36:41 +01:00
MartinBraquet
9a91eab13f Catch user not found 2025-10-31 11:52:32 +01:00
MartinBraquet
5f6722b917 Add message notif 2025-10-31 11:52:24 +01:00
MartinBraquet
fdcd4d46ac Add allowNavigation 2025-10-31 11:52:06 +01:00
MartinBraquet
27a72170b8 Url redirect 2025-10-31 11:51:58 +01:00
MartinBraquet
b3021e60ec Add sync_android.sh 2025-10-31 10:41:22 +01:00
MartinBraquet
6d8834bd87 Add little emoji 2025-10-31 10:41:12 +01:00
MartinBraquet
ab8a9d95d8 Add notif perms 2025-10-31 00:15:42 +01:00
MartinBraquet
563ee3f5df Remove unused social login 2025-10-31 00:15:34 +01:00
MartinBraquet
c3389a7fcf Uncomment 2025-10-31 00:14:57 +01:00
MartinBraquet
e824bbb533 Add webview push notif 2025-10-30 23:45:04 +01:00
MartinBraquet
7de0e351f3 Skip notif verif in client 2025-10-30 23:44:41 +01:00
MartinBraquet
2df5f55390 Add androidpush check 2025-10-30 23:31:57 +01:00
MartinBraquet
21038cc5ac Clean and refactor oauth redirect 2025-10-30 23:28:18 +01:00
MartinBraquet
d6749bcd41 Add way to access profiles without log in during dev 2025-10-30 23:18:50 +01:00
MartinBraquet
005c9ccdef Add info 2025-10-30 23:18:31 +01:00
MartinBraquet
d47cb53e59 Fix filter top touching top nav bar 2025-10-30 23:18:18 +01:00
MartinBraquet
adfb3ca4f0 Add build_sync_android.sh 2025-10-30 23:03:16 +01:00
MartinBraquet
88b7e4edda Update android readmes 2025-10-30 22:59:09 +01:00
MartinBraquet
e33b57f0fd Comment sensitive logs 2025-10-30 22:52:33 +01:00
MartinBraquet
777825b73f Better log 2025-10-30 22:50:16 +01:00
MartinBraquet
04ca9b6f9a Remove PKCE as using google client secret 2025-10-30 22:39:10 +01:00
MartinBraquet
7f3d3eeb9c Hide background in bottom nav bar 2025-10-30 22:24:47 +01:00
MartinBraquet
75ac16d43c Rename 2025-10-30 21:59:33 +01:00
MartinBraquet
5f1120c718 Fix 2025-10-30 21:55:21 +01:00
MartinBraquet
806a0694c6 Add status bar 2025-10-30 21:50:36 +01:00
MartinBraquet
c506ae3242 Fix statusbard 2025-10-30 21:48:31 +01:00
MartinBraquet
444fa529fb Fix margin 2025-10-30 21:47:17 +01:00
MartinBraquet
62ced3eb04 Add top and bottom margin for mobile 2025-10-30 21:40:36 +01:00
MartinBraquet
31718f1c4d Refactor redirect url 2025-10-30 20:37:12 +01:00
MartinBraquet
98ab8971b4 Clean 2025-10-30 19:52:55 +01:00
MartinBraquet
870c86f9af Cleanup 2025-10-30 19:39:03 +01:00
MartinBraquet
a4f6aabee9 Cleanup 2025-10-30 19:37:11 +01:00
MartinBraquet
5584ad0a10 Remove unused cst 2025-10-30 19:26:26 +01:00
MartinBraquet
6b4932b4c5 Add error handling 2025-10-30 19:26:19 +01:00
MartinBraquet
7f1cb0aaf3 Add firebase login after receiving tokens 2025-10-30 19:26:08 +01:00
MartinBraquet
d42a5a48e9 Send code verifier to backend 2025-10-30 18:44:22 +01:00
MartinBraquet
8a1b762c35 Fix readme 2025-10-30 18:44:12 +01:00
MartinBraquet
d2d1de41d2 Fix 2025-10-30 18:08:22 +01:00
MartinBraquet
d1366af2a0 Fix 2025-10-30 18:06:14 +01:00
MartinBraquet
cae4b15bbb Fix 2025-10-30 18:01:49 +01:00
MartinBraquet
e41bc64b0c Fix 2025-10-30 17:21:19 +01:00
MartinBraquet
3d03ebe487 Fix 2025-10-30 17:18:00 +01:00
MartinBraquet
1fcb431d1b Fix bridge 2025-10-30 17:15:49 +01:00
MartinBraquet
4f321490af Fix 2025-10-30 17:11:53 +01:00
MartinBraquet
04c7469e68 Fix 2025-10-30 17:06:19 +01:00
MartinBraquet
98bc0a9309 Fix 2025-10-30 17:01:21 +01:00
MartinBraquet
6e537f4cdf Fix 2025-10-30 16:48:52 +01:00
MartinBraquet
b121d61852 Fix 2025-10-30 16:24:22 +01:00
MartinBraquet
6ba3c3ffbd Fix 2025-10-30 13:53:59 +01:00
MartinBraquet
67b3efad4c Fix 2025-10-30 13:50:05 +01:00
MartinBraquet
1282e468e3 Fix 2025-10-30 13:47:12 +01:00
MartinBraquet
67b2e78a63 Fix oauthRedirect 2025-10-30 13:44:40 +01:00
MartinBraquet
213c56f945 Fix 2025-10-30 13:20:27 +01:00
MartinBraquet
ccde6e4f4b Fix 2025-10-30 12:46:25 +01:00
MartinBraquet
2c1c94d24c Update android webview 2025-10-30 12:40:20 +01:00
MartinBraquet
56edb51f36 Add bridge 2025-10-30 12:40:00 +01:00
MartinBraquet
b6ed2c7dd8 Change GOOGLE_CLIENT_ID 2025-10-30 12:25:46 +01:00
MartinBraquet
74c7c5c423 Fix GOOGLE_CLIENT_ID 2025-10-30 12:00:04 +01:00
MartinBraquet
314d774bde Fix 2025-10-30 11:51:56 +01:00
MartinBraquet
d94091ae4e Add webviewGoogleSignin 2025-10-30 11:33:23 +01:00
MartinBraquet
8bf3c4fcd7 Remove sign up button if logged in 2025-10-29 20:43:36 +01:00
MartinBraquet
4358c15432 Update TODOS 2025-10-29 20:30:56 +01:00
MartinBraquet
fdf8d649fe Link logo to /home page 2025-10-29 20:04:50 +01:00
MartinBraquet
a4e22ec4b1 Add email change 2025-10-29 20:01:42 +01:00
MartinBraquet
783bc43547 Move delete pinned photo 2025-10-29 19:23:33 +01:00
MartinBraquet
2e6aec175a Comment 2025-10-29 17:51:22 +01:00
MartinBraquet
ee41aaa112 Use WithPrivateUser 2025-10-29 17:51:12 +01:00
MartinBraquet
32e8a8b1b9 Add donation name 2025-10-29 17:30:57 +01:00
MartinBraquet
6b3def230b Add liberapay donation link 2025-10-29 17:20:47 +01:00
MartinBraquet
51c46db106 Fix 2025-10-29 17:06:39 +01:00
MartinBraquet
64c18179ac Fix 2025-10-29 16:50:06 +01:00
MartinBraquet
5ee0b39e07 Hide settings when logged out 2025-10-29 16:29:14 +01:00
MartinBraquet
6470319fd6 Add settings page 2025-10-29 16:25:03 +01:00
MartinBraquet
4ca3f3c8ee Delete auth user right away 2025-10-29 16:24:49 +01:00
MartinBraquet
07e2d2d509 Fix red color 2025-10-29 16:21:58 +01:00
MartinBraquet
07d2a143a2 Create base settings page 2025-10-29 12:24:08 +01:00
MartinBraquet
968845492f Comment unused function 2025-10-29 12:23:53 +01:00
MartinBraquet
c2106b64f9 Add webview_oauth_signin.md 2025-10-29 12:08:20 +01:00
MartinBraquet
0d73d1d258 Delete 2025-10-28 18:43:39 +01:00
MartinBraquet
423d425950 Roolback 2025-10-28 18:42:37 +01:00
MartinBraquet
7ab0093fec Sign in failing for android 2025-10-28 18:41:54 +01:00
MartinBraquet
56d2757448 Add styles 2025-10-28 18:32:21 +01:00
MartinBraquet
f5b6037367 Move log 2025-10-28 18:02:45 +01:00
MartinBraquet
2c4ce6c8d1 Add isAndroidWebView 2025-10-28 17:33:25 +01:00
MartinBraquet
f9ccd3628a Add logs 2025-10-28 16:56:45 +01:00
MartinBraquet
abef2b394e Add aiexclude 2025-10-28 16:51:00 +01:00
MartinBraquet
97ff6f1de9 Add base android webview app 2025-10-28 16:05:22 +01:00
MartinBraquet
7fad4435cb Add android readme 2025-10-28 15:16:19 +01:00
MartinBraquet
92a97209ca Add yarn build-web 2025-10-28 15:16:13 +01:00
MartinBraquet
e7c3f083b4 Update NEXT_PUBLIC_LOCAL_ANDROID check 2025-10-28 15:16:02 +01:00
MartinBraquet
7b5961f941 Add log 2025-10-28 01:34:29 +01:00
MartinBraquet
85d4b411b5 Add log 2025-10-28 01:10:51 +01:00
MartinBraquet
c9ec32aca7 Fix APK discrimination 2025-10-28 00:37:02 +01:00
MartinBraquet
13a3013a8e Add @capgo/capacitor-social-login 2025-10-28 00:16:28 +01:00
MartinBraquet
0dd3bac855 Add googleNativeLogin 2025-10-28 00:10:44 +01:00
MartinBraquet
d7a716a5cb Add -H 0.0.0.0 2025-10-28 00:04:02 +01:00
MartinBraquet
0bbc9cbe81 Add IS_LOCAL_ANDROID 2025-10-28 00:03:54 +01:00
MartinBraquet
df766d8d1f Fix typo 2025-10-28 00:02:28 +01:00
MartinBraquet
20a150a228 Remove ; 2025-10-27 21:45:17 +01:00
MartinBraquet
010292a440 Remove cached dataa 2025-10-27 21:44:29 +01:00
MartinBraquet
394dae18e9 Move FCM inside 2025-10-27 21:44:19 +01:00
MartinBraquet
f1676c52f0 Add mobile push notifications 2025-10-27 16:12:51 +01:00
MartinBraquet
05f6f3c79b Install capacitor android 2025-10-27 13:13:52 +01:00
MartinBraquet
9942b488ea Serve mobile app from url 2025-10-27 13:13:41 +01:00
MartinBraquet
5d83f4bf2d Add build web script 2025-10-27 13:13:25 +01:00
MartinBraquet
d9afd914ff Make SEA description dynamic 2025-10-27 12:56:06 +01:00
MartinBraquet
990d8160f8 Use SSR for news 2025-10-27 12:49:08 +01:00
MartinBraquet
80c321b66f Remove log 2025-10-27 12:48:12 +01:00
MartinBraquet
67b45f3e5c Add capacitor 2025-10-27 11:49:39 +01:00
MartinBraquet
ca3cee5673 Move paragraph 2025-10-27 01:47:23 +01:00
MartinBraquet
ae0d170244 Improve wording 2025-10-27 01:46:10 +01:00
MartinBraquet
9a31cfa938 Fixes 2025-10-27 01:34:30 +01:00
MartinBraquet
cdbc9c305e Allow null for profiles 2025-10-27 01:23:20 +01:00
MartinBraquet
cdbce13c49 Sort zod types 2025-10-27 00:39:48 +01:00
MartinBraquet
0a41ebbcda Add an option to disable your profile 2025-10-27 00:37:54 +01:00
MartinBraquet
476fe1602b Clean 2025-10-27 00:37:18 +01:00
MartinBraquet
2f482e9afc Fix links 2025-10-26 18:50:14 +01:00
MartinBraquet
d59e6e0691 Fix news link (open in no tab) 2025-10-26 18:37:44 +01:00
MartinBraquet
7ec6866f26 Add news to FAQ 2025-10-26 16:37:16 +01:00
MartinBraquet
3686e7facf Remove changelog 2025-10-26 16:32:00 +01:00
MartinBraquet
1aba1894ea Remove unused react 2025-10-26 16:24:38 +01:00
MartinBraquet
14503c9b8f Fix header color 2025-10-26 16:24:12 +01:00
MartinBraquet
a315668d31 Fix format 2025-10-26 16:19:42 +01:00
MartinBraquet
db9ea63210 Add what's new page 2025-10-26 16:07:57 +01:00
MartinBraquet
51ecbd5b53 Add dark swagger docs 2025-10-26 15:45:47 +01:00
MartinBraquet
45ef0d9809 Add /donate to /support 2025-10-26 14:37:05 +01:00
MartinBraquet
356702b50d Hide local endpoints when deployed 2025-10-26 14:34:09 +01:00
MartinBraquet
e72ce5376c Remove unused pages 2025-10-26 14:24:48 +01:00
MartinBraquet
0d35f3fbd2 Add bottom margin 2025-10-26 14:24:17 +01:00
MartinBraquet
28c22c1eae Add qr code to FAQ 2025-10-26 14:24:04 +01:00
MartinBraquet
7cf83f65c3 Cache vote creators 2025-10-26 13:53:46 +01:00
MartinBraquet
4c4f2e720d Add status on proposals 2025-10-26 10:51:52 +01:00
MartinBraquet
0fa562e6fd Rename order by 2025-10-26 10:01:36 +01:00
MartinBraquet
bcd0f778cf Rename vote title 2025-10-26 10:01:20 +01:00
MartinBraquet
401ab9f706 Add bottom margin 2025-10-26 09:42:20 +01:00
MartinBraquet
8b09a81d5a Add keyword example 2025-10-25 23:54:50 +02:00
MartinBraquet
86718cc406 Sleep more more more 2025-10-25 23:20:17 +02:00
MartinBraquet
ccb72364e1 Sleep more 2025-10-25 23:08:55 +02:00
MartinBraquet
bfd6a59d87 Increase cache check 2025-10-25 23:08:48 +02:00
MartinBraquet
af4caa455a Fix typos 2025-10-25 23:00:54 +02:00
MartinBraquet
d511e4a75c Comment log 2025-10-25 23:00:46 +02:00
MartinBraquet
8fd906223c Re enable dev cache 2025-10-25 23:00:40 +02:00
MartinBraquet
deadb56aaa Try to load profiles times if not found 2025-10-25 23:00:25 +02:00
MartinBraquet
1ff867879c Throttle discord message 2025-10-25 23:00:02 +02:00
MartinBraquet
f3630dd868 Send generic error message instead of error log 2025-10-25 22:30:08 +02:00
MartinBraquet
39143525c3 Revalidates ISR pages 2025-10-25 22:18:48 +02:00
MartinBraquet
e8dd1f8f8b Add log 2025-10-25 21:48:25 +02:00
MartinBraquet
28e5d2e3f2 Fix profile not found due to cache 2025-10-25 21:40:29 +02:00
MartinBraquet
21def91427 Fix plus icon color 2025-10-25 21:22:29 +02:00
MartinBraquet
bc5d04c662 Set user_id?: string 2025-10-25 16:57:51 +02:00
MartinBraquet
c736227448 Add response to send test email 2025-10-25 16:52:11 +02:00
MartinBraquet
168285cb64 Add contact info to /contact 2025-10-25 16:37:15 +02:00
MartinBraquet
3411f50d29 Add /local/send-test-email endpoint 2025-10-25 16:33:30 +02:00
MartinBraquet
319c14b0e0 Add react import in email
Required since using tsx watch src/serve.ts and not nodemon
2025-10-25 16:32:55 +02:00
MartinBraquet
64c077396f Rename fromEmail 2025-10-25 16:32:07 +02:00
MartinBraquet
65f0d448a1 Fix messages not sent 2025-10-25 15:48:42 +02:00
MartinBraquet
2fdaa464dd Remove import 2025-10-25 15:33:23 +02:00
MartinBraquet
f86a6a10ac Add political_details 2025-10-25 15:32:16 +02:00
MartinBraquet
08a2438e79 Add Animist 2025-10-25 15:31:53 +02:00
MartinBraquet
60cc47f7ca Improve rendering of religion in about me 2025-10-25 14:45:53 +02:00
MartinBraquet
7e4f606492 Improve filter formatting for politics and religion 2025-10-25 14:36:55 +02:00
MartinBraquet
8ff58534d9 Update docs 2025-10-25 14:24:14 +02:00
MartinBraquet
a4bb184e95 Add filter: religion 2025-10-25 14:22:11 +02:00
MartinBraquet
940c1f5692 Add profile field: religion 2025-10-25 14:21:58 +02:00
MartinBraquet
0430733b58 Add internal endpoints to API docs 2025-10-25 13:44:00 +02:00
MartinBraquet
33136816af Improve API docs 2025-10-25 13:31:44 +02:00
MartinBraquet
469a235799 Remove logs 2025-10-25 04:40:11 +02:00
MartinBraquet
2d71c827b3 Fix 2025-10-25 04:30:21 +02:00
MartinBraquet
17f9e72a9f Release 2025-10-25 04:19:29 +02:00
MartinBraquet
120aeed56f Add API to FAQ 2025-10-25 04:19:19 +02:00
MartinBraquet
8128c3b2d7 Update docs 2025-10-25 04:10:37 +02:00
MartinBraquet
4581a33cae Fix package import 2025-10-25 04:10:32 +02:00
MartinBraquet
d43e2af3ae Remove unused openapi 2025-10-25 04:07:46 +02:00
MartinBraquet
0283eb4d85 Massive upgrade to the API Swagger UI @ api.compassmeet.com 2025-10-25 03:42:23 +02:00
MartinBraquet
f483ae42a8 Add smoker to your filters 2025-10-25 01:31:43 +02:00
MartinBraquet
f974eba465 Move Reset filter to top 2025-10-25 01:25:24 +02:00
MartinBraquet
7d7969fe0f Reset filter 2025-10-25 01:22:01 +02:00
MartinBraquet
2a3d7e8362 Fix you filter 2025-10-25 01:21:46 +02:00
MartinBraquet
a38c03c4e0 Fix 2025-10-25 01:02:21 +02:00
MartinBraquet
342a0c612a Add filter for smoking 2025-10-25 00:58:53 +02:00
MartinBraquet
f1f9970407 Either instead of Any Kids 2025-10-24 23:03:41 +02:00
MartinBraquet
c83a3e6315 Log error 2025-10-24 23:03:17 +02:00
MartinBraquet
fbc65e7e2a Update helpers for local run 2025-10-24 23:03:04 +02:00
MartinBraquet
d9e9407cab Update doc 2025-10-24 23:02:36 +02:00
MartinBraquet
d0881b76e0 Update migrate 2025-10-24 23:02:28 +02:00
MartinBraquet
61c867b49c Add info for add new profile field 2025-10-24 23:02:17 +02:00
MartinBraquet
87de30d257 Create share notif 2025-10-24 23:02:03 +02:00
MartinBraquet
817605417c Default to DEV in local 2025-10-24 23:01:50 +02:00
MartinBraquet
65b018db2a Add contact links 2025-10-24 20:48:45 +02:00
MartinBraquet
addb52e3fa Add /donate page with OC widget 2025-10-24 19:24:58 +02:00
MartinBraquet
c3124ec7c3 Clean 2025-10-24 18:52:00 +02:00
MartinBraquet
b1caa6dfdc Remove log 2025-10-24 18:34:55 +02:00
MartinBraquet
26f28d55d9 Add filters for drinks 2025-10-24 18:31:43 +02:00
MartinBraquet
cb66688529 Make left nav bar scrollable 2025-10-24 17:20:52 +02:00
MartinBraquet
40c61f11be Speed update local run 2025-10-24 17:10:47 +02:00
MartinBraquet
9b45c75a5b Remove none 2025-10-24 16:25:16 +02:00
MartinBraquet
09425c1910 Rename college 2025-10-24 16:24:01 +02:00
MartinBraquet
591798e98c Add filter for education level 2025-10-24 16:19:37 +02:00
MartinBraquet
acdd82a680 Clean and factor out get-private-messages 2025-10-24 15:18:04 +02:00
MartinBraquet
5719ac3209 Fix age input 2025-10-24 14:20:55 +02:00
MartinBraquet
2ac687b0c2 Fix 2025-10-24 02:41:10 +02:00
MartinBraquet
a86a249f05 Add height in cm 2025-10-24 02:36:30 +02:00
MartinBraquet
e49a7b0bb4 Add backend support for setting compatibility answers 2025-10-24 02:09:46 +02:00
MartinBraquet
e904a7949c Fix RLS 2025-10-24 01:57:01 +02:00
MartinBraquet
080d8110df Delete bookmarked searches in the backend 2025-10-24 01:35:16 +02:00
MartinBraquet
d90826e851 Create bookmarked searches in the backend 2025-10-24 01:30:03 +02:00
MartinBraquet
e495da692b Improve wording in saved searches 2025-10-24 01:29:32 +02:00
MartinBraquet
52970ef93e Add fkeys to schema 2025-10-24 01:26:50 +02:00
MartinBraquet
8f641d117a Cancel deletions automated by cascade delete 2025-10-24 01:08:39 +02:00
MartinBraquet
d164ebc7da Add foreign keys 2025-10-24 01:04:18 +02:00
MartinBraquet
632cc5810d Add icons 2025-10-23 20:15:21 +02:00
MartinBraquet
e565a6c77f Set last online upon user creation 2025-10-23 18:45:37 +02:00
MartinBraquet
c1fe700d7a Add lib clean 2025-10-23 18:18:01 +02:00
MartinBraquet
06ee267804 Remove log 2025-10-23 18:00:11 +02:00
MartinBraquet
aad722c723 Add chat messages AES encryption 2025-10-23 17:57:19 +02:00
MartinBraquet
aefc58b636 Make mobile filter modal scrollable 2025-10-23 15:52:18 +02:00
MartinBraquet
fdd96507b8 Add politics filter 2025-10-23 15:41:46 +02:00
MartinBraquet
2ad87a5ec5 Update SQL schema 2025-10-23 15:25:06 +02:00
MartinBraquet
b94cdba5af Add diet 2025-10-23 14:58:57 +02:00
MartinBraquet
725261335c Reorder about 2025-10-23 14:16:50 +02:00
MartinBraquet
5fb0051fc6 Update README.md for clarity and minor corrections 2025-10-23 02:25:47 +02:00
MartinBraquet
1247847739 Update FAQ 2025-10-23 02:25:34 +02:00
MartinBraquet
18cb4e74d6 Refactor notification process 2025-10-23 02:07:45 +02:00
MartinBraquet
e07cb7fca9 Show all notifs types 2025-10-23 01:46:57 +02:00
MartinBraquet
dc54ed46f8 Add notif for New Vote Page 2025-10-23 01:38:34 +02:00
MartinBraquet
0415d86d71 Replace DROP INDEX with IF NOT EXISTS 2025-10-23 00:34:34 +02:00
MartinBraquet
b8b95be5ce Add on delete cascade 2025-10-23 00:29:14 +02:00
MartinBraquet
46820f0986 Clean 2025-10-23 00:08:54 +02:00
MartinBraquet
dcc022ac7f Add open Collective 2025-10-23 00:05:27 +02:00
MartinBraquet
9142f0d633 Autofocus 2025-10-22 23:49:28 +02:00
MartinBraquet
181c72befe Format 2025-10-22 23:49:18 +02:00
MartinBraquet
99f3459978 Remove user followers in search-users 2025-10-22 23:33:25 +02:00
MartinBraquet
75fbc9679c Fix page props 2025-10-22 23:33:12 +02:00
MartinBraquet
700b7774b1 Margin 404 2025-10-22 23:32:37 +02:00
MartinBraquet
d9f0a9b1ca Fix "younger than 99" 2025-10-22 22:59:25 +02:00
MartinBraquet
70644ff26d Prevent editor focus on mobile devices 2025-10-22 16:22:50 +02:00
MartinBraquet
bbefcc3bc8 Add indexes to push_subscriptions 2025-10-22 15:33:55 +02:00
MartinBraquet
09767dbae3 Add WPA to faq 2025-10-22 15:33:42 +02:00
MartinBraquet
57eafa95ba Fire and forget other user notif 2025-10-22 15:00:41 +02:00
MartinBraquet
f4f28a411e Typo 2025-10-22 15:00:27 +02:00
MartinBraquet
f6059ef5c7 Focus on editor upon page loading 2025-10-22 14:47:07 +02:00
MartinBraquet
e3fa4efa95 Remove expired subscriptions 2025-10-22 14:28:59 +02:00
MartinBraquet
6884a91eb8 Ignore ico 2025-10-22 14:28:23 +02:00
MartinBraquet
71ba018a42 Remove misplace favicon 2025-10-22 13:57:24 +02:00
MartinBraquet
10f5232ac3 Clean 2025-10-22 13:54:56 +02:00
MartinBraquet
78d707484d Update badge 2 2025-10-22 13:54:47 +02:00
MartinBraquet
69db66fbbb Update badge 2025-10-22 13:32:53 +02:00
MartinBraquet
99691cd7ee Factor out vapid key 2025-10-22 13:23:14 +02:00
MartinBraquet
47cef359ca Put back icons 2025-10-22 03:57:57 +02:00
MartinBraquet
046105498f Hide icons 2025-10-22 03:53:07 +02:00
MartinBraquet
4d3ef5dd2a Log payload 2025-10-22 03:43:56 +02:00
MartinBraquet
8bcd5623bf Add service worker info 2025-10-22 03:43:46 +02:00
MartinBraquet
a29b4a3a8e Remove badge 2025-10-22 03:40:15 +02:00
MartinBraquet
dee0fb396b Remove cache 2025-10-22 03:39:35 +02:00
MartinBraquet
b5c707e07f Send notif for every single private message 2025-10-22 03:25:04 +02:00
MartinBraquet
8fe35bd1d7 Add VAPID secrets 2025-10-22 03:02:34 +02:00
MartinBraquet
6c864c35cd Implement subscriptions for mobile notifications 2025-10-22 02:52:41 +02:00
MartinBraquet
f00acf6af1 Improve link accessibility 2025-10-22 01:18:45 +02:00
MartinBraquet
49e1599bc4 Fix react error 2025-10-22 00:44:22 +02:00
MartinBraquet
7311d4b724 Remove cache 2025-10-22 00:38:56 +02:00
MartinBraquet
fa44e348a2 Fix WPA theme color 2025-10-22 00:22:28 +02:00
MartinBraquet
8cba02741c Fix WPA 2025-10-22 00:01:58 +02:00
MartinBraquet
48d04d5e72 Add WPA 2025-10-21 23:37:20 +02:00
MartinBraquet
7cac25c0e2 Remove react import (2) 2025-10-21 20:50:56 +02:00
MartinBraquet
88b0fa0163 Remove react import 2025-10-21 20:32:35 +02:00
MartinBraquet
3fcef24cc9 Add SEO (including better tab titles) 2025-10-21 20:29:10 +02:00
MartinBraquet
d9fba6ce6b Fix link 2025-10-21 12:40:28 +02:00
MartinBraquet
8bc2f0c40e Fix flashing submit button 2025-10-21 12:36:54 +02:00
MartinBraquet
21254695d5 Remove unused import 2025-10-21 12:32:39 +02:00
MartinBraquet
f063f0a6f4 Add donate links 2025-10-21 12:32:00 +02:00
MartinBraquet
2d847cbcdb Add Github sponsor 2025-10-21 12:15:27 +02:00
MartinBraquet
547e99f526 Add vercel config for backup info 2025-10-20 17:36:55 +02:00
MartinBraquet
a9794cd2ee Add packages 2025-10-20 17:24:42 +02:00
MartinBraquet
c651abd8ae Add unmet deps 2025-10-20 16:45:02 +02:00
MartinBraquet
15781475b6 Rename Love to Profile 2025-10-20 16:35:59 +02:00
MartinBraquet
26a28175fd Rename LovePage 2025-10-20 16:24:22 +02:00
MartinBraquet
aa3680934b Rename lover to profile 2025-10-20 16:18:49 +02:00
MartinBraquet
0b36586ddf Rename love folder in backend 2025-10-20 16:14:23 +02:00
MartinBraquet
7b58acac0d Rename love folder 2025-10-20 16:13:43 +02:00
MartinBraquet
27bf4eadf9 Rename compatibility_answers_free 2025-10-20 16:10:41 +02:00
MartinBraquet
c8d4353888 Rename love_waitlist 2025-10-20 16:05:14 +02:00
MartinBraquet
4876ca2643 Rename love_stars 2025-10-20 16:03:28 +02:00
MartinBraquet
e06a382c94 Rename love_ships 2025-10-20 16:01:33 +02:00
MartinBraquet
d1a421ca15 Rename compatibility_prompts 2025-10-20 15:59:44 +02:00
MartinBraquet
cd3c8d89d0 Rename love_likes 2025-10-20 15:48:22 +02:00
MartinBraquet
1f943ccead Rename love_compatibility_answers 2025-10-20 15:46:16 +02:00
MartinBraquet
753776fa9a Fix 2025-10-20 15:37:33 +02:00
MartinBraquet
9787a2446e Fix 2025-10-20 15:31:39 +02:00
MartinBraquet
4cb29d274b Add floating info box 2025-10-20 15:28:59 +02:00
MartinBraquet
df55d63f99 Add security and help pages 2025-10-20 13:31:58 +02:00
MartinBraquet
236e2d48c5 Update security email 2025-10-20 13:20:31 +02:00
MartinBraquet
30d45d834f Add compat prompts number to stats 2025-10-20 13:03:26 +02:00
MartinBraquet
edf30897f2 Fix effect 2025-10-20 12:56:42 +02:00
MartinBraquet
3d31ebb576 Fix hooks 2025-10-20 12:55:16 +02:00
MartinBraquet
d3bac8bcc0 Fix vercel badge 2025-10-20 12:53:37 +02:00
MartinBraquet
a360f80cdf Add warning message about short bio 2025-10-20 12:51:28 +02:00
MartinBraquet
0cc7549546 remove log 2025-10-20 12:51:10 +02:00
MartinBraquet
283d2743e0 Fix mod 2025-10-19 22:50:09 +02:00
MartinBraquet
b431fa11fa Fix pref_gender filtering 2025-10-19 21:42:37 +02:00
MartinBraquet
648e00867f Reduce v space 2025-10-19 21:30:44 +02:00
MartinBraquet
552af7bb6b Add romantic type in filters 2025-10-19 21:25:17 +02:00
MartinBraquet
92980f7c79 Increase contrast 2025-10-19 20:19:52 +02:00
MartinBraquet
09a563bf73 Clean 2025-10-19 20:19:45 +02:00
MartinBraquet
141fa12a20 Add kids strength and other relationship filters 2025-10-19 12:36:48 +02:00
MartinBraquet
6e0035d4f3 Show profiles that don't set kids strength 2025-10-19 12:36:09 +02:00
MartinBraquet
97bac4132c Fix loc filter clearing 2025-10-19 11:11:51 +02:00
MartinBraquet
b23b0280cd Fix mobile nav contrast 2025-10-19 11:00:49 +02:00
MartinBraquet
7ac093a8d0 Fix socials contrast 2025-10-19 10:58:12 +02:00
MartinBraquet
dfc524b957 Fix 2025-10-19 00:32:48 +02:00
MartinBraquet
65ba0d348b Increase contrast for better accessibility 2025-10-19 00:11:59 +02:00
MartinBraquet
ed07031539 Fix primary key 2025-10-18 23:31:32 +02:00
MartinBraquet
93f3690344 Speed up init theme 2025-10-18 22:47:43 +02:00
MartinBraquet
1341d1356a Fix warning 2025-10-18 22:37:46 +02:00
MartinBraquet
38dcf16c03 customlink -> custom-link 2025-10-18 22:36:37 +02:00
MartinBraquet
8696a42959 SSH and view logs in one click 2025-10-18 22:36:12 +02:00
MartinBraquet
c6fc7db1e9 Move up 2025-10-18 13:17:44 +02:00
MartinBraquet
58540aca57 Add proposals and votes number to stats 2025-10-18 13:16:05 +02:00
MartinBraquet
b7b75279c2 Add warning message upon console opening 2025-10-18 12:49:36 +02:00
MartinBraquet
204a35d026 Release 1.4 2025-10-18 12:24:13 +02:00
MartinBraquet
fb2841f198 Update compat answer box 2025-10-18 12:23:47 +02:00
MartinBraquet
5de055c977 Improve importance radio contrast 2025-10-18 12:14:35 +02:00
MartinBraquet
084659ea3d Remove debug 2025-10-18 12:14:20 +02:00
MartinBraquet
c1a414afab Make votes sortable 2025-10-18 11:49:54 +02:00
MartinBraquet
a5747034d6 Fix props name 2025-10-18 10:40:35 +02:00
MartinBraquet
fda52fec97 Move proposal up and hide by default 2025-10-18 10:39:50 +02:00
MartinBraquet
e38ec79618 remove quote 2025-10-18 02:50:05 +02:00
MartinBraquet
1ef125db12 Fix md format 2025-10-18 02:46:07 +02:00
MartinBraquet
b580b640bd Remove unused react 2025-10-18 02:39:24 +02:00
MartinBraquet
214bddaca4 Add contact links 2025-10-18 02:36:54 +02:00
MartinBraquet
065d489869 Add contact form 2025-10-18 02:20:31 +02:00
MartinBraquet
46ffefbbb9 Add anonymous option for votes 2025-10-18 00:53:35 +02:00
MartinBraquet
a19db3bca9 Clean 2025-10-18 00:20:32 +02:00
MartinBraquet
2c8d8d9989 Clean 2025-10-18 00:12:53 +02:00
MartinBraquet
d52943e31e Fix 2025-10-17 23:24:08 +02:00
MartinBraquet
3eababb742 Fix 2025-10-17 23:19:20 +02:00
MartinBraquet
8a954d3c20 Add voting / proposal page 2025-10-17 23:15:15 +02:00
MartinBraquet
8516901032 Allow get notified for anyone 2025-10-17 19:04:57 +02:00
MartinBraquet
3f2d246fec Fix short bios not showing when sorting by compatibility 2025-10-17 16:55:44 +02:00
MartinBraquet
58fdaa26ca Move to distance filtering to improve accuracy and speed 2025-10-17 16:43:27 +02:00
MartinBraquet
7dc1a8790d Fix 2025-10-17 14:53:55 +02:00
MartinBraquet
70c9ec1d73 Show full political names 2025-10-17 14:02:04 +02:00
MartinBraquet
2bcbbc96ad Add political options 2025-10-17 13:55:48 +02:00
MartinBraquet
527d36a159 Move want kids closer to connection type 2025-10-17 13:50:46 +02:00
MartinBraquet
2ce21247ee Add romantic style (poly, mono, other) 2025-10-17 13:42:32 +02:00
MartinBraquet
8ea6c406e0 Add webhook to report to discord 2025-10-16 20:59:46 +02:00
MartinBraquet
e22f50ecd3 Show loading indicator 2025-10-16 15:28:29 +02:00
MartinBraquet
20dcd98fdf Allow unauth requests to get-messages-count (used in public stats) 2025-10-16 15:16:21 +02:00
MartinBraquet
bc5708857a Improve onboarding UI 2025-10-16 14:37:17 +02:00
MartinBraquet
b9c045ebfb Do not render sign up button, redirect to home 2025-10-16 14:22:53 +02:00
MartinBraquet
c69bd7018e Use compass loading sign 2025-10-16 14:08:34 +02:00
MartinBraquet
078d149175 Redirect to profiles grid after sign up 2025-10-16 14:08:24 +02:00
MartinBraquet
be9f0bd061 Add 20 core compatibility prompts 2025-10-16 13:48:00 +02:00
MartinBraquet
a4723563f5 Keep blue loading circle for buttons 2025-10-16 13:41:03 +02:00
MartinBraquet
1fdcd24f28 Improve design of loading indicator 2025-10-16 12:42:38 +02:00
MartinBraquet
a43480db92 Increase API_RATE_LIMIT_PER_MIN_UNAUTHED 2025-10-16 01:27:24 +02:00
MartinBraquet
e85a072f1c Add user loaded log 2025-10-16 01:24:05 +02:00
MartinBraquet
bbfa2a4eab Wait longer for user to appear 2025-10-16 01:23:12 +02:00
MartinBraquet
2f2db4ded8 Rollback toast error as it shows randomly 2025-10-16 00:48:38 +02:00
MartinBraquet
7296a0d2cd Remove rate limit for endpoints not prone to scraping 2025-10-16 00:41:21 +02:00
MartinBraquet
08e02b6ac0 Add too many requests toast 2025-10-16 00:28:25 +02:00
MartinBraquet
715811d7fd Commetn 2025-10-16 00:28:10 +02:00
MartinBraquet
c7d6ae6995 Hide log 2025-10-16 00:27:58 +02:00
MartinBraquet
b1d1396944 Fix import 2025-10-15 23:52:46 +02:00
MartinBraquet
25a319710e Fix import 2025-10-15 23:50:27 +02:00
MartinBraquet
796b13dd62 Add toast error for too many requests 2025-10-15 23:47:01 +02:00
MartinBraquet
8197863ac5 Clean auth and rate limiting 2025-10-15 23:37:24 +02:00
MartinBraquet
89bd164d43 Add authed 2025-10-15 23:20:20 +02:00
MartinBraquet
80d7061e5f Pre commit 2025-10-15 22:50:50 +02:00
MartinBraquet
c49bac3a09 Make API calls authed 2025-10-15 22:42:26 +02:00
MartinBraquet
06d53fe801 Redirect if logged out in /notifications 2025-10-15 22:32:58 +02:00
MartinBraquet
15ba529938 Fix "column reference "user_id" is ambiguous" 2025-10-15 19:26:10 +02:00
MartinBraquet
83054d0cd1 Fix link opening in same tab 2025-10-15 17:04:00 +02:00
MartinBraquet
8da486adf2 Optimistically remove starred profile upon deletion 2025-10-15 16:01:11 +02:00
MartinBraquet
32bc3847fa Add option to save / bookmark profiles 2025-10-15 15:45:47 +02:00
MartinBraquet
5d763c18c8 Comment log 2025-10-15 15:45:27 +02:00
MartinBraquet
bd3920cfff Fix pagination for last active sorting 2025-10-14 22:09:20 +02:00
MartinBraquet
06d94332b6 Update keyword search placeholder names 2025-10-14 21:35:37 +02:00
MartinBraquet
50614484d8 Move last above social links 2025-10-14 21:06:08 +02:00
MartinBraquet
c29d3d8c92 Clean 2025-10-14 20:51:06 +02:00
MartinBraquet
26f46af375 Fix unused botBadge 2025-10-14 20:49:36 +02:00
MartinBraquet
32b1491dd0 Fix unused node 2025-10-14 20:48:50 +02:00
MartinBraquet
51b8a6c80a Fix unused open 2025-10-14 20:48:21 +02:00
MartinBraquet
0f63d6d3a0 Remove unused react imports 2025-10-14 20:42:41 +02:00
MartinBraquet
4771b08773 Show and write when the user was last online in their profile 2025-10-14 20:36:58 +02:00
MartinBraquet
9b880101fd Make plot larger on mobile 2025-10-14 19:56:35 +02:00
MartinBraquet
594806d6e8 Improve charts design 2025-10-14 19:37:38 +02:00
MartinBraquet
e9afd4db2f Regen supabase types 2025-10-14 19:11:26 +02:00
MartinBraquet
b23efe4089 Add active members tile and move /charts to /stats 2025-10-14 19:09:56 +02:00
MartinBraquet
e33be41a93 Store last_online_time in user_activity.sql table instead of profiles 2025-10-14 19:09:08 +02:00
MartinBraquet
33b09df872 Reduce chart height 2025-10-14 19:03:30 +02:00
MartinBraquet
e9050d0aa0 Simplify and make grid for organization.tsx 2025-10-14 18:57:59 +02:00
MartinBraquet
baeb2a33fe Simplify and make grid for social 2025-10-14 18:57:52 +02:00
MartinBraquet
4ad89acdc7 Use last_online_time from user_activity instead of profiles 2025-10-14 17:53:29 +02:00
MartinBraquet
7d87af8f5c Add user_activity.sql 2025-10-14 17:52:09 +02:00
MartinBraquet
65c0e84e2a Do not prepend social url if full url is provided 2025-10-14 11:28:48 +02:00
MartinBraquet
7b15d85871 Do not render protocol and subdomain in socials 2025-10-14 11:28:21 +02:00
MartinBraquet
ad8ec0f4fd Add browser dev info 2025-10-14 11:27:39 +02:00
MartinBraquet
2d05d83dd0 Always show relationship questions when connection type includes relationships 2025-10-14 11:02:50 +02:00
MartinBraquet
bd45066b13 Add IDE note 2025-10-13 19:43:06 +02:00
MartinBraquet
8ee4274054 Rename voting members 2025-10-13 19:13:13 +02:00
MartinBraquet
83a7ed4d6b Add stats to organization page 2025-10-13 19:05:41 +02:00
MartinBraquet
07dbd86ac6 Add How fast is Compass growing? to FAQ 2025-10-13 18:49:05 +02:00
MartinBraquet
0e671d2cc0 Add nice stats 2025-10-13 18:42:39 +02:00
MartinBraquet
2d6d3c04ce Add stat box 2025-10-13 18:42:06 +02:00
MartinBraquet
b0148963c7 Remove bookmarked_searches and love_compatibility_answers upon account deletion 2025-10-13 17:35:12 +02:00
MartinBraquet
13356950f3 Wait instead of thread resend emails 2025-10-13 17:27:59 +02:00
MartinBraquet
629bcb30a7 Add health discord webhook and send error message there 2025-10-13 15:13:05 +02:00
MartinBraquet
03721fff1c Do not pass orderBy when processing saved searches 2025-10-13 15:12:24 +02:00
MartinBraquet
2a6911ae3d Move email links to our domain 2025-10-13 13:34:10 +02:00
MartinBraquet
164eddecab Release v1.3.0 2025-10-13 12:39:38 +02:00
MartinBraquet
9eacb38eb9 Show bio in discord message of profile creation 2025-10-13 12:26:40 +02:00
MartinBraquet
20f5cfb9a7 Fix demo 2025-10-13 10:49:19 +02:00
Martin Braquet
6c6c1cc90a Upgrade geodb plan to increase radius and page limit (#14)
* Upgrade geodb plan to increase radius and page limit

* Speed location debounce
2025-10-12 15:58:52 +02:00
MartinBraquet
a32c099cc1 Rename social 2025-10-12 14:54:58 +02:00
MartinBraquet
fe2f832e83 Improve support page 2025-10-12 14:52:39 +02:00
MartinBraquet
868746cc23 Use Atkinson Hyperlegible font 2025-10-12 14:35:59 +02:00
MartinBraquet
3be7a54284 Fix compat modal (had to scroll to see Next on mobile) 2025-10-12 13:13:01 +02:00
MartinBraquet
635e1ec8e2 Add TODO readme info 2025-10-11 23:23:52 +02:00
MartinBraquet
a638a35a76 Upgrade ban logic 2025-10-11 22:58:50 +02:00
MartinBraquet
8cc33d3418 Add massive upgrade text 2025-10-11 21:42:46 +02:00
MartinBraquet
9947f7b967 Fix 2025-10-11 21:42:33 +02:00
MartinBraquet
daf5350f41 Add stem vector search in bio 2025-10-11 21:15:03 +02:00
MartinBraquet
020b9ddb8d Fix 2025-10-11 19:56:54 +02:00
MartinBraquet
23aff9497a Fix 2025-10-11 19:54:25 +02:00
MartinBraquet
3c119396f3 Add demo 2025-10-11 19:51:48 +02:00
MartinBraquet
f7c7c47ac0 Remove backup info from git 2025-10-11 19:44:38 +02:00
MartinBraquet
dbe2369bbe Fix avatar link 2025-10-11 19:44:12 +02:00
MartinBraquet
4e8033d221 Add info about contact 2025-10-11 19:44:02 +02:00
MartinBraquet
97a0f87cbd Use georgia font 2025-10-11 12:15:26 +02:00
MartinBraquet
bfa2713d43 Fix wording 2025-10-11 11:46:38 +02:00
MartinBraquet
fe5e109751 Improve reading 2025-10-10 22:53:30 +02:00
MartinBraquet
8cc96030b1 Speed up placeholder 2025-10-10 22:52:09 +02:00
MartinBraquet
a2b172ad58 Improve charts 2025-10-10 21:31:30 +02:00
MartinBraquet
e756225d8b Move pics above endorsements 2025-10-10 20:58:06 +02:00
MartinBraquet
dd803b604f Update reserved paths 2025-10-10 20:20:10 +02:00
MartinBraquet
b5c961c8ee Hide complete profile button 2025-10-10 20:05:01 +02:00
MartinBraquet
47cd9d227e Add shortBio filter to mobile filters 2025-10-10 19:13:32 +02:00
MartinBraquet
e2be3aafcd Add shortBio filter 2025-10-10 19:03:57 +02:00
MartinBraquet
015fe76c44 Hide profiles with small bio 2025-10-10 18:33:47 +02:00
MartinBraquet
44666aec03 Update post install 2025-10-10 18:33:11 +02:00
MartinBraquet
6a265e4f35 Do not render home before user loads 2025-10-10 18:32:37 +02:00
MartinBraquet
12c7316524 Refactor buttons 2025-10-10 17:04:26 +02:00
MartinBraquet
dcf9741d69 Format required form as step by step onboarding 2025-10-10 16:46:17 +02:00
MartinBraquet
63dd1fdd50 Replace user with voting member and member with volunteer for clarity and inclusion 2025-10-10 15:22:55 +02:00
MartinBraquet
5aa166bbfd Open links in same tab 2025-10-10 15:12:48 +02:00
MartinBraquet
34cbf7093e Skip welcome email if local 2025-10-10 14:51:22 +02:00
MartinBraquet
159d58949e Reformat 2025-10-09 21:51:21 +02:00
MartinBraquet
fcf802b7e3 Refactor bios and add character counter 2025-10-09 21:51:08 +02:00
MartinBraquet
92ff6dadb0 Add email 2025-10-09 20:00:01 +02:00
MartinBraquet
05fa2f9883 Add socials and organization pages 2025-10-09 19:47:32 +02:00
MartinBraquet
71bb8fd784 Commetn 2025-10-09 19:33:51 +02:00
MartinBraquet
16ffd6dfab Fix message view without sign in 2025-10-09 19:30:24 +02:00
MartinBraquet
2661d15910 Remove waitlist 2025-10-09 19:17:15 +02:00
MartinBraquet
394102bb93 Fix avatar icon 2025-10-09 18:37:11 +02:00
MartinBraquet
3585b12dfd Remove maintenance banner 2025-10-09 18:20:53 +02:00
MartinBraquet
423d87d5f1 Remove logs 2025-10-09 18:19:14 +02:00
MartinBraquet
13b13b1104 Fix 2025-10-09 18:02:15 +02:00
MartinBraquet
a77e7b96b7 Move logs to debug status 2025-10-09 17:59:10 +02:00
MartinBraquet
d7213c255c Add client side heartbeat 2025-10-09 17:50:43 +02:00
MartinBraquet
ddeb1dcdb7 Improve ping pong connection duration 2025-10-09 17:38:05 +02:00
MartinBraquet
221cfa3528 Fix websockets not reaching the container and remove v0/ prefix 2025-10-09 16:58:20 +02:00
MartinBraquet
d6f6348ff1 Add maintenance banner 2025-10-09 16:25:47 +02:00
MartinBraquet
0c6afdc98e Add star 2025-10-09 15:14:38 +02:00
MartinBraquet
02a2148b3f Improve messages width 2025-10-09 13:14:38 +02:00
MartinBraquet
36a02268d8 Fix 2025-10-09 11:28:11 +02:00
MartinBraquet
450f07f505 Add private backup 2025-10-09 11:27:24 +02:00
MartinBraquet
777eba9fed Move backup to private storage 2025-10-09 11:23:30 +02:00
MartinBraquet
eaa8fa57d1 Add private bucket 2025-10-09 11:18:37 +02:00
MartinBraquet
200bf479e1 Clean 2025-10-09 00:43:27 +02:00
MartinBraquet
331f409af9 Increase debounce 2025-10-09 00:28:27 +02:00
MartinBraquet
ce875a5e63 Fix Porto not showing 2025-10-09 00:16:58 +02:00
MartinBraquet
638013f835 Update email address 2025-10-08 23:44:37 +02:00
MartinBraquet
1de87cbfec Add welcome email 2025-10-08 23:40:53 +02:00
MartinBraquet
7f3428b36a Factor out unsubscribe url 2025-10-08 23:40:37 +02:00
MartinBraquet
35595ded47 Fix bullets 2025-10-08 23:40:18 +02:00
MartinBraquet
35e9264017 Show profiles number, not users number 2025-10-08 23:39:42 +02:00
MartinBraquet
02d33c8f83 Rename mock user 2025-10-08 20:38:31 +02:00
MartinBraquet
f229ebc3a8 Add email confirmation 2025-10-08 20:38:20 +02:00
MartinBraquet
0062351f6d Add welcome email 2025-10-08 20:38:09 +02:00
MartinBraquet
e86f6798ec Fix bullet 2025-10-08 20:37:35 +02:00
MartinBraquet
4f53f7136b Add members 2025-10-08 20:35:57 +02:00
MartinBraquet
d80b982dde Simplify tab title 2025-10-08 17:32:19 +02:00
MartinBraquet
24788aa9af Add optional Garamond font 2025-10-08 14:11:51 +02:00
MartinBraquet
9ffae658df Clean 2025-10-08 11:58:58 +02:00
MartinBraquet
82ad573cac Add stoat link 2025-10-08 11:58:52 +02:00
MartinBraquet
36bf7ad65b Fix 2025-10-07 22:53:37 +02:00
MartinBraquet
b30af128c7 Release 2025-10-07 22:11:34 +02:00
MartinBraquet
72c31ae097 Add compat score math video link 2025-10-07 21:25:08 +02:00
Martin Braquet
d2c608021d Improve home (#10) 2025-10-05 10:38:04 +02:00
Martin Braquet
1f36fb2413 Update faq.md 2025-10-05 10:14:09 +02:00
Martin Braquet
16a0cbcecf Update about.tsx 2025-10-05 10:00:30 +02:00
Martin Braquet
e068e246aa Update faq.md 2025-10-03 13:58:31 +02:00
MartinBraquet
ec7c77fcf9 Log 2025-10-02 15:14:13 +02:00
MartinBraquet
46a338b874 Clean 2025-10-02 14:54:47 +02:00
MartinBraquet
bfee7ff09d Fix age rendering 2025-10-02 14:14:24 +02:00
MartinBraquet
ce1305d8ae Host logos 2025-10-02 13:17:22 +02:00
MartinBraquet
aaebf88438 Log profile count 2025-10-01 09:02:50 +02:00
MartinBraquet
dde2c99e36 Fix 2025-10-01 08:58:11 +02:00
MartinBraquet
4dc2f3b9b9 Add Community Growth over Time 2025-10-01 08:56:03 +02:00
MartinBraquet
f30cfffb86 Fix bio parsing grid 2025-09-30 22:17:28 +02:00
MartinBraquet
ca3eb62ba7 Fix 2 2025-09-30 21:48:15 +02:00
MartinBraquet
c8e55ca4ce Fix 2025-09-30 21:46:31 +02:00
MartinBraquet
e4acb25a40 Better render headings and lists in profile grid 2025-09-30 21:45:09 +02:00
MartinBraquet
c741e10139 Stop spamming prod discord channels 2025-09-30 21:30:38 +02:00
MartinBraquet
28d0b35f8e Move discord move to create profile 2025-09-30 20:53:26 +02:00
MartinBraquet
f7f09cd9e5 Send message to Discord when reaching 50, 100, ..., users 2025-09-28 21:09:11 +02:00
MartinBraquet
501c92c350 Send discord message at every profile creation 2025-09-28 20:47:31 +02:00
MartinBraquet
f021101322 Remove unused React 2025-09-26 22:17:24 +02:00
MartinBraquet
369265bc2c Add firebase storage backup script 2025-09-26 22:12:08 +02:00
Martin Braquet
b1f1e5db1f Fully delete profile in database and Firebase auth + storage (#7) 2025-09-26 22:10:38 +02:00
MartinBraquet
51d32e5afb Link donations in FAQ 2025-09-26 13:52:40 +02:00
MartinBraquet
f396e8e482 Your filters 2025-09-26 13:51:08 +02:00
MartinBraquet
077321731e Add bio warning message 2025-09-25 23:02:04 +02:00
MartinBraquet
60eb0c6978 Add 'datingdoc', 'friendshipdoc', 'connectiondoc', 'workdoc' 2025-09-25 22:49:54 +02:00
MartinBraquet
475f0af78a Add supabase backup VM to Firebase storage and discord error hook 2025-09-24 15:59:46 +02:00
MartinBraquet
206fa07035 Ignore tf 2025-09-24 15:13:33 +02:00
MartinBraquet
aff949714c Add charts example 2025-09-24 13:28:46 +02:00
Martin Braquet
7e834b9ff6 Update FUNDING.yml 2025-09-24 12:46:20 +02:00
Martin Braquet
19bad26a98 Create FUNDING.yml 2025-09-24 12:44:37 +02:00
MartinBraquet
7cc7c8d27b Hide age, city and gender if null 2025-09-22 11:06:52 +02:00
MartinBraquet
ae5a8c7cfa Add page loading warning 2025-09-22 11:02:53 +02:00
MartinBraquet
5004b73210 Fix calendly link 2025-09-22 10:52:46 +02:00
MartinBraquet
02f613d269 Add Ko-fi donation link 2025-09-22 10:42:23 +02:00
MartinBraquet
439ac0310b Add option to delete an answered compatibility prompt 2025-09-22 10:09:13 +02:00
MartinBraquet
3e95467819 Add okcupid and calendly links 2025-09-22 00:06:05 +02:00
MartinBraquet
90522cb88b Comment log 2025-09-22 00:05:15 +02:00
MartinBraquet
af39b01d4a Reload env config after setting env vars 2025-09-21 16:56:42 +02:00
MartinBraquet
73a0a5ff0b Fix vercel env 2025-09-21 16:08:29 +02:00
MartinBraquet
e157f500bc Fix dev firebase admin key 2025-09-21 16:00:29 +02:00
MartinBraquet
274ee5ed5f Remove json 2025-09-21 15:44:33 +02:00
MartinBraquet
4cb11ba8c0 Remove log 2025-09-21 15:00:26 +02:00
MartinBraquet
7b8e775139 Fix google cloud env 2025-09-21 14:55:32 +02:00
MartinBraquet
86a7d26bfd Clean readme 2025-09-20 23:54:56 +02:00
MartinBraquet
84a437772d Make local DEV work out of the box 2025-09-20 23:51:28 +02:00
MartinBraquet
d7c95e2ae0 Clean ENV 2025-09-20 18:26:03 +02:00
MartinBraquet
b4f0ef8b43 Move supabase dev pwd inside code 2025-09-20 18:12:46 +02:00
MartinBraquet
6d30cd7ae4 Add open source to FAQ 2025-09-19 14:47:58 +02:00
MartinBraquet
f631236ee7 Add platform to FAQ 2025-09-19 14:42:36 +02:00
MartinBraquet
1a58ff5c4c Shuffle prompts 2025-09-18 15:33:23 +02:00
MartinBraquet
73aca913a1 Fix 2025-09-18 14:14:58 +02:00
MartinBraquet
24dee0cad6 Add bio char template 2025-09-18 13:34:29 +02:00
MartinBraquet
2d2de75372 Add bio tips 2025-09-18 13:12:48 +02:00
MartinBraquet
d98982e6fd Factor out links 2025-09-18 11:30:59 +02:00
MartinBraquet
14c12ffb08 Rename 2025-09-18 11:19:09 +02:00
MartinBraquet
f260afca11 Ignore 2025-09-18 11:18:22 +02:00
MartinBraquet
5bcbe25d97 Rm 2025-09-18 11:18:04 +02:00
MartinBraquet
2eee366fbd Update financials 2025-09-18 11:12:33 +02:00
MartinBraquet
85d57ec5e6 Fix resend email limit 2/sec 2025-09-17 18:35:29 +02:00
MartinBraquet
502c878f82 Fix 2025-09-17 18:15:02 +02:00
MartinBraquet
1136c3f767 Release 2025-09-17 17:58:59 +02:00
MartinBraquet
42b496cd77 Fix 2025-09-17 17:58:11 +02:00
MartinBraquet
4acb5ee020 Fix warnings 2025-09-17 15:56:17 +02:00
MartinBraquet
ea18781cc6 Rename lover -> profile 2025-09-17 15:51:19 +02:00
MartinBraquet
593617c0ff Rename Lover -> Profile 2025-09-17 15:43:19 +02:00
MartinBraquet
c6a139d88d Rename Lovers -> Profiles 2025-09-17 15:42:23 +02:00
MartinBraquet
b7357a4546 Set 3 words 2025-09-17 15:25:27 +02:00
MartinBraquet
5eac959d15 FIx paypal path 2025-09-17 15:20:49 +02:00
MartinBraquet
74c86ecfbe FIx path 2025-09-17 15:19:17 +02:00
MartinBraquet
f353e590e1 Rename lovers -> profiles 2025-09-17 15:11:53 +02:00
MartinBraquet
a4cc3e10c2 Upgrade keywords examples 2025-09-17 12:00:24 +02:00
MartinBraquet
7321f56ee2 Upgrade docs 2025-09-16 23:49:17 +02:00
MartinBraquet
8800d9adc6 Upgrade README.md 2025-09-16 23:37:45 +02:00
MartinBraquet
22cd535527 Upgrade README.md 2025-09-16 23:35:02 +02:00
MartinBraquet
1d0e9592df Replace github sponsors 2025-09-16 22:53:53 +02:00
MartinBraquet
2ef4af0ff2 Fix links submit your own 2025-09-16 22:49:53 +02:00
MartinBraquet
542a6b1592 Fix endpoint 2025-09-16 22:42:37 +02:00
MartinBraquet
613ef94dba Upgrade docs endpoint 2025-09-16 22:28:40 +02:00
MartinBraquet
1dc2a1fadf Fix 2025-09-16 22:12:19 +02:00
MartinBraquet
41a606f5c1 Fix ages in filters 2025-09-16 22:09:43 +02:00
MartinBraquet
7b2b9855f9 Fix notifs page 2025-09-16 21:54:29 +02:00
MartinBraquet
b2b519ba2e Clean 2025-09-16 21:38:52 +02:00
MartinBraquet
5cf89392ff Fix username not being updated when loading their profile after registration 2025-09-16 21:38:19 +02:00
MartinBraquet
0f05304ec3 Fix typo 2025-09-16 21:07:02 +02:00
MartinBraquet
87bc962c88 Fix Rename github org 2025-09-16 18:59:21 +02:00
MartinBraquet
546ce6e229 Revert "Rename github org"
This reverts commit 2163d5aaf6.
2025-09-16 18:58:29 +02:00
MartinBraquet
2163d5aaf6 Rename github org 2025-09-16 18:49:36 +02:00
MartinBraquet
905ea160f2 Fix link 2025-09-16 18:42:48 +02:00
MartinBraquet
675f4a372b Update paypal links 2025-09-16 18:41:32 +02:00
MartinBraquet
7ff42db0c6 Curl API 2025-09-16 18:41:25 +02:00
MartinBraquet
a01283a446 Fix links 2025-09-16 18:18:57 +02:00
MartinBraquet
fefa261e7d Restrict matched users to last 24h 2025-09-16 18:09:48 +02:00
MartinBraquet
0447b22dd2 Restrict internal/send-search-notifications with API key 2025-09-16 17:54:13 +02:00
MartinBraquet
cf125c1b48 Fix packages 2025-09-16 16:48:52 +02:00
MartinBraquet
81a9d8257c Trigger filter change only on slide commit to avoid API overload 2025-09-16 16:15:07 +02:00
MartinBraquet
ee3f471300 Skip 2025-09-16 16:14:39 +02:00
MartinBraquet
5c2e5f626d Upgrade subject 2025-09-16 16:14:34 +02:00
MartinBraquet
a0f4b62361 Allow for empty orderBy 2025-09-16 16:14:26 +02:00
MartinBraquet
786166b448 Fix connection clause 2025-09-16 16:14:16 +02:00
MartinBraquet
66e198b4ef Ignore 2025-09-16 15:13:40 +02:00
MartinBraquet
4919240242 Update readme info 2025-09-16 15:13:36 +02:00
MartinBraquet
d7e6a41e3f Clean installs 2025-09-16 14:31:39 +02:00
MartinBraquet
202ef737dd Add react email as dev 2025-09-16 14:08:36 +02:00
MartinBraquet
04993224dc Nice emails 2025-09-16 14:03:08 +02:00
MartinBraquet
bebe7c28f8 Fix 2025-09-16 12:48:44 +02:00
MartinBraquet
639991dde4 Add bookmarked search emails and factor out utils from web to common 2025-09-16 12:36:18 +02:00
MartinBraquet
31404cb89a Add allowSyntheticDefaultImports 2025-09-16 12:35:19 +02:00
MartinBraquet
f6205ca1dd Add source.sh 2025-09-16 12:35:12 +02:00
MartinBraquet
6e86fc0593 Add build_api.sh 2025-09-16 12:35:01 +02:00
MartinBraquet
f39a9845a3 Fix tsconfig include jsonapi 2025-09-15 21:08:27 +02:00
MartinBraquet
ba17582945 Factor out loadProfiles 2025-09-15 21:08:14 +02:00
MartinBraquet
02a1cbd467 Rename getLovers and add base send-search-notifications.ts 2025-09-15 18:48:34 +02:00
MartinBraquet
2cd102ef0b Add API docs 2025-09-15 18:45:03 +02:00
MartinBraquet
240361b55b Access API at / install of /v0/ 2025-09-15 18:07:25 +02:00
MartinBraquet
9beabc93cd Clean 2025-09-15 16:13:11 +02:00
MartinBraquet
8f4c6b911a Meke age optional in API requests 2025-09-15 16:04:30 +02:00
MartinBraquet
083ef3010d Add location column 2025-09-15 14:24:47 +02:00
MartinBraquet
e6c2253219 Fix location pretty print 2025-09-14 22:45:04 +02:00
MartinBraquet
d802eb3f28 Fix 2025-09-14 22:36:16 +02:00
MartinBraquet
a342d5d5ad Fix hidden close button 2025-09-14 22:35:00 +02:00
MartinBraquet
99adb77fcb Pretty print bookmarked searches 2025-09-14 22:18:21 +02:00
MartinBraquet
2ea4eae9d6 Add wantsKidsNames 2025-09-14 22:16:45 +02:00
MartinBraquet
9b079b2c3a Add hasKidsNames 2025-09-14 22:16:30 +02:00
MartinBraquet
8648e8569e Add list style 2025-09-14 22:16:03 +02:00
MartinBraquet
1be0ab8bcb Fix 2025-09-14 16:37:57 +02:00
MartinBraquet
718f76c1f2 Hide likes 2025-09-14 16:37:52 +02:00
MartinBraquet
155d1f4c06 Add bookmarked filters for notifications 2025-09-13 23:21:45 +02:00
MartinBraquet
cb79e27d5a Fix hover button gray 2025-09-13 23:19:51 +02:00
MartinBraquet
26991f8dd8 Remove blue message 2025-09-13 23:19:29 +02:00
MartinBraquet
2375330d76 Fix modal size on mobile 2025-09-13 23:18:56 +02:00
MartinBraquet
94e9b6d99b Fix UI 2025-09-13 23:18:33 +02:00
MartinBraquet
b516d24101 Fix font 2025-09-13 23:18:11 +02:00
MartinBraquet
1b131d9371 Fix typo 2025-09-13 23:18:01 +02:00
MartinBraquet
3f45ef192d Update db schema 2025-09-13 23:17:46 +02:00
MartinBraquet
c6684af521 Improve colors 2025-09-13 23:17:23 +02:00
MartinBraquet
52f12b81ff Move script 2025-09-13 23:17:10 +02:00
MartinBraquet
6630f787bf Debug log 2025-09-13 23:16:58 +02:00
MartinBraquet
2d7b2da3e2 Improve wording 2025-09-13 23:16:53 +02:00
MartinBraquet
d3b008fcd9 Add bookmarked_searches.sql 2025-09-13 18:12:20 +02:00
MartinBraquet
8a62fd0e6a Add migration 2025-09-13 18:11:57 +02:00
MartinBraquet
b044860f05 Add last_modification_time 2025-09-13 18:11:46 +02:00
MartinBraquet
1c5786dfb6 Add web readme 2025-09-13 16:55:09 +02:00
MartinBraquet
6bc9e3d695 Downgrade next 2025-09-13 16:44:14 +02:00
MartinBraquet
b74fe59f12 Upgrade next 2025-09-13 16:19:23 +02:00
MartinBraquet
6b57aa7f14 Add info 2025-09-13 16:12:05 +02:00
MartinBraquet
227125b35c Update packages 2025-09-13 16:11:59 +02:00
MartinBraquet
c4012d8dfc Fix 2025-09-13 15:38:08 +02:00
MartinBraquet
cf3fa9ffbc Fix 2025-09-13 15:36:04 +02:00
MartinBraquet
40640d029a Update packages 2025-09-13 15:33:23 +02:00
MartinBraquet
01eb7038dc Add /support page 2025-09-13 15:12:45 +02:00
MartinBraquet
58115bfd11 Dynamic filename finding 2025-09-13 15:12:32 +02:00
MartinBraquet
f1ea5031fb Remove supabase token 2025-09-13 14:54:10 +02:00
MartinBraquet
26d15a9fb3 Remove autogenerated line 2025-09-13 13:14:56 +02:00
MartinBraquet
54ba8e6047 Upgrade next 2025-09-13 12:39:24 +02:00
MartinBraquet
eca063ab75 Clean 2025-09-13 12:14:17 +02:00
MartinBraquet
8892f4144e Remove logs 2025-09-13 12:14:09 +02:00
MartinBraquet
d2c25f9d6c Remove firebase warning 2025-09-13 12:03:08 +02:00
MartinBraquet
b57457dc2f Add yarn clean-install 2025-09-13 11:57:41 +02:00
MartinBraquet
2861b0cfa2 Update caniuse 2025-09-13 11:47:58 +02:00
MartinBraquet
0c45dbb884 Fix warning font 2025-09-13 11:46:58 +02:00
MartinBraquet
a9f9261fb7 Update description 2025-09-12 21:42:18 +02:00
MartinBraquet
7e5f54a4f1 Add anim in search bar 2025-09-12 21:40:02 +02:00
MartinBraquet
1228e8759c Add multiple keywords per search 2025-09-12 21:39:44 +02:00
MartinBraquet
1daf771218 Clean file architecture 2025-09-12 20:47:37 +02:00
MartinBraquet
880cb08c3d Update FAQ 2025-09-12 20:17:24 +02:00
MartinBraquet
e2cbae3089 Add supabase dev key 2025-09-12 20:06:20 +02:00
MartinBraquet
42dcc3318c Comment 2025-09-12 18:33:46 +02:00
MartinBraquet
b32a85ae7e Fix cookie 2025-09-12 18:27:39 +02:00
MartinBraquet
af85edddca Ignore pics 2025-09-12 18:18:55 +02:00
MartinBraquet
eccd88e3c2 Debug cookie 2025-09-12 18:14:34 +02:00
MartinBraquet
e0e11629a1 Set up PostHog 2025-09-12 18:03:21 +02:00
MartinBraquet
968095c183 Fix LOCAL_DEV import in shared 2025-09-12 17:40:54 +02:00
MartinBraquet
d32b5115c5 Clean images and SEO 2025-09-12 17:21:03 +02:00
MartinBraquet
d3001ec887 Fix 2025-09-12 17:00:53 +02:00
MartinBraquet
fef6a52008 Fix og url for local 2025-09-12 17:00:49 +02:00
MartinBraquet
048e6affbc Update dev info 2025-09-12 16:42:53 +02:00
MartinBraquet
c653d49691 Update doc 2025-09-12 15:56:42 +02:00
MartinBraquet
6f5c9bd054 Update docs 2025-09-12 15:55:45 +02:00
MartinBraquet
9e5576244d Fix readme 2025-09-12 15:36:23 +02:00
MartinBraquet
ef91317232 Add local dev info 2025-09-12 15:29:17 +02:00
MartinBraquet
10c44d050f Fix 2025-09-12 15:03:44 +02:00
MartinBraquet
1845ea7170 Fix 2025-09-12 15:03:12 +02:00
MartinBraquet
d453294622 Fix 2025-09-12 15:02:47 +02:00
MartinBraquet
d11f9e4971 Fix font 2025-09-12 15:01:57 +02:00
MartinBraquet
08272dd04e Fix typo 2025-09-12 15:01:50 +02:00
MartinBraquet
42441b9b42 Add info 2025-09-12 14:54:06 +02:00
MartinBraquet
e4a293c046 Add todos 2025-09-12 14:50:47 +02:00
MartinBraquet
0cc5a39d63 Add todos 2025-09-12 14:48:32 +02:00
MartinBraquet
942ea3f125 Update readme todo 2025-09-12 14:36:03 +02:00
MartinBraquet
a8a70bb71c Make avatar pic if no pictures 2025-09-12 12:01:04 +02:00
MartinBraquet
0d7c3fb4b2 Allow everyone to message everyone for now 2025-09-12 12:00:41 +02:00
MartinBraquet
77c682454e Reduce like button size 2025-09-12 12:00:15 +02:00
MartinBraquet
dd3473f5d8 Improve prompts link 2025-09-12 03:39:50 +02:00
MartinBraquet
cceadc5e04 Center the icons, even in gmail 2025-09-12 03:15:17 +02:00
MartinBraquet
e48c3a3f9c Add links to emails 2025-09-12 02:42:00 +02:00
MartinBraquet
14981ef077 Fix 2025-09-12 02:15:06 +02:00
MartinBraquet
a7858d44bd Fix empty content 2025-09-12 02:10:19 +02:00
MartinBraquet
9ae5f27c04 Remove free responses for now as not implemented in the db 2025-09-12 02:03:30 +02:00
MartinBraquet
d691129842 Hide location if null 2025-09-12 01:57:22 +02:00
MartinBraquet
e26d551263 Fix overrides 2025-09-12 01:47:11 +02:00
MartinBraquet
277c6a444f Release 2025-09-12 01:40:58 +02:00
MartinBraquet
f344800fd6 Fix yarn install warnings 2025-09-12 01:38:48 +02:00
MartinBraquet
39a6fba33f Remove unused and confusing sub lock files 2025-09-12 01:30:50 +02:00
MartinBraquet
8e11657bd2 Update install 2025-09-12 01:26:49 +02:00
MartinBraquet
dfbeaa4edf Clean lock 2025-09-12 01:23:16 +02:00
MartinBraquet
e90dc3b7f4 Remove log 2025-09-12 01:23:06 +02:00
MartinBraquet
dba89e611a Fix package backend 2025-09-12 01:17:39 +02:00
MartinBraquet
1a3fecc89e Fix 2025-09-12 00:46:38 +02:00
MartinBraquet
407e6a3d06 Fix yarn.lock 2025-09-12 00:40:46 +02:00
MartinBraquet
6ee19d5359 Back to working on vercel 2025-09-12 00:31:11 +02:00
MartinBraquet
2df424dbac Remove log 2025-09-12 00:14:23 +02:00
MartinBraquet
9874be6bf1 Fix packages 2025-09-12 00:10:50 +02:00
MartinBraquet
a3d4199d1d Fix vercel build 2025-09-11 23:56:41 +02:00
MartinBraquet
247fa146a9 Fix 2025-09-11 23:26:38 +02:00
MartinBraquet
f2b2c02cd6 Add install.sh 2025-09-11 22:56:29 +02:00
MartinBraquet
a915f27f00 Roolback 2025-09-11 22:56:21 +02:00
MartinBraquet
e14a488934 Revert "Failed attempt to use react icons in emails"
This reverts commit e82a8d9bc3.
2025-09-11 22:35:54 +02:00
MartinBraquet
e82a8d9bc3 Failed attempt to use react icons in emails 2025-09-11 22:35:48 +02:00
MartinBraquet
4527a0d12b Rm add tsconfig 2025-09-11 22:34:00 +02:00
MartinBraquet
01be202484 Cd cur dir 2025-09-11 19:20:40 +02:00
MartinBraquet
d1fe99edc3 Remove log 2025-09-11 19:20:32 +02:00
MartinBraquet
fa629591e9 Fix unsubscribe URL 2025-09-11 18:45:42 +02:00
MartinBraquet
4ab3edc97b Add email footer 2025-09-11 18:37:31 +02:00
MartinBraquet
f1bfc6bf55 Fix connection type (2) 2025-09-11 16:51:13 +02:00
MartinBraquet
3283843ef3 Fix connection type 2025-09-11 16:21:01 +02:00
MartinBraquet
4cb14ec8cc Fix gender 2025-09-11 16:20:11 +02:00
MartinBraquet
41535a68be Improve email UI 2025-09-11 16:00:10 +02:00
MartinBraquet
d62447a12a Unstick like 2025-09-11 14:48:36 +02:00
MartinBraquet
802367c914 Search users 2025-09-11 14:48:24 +02:00
MartinBraquet
ff9b2c6ee8 Add compatibility score FAQ 2025-09-11 14:12:44 +02:00
MartinBraquet
a0e25c941a Smoot login 2025-09-11 13:59:23 +02:00
MartinBraquet
091c99e784 Reduce sidebar width 2025-09-11 13:27:44 +02:00
MartinBraquet
e264bb407b Improve sign in / up UI 2025-09-11 13:14:20 +02:00
MartinBraquet
16625210fc Fix 2025-09-11 12:51:37 +02:00
MartinBraquet
2550453ee4 Add keyword search 2025-09-10 22:50:20 +02:00
MartinBraquet
d1c480f23f Change genders 2025-09-10 21:54:03 +02:00
MartinBraquet
b4b0397589 Hide gender they are interested in 2025-09-10 21:53:55 +02:00
MartinBraquet
ab6b34e84c Hide supabase annon key (env) 2025-09-10 21:41:08 +02:00
MartinBraquet
87af9d5078 Hide supabase annon key 2025-09-10 21:40:31 +02:00
MartinBraquet
95fab7c395 Add reports table 2025-09-10 21:40:23 +02:00
MartinBraquet
90825925ff Improve share button layout 2025-09-10 21:39:57 +02:00
MartinBraquet
7036cf9e49 Profile view when signed up: no pic, see first lines of bio 2025-09-10 20:42:09 +02:00
MartinBraquet
53123eb0ee Improve UI 2025-09-10 18:24:44 +02:00
MartinBraquet
3c5407dd51 Fix colors 2025-09-10 17:09:35 +02:00
MartinBraquet
1ffe81f740 Fix link 2025-09-10 16:20:50 +02:00
MartinBraquet
6bb35d61e1 Clean 2025-09-10 16:16:11 +02:00
MartinBraquet
f36ccf7bdc Fix colors 2025-09-10 16:16:08 +02:00
MartinBraquet
4632e68a00 Add bookish font 2025-09-10 16:14:56 +02:00
MartinBraquet
09858d0783 Change md path 2025-09-10 16:14:33 +02:00
MartinBraquet
9d1423c41b Add todo 2025-09-10 14:29:11 +02:00
MartinBraquet
1a4b7786dd Fix blue links 2025-09-10 12:53:34 +02:00
MartinBraquet
77c0a21ad0 Add info about Martin 2025-09-10 12:23:33 +02:00
MartinBraquet
7cedf14121 Add members page 2025-09-10 12:23:25 +02:00
MartinBraquet
235346f3dd Add financials link 2025-09-10 11:26:14 +02:00
MartinBraquet
34c36b7c3a Fix 2025-09-09 19:24:08 +02:00
MartinBraquet
3e0f788ec3 Add FAQ and financials 2025-09-09 19:18:44 +02:00
MartinBraquet
867bb8a072 Fix 2025-09-09 18:55:47 +02:00
MartinBraquet
31a400158a Do not render your own profile in Profiles 2025-09-09 16:58:07 +02:00
MartinBraquet
8106ff6489 Show compatibility score of no preferred gender 2025-09-09 16:57:02 +02:00
MartinBraquet
de3508993c Factor out geodbFetch 2025-09-09 16:07:36 +02:00
MartinBraquet
fd3e7a6f8a Fix location filtering 2025-09-09 15:55:33 +02:00
MartinBraquet
4cf97a6054 Fix 2025-09-09 15:25:21 +02:00
MartinBraquet
75036e3ec7 Fix 2025-09-09 14:57:08 +02:00
MartinBraquet
a157f6ce27 Release 2025-09-09 14:50:36 +02:00
MartinBraquet
d27cc94dd0 Update readme 2025-09-09 14:48:47 +02:00
MartinBraquet
05b1416d39 Fix link 2025-09-09 13:58:41 +02:00
MartinBraquet
001d6ed968 Add constitution 2025-09-09 13:53:24 +02:00
MartinBraquet
3ed3eecb00 Ignore test 2025-09-09 13:30:24 +02:00
MartinBraquet
8638e6cdeb Clean about blocks 2025-09-09 13:28:26 +02:00
MartinBraquet
15c7a9c22e Add react-md 2025-09-09 03:31:58 +02:00
MartinBraquet
20566e42ec Fix sign in loading page 2025-09-09 03:25:14 +02:00
MartinBraquet
92d1cac254 Fix sign up loading page 2025-09-09 03:03:02 +02:00
MartinBraquet
72e879c424 Add terms and privacy notices 2025-09-09 02:08:12 +02:00
Martin Braquet
3b2516fea2 Create NOTICE 2025-09-09 01:44:06 +02:00
Martin Braquet
1151f98fd3 Update LICENSE-MIT 2025-09-09 01:42:54 +02:00
Martin Braquet
d608fdb6f0 Update LICENSE-MIT 2025-09-09 01:42:33 +02:00
Martin Braquet
5a2c172f96 Update LICENSE-MIT 2025-09-09 01:38:40 +02:00
Martin Braquet
6e1f7bdd7b Update LICENSE-MIT 2025-09-09 01:38:21 +02:00
Martin Braquet
bf96227b8b Update LICENSE-MIT 2025-09-09 01:35:38 +02:00
MartinBraquet
e7195dd68d Log 2025-09-09 01:16:14 +02:00
MartinBraquet
6a374b0c5a Fix compatibility questions 2025-09-08 23:46:50 +02:00
MartinBraquet
a0a876a282 Fix link going over max width 2025-09-08 23:12:41 +02:00
MartinBraquet
0a538375b2 Fix hanging button 2025-09-08 23:03:57 +02:00
MartinBraquet
ac5d60cd58 Add home line 2025-09-08 22:54:01 +02:00
MartinBraquet
8498480b9c Clean forms and prevent from changing username after creation 2025-09-08 22:35:57 +02:00
MartinBraquet
4c1c7fc514 Remove onlyfans (2)... 2025-09-08 22:35:33 +02:00
MartinBraquet
1c3a0f9c71 Remove onlyfans... 2025-09-08 22:35:28 +02:00
MartinBraquet
39b5068370 Fix tracker 2025-09-08 22:34:50 +02:00
MartinBraquet
4815cc7682 Make all complete-profile fields optional and Pre-Save lover 2025-09-07 23:54:37 +02:00
MartinBraquet
7886d32933 Refactor 2025-09-07 21:40:56 +02:00
MartinBraquet
2dab88c7a9 Rollback packages 2025-09-07 21:33:44 +02:00
MartinBraquet
35b83dcb9a Fix 2025-09-07 21:28:37 +02:00
MartinBraquet
5194b5f6bf Clean home and fix profiles not loading 2025-09-07 21:12:06 +02:00
MartinBraquet
d7c49fe19f Refactor into AboutBox 2025-09-07 20:39:09 +02:00
MartinBraquet
e3dadd2ce8 Fix dynamic "search" 2025-09-07 20:28:50 +02:00
MartinBraquet
d2d08bc77c Update packages 2025-09-07 20:04:56 +02:00
MartinBraquet
41da374d93 Fix github CI 2025-09-07 16:23:24 +02:00
MartinBraquet
c2f48fc90c Fix scaling 2025-09-07 16:16:57 +02:00
MartinBraquet
18f2b61545 Render bio and name only in profiles grid 2025-09-07 16:15:01 +02:00
MartinBraquet
a71c7adaf7 Move mutual likes down 2025-09-06 20:33:42 +02:00
MartinBraquet
3c050bee3b Show previous profile while loading another one 2025-09-06 20:30:46 +02:00
MartinBraquet
50be1ba510 Add logs 2025-09-06 20:13:44 +02:00
MartinBraquet
445f62ca53 Put profile pic down 2025-09-06 20:13:17 +02:00
MartinBraquet
4810904aa8 Add logs 2025-09-06 20:13:05 +02:00
MartinBraquet
3a6d459ebd Skip jwt supabase update 2025-09-06 20:12:37 +02:00
MartinBraquet
5c23380de9 Clean 2025-09-06 19:35:13 +02:00
MartinBraquet
a0285d970f Clean 2025-09-06 19:34:52 +02:00
MartinBraquet
0e7e0f52f1 Fix 2025-09-06 15:56:54 +02:00
MartinBraquet
5054a9552b Update env.example 2025-09-06 15:56:47 +02:00
MartinBraquet
a7c55530a4 Hide and regenerate Firebase API key 2025-09-06 14:58:30 +02:00
MartinBraquet
0f03746c6a Fix register redirect and add error message for email already in use 2025-09-06 12:02:06 +02:00
MartinBraquet
4a891d9c9a Fix 2025-09-06 11:31:27 +02:00
MartinBraquet
209e233ee2 Add error message 2025-09-06 11:29:37 +02:00
MartinBraquet
519ec081b5 Add 404 info 2025-09-06 11:07:49 +02:00
MartinBraquet
a73c7ff8b6 Fix 2025-09-06 10:53:42 +02:00
MartinBraquet
ea74e0514e Make sign in and sign up pages and allow for email/pwd registration 2025-09-06 10:50:47 +02:00
MartinBraquet
523bbd11cc Update readme 2025-09-05 20:29:01 +02:00
MartinBraquet
becf6ad7a4 Fix wrong firebase bucket name 2025-09-03 12:00:15 +02:00
MartinBraquet
364f58b186 Fix 2025-09-01 22:54:52 +02:00
MartinBraquet
ef36b78399 Comment 2025-09-01 22:51:34 +02:00
MartinBraquet
1c77e6dc2c Hide profiles if not logged in and put nice home page 2025-09-01 22:46:02 +02:00
MartinBraquet
d581ce054c Clean more files 2025-09-01 22:19:23 +02:00
MartinBraquet
2956ec073f Add about in mobile 2025-09-01 22:19:14 +02:00
MartinBraquet
3ace876b66 Clean text 2025-09-01 22:19:01 +02:00
MartinBraquet
d8722a8274 Clean NEXT_PUBLIC_FIREBASE_ENV 2025-09-01 22:18:48 +02:00
MartinBraquet
d5216a8e8c Remove unused component 2025-09-01 18:29:29 +02:00
MartinBraquet
152521e9e5 Fix color style 2025-09-01 18:27:58 +02:00
MartinBraquet
a2d8518f14 Even nicer /about 2025-09-01 18:21:16 +02:00
MartinBraquet
e88a384c5e Get nice about page 2025-09-01 18:12:41 +02:00
MartinBraquet
063663d5b0 Fix discord link 2025-09-01 17:54:15 +02:00
MartinBraquet
0a48263013 Remove supabase log 2025-09-01 17:54:05 +02:00
MartinBraquet
2877bc8239 Fix old about 2025-09-01 17:51:14 +02:00
MartinBraquet
d5886fe3f2 Remove pink border 2025-09-01 17:50:58 +02:00
MartinBraquet
c0bacb104f Clean 2025-09-01 17:45:53 +02:00
MartinBraquet
27b851dca1 Clean 2025-09-01 17:40:38 +02:00
MartinBraquet
bd358b38f7 Fix 2025-09-01 17:33:53 +02:00
MartinBraquet
f6c895fe78 Clean manifold names and update discord invite and improve email notifs 2025-09-01 17:30:45 +02:00
MartinBraquet
23c8f175bb Fix supabase IPV6 only host 2025-09-01 16:49:53 +02:00
MartinBraquet
2949871ba1 Fixes 2025-09-01 15:01:23 +02:00
MartinBraquet
34298fcfa1 Clean 2025-09-01 14:58:29 +02:00
MartinBraquet
83c6973d8e Add setup script 2025-09-01 14:56:23 +02:00
MartinBraquet
3b932fd52d Remove build 2025-09-01 14:39:49 +02:00
MartinBraquet
e0291b8e5a Fix tests 2025-09-01 14:36:47 +02:00
MartinBraquet
8fe3736411 Add base tests 2025-09-01 14:33:04 +02:00
MartinBraquet
70e46c2b69 Upgrade API deploy script 2025-09-01 14:17:21 +02:00
MartinBraquet
c5a7d823c8 Upgrade API readme 2025-09-01 14:17:09 +02:00
MartinBraquet
56115d34a4 Fix SSL 2025-09-01 14:16:57 +02:00
MartinBraquet
af3b91037e Copy tsconfig to docker 2025-09-01 14:16:30 +02:00
MartinBraquet
549161586e Fix tsc-alias 2025-09-01 14:15:31 +02:00
MartinBraquet
20ac60219f Add version 2025-09-01 14:15:14 +02:00
MartinBraquet
ef8b17f5c1 Source from .env 2025-09-01 14:15:06 +02:00
MartinBraquet
2c5339aa92 Use supabase ipv4 2025-09-01 14:14:41 +02:00
MartinBraquet
720fa70d60 Use prod by default 2025-09-01 14:14:17 +02:00
MartinBraquet
b2eef1279f use LOCAL_DEV 2025-09-01 14:14:07 +02:00
MartinBraquet
da5e5aedb2 rename secret 2025-09-01 14:13:35 +02:00
MartinBraquet
51c505f9a6 Update lock 2025-09-01 14:13:18 +02:00
MartinBraquet
4f97a61e94 Remove package lock as using yarn lock 2025-09-01 14:13:09 +02:00
MartinBraquet
7af0f28bd7 Acknowledge manifold.love 2025-08-31 12:17:46 +02:00
MartinBraquet
a9f4e95b77 Set up google cloud server 2025-08-28 22:16:42 +02:00
MartinBraquet
3d3420b1aa Add lock file 2025-08-28 17:57:03 +02:00
MartinBraquet
26915ea94f Fix vercel env key 2025-08-28 17:29:03 +02:00
MartinBraquet
dfa1d1c76e Remove pwd 2025-08-28 16:56:46 +02:00
MartinBraquet
87c1870770 Make PROD work 2025-08-28 16:42:51 +02:00
MartinBraquet
626c27b635 Fix color 2025-08-28 12:49:09 +02:00
MartinBraquet
7572d1a6ff Add email sending 2025-08-28 12:49:05 +02:00
MartinBraquet
b707439de7 Rename to Compass (2) 2025-08-27 22:17:07 +02:00
MartinBraquet
2e332707ff Rename to Compass 2025-08-27 22:12:04 +02:00
MartinBraquet
2f60efe273 Set grey color scales 2025-08-27 22:07:43 +02:00
MartinBraquet
632f8477fd Fix favicon 2025-08-27 22:07:35 +02:00
MartinBraquet
53432520cd Pull up features from manifold.love 2025-08-27 21:30:05 +02:00
MartinBraquet
078893f7d1 Add books feature 2025-08-11 17:38:26 +02:00
MartinBraquet
131cb0ff79 Disable interest cache 2025-08-11 17:34:00 +02:00
MartinBraquet
c49117ffd5 Rename about 2025-08-11 15:40:08 +02:00
MartinBraquet
cb52aa264d Update desc 2025-08-11 15:09:39 +02:00
MartinBraquet
1c1a02757f Add loading... info 2025-08-10 13:04:29 +02:00
MartinBraquet
4558c4a29c Make social style slider in /complete-profile 2025-08-10 12:59:09 +02:00
MartinBraquet
21832a1885 Update introversion bar 2025-08-09 22:19:05 +02:00
MartinBraquet
9b45cc087f Fix 2025-08-09 21:56:52 +02:00
MartinBraquet
b131e6ee8c Hide minor onboarding questions 2025-08-09 21:51:01 +02:00
MartinBraquet
306a297837 Change connections options 2025-08-09 21:50:38 +02:00
MartinBraquet
380eda64a0 Update schema.prisma 2025-08-08 01:51:29 +02:00
MartinBraquet
2dd8e3016f Undo schema 2025-08-08 01:43:55 +02:00
MartinBraquet
fd9c61a1c7 Redirect new users to onboarding page 2025-08-08 01:19:36 +02:00
MartinBraquet
4a2dba6e2e Hide register buttons if logged in 2025-08-07 23:28:10 +02:00
MartinBraquet
034d94ee22 Move sign in button from header to /profile 2025-08-07 23:16:07 +02:00
MartinBraquet
65139094cb Edit readme 2025-08-07 23:03:02 +02:00
emilyokeefe
b607cf22a7 More concise wording on the about page (#5)
* Changed wording

* remove share button because it doesn't function yet

* more concise wording
2025-08-07 19:16:10 +02:00
emilyokeefe
b79f8d05be Added onboarding questions and worked on style (#3) 2025-08-07 13:52:19 +02:00
Martin Braquet
520d157a0f Move LICENSE from MIT to AGPL 2025-08-07 13:39:33 +02:00
MartinBraquet
f73be7e38f Delete license 2025-08-07 13:32:22 +02:00
MartinBraquet
44834d1d27 Hide registration options until 3 chars typed 2025-08-07 01:00:25 +02:00
MartinBraquet
d07422060a Add local dev prisma db 2025-08-07 00:35:22 +02:00
emilyokeefe
0ee9437cf8 Changed wording (#4)
* Changed wording

* remove share button because it doesn't function yet

---------

Co-authored-by: Martin Braquet <martin.braquet@gmail.com>
2025-08-06 23:05:10 +02:00
MartinBraquet
6fe646a32b Hide stats for now 2025-08-06 14:16:21 +02:00
MartinBraquet
8e1f643612 Add basic multi-step form 2025-08-05 20:19:11 +02:00
MartinBraquet
7f2ba4d727 Fix test 2025-08-05 17:47:40 +02:00
MartinBraquet
b8706eae10 Clean 2025-08-05 17:42:21 +02:00
MartinBraquet
34600ab0cf Fix favicon 2025-08-05 17:33:35 +02:00
MartinBraquet
b3031b79d1 Restrict userbase access to logged-in users 2025-08-05 17:12:53 +02:00
MartinBraquet
cc679ddcfa Fix image 2025-08-05 17:11:55 +02:00
MartinBraquet
3633d469c1 Switch from home to logo on narrow screens 2025-08-05 16:46:49 +02:00
MartinBraquet
46fa721a6c Add prisma config 2025-08-05 16:26:57 +02:00
emilyokeefe
7e4116b1a2 UI redesign: Add compass branding, improve navigation, & simplify content (#1)
* UI redesign: Add compass branding, improve navigation, and simplify content

* Add homepage improvements
- Created "Why Compass" section with three value props
- Worked on spacing
- Made word "Search" styled with typing effect

* Delete pic

* Fix lint

* Compress favicon

---------

Co-authored-by: MartinBraquet <martin.braquet@gmail.com>
2025-08-05 16:14:07 +02:00
MartinBraquet
2a043dbd53 Add AND multi keyword search 2025-08-05 02:18:37 +02:00
MartinBraquet
4aa23cf755 Fix 2025-08-05 01:59:24 +02:00
MartinBraquet
4f261116de Add filter params to url query params 2025-08-05 01:55:31 +02:00
MartinBraquet
7ecd5481a0 Move button 2025-08-04 23:21:03 +02:00
MartinBraquet
1cea1b7bcf Hide cause areas 2025-08-04 23:09:26 +02:00
MartinBraquet
ccba688ec4 Rename 2025-08-04 22:55:29 +02:00
MartinBraquet
4eabe72078 Remove conflict style 2025-08-04 22:46:28 +02:00
MartinBraquet
fbbe97d297 Add sign in 2025-08-04 22:42:15 +02:00
MartinBraquet
26a80d1313 Change description 2025-08-04 22:19:55 +02:00
MartinBraquet
24b5edba9f Remove age slider 2025-08-04 22:14:16 +02:00
MartinBraquet
9113e0e372 Update name 2025-08-04 22:09:14 +02:00
MartinBraquet
9f944dd171 Clean texts 2025-08-04 19:34:50 +02:00
MartinBraquet
b91f76914c Rename manifesto 2025-08-04 19:31:20 +02:00
MartinBraquet
8593d90209 Remove core 2025-08-04 19:30:41 +02:00
MartinBraquet
df441b7236 Remove age slider 2025-08-04 19:29:04 +02:00
MartinBraquet
43e2798a9b Rename desired connection 2025-08-04 19:27:38 +02:00
MartinBraquet
7ee14a48a7 Change search bar placeholder 2025-08-04 19:24:42 +02:00
MartinBraquet
1e5cf0ca5b Move user count 2025-08-04 19:24:30 +02:00
MartinBraquet
12bac6d305 Fix spinner 2025-08-04 19:18:35 +02:00
MartinBraquet
1518cd50ec Clean 2025-08-04 18:46:33 +02:00
MartinBraquet
9dfc82c106 Clean 2025-08-04 14:43:23 +02:00
MartinBraquet
087f10f7bb Rename 2025-08-04 14:39:43 +02:00
MartinBraquet
6971eac21f Add badges 2025-08-04 14:38:19 +02:00
MartinBraquet
da7cde91b3 Fix 2025-08-04 14:28:24 +02:00
MartinBraquet
814b4fe0ae Add tests 2025-08-04 14:25:44 +02:00
MartinBraquet
a2abc4fda9 Fix bad useState practices 2025-08-04 14:25:33 +02:00
MartinBraquet
9bcba9895e Clean bar info 2025-08-04 14:24:54 +02:00
MartinBraquet
ff13ea71a6 Fix spinner 2025-08-04 14:24:03 +02:00
MartinBraquet
7993dcb0b1 Update gitig 2025-08-04 14:23:42 +02:00
MartinBraquet
9a527fef20 Update lint 2025-08-04 14:23:30 +02:00
MartinBraquet
e6de25c0a4 Fix some lints 2025-08-04 11:44:24 +02:00
MartinBraquet
c45adc1a8a Remove slack 2025-08-04 11:34:17 +02:00
MartinBraquet
c06f86edbb Fix loading spinner 2025-08-04 11:34:01 +02:00
MartinBraquet
ed515fa3fc Add base tests 2025-08-04 10:26:19 +02:00
MartinBraquet
c284983b4b Update dev docs 2025-08-04 10:16:24 +02:00
MartinBraquet
5fed099034 Update README.md 2025-08-04 10:16:19 +02:00
MartinBraquet
c8dd335d65 Update README.md 2025-08-04 09:58:51 +02:00
MartinBraquet
17d2c6aa57 Clean 2025-08-04 09:56:30 +02:00
764 changed files with 66880 additions and 13554 deletions

View File

@@ -1,20 +1,15 @@
# Use the Prisma Postgres integration from Vercel Marketeplace to automatically connect a Prisma Postgres instance # You already have access to basic local functionality (UI, authentication, database read access).
# Or manually run `npx prisma init --db` to create a Prisma Postgres and manually set the `DATABASE_URL` below
# Create a random 32-character string or run `npx auth secret` to obtain one and set it as the `AUTH_SECRET` below # openssl enc -aes-256-cbc -salt -pbkdf2 -iter 100000 -in backend/shared/src/googleApplicationCredentials-dev.json -out secrets/googleApplicationCredentials-dev.json.enc
GOOGLE_CREDENTIALS_ENC_PWD=nP7s3274uzOG4c2t
GOOGLE_CLIENT_ID=
GOOGLE_CLIENT_SECRET=
NEXTAUTH_SECRET=
NEXTAUTH_URL=http://localhost:3000
# Email configuration # Optional variables for full local functionality
EMAIL_SERVER_HOST=smtp.resend.dev
EMAIL_SERVER_PORT=587
EMAIL_SERVER_USER=BayesBond
EMAIL_SERVER_PASSWORD=
RESEND_API_KEY=
EMAIL_FROM=
# Development (SQLite) # For the location / distance filtering features.
DATABASE_URL=file:./dev.db # Create a free account at https://rapidapi.com/wirefreethought/api/geodb-cities and get an API key.
GEODB_API_KEY=
# For sending emails (e.g. for user sign up, password reset, notifications, etc.).
# Create a free account at https://resend.com and get an API key. Should start with "re_".
RESEND_KEY=

15
.github/FUNDING.yml vendored Normal file
View File

@@ -0,0 +1,15 @@
# These are supported funding model platforms
github: [CompassConnections] # Replace with up to 4 GitHub Sponsors-enabled usernames e.g., [user1, user2]
patreon: CompassMeet # Replace with a single Patreon username
open_collective: compass-connection # Replace with a single Open Collective username
ko_fi: compassconnections # Replace with a single Ko-fi username
tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel
community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry
liberapay: CompassConnections # Replace with a single Liberapay username
issuehunt: # Replace with a single IssueHunt username
lfx_crowdfunding: # Replace with a single LFX Crowdfunding project-name e.g., cloud-foundry
polar: # Replace with a single Polar username
buy_me_a_coffee: # Replace with a single Buy Me a Coffee username
thanks_dev: # Replace with a single thanks.dev username
custom: # Replace with up to 4 custom sponsorship URLs e.g., ['link1', 'link2']

View File

@@ -1,31 +0,0 @@
name: Check Next.js
on:
push:
branches:
- main
pull_request:
branches:
- main
jobs:
build:
name: Build
runs-on: ubuntu-latest
steps:
- name: Checkout repository
uses: actions/checkout@v4
- name: Setup Node.js
uses: actions/setup-node@v4
with:
node-version: '22'
- name: Install dependencies
run: npm ci
- name: Build the app
env:
DATABASE_URL: ${{ secrets.DATABASE_URL }}
run: npm run build

View File

@@ -9,6 +9,7 @@ on:
jobs: jobs:
release: release:
name: Release
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- name: Checkout repo - name: Checkout repo

60
.github/workflows/ci.yml vendored Normal file
View File

@@ -0,0 +1,60 @@
name: CI
on:
push:
branches:
- main
pull_request:
branches:
- main
jobs:
ci:
name: All
runs-on: ubuntu-latest
steps:
- name: Checkout repository
uses: actions/checkout@v4
- name: Setup Node.js
uses: actions/setup-node@v4
with:
node-version: '22'
- name: Install dependencies
run: yarn install
- name: Type check
run: echo skipping #npx tsc --noEmit
- name: Lint
run: npm run lint
- name: Run Jest tests
run: npm run test tests/jest
# - name: Build app
# env:
# DATABASE_URL: ${{ secrets.DATABASE_URL }}
# run: npm run build
# Optional: Playwright E2E tests
- name: Install Playwright deps
run: npx playwright install --with-deps
# npm install @playwright/test
# npx playwright install
- name: Run E2E tests
env:
NEXT_PUBLIC_API_URL: localhost:8088
NEXT_PUBLIC_FIREBASE_ENV: DEV
NEXT_PUBLIC_FIREBASE_API_KEY: ${{ secrets.NEXT_PUBLIC_FIREBASE_API_KEY }}
NEXT_PUBLIC_SUPABASE_KEY: ${{ secrets.NEXT_PUBLIC_SUPABASE_KEY }}
run: |
yarn --cwd=web serve &
npx wait-on http://localhost:3000
npx playwright test tests/e2e
SERVER_PID=$(fuser -k 3000/tcp)
echo $SERVER_PID
kill $SERVER_PID

50
.gitignore vendored
View File

@@ -13,6 +13,9 @@
# testing # testing
/coverage /coverage
# Playwright
/tests/reports/playwright-report
# next.js # next.js
/.next/ /.next/
/out/ /out/
@@ -33,6 +36,9 @@ yarn-error.log*
# env files (can opt-in for committing if needed) # env files (can opt-in for committing if needed)
.env .env
.env.local .env.local
.env.*
.envrc
supabase/*
# vercel # vercel
.vercel .vercel
@@ -41,9 +47,51 @@ yarn-error.log*
*.tsbuildinfo *.tsbuildinfo
next-env.d.ts next-env.d.ts
.idea/
node_modules
yarn-error.log
dev
firebase-debug.log
tsconfig.tsbuildinfo
*.db *.db
*prisma/migrations *prisma/migrations
martin martin
email-preview
.obsidian .obsidian
.idea .idea
*.last-run.json
*lock.hcl
/web/pages/test.tsx
*.png
*.jpg
*.jpeg
*.gif
*.svg
*.ico
*.mp4
*.mov
*.webp
*.avi
*.wmv
*.mp3
*.wav
*.flac
*.aac
*.zip
*.tar.gz
*.rar
/favicon_color.ico
/backend/shared/src/googleApplicationCredentials-dev.json
*.tfstate
*.tfstate.backup
*.terraform
/backups/firebase/auth/data/
/backups/firebase/storage/data/
android/app/release*
icons/
*.bak

24
.prettierrc Normal file
View File

@@ -0,0 +1,24 @@
{
"tabWidth": 2,
"useTabs": false,
"semi": false,
"trailingComma": "es5",
"singleQuote": true,
"plugins": ["prettier-plugin-sql"],
"overrides": [
{
"files": "*.sql",
"options": {
"language": "postgresql",
"keywordCase": "lower",
"logicalOperatorNewline": "before"
}
},
{
"files": "*.svg",
"options": {
"parser": "html"
}
}
]
}

1
.yarnrc Normal file
View File

@@ -0,0 +1 @@
save-exact true

View File

@@ -13,13 +13,13 @@ We welcome pull requests, but only if they meet the project's quality and design
1. **Fork the repository** using the GitHub UI. 1. **Fork the repository** using the GitHub UI.
2. **Clone your fork** locally: 2. **Clone your fork** locally:
```bash ```bash
git clone https://github.com/your-username/BayesBond.git git clone https://github.com/your-username/Compass.git
cd your-fork cd your-fork
3. **Add the upstream remote**: 3. **Add the upstream remote**:
```bash ```bash
git remote add upstream https://github.com/BayesBond/BayesBond.git git remote add upstream https://github.com/CompassConnections/Compass.git
``` ```
## Create a New Branch ## Create a New Branch

674
LICENSE
View File

@@ -1,21 +1,661 @@
MIT License GNU AFFERO GENERAL PUBLIC LICENSE
Version 3, 19 November 2007
Copyright (c) 2025 BayesBond Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
Everyone is permitted to copy and distribute verbatim copies
of this license document, but changing it is not allowed.
Permission is hereby granted, free of charge, to any person obtaining a copy Preamble
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all The GNU Affero General Public License is a free, copyleft license for
copies or substantial portions of the Software. software and other kinds of works, specifically designed to ensure
cooperation with the community in the case of network server software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR The licenses for most software and other practical works are designed
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, to take away your freedom to share and change the works. By contrast,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE our General Public Licenses are intended to guarantee your freedom to
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER share and change all versions of a program--to make sure it remains free
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, software for all its users.
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE. When we speak of free software, we are referring to freedom, not
price. Our General Public Licenses are designed to make sure that you
have the freedom to distribute copies of free software (and charge for
them if you wish), that you receive source code or can get it if you
want it, that you can change the software or use pieces of it in new
free programs, and that you know you can do these things.
Developers that use our General Public Licenses protect your rights
with two steps: (1) assert copyright on the software, and (2) offer
you this License which gives you legal permission to copy, distribute
and/or modify the software.
A secondary benefit of defending all users' freedom is that
improvements made in alternate versions of the program, if they
receive widespread use, become available for other developers to
incorporate. Many developers of free software are heartened and
encouraged by the resulting cooperation. However, in the case of
software used on network servers, this result may fail to come about.
The GNU General Public License permits making a modified version and
letting the public access it on a server without ever releasing its
source code to the public.
The GNU Affero General Public License is designed specifically to
ensure that, in such cases, the modified source code becomes available
to the community. It requires the operator of a network server to
provide the source code of the modified version running there to the
users of that server. Therefore, public use of a modified version, on
a publicly accessible server, gives the public access to the source
code of the modified version.
An older license, called the Affero General Public License and
published by Affero, was designed to accomplish similar goals. This is
a different license, not a version of the Affero GPL, but Affero has
released a new version of the Affero GPL which permits relicensing under
this license.
The precise terms and conditions for copying, distribution and
modification follow.
TERMS AND CONDITIONS
0. Definitions.
"This License" refers to version 3 of the GNU Affero General Public License.
"Copyright" also means copyright-like laws that apply to other kinds of
works, such as semiconductor masks.
"The Program" refers to any copyrightable work licensed under this
License. Each licensee is addressed as "you". "Licensees" and
"recipients" may be individuals or organizations.
To "modify" a work means to copy from or adapt all or part of the work
in a fashion requiring copyright permission, other than the making of an
exact copy. The resulting work is called a "modified version" of the
earlier work or a work "based on" the earlier work.
A "covered work" means either the unmodified Program or a work based
on the Program.
To "propagate" a work means to do anything with it that, without
permission, would make you directly or secondarily liable for
infringement under applicable copyright law, except executing it on a
computer or modifying a private copy. Propagation includes copying,
distribution (with or without modification), making available to the
public, and in some countries other activities as well.
To "convey" a work means any kind of propagation that enables other
parties to make or receive copies. Mere interaction with a user through
a computer network, with no transfer of a copy, is not conveying.
An interactive user interface displays "Appropriate Legal Notices"
to the extent that it includes a convenient and prominently visible
feature that (1) displays an appropriate copyright notice, and (2)
tells the user that there is no warranty for the work (except to the
extent that warranties are provided), that licensees may convey the
work under this License, and how to view a copy of this License. If
the interface presents a list of user commands or options, such as a
menu, a prominent item in the list meets this criterion.
1. Source Code.
The "source code" for a work means the preferred form of the work
for making modifications to it. "Object code" means any non-source
form of a work.
A "Standard Interface" means an interface that either is an official
standard defined by a recognized standards body, or, in the case of
interfaces specified for a particular programming language, one that
is widely used among developers working in that language.
The "System Libraries" of an executable work include anything, other
than the work as a whole, that (a) is included in the normal form of
packaging a Major Component, but which is not part of that Major
Component, and (b) serves only to enable use of the work with that
Major Component, or to implement a Standard Interface for which an
implementation is available to the public in source code form. A
"Major Component", in this context, means a major essential component
(kernel, window system, and so on) of the specific operating system
(if any) on which the executable work runs, or a compiler used to
produce the work, or an object code interpreter used to run it.
The "Corresponding Source" for a work in object code form means all
the source code needed to generate, install, and (for an executable
work) run the object code and to modify the work, including scripts to
control those activities. However, it does not include the work's
System Libraries, or general-purpose tools or generally available free
programs which are used unmodified in performing those activities but
which are not part of the work. For example, Corresponding Source
includes interface definition files associated with source files for
the work, and the source code for shared libraries and dynamically
linked subprograms that the work is specifically designed to require,
such as by intimate data communication or control flow between those
subprograms and other parts of the work.
The Corresponding Source need not include anything that users
can regenerate automatically from other parts of the Corresponding
Source.
The Corresponding Source for a work in source code form is that
same work.
2. Basic Permissions.
All rights granted under this License are granted for the term of
copyright on the Program, and are irrevocable provided the stated
conditions are met. This License explicitly affirms your unlimited
permission to run the unmodified Program. The output from running a
covered work is covered by this License only if the output, given its
content, constitutes a covered work. This License acknowledges your
rights of fair use or other equivalent, as provided by copyright law.
You may make, run and propagate covered works that you do not
convey, without conditions so long as your license otherwise remains
in force. You may convey covered works to others for the sole purpose
of having them make modifications exclusively for you, or provide you
with facilities for running those works, provided that you comply with
the terms of this License in conveying all material for which you do
not control copyright. Those thus making or running the covered works
for you must do so exclusively on your behalf, under your direction
and control, on terms that prohibit them from making any copies of
your copyrighted material outside their relationship with you.
Conveying under any other circumstances is permitted solely under
the conditions stated below. Sublicensing is not allowed; section 10
makes it unnecessary.
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
No covered work shall be deemed part of an effective technological
measure under any applicable law fulfilling obligations under article
11 of the WIPO copyright treaty adopted on 20 December 1996, or
similar laws prohibiting or restricting circumvention of such
measures.
When you convey a covered work, you waive any legal power to forbid
circumvention of technological measures to the extent such circumvention
is effected by exercising rights under this License with respect to
the covered work, and you disclaim any intention to limit operation or
modification of the work as a means of enforcing, against the work's
users, your or third parties' legal rights to forbid circumvention of
technological measures.
4. Conveying Verbatim Copies.
You may convey verbatim copies of the Program's source code as you
receive it, in any medium, provided that you conspicuously and
appropriately publish on each copy an appropriate copyright notice;
keep intact all notices stating that this License and any
non-permissive terms added in accord with section 7 apply to the code;
keep intact all notices of the absence of any warranty; and give all
recipients a copy of this License along with the Program.
You may charge any price or no price for each copy that you convey,
and you may offer support or warranty protection for a fee.
5. Conveying Modified Source Versions.
You may convey a work based on the Program, or the modifications to
produce it from the Program, in the form of source code under the
terms of section 4, provided that you also meet all of these conditions:
a) The work must carry prominent notices stating that you modified
it, and giving a relevant date.
b) The work must carry prominent notices stating that it is
released under this License and any conditions added under section
7. This requirement modifies the requirement in section 4 to
"keep intact all notices".
c) You must license the entire work, as a whole, under this
License to anyone who comes into possession of a copy. This
License will therefore apply, along with any applicable section 7
additional terms, to the whole of the work, and all its parts,
regardless of how they are packaged. This License gives no
permission to license the work in any other way, but it does not
invalidate such permission if you have separately received it.
d) If the work has interactive user interfaces, each must display
Appropriate Legal Notices; however, if the Program has interactive
interfaces that do not display Appropriate Legal Notices, your
work need not make them do so.
A compilation of a covered work with other separate and independent
works, which are not by their nature extensions of the covered work,
and which are not combined with it such as to form a larger program,
in or on a volume of a storage or distribution medium, is called an
"aggregate" if the compilation and its resulting copyright are not
used to limit the access or legal rights of the compilation's users
beyond what the individual works permit. Inclusion of a covered work
in an aggregate does not cause this License to apply to the other
parts of the aggregate.
6. Conveying Non-Source Forms.
You may convey a covered work in object code form under the terms
of sections 4 and 5, provided that you also convey the
machine-readable Corresponding Source under the terms of this License,
in one of these ways:
a) Convey the object code in, or embodied in, a physical product
(including a physical distribution medium), accompanied by the
Corresponding Source fixed on a durable physical medium
customarily used for software interchange.
b) Convey the object code in, or embodied in, a physical product
(including a physical distribution medium), accompanied by a
written offer, valid for at least three years and valid for as
long as you offer spare parts or customer support for that product
model, to give anyone who possesses the object code either (1) a
copy of the Corresponding Source for all the software in the
product that is covered by this License, on a durable physical
medium customarily used for software interchange, for a price no
more than your reasonable cost of physically performing this
conveying of source, or (2) access to copy the
Corresponding Source from a network server at no charge.
c) Convey individual copies of the object code with a copy of the
written offer to provide the Corresponding Source. This
alternative is allowed only occasionally and noncommercially, and
only if you received the object code with such an offer, in accord
with subsection 6b.
d) Convey the object code by offering access from a designated
place (gratis or for a charge), and offer equivalent access to the
Corresponding Source in the same way through the same place at no
further charge. You need not require recipients to copy the
Corresponding Source along with the object code. If the place to
copy the object code is a network server, the Corresponding Source
may be on a different server (operated by you or a third party)
that supports equivalent copying facilities, provided you maintain
clear directions next to the object code saying where to find the
Corresponding Source. Regardless of what server hosts the
Corresponding Source, you remain obligated to ensure that it is
available for as long as needed to satisfy these requirements.
e) Convey the object code using peer-to-peer transmission, provided
you inform other peers where the object code and Corresponding
Source of the work are being offered to the general public at no
charge under subsection 6d.
A separable portion of the object code, whose source code is excluded
from the Corresponding Source as a System Library, need not be
included in conveying the object code work.
A "User Product" is either (1) a "consumer product", which means any
tangible personal property which is normally used for personal, family,
or household purposes, or (2) anything designed or sold for incorporation
into a dwelling. In determining whether a product is a consumer product,
doubtful cases shall be resolved in favor of coverage. For a particular
product received by a particular user, "normally used" refers to a
typical or common use of that class of product, regardless of the status
of the particular user or of the way in which the particular user
actually uses, or expects or is expected to use, the product. A product
is a consumer product regardless of whether the product has substantial
commercial, industrial or non-consumer uses, unless such uses represent
the only significant mode of use of the product.
"Installation Information" for a User Product means any methods,
procedures, authorization keys, or other information required to install
and execute modified versions of a covered work in that User Product from
a modified version of its Corresponding Source. The information must
suffice to ensure that the continued functioning of the modified object
code is in no case prevented or interfered with solely because
modification has been made.
If you convey an object code work under this section in, or with, or
specifically for use in, a User Product, and the conveying occurs as
part of a transaction in which the right of possession and use of the
User Product is transferred to the recipient in perpetuity or for a
fixed term (regardless of how the transaction is characterized), the
Corresponding Source conveyed under this section must be accompanied
by the Installation Information. But this requirement does not apply
if neither you nor any third party retains the ability to install
modified object code on the User Product (for example, the work has
been installed in ROM).
The requirement to provide Installation Information does not include a
requirement to continue to provide support service, warranty, or updates
for a work that has been modified or installed by the recipient, or for
the User Product in which it has been modified or installed. Access to a
network may be denied when the modification itself materially and
adversely affects the operation of the network or violates the rules and
protocols for communication across the network.
Corresponding Source conveyed, and Installation Information provided,
in accord with this section must be in a format that is publicly
documented (and with an implementation available to the public in
source code form), and must require no special password or key for
unpacking, reading or copying.
7. Additional Terms.
"Additional permissions" are terms that supplement the terms of this
License by making exceptions from one or more of its conditions.
Additional permissions that are applicable to the entire Program shall
be treated as though they were included in this License, to the extent
that they are valid under applicable law. If additional permissions
apply only to part of the Program, that part may be used separately
under those permissions, but the entire Program remains governed by
this License without regard to the additional permissions.
When you convey a copy of a covered work, you may at your option
remove any additional permissions from that copy, or from any part of
it. (Additional permissions may be written to require their own
removal in certain cases when you modify the work.) You may place
additional permissions on material, added by you to a covered work,
for which you have or can give appropriate copyright permission.
Notwithstanding any other provision of this License, for material you
add to a covered work, you may (if authorized by the copyright holders of
that material) supplement the terms of this License with terms:
a) Disclaiming warranty or limiting liability differently from the
terms of sections 15 and 16 of this License; or
b) Requiring preservation of specified reasonable legal notices or
author attributions in that material or in the Appropriate Legal
Notices displayed by works containing it; or
c) Prohibiting misrepresentation of the origin of that material, or
requiring that modified versions of such material be marked in
reasonable ways as different from the original version; or
d) Limiting the use for publicity purposes of names of licensors or
authors of the material; or
e) Declining to grant rights under trademark law for use of some
trade names, trademarks, or service marks; or
f) Requiring indemnification of licensors and authors of that
material by anyone who conveys the material (or modified versions of
it) with contractual assumptions of liability to the recipient, for
any liability that these contractual assumptions directly impose on
those licensors and authors.
All other non-permissive additional terms are considered "further
restrictions" within the meaning of section 10. If the Program as you
received it, or any part of it, contains a notice stating that it is
governed by this License along with a term that is a further
restriction, you may remove that term. If a license document contains
a further restriction but permits relicensing or conveying under this
License, you may add to a covered work material governed by the terms
of that license document, provided that the further restriction does
not survive such relicensing or conveying.
If you add terms to a covered work in accord with this section, you
must place, in the relevant source files, a statement of the
additional terms that apply to those files, or a notice indicating
where to find the applicable terms.
Additional terms, permissive or non-permissive, may be stated in the
form of a separately written license, or stated as exceptions;
the above requirements apply either way.
8. Termination.
You may not propagate or modify a covered work except as expressly
provided under this License. Any attempt otherwise to propagate or
modify it is void, and will automatically terminate your rights under
this License (including any patent licenses granted under the third
paragraph of section 11).
However, if you cease all violation of this License, then your
license from a particular copyright holder is reinstated (a)
provisionally, unless and until the copyright holder explicitly and
finally terminates your license, and (b) permanently, if the copyright
holder fails to notify you of the violation by some reasonable means
prior to 60 days after the cessation.
Moreover, your license from a particular copyright holder is
reinstated permanently if the copyright holder notifies you of the
violation by some reasonable means, this is the first time you have
received notice of violation of this License (for any work) from that
copyright holder, and you cure the violation prior to 30 days after
your receipt of the notice.
Termination of your rights under this section does not terminate the
licenses of parties who have received copies or rights from you under
this License. If your rights have been terminated and not permanently
reinstated, you do not qualify to receive new licenses for the same
material under section 10.
9. Acceptance Not Required for Having Copies.
You are not required to accept this License in order to receive or
run a copy of the Program. Ancillary propagation of a covered work
occurring solely as a consequence of using peer-to-peer transmission
to receive a copy likewise does not require acceptance. However,
nothing other than this License grants you permission to propagate or
modify any covered work. These actions infringe copyright if you do
not accept this License. Therefore, by modifying or propagating a
covered work, you indicate your acceptance of this License to do so.
10. Automatic Licensing of Downstream Recipients.
Each time you convey a covered work, the recipient automatically
receives a license from the original licensors, to run, modify and
propagate that work, subject to this License. You are not responsible
for enforcing compliance by third parties with this License.
An "entity transaction" is a transaction transferring control of an
organization, or substantially all assets of one, or subdividing an
organization, or merging organizations. If propagation of a covered
work results from an entity transaction, each party to that
transaction who receives a copy of the work also receives whatever
licenses to the work the party's predecessor in interest had or could
give under the previous paragraph, plus a right to possession of the
Corresponding Source of the work from the predecessor in interest, if
the predecessor has it or can get it with reasonable efforts.
You may not impose any further restrictions on the exercise of the
rights granted or affirmed under this License. For example, you may
not impose a license fee, royalty, or other charge for exercise of
rights granted under this License, and you may not initiate litigation
(including a cross-claim or counterclaim in a lawsuit) alleging that
any patent claim is infringed by making, using, selling, offering for
sale, or importing the Program or any portion of it.
11. Patents.
A "contributor" is a copyright holder who authorizes use under this
License of the Program or a work on which the Program is based. The
work thus licensed is called the contributor's "contributor version".
A contributor's "essential patent claims" are all patent claims
owned or controlled by the contributor, whether already acquired or
hereafter acquired, that would be infringed by some manner, permitted
by this License, of making, using, or selling its contributor version,
but do not include claims that would be infringed only as a
consequence of further modification of the contributor version. For
purposes of this definition, "control" includes the right to grant
patent sublicenses in a manner consistent with the requirements of
this License.
Each contributor grants you a non-exclusive, worldwide, royalty-free
patent license under the contributor's essential patent claims, to
make, use, sell, offer for sale, import and otherwise run, modify and
propagate the contents of its contributor version.
In the following three paragraphs, a "patent license" is any express
agreement or commitment, however denominated, not to enforce a patent
(such as an express permission to practice a patent or covenant not to
sue for patent infringement). To "grant" such a patent license to a
party means to make such an agreement or commitment not to enforce a
patent against the party.
If you convey a covered work, knowingly relying on a patent license,
and the Corresponding Source of the work is not available for anyone
to copy, free of charge and under the terms of this License, through a
publicly available network server or other readily accessible means,
then you must either (1) cause the Corresponding Source to be so
available, or (2) arrange to deprive yourself of the benefit of the
patent license for this particular work, or (3) arrange, in a manner
consistent with the requirements of this License, to extend the patent
license to downstream recipients. "Knowingly relying" means you have
actual knowledge that, but for the patent license, your conveying the
covered work in a country, or your recipient's use of the covered work
in a country, would infringe one or more identifiable patents in that
country that you have reason to believe are valid.
If, pursuant to or in connection with a single transaction or
arrangement, you convey, or propagate by procuring conveyance of, a
covered work, and grant a patent license to some of the parties
receiving the covered work authorizing them to use, propagate, modify
or convey a specific copy of the covered work, then the patent license
you grant is automatically extended to all recipients of the covered
work and works based on it.
A patent license is "discriminatory" if it does not include within
the scope of its coverage, prohibits the exercise of, or is
conditioned on the non-exercise of one or more of the rights that are
specifically granted under this License. You may not convey a covered
work if you are a party to an arrangement with a third party that is
in the business of distributing software, under which you make payment
to the third party based on the extent of your activity of conveying
the work, and under which the third party grants, to any of the
parties who would receive the covered work from you, a discriminatory
patent license (a) in connection with copies of the covered work
conveyed by you (or copies made from those copies), or (b) primarily
for and in connection with specific products or compilations that
contain the covered work, unless you entered into that arrangement,
or that patent license was granted, prior to 28 March 2007.
Nothing in this License shall be construed as excluding or limiting
any implied license or other defenses to infringement that may
otherwise be available to you under applicable patent law.
12. No Surrender of Others' Freedom.
If conditions are imposed on you (whether by court order, agreement or
otherwise) that contradict the conditions of this License, they do not
excuse you from the conditions of this License. If you cannot convey a
covered work so as to satisfy simultaneously your obligations under this
License and any other pertinent obligations, then as a consequence you may
not convey it at all. For example, if you agree to terms that obligate you
to collect a royalty for further conveying from those to whom you convey
the Program, the only way you could satisfy both those terms and this
License would be to refrain entirely from conveying the Program.
13. Remote Network Interaction; Use with the GNU General Public License.
Notwithstanding any other provision of this License, if you modify the
Program, your modified version must prominently offer all users
interacting with it remotely through a computer network (if your version
supports such interaction) an opportunity to receive the Corresponding
Source of your version by providing access to the Corresponding Source
from a network server at no charge, through some standard or customary
means of facilitating copying of software. This Corresponding Source
shall include the Corresponding Source for any work covered by version 3
of the GNU General Public License that is incorporated pursuant to the
following paragraph.
Notwithstanding any other provision of this License, you have
permission to link or combine any covered work with a work licensed
under version 3 of the GNU General Public License into a single
combined work, and to convey the resulting work. The terms of this
License will continue to apply to the part which is the covered work,
but the work with which it is combined will remain governed by version
3 of the GNU General Public License.
14. Revised Versions of this License.
The Free Software Foundation may publish revised and/or new versions of
the GNU Affero General Public License from time to time. Such new versions
will be similar in spirit to the present version, but may differ in detail to
address new problems or concerns.
Each version is given a distinguishing version number. If the
Program specifies that a certain numbered version of the GNU Affero General
Public License "or any later version" applies to it, you have the
option of following the terms and conditions either of that numbered
version or of any later version published by the Free Software
Foundation. If the Program does not specify a version number of the
GNU Affero General Public License, you may choose any version ever published
by the Free Software Foundation.
If the Program specifies that a proxy can decide which future
versions of the GNU Affero General Public License can be used, that proxy's
public statement of acceptance of a version permanently authorizes you
to choose that version for the Program.
Later license versions may give you additional or different
permissions. However, no additional obligations are imposed on any
author or copyright holder as a result of your choosing to follow a
later version.
15. Disclaimer of Warranty.
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
16. Limitation of Liability.
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
SUCH DAMAGES.
17. Interpretation of Sections 15 and 16.
If the disclaimer of warranty and limitation of liability provided
above cannot be given local legal effect according to their terms,
reviewing courts shall apply local law that most closely approximates
an absolute waiver of all civil liability in connection with the
Program, unless a warranty or assumption of liability accompanies a
copy of the Program in return for a fee.
END OF TERMS AND CONDITIONS
How to Apply These Terms to Your New Programs
If you develop a new program, and you want it to be of the greatest
possible use to the public, the best way to achieve this is to make it
free software which everyone can redistribute and change under these terms.
To do so, attach the following notices to the program. It is safest
to attach them to the start of each source file to most effectively
state the exclusion of warranty; and each file should have at least
the "copyright" line and a pointer to where the full notice is found.
<one line to give the program's name and a brief idea of what it does.>
Copyright (C) <year> <name of author>
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
Also add information on how to contact you by electronic and paper mail.
If your software can interact with users remotely through a computer
network, you should also make sure that it provides a way for users to
get its source. For example, if your program is a web application, its
interface could display a "Source" link that leads users to an archive
of the code. There are many ways you could offer source, and different
solutions will be better for different programs; see section 13 for the
specific requirements.
You should also get your employer (if you work as a programmer) or school,
if any, to sign a "copyright disclaimer" for the program, if necessary.
For more information on this, and how to apply and follow the GNU AGPL, see
<https://www.gnu.org/licenses/>.

21
LICENSE-MIT Normal file
View File

@@ -0,0 +1,21 @@
MIT License
Copyright (c) 2025 polylove, LLC
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.

6
NOTICE Normal file
View File

@@ -0,0 +1,6 @@
Modifications License (AGPL-3.0)
Portions of this software have been modified by Compass (c) 2025.
These modifications are licensed under the GNU Affero General Public License v3.0 (AGPL-3.0).
The original software remains MIT-licensed (c) 2025 polylove, LLC.

194
README.md
View File

@@ -1,91 +1,191 @@
# BayesBond
This repository provides the source code for [BayesBond](https://bayesbond.vercel.app), a web application where rational thinkers can bond and form deep 1-1 [![CI](https://github.com/CompassConnections/Compass/actions/workflows/ci.yml/badge.svg)](https://github.com/CompassConnections/Compass/actions/workflows/ci.yml)
relationships in a fully transparent and efficient way. It just got released—please share it with anyone who would benefit from it! [![CD](https://github.com/CompassConnections/Compass/actions/workflows/cd.yml/badge.svg)](https://github.com/CompassConnections/Compass/actions/workflows/cd.yml)
![Vercel](https://deploy-badge.vercel.app/vercel/compass)
To contribute, please submit a pull request or issue, or fill out this [form](https://forms.gle/tKnXUMAbEreMK6FC6) for suggestions and collaborations. # Compass
This repository contains the source code for [Compass](https://compassmeet.com) — an open platform for forming deep, authentic 1-on-1 connections with clarity and efficiency.
## Features ## Features
- Extremely detailed profiles for deep connections - Extremely detailed profiles for deep connections
- Radically transparent: user base fully searchable - Radically transparent: user base fully searchable
- Free, ad-free, not for profit - Free, ad-free, not for profit (supported by donations)
- Supported by donation - Created, hosted, maintained, and moderated by volunteers
- Open source - Open source
- Democratically governed - Democratically governed
The full description is available [here](https://martinbraquet.com/meeting-rational). You can find a lot of interesting info in the [About page](https://www.compassmeet.com/about) and the [FAQ](https://www.compassmeet.com/faq) as well.
A detailed description of the early vision is also available in this [blog post](https://martinbraquet.com/meeting-rational) (you can disregard the parts about rationality, as Compass shifted to a more general audience).
**We cant do this alone.** Whatever your skills—coding, design, writing, moderation, marketing, or even small donations—you can make a real difference. [Contribute](https://www.compassmeet.com/support) in any way you can and help our community thrive!
![Demo](https://raw.githubusercontent.com/CompassConnections/assets/refs/heads/main/assets/demo-2x.gif)
## To Do ## To Do
No contribution is too small—whether its changing a color, resizing a button, tweaking a font, or improving wording. Bigger contributions like adding new profile fields, building modules, or improving onboarding are equally welcome. The goal is to make the platform better step by step, and every improvement counts. If you see something that could be clearer, smoother, or more engaging, **please jump in**!
The complete, official list of tasks is available [here on ClickUp](https://sharing.clickup.com/90181043445/l/h/6-901810339879-1/bbfd32f4f4bf64b). If you are working on one task, just assign it to yourself and move its status to "in progress". If there is also a GitHub issue for that task, assign it to yourself as well.
To have edit access to the ClickUp workspace, you need an admin to manually give you permission (one time thing). To do so, use your preferred option:
- Ask or DM an admin on [Discord](https://discord.gg/8Vd7jzqjun)
- Email hello@compassmeet.com
- Raise an issue on GitHub
If you want to add tasks without creating an account, you can simply email
```
a.t.901810339879.u-276866260.b847aba1-2709-4f17-b4dc-565a6967c234@tasks.clickup.com
```
Put the task title in the email subject and the task description in the email content.
Here is a tailored selection of things that would be very useful. If you want to help but dont know where to start, just ask us on [Discord](https://discord.gg/8Vd7jzqjun).
- [x] Authentication (user/password and Google Sign In) - [x] Authentication (user/password and Google Sign In)
- [x] Set up PostgreSQL in Production with supabase or prisma console (can stick with SQLite in dev / local) - [x] Set up PostgreSQL in Production with supabase
- [x] Set up hosting (vercel) - [x] Set up web hosting (vercel)
- [x] Ask for detailed info per profile upon registration (intellectual interests, location, cause areas, personality type, conflict style, desired type of connection, prompt answers, gender, etc.) - [x] Set up backend hosting (google cloud)
- [x] Ask for detailed info upon registration (location, desired type of connection, prompt answers, gender, etc.)
- [x] Set up page listing all the profiles - [x] Set up page listing all the profiles
- [x] Search through all the profile variables - [x] Search through most profile variables
- [ ] (Set up chat / direct messaging) - [x] Set up chat / direct messaging
- [ ] Set up domain name (https://bayesbond.com) - [x] Set up domain name (compassmeet.com)
- [ ] Add mobile app (React Native on Android and iOS)
- [ ] Add better onboarding (tooltips, modals, etc.)
- [ ] Add modules to learn more about each other (personality test, conflict style, love languages, etc.)
- [ ] Add modules to improve interpersonal skills (active listening, nonviolent communication, etc.)
- [ ] Add calendar integration and scheduling
- [ ] Add events (group calls, in-person meetups, etc.)
#### Secondary To Do #### Secondary To Do
Any action item is open to anyone for collaboration, but the following ones are particularly easy to do for first-time contributors. Everything is open to anyone for collaboration, but the following ones are particularly easy to do for first-time contributors.
- [ ] Clean up terms and conditions - [x] Clean up learn more page
- [ ] Clean up privacy notice
- [ ] Clean up learn more page
- [x] Add dark theme - [x] Add dark theme
- [ ] Cover with tests - [ ] Add profile fields (intellectual interests, cause areas, personality type, conflict style, timezone, etc.)
- [ ] Add filters to search through remaining profile fields (politics, religion, education level, etc.)
- [ ] Cover with tests (crucial, just the test template and framework are ready)
- [ ] Make the app more user-friendly and appealing (UI/UX)
- [ ] Clean up terms and conditions (convert to Markdown)
- [ ] Clean up privacy notice (convert to Markdown)
- [ ] Add other authentication methods (GitHub, Facebook, Apple, phone, etc.)
- [x] Add email verification
- [x] Add password reset
- [x] Add automated welcome email
- [ ] Security audit and penetration testing
- [ ] Make `deploy-api.sh` run automatically on push to `main` branch
- [x] Create settings page (change email, password, delete account, etc.)
- [ ] Improve [financials](web/public/md/financials.md) page (donor / acknowledgments, etc.)
- [x] Improve loading sign (e.g., animation of a compass moving around)
- [ ] Show compatibility score in profile page
## Implementation ## Implementation
The web app is coded in Typescript using React as front-end and Prisma as back-end. It includes: The web app is coded in Typescript using React as front-end. It includes:
- [NextAuth.js v4](https://next-auth.js.org/) - [Supabase](https://supabase.com/) for the PostgreSQL database
- [Prisma Postgres](https://www.prisma.io/postgres) - [Google Cloud](https://console.cloud.google.com) for hosting the backend API
- [Prisma ORM](https://www.prisma.io/orm) - [Firebase](https://firebase.google.com/) for authentication and media storage
- Vercel - [Vercel](https://vercel.com/) for hosting the front-end
## Development ## Development
After cloning the repo and navigating into it, install dependencies: Below are the steps to contribute. If you have any trouble or questions, please don't hesitate to open an issue or contact us on [Discord](https://discord.gg/8Vd7jzqjun)! We're responsive and happy to help.
``` ### Installation
npm install
```
You now need to configure your database connection via an environment variable.
First, create an `.env` file:
Fork the [repo](https://github.com/CompassConnections/Compass) on GitHub (button in top right). Then, clone your repo and navigating into it:
```bash ```bash
cp .env.example .env git clone https://github.com/<your-username>/Compass.git
cd Compass
``` ```
To ensure your authentication works properly, you'll also need to set the `AUTH_SECRET` [env var for NextAuth.js] Install `opentofu`, `docker`, and `yarn`. Try running this on Linux or macOS for a faster install:
(https://next-auth.js.org/configuration/options). You can generate such a random 32-character string with:
```bash ```bash
npx auth secret ./setup.sh
```
If it doesn't work, you can install them manually (google how to install `opentofu`, `docker`, and `yarn` for your OS).
Then, install the dependencies for this project:
```bash
yarn install
``` ```
In the end, your entire `.env` file should look similar to this (but using _your own values_ for the env vars): ### Environment Variables
```bash
DATABASE_URL="file:./dev.db"
AUTH_SECRET="gTwLSXFeNWFRpUTmxlRniOfegXYw445pd0k6JqXd7Ag="
```
Run the following commands to set up your local development database and Prisma schema: Almost all the features will work out of the box, so you can skip this step and come back later if you need to test the following services: email, geolocation.
We can't make the following information public, for security and privacy reasons:
- Database, otherwise anyone could access all the user data (including private messages)
- Firebase, otherwise anyone could remove users or modify the media files
- Email, analytics, and location services, otherwise anyone could use the service plans Compass paid for and run up the bill.
That's why we separate all those services between production and development environments, so that you can code freely without impacting the functioning of the deployed platform.
Contributors should use the default keys for local development. Production uses a separate environment with stricter rules and private keys that are not shared.
If you do need one of the few remaining services, you need to set them up and store your own secrets as environment variables. To do so, simply open `.env` and fill in the variables according to the instructions in the file.
### Tests
Make sure the tests pass:
```bash ```bash
npx prisma migrate dev --name init yarn test tests/jest/
``` ```
Note that your local database will be made of synthetic data, not real users. This is fine for development and testing. TODO: make `yarn test` run all the tests, not just the ones in `tests/jest/`.
### Running the Development Server
Start the development server: Start the development server:
```bash ```bash
npm run dev yarn dev
``` ```
Once the server is running, visit http://localhost:3000 to start using the app. Once the server is running, visit http://localhost:3000 to start using the app. You can sign up and visit the profiles; you should see a few synthetic profiles.
See [development.md](docs/development.md) for additional instructions, such as adding new profile features. Note: it's normal if page loading locally is much slower than the deployed version. It can take up to 10 seconds, it would be great to improve that though!
### Contributing
Now you can start contributing by making changes and submitting pull requests!
We recommend using a good code editor (VSCode, WebStorm, Cursor, etc.) with Typescript support and a good AI assistant (GitHub Copilot, etc.) to make your life easier. To debug, you can use the browser developer tools (F12), specifically:
- Components tab to see the React component tree and props (you need to install the [React Developer Tools](https://react.dev/learn/react-developer-tools) extension)
- Console tab for errors and logs
- Network tab to see the requests and responses
- Storage tab to see cookies and local storage
You can also add `console.log()` statements in the code.
If you are new to Typescript or the open-source space, you could start with small changes, such as tweaking some web components or improving wording in some pages. You can find those files in `web/public/md/`.
See [development.md](docs/development.md) for additional instructions, such as adding new profile fields.
### Submission
Add the original repo as upstream for syncing:
```bash
git remote add upstream https://github.com/CompassConnections/Compass.git
```
Create a new branch for your changes:
```bash
git checkout -b <branch-name>
```
Make changes, then stage and commit:
```bash
git add .
git commit -m "Describe your changes"
```
Push branch to your fork:
```bash
git push origin <branch-name>
```
Finally, open a Pull Request on GitHub from your `fork/<branch-name>``CompassConnections/Compass` main branch.
## Acknowledgements
This project is built on top of [manifold.love](https://github.com/sipec/polylove), an open-source dating platform licensed under the MIT License. We greatly appreciate their work and contributions to open-source, which have significantly aided in the development of some core features such as direct messaging, prompts, and email notifications. We invite the community to explore and contribute to other open-source projects like manifold.love as well, especially if you're interested in functionalities that deviate from Compass' ideals of deep, intentional connections.

View File

@@ -8,5 +8,5 @@
## Reporting a Vulnerability ## Reporting a Vulnerability
Contact the development team to report a vulnerability. You should receive updates within a week. Contact the development team at hello@compassmeet.com to report a vulnerability. You should receive updates within a week.

9
_old/.eslintrc.js Normal file
View File

@@ -0,0 +1,9 @@
// .eslintrc.js
module.exports = {
root: true,
extends: ['next', 'next/core-web-vitals'],
rules: {
'@typescript-eslint/no-explicit-any': 'off',
'react/no-unescaped-entities': 'off',
},
}

View File

@@ -1,26 +1,26 @@
"use client"; "use client";
import { useState, useEffect } from "react"; import {useEffect, useState} from "react";
import Link from "next/link"; import Link from "next/link";
import { signOut, useSession } from "next-auth/react"; import {useSession} from "next-auth/react";
import { FaHome } from "react-icons/fa";
import ThemeToggle from "@/lib/client/theme"; import ThemeToggle from "@/lib/client/theme";
import FavIcon from "@/components/FavIcon";
export default function Header() { export default function Header() {
const { data: session } = useSession(); const {data: session} = useSession();
const [isSmallScreen, setIsSmallScreen] = useState(false); const [isSmallScreen, setIsSmallScreen] = useState(false);
useEffect(() => { useEffect(() => {
const checkScreenSize = () => { const checkScreenSize = () => {
setIsSmallScreen(window.innerWidth < 640); // Tailwind's 'sm' breakpoint is 640px setIsSmallScreen(window.innerWidth < 640); // Tailwind's 'sm' breakpoint is 640px
}; };
// Initial check // Initial check
checkScreenSize(); checkScreenSize();
// Add event listener for window resize // Add event listener for window resize
window.addEventListener('resize', checkScreenSize); window.addEventListener('resize', checkScreenSize);
// Clean up the event listener when the component unmounts // Clean up the event listener when the component unmounts
return () => window.removeEventListener('resize', checkScreenSize); return () => window.removeEventListener('resize', checkScreenSize);
}, []); }, []);
@@ -30,25 +30,30 @@ export default function Header() {
return ( return (
<header className="w-full <header className="w-full
{/*shadow-md*/} {/*shadow-md*/}
py-4 px-8 xs:px-4"> py-5 px-8 xs:px-4">
<nav className="flex justify-between items-center"> <nav className="flex justify-between items-center">
<Link <Link
href="/" href="/"
className="text-xl font-bold hover:text-blue-600 transition-colors flex items-center" className="text-4xl font-bold hover:text-blue-600 transition-colors flex items-center"
aria-label={isSmallScreen ? "Home" : "IntentionalBond"} aria-label={isSmallScreen ? "Home" : "Compass"}
> >
{isSmallScreen ? <FaHome className="w-5 h-5" /> : 'IntentionalBond'} <FavIcon className="dark:invert"/>
{!isSmallScreen && (
<span className="flex items-center gap-2">
Compass
</span>
)}
</Link> </Link>
<div className="flex items-center space-x-2"> <div className="flex items-center space-x-3">
<ThemeToggle/> <ThemeToggle/>
<div className="flex items-center space-x-2"> <div className="flex items-center space-x-2">
<Link <Link
href="/learn-more" href="/about"
className={`${fontStyle} bg-gray-200 dark:bg-gray-600 text-gray-800 dark:text-white rounded-lg hover:bg-gray-300 dark:hover:bg-gray-500`} className={`${fontStyle} bg-gray-200 dark:bg-gray-600 text-gray-800 dark:text-white rounded-full hover:bg-gray-300 dark:hover:bg-gray-500`}
> >
Learn More About
</Link> </Link>
</div> </div>
{session ? ( {session ? (
@@ -62,27 +67,22 @@ export default function Header() {
</Link> </Link>
{/*<Link*/} {/*<Link*/}
{/* href="/profiles"*/} {/* href="/profiles"*/}
{/* className="bg-blue-500 text-white px-4 py-2 rounded-lg hover:bg-blue-600 transition"*/} {/* className="bg-blue-500 text-white px-4 py-2 rounded-full hover:bg-blue-600 transition"*/}
{/*>*/} {/*>*/}
{/* Dashboard*/} {/* Dashboard*/}
{/*</Link>*/} {/*</Link>*/}
<button
onClick={() => signOut({callbackUrl: "/"})}
className={`${fontStyle} bg-red-500 text-white rounded-lg hover:bg-red-600`}
>
Sign Out
</button>
</div> </div>
</> </>
) : ( ) : (
<> <>
<Link href="/login" className={`${fontStyle} bg-blue-500 text-white rounded-lg hover:bg-blue-600 `}> <Link href="/login"
className={`${fontStyle} bg-gray-200 dark:bg-gray-600 text-gray-800 dark:text-white rounded-full hover:bg-gray-300 dark:hover:bg-gray-500`}>
Sign In Sign In
</Link> </Link>
<Link href="/register" {/*<Link href="/register"
className={`${fontStyle} bg-blue-500 text-white rounded-lg hover:bg-blue-600`}> className={`${fontStyle} bg-blue-500 text-white rounded-full hover:bg-blue-600`}>
Sign Up Sign Up
</Link> </Link> */}
</> </>
)} )}
</div> </div>

View File

@@ -46,7 +46,7 @@ export async function POST(req: Request) {
// const emailHtml = await render(VerificationEmail({ url: verificationUrl })); // const emailHtml = await render(VerificationEmail({ url: verificationUrl }));
// try { // try {
// let payload = { // let payload = {
// from: `BayesBond <${process.env.EMAIL_FROM!}>`, // from: `Compass <${process.env.EMAIL_FROM!}>`,
// to: email, // to: email,
// subject: 'Verify your email', // subject: 'Verify your email',
// html: emailHtml, // html: emailHtml,

View File

@@ -1,10 +1,11 @@
import { prisma } from "@/lib/server/prisma"; import {prisma} from "@/lib/server/prisma";
import { NextResponse } from "next/server"; import {NextResponse} from "next/server";
export async function GET() { export async function GET() {
try { try {
// Get all interests from the database // Get all interests from the database
const cacheStrategy = { swr: 60, ttl: 60, tags: ["interests"] }; // Disable cache for now as it bugs when saving profile with new interest and clicking on "Edit Profile" just after
const cacheStrategy = {swr: 0, ttl: 0, tags: ["interests"]};
const interests = await prisma.interest.findMany({ const interests = await prisma.interest.findMany({
select: { select: {
id: true, id: true,
@@ -27,6 +28,17 @@ export async function GET() {
cacheStrategy: cacheStrategy, cacheStrategy: cacheStrategy,
}); });
const books = await prisma.book.findMany({
select: {
id: true,
name: true,
},
orderBy: {
name: 'asc'
},
cacheStrategy: cacheStrategy,
});
const causeAreas = await prisma.causeArea.findMany({ const causeAreas = await prisma.causeArea.findMany({
select: { select: {
id: true, id: true,
@@ -49,12 +61,12 @@ export async function GET() {
cacheStrategy: cacheStrategy, cacheStrategy: cacheStrategy,
}); });
return NextResponse.json({ interests, coreValues, causeAreas, connections }); return NextResponse.json({interests, coreValues, books, causeAreas, connections});
} catch (error) { } catch (error) {
console.error('Error fetching interests:', error); console.error('Error fetching interests:', error);
return NextResponse.json( return NextResponse.json(
{ error: "Failed to fetch interests" }, {error: "Failed to fetch interests"},
{ status: 500 } {status: 500}
); );
} }
} }

View File

@@ -82,7 +82,7 @@ export async function DELETE(
prisma.profileCauseArea.deleteMany({ prisma.profileCauseArea.deleteMany({
where: { profileId: id }, where: { profileId: id },
}), }),
// Delete desired connections // Delete Type of Connection
prisma.profileConnection.deleteMany({ prisma.profileConnection.deleteMany({
where: { profileId: id }, where: { profileId: id },
}), }),

View File

@@ -12,9 +12,10 @@ export async function GET(request: Request) {
const maxIntroversion = url.searchParams.get("maxIntroversion"); const maxIntroversion = url.searchParams.get("maxIntroversion");
const interests = url.searchParams.get("interests")?.split(",").filter(Boolean) || []; const interests = url.searchParams.get("interests")?.split(",").filter(Boolean) || [];
const coreValues = url.searchParams.get("coreValues")?.split(",").filter(Boolean) || []; const coreValues = url.searchParams.get("coreValues")?.split(",").filter(Boolean) || [];
const books = url.searchParams.get("books")?.split(",").filter(Boolean) || [];
const causeAreas = url.searchParams.get("causeAreas")?.split(",").filter(Boolean) || []; const causeAreas = url.searchParams.get("causeAreas")?.split(",").filter(Boolean) || [];
const connections = url.searchParams.get("connections")?.split(",").filter(Boolean) || []; const connections = url.searchParams.get("connections")?.split(",").filter(Boolean) || [];
const searchQuery = url.searchParams.get("search") || ""; const searchQueries = url.searchParams.get("searchQuery")?.split(",").map(q => q.trim()).filter(Boolean) || [];
const profilesPerPage = 100; const profilesPerPage = 100;
const offset = (page - 1) * profilesPerPage; const offset = (page - 1) * profilesPerPage;
@@ -116,6 +117,22 @@ export async function GET(request: Request) {
]; ];
} }
// AND
if (books.length > 0) {
where.profile.AND = [
...where.profile.AND,
...books.map((name) => ({
books: {
some: {
value: {
name: name,
},
},
},
})),
];
}
if (causeAreas.length > 0) { if (causeAreas.length > 0) {
where.profile.AND = [ where.profile.AND = [
...where.profile.AND, ...where.profile.AND,
@@ -145,93 +162,110 @@ export async function GET(request: Request) {
}; };
} }
if (searchQuery) { if (searchQueries.length > 0) {
where.OR = [ where.AND = [
{name: {contains: searchQuery, mode: 'insensitive'}}, ...(where.AND ?? []),
// {email: {contains: searchQuery, mode: 'insensitive'}}, ...searchQueries.map(query => ({
{ OR: [
profile: { {name: {contains: query, mode: 'insensitive'}},
description: {contains: searchQuery, mode: 'insensitive'}, // {email: {contains: searchQuery, mode: 'insensitive'}},
}, {
}, profile: {
{ description: {contains: query, mode: 'insensitive'},
profile: { },
occupation: {contains: searchQuery, mode: 'insensitive'}, },
}, {
}, profile: {
{ occupation: {contains: query, mode: 'insensitive'},
profile: { },
location: {contains: searchQuery, mode: 'insensitive'}, },
}, {
}, profile: {
{ location: {contains: query, mode: 'insensitive'},
profile: { },
contactInfo: {contains: searchQuery, mode: 'insensitive'}, },
}, {
}, profile: {
{ contactInfo: {contains: query, mode: 'insensitive'},
profile: { },
intellectualInterests: { },
some: { {
interest: { profile: {
name: {contains: searchQuery, mode: "insensitive"}, intellectualInterests: {
some: {
interest: {
name: {contains: query, mode: "insensitive"},
},
},
}, },
}, },
}, },
}, {
}, profile: {
{ coreValues: {
profile: { some: {
coreValues: { value: {
some: { name: {contains: query, mode: "insensitive"},
value: { },
name: {contains: searchQuery, mode: "insensitive"}, },
}, },
}, },
}, },
}, {
}, profile: {
{ books: {
profile: { some: {
causeAreas: { value: {
some: { name: {contains: query, mode: "insensitive"},
causeArea: { },
name: {contains: searchQuery, mode: "insensitive"}, },
}, },
}, },
}, },
}, {
}, profile: {
{ causeAreas: {
profile: { some: {
desiredConnections: { causeArea: {
some: { name: {contains: query, mode: "insensitive"},
connection: { },
name: {contains: searchQuery, mode: "insensitive"}, },
}, },
}, },
}, },
}, {
}, profile: {
{ desiredConnections: {
profile: { some: {
promptAnswers: { connection: {
some: { name: {contains: query, mode: "insensitive"},
answer: {contains: searchQuery, mode: "insensitive"}, },
},
},
}, },
}, },
}, {
}, profile: {
{ promptAnswers: {
profile: { some: {
promptAnswers: { answer: {contains: query, mode: "insensitive"},
some: { },
prompt: {contains: searchQuery, mode: "insensitive"}, },
}, },
}, },
}, {
}, profile: {
]; promptAnswers: {
some: {
prompt: {contains: query, mode: "insensitive"},
},
},
},
},
]
}))
]
} }
console.log(where.profile); console.log(where.profile);
@@ -253,6 +287,7 @@ export async function GET(request: Request) {
include: { include: {
intellectualInterests: {include: {interest: true}}, intellectualInterests: {include: {interest: true}},
coreValues: {include: {value: true}}, coreValues: {include: {value: true}},
books: {include: {value: true}},
causeAreas: {include: {causeArea: true}}, causeAreas: {include: {causeArea: true}},
desiredConnections: {include: {connection: true}}, desiredConnections: {include: {connection: true}},
promptAnswers: true, promptAnswers: true,

View File

@@ -14,8 +14,9 @@ export async function POST(req: Request) {
} }
const data = await req.json(); const data = await req.json();
const {profile, image, name, interests = [], connections = [], coreValues = [], causeAreas = []} = data; const {profile, image, name, interests = [], connections = [], coreValues = [], books = [], causeAreas = []} = data;
console.log('books: ', books)
Object.keys(profile).forEach(key => { Object.keys(profile).forEach(key => {
if (profile[key] === '' || !profile[key]) { if (profile[key] === '' || !profile[key]) {
delete profile[key]; delete profile[key];
@@ -71,6 +72,8 @@ export async function POST(req: Request) {
profileConnection: prisma.profileConnection, profileConnection: prisma.profileConnection,
value: prisma.value, value: prisma.value,
profileValue: prisma.profileValue, profileValue: prisma.profileValue,
book: prisma.book,
profileBook: prisma.profileBook,
causeArea: prisma.causeArea, causeArea: prisma.causeArea,
profileCauseArea: prisma.profileCauseArea, profileCauseArea: prisma.profileCauseArea,
} as const; } as const;
@@ -79,7 +82,7 @@ export async function POST(req: Request) {
async function handleFeatures(features: any, attribute: ModelKey, profileAttribute: string, idName: string) { async function handleFeatures(features: any, attribute: ModelKey, profileAttribute: string, idName: string) {
// Add new features // Add new features
if (features.length > 0 && updatedUser.profile) { if (features !== null && updatedUser.profile) {
// First, find or create all features // First, find or create all features
console.log('profile', profileAttribute, profileAttribute); console.log('profile', profileAttribute, profileAttribute);
const operations = features.map((feat: { id?: string; name: string }) => const operations = features.map((feat: { id?: string; name: string }) =>
@@ -95,25 +98,31 @@ export async function POST(req: Request) {
// Get the IDs of all created/updated features // Get the IDs of all created/updated features
const ids = createdFeatures.map(v => v.id); const ids = createdFeatures.map(v => v.id);
// First, remove all existing interests for this profile const profileId = updatedUser.profile.id;
await modelMap[profileAttribute].deleteMany({ console.log('profile ID:', profileId);
where: {profileId: updatedUser.profile.id},
});
// Then, create new connections // First, remove all existing features for this profile
const res = await modelMap[profileAttribute].deleteMany({
where: {profileId: profileId},
});
console.log('deleted profile:', profileAttribute, res);
// Then, create new features
if (ids.length > 0) { if (ids.length > 0) {
await modelMap[profileAttribute].createMany({ const create_res =await modelMap[profileAttribute].createMany({
data: ids.map(id => ({ data: ids.map(id => ({
profileId: updatedUser.profile!.id, profileId: profileId,
[idName]: id, [idName]: id,
})), })),
skipDuplicates: true, skipDuplicates: true,
}); });
console.log('created many:', profileAttribute, create_res);
} }
} }
} }
await handleFeatures(interests, 'interest', 'profileInterest', 'interestId') await handleFeatures(interests, 'interest', 'profileInterest', 'interestId')
await handleFeatures(books, 'book', 'profileBook', 'valueId')
await handleFeatures(connections, 'connection', 'profileConnection', 'connectionId') await handleFeatures(connections, 'connection', 'profileConnection', 'connectionId')
await handleFeatures(coreValues, 'value', 'profileValue', 'valueId') await handleFeatures(coreValues, 'value', 'profileValue', 'valueId')
await handleFeatures(causeAreas, 'causeArea', 'profileCauseArea', 'causeAreaId') await handleFeatures(causeAreas, 'causeArea', 'profileCauseArea', 'causeAreaId')

View File

@@ -1,6 +1,6 @@
'use client'; 'use client';
import {ChangeEvent, ReactNode, Suspense, useEffect, useRef, useState} from 'react'; import React, {ChangeEvent, ReactNode, Suspense, useEffect, useRef, useState} from 'react';
import {useRouter, useSearchParams} from 'next/navigation'; import {useRouter, useSearchParams} from 'next/navigation';
import {signOut, useSession} from 'next-auth/react'; import {signOut, useSession} from 'next-auth/react';
import Image from 'next/image'; import Image from 'next/image';
@@ -10,6 +10,11 @@ import {DeleteProfileButton} from "@/lib/client/profile";
import PromptAnswer from '@/components/ui/PromptAnswer'; import PromptAnswer from '@/components/ui/PromptAnswer';
import imageCompression from 'browser-image-compression'; import imageCompression from 'browser-image-compression';
import {Item} from '@/lib/client/schema';
import {fetchFeatures} from "@/lib/client/fetching";
import {errorBlock} from "@/lib/client/errors";
import Slider from "@mui/material/Slider";
export default function CompleteProfile() { export default function CompleteProfile() {
return ( return (
@@ -51,28 +56,46 @@ function RegisterComponent() {
const router = useRouter(); const router = useRouter();
const {data: session, update} = useSession(); const {data: session, update} = useSession();
const hooks = Object.fromEntries(['interests', 'coreValues', 'description', 'connections', 'causeAreas'].map((id) => { const featureNames = ['interests', 'coreValues', 'description', 'connections', 'causeAreas', 'books'];
const [showMoreInfo, setShowMoreInfo] = useState(false);
const [newFeature, setNewFeature] = useState('');
const [allFeatures, setAllFeatures] = useState<{ id: string, name: string }[]>([]);
const [selectedFeatures, setSelectedFeatures] = useState<Set<string>>(new Set());
const dropdownRef = useRef<HTMLDivElement>(null);
const [showDropdown, setShowDropdown] = useState(false);
return [id, { const [showMoreInfo, _setShowMoreInfo] = useState(() =>
showMoreInfo, Object.fromEntries(featureNames.map((id) => [id, false]))
setShowMoreInfo, );
newFeature, const setShowMoreInfo = (id: string, value: boolean) => {
setNewFeature, _setShowMoreInfo((prev) => ({...prev, [id]: value}));
allFeatures, };
setAllFeatures,
selectedFeatures, const [newFeature, _setNewFeature] = useState(() =>
setSelectedFeatures, Object.fromEntries(featureNames.map((id) => [id, '']))
dropdownRef, );
showDropdown, const setNewFeature = (id: string, value: string) => {
setShowDropdown _setNewFeature((prev) => ({...prev, [id]: value}));
}] };
}));
const [allFeatures, _setAllFeatures] = useState(() =>
Object.fromEntries(featureNames.map((id) => [id, [] as Item[]]))
);
const setAllFeatures = (id: string, value: any) => {
_setAllFeatures((prev) => ({...prev, [id]: value}));
};
const [selectedFeatures, _setSelectedFeatures] = useState(() =>
Object.fromEntries(featureNames.map((id) => [id, new Set<string>()]))
);
const setSelectedFeatures = (id: string, value: Set<string>) => {
_setSelectedFeatures((prev) => ({...prev, [id]: value}));
}
const [showDropdown, _setShowDropdown] = useState(() =>
Object.fromEntries(featureNames.map((id) => [id, false]))
);
const setShowDropdown = (id: string, value: boolean) => {
_setShowDropdown((prev) => ({...prev, [id]: value}));
};
const refDropdown = useRef<any>(
Object.fromEntries(featureNames.map((id) => [id, React.createRef<HTMLDivElement>()]))
);
const id = session?.user.id const id = session?.user.id
@@ -106,18 +129,19 @@ function RegisterComponent() {
} }
// Set selected interests if any // Set selected interests if any
function setSelectedFeatures(id: string, attribute: string, subAttribute: string) { function setSelFeat(id: string, attribute: string, subAttribute: string) {
const feature = profile[attribute]; const feature = profile[attribute];
if (feature?.length > 0) { if (feature?.length > 0) {
const ids = feature.map((pi: any) => pi[subAttribute].id); const ids = feature.map((pi: any) => pi[subAttribute].id);
hooks[id].setSelectedFeatures(new Set(ids)); setSelectedFeatures(id, new Set(ids));
} }
} }
setSelectedFeatures('interests', 'intellectualInterests', 'interest') setSelFeat('interests', 'intellectualInterests', 'interest')
setSelectedFeatures('coreValues', 'coreValues', 'value') setSelFeat('coreValues', 'coreValues', 'value')
setSelectedFeatures('connections', 'desiredConnections', 'connection') setSelFeat('connections', 'desiredConnections', 'connection')
setSelectedFeatures('causeAreas', 'causeAreas', 'causeArea') setSelFeat('causeAreas', 'causeAreas', 'causeArea')
setSelFeat('books', 'books', 'value')
setImages([]) setImages([])
setKeys(profile?.images) setKeys(profile?.images)
@@ -156,45 +180,37 @@ function RegisterComponent() {
}, []); }, []);
// Load existing interests and set up click-outside handler
useEffect(() => { useEffect(() => {
async function fetchFeatures() { fetchFeatures(setAllFeatures)
try {
const res = await fetch('/api/interests');
if (res.ok) {
const data = await res.json();
for (const id of ['interests', 'coreValues', 'connections', 'causeAreas']) {
hooks[id].setAllFeatures(data[id] || []);
// Close dropdown when clicking outside
const handleClickOutside = (event: MouseEvent) => {
const hook = hooks[id];
const current = hook.dropdownRef.current;
if (current && !current.contains(event.target as Node)) {
hook.setShowDropdown(false);
}
};
document.addEventListener('mousedown', handleClickOutside);
// return () => {
// document.removeEventListener('mousedown', handleClickOutside);
// };
}
}
} catch (error) {
console.error('Error loading' + id, error);
}
}
fetchFeatures();
}, []); }, []);
useEffect(() => {
// Close dropdown when clicking outside
const handleClickOutside = (event: MouseEvent) => {
for (const id in showDropdown) {
const ref = refDropdown.current[id];
if (
ref?.current &&
!ref.current.contains(event.target as Node)
) {
setShowDropdown(id, false);
}
}
};
document.addEventListener('mousedown', handleClickOutside);
return () => {
document.removeEventListener('mousedown', handleClickOutside);
};
}, [showDropdown]);
if (isLoading) { if (isLoading) {
return ( return (
<div className="flex justify-center items-center min-h-screen"> <div className="flex justify-center min-h-screen py-8">
<div className="animate-spin rounded-full h-12 w-12 border-t-2 border-b-2 border-blue-500"></div> <div data-testid="spinner"
className="animate-spin rounded-full h-12 w-12 border-t-2 border-b-2 border-blue-500"></div>
</div> </div>
); )
} }
const handleImagesUpload = async (e: ChangeEvent<HTMLInputElement>) => { const handleImagesUpload = async (e: ChangeEvent<HTMLInputElement>) => {
@@ -294,10 +310,11 @@ function RegisterComponent() {
...(key && {image: key}), ...(key && {image: key}),
...(name && {name}), ...(name && {name}),
}; };
for (const name of ['interests', 'connections', 'coreValues', 'causeAreas']) { for (const name of ['books', 'interests', 'connections', 'coreValues', 'causeAreas']) {
data[name] = Array.from(hooks[name].selectedFeatures).map(id => ({ // if (!selectedFeatures[name].size) continue;
data[name] = Array.from(selectedFeatures[name]).map(id => ({
id: id.startsWith('new-') ? undefined : id, id: id.startsWith('new-') ? undefined : id,
name: hooks[name].allFeatures.find(i => i.id === id)?.name || id.replace('new-', '') name: allFeatures[name].find(i => i.id === id)?.name
})); }));
} }
console.log('data', data) console.log('data', data)
@@ -327,24 +344,21 @@ function RegisterComponent() {
const genderOptions = Object.values(Gender); const genderOptions = Object.values(Gender);
// const personalityOptions = Object.values(PersonalityType); // const personalityOptions = Object.values(PersonalityType);
const conflictOptions = Object.values(ConflictStyle); // const conflictOptions = Object.values(ConflictStyle);
const headingStyle = "block text-base font-medium text-gray-700 dark:text-white mb-1"; const headingStyle = "block text-base font-medium text-gray-700 dark:text-white mb-1";
function getDetails(id: string, brief: string, text: ReactNode) { function getDetails(id: string, brief: string, text: ReactNode) {
const hook = hooks[id];
const showMoreInfo = hook.showMoreInfo;
const setShowMoreInfo = hook.setShowMoreInfo;
return <> return <>
<div className="mt-2 mb-4"> <div className="mt-2 mb-4">
<button <button
type="button" type="button"
onClick={() => setShowMoreInfo(!showMoreInfo)} onClick={() => setShowMoreInfo(id, !showMoreInfo[id])}
className="text-sm text-blue-600 hover:text-blue-800 dark:text-blue-400 dark:hover:text-blue-300 flex items-center" className="text-sm text-blue-600 hover:text-blue-800 dark:text-blue-400 dark:hover:text-blue-300 flex items-center"
> >
{showMoreInfo ? 'Hide info' : brief} {showMoreInfo[id] ? 'Hide info' : brief}
<svg <svg
className={`w-4 h-4 ml-1 transition-transform ${showMoreInfo ? 'rotate-180' : ''}`} className={`w-4 h-4 ml-1 transition-transform ${showMoreInfo[id] ? 'rotate-180' : ''}`}
fill="none" fill="none"
stroke="currentColor" stroke="currentColor"
viewBox="0 0 24 24" viewBox="0 0 24 24"
@@ -352,7 +366,7 @@ function RegisterComponent() {
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M19 9l-7 7-7-7"/> <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M19 9l-7 7-7-7"/>
</svg> </svg>
</button> </button>
{showMoreInfo && ( {showMoreInfo[id] && (
<div className="mt-2 p-3 bg-gray-50 dark:bg-gray-800 rounded-md text-sm text-gray-700 dark:text-gray-300"> <div className="mt-2 p-3 bg-gray-50 dark:bg-gray-800 rounded-md text-sm text-gray-700 dark:text-gray-300">
{text} {text}
</div> </div>
@@ -370,11 +384,11 @@ function RegisterComponent() {
const dropdownConfig: DropdownConfig[] = [ const dropdownConfig: DropdownConfig[] = [
{ {
id: 'connections', title: 'Desired Connections', allowAdd: false, id: 'connections', title: 'Connection Type', allowAdd: false,
content: null content: null
}, },
{ {
id: 'coreValues', title: 'Core Values', allowAdd: true, id: 'coreValues', title: 'Values', allowAdd: true,
content: <> content: <>
<p className="mt-2"> <p className="mt-2">
When defining your core values on a platform meant for forming deep, lasting bonds, focus on what governs your When defining your core values on a platform meant for forming deep, lasting bonds, focus on what governs your
@@ -390,7 +404,7 @@ function RegisterComponent() {
</> </>
}, },
{ {
id: 'interests', title: 'Core Interests', allowAdd: true, id: 'interests', title: 'Interests', allowAdd: true,
content: <> content: <>
<p className="mt-2"> <p className="mt-2">
When selecting your core interests on a platform designed to foster deep, lasting When selecting your core interests on a platform designed to foster deep, lasting
@@ -405,45 +419,50 @@ function RegisterComponent() {
</> </>
}, },
{ {
id: 'causeAreas', title: 'Cause Areas', allowAdd: true, id: 'books', title: 'Works to discuss', allowAdd: true,
content: <> content: <>
<p className="mt-2"> <p className="mt-2">
When choosing your cause areas on a platform designed for deep, lasting connection, focus on the issues that List the works (books, articles, essays, reports, etc.) you would like to bring up.
you feel personally compelled to engage withnot just whats socially approved or intellectually interesting. For each work, include the exact title (as it appears on the cover), the
Good cause areas reveal what breaks your heart, what energizes your long-term thinking, or what youd authors full name, and, if necessary, the edition or publication year. For example: <i>Peter Singer - Animal
willingly struggle for even if no one noticed. Be honest about what genuinely matters to you: that might be Liberation</i>. If you want to focus on specific
existential risk reduction, prison abolition, mental health reform, animal welfare, epistemic integrity, or chapters, themes, or questions, note them in your descriptionit helps keep the discussion targeted. Dont just write
education equity. You dont need to be an expert or activist to list a causejust sincerely invested. The something by Orwell or that new mystery; vague entries waste time and make it harder for others to find
point isnt to posture but to expose what kind of future you want to help shape, and to find others whose the right work. Be explicit so everyone is literally on the same page!
moral intuitions and sense of responsibility resonate with your own. That kind of alignment creates not just
shared goals, but durable trust.
</p> </p>
</> </>
}, },
// {
// id: 'causeAreas', title: 'Cause Areas', allowAdd: true,
// content: <>
// <p className="mt-2">
// When choosing your cause areas on a platform designed for deep, lasting connection, focus on the issues that
// you feel personally compelled to engage with—not just whats socially approved or intellectually interesting.
// Good cause areas reveal what breaks your heart, what energizes your long-term thinking, or what youd
// willingly struggle for even if no one noticed. Be honest about what genuinely matters to you: that might be
// existential risk reduction, prison abolition, mental health reform, animal welfare, epistemic integrity, or
// education equity. You dont need to be an expert or activist to list a cause—just sincerely invested. The
// point isnt to posture but to expose what kind of future you want to help shape, and to find others whose
// moral intuitions and sense of responsibility resonate with your own. That kind of alignment creates not just
// shared goals, but durable trust.
// </p>
// </>
// },
] ]
function getDropdown({id, title, allowAdd, content}: DropdownConfig) { function getDropdown({id, title, allowAdd, content}: DropdownConfig) {
const hook = hooks[id]; const newFeat = newFeature[id];
const newFeature = hook.newFeature; const allFeat = allFeatures[id];
const setNewFeature = hook.setNewFeature; const selectedFeat = selectedFeatures[id];
const showDropdown = hook.showDropdown;
const setShowDropdown = hook.setShowDropdown;
const allFeatures = hook.allFeatures;
const setSelectedFeatures = hook.setSelectedFeatures;
const setAllFeatures = hook.setAllFeatures;
const dropdownRef = hook.dropdownRef;
const selectedFeatures = hook.selectedFeatures;
const toggleFeature = (featureId: string) => { const toggleFeature = (featureId: string) => {
setSelectedFeatures(prev => { const newSet = new Set(selectedFeat);
const newSet = new Set(prev); if (newSet.has(featureId)) {
if (newSet.has(featureId)) { newSet.delete(featureId);
newSet.delete(featureId); } else {
} else { newSet.add(featureId);
newSet.add(featureId); }
} setSelectedFeatures(id, newSet);
return newSet;
});
}; };
const handleKeyDown = (e: React.KeyboardEvent) => { const handleKeyDown = (e: React.KeyboardEvent) => {
@@ -451,17 +470,17 @@ function RegisterComponent() {
e.preventDefault(); e.preventDefault();
addNewFeature(); addNewFeature();
} else if (e.key === 'Escape') { } else if (e.key === 'Escape') {
setShowDropdown(false); setShowDropdown(id, false);
} }
}; };
const addNewFeature = (e?: React.FormEvent) => { const addNewFeature = (e?: React.FormEvent) => {
if (e) e.preventDefault(); if (e) e.preventDefault();
const toAdd = newFeature.trim(); const toAdd = newFeat.trim();
if (!toAdd) return; if (!toAdd) return;
// Check if interest already exists (case-insensitive) // Check if interest already exists (case-insensitive)
const existingFeature = allFeatures.find( const existingFeature = allFeat.find(
i => i.name.toLowerCase() === toAdd.toLowerCase() i => i.name.toLowerCase() === toAdd.toLowerCase()
); );
@@ -471,36 +490,36 @@ function RegisterComponent() {
} else { } else {
// Add new feature // Add new feature
const newObj = {id: `new-${Date.now()}`, name: toAdd}; const newObj = {id: `new-${Date.now()}`, name: toAdd};
setAllFeatures(prev => [...prev, newObj]); setAllFeatures(id, [...allFeat, newObj]);
setSelectedFeatures(prev => new Set(prev).add(newObj.id)); setSelectedFeatures(id, new Set(selectedFeat).add(newObj.id));
} }
setNewFeature(''); setNewFeature(id, '');
setShowDropdown(false); setShowDropdown(id, false);
}; };
return <> return <>
<div className="relative" ref={dropdownRef}> <div className="relative" ref={refDropdown.current[id]}>
<label className={headingStyle}> <label className={headingStyle}>
{title} {title}
</label> </label>
{content && getDetails(id, 'Guidance', content)} {content && getDetails(id, 'Tips', content)}
<div className="relative"> <div className="relative">
<div className="flex items-center border border-gray-300 rounded-md shadow-sm"> <div className="flex items-center border border-gray-300 rounded-md shadow-sm">
<input <input
type="text" type="text"
value={newFeature} value={newFeat}
maxLength={100} maxLength={100}
onChange={(e) => setNewFeature(e.target.value)} onChange={(e) => setNewFeature(id, e.target.value)}
onFocus={() => setShowDropdown(true)} onFocus={() => setShowDropdown(id, true)}
onKeyDown={handleKeyDown} onKeyDown={handleKeyDown}
className="flex-1 min-w-0 block w-full px-3 py-2 rounded-l-md border-0 focus:ring-blue-500 focus:border-blue-500 sm:text-sm" className="flex-1 min-w-0 block w-full px-3 py-2 rounded-l-md border-0 focus:ring-blue-500 focus:border-blue-500 sm:text-sm"
placeholder={allowAdd ? "Type to search or add" : "Type to search"} placeholder={allowAdd ? "Search or add your own" : "Type to search"}
/> />
<button <button
type="button" type="button"
onClick={() => setShowDropdown(!showDropdown)} onClick={() => setShowDropdown(id, !showDropdown[id])}
className="px-3 py-2 border-l border-gray-300 text-gray-500 hover:bg-gray-100 dark:hover:bg-gray-700 focus:outline-none focus:ring-1 focus:ring-blue-500 focus:border-blue-500 " className="px-3 py-2 border-l border-gray-300 text-gray-500 hover:bg-gray-100 dark:hover:bg-gray-700 focus:outline-none focus:ring-1 focus:ring-blue-500 focus:border-blue-500 "
> >
<svg className="h-5 w-5 text-gray-400" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" <svg className="h-5 w-5 text-gray-400" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20"
@@ -514,7 +533,7 @@ function RegisterComponent() {
<button <button
type="button" type="button"
onClick={addNewFeature} onClick={addNewFeature}
disabled={!newFeature.trim()} disabled={!newFeat.trim()}
className="inline-flex items-center px-4 py-2 border border-transparent text-sm font-medium rounded-r-md text-white bg-blue-600 hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 disabled:opacity-50 disabled:cursor-not-allowed" className="inline-flex items-center px-4 py-2 border border-transparent text-sm font-medium rounded-r-md text-white bg-blue-600 hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 disabled:opacity-50 disabled:cursor-not-allowed"
> >
Add Add
@@ -522,12 +541,12 @@ function RegisterComponent() {
} }
</div> </div>
{(showDropdown || newFeature) && ( {(showDropdown[id] || newFeat) && (
<div <div
className="absolute z-10 mt-1 w-full bg-white dark:bg-gray-800 shadow-lg max-h-60 rounded-md py-1 text-base ring-1 ring-black ring-opacity-5 overflow-auto focus:outline-none sm:text-sm"> className="absolute z-10 mt-1 w-full bg-white dark:bg-gray-800 shadow-lg max-h-60 rounded-md py-1 text-base ring-1 ring-black ring-opacity-5 overflow-auto focus:outline-none sm:text-sm">
{/* New interest option */} {/* New interest option */}
{allowAdd && newFeature && !allFeatures.some(i => {allowAdd && newFeat && !allFeat.some(i =>
i.name.toLowerCase() === newFeature.toLowerCase() i.name.toLowerCase() === newFeat.toLowerCase()
) && ( ) && (
<div <div
className=" cursor-default select-none relative py-2 pl-3 pr-9 hover:bg-blue-50 dark:hover:bg-gray-700" className=" cursor-default select-none relative py-2 pl-3 pr-9 hover:bg-blue-50 dark:hover:bg-gray-700"
@@ -535,36 +554,36 @@ function RegisterComponent() {
> >
<div className="flex items-center"> <div className="flex items-center">
<span className="font-normal ml-3 block truncate"> <span className="font-normal ml-3 block truncate">
Add "{newFeature}" Add "{newFeat}"
</span> </span>
</div> </div>
</div> </div>
)} )}
{/* Filtered interests */} {/* Filtered features */}
{allFeatures {(!allowAdd || newFeat.length >= 3) && allFeat
.filter(interest => .filter(feature =>
interest.name.toLowerCase().includes(newFeature.toLowerCase()) feature.name.toLowerCase().includes(newFeat.toLowerCase())
) )
.map((interest) => ( .map((feature) => (
<div <div
key={interest.id} key={feature.id}
className="cursor-default select-none relative py-2 pl-3 pr-9 hover:bg-blue-50 dark:hover:bg-gray-700" className="cursor-default select-none relative py-2 pl-3 pr-9 hover:bg-blue-50 dark:hover:bg-gray-700"
onClick={() => { onClick={() => {
toggleFeature(interest.id); toggleFeature(feature.id);
setNewFeature(''); setNewFeature(id, '');
}} }}
> >
<div className="flex items-center"> <div className="flex items-center">
<input <input
type="checkbox" type="checkbox"
className="h-4 w-4 text-blue-600 focus:ring-blue-500 border-gray-300 rounded " className="h-4 w-4 text-blue-600 focus:ring-blue-500 border-gray-300 rounded "
checked={selectedFeatures.has(interest.id)} checked={selectedFeat.has(feature.id)}
onChange={(e) => { onChange={(e) => {
e.stopPropagation() e.stopPropagation()
}} }}
/> />
<span className="font-normal ml-3 block truncate">{interest.name}</span> <span className="font-normal ml-3 block truncate">{feature.name}</span>
</div> </div>
</div> </div>
))} ))}
@@ -574,8 +593,8 @@ function RegisterComponent() {
{/* Selected interests */} {/* Selected interests */}
<div className="flex flex-wrap gap-2 mt-3"> <div className="flex flex-wrap gap-2 mt-3">
{Array.from(selectedFeatures).map(featureId => { {Array.from(selectedFeat).map(featureId => {
const interest = allFeatures.find(i => i.id === featureId); const interest = allFeat.find(i => i.id === featureId);
if (!interest) return null; if (!interest) return null;
return ( return (
<span <span
@@ -605,24 +624,6 @@ function RegisterComponent() {
</> </>
} }
function errorBlock() {
return <div className="bg-red-50 border-l-4 border-red-400 p-4">
<div className="flex">
<div className="flex-shrink-0">
<svg className="h-5 w-5 text-red-400" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20"
fill="currentColor">
<path fillRule="evenodd"
d="M10 18a8 8 0 100-16 8 8 0 000 16zM8.707 7.293a1 1 0 00-1.414 1.414L8.586 10l-1.293 1.293a1 1 0 101.414 1.414L10 11.414l1.293 1.293a1 1 0 001.414-1.414L11.414 10l1.293-1.293a1 1 0 00-1.414-1.414L10 8.586 8.707 7.293z"
clipRule="evenodd"/>
</svg>
</div>
<div className="ml-3">
<p className="text-sm text-red-700">{error}</p>
</div>
</div>
</div>
}
return ( return (
<div className="min-h-screen flex items-center justify-center py-12 px-4 sm:px-6 lg:px-8"> <div className="min-h-screen flex items-center justify-center py-12 px-4 sm:px-6 lg:px-8">
<div className="max-w-3xl w-full space-y-8"> <div className="max-w-3xl w-full space-y-8">
@@ -632,7 +633,7 @@ function RegisterComponent() {
</h2> </h2>
</div> </div>
{error && errorBlock()} {error && errorBlock(error)}
<div className="flex justify-center mb-6"> <div className="flex justify-center mb-6">
<div className="relative"> <div className="relative">
@@ -747,26 +748,32 @@ function RegisterComponent() {
/> />
</div> </div>
{getDropdown(dropdownConfig[0])} {dropdownConfig.map((v, i) => (
{getDropdown(dropdownConfig[1])} <React.Fragment key={i}>{getDropdown(v)}</React.Fragment>
{getDropdown(dropdownConfig[2])} ))}
{getDropdown(dropdownConfig[3])}
<div> <div>
<label htmlFor="introversion" className={headingStyle}> <label htmlFor="introversion" className={headingStyle}>Social Style</label>
Introversion (0) - Extroversion (100) <div className="flex items-center w-full max-w-xl gap-4">
</label> <span className={headingStyle}>Introverted</span>
<input <Slider
id="introversion" value={introversion ? 100 - introversion : 50}
name="introversion" onChange={(e, value) => setIntroversion(100 - Number(value))}
type="number" valueLabelDisplay="auto"
min="0" min={0}
max="100" max={100}
value={introversion ? 100 - introversion : ''} sx={{
onChange={(e) => setIntroversion(100 - Number(e.target.value))} color: '#3B82F6',
className="appearance-none rounded-md relative block w-full px-3 py-2 border border-gray-300 placeholder-gray-500 focus:outline-none focus:ring-blue-500 focus:border-blue-500 focus:z-10 sm:text-sm" '& .MuiSlider-valueLabel': {
// placeholder="" backgroundColor: '#3B82F6',
/> color: '#fff',
},
}}
className="flex-1"
/>
<span className={headingStyle}>Extroverted</span>
</div>
</div> </div>
{/*<div>*/} {/*<div>*/}
@@ -789,39 +796,38 @@ function RegisterComponent() {
{/* </select>*/} {/* </select>*/}
{/*</div>*/} {/*</div>*/}
<div> {/*<div>*/}
<label htmlFor="conflictStyle" className={headingStyle}> {/* <label htmlFor="conflictStyle" className={headingStyle}>*/}
Conflict Style {/* Conflict Style*/}
</label> {/* </label>*/}
<select {/* <select*/}
id="conflictStyle" {/* id="conflictStyle"*/}
name="conflictStyle" {/* name="conflictStyle"*/}
value={conflictStyle || ''} {/* value={conflictStyle || ''}*/}
onChange={(e) => setConflictStyle(e.target.value as ConflictStyle)} {/* onChange={(e) => setConflictStyle(e.target.value as ConflictStyle)}*/}
className="appearance-none rounded-md relative block w-full px-3 py-2 border border-gray-300 placeholder-gray-500 focus:outline-none focus:ring-blue-500 focus:border-blue-500 focus:z-10 sm:text-sm" {/* className="appearance-none rounded-md relative block w-full px-3 py-2 border border-gray-300 placeholder-gray-500 focus:outline-none focus:ring-blue-500 focus:border-blue-500 focus:z-10 sm:text-sm"*/}
> {/* >*/}
<option value="">Select your conflict style</option> {/* <option value="">Select your conflict style</option>*/}
{conflictOptions.map((style) => ( {/* {conflictOptions.map((style) => (*/}
<option key={style} value={style}> {/* <option key={style} value={style}>*/}
{style} {/* {style}*/}
</option> {/* </option>*/}
))} {/* ))}*/}
</select> {/* </select>*/}
</div> {/*</div>*/}
<div className="max-w-3xl w-full"> <div className="max-w-3xl w-full">
<label htmlFor="description" className={headingStyle}> <label htmlFor="description" className={headingStyle}>
About You About You
<p className="text-sm italic"> <p className="text-sm italic">
Feel free to include any relevant links (dating / friends docs, etc.), but consider copy-pasting People search by keyword, so include as many words that feel true to you!
the content here so that people can find you by keyword search.
</p> </p>
</label> </label>
{getDetails( {getDetails(
'description', 'description',
'Guidance', 'Tips',
<> <>
<p className="mt-2">To the extent that you are comfortable sharing, consider writing about:</p> <p className="mt-2">Consider adding:</p>
<ul className="list-disc pl-5 space-y-1"> <ul className="list-disc pl-5 space-y-1">
<li>Your interests and what you're looking for: type of connection, activities to do, etc.</li> <li>Your interests and what you're looking for: type of connection, activities to do, etc.</li>
<li>Your availability and timezone</li> <li>Your availability and timezone</li>
@@ -963,7 +969,7 @@ function RegisterComponent() {
)} )}
</div> </div>
{error && errorBlock()} {error && errorBlock(error)}
<div> <div>
<button <button

View File

@@ -1,6 +1,7 @@
'use client'; 'use client';
import {DropdownKey} from "@/app/profiles/page"; import { DropdownKey } from "@/lib/client/schema";
type DropdownProps = { type DropdownProps = {
id: DropdownKey id: DropdownKey

View File

@@ -66,3 +66,20 @@ ul {
margin-top: 0.5rem; margin-top: 0.5rem;
} }
@keyframes spin {
to {
transform: rotate(360deg);
}
}
.spinner {
width: 48px; /* 12 * 4px */
height: 48px;
border: 4px solid #d1d5db; /* gray-300 */
border-top-color: #1f2937; /* gray-800 */
border-radius: 50%;
animation: spin 1s linear infinite;
}

View File

@@ -6,8 +6,8 @@ import Header from "@/app/Header";
import Providers from "@/app/providers"; import Providers from "@/app/providers";
export const metadata: Metadata = { export const metadata: Metadata = {
title: "IntentionalBond", title: "Compass",
description: "A bonding platform for rational thinkers", description: "A social platform to form intentional bonds",
}; };
@@ -30,7 +30,7 @@ export default function RootLayout(
<footer className="p-6 text-center text-gray-500"> <footer className="p-6 text-center text-gray-500">
<div className="mb-2"> <div className="mb-2">
<a <a
href="https://github.com/BayesBond/BayesBond" href="https://github.com/CompassConnections/Compass"
target="_blank" target="_blank"
rel="noopener noreferrer" rel="noopener noreferrer"
className="inline-flex items-center text-gray-500 hover:text-gray-700 transition" className="inline-flex items-center text-gray-500 hover:text-gray-700 transition"
@@ -41,7 +41,7 @@ export default function RootLayout(
View on GitHub View on GitHub
</a> </a>
</div> </div>
<div>© {new Date().getFullYear()} IntentionalBond. All rights reserved.</div> <div>© {new Date().getFullYear()} Compass. All rights reserved.</div>
</footer> </footer>
</div> </div>
</ThemeProvider> </ThemeProvider>

View File

@@ -1,6 +1,6 @@
'use client'; 'use client';
import {aColor, supportEmail} from "@/lib/client/constants"; import Image from 'next/image';
export default function PrivacyPage() { export default function PrivacyPage() {
return ( return (
@@ -169,7 +169,7 @@ export default function PrivacyPage() {
<p>If the platform is very selective, there will be only a couple people with very aligned values. Their well-being will increase a lot, but there are only a few of them. So there is not much overall benefit for society.</p> <p>If the platform is very selective, there will be only a couple people with very aligned values. Their well-being will increase a lot, but there are only a few of them. So there is not much overall benefit for society.</p>
<p>If the platform is fully open (everyone can join), there are two possibilities depending on the connection mechanism. If the list of members is opaque and poorly searchable, as in traditional dating apps, it becomes very unlikely to find value-aligned people. Hence, each individuals increase in well-being is negligible, even though there are plenty of people, and there is not much overall benefit for society either.</p> <p>If the platform is fully open (everyone can join), there are two possibilities depending on the connection mechanism. If the list of members is opaque and poorly searchable, as in traditional dating apps, it becomes very unlikely to find value-aligned people. Hence, each individuals increase in well-being is negligible, even though there are plenty of people, and there is not much overall benefit for society either.</p>
<p>Qualitatively, the figure below illustrates this trade-off between selectivity and openness for poorly searchable (i.e., Tinder-type) platforms. The quality is the individual increase in well-being, which increases as the platform selects for people strictly following the core values. The quantity is simply the number of users. The platforms overall benefit (read, increase in total well-being) is then the product of quantity and qualityreaching a maximum with a non-extreme selectivity.</p> <p>Qualitatively, the figure below illustrates this trade-off between selectivity and openness for poorly searchable (i.e., Tinder-type) platforms. The quality is the individual increase in well-being, which increases as the platform selects for people strictly following the core values. The quantity is simply the number of users. The platforms overall benefit (read, increase in total well-being) is then the product of quantity and qualityreaching a maximum with a non-extreme selectivity.</p>
<p><img src="https://martinbraquet.com/wp-content/uploads/rational_qualitative.png" alt="" /></p> <p><Image src="https://martinbraquet.com/wp-content/uploads/rational_qualitative.png" alt="" /></p>
<p>The second possibility appears when the members are fully visible and searchable by anyone. In that case, each member can filter and meaningfully engage with the few people aligning with their values. Each individuals increase in well-being becomes important, and there are plenty of people. So, a fully open and searchable platform may bring a lot of overall benefit for society.</p> <p>The second possibility appears when the members are fully visible and searchable by anyone. In that case, each member can filter and meaningfully engage with the few people aligning with their values. Each individuals increase in well-being becomes important, and there are plenty of people. So, a fully open and searchable platform may bring a lot of overall benefit for society.</p>
<p>Of course, in practice, there will always be interferences within big communities. And, more importantly, a larger platform requires more resources (i.e., funding) and moderation. Thats why the focus of this article is on a specific community for now. Not only do I identify with the rational / intellectual community, but it is also composed of members who are much more likely to contribute (especially on the tech side), making it a very convenient community. But if it creates much greater good, I think it would be worth considering extending the platform at some point in the far futureprovided that it doesnt negatively dilute the community or create brand identity confusion. More than just creating a higher good, a larger user base means economy of scale: donations may scale proportionally while expense per user would diminishmaking the platform more likely to survive financially.</p> <p>Of course, in practice, there will always be interferences within big communities. And, more importantly, a larger platform requires more resources (i.e., funding) and moderation. Thats why the focus of this article is on a specific community for now. Not only do I identify with the rational / intellectual community, but it is also composed of members who are much more likely to contribute (especially on the tech side), making it a very convenient community. But if it creates much greater good, I think it would be worth considering extending the platform at some point in the far futureprovided that it doesnt negatively dilute the community or create brand identity confusion. More than just creating a higher good, a larger user base means economy of scale: donations may scale proportionally while expense per user would diminishmaking the platform more likely to survive financially.</p>
<h3 id="viability">Viability</h3> <h3 id="viability">Viability</h3>

View File

@@ -0,0 +1,23 @@
"use client";
import {Suspense} from "react";
import OnboardingForm from "@/components/onboarding";
export default function RegisterPage() {
return (
<Suspense fallback={<div></div>}>
<RegisterComponent/>
</Suspense>
);
}
function RegisterComponent() {
return (
<div className="min-h-screen flex items-center justify-center py-12 px-4 sm:px-6 lg:px-8">
<OnboardingForm/>
</div>
);
}

125
_old/app/page.tsx Normal file
View File

@@ -0,0 +1,125 @@
'use client';
import ProfilePage from "@/app/profiles/page";
import Link from "next/link";
import {useEffect} from 'react';
import {useSession} from "next-auth/react"; // ← Add this line here
export const dynamic = "force-dynamic"; // This disables SSG and ISR
export default function HomePage() {
const {data: session} = useSession();
const userId = session?.user?.id
console.log("session:", userId)
const fontStyle = "transition px-5 py-3 text-3xl font-medium xs:text-sm"
useEffect(() => {
const text = "Search.";
const typewriter = document.getElementById("typewriter");
let i = 0;
let timeoutId: any;
let intervalId;
// Clear any existing content
if (typewriter) typewriter.textContent = ""
function typeWriter() {
if (i < text.length && typewriter) {
typewriter.textContent = text.substring(0, i + 1);
i++;
timeoutId = setTimeout(typeWriter, 150);
}
}
// Start typing after delay
intervalId = setTimeout(() => typeWriter(), 500);
// Cleanup function - this runs when component unmounts
return () => {
clearTimeout(timeoutId);
clearTimeout(intervalId);
if (typewriter) typewriter.textContent = "Search."
};
}, []);
return (
<main className="min-h-screen flex flex-col">
{/* Header */}
{/*<header className="flex justify-between items-center p-2 max-w-6xl mx-auto w-full">*/}
{/* <a */}
{/* href="https://github.com/CompassConnections/Compass" */}
{/* target="_blank" */}
{/* rel="noopener noreferrer"*/}
{/* className="text-gray-700 hover: transition"*/}
{/* >*/}
{/* <svg className="w-6 h-6" fill="currentColor" viewBox="0 0 24 24" aria-hidden="true">*/}
{/* <path fillRule="evenodd" d="M12 2C6.477 2 2 6.484 2 12.017c0 4.425 2.865 8.18 6.839 9.504.5.092.682-.217.682-.483 0-.237-.008-.868-.013-1.703-2.782.605-3.369-1.343-3.369-1.343-.454-1.158-1.11-1.466-1.11-1.466-.908-.62.069-.608.069-.608 1.003.07 1.531 1.032 1.531 1.032.892 1.53 2.341 1.088 2.91.832.092-.647.35-1.088.636-1.338-2.22-.253-4.555-1.11-4.555-4.943 0-1.091.39-1.984 1.029-2.683-.103-.253-.446-1.27.098-2.647 0 0 .84-.269 2.75 1.026A9.564 9.564 0 0112 6.844c.85.004 1.705.115 2.504.337 1.909-1.296 2.747-1.027 2.747-1.027.546 1.379.202 2.398.1 2.651.64.699 1.028 1.595 1.028 2.688 0 3.842-2.339 4.687-4.566 4.935.359.309.678.919.678 1.852 0 1.336-.012 2.415-.012 2.743 0 .267.18.578.688.48A10.019 10.019 0 0022 12.017C22 6.484 17.522 2 12 2z" clipRule="evenodd" />*/}
{/* </svg>*/}
{/* </a>*/}
{/*</header>*/}
{/* Hero Section */}
<section className="flex flex-col items-center justify-start flex-1 text-center px-4">
<div className="h-10"></div>
<h1
className="pt-12 pb-2 text-7xl md:text-8xl xs:text-6xl font-extrabold max-w-4xl leading-tight xl:whitespace-nowrap md:whitespace-nowrap ">
Don't Swipe. <span id="typewriter"></span><span id="cursor" className="animate-pulse">|</span>
</h1>
{/*<p className="mt-6 text-lg md:text-xl text-gray-400 max-w-2xl">*/}
{/* {"Tired of swiping? Search what you're looking for!"}*/}
{/*</p>*/}
{/* Spacer */}
<div className="h-10"></div>
{!userId && <div className="py-8">
<Link href="/register"
className={`${fontStyle} bg-gradient-to-r from-red-600 to-red-800 text-white rounded-full hover:from-red-700 hover:to-red-900`}>
Join Compass
</Link>
{/* Spacer */}
{/*<div className="h-16"></div>*/}
</div>}
{/* Why Compass Bar */}
<div className="w-full bg-gray-50 dark:bg-gray-900 py-8 mt-20">
<div className="max-w-6xl mx-auto px-4">
<div className="grid md:grid-cols-3 gap-8 text-center">
<div className="space-y-2">
<h3 className="text-lg font-bold">Radically Transparent</h3>
<p className="text-gray-600 dark:text-gray-400">
No algorithms. Every profile searchable.
</p>
</div>
<div className="space-y-2">
<h3 className="text-lg font-bold">Built for Depth</h3>
<p className="text-gray-600 dark:text-gray-400">
Filter by any keyword and what matters most.
</p>
</div>
<div className="space-y-2">
<h3 className="text-lg font-bold">Community Owned</h3>
<p className="text-gray-600 dark:text-gray-400">
Free forever. Built by users, for users.
</p>
</div>
</div>
</div>
</div>
{/* Spacer */}
{userId &&
<>
{/*<div className="h-20"></div>*/}
<div className=" w-full py-10">
<main className="min-h-screen flex flex-col">
<ProfilePage/>
</main>
</div>
</>
}
</section>
</main>
);
}

View File

@@ -2,9 +2,9 @@
import Link from 'next/link'; import Link from 'next/link';
import {usePathname, useRouter} from "next/navigation"; import {usePathname, useRouter} from "next/navigation";
import {getProfile} from "@/lib/client/profile"; import {Profile} from "@/lib/client/profile";
import {useEffect} from "react"; import {useEffect} from "react";
import {useSession} from "next-auth/react"; import {signOut, useSession} from "next-auth/react";
export default function ProfilePage() { export default function ProfilePage() {
const pathname = usePathname(); // Get the current route const pathname = usePathname(); // Get the current route
@@ -30,16 +30,22 @@ export default function ProfilePage() {
</div> </div>
<Link <Link
href={`/complete-profile?redirect=${encodeURIComponent(pathname)}`} href={`/complete-profile?redirect=${encodeURIComponent(pathname)}`}
className="inline-flex items-center px-4 py-2 border border-transparent text-sm font-medium rounded-md shadow-sm text-white bg-blue-600 hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500" className="mx-1 transition px-2 py-2 text-sm font-medium xs:text-xs bg-blue-500 text-white rounded-full hover:bg-blue-600 min-w-20"
> >
Edit Profile Edit Profile
</Link> </Link>
<button
onClick={() => signOut({callbackUrl: "/"})}
className="mx-1 transition px-2 py-2 text-sm font-medium xs:text-xs bg-red-500 text-white rounded-full hover:bg-red-600 min-w-20"
>
Sign Out
</button>
</div> </div>
) )
return ( return (
<div className="min-h-screen py-12 px-4 sm:px-6 lg:px-8"> <div className="min-h-screen py-12 px-4 sm:px-6 lg:px-8">
{getProfile('/api/profile', header)} {Profile('/api/profile', header)}
</div> </div>
) )
; ;

View File

@@ -1,11 +1,12 @@
'use client'; 'use client';
import {useEffect, useRef, useState} from 'react'; import React, {useEffect, useRef, useState} from 'react';
import {Gender} from "@prisma/client"; import {Gender} from "@prisma/client";
import Dropdown from "@/app/components/dropdown"; import Dropdown from "@/app/components/dropdown";
import Slider from '@mui/material/Slider'; import Slider from '@mui/material/Slider';
import {DropdownKey, RangeKey} from "@/app/profiles/page"; import {DropdownKey, Item, RangeKey} from "@/lib/client/schema";
import {capitalize} from "@/lib/format"; import {capitalize} from "@/lib/format";
import {fetchFeatures} from "@/lib/client/fetching";
interface FilterProps { interface FilterProps {
filters: { filters: {
@@ -27,10 +28,11 @@ interface FilterProps {
} }
export const dropdownConfig: { id: DropdownKey, name: string }[] = [ export const dropdownConfig: { id: DropdownKey, name: string }[] = [
{id: "connections", name: "Desired Connections"}, {id: "connections", name: "Connection Type"},
{id: "coreValues", name: "Core Values"}, {id: "coreValues", name: "Values"},
{id: "interests", name: "Core Interests"}, {id: "interests", name: "Interests"},
{id: "causeAreas", name: "Cause Areas"}, {id: "books", name: "Works"},
// {id: "causeAreas", name: "Cause Areas"},
] ]
export const rangeConfig: { id: RangeKey, name: string, min: number, max: number }[] = [ export const rangeConfig: { id: RangeKey, name: string, min: number, max: number }[] = [
@@ -39,59 +41,87 @@ export const rangeConfig: { id: RangeKey, name: string, min: number, max: number
] ]
export function ProfileFilters({filters, onFilterChange, onShowFilters, onToggleFilter, onReset}: FilterProps) { export function ProfileFilters({filters, onFilterChange, onShowFilters, onToggleFilter, onReset}: FilterProps) {
interface Item {
id: DropdownKey;
name: string;
}
const [showFilters, setShowFilters] = useState(true); const [showFilters, setShowFilters] = useState(true);
const dropDownStates = Object.fromEntries(dropdownConfig.map(({id}) => { // Initialize state for all dropdowns as an object with keys from dropdownConfig ids
const [all, setAll] = useState<Item[]>([]); const [optionsDropdown, setOptionsDropdown] = useState(() =>
const [selected, setSelected] = useState<Set<string>>(new Set()); Object.fromEntries(dropdownConfig.map(({id}) => [id, [] as Item[]]))
const [newValue, setNewValue] = useState(''); );
const ref = useRef<HTMLDivElement>(null); const setOptionsDropdownId = (id: string, value: any) => {
const [show, setShow] = useState(false); setOptionsDropdown((prev) => ({...prev, [id]: value}));
};
const [selectedDropdown, setSelectedDropdown] = useState(() =>
Object.fromEntries(dropdownConfig.map(({id}) => [id, new Set<string>()]))
);
const [newDropdown, setNewDropdown] = useState(() =>
Object.fromEntries(dropdownConfig.map(({id}) => [id, '']))
);
const [showDropdown, setShowDropdown] = useState(() =>
Object.fromEntries(dropdownConfig.map(({id}) => [id, false]))
);
// refs cannot be in state; create refs map outside state
const refDropdown = useRef<any>(
Object.fromEntries(dropdownConfig.map(({id}) => [id, React.createRef<HTMLDivElement>()]))
);
return [id, {
options: {value: all, set: setAll},
selected: {value: selected, set: setSelected},
new: {value: newValue, set: setNewValue},
ref: ref,
show: {value: show, set: setShow},
}]
}))
console.log(dropDownStates)
useEffect(() => { useEffect(() => {
async function fetchOptions() { fetchFeatures(setOptionsDropdownId);
try { }, []);
const res = await fetch('/api/interests');
if (res.ok) { useEffect(() => {
const data = await res.json() as Record<string, Item[]>; console.log('selectedDropdown changed:', selectedDropdown);
console.log(data); }, [selectedDropdown]);
for (const [id, values] of Object.entries(data)) {
console.log(id) useEffect(() => {
dropDownStates[id].options.set(values); const params = new URLSearchParams(window.location.search);
console.log('optionsDropdown changed:', optionsDropdown, params);
for (const [key, value] of params.entries()) {
let v: any = value
if (key === 'minAge') {
setMinRange({...minRange, age: v});
} else if (key === 'maxAge') {
setMaxRange({...maxRange, age: v});
} else if (key === 'minIntroversion') {
setMinRange({...minRange, introversion: v});
} else if (key === 'maxIntroversion') {
setMaxRange({...maxRange, introversion: v});
} else if (['interests', 'coreValues', 'causeAreas', 'connections'].includes(key)) {
v = v.split(",").filter(Boolean) || []
console.log(v)
for (const n of v) {
const option = optionsDropdown[key].find(i => i.name === n);
if (option) {
console.log(option);
setSelectedDropdown(prev => {
const newSet = new Set(prev[key]);
newSet.add(option.id);
return {...prev, [key]: newSet};
});
} }
} }
} catch (error) {
console.error('Error loading options:', error);
} }
} }
}, [optionsDropdown]);
fetchOptions(); useEffect(() => {
// Close dropdown when clicking outside // Close dropdown when clicking outside
const handleClickOutside = (event: MouseEvent) => { const handleClickOutside = (event: MouseEvent) => {
for (const id in dropDownStates) { for (const id in showDropdown) {
const dropdown = dropDownStates[id]; const ref = refDropdown.current[id];
const ref = dropdown.ref;
if ( if (
ref?.current && ref?.current &&
!ref.current.contains(event.target as Node) !ref.current.contains(event.target as Node)
) { ) {
dropdown.show?.set?.(false); // Defensive chaining setShowDropdown(prev => ({...prev, [id]: false}));
} }
} }
}; };
@@ -101,61 +131,61 @@ export function ProfileFilters({filters, onFilterChange, onShowFilters, onToggle
return () => { return () => {
document.removeEventListener('mousedown', handleClickOutside); document.removeEventListener('mousedown', handleClickOutside);
}; };
}, []); }, [showDropdown]);
const toggle = (id: DropdownKey, optionId: string) => { const toggle = (id: DropdownKey, optionId: string) => {
dropDownStates[id].selected.set(prev => { setSelectedDropdown(prev => {
const newSet = new Set(prev); const newSet = new Set(prev[id]);
if (newSet.has(optionId)) { if (newSet.has(optionId)) {
newSet.delete(optionId); newSet.delete(optionId);
} else { } else {
newSet.add(optionId); newSet.add(optionId);
} }
return newSet; // console.log(newSet);
return {...prev, [id]: newSet};
}); });
}; };
const handleKeyDown = (id: DropdownKey, key: string) => { const handleKeyDown = (id: DropdownKey, key: string) => {
if (key === 'Escape') dropDownStates[id].show.set(false); if (key === 'Escape') setShowDropdown(prev => ({...prev, [id]: false}));
}; };
const handleChange = (id: DropdownKey, e: string) => { const handleChange = (id: DropdownKey, e: string) => {
dropDownStates[id].new.set(e); setNewDropdown(prev => ({...prev, [id]: e}));
} }
const handleFocus = (id: DropdownKey) => { const handleFocus = (id: DropdownKey) => {
dropDownStates[id].show.set(true); setShowDropdown(prev => ({...prev, [id]: true}))
} }
const handleClick = (id: DropdownKey) => { const handleClick = (id: DropdownKey) => {
const shown = dropDownStates[id].show.value; setShowDropdown(prev => ({...prev, [id]: !showDropdown[id]}))
dropDownStates[id].show.set(!shown);
} }
function getDrowDown(id: DropdownKey, name: string) { function getDrowDown(id: DropdownKey, name: string) {
return ( return (
<div key={id + '.div'}> <div key={id + '.div'}>
<div className="relative" ref={dropDownStates[id].ref}> <div className="relative" ref={refDropdown.current[id]}>
<label className="block text-sm font-medium text-gray-700 dark:text-white mb-2"> <label className="block text-sm font-medium text-gray-700 dark:text-white mb-2">
{name} {name}
</label> </label>
<Dropdown <Dropdown
key={id} key={id}
id={id} id={id}
value={dropDownStates[id].new.value} value={newDropdown[id]}
onChange={handleChange} onChange={handleChange}
onFocus={handleFocus} onFocus={handleFocus}
onKeyDown={handleKeyDown} onKeyDown={handleKeyDown}
onClick={handleClick} onClick={handleClick}
/> />
{(dropDownStates[id].show.value) && ( {(showDropdown[id]) && (
<div <div
className="absolute z-10 mt-1 w-full bg-white dark:bg-gray-900 shadow-lg max-h-60 rounded-md py-1 text-base ring-1 ring-black dark:ring-white ring-opacity-5 overflow-auto focus:outline-none sm:text-sm"> className="absolute z-10 mt-1 w-full bg-white dark:bg-gray-900 shadow-lg max-h-60 rounded-md py-1 text-base ring-1 ring-black dark:ring-white ring-opacity-5 overflow-auto focus:outline-none sm:text-sm">
{dropDownStates[id].options.value {optionsDropdown[id]
.filter(v => v.name.toLowerCase().includes(dropDownStates[id].new.value.toLowerCase())) .filter(v => v.name.toLowerCase().includes(newDropdown[id].toLowerCase()))
.map((v) => ( .map((v) => (
<div <div
key={v.id} key={v.id}
@@ -169,7 +199,7 @@ export function ProfileFilters({filters, onFilterChange, onShowFilters, onToggle
<input <input
type="checkbox" type="checkbox"
className="h-4 w-4 text-blue-600 focus:ring-blue-500 border-gray-300 rounded" className="h-4 w-4 text-blue-600 focus:ring-blue-500 border-gray-300 rounded"
checked={dropDownStates[id].selected.value.has(v.id)} checked={selectedDropdown[id].has(v.id)}
onChange={() => { onChange={() => {
}} }}
onClick={(e) => e.stopPropagation()} onClick={(e) => e.stopPropagation()}
@@ -184,8 +214,8 @@ export function ProfileFilters({filters, onFilterChange, onShowFilters, onToggle
)} )}
</div> </div>
<div className="flex flex-wrap gap-2 mt-3"> <div className="flex flex-wrap gap-2 mt-3">
{Array.from(dropDownStates[id].selected.value).map(vId => { {Array.from(selectedDropdown[id]).map(vId => {
const value = dropDownStates[id].options.value.find(i => i.id === vId); const value = optionsDropdown[id].find(i => i.id === vId);
if (!value) return null; if (!value) return null;
return ( return (
<span <span
@@ -223,50 +253,50 @@ export function ProfileFilters({filters, onFilterChange, onShowFilters, onToggle
max: number; max: number;
} }
const rangeStates = Object.fromEntries(rangeConfig.map(({id}) => { const [minRange, setMinRange] = useState(() =>
const [minVal, setMinVal] = useState<number | undefined>(undefined); Object.fromEntries(rangeConfig.map(({id}) => [id, undefined]))
const [maxVal, setMaxVal] = useState<number | undefined>(undefined); );
return [id, { const [maxRange, setMaxRange] = useState(() =>
minVal, Object.fromEntries(rangeConfig.map(({id}) => [id, undefined]))
maxVal, );
setMinVal,
setMaxVal,
}];
}))
function getSlider({id, name, min, max}: Range) { function getSlider({id, name, min, max}: Range, showSlider: boolean = true) {
const minStr = 'min' + capitalize(id); const minStr = 'min' + capitalize(id);
const maxStr = 'max' + capitalize(id); const maxStr = 'max' + capitalize(id);
const {minVal, maxVal, setMinVal, setMaxVal} = rangeStates[id]; const minVal = minRange[id];
const maxVal = maxRange[id];
const setMinVal = (v: any) => setMinRange({...minRange, [id]: v});
const setMaxVal = (v: any) => setMaxRange({...maxRange, [id]: v});
return ( return (
<div key={id + '.div'}> <div key={id + '.div'}>
<div className="w-full px-2"> {showSlider &&
<label className="block text-sm font-medium text-gray-700 dark:text-white mb-2">{name}</label> <div className="w-full px-2">
<Slider <label className="block text-sm font-medium text-gray-700 dark:text-white mb-2">{name}</label>
value={[minVal || min, maxVal || max]} <Slider
onChange={(e, value) => { value={[minVal || min, maxVal || max]}
let [_min, _max] = value; onChange={(e, value) => {
setMinVal((_min || min) > min ? _min : undefined); let [_min, _max] = value;
setMaxVal((_max || max) < max ? _max : undefined); setMinVal((_min || min) > min ? _min : undefined);
}} setMaxVal((_max || max) < max ? _max : undefined);
onChangeCommitted={(e, value) => { }}
let [_min, _max] = value; onChangeCommitted={(e, value) => {
onFilterChange(minStr, (_min || min) > min ? _min : undefined); let [_min, _max] = value;
onFilterChange(maxStr, (_max || max) < max ? _max : undefined); onFilterChange(minStr, (_min || min) > min ? _min : undefined);
}} onFilterChange(maxStr, (_max || max) < max ? _max : undefined);
valueLabelDisplay="auto" }}
min={min} valueLabelDisplay="auto"
max={max} min={min}
sx={{ max={max}
color: '#3B82F6', sx={{
'& .MuiSlider-valueLabel': { color: '#3B82F6',
backgroundColor: '#3B82F6', '& .MuiSlider-valueLabel': {
color: '#fff', backgroundColor: '#3B82F6',
}, color: '#fff',
}} },
/> }}
</div> />
</div>}
<div className="grid grid-cols-2 gap-4"> <div className="grid grid-cols-2 gap-4">
<div> <div>
{/*<label className="block text-sm font-medium text-gray-700 dark:text-white mb-1">Min Age</label>*/} {/*<label className="block text-sm font-medium text-gray-700 dark:text-white mb-1">Min Age</label>*/}
@@ -347,7 +377,7 @@ export function ProfileFilters({filters, onFilterChange, onShowFilters, onToggle
</div> </div>
</div> </div>
{getSlider(rangeConfig[0])} {getSlider(rangeConfig[0], false)}
{dropdownConfig.map(({id, name}) => getDrowDown(id, name))} {dropdownConfig.map(({id, name}) => getDrowDown(id, name))}
{getSlider(rangeConfig[1])} {getSlider(rangeConfig[1])}
@@ -376,13 +406,15 @@ export function ProfileFilters({filters, onFilterChange, onShowFilters, onToggle
<button <button
onClick={() => { onClick={() => {
onReset(); onReset();
Object.values(dropDownStates).map((v) => { setSelectedDropdown(() =>
v.selected.set(new Set()); Object.fromEntries(dropdownConfig.map(({id}) => [id, new Set<string>()]))
}); );
Object.values(rangeStates).map((v) => { setMinRange(() =>
v.setMaxVal(undefined); Object.fromEntries(rangeConfig.map(({id}) => [id, undefined]))
v.setMinVal(undefined); );
}); setMaxRange(() =>
Object.fromEntries(rangeConfig.map(({id}) => [id, undefined]))
);
}} }}
className="px-4 py-2 text-sm text-gray-600 dark:text-white hover:text-gray-800" className="px-4 py-2 text-sm text-gray-600 dark:text-white hover:text-gray-800"
> >

View File

@@ -1,7 +1,7 @@
'use client'; 'use client';
import {useParams} from "next/navigation"; import {useParams} from "next/navigation";
import {getProfile} from "@/lib/client/profile"; import {Profile} from "@/lib/client/profile";
export const dynamic = "force-dynamic"; // This disables SSG and ISR export const dynamic = "force-dynamic"; // This disables SSG and ISR
@@ -11,7 +11,7 @@ export default function Post() {
return ( return (
<div className="min-h-screen py-12 px-4 sm:px-6 lg:px-8"> <div className="min-h-screen py-12 px-4 sm:px-6 lg:px-8">
{getProfile(`/api/profiles/${id}`)} {Profile(`/api/profiles/${id}`)}
</div> </div>
); );
} }

View File

@@ -1,10 +1,11 @@
'use client'; 'use client';
import Link from "next/link"; import Link from "next/link";
import {useCallback, useEffect, useState} from "react"; import React, {useCallback, useEffect, useState} from "react";
import LoadingSpinner from "@/lib/client/LoadingSpinner"; import {DropdownKey, ProfileData} from "@/lib/client/schema";
import {ProfileData} from "@/lib/client/schema";
import {dropdownConfig, ProfileFilters} from "./ProfileFilters"; import {dropdownConfig, ProfileFilters} from "./ProfileFilters";
import Image from "next/image";
import {useSession} from "next-auth/react";
// Disable static generation // Disable static generation
export const dynamic = "force-dynamic"; export const dynamic = "force-dynamic";
@@ -19,42 +20,83 @@ const initialState = {
maxIntroversion: null as number | null, maxIntroversion: null as number | null,
interests: [] as string[], interests: [] as string[],
coreValues: [] as string[], coreValues: [] as string[],
books: [] as string[],
causeAreas: [] as string[], causeAreas: [] as string[],
connections: [] as string[], connections: [] as string[],
searchQuery: '', searchQuery: '',
forceRun: false, forceRun: false,
}; };
export type DropdownKey = 'interests' | 'causeAreas' | 'connections' | 'coreValues'; type ProfileFilters = {
export type RangeKey = 'age' | 'introversion'; gender: string;
// type OtherKey = 'gender' | 'searchQuery'; minAge: number | null;
maxAge: number | null;
minIntroversion: number | null;
maxIntroversion: number | null;
interests: string[];
books: string[];
coreValues: string[];
causeAreas: string[];
connections: string[];
searchQuery: string;
forceRun: boolean;
};
export default function ProfilePage() { export default function ProfilePage() {
const {data: session} = useSession();
const userId = session?.user?.id
console.log("session:", userId)
// if (!userId) return <div/>
const [profiles, setProfiles] = useState<ProfileData[]>([]); const [profiles, setProfiles] = useState<ProfileData[]>([]);
const [loading, setLoading] = useState(true); const [loading, setLoading] = useState(true);
const [error, setError] = useState(''); const [error, setError] = useState('');
const [_, setShowFilters] = useState(true); const [_, setShowFilters] = useState(true);
const [totalUsers, setTotalUsers] = useState<number>(0); const [images, setImages] = useState<string[]>([]);
const [images, setImages] = useState<string[]>([])
const [text, setText] = useState<string>('');
const [filters, setFilters] = useState(initialState); const [filters, setFilters] = useState(initialState);
useEffect(() => { useEffect(() => {
const getCount = async () => { const params = new URLSearchParams(window.location.search);
const countResponse = await fetch('/api/profiles/count'); const newFilters = {...initialState};
if (countResponse.ok) {
const {count} = await countResponse.json();
setTotalUsers(count);
}
};
getCount(); for (const [key, value] of params.entries()) {
}, []); // <- runs once after initial mount // Type guard to check if the key is a valid filter key
if (key in newFilters) {
const filterKey = key as keyof ProfileFilters;
if (key === 'searchQuery') {
setText(value);
newFilters[filterKey] = value as never;
} else if (['interests', 'coreValues', 'causeAreas', 'connections'].includes(key)) {
const arrayKey = filterKey as 'interests' | 'coreValues' | 'causeAreas' | 'connections';
newFilters[arrayKey] = [...newFilters[arrayKey], value];
} else {
newFilters[filterKey] = value as never;
}
}
}
console.log(newFilters);
setFilters(newFilters);
}, []);
const [isStart, setIsStart] = useState(true);
const fetchProfiles = useCallback(async () => { const fetchProfiles = useCallback(async () => {
try { try {
setLoading(true); setLoading(true);
const params = new URLSearchParams(); let params = new URLSearchParams();
if (isStart) {
params = new URLSearchParams(window.location.search);
setIsStart(false);
}
console.log('fetchProfiles', params);
if (filters.gender) params.append('gender', filters.gender); if (filters.gender) params.append('gender', filters.gender);
if (filters.minAge) params.append('minAge', filters.minAge.toString()); if (filters.minAge) params.append('minAge', filters.minAge.toString());
@@ -70,9 +112,12 @@ export default function ProfilePage() {
} }
} }
if (filters.searchQuery) params.append('search', filters.searchQuery); if (filters.searchQuery) params.append('searchQuery', filters.searchQuery);
const response = await fetch(`/api/profiles?${params.toString()}`); let s = params.toString();
window.history.pushState({}, '', `?${s}`);
const response = await fetch(`/api/profiles?${s}`);
const data = await response.json(); const data = await response.json();
if (!response.ok) { if (!response.ok) {
console.log(response); console.log(response);
@@ -111,7 +156,7 @@ export default function ProfilePage() {
} finally { } finally {
setLoading(false); setLoading(false);
} }
}, [filters]); }, [filters, images]);
useEffect(() => { useEffect(() => {
fetchProfiles(); fetchProfiles();
@@ -140,12 +185,13 @@ export default function ProfilePage() {
const resetFilters = () => { const resetFilters = () => {
setFilters(initialState); setFilters(initialState);
setText(''); setText('');
// window.history.pushState({}, '', '');
}; };
const [text, setText] = useState<string>('');
const onFilterChange = handleFilterChange const onFilterChange = handleFilterChange
if (!userId) return <div/>
return ( return (
<div className="min-h-screen px-4 sm:px-6 lg:px-8"> <div className="min-h-screen px-4 sm:px-6 lg:px-8">
<div className="w-full max-w-7xl mx-auto"> <div className="w-full max-w-7xl mx-auto">
@@ -165,7 +211,7 @@ export default function ProfilePage() {
<div className="relative"> <div className="relative">
<input <input
type="text" type="text"
placeholder={totalUsers > 0 ? `Search anything among the ${totalUsers} users...` : "Search anything..."} placeholder='Try "meditation", "hiking", or multiple words like "writing, nature"'
className="w-full pl-10 pr-10 py-2 border rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-transparent" className="w-full pl-10 pr-10 py-2 border rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-transparent"
value={text} value={text}
onChange={(e) => setText(e.target.value)} onChange={(e) => setText(e.target.value)}
@@ -220,8 +266,11 @@ export default function ProfilePage() {
{/* Profiles Grid */} {/* Profiles Grid */}
<div className="flex-1"> <div className="flex-1">
{loading ? ( {loading ? (
<div className="flex justify-center py-2"> <div className="flex justify-center py-8">
<LoadingSpinner/> <div className="flex justify-center min-h-screen py-8">
<div data-testid="spinner"
className="animate-spin rounded-full h-12 w-12 border-t-2 border-b-2 border-blue-500"></div>
</div>
</div> </div>
) : error ? ( ) : error ? (
<div className="flex justify-center py-2"> <div className="flex justify-center py-2">
@@ -241,7 +290,7 @@ export default function ProfilePage() {
<div className="p-4 h-full flex flex-col"> <div className="p-4 h-full flex flex-col">
<div className="flex items-center space-x-4"> <div className="flex items-center space-x-4">
{renderImages && (<div className="flex-shrink-0"> {renderImages && (<div className="flex-shrink-0">
<img <Image
className="h-16 w-16 rounded-full object-cover" className="h-16 w-16 rounded-full object-cover"
src={images[idx]} src={images[idx]}
alt={``} alt={``}
@@ -280,6 +329,18 @@ export default function ProfilePage() {
</div> </div>
)} )}
</div> </div>
<div className="mt-4 space-y-2 flex-grow">
{user.profile.books?.length > 0 && (
<div className="flex flex-wrap gap-1">
{user.profile.books.slice(0, 6).map(({value}) => (
<span key={value?.id}
className="inline-block text-xs px-2 py-1 bg-blue-50 text-blue-700 dark:text-white dark:bg-gray-700 rounded-full">
{value?.name}
</span>
))}
</div>
)}
</div>
</div> </div>
</Link> </Link>
))} ))}
@@ -294,7 +355,7 @@ export default function ProfilePage() {
</svg> </svg>
<h3 className="mt-2 text-sm font-medium">No profiles found</h3> <h3 className="mt-2 text-sm font-medium">No profiles found</h3>
<p className="mt-1 text-sm"> <p className="mt-1 text-sm">
Try adjusting your search or filter to find what you're looking for. {"Try adjusting your search or filter to find what you're looking for."}
</p> </p>
</div> </div>
)} )}

View File

@@ -5,7 +5,7 @@ export default function SetupPage() {
<div className="min-h-screen flex flex-col items-center justify-center p-8"> <div className="min-h-screen flex flex-col items-center justify-center p-8">
<div className="max-w-3xl w-full rounded-lg shadow-lg p-8"> <div className="max-w-3xl w-full rounded-lg shadow-lg p-8">
<h1 className="text-3xl font-bold text-center mb-6 "> <h1 className="text-3xl font-bold text-center mb-6 ">
Welcome to BayesBond Welcome to Compass
</h1> </h1>
<p className="text-gray-600 mb-8 text-center"> <p className="text-gray-600 mb-8 text-center">
It looks like your database isn&apos;t set up yet. Follow the It looks like your database isn&apos;t set up yet. Follow the

View File

@@ -0,0 +1,695 @@
import React, {useEffect, useState} from "react";
import {SubmitHandler, useForm} from "react-hook-form";
import {Gender} from "@prisma/client";
import {useSession} from "next-auth/react";
import {useRouter, useSearchParams} from "next/navigation";
import Slider from "@mui/material/Slider";
import {errorBlock} from "@/lib/client/errors";
import {Item} from "@/lib/client/schema";
import {fetchFeatures} from "@/lib/client/fetching";
// Updated Question type to support more field types and conditions
type Question = {
name: string;
label: string;
type?: string;
options?: string[];
range?: string[];
optional?: boolean;
group?: boolean;
fields?: Question[];
condition?: (values: any) => boolean;
};
// Onboarding questions with conditional logic and grouping
const questions: Question[] = [
{
name: "name",
label: "Hi! What's your first name?",
},
{
name: "location",
label: "Where are you located?",
group: true,
fields: [
{name: "country", label: "Country"},
// { name: "zipCode", label: "Zip Code" },
],
},
{
name: "gender",
label: "What's your gender?",
type: "select",
options: ["Man", "Woman", "Other"],
},
// {
// name: "genderOther",
// label: "How do you identify?",
// type: "select",
// options: [
// "Agender",
// "Androgynous",
// "Bigender",
// "Cis Man",
// "Cis Woman",
// "Genderfluid",
// "Genderqueer",
// "Gender Nonconforming",
// "Hijra",
// "Intersex",
// "Non-binary",
// "Other gender",
// "Pangender",
// "Transfeminine",
// "Transgender",
// "Trans Man",
// "Transmasculine",
// "Transsexual",
// "Trans Woman",
// "Two Spirit"
// ],
// optional: true,
// condition: (values) => values.gender === "See all",
// },
{
name: "birthday",
label: "When's your birthday?",
type: "date",
},
{
name: "connections",
label: "What kind of relationship are you looking for?",
type: "multiselect",
options: [
// "Debate Partner",
"Friendship",
"Short-term relationship",
"Long-term Relationship",
// "Other",
],
},
// {
// name: "kids",
// label: "What are your ideal plans for children? (optional)",
// type: "select",
// options: [
// "Skip",
// "Want someday",
// "Don't want",
// "Have and want more",
// "Have and don't want more",
// "Not sure yet",
// "Have kids",
// "Open to kids",
// ],
// optional: true,
// condition: (values) =>
// ["Short-term dating", "Hookups", "Long-term dating"].includes(
// values.relationshipType
// ),
// },
// {
// name: "nonMonogamy",
// label: "Non-Monogamy Options",
// type: "select",
// options: ["Monogamous", "Non-monogamous", "Open to either"],
// condition: (values) =>
// ["Short-term dating", "Hookups", "Long-term dating"].includes(
// values.relationshipType
// ),
// },
// {
// name: "photos",
// label: "Add photos (optional)",
// type: "file",
// optional: true,
// },
{
name: "description",
label: "Tell us about yourself",
type: "textarea",
},
{
name: "contactInfo",
label: "How can people contact you?",
type: "textarea",
},
// Personality questions
// {
// name: "intenseOrCarefree",
// label: "Which word describes you better?",
// type: "select",
// options: ["Intense", "Carefree"],
// },
// {
// name: "religion",
// label: "How important is religion/God in your life?",
// type: "select",
// options: [
// "Not at all important",
// "Slightly important",
// "Moderately important",
// "Very important",
// "Extremely important"
// ]
// },
// {
// name: "politics",
// label: "Which best describes your political beliefs?",
// type: "select",
// options: [
// "Very liberal",
// "Liberal",
// "Moderate",
// "Conservative",
// "Very conservative",
// "Other"
// ]
// },
{
name: "introversion",
label: "How would you describe your social style?",
type: "slider",
range: ['Introverted', 'Extroverted'],
},
// {
// name: "introversion",
// label: "How would you describe your social style?",
// type: "select",
// options: [
// "Very introverted",
// "Somewhat introverted",
// "In the middle",
// "Somewhat extroverted",
// "Very extroverted"
// ]
// },
];
// List of valid countries (shortened for brevity, add more as needed)
const countryOptions = [
"United States", "Canada", "United Kingdom", "Australia", "Germany", "France", "India", "China", "Japan", "Brazil", "Mexico", "Italy", "Spain", "Netherlands", "Sweden", "Norway", "Denmark", "Finland", "Ireland", "New Zealand"
// ...add more countries as needed
];
type FormValues = {
[key: string]: any;
};
// Helper to get visible questions based on current form values
const getVisibleQuestions = (values: FormValues) =>
questions.filter((q) => !q.condition || q.condition(values));
const OnboardingForm: React.FC = () => {
const searchParams = useSearchParams();
const redirect = searchParams.get('redirect') || '/';
const router = useRouter();
const {update} = useSession();
const [error, setError] = useState('');
const [step, setStep] = useState(0);
const {register, handleSubmit, getValues, formState: {errors}} = useForm<FormValues>();
const [formValues, setFormValues] = useState<FormValues>({});
const [showGenderDefs, setShowGenderDefs] = useState(false);
const [sliderValue, setSliderValue] = useState(0);
const [isSubmitting, setIsSubmitting] = useState(false);
const featureNames = ['connections'];
const [allFeatures, _setAllFeatures] = useState(() =>
Object.fromEntries(featureNames.map((id) => [id, [] as Item[]]))
);
const setAllFeatures = (id: string, value: any) => {
_setAllFeatures((prev) => ({...prev, [id]: value}));
};
useEffect(() => {
console.log('Fetching features...');
fetchFeatures(setAllFeatures);
}, []);
// Helper to calculate age
const getAge = (month: string, day: string, year: string) => {
if (!month || !day || !year) return null;
const mm = parseInt(month, 10);
const dd = parseInt(day, 10);
const yyyy = parseInt(year, 10);
if (isNaN(mm) || isNaN(dd) || isNaN(yyyy)) return null;
const today = new Date();
// const birthDate = new Date(yyyy, mm - 1, dd);
let age = today.getFullYear() - yyyy;
const m = today.getMonth() - (mm - 1);
if (m < 0 || (m === 0 && today.getDate() < dd)) {
age--;
}
return age;
};
const visibleQuestions = getVisibleQuestions(formValues);
const isLastStep = step === visibleQuestions.length - 1;
const currentQuestion = visibleQuestions[step];
const onSubmit: SubmitHandler<FormValues> = async () => {
setIsSubmitting(true);
setError('');
const allValues = {...formValues, ...getValues()};
console.log(JSON.stringify(allValues, null, 2));
try {
const data: any = {
profile: {
description: allValues.description,
contactInfo: allValues.contactInfo,
location: allValues.country,
gender: allValues.gender as Gender,
birthYear: parseInt(allValues.birthYear),
introversion: 100 - allValues.introversion,
// images: keys,
},
// ...(key && {image: key}),
name: allValues.name,
};
for (const t of ['connections']) {
if (!allValues[t]) continue;
data[t] = Array.from(allValues[t]).map(name => ({
id: allFeatures[t].find(i => i.name === name)?.id,
name: name
}));
}
console.log('data:', data)
const response = await fetch('/api/user/update-profile', {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify(data),
});
if (!response.ok) {
const errorData = await response.json();
setError(errorData.error || 'Failed to update profile');
return;
}
await update();
router.push(redirect);
} catch (error) {
console.error('Profile update error:', error);
setError(error instanceof Error ? error.message : 'Failed to update profile');
}
};
const nextStep = () => {
const values = getValues();
setFormValues((prev) => ({...prev, ...values}));
setStep((s) => Math.min(s + 1, visibleQuestions.length - 1));
};
const prevStep = () => setStep((s) => Math.max(s - 1, 0));
// Questions where skip is allowed (from 'relationshipDuration' onward)
const skipFrom = visibleQuestions.findIndex(q => q.name === 'relationshipDuration');
const canSkip = step >= skipFrom && skipFrom !== -1;
return (
<div>
{error && errorBlock(error)}
<form onSubmit={handleSubmit(onSubmit)} className="mt-0 space-y-6 flex flex-col items-center">
<div className="w-full flex flex-col items-center">
<label className="px-4 text-center block w-full text-3xl font-bold mb-8 mt-0"
style={{marginTop: '0rem'}}>{currentQuestion.label}</label>
{currentQuestion.group && currentQuestion.fields ? (
currentQuestion.fields.map((field) => {
let fieldInput;
if (field.name === "country") {
fieldInput = (
<>
<input
list="country-list"
{...register(field.name, {
required: !field.optional,
// validate: value => countryOptions.includes(value) || "Please select a valid country." // Skip until all countries are listed
})}
defaultValue={getValues(field.name) || ""}
onKeyDown={(e) => {
if (e.key === "Enter") {
e.preventDefault();
nextStep();
}
}}
className="text-center text-lg py-3 px-5 h-14 rounded-full border w-72"
/>
<datalist id="country-list">
{countryOptions.map((country) => (
<option key={country} value={country}/>
))}
</datalist>
</>
);
} else if (field.name === "zipCode") {
fieldInput = (
<input
type="text"
{...register(field.name, {
required: !field.optional,
pattern: {value: /^\d{5}(-\d{4})?$/, message: "Please enter a valid US zip code."}
})}
defaultValue={getValues(field.name) || ""}
onKeyDown={(e) => {
if (e.key === "Enter") {
e.preventDefault();
nextStep();
}
}}
className="text-center text-lg py-3 px-5 h-14 rounded-full border w-48"
/>
);
} else {
fieldInput = (
<input
type={field.type || "text"}
{...register(field.name, {required: !field.optional})}
defaultValue={getValues(field.name) || ""}
onKeyDown={(e) => {
if (e.key === "Enter") {
e.preventDefault();
nextStep();
}
}}
className="text-center text-lg py-3 px-5 h-14 rounded-full border"
/>
);
}
return (
<div key={field.name} className="w-full flex flex-col items-center">
<label className="text-center w-full">{field.label}</label>
{fieldInput}
{errors[field.name] && (
<span className="block text-center mt-4">
{(errors[field.name] as any)?.message || "This field is required"}
</span>
)}
</div>
);
})
) : currentQuestion.name === "birthday" ? (
<div className="flex flex-row gap-2 justify-center items-center mt-2">
<input
type="text"
maxLength={2}
placeholder="MM"
{...register("birthMonth", {required: true, pattern: /^(0?[1-9]|1[0-2])$/})}
className="rounded-full border px-4 py-2 text-lg text-center w-20"
/>
<input
type="text"
maxLength={2}
placeholder="DD"
{...register("birthDay", {required: true, pattern: /^(0?[1-9]|[12][0-9]|3[01])$/})}
className="rounded-full border px-4 py-2 text-lg text-center w-20"
/>
<input
type="text"
maxLength={4}
placeholder="YYYY"
{...register("birthYear", {required: true, pattern: /^[0-9]{4}$/})}
className="rounded-full border px-4 py-2 text-lg text-center w-28"
/>
{(() => {
const month = getValues("birthMonth");
const day = getValues("birthDay");
const year = getValues("birthYear");
const age = getAge(month, day, year);
if (age && age > 0 && age < 120) {
return <span className="ml-4 text-lg font-semibold text-blue-700">You are {age}!</span>;
}
return null;
})()}
</div>
) : (currentQuestion.type === "select" && currentQuestion.options && currentQuestion.options.length <= 6) ? (
<div className="flex flex-wrap gap-2 mt-2 justify-center">
{currentQuestion.options.map((opt) => (
<button
key={opt}
type="button"
className={`px-4 py-2 rounded-full border ${formValues[currentQuestion.name] === opt ? 'bg-red-700 text-white' : 'bg-gray-100 dark:bg-gray-700'} hover:bg-red-100 dark:hover:bg-red-800 transition-colors`}
onClick={() => {
setFormValues({...formValues, [currentQuestion.name]: opt});
setStep((s) => Math.min(s + 1, visibleQuestions.length - 1));
}}
>
{opt}
</button>
))}
</div>
) : (currentQuestion.type === "multiselect" && currentQuestion.options && currentQuestion.options.length <= 6) ? (
<div className="flex flex-wrap gap-2 mt-2 justify-center">
{currentQuestion.options.map((opt) => {
const selected = (formValues[currentQuestion.name] || []).includes(opt);
return (
<button
key={opt}
type="button"
className={`px-4 py-2 rounded-full border ${selected ? 'bg-red-700 text-white' : 'bg-gray-100 dark:bg-gray-700'} hover:bg-red-100 dark:hover:bg-red-800 transition-colors`}
onClick={() => {
const prev = formValues[currentQuestion.name] || [];
const next = selected ? prev.filter((v: string) => v !== opt) : [...prev, opt];
setFormValues({...formValues, [currentQuestion.name]: next});
}}
>
{opt}
</button>
);
})}
</div>
) : currentQuestion.type === "slider" ? (
<div className="flex items-center w-full max-w-xl gap-4">
{currentQuestion.range && (
<span className="min-w-[80px] text-gray-500 text-left">
{currentQuestion.range[0]}
</span>
)}
<Slider
value={sliderValue}
onChange={(e, value) => {
setSliderValue(value);
setFormValues({...formValues, [currentQuestion.name]: value});
}}
valueLabelDisplay="auto"
min={0}
max={100}
sx={{
color: '#3B82F6',
'& .MuiSlider-valueLabel': {
backgroundColor: '#3B82F6',
color: '#fff',
},
}}
className="flex-1"
/>
{currentQuestion.range && (
<span className="min-w-[80px] text-gray-500 text-right">
{currentQuestion.range[1]}
</span>
)}
</div>
) : currentQuestion.type === "select" ? (
<select
{...register(currentQuestion.name, {required: !currentQuestion.optional})}
className="px-4 text-center text-lg py-3 h-14 rounded-full border"
onKeyDown={(e) => {
if (e.key === "Enter") {
e.preventDefault();
nextStep();
}
}}
>
<option value="">Select...</option>
{currentQuestion.options?.map((opt) => (
<option key={opt} value={opt}>
{opt}
</option>
))}
</select>
) : currentQuestion.type === "multiselect" ? (
<select
multiple
{...register(currentQuestion.name, {required: !currentQuestion.optional})}
className="px-4 border rounded-full w-full min-h-[44px] focus:outline-none focus:ring-2 focus:ring-blue-400 text-center text-lg py-3 h-14"
style={{minHeight: 44, boxSizing: 'border-box', cursor: 'pointer'}}
onKeyDown={(e) => {
if (e.key === "Enter") {
e.preventDefault();
nextStep();
}
}}
>
{currentQuestion.options?.map((opt) => (
<option key={opt} value={opt}>
{opt}
</option>
))}
</select>
// For a more advanced UI, consider using react-select's MultiSelect component
) : currentQuestion.type === "textarea" ? (
<textarea
{...register(currentQuestion.name, {required: !currentQuestion.optional})}
defaultValue={getValues(currentQuestion.name) || ""}
className="text-center text-lg py-3 px-5 h-28 rounded-lg border"
/>
) : currentQuestion.type === "file" ? (
<input
type="file"
{...register(currentQuestion.name)}
className="text-center text-lg py-3 px-5 h-14 rounded-full border"
/>
) : (currentQuestion.name === "genderOther") ? (
<div className="flex flex-col items-center w-full">
<select
{...register(currentQuestion.name, {required: !currentQuestion.optional})}
className="px-4 text-center text-lg py-3 h-14 rounded-full border w-full max-w-xl"
>
<option value="">Select...</option>
{questions.find(q => q.name === "genderOther")?.options?.map((opt) => (
<option key={opt} value={opt}>{opt}</option>
))}
</select>
<button
type="button"
className="mt-4 mb-2 px-4 py-2 rounded-full border bg-gray-100 hover:bg-gray-200 text-gray-700 transition-colors"
onClick={() => setShowGenderDefs((v) => !v)}
>
{showGenderDefs ? "Hide definitions" : "Show definitions"}
</button>
{showGenderDefs && (
<div
className="mt-2 p-4 bg-white border rounded-lg shadow max-w-xl text-left text-sm overflow-y-auto max-h-96">
<ul className="list-disc pl-5">
<li><b>Agender</b>: Individuals with no gender identity or a neutral gender identity.</li>
<li><b>Androgynous</b>: Individuals with both male & female presentation or nature.</li>
<li><b>Bigender</b>: Individuals who identify as multiple genders/identities, either simultaneously
or
at different times.
</li>
<li><b>Cis Man</b>: Individuals whose gender identity matches the male sex they were assigned at
birth.
</li>
<li><b>Cis Woman</b>: Individuals whose gender identity matches the female sex they were assigned at
birth.
</li>
<li><b>Genderfluid</b>: Individuals who dont have a fixed gender identity.</li>
<li><b>Genderqueer</b>: Individuals who dont identify with binary gender identity norms.</li>
<li><b>Gender Nonconforming</b>: Individuals whose gender expressions dont match masculine &
feminine
gender norms.
</li>
<li><b>Hijra</b>: A third gender identity, largely used in the Indian subcontinent, which typically
reflects people who were assigned male at birth, with feminine gender expression, who identify as
neither male nor female.
</li>
<li><b>Intersex</b>: Individuals born with a reproductive or sexual anatomy that doesnt fit the
typical definitions of female or male.
</li>
<li><b>Non-binary</b>: A term covering any gender identity or expression that doesnt fit within the
gender binary.
</li>
<li><b>Other gender</b>: Individuals who identify with any other gender expressions.</li>
<li><b>Pangender</b>: Individuals who identify with a wide multiplicity of gender identities.</li>
<li><b>Transfeminine</b>: Transgender individuals whose gender expression is more feminine
presenting.
</li>
<li><b>Transgender</b>: Individuals whose gender identity differs from the sex they were assigned at
birth.
</li>
<li><b>Trans Man</b>: Individuals who were assigned female at birth (AFAB) but have a male gender
identity.
</li>
<li><b>Transmasculine</b>: Transgender individuals whose gender expression is more masculine
presenting.
</li>
<li><b>Transsexual</b>: This term is sometimes used to describe trans individuals (who do not
identify
with the sex they were assigned at birth) who wish to align their gender identity & sex through
medical intervention.
</li>
<li><b>Trans Woman</b>: Individuals who were assigned male at birth (AMAB) but have a female gender
identity.
</li>
<li><b>Two Spirit</b>: Term largely used in Indigenous, Native American, and First Nation cultures,
reflecting individuals who identify with multiple genders/gender identities that are neither male
nor female.
</li>
</ul>
</div>
)}
</div>
) : (
<input
type={currentQuestion.type || "text"}
{...register(currentQuestion.name, {required: !currentQuestion.optional})}
defaultValue={getValues(currentQuestion.name) || ""}
onKeyDown={(e) => {
if (e.key === "Enter") {
e.preventDefault();
nextStep();
}
}}
className="text-center text-lg py-3 px-5 h-14 rounded-full border"
/>
)}
{errors[currentQuestion.name] && <span className="block text-center mt-4">This field is required</span>}
</div>
<div className="flex flex-row justify-center items-center w-full mt-16 gap-4">
{step > 0 && (
<button
type="button"
onClick={prevStep}
className="rounded-full px-6 py-2 font-semibold bg-gray-200 dark:bg-gray-700 hover:bg-gray-300 dark:hover:bg-gray-600 transition-colors"
>
Back
</button>
)}
{canSkip && (
<button
type="button"
onClick={() => {
setFormValues((prev) => ({...prev, [currentQuestion.name]: ""}));
setStep((s) => Math.min(s + 1, visibleQuestions.length - 1));
}}
className="rounded-full px-6 py-2 font-semibold bg-gray-100 text-gray-700 hover:bg-gray-200 transition-colors"
>
Skip
</button>
)}
{!isLastStep ? (
<button
key={currentQuestion.name}
type="button"
onClick={nextStep}
className="rounded-full px-6 py-2 font-semibold bg-red-700 text-white hover:bg-red-800 transition-colors"
>
Next
</button>
) : (
<button
type="submit"
className="rounded-full px-6 py-2 font-semibold bg-red-700 text-white hover:bg-red-800 transition-colors"
>
{isSubmitting ? 'Saving...' : 'Submit'}
</button>
)}
</div>
</form>
</div>
);
};
export default OnboardingForm;

View File

@@ -3,7 +3,7 @@
import {useEffect, useState} from 'react'; import {useEffect, useState} from 'react';
import {Textarea} from '@/components/ui/textarea'; import {Textarea} from '@/components/ui/textarea';
import {Select, SelectContent, SelectItem, SelectTrigger, SelectValue,} from '@/components/ui/select'; import {Select, SelectContent, SelectItem, SelectTrigger, SelectValue,} from '@/components/ui/select';
import {cons} from "effect/List"; // import {cons} from "effect/List";
type Prompt = { type Prompt = {
id: string; id: string;

View File

@@ -0,0 +1,9 @@
// For som unknown reasons, the spinner does not render when using LoadingSpinner(), so I copy paste the div block everywhere (TODO)
export default function LoadingSpinner() {
return (
<div className="flex justify-center min-h-screen py-8">
<div data-testid="spinner" className="animate-spin rounded-full h-12 w-12 border-t-2 border-b-2 border-blue-500"></div>
</div>
)
}

View File

@@ -0,0 +1,2 @@
'use client';

View File

@@ -0,0 +1,20 @@
import React from "react";
export function errorBlock(error: string = '') {
return <div className="bg-red-50 border-l-4 border-red-400 p-4">
<div className="flex">
<div className="flex-shrink-0">
<svg className="h-5 w-5 text-red-400" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20"
fill="currentColor">
<path fillRule="evenodd"
d="M10 18a8 8 0 100-16 8 8 0 000 16zM8.707 7.293a1 1 0 00-1.414 1.414L8.586 10l-1.293 1.293a1 1 0 101.414 1.414L10 11.414l1.293 1.293a1 1 0 001.414-1.414L11.414 10l1.293-1.293a1 1 0 00-1.414-1.414L10 8.586 8.707 7.293z"
clipRule="evenodd"/>
</svg>
</div>
<div className="ml-3">
<p className="text-sm text-red-700">{error}</p>
</div>
</div>
</div>
}

View File

@@ -0,0 +1,15 @@
export async function fetchFeatures(setAllFeatures: any) {
// results = []
try {
const res = await fetch('/api/interests');
if (res.ok) {
const data = await res.json();
for (const [id, values] of Object.entries(data)) {
setAllFeatures(id, values || []);
// results.push({id, values});
}
}
} catch (error) {
console.error('Error fetching feature options:', error);
}
}

View File

@@ -20,5 +20,4 @@ export async function parseImage(img: string, setImage: any, batch = false) {
setImage(url); setImage(url);
} }
} }
}
}

View File

@@ -2,9 +2,8 @@
import Image from "next/image"; import Image from "next/image";
import {pStyle} from "@/lib/client/constants"; import {pStyle} from "@/lib/client/constants";
import {useEffect, useState} from "react"; import React, {useEffect, useState} from "react";
import {parseImage} from "@/lib/client/media"; import {parseImage} from "@/lib/client/media";
import LoadingSpinner from "@/lib/client/LoadingSpinner";
import {useRouter} from 'next/navigation'; import {useRouter} from 'next/navigation';
interface DeleteProfileButtonProps { interface DeleteProfileButtonProps {
@@ -59,7 +58,7 @@ export function DeleteProfileButton({profileId, onDelete}: DeleteProfileButtonPr
); );
} }
export function getProfile(url: string, header: any = null) { export function Profile(url: string, header: any = null) {
const [loading, setLoading] = useState(true); const [loading, setLoading] = useState(true);
const [userData, setUserData] = useState<any>(null); const [userData, setUserData] = useState<any>(null);
@@ -95,10 +94,15 @@ export function getProfile(url: string, header: any = null) {
} }
fetchImage(); fetchImage();
}, []); }, [url]);
if (loading) { if (loading) {
return <LoadingSpinner/>; return (
<div className="flex justify-center min-h-screen py-8">
<div data-testid="spinner"
className="animate-spin rounded-full h-12 w-12 border-t-2 border-b-2 border-blue-500"></div>
</div>
)
} }
if (!userData) { if (!userData) {
@@ -111,19 +115,21 @@ export function getProfile(url: string, header: any = null) {
interface Tags { interface Tags {
profileAttribute: string; profileAttribute: string;
attribute: string; attribute?: string;
title: string; title: string;
} }
const tagsConfig: Tags[] = [ const tagsConfig: Tags[] = [
{profileAttribute: 'desiredConnections', attribute: 'connection', title: 'Desired Connections'}, {profileAttribute: 'desiredConnections', attribute: 'connection', title: 'Connection Type'},
{profileAttribute: 'coreValues', attribute: 'value', title: 'Core Values'}, {profileAttribute: 'coreValues', title: 'Values'},
{profileAttribute: 'intellectualInterests', attribute: 'interest', title: 'Core Interests'}, {profileAttribute: 'intellectualInterests', attribute: 'interest', title: 'Interests'},
{profileAttribute: 'causeAreas', attribute: 'causeArea', title: 'Cause Areas'}, {profileAttribute: 'books', title: 'Works to Discuss'},
// {profileAttribute: 'causeAreas', attribute: 'causeArea', title: 'Cause Areas'},
] ]
function getTags({profileAttribute, attribute, title}: Tags) { function getTags({profileAttribute, attribute = 'value', title}: Tags) {
const values = userData?.profile?.[profileAttribute]; const values = userData?.profile?.[profileAttribute];
console.log('values', values);
return <div key={profileAttribute + '.div'}> return <div key={profileAttribute + '.div'}>
{values?.length > 0 && ( {values?.length > 0 && (
<div className="mt-3">< <div className="mt-3"><
@@ -161,18 +167,18 @@ export function getProfile(url: string, header: any = null) {
<div className="flex-1"> <div className="flex-1">
<div className="h-32 w-32 rounded-full border-4 border-white overflow-hidden "> <div className="h-32 w-32 rounded-full border-4 border-white overflow-hidden ">
<a href={image} target="_blank" rel="noopener noreferrer"> <a href={image} target="_blank" rel="noopener noreferrer">
<Image <Image
src={image} src={image}
alt={userData.name || 'Profile picture'} alt={userData.name || 'Profile picture'}
className="h-full w-full object-cover" className="h-full w-full object-cover"
width={200} width={200}
height={200} height={200}
// onError={(e) => { // onError={(e) => {
// const target = e.target as HTMLImageElement; // const target = e.target as HTMLImageElement;
// target.onerror = null; // target.onerror = null;
// target.src = `https://ui-avatars.com/api/?name=${encodeURIComponent(profile.name || 'U')}&background=random`; // target.src = `https://ui-avatars.com/api/?name=${encodeURIComponent(profile.name || 'U')}&background=random`;
// }} // }}
/> />
</a> </a>
</div> </div>
</div> </div>
@@ -234,10 +240,21 @@ export function getProfile(url: string, header: any = null) {
{ {
userData?.profile?.introversion && ( userData?.profile?.introversion && (
<div> <div>
<h2 className="text-sm font-medium text-gray-500 uppercase tracking-wider"> Introversion - <h2 className="text-sm font-medium text-gray-500 uppercase tracking-wider">
Extroversion </h2> Social Style
< p </h2>
className="mt-1 capitalize"> {100 - userData.profile.introversion}% </p> <div className="flex items-center w-full max-w-xl gap-4">
<span className={pStyle}>Introverted</span>
<div className="mt-1 flex items-center gap-4 w-32">
<div className="w-full bg-gray-200 rounded-full h-2">
<div
className="bg-blue-600 h-2 rounded-full transition-all duration-200"
style={{width: `${100 - userData.profile.introversion}%`}}
/>
</div>
</div>
<span className={pStyle}>Extroverted</span>
</div>
</div> </div>
) )
} }
@@ -253,16 +270,15 @@ export function getProfile(url: string, header: any = null) {
) )
} }
{ {/*{*/}
userData?.profile?.conflictStyle && ( {/* userData?.profile?.conflictStyle && (*/}
<div> {/* <div>*/}
<h2 className="text-sm font-medium text-gray-500 uppercase tracking-wider"> Conflict {/* <h2 className="text-sm font-medium text-gray-500 uppercase tracking-wider"> Conflict Style </h2>*/}
Style </h2> {/* < p*/}
< p {/* className={pStyle}> {userData.profile.conflictStyle} </p>*/}
className={pStyle}> {userData.profile.conflictStyle} </p> {/* </div>*/}
</div> {/* )*/}
) {/*}*/}
}
{tagsConfig.map((tag: any) => getTags(tag))} {tagsConfig.map((tag: any) => getTags(tag))}

View File

@@ -16,9 +16,20 @@ export interface ProfileData {
contactInfo: string; contactInfo: string;
intellectualInterests: { interest?: { name?: string, id?: string } }[]; intellectualInterests: { interest?: { name?: string, id?: string } }[];
coreValues: { value?: { name?: string, id?: string } }[]; coreValues: { value?: { name?: string, id?: string } }[];
books: { value?: { name?: string, id?: string } }[];
causeAreas: { causeArea?: { name?: string, id?: string } }[]; causeAreas: { causeArea?: { name?: string, id?: string } }[];
desiredConnections: { connection?: { name?: string, id?: string } }[]; desiredConnections: { connection?: { name?: string, id?: string } }[];
promptAnswers: { prompt?: string; answer?: string, id?: string }[]; promptAnswers: { prompt?: string; answer?: string, id?: string }[];
images: string[]; images: string[];
}; };
}
export type DropdownKey = 'interests' | 'causeAreas' | 'connections' | 'coreValues' | 'books';
export type RangeKey = 'age' | 'introversion';
// type OtherKey = 'gender' | 'searchQuery';
export interface Item {
id: DropdownKey;
name: string;
} }

View File

View File

@@ -29,6 +29,7 @@ export async function retrieveUser(id: string) {
intellectualInterests: {include: {interest: true}}, intellectualInterests: {include: {interest: true}},
causeAreas: {include: {causeArea: true}}, causeAreas: {include: {causeArea: true}},
coreValues: {include: {value: true}}, coreValues: {include: {value: true}},
books: {include: {value: true}},
desiredConnections: {include: {connection: true}}, desiredConnections: {include: {connection: true}},
promptAnswers: true, promptAnswers: true,
}, },

View File

@@ -6,5 +6,5 @@ import 'server-only';
// //
// export const supabase = createClient( // export const supabase = createClient(
// process.env.NEXT_PUBLIC_SUPABASE_URL!, // process.env.NEXT_PUBLIC_SUPABASE_URL!,
// process.env.NEXT_PUBLIC_SUPABASE_ANON_KEY! // process.env.SUPABASE_KEY!
// ); // );

View File

View File

View File

78
_old/package.json Normal file
View File

@@ -0,0 +1,78 @@
{
"name": "Compass",
"version": "1.0.0",
"private": true,
"scripts": {
"dev": "next dev --turbopack",
"postinstall": "npx prisma generate --no-engine",
"build": "npx prisma migrate deploy && next build",
"start": "next start",
"lint": "next lint",
"test": "jest",
"test:watch": "jest --watch",
"test:coverage": "jest --coverage",
"test:update": "jest --updateSnapshot"
},
"dependencies": {
"@auth/prisma-adapter": "^2.10.0",
"@aws-sdk/client-s3": "^3.855.0",
"@aws-sdk/s3-request-presigner": "^3.855.0",
"@emotion/react": "^11.14.0",
"@emotion/styled": "^11.14.1",
"@heroicons/react": "^2.2.0",
"@mui/material": "^7.2.0",
"@next-auth/prisma-adapter": "^1.0.7",
"@playwright/test": "^1.54.2",
"@prisma/client": "^6.12.0",
"@prisma/extension-accelerate": "^2.0.2",
"@radix-ui/react-select": "^2.2.5",
"@react-email/render": "^1.1.3",
"@supabase/supabase-js": "^2.53.0",
"@types/uuid": "^10.0.0",
"@upstash/ratelimit": "^2.0.6",
"@upstash/redis": "^1.35.3",
"bcryptjs": "^3.0.2",
"browser-image-compression": "^2.0.2",
"class-variance-authority": "^0.7.1",
"clsx": "^2.1.1",
"express-rate-limit": "^8.0.1",
"heroicons": "^2.2.0",
"jsonwebtoken": "^9.0.2",
"lucide-react": "^0.503.0",
"next": "^15.4.4",
"next-auth": "^4.24.11",
"next-rate-limit": "^0.0.3",
"next-themes": "^0.4.6",
"react": "^19.1.0",
"react-dom": "^19.0.0",
"react-hook-form": "^7.62.0",
"react-icons": "^5.5.0",
"resend": "^4.7.0",
"tailwind-merge": "^3.3.1",
"use-debounce": "^10.0.5",
"uuid": "^11.1.0",
"wait-on": "^8.0.4"
},
"devDependencies": {
"@eslint/eslintrc": "^3",
"@testing-library/jest-dom": "^6.6.4",
"@testing-library/react": "^16.3.0",
"@testing-library/user-event": "^14.6.1",
"@types/jest": "^30.0.0",
"@types/node": "^20",
"@types/react": "^19",
"@types/react-dom": "^19",
"eslint": "^9",
"eslint-config-next": "15.1.7",
"jest": "^30.0.5",
"jest-environment-jsdom": "^30.0.5",
"postcss": "^8",
"prisma": "^6.13.0",
"tailwindcss": "^3.4.1",
"tsx": "^4.19.2",
"typescript": "^5"
},
"prisma": {
"seed": "npx tsx prisma/seed.ts"
}
}

17
_old/prisma.config.ts Normal file
View File

@@ -0,0 +1,17 @@
import path from "node:path";
import 'dotenv/config'; // <-- This loads your .env file before Prisma runs
import type {PrismaConfig} from "prisma";
export default {
schema: path.join("prisma", "schema.prisma"),
migrations: {
path: path.join("prisma", "migrations"),
},
// views: {
// path: path.join("prisma", "views"),
// },
// typedSql: {
// path: path.join("prisma", "queries"),
// }
} satisfies PrismaConfig;

View File

@@ -0,0 +1,245 @@
-- CreateSchema
CREATE SCHEMA IF NOT EXISTS "public";
-- CreateEnum
CREATE TYPE "public"."Gender" AS ENUM ('Male', 'Female', 'Other');
-- CreateEnum
CREATE TYPE "public"."PersonalityType" AS ENUM ('Introvert', 'Extrovert', 'Ambivert');
-- CreateEnum
CREATE TYPE "public"."ConflictStyle" AS ENUM ('Competing', 'Avoidant', 'Compromising', 'Accommodating', 'Collaborating');
-- CreateTable
CREATE TABLE "public"."User" (
"id" TEXT NOT NULL,
"name" TEXT,
"email" TEXT,
"password" TEXT,
"emailVerified" TIMESTAMP(3),
"verificationToken" TEXT,
"verificationTokenExpires" TIMESTAMP(3),
"image" TEXT,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "User_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "public"."Profile" (
"id" TEXT NOT NULL,
"userId" TEXT NOT NULL,
"location" TEXT,
"description" TEXT,
"contactInfo" TEXT,
"birthYear" INTEGER,
"occupation" TEXT,
"gender" "public"."Gender",
"personalityType" "public"."PersonalityType",
"introversion" INTEGER,
"conflictStyle" "public"."ConflictStyle",
"images" TEXT[],
CONSTRAINT "Profile_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "public"."Connection" (
"id" TEXT NOT NULL,
"name" TEXT NOT NULL,
CONSTRAINT "Connection_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "public"."Interest" (
"id" TEXT NOT NULL,
"name" TEXT NOT NULL,
CONSTRAINT "Interest_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "public"."Value" (
"id" TEXT NOT NULL,
"name" TEXT NOT NULL,
CONSTRAINT "Value_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "public"."CauseArea" (
"id" TEXT NOT NULL,
"name" TEXT NOT NULL,
CONSTRAINT "CauseArea_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "public"."ProfileConnection" (
"profileId" TEXT NOT NULL,
"connectionId" TEXT NOT NULL,
CONSTRAINT "ProfileConnection_pkey" PRIMARY KEY ("profileId","connectionId")
);
-- CreateTable
CREATE TABLE "public"."ProfileInterest" (
"profileId" TEXT NOT NULL,
"interestId" TEXT NOT NULL,
CONSTRAINT "ProfileInterest_pkey" PRIMARY KEY ("profileId","interestId")
);
-- CreateTable
CREATE TABLE "public"."ProfileValue" (
"profileId" TEXT NOT NULL,
"valueId" TEXT NOT NULL,
CONSTRAINT "ProfileValue_pkey" PRIMARY KEY ("profileId","valueId")
);
-- CreateTable
CREATE TABLE "public"."ProfileCauseArea" (
"profileId" TEXT NOT NULL,
"causeAreaId" TEXT NOT NULL,
CONSTRAINT "ProfileCauseArea_pkey" PRIMARY KEY ("profileId","causeAreaId")
);
-- CreateTable
CREATE TABLE "public"."PromptAnswer" (
"id" TEXT NOT NULL,
"profileId" TEXT NOT NULL,
"prompt" TEXT NOT NULL,
"answer" TEXT NOT NULL,
CONSTRAINT "PromptAnswer_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "public"."Account" (
"id" TEXT NOT NULL,
"userId" TEXT NOT NULL,
"type" TEXT NOT NULL,
"provider" TEXT NOT NULL,
"providerAccountId" TEXT NOT NULL,
"refresh_token" TEXT,
"access_token" TEXT,
"expires_at" INTEGER,
"token_type" TEXT,
"scope" TEXT,
"id_token" TEXT,
"session_state" TEXT,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "Account_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "public"."Session" (
"id" TEXT NOT NULL,
"sessionToken" TEXT NOT NULL,
"userId" TEXT NOT NULL,
"expires" TIMESTAMP(3) NOT NULL,
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" TIMESTAMP(3) NOT NULL,
CONSTRAINT "Session_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "public"."VerificationToken" (
"identifier" TEXT NOT NULL,
"token" TEXT NOT NULL,
"expires" TIMESTAMP(3) NOT NULL
);
-- CreateTable
CREATE TABLE "public"."Authenticator" (
"credentialID" TEXT NOT NULL,
"userId" TEXT NOT NULL,
"providerAccountId" TEXT NOT NULL,
"credentialPublicKey" TEXT NOT NULL,
"counter" INTEGER NOT NULL,
"credentialDeviceType" TEXT NOT NULL,
"credentialBackedUp" BOOLEAN NOT NULL,
"transports" TEXT,
CONSTRAINT "Authenticator_pkey" PRIMARY KEY ("userId","credentialID")
);
-- CreateIndex
CREATE UNIQUE INDEX "User_email_key" ON "public"."User"("email");
-- CreateIndex
CREATE UNIQUE INDEX "User_verificationToken_key" ON "public"."User"("verificationToken");
-- CreateIndex
CREATE UNIQUE INDEX "Profile_userId_key" ON "public"."Profile"("userId");
-- CreateIndex
CREATE UNIQUE INDEX "Connection_name_key" ON "public"."Connection"("name");
-- CreateIndex
CREATE UNIQUE INDEX "Interest_name_key" ON "public"."Interest"("name");
-- CreateIndex
CREATE UNIQUE INDEX "Value_name_key" ON "public"."Value"("name");
-- CreateIndex
CREATE UNIQUE INDEX "CauseArea_name_key" ON "public"."CauseArea"("name");
-- CreateIndex
CREATE UNIQUE INDEX "Account_provider_providerAccountId_key" ON "public"."Account"("provider", "providerAccountId");
-- CreateIndex
CREATE UNIQUE INDEX "Session_sessionToken_key" ON "public"."Session"("sessionToken");
-- CreateIndex
CREATE UNIQUE INDEX "VerificationToken_identifier_token_key" ON "public"."VerificationToken"("identifier", "token");
-- CreateIndex
CREATE UNIQUE INDEX "Authenticator_credentialID_key" ON "public"."Authenticator"("credentialID");
-- AddForeignKey
ALTER TABLE "public"."Profile" ADD CONSTRAINT "Profile_userId_fkey" FOREIGN KEY ("userId") REFERENCES "public"."User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "public"."ProfileConnection" ADD CONSTRAINT "ProfileConnection_connectionId_fkey" FOREIGN KEY ("connectionId") REFERENCES "public"."Connection"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "public"."ProfileConnection" ADD CONSTRAINT "ProfileConnection_profileId_fkey" FOREIGN KEY ("profileId") REFERENCES "public"."Profile"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "public"."ProfileInterest" ADD CONSTRAINT "ProfileInterest_interestId_fkey" FOREIGN KEY ("interestId") REFERENCES "public"."Interest"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "public"."ProfileInterest" ADD CONSTRAINT "ProfileInterest_profileId_fkey" FOREIGN KEY ("profileId") REFERENCES "public"."Profile"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "public"."ProfileValue" ADD CONSTRAINT "ProfileValue_profileId_fkey" FOREIGN KEY ("profileId") REFERENCES "public"."Profile"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "public"."ProfileValue" ADD CONSTRAINT "ProfileValue_valueId_fkey" FOREIGN KEY ("valueId") REFERENCES "public"."Value"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "public"."ProfileCauseArea" ADD CONSTRAINT "ProfileCauseArea_causeAreaId_fkey" FOREIGN KEY ("causeAreaId") REFERENCES "public"."CauseArea"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "public"."ProfileCauseArea" ADD CONSTRAINT "ProfileCauseArea_profileId_fkey" FOREIGN KEY ("profileId") REFERENCES "public"."Profile"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "public"."PromptAnswer" ADD CONSTRAINT "PromptAnswer_profileId_fkey" FOREIGN KEY ("profileId") REFERENCES "public"."Profile"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "public"."Account" ADD CONSTRAINT "Account_userId_fkey" FOREIGN KEY ("userId") REFERENCES "public"."User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "public"."Session" ADD CONSTRAINT "Session_userId_fkey" FOREIGN KEY ("userId") REFERENCES "public"."User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "public"."Authenticator" ADD CONSTRAINT "Authenticator_userId_fkey" FOREIGN KEY ("userId") REFERENCES "public"."User"("id") ON DELETE CASCADE ON UPDATE CASCADE;

View File

@@ -0,0 +1,14 @@
/*
Warnings:
- The values [Male,Female] on the enum `Gender` will be removed. If these variants are still used in the database, this will fail.
*/
-- AlterEnum
BEGIN;
CREATE TYPE "public"."Gender_new" AS ENUM ('Man', 'Woman', 'Other');
ALTER TABLE "public"."Profile" ALTER COLUMN "gender" TYPE "public"."Gender_new" USING ("gender"::text::"public"."Gender_new");
ALTER TYPE "public"."Gender" RENAME TO "Gender_old";
ALTER TYPE "public"."Gender_new" RENAME TO "Gender";
DROP TYPE "public"."Gender_old";
COMMIT;

View File

@@ -0,0 +1,24 @@
-- CreateTable
CREATE TABLE "public"."Book" (
"id" TEXT NOT NULL,
"name" TEXT NOT NULL,
CONSTRAINT "Book_pkey" PRIMARY KEY ("id")
);
-- CreateTable
CREATE TABLE "public"."ProfileBook" (
"profileId" TEXT NOT NULL,
"valueId" TEXT NOT NULL,
CONSTRAINT "ProfileBook_pkey" PRIMARY KEY ("profileId","valueId")
);
-- CreateIndex
CREATE UNIQUE INDEX "Book_name_key" ON "public"."Book"("name");
-- AddForeignKey
ALTER TABLE "public"."ProfileBook" ADD CONSTRAINT "ProfileBook_valueId_fkey" FOREIGN KEY ("valueId") REFERENCES "public"."Book"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- AddForeignKey
ALTER TABLE "public"."ProfileBook" ADD CONSTRAINT "ProfileBook_profileId_fkey" FOREIGN KEY ("profileId") REFERENCES "public"."Profile"("id") ON DELETE CASCADE ON UPDATE CASCADE;

View File

@@ -0,0 +1,3 @@
# Please do not edit this file manually
# It should be added in your version-control system (e.g., Git)
provider = "postgresql"

202
_old/prisma/schema.prisma Normal file
View File

@@ -0,0 +1,202 @@
generator client {
provider = "prisma-client-js"
}
datasource db {
provider = "postgresql"
url = env("DATABASE_URL")
}
model User {
id String @id @default(cuid())
name String?
email String? @unique
password String?
emailVerified DateTime?
verificationToken String? @unique
verificationTokenExpires DateTime?
image String?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
accounts Account[]
Authenticator Authenticator[]
profile Profile?
sessions Session[]
}
model Profile {
id String @id @default(cuid())
userId String @unique
location String?
description String?
contactInfo String?
birthYear Int?
occupation String?
gender Gender?
personalityType PersonalityType?
introversion Int?
conflictStyle ConflictStyle?
images String[]
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
causeAreas ProfileCauseArea[]
desiredConnections ProfileConnection[]
intellectualInterests ProfileInterest[]
coreValues ProfileValue[]
books ProfileBook[]
promptAnswers PromptAnswer[]
}
model Connection {
id String @id @default(cuid())
name String @unique
users ProfileConnection[]
}
model Interest {
id String @id @default(cuid())
name String @unique
users ProfileInterest[]
}
model Value {
id String @id @default(cuid())
name String @unique
users ProfileValue[]
}
model Book {
id String @id @default(cuid())
name String @unique
users ProfileBook[]
}
model CauseArea {
id String @id @default(cuid())
name String @unique
users ProfileCauseArea[]
}
model ProfileConnection {
profileId String
connectionId String
connection Connection @relation(fields: [connectionId], references: [id], onDelete: Cascade)
profile Profile @relation(fields: [profileId], references: [id], onDelete: Cascade)
@@id([profileId, connectionId])
}
model ProfileInterest {
profileId String
interestId String
interest Interest @relation(fields: [interestId], references: [id], onDelete: Cascade)
profile Profile @relation(fields: [profileId], references: [id], onDelete: Cascade)
@@id([profileId, interestId])
}
model ProfileValue {
profileId String
valueId String
value Value @relation(fields: [valueId], references: [id], onDelete: Cascade)
profile Profile @relation(fields: [profileId], references: [id], onDelete: Cascade)
@@id([profileId, valueId])
}
model ProfileBook {
profileId String
valueId String
value Book @relation(fields: [valueId], references: [id], onDelete: Cascade)
profile Profile @relation(fields: [profileId], references: [id], onDelete: Cascade)
@@id([profileId, valueId])
}
model ProfileCauseArea {
profileId String
causeAreaId String
causeArea CauseArea @relation(fields: [causeAreaId], references: [id], onDelete: Cascade)
profile Profile @relation(fields: [profileId], references: [id], onDelete: Cascade)
@@id([profileId, causeAreaId])
}
model PromptAnswer {
id String @id @default(cuid())
profileId String
prompt String
answer String
profile Profile @relation(fields: [profileId], references: [id], onDelete: Cascade)
}
model Account {
id String @id @default(cuid())
userId String
type String
provider String
providerAccountId String
refresh_token String?
access_token String?
expires_at Int?
token_type String?
scope String?
id_token String?
session_state String?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
@@unique([provider, providerAccountId])
}
model Session {
id String @id @default(cuid())
sessionToken String @unique
userId String
expires DateTime
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
}
model VerificationToken {
identifier String
token String
expires DateTime
@@unique([identifier, token])
}
model Authenticator {
credentialID String @unique
userId String
providerAccountId String
credentialPublicKey String
counter Int
credentialDeviceType String
credentialBackedUp Boolean
transports String?
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
@@id([userId, credentialID])
}
enum Gender {
Man
Woman
Other
}
enum PersonalityType {
Introvert
Extrovert
Ambivert
}
enum ConflictStyle {
Competing
Avoidant
Compromising
Accommodating
Collaborating
}

View File

@@ -16,6 +16,7 @@ async function main() {
bio: string; bio: string;
interests: string[]; interests: string[];
values: string[]; values: string[];
books: string[];
}; };
const profiles: ProfileBio[] = [ const profiles: ProfileBio[] = [
@@ -27,7 +28,8 @@ async function main() {
location: "Berlin, Germany", location: "Berlin, Germany",
bio: "Im passionate about understanding the limits and mechanics of human reasoning. I spend weekends dissecting papers on decision theory and evenings debating moral uncertainty. If you know your way around LessWrong and thought experiments, well get along.", bio: "Im passionate about understanding the limits and mechanics of human reasoning. I spend weekends dissecting papers on decision theory and evenings debating moral uncertainty. If you know your way around LessWrong and thought experiments, well get along.",
interests: ["Bayesian epistemology", "AI alignment", "Effective Altruism", "Meditation", "Game Theory"], interests: ["Bayesian epistemology", "AI alignment", "Effective Altruism", "Meditation", "Game Theory"],
values: ["Intellectualism", "Rationality", "Autonomy"] values: ["Intellectualism", "Rationality", "Autonomy"],
books: ["Daniel Kahneman - Thinking, Fast and Slow"]
}, },
{ {
name: "Marcus", name: "Marcus",
@@ -37,7 +39,8 @@ async function main() {
location: "San Francisco, USA", location: "San Francisco, USA",
bio: "Practicing instrumental rationality one well-calibrated belief at a time. Stoicism and startup life have taught me a lot about tradeoffs. Looking for someone who can argue in good faith and loves truth-seeking as much as I do.", bio: "Practicing instrumental rationality one well-calibrated belief at a time. Stoicism and startup life have taught me a lot about tradeoffs. Looking for someone who can argue in good faith and loves truth-seeking as much as I do.",
interests: ["Stoicism", "Predictive processing", "Rational fiction", "Startups", "Causal inference"], interests: ["Stoicism", "Predictive processing", "Rational fiction", "Startups", "Causal inference"],
values: ["Diplomacy", "Rationality", "Community"] values: ["Diplomacy", "Rationality", "Community"],
books: ["Daniel Kahneman - Thinking, Fast and Slow"]
}, },
{ {
name: "Aya", name: "Aya",
@@ -47,7 +50,8 @@ async function main() {
location: "Oxford, UK", location: "Oxford, UK",
bio: "My research focuses on metaethics and formal logic, but my heart belongs to moral philosophy. I think a lot about personhood, consciousness, and the ethics of future civilizations. Let's talk about Rawls or Parfit over tea.", bio: "My research focuses on metaethics and formal logic, but my heart belongs to moral philosophy. I think a lot about personhood, consciousness, and the ethics of future civilizations. Let's talk about Rawls or Parfit over tea.",
interests: ["Metaethics", "Consciousness", "Transhumanism", "Moral realism", "Formal logic"], interests: ["Metaethics", "Consciousness", "Transhumanism", "Moral realism", "Formal logic"],
values: ["Radical Honesty", "Structure", "Sufficiency"] values: ["Radical Honesty", "Structure", "Sufficiency"],
books: ["Daniel Kahneman - Thinking, Fast and Slow"]
}, },
{ {
name: "David", name: "David",
@@ -57,7 +61,8 @@ async function main() {
location: "Toronto, Canada", location: "Toronto, Canada",
bio: "Former humanities major turned quant. Still fascinated by existential risk, the philosophy of science, and how to stay sane in an uncertain world. I'm here to meet people who think weird is a compliment.", bio: "Former humanities major turned quant. Still fascinated by existential risk, the philosophy of science, and how to stay sane in an uncertain world. I'm here to meet people who think weird is a compliment.",
interests: ["Probability theory", "Longtermism", "Epistemic humility", "Futurology", "Meditation"], interests: ["Probability theory", "Longtermism", "Epistemic humility", "Futurology", "Meditation"],
values: ["Conservatism", "Ambition", "Idealism"] values: ["Conservatism", "Ambition", "Idealism"],
books: ["Daniel Kahneman - Thinking, Fast and Slow"]
}, },
{ {
name: "Mei", name: "Mei",
@@ -67,12 +72,14 @@ async function main() {
location: "Singapore", location: "Singapore",
bio: "Writing essays on intellectual humility, the philosophy of language, and how thinking styles shape our lives. I appreciate calm reasoning, rigorous curiosity, and the beauty of well-defined concepts. Let's try to model each other's minds.", bio: "Writing essays on intellectual humility, the philosophy of language, and how thinking styles shape our lives. I appreciate calm reasoning, rigorous curiosity, and the beauty of well-defined concepts. Let's try to model each other's minds.",
interests: ["Philosophy of language", "Bayesian reasoning", "Writing", "Dialectics", "Systems thinking"], interests: ["Philosophy of language", "Bayesian reasoning", "Writing", "Dialectics", "Systems thinking"],
values: ["Emotional Merging", "Sufficiency", "Pragmatism"] values: ["Emotional Merging", "Sufficiency", "Pragmatism"],
books: ["Daniel Kahneman - Thinking, Fast and Slow"]
} }
]; ];
const interests = new Set<string>(); const interests = new Set<string>();
const values = new Set<string>(); const values = new Set<string>();
const books = new Set<string>();
profiles.forEach(profile => { profiles.forEach(profile => {
profile.interests.forEach(v => interests.add(v)); profile.interests.forEach(v => interests.add(v));
@@ -82,8 +89,13 @@ async function main() {
profile.values.forEach(v => values.add(v)); profile.values.forEach(v => values.add(v));
}); });
profiles.forEach(profile => {
profile.books.forEach(v => books.add(v));
});
console.log('Interests:', [...interests]); console.log('Interests:', [...interests]);
console.log('Values:', [...values]); console.log('Values:', [...values]);
console.log('Books:', [...books]);
await prisma.interest.createMany({ await prisma.interest.createMany({
data: [...interests].map(v => ({name: v})), data: [...interests].map(v => ({name: v})),
@@ -95,6 +107,11 @@ async function main() {
skipDuplicates: true, skipDuplicates: true,
}); });
await prisma.book.createMany({
data: [...books].map(v => ({name: v})),
skipDuplicates: true,
});
await prisma.causeArea.createMany({ await prisma.causeArea.createMany({
data: [ data: [
{name: 'Climate Change'}, {name: 'Climate Change'},
@@ -106,9 +123,10 @@ async function main() {
await prisma.connection.createMany({ await prisma.connection.createMany({
data: [ data: [
{name: 'Debate Partner'}, // {name: 'Debate Partner'},
{name: 'Friendship'}, {name: 'Friendship'},
{name: 'Relationship'}, {name: 'Short-term relationship'},
{name: 'Long-term relationship'},
], ],
skipDuplicates: true, skipDuplicates: true,
}); });
@@ -117,6 +135,7 @@ async function main() {
// Get actual Interest & CauseArea objects // Get actual Interest & CauseArea objects
const allInterests = await prisma.interest.findMany(); const allInterests = await prisma.interest.findMany();
const allValues = await prisma.value.findMany(); const allValues = await prisma.value.findMany();
const allBooks = await prisma.book.findMany();
const allCauseAreas = await prisma.causeArea.findMany(); const allCauseAreas = await prisma.causeArea.findMany();
const allConnections = await prisma.connection.findMany(); const allConnections = await prisma.connection.findMany();
@@ -125,7 +144,7 @@ async function main() {
const profile = profiles[i % profiles.length]; const profile = profiles[i % profiles.length];
const user = await prisma.user.create({ const user = await prisma.user.create({
data: { data: {
email: `user${i + 1}@bayesbond.com`, email: `user${i + 1}@Compassmeet.com`,
name: profile.name, name: profile.name,
image: 'profile-pictures/57a821c0-cda0-4797-8654-f54f26fed414.jpg', image: 'profile-pictures/57a821c0-cda0-4797-8654-f54f26fed414.jpg',
profile: { profile: {
@@ -134,10 +153,10 @@ async function main() {
birthYear: 2025 - profile.age, birthYear: 2025 - profile.age,
introversion: profile.introversion, introversion: profile.introversion,
description: `[Dummy profile for demo purposes]\n${profile.bio}`, description: `[Dummy profile for demo purposes]\n${profile.bio}`,
gender: i % 2 === 0 ? 'Male' : 'Female', gender: i % 2 === 0 ? 'Man' : 'Woman',
personalityType: i % 3 === 0 ? 'Extrovert' : 'Introvert', personalityType: i % 3 === 0 ? 'Extrovert' : 'Introvert',
conflictStyle: 'Avoidant', conflictStyle: 'Avoidant',
contactInfo: `Email: user${i}@bayesbond.com\nPhone: +1 (123) 456-7890`, contactInfo: `Email: user${i}@Compassmeet.com\nPhone: +1 (123) 456-7890`,
occupation: profile.occupation, occupation: profile.occupation,
desiredConnections: { desiredConnections: {
create: [ create: [
@@ -154,6 +173,11 @@ async function main() {
.filter(e => (new Set(profile.values)).has(e.name)) .filter(e => (new Set(profile.values)).has(e.name))
.map(e => ({valueId: e.id})) .map(e => ({valueId: e.id}))
}, },
books: {
create: allBooks
.filter(e => (new Set(profile.books)).has(e.name))
.map(e => ({valueId: e.id}))
},
causeAreas: { causeAreas: {
create: [ create: [
{causeAreaId: allCauseAreas[i % allCauseAreas.length].id}, {causeAreaId: allCauseAreas[i % allCauseAreas.length].id},

View File

@@ -17,6 +17,9 @@ export default {
}, },
}, },
}, },
corePlugins: {
animation: true,
},
darkMode: 'class', // required for next-themes darkMode: 'class', // required for next-themes
plugins: [ ], plugins: [ ],
} satisfies Config; } satisfies Config;

View File

@@ -24,5 +24,8 @@
}, },
"include": ["next-env.d.ts", "**/*.ts", "**/*.tsx", ".next/types/**/*.ts"], "include": ["next-env.d.ts", "**/*.ts", "**/*.tsx", ".next/types/**/*.ts"],
"exclude": ["node_modules"], "exclude": ["node_modules"],
"types": ["jest", "@testing-library/jest-dom"],
"typeRoots": ["./node_modules/@types", "./types"],
"jsx": "react-jsx",
"sourceMap": true "sourceMap": true
} }

0
android/.aiexclude Normal file
View File

102
android/.gitignore vendored Normal file
View File

@@ -0,0 +1,102 @@
# Using Android gitignore template: https://github.com/github/gitignore/blob/HEAD/Android.gitignore
# Built application files
*.apk
*.aar
*.ap_
*.aab
# Files for the ART/Dalvik VM
*.dex
# Java class files
*.class
# Generated files
bin/
gen/
out/
# Uncomment the following line in case you need and you don't have the release build type files in your app
# release/
# Gradle files
.gradle/
build/
# Local configuration file (sdk path, etc)
local.properties
# Proguard folder generated by Eclipse
proguard/
# Log Files
*.log
# Android Studio Navigation editor temp files
.navigation/
# Android Studio captures folder
captures/
# IntelliJ
*.iml
.idea/workspace.xml
.idea/tasks.xml
.idea/gradle.xml
.idea/assetWizardSettings.xml
.idea/dictionaries
.idea/libraries
# Android Studio 3 in .gitignore file.
.idea/caches
.idea/modules.xml
# Comment next line if keeping position of elements in Navigation Editor is relevant for you
.idea/navEditor.xml
# Keystore files
# Uncomment the following lines if you do not want to check your keystore files in.
#*.jks
#*.keystore
# External native build folder generated in Android Studio 2.2 and later
.externalNativeBuild
.cxx/
# Google Services (e.g. APIs or Firebase)
# google-services.json
# Freeline
freeline.py
freeline/
freeline_project_description.json
# fastlane
fastlane/report.xml
fastlane/Preview.html
fastlane/screenshots
fastlane/test_output
fastlane/readme.md
# Version control
vcs.xml
# lint
lint/intermediates/
lint/generated/
lint/outputs/
lint/tmp/
# lint/reports/
# Android Profiling
*.hprof
# Cordova plugins for Capacitor
capacitor-cordova-android-plugins
# Copied web assets
app/src/main/assets/public
# Generated Config files
app/src/main/assets/capacitor.config.json
app/src/main/assets/capacitor.plugins.json
app/src/main/res/xml/config.xml
/app/release/

364
android/README.md Normal file
View File

@@ -0,0 +1,364 @@
# Compass Android WebView App
This folder contains the source code for the Android application of Compass.
A hybrid mobile app built with **Next.js (TypeScript)** frontend, **Firebase backend**, and wrapped as a **Capacitor WebView** for Android. In the future it may contain native code as well.
This document describes how to:
1. Build and run the web frontend and backend locally
2. Sync and build the Android WebView wrapper
3. Debug, sign, and publish the APK
4. Enable Google Sign-In and push notifications
---
## 1. Project Overview
The app is a Capacitor Android project that loads the local Next.js assests inside a WebView.
During development, it can instead load the local frontend (`http://10.0.2.2:3000`) and backend (`http://10.0.2.2:8088`).
Firebase handles authentication and push notifications.
Google Sign-In is supported natively in the WebView via the Capacitor Social Login plugin.
Project Structure
- `app/src/main/java/com/compass/app`: Contains the Java/Kotlin source code for the Android application.
- `app/src/main/res`: Contains the resources for the application, such as layouts, strings, and images.
- `app/build.gradle`: The Gradle build file for the Android application module.
- `build.gradle`: The top-level Gradle build file for the project.
- `AndroidManifest.xml`: The manifest file that describes essential information about the application.
### **Why Local Is the Default**
- **Performance:** Local assets load instantly, without network latency.
- **Reliability:** Works offline or in poor connectivity environments.
- **App Store policy compliance:** Apple and Google generally prefer that the main experience doesnt depend on a remote site (for security, review, and performance reasons).
- **Version consistency:** The web bundle is versioned with the app, ensuring no breaking updates outside your control.
When Remote (No Local Assets) Is sometimes Used
Loading from a **remote URL** (e.g. `https://compassmeet.com`) is **less common**, but seen in a few cases:
- **Internal enterprise apps** where the WebView just wraps an existing web portal.
- **Dynamic content** or **frequent updates** where pushing a new web build every time through app stores would be too slow.
- To leverage the low latency of ISR and SSR.
However, this approach requires:
- Careful handling of **CORS**, **SSL**, and **login/session** persistence.
- Compliance with **Google Play policies** (they may reject apps that are “just a webview of a website” unless theres meaningful native integration).
**A middle ground we use:**
- The app ships with **local assets** for core functionality.
- The app **fetches remote content or updates** (e.g., via Capacitor Live Updates, Ionic Appflow).
## 2. Prerequisites
### Required Software
| Tool | Version | Purpose |
| -------------- | ------- | ---------------------------------- |
| Node.js | 22+ | For building frontend/backend |
| yarn | latest | Package manager |
| Java | 21 | Required for Android Gradle plugin |
| Android Studio | latest | For building and signing APKs |
| Capacitor CLI | latest | Android bridge |
| OpenJDK | 21 | JDK for Gradle |
### Environment Setup
```bash
sudo apt install openjdk-21-jdk
sudo update-alternatives --config java
# Select Java 21
export JAVA_HOME=/usr/lib/jvm/java-21-openjdk-amd64
java -version
javac -version
````
---
## 3. Build and Run the Web App
```
yarn install
yarn build-web
```
### Local mode
If you want the webview to load from your local web version of Compass, run the web app.
In root directory:
```bash
export NEXT_PUBLIC_LOCAL_ANDROID=1
yarn dev # or prod
```
* Runs Next.js frontend at `http://localhost:3000`
* Runs backend at `http://10.0.2.2:8088`
### Deployed mode
If you want the webview to load from the deployed web version of Compass (like at www.compassmeet.com), no web app to run.
---
## 5. Android WebView App Setup
### Install dependencies
```
cd android
./gradlew clean
```
Sync web files and native plugins with Android, for offline fallback. In root:
```
export NEXT_PUBLIC_LOCAL_ANDROID=1 # if running your local web Compass
yarn build-web # if you made changes to web app
npx cap sync android
```
### Load from site
During local development, open Android Studio project and run the app on an emulator or your physical device. Note that right now you can't use a physical device for the local web version (`10.0.2.2:3000 time out` )
```
npx cap open android
```
Building the Application:
1. Open Android Studio.
2. Click on "Open an existing Android Studio project".
3. Navigate to the `android` folder in this repository and select it.
4. Wait for Android Studio to index the project and download any necessary dependencies.
5. Connect your Android device via USB or set up an Android emulator.
6. Click on the "Run" button (green play button) in Android Studio to build and run the application.
7. Select your device or emulator and click "OK".
8. The application should now build and launch on your device or emulator.
---
## 6. Building the APK
### From Android Studio
- If you want to generate a signed APK for release, go to "Build" > "Generate Signed Bundle / APK..." and follow the prompts.
- Make sure to test the application thoroughly on different devices and Android versions to ensure compatibility.
### Debug build
```bash
cd android
./gradlew assembleDebug
```
Outputs:
```
android/app/build/outputs/apk/debug/app-debug.apk
```
### Install on emulator
```bash
adb install -r app/build/outputs/apk/debug/app-debug.apk
```
### Release build (signed)
1. Generate a release keystore:
```bash
keytool -genkey -v -keystore release-key.keystore -alias compass \
-keyalg RSA -keysize 2048 -validity 10000
```
2. Add signing config to `android/app/build.gradle`
3. Build:
```bash
./gradlew assembleRelease
```
---
## 9. Debugging
Client logs from the emulator on Chrome can be accessed at:
```
chrome://inspect/#devices
```
Backend logs can be accessed from the output of `yarn prod / dev` like in the web application.
Java/Kotlin logs can be accessed via Android Studio's Logcat.
```
adb logcat | grep CompassApp
adb logcat | grep com.compassconnections.app
adb logcat | grep Capacitor
```
You can also add this inside `MainActivity.java`:
```java
webView.setWebChromeClient(new WebChromeClient() {
@Override
public boolean onConsoleMessage(ConsoleMessage consoleMessage) {
Log.d("WebView", consoleMessage.message());
return true;
}
});
```
---
## 10. Deploy to Play Store
1. Sign the release APK or AAB.
2. Verify package name matches Firebase settings (`com.compassconnections.app`).
3. Upload to Google Play Console.
4. Add Privacy Policy and content rating.
5. Submit for review.
---
## 11. Common Issues
| Problem | Cause | Fix |
| -------------------------------------- | -------------------------------------- | ------------------------------------------------------------------- |
| `INSTALL_FAILED_UPDATE_INCOMPATIBLE` | Old APK signed with different key | Uninstall old app first |
| `Account reauth failed [16]` | Missing or incorrect SHA-1 in Firebase | Re-add SHA-1 of keystore |
| App opens in Firefox | Missing `WebViewClient` override | Fix `shouldOverrideUrlLoading` |
| APK > 1 GB | Cached webpack artifacts included | Add `.next/` and `/public/cache` to `.gitignore` and build excludes |
---
## 13. Local Development Workflow
```bash
# Terminal 1
export NEXT_PUBLIC_LOCAL_ANDROID=1
yarn dev # or prod
# Terminal 2: start frontend
export NEXT_PUBLIC_LOCAL_ANDROID=1
yarn build-web # if you made changes to web app
npx cap sync android
# Run on emulator or device
```
---
## 14. Deployment Workflow
```bash
# 1. Build web app for production
yarn build-web
# 2. Sync assets to Android
npx cap sync android
# 3. Build signed release APK in Android Studio
```
---
## 15. Resources
* [Capacitor Docs](https://capacitorjs.com/docs)
* [Firebase Android Setup](https://firebase.google.com/docs/android/setup)
* [FCM HTTP API](https://firebase.google.com/docs/cloud-messaging/send-message)
* [Next.js Deployment](https://nextjs.org/docs/deployment)
# Useful Commands
- To build the project: `./gradlew assembleDebug`
- To run unit tests: `./gradlew test`
- To run instrumentation tests: `./gradlew connectedAndroidTest`
- To clean the project: `./gradlew clean`
- To install dependencies: Open Android Studio and it will handle dependencies automatically.
- To update dependencies: Modify the `build.gradle` files and sync the project in Android Studio.
- To generate a signed APK: Use the "Generate Signed Bundle / APK..." option in Android Studio.
- To lint the project: `./gradlew lint`
- To check for updates to the Android Gradle Plugin: `./gradlew dependencyUpdates`
- To run the application on a connected device or emulator: `./gradlew installDebug`
- To view the project structure: Use the "Project" view in Android Studio.
- To analyze the APK: `./gradlew analyzeRelease`
- To run ProGuard/R8: `./gradlew minifyRelease`
- To generate documentation: `./gradlew javadoc`
# One time setups
Was already done for Compass, so you only need to do the steps below if you create a project separated from Compass.
## Configure Firebase
### In Firebase Console
1. Add a **Web app** → obtain `firebaseConfig`
2. Add an **Android app**
* Package name: `com.compassconnections.app`
* Add your SHA-1 and SHA-256 fingerprints (see below)
* Download `google-services.json` and put it in:
```
android/app/google-services.json
```
### To get SHA-1 for debug keystore
```bash
keytool -list -v \
-keystore ~/.android/debug.keystore \
-alias androiddebugkey \
-storepass android \
-keypass android
```
Add both SHA-1 and SHA-256 to Firebase.
## 7. Google Sign-In (Web + Native)
In Firebase Console:
* Enable **Google** provider under *Authentication → Sign-in method*
* Add your **Android SHA-1**
* Add your **Web OAuth client ID**
In your code:
```ts
import { googleNativeLogin } from 'web/lib/service/android-push'
```
## 8. Push Notifications (FCM)
### Setup FCM
* Add Firebase Cloud Messaging to your project
* Include `google-services.json` under `android/app/`
* Add in `android/build.gradle`:
```gradle
classpath 'com.google.gms:google-services:4.3.15'
```
* Add at the bottom of `android/app/build.gradle`:
```gradle
apply plugin: 'com.google.gms.google-services'
```
### Test notification
```ts
const message = {
notification: {
title: "Test Notification",
body: "Hello from Firebase Admin SDK"
},
token: "..."
};
initAdmin()
await admin.messaging().send(message)
.then(response => console.log("Successfully sent message:", response))
.catch(error => console.error("Error sending message:", error));
```
---

3
android/app/.gitignore vendored Normal file
View File

@@ -0,0 +1,3 @@
/build/*
!/build/.npmkeep
/google-services.json

66
android/app/build.gradle Normal file
View File

@@ -0,0 +1,66 @@
apply plugin: 'com.android.application'
apply plugin: 'com.google.gms.google-services'
android {
namespace "com.compassconnections.app"
compileSdk 36
defaultConfig {
applicationId "com.compassconnections.app"
minSdkVersion rootProject.ext.minSdkVersion
targetSdkVersion rootProject.ext.targetSdkVersion
versionCode 13
versionName "1.1.2"
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
aaptOptions {
// Files and dirs to omit from the packaged assets dir, modified to accommodate modern web apps.
// Default: https://android.googlesource.com/platform/frameworks/base/+/282e181b58cf72b6ca770dc7ca5f91f135444502/tools/aapt/AaptAssets.cpp#61
ignoreAssetsPattern '!.svn:!.git:!.ds_store:!*.scc:.*:!CVS:!thumbs.db:!picasa.ini:!*~'
}
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}
}
repositories {
flatDir{
dirs '../capacitor-cordova-android-plugins/src/main/libs', 'libs'
}
}
dependencies {
implementation fileTree(include: ['*.jar'], dir: 'libs')
implementation "androidx.appcompat:appcompat:$androidxAppCompatVersion"
implementation "androidx.coordinatorlayout:coordinatorlayout:$androidxCoordinatorLayoutVersion"
implementation "androidx.core:core-splashscreen:$coreSplashScreenVersion"
implementation project(':capacitor-android')
testImplementation "junit:junit:$junitVersion"
androidTestImplementation "androidx.test.ext:junit:$androidxJunitVersion"
androidTestImplementation "androidx.test.espresso:espresso-core:$androidxEspressoCoreVersion"
implementation project(':capacitor-cordova-android-plugins')
// Import the Firebase BoM
implementation platform('com.google.firebase:firebase-bom:34.4.0')
// TODO: Add the dependencies for Firebase products you want to use
// When using the BoM, don't specify versions in Firebase dependencies
implementation 'com.google.firebase:firebase-analytics'
// Add the dependencies for any other desired Firebase products
// https://firebase.google.com/docs/android/setup#available-libraries
implementation 'com.google.android.gms:play-services-auth:21.4.0'
implementation 'com.google.firebase:firebase-auth:24.0.1'
}
apply from: 'capacitor.build.gradle'
try {
def servicesJSON = file('google-services.json')
if (servicesJSON.text) {
apply plugin: 'com.google.gms.google-services'
}
} catch(Exception e) {
logger.info("google-services.json not found, google-services plugin not applied. Push Notifications won't work")
}

View File

@@ -0,0 +1,23 @@
// DO NOT EDIT THIS FILE! IT IS GENERATED EACH TIME "capacitor update" IS RUN
android {
compileOptions {
sourceCompatibility JavaVersion.VERSION_21
targetCompatibility JavaVersion.VERSION_21
}
}
apply from: "../capacitor-cordova-android-plugins/cordova.variables.gradle"
dependencies {
implementation project(':capacitor-app')
implementation project(':capacitor-keyboard')
implementation project(':capacitor-push-notifications')
implementation project(':capacitor-status-bar')
implementation project(':capgo-capacitor-social-login')
}
if (hasProperty('postBuildExtras')) {
postBuildExtras()
}

21
android/app/proguard-rules.pro vendored Normal file
View File

@@ -0,0 +1,21 @@
# Add project specific ProGuard rules here.
# You can control the set of applied configuration files using the
# proguardFiles setting in build.gradle.
#
# For more details, see
# http://developer.android.com/guide/developing/tools/proguard.html
# If your project uses WebView with JS, uncomment the following
# and specify the fully qualified class name to the JavaScript interface
# class:
#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
# public *;
#}
# Uncomment this to preserve the line number information for
# debugging stack traces.
#-keepattributes SourceFile,LineNumberTable
# If you keep the line number information, uncomment this to
# hide the original source file name.
#-renamesourcefileattribute SourceFile

View File

@@ -0,0 +1,37 @@
{
"version": 3,
"artifactType": {
"type": "APK",
"kind": "Directory"
},
"applicationId": "com.compassconnections.app",
"variantName": "release",
"elements": [
{
"type": "SINGLE",
"filters": [],
"attributes": [],
"versionCode": 13,
"versionName": "1.1.2",
"outputFile": "app-release.apk"
}
],
"elementType": "File",
"baselineProfiles": [
{
"minApi": 28,
"maxApi": 30,
"baselineProfiles": [
"baselineProfiles/1/app-release.dm"
]
},
{
"minApi": 31,
"maxApi": 2147483647,
"baselineProfiles": [
"baselineProfiles/0/app-release.dm"
]
}
],
"minSdkVersionForDexing": 23
}

View File

@@ -0,0 +1,26 @@
package com.getcapacitor.myapp;
import static org.junit.Assert.*;
import android.content.Context;
import androidx.test.ext.junit.runners.AndroidJUnit4;
import androidx.test.platform.app.InstrumentationRegistry;
import org.junit.Test;
import org.junit.runner.RunWith;
/**
* Instrumented test, which will execute on an Android device.
*
* @see <a href="http://d.android.com/tools/testing">Testing documentation</a>
*/
@RunWith(AndroidJUnit4.class)
public class ExampleInstrumentedTest {
@Test
public void useAppContext() throws Exception {
// Context of the app under test.
Context appContext = InstrumentationRegistry.getInstrumentation().getTargetContext();
assertEquals("com.getcapacitor.app", appContext.getPackageName());
}
}

View File

@@ -0,0 +1,76 @@
<?xml version="1.1" encoding="utf-8" ?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools">
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:roundIcon="@mipmap/ic_launcher_round"
android:label="@string/app_name"
android:supportsRtl="true"
android:usesCleartextTraffic="true"
android:fitsSystemWindows="true"
android:theme="@style/AppTheme">
<activity
android:configChanges="orientation|keyboardHidden|keyboard|screenSize|locale|smallestScreenSize|screenLayout|uiMode|navigation"
android:name=".MainActivity"
android:label="@string/title_activity_main"
android:theme="@style/AppTheme.NoActionBarLaunch"
android:launchMode="singleTask"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
<!-- <intent-filter>-->
<!-- <action android:name="openapp" />-->
<!-- <category android:name="android.intent.category.DEFAULT" />-->
<!-- <category android:name="android.intent.category.BROWSABLE" />-->
<!-- <data android:scheme="com.compassconnections.app" android:host="details"/>-->
<!-- </intent-filter>-->
<!-- <intent-filter android:autoVerify="true">-->
<!-- <action android:name="android.intent.action.VIEW" />-->
<!-- <category android:name="android.intent.category.DEFAULT" />-->
<!-- <category android:name="android.intent.category.BROWSABLE" />-->
<!-- <data android:scheme="com.compassconnections.app" android:host="auth" />-->
<!-- </intent-filter>-->
</activity>
<!-- <service-->
<!-- android:name=".MyMessagingService"-->
<!-- android:exported="false">-->
<!-- <intent-filter>-->
<!-- <action android:name="com.google.firebase.MESSAGING_EVENT" />-->
<!-- </intent-filter>-->
<!--&lt;!&ndash; <meta-data&ndash;&gt;-->
<!--&lt;!&ndash; android:name="com.google.firebase.messaging.default_notification_channel_id"&ndash;&gt;-->
<!--&lt;!&ndash; android:value="@string/default_notification_channel_id" />&ndash;&gt;-->
<!-- </service>-->
<provider
android:name="androidx.core.content.FileProvider"
android:authorities="${applicationId}.fileprovider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data android:name="android.support.FILE_PROVIDER_PATHS" android:resource="@xml/file_paths" />
</provider>
</application>
<!-- Permissions -->
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
<uses-permission android:name="android.permission.POST_NOTIFICATIONS" />
<uses-permission android:name="android.permission.VIBRATE" />
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
<uses-permission android:name="com.google.android.gms.permission.AD_ID" tools:node="remove" />
<!-- Firebase Cloud Messaging -->
<permission android:name="${applicationId}.permission.C2D_MESSAGE" android:protectionLevel="signature" />
<uses-permission android:name="${applicationId}.permission.C2D_MESSAGE" />
<!-- Old, can be removed ?-->
<uses-permission android:name="com.google.android.c2dm.permission.RECEIVE" />
</manifest>

View File

@@ -0,0 +1,173 @@
package com.compassconnections.app;
import android.Manifest;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.os.Build;
import android.os.Bundle;
import android.util.Log;
import android.webkit.JavascriptInterface;
import android.webkit.WebSettings;
import android.webkit.WebView;
import androidx.activity.result.ActivityResultLauncher;
import androidx.activity.result.contract.ActivityResultContracts;
import androidx.core.content.ContextCompat;
import com.capacitorjs.plugins.pushnotifications.PushNotificationsPlugin;
import com.getcapacitor.BridgeActivity;
import com.getcapacitor.BridgeWebViewClient;
import com.getcapacitor.Plugin;
import com.getcapacitor.PluginHandle;
import org.json.JSONException;
import org.json.JSONObject;
import ee.forgr.capacitor.social.login.GoogleProvider;
import ee.forgr.capacitor.social.login.ModifiedMainActivityForSocialLoginPlugin;
import ee.forgr.capacitor.social.login.SocialLoginPlugin;
//import android.app.NotificationChannel;
//import android.app.NotificationManager;
//import android.os.Build;
//import com.google.firebase.messaging.RemoteMessage;
//import com.capacitorjs.plugins.pushnotifications.MessagingService;
//public class MyMessagingService extends MessagingService {
//
// @Override
// public void onMessageReceived(RemoteMessage remoteMessage) {
// // TODO(developer): Handle FCM messages here.
// // Not getting messages here? See why this may be: https://goo.gl/39bRNJ
// Log.d(TAG, "From: " + remoteMessage.getFrom());
//
// // Check if message contains a data payload.
// if (remoteMessage.getData().size() > 0) {
// Log.d(TAG, "Message data payload: " + remoteMessage.getData());
//
// if (/* Check if data needs to be processed by long running job */ true) {
// // For long-running tasks (10 seconds or more) use WorkManager.
// scheduleJob();
// } else {
// // Handle message within 10 seconds
// handleNow();
// }
//
// }
//
// // Check if message contains a notification payload.
// if (remoteMessage.getNotification() != null) {
// Log.d(TAG, "Message Notification Body: " + remoteMessage.getNotification().getBody());
// }
//
// // Also if you intend on generating your own notifications as a result of a received FCM
// // message, here is where that should be initiated. See sendNotification method below.
// }
//}
public class MainActivity extends BridgeActivity implements ModifiedMainActivityForSocialLoginPlugin {
// Declare this at class level
private final ActivityResultLauncher<String> requestPermissionLauncher =
registerForActivityResult(new ActivityResultContracts.RequestPermission(), isGranted -> {
if (isGranted) {
Log.i("CompassApp", "Permission granted");
// Permission granted you can show notifications
} else {
Log.i("CompassApp", "Permission denied");
// Permission denied handle gracefully
}
});
private void askNotificationPermission() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) { // API 33
if (ContextCompat.checkSelfPermission(this, Manifest.permission.POST_NOTIFICATIONS)
!= PackageManager.PERMISSION_GRANTED) {
// Permission not yet granted; request it
requestPermissionLauncher.launch(Manifest.permission.POST_NOTIFICATIONS);
}
}
}
public static class NativeBridge {
@JavascriptInterface
public boolean isNativeApp() {
return true;
}
}
@Override
protected void onNewIntent(Intent intent) {
super.onNewIntent(intent);
// String data = intent.getDataString();
String endpoint = intent.getStringExtra("endpoint");
Log.i("CompassApp", "onNewIntent called with endpoint: " + endpoint);
if (endpoint != null) {
Log.i("CompassApp", "redirecting to endpoint: " + endpoint);
try {
String payload = new JSONObject().put("endpoint", endpoint).toString();
Log.i("CompassApp", "Payload: " + payload);
bridge.getWebView().post(() -> bridge.getWebView().evaluateJavascript("bridgeRedirect(" + payload + ");", null));
} catch (JSONException e) {
Log.i("CompassApp", "Failed to encode JSON payload", e);
}
} else {
Log.i("CompassApp", "No relevant data");
}
}
@Override
public void onCreate(Bundle savedInstanceState) {
Log.i("CompassApp", "onCreate called");
super.onCreate(savedInstanceState);
WebView webView = this.bridge.getWebView();
webView.setWebViewClient(new BridgeWebViewClient(this.bridge));
WebView.setWebContentsDebuggingEnabled(true);
// Set a recognizable User-Agent (always reliable)
WebSettings settings = webView.getSettings();
settings.setUserAgentString(settings.getUserAgentString() + " CompassAppWebView");
settings.setJavaScriptEnabled(true);
webView.addJavascriptInterface(new NativeBridge(), "AndroidBridge");
registerPlugin(PushNotificationsPlugin.class);
// Initialize the Bridge with Push Notifications plugin
// this.init(savedInstanceState, new ArrayList<Class<? extends Plugin>>() {{
// add(com.getcapacitor.plugin.PushNotifications.class);
// }});
askNotificationPermission();
}
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode >= GoogleProvider.REQUEST_AUTHORIZE_GOOGLE_MIN && requestCode < GoogleProvider.REQUEST_AUTHORIZE_GOOGLE_MAX) {
PluginHandle pluginHandle = getBridge().getPlugin("SocialLogin");
if (pluginHandle == null) {
Log.i("CompassApp", "SocialLogin login handle is null");
return;
}
Plugin plugin = pluginHandle.getInstance();
if (!(plugin instanceof SocialLoginPlugin)) {
Log.i("CompassApp", "SocialLogin plugin instance is not SocialLoginPlugin");
return;
}
Log.i("CompassApp", "handleGoogleLoginIntent");
((SocialLoginPlugin) plugin).handleGoogleLoginIntent(requestCode, data);
}
}
// This function will never be called, leave it empty
@Override
public void IHaveModifiedTheMainActivityForTheUseWithSocialLoginPlugin() {
}
}

View File

@@ -0,0 +1,34 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:aapt="http://schemas.android.com/aapt"
android:width="108dp"
android:height="108dp"
android:viewportHeight="108"
android:viewportWidth="108">
<path
android:fillType="evenOdd"
android:pathData="M32,64C32,64 38.39,52.99 44.13,50.95C51.37,48.37 70.14,49.57 70.14,49.57L108.26,87.69L108,109.01L75.97,107.97L32,64Z"
android:strokeColor="#00000000"
android:strokeWidth="1">
<aapt:attr name="android:fillColor">
<gradient
android:endX="78.5885"
android:endY="90.9159"
android:startX="48.7653"
android:startY="61.0927"
android:type="linear">
<item
android:color="#44000000"
android:offset="0.0" />
<item
android:color="#00000000"
android:offset="1.0" />
</gradient>
</aapt:attr>
</path>
<path
android:fillColor="#FFFFFF"
android:fillType="nonZero"
android:pathData="M66.94,46.02L66.94,46.02C72.44,50.07 76,56.61 76,64L32,64C32,56.61 35.56,50.11 40.98,46.06L36.18,41.19C35.45,40.45 35.45,39.3 36.18,38.56C36.91,37.81 38.05,37.81 38.78,38.56L44.25,44.05C47.18,42.57 50.48,41.71 54,41.71C57.48,41.71 60.78,42.57 63.68,44.05L69.11,38.56C69.84,37.81 70.98,37.81 71.71,38.56C72.44,39.3 72.44,40.45 71.71,41.19L66.94,46.02ZM62.94,56.92C64.08,56.92 65,56.01 65,54.88C65,53.76 64.08,52.85 62.94,52.85C61.8,52.85 60.88,53.76 60.88,54.88C60.88,56.01 61.8,56.92 62.94,56.92ZM45.06,56.92C46.2,56.92 47.13,56.01 47.13,54.88C47.13,53.76 46.2,52.85 45.06,52.85C43.92,52.85 43,53.76 43,54.88C43,56.01 43.92,56.92 45.06,56.92Z"
android:strokeColor="#00000000"
android:strokeWidth="1" />
</vector>

View File

@@ -0,0 +1,170 @@
<?xml version="1.0" encoding="utf-8"?>
<vector xmlns:android="http://schemas.android.com/apk/res/android"
android:width="108dp"
android:height="108dp"
android:viewportHeight="108"
android:viewportWidth="108">
<path
android:fillColor="#26A69A"
android:pathData="M0,0h108v108h-108z" />
<path
android:fillColor="#00000000"
android:pathData="M9,0L9,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M19,0L19,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M29,0L29,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M39,0L39,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M49,0L49,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M59,0L59,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M69,0L69,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M79,0L79,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M89,0L89,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M99,0L99,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M0,9L108,9"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M0,19L108,19"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M0,29L108,29"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M0,39L108,39"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M0,49L108,49"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M0,59L108,59"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M0,69L108,69"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M0,79L108,79"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M0,89L108,89"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M0,99L108,99"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M19,29L89,29"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M19,39L89,39"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M19,49L89,49"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M19,59L89,59"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M19,69L89,69"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M19,79L89,79"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M29,19L29,89"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M39,19L39,89"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M49,19L49,89"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M59,19L59,89"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M69,19L69,89"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
<path
android:fillColor="#00000000"
android:pathData="M79,19L79,89"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8" />
</vector>

View File

@@ -0,0 +1,12 @@
<?xml version="1.0" encoding="utf-8"?>
<androidx.coordinatorlayout.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity">
<WebView
android:layout_width="match_parent"
android:layout_height="match_parent" />
</androidx.coordinatorlayout.widget.CoordinatorLayout>

View File

@@ -0,0 +1,9 @@
<?xml version="1.0" encoding="utf-8"?>
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
<background>
<inset android:drawable="@mipmap/ic_launcher_background" android:inset="16.7%" />
</background>
<foreground>
<inset android:drawable="@mipmap/ic_launcher_foreground" android:inset="16.7%" />
</foreground>
</adaptive-icon>

View File

@@ -0,0 +1,9 @@
<?xml version="1.0" encoding="utf-8"?>
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
<background>
<inset android:drawable="@mipmap/ic_launcher_background" android:inset="16.7%" />
</background>
<foreground>
<inset android:drawable="@mipmap/ic_launcher_foreground" android:inset="16.7%" />
</foreground>
</adaptive-icon>

View File

@@ -0,0 +1,4 @@
<?xml version="1.0" encoding="utf-8"?>
<resources>
<color name="ic_launcher_background">#FFFFFF</color>
</resources>

View File

@@ -0,0 +1,7 @@
<?xml version='1.0' encoding='utf-8'?>
<resources>
<string name="app_name">Compass</string>
<string name="title_activity_main">Compass</string>
<string name="package_name">com.compassconnections.app</string>
<string name="custom_url_scheme">com.compassconnections.app</string>
</resources>

View File

@@ -0,0 +1,25 @@
<?xml version="1.0" encoding="utf-8"?>
<resources>
<!-- Base application theme. -->
<style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar">
<!-- Customize your theme here. -->
<item name="colorPrimary">@color/colorPrimary</item>
<item name="colorPrimaryDark">@color/colorPrimaryDark</item>
<item name="colorAccent">@color/colorAccent</item>
<item name="android:windowTranslucentStatus">true</item>
</style>
<style name="AppTheme.NoActionBar" parent="Theme.AppCompat.DayNight.NoActionBar">
<item name="windowActionBar">false</item>
<item name="windowNoTitle">true</item>
<item name="android:background">@null</item>
<item name="android:windowTranslucentStatus">true</item>
<item name="android:windowTranslucentNavigation">true</item>
</style>
<style name="AppTheme.NoActionBarLaunch" parent="Theme.SplashScreen">
<item name="android:background">@drawable/splash</item>
</style>
</resources>

View File

@@ -0,0 +1,5 @@
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<external-path name="my_images" path="." />
<cache-path name="my_cache_images" path="." />
</paths>

Some files were not shown because too many files have changed in this diff Show More