solid-panes 4.2.6 → 4.4.1-test.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (143) hide show
  1. package/README.md +45 -1
  2. package/dist/0314353e28ce6e5539af.svg +9 -0
  3. package/dist/04567ff683933c35c465.png +0 -0
  4. package/dist/10163fd9b5a0e00d63a0.png +0 -0
  5. package/dist/1234dcb2eec2e45f252b.png +0 -0
  6. package/dist/20899934157df4db56cb.png +0 -0
  7. package/dist/33760bf79f097f449da5.png +0 -0
  8. package/dist/4cceba29ab33b1ddd9bb.svg +6 -0
  9. package/dist/578d2b6ed32e7624164e.png +0 -0
  10. package/dist/5f62a5b2b7e99b9640c7.png +0 -0
  11. package/dist/6525766ecd288ec60129.png +0 -0
  12. package/dist/7800be6f6c4b5b0f4f20.png +0 -0
  13. package/dist/7b7538c6f6b317968009.svg +9 -0
  14. package/dist/92d03142abe6efc0b42d.svg +6 -0
  15. package/dist/976473cf5fe24d657d4b.png +0 -0
  16. package/dist/RDFXMLPane.css +70 -0
  17. package/dist/RDFXMLPane.d.ts +13 -0
  18. package/dist/RDFXMLPane.d.ts.map +1 -0
  19. package/dist/RDFXMLPane.js +46 -4
  20. package/dist/bda84f59e7216675a208.png +0 -0
  21. package/dist/cd68e8f3990ba8b2139e.png +0 -0
  22. package/dist/dashboard/basicPreferences.d.ts.map +1 -1
  23. package/dist/dashboard/basicPreferences.js +1 -0
  24. package/dist/dashboard/dashboardPane.d.ts.map +1 -1
  25. package/dist/dashboard/dashboardPane.js +9 -3
  26. package/dist/dashboard/homepage.d.ts +1 -1
  27. package/dist/dashboard/homepage.d.ts.map +1 -1
  28. package/dist/dashboard/homepage.js +5 -35
  29. package/dist/dataContentPane.css +271 -0
  30. package/dist/dataContentPane.d.ts +14 -0
  31. package/dist/dataContentPane.d.ts.map +1 -0
  32. package/dist/dataContentPane.js +68 -101
  33. package/dist/defaultPane.css +97 -0
  34. package/dist/defaultPane.d.ts +14 -0
  35. package/dist/defaultPane.d.ts.map +1 -0
  36. package/dist/defaultPane.js +9 -2
  37. package/dist/e7074a7e2cb69e51cfd3.png +0 -0
  38. package/dist/f3772696fb7ee53c23d8.png +0 -0
  39. package/dist/form/formPane.css +120 -0
  40. package/dist/form/pane.d.ts +4 -0
  41. package/dist/form/pane.d.ts.map +1 -0
  42. package/dist/form/pane.js +120 -54
  43. package/dist/home/homePane.d.ts.map +1 -1
  44. package/dist/home/homePane.js +2 -0
  45. package/dist/humanReadablePane.css +129 -0
  46. package/dist/humanReadablePane.d.ts +13 -0
  47. package/dist/humanReadablePane.d.ts.map +1 -0
  48. package/dist/humanReadablePane.js +53 -16
  49. package/dist/icons/clock.svg +7 -0
  50. package/dist/icons/comment.svg +6 -0
  51. package/dist/icons/dashboard.svg +9 -0
  52. package/dist/icons/downArrow.svg +6 -0
  53. package/dist/icons/folder.svg +6 -0
  54. package/dist/icons/friends.svg +9 -0
  55. package/dist/icons/help.svg +8 -0
  56. package/dist/icons/iconHelper.d.ts +2 -0
  57. package/dist/icons/iconHelper.d.ts.map +1 -0
  58. package/dist/icons/iconHelper.js +23 -0
  59. package/dist/icons/menu.svg +8 -0
  60. package/dist/icons/person.svg +6 -0
  61. package/dist/icons/personInCircle.svg +8 -0
  62. package/dist/icons/sharing.svg +10 -0
  63. package/dist/icons/signOut.svg +8 -0
  64. package/dist/icons/signUp.svg +9 -0
  65. package/dist/icons/star.svg +3 -0
  66. package/dist/imagePane.css +4 -0
  67. package/dist/imagePane.d.ts +12 -0
  68. package/dist/imagePane.d.ts.map +1 -0
  69. package/dist/imagePane.js +19 -21
  70. package/dist/index.d.ts +5 -4
  71. package/dist/index.d.ts.map +1 -1
  72. package/dist/index.js +19 -4
  73. package/dist/internal/internalPane.css +14 -0
  74. package/dist/internal/internalPane.d.ts +1 -0
  75. package/dist/internal/internalPane.d.ts.map +1 -1
  76. package/dist/internal/internalPane.js +6 -6
  77. package/dist/mainPage/footer.d.ts +14 -2
  78. package/dist/mainPage/footer.d.ts.map +1 -1
  79. package/dist/mainPage/footer.js +21 -13
  80. package/dist/mainPage/header.d.ts +16 -1
  81. package/dist/mainPage/header.d.ts.map +1 -1
  82. package/dist/mainPage/header.js +179 -61
  83. package/dist/mainPage/index.d.ts +16 -1
  84. package/dist/mainPage/index.d.ts.map +1 -1
  85. package/dist/mainPage/index.js +61 -7
  86. package/dist/mainPage/menu.css +243 -0
  87. package/dist/mainPage/menu.d.ts +7 -0
  88. package/dist/mainPage/menu.d.ts.map +1 -0
  89. package/dist/mainPage/menu.js +436 -0
  90. package/dist/n3Pane.css +49 -0
  91. package/dist/n3Pane.d.ts +13 -0
  92. package/dist/n3Pane.d.ts.map +1 -0
  93. package/dist/n3Pane.js +36 -4
  94. package/dist/outline/context.d.ts +2 -2
  95. package/dist/outline/context.d.ts.map +1 -1
  96. package/dist/outline/context.js +5 -2
  97. package/dist/outline/manager.css +12 -14
  98. package/dist/outline/manager.js +172 -82
  99. package/dist/outline/userInput.js +6 -3
  100. package/dist/pad/padPane.css +40 -0
  101. package/dist/pad/padPane.d.ts +1 -0
  102. package/dist/pad/padPane.d.ts.map +1 -1
  103. package/dist/pad/padPane.js +33 -22
  104. package/dist/playlist/playlistPane.js +2 -6
  105. package/dist/profileUtils/ownerProfile.d.ts +5 -0
  106. package/dist/profileUtils/ownerProfile.d.ts.map +1 -0
  107. package/dist/profileUtils/ownerProfile.js +84 -0
  108. package/dist/registerPanes.js +12 -12
  109. package/dist/schedule/schedulePane.css +294 -0
  110. package/dist/schedule/schedulePane.d.ts +23 -0
  111. package/dist/schedule/schedulePane.d.ts.map +1 -0
  112. package/dist/schedule/schedulePane.js +161 -61
  113. package/dist/slideshow/slideshowPane.js +1 -1
  114. package/dist/social/editProfileDetails.d.ts +19 -0
  115. package/dist/social/editProfileDetails.d.ts.map +1 -0
  116. package/dist/social/editProfileDetails.js +362 -0
  117. package/dist/social/icons.d.ts +7 -0
  118. package/dist/social/icons.d.ts.map +1 -0
  119. package/dist/social/icons.js +95 -0
  120. package/dist/social/socialPane.css +1464 -0
  121. package/dist/social/socialPane.d.ts +30 -0
  122. package/dist/social/socialPane.d.ts.map +1 -0
  123. package/dist/social/socialPane.js +651 -0
  124. package/dist/social/socialSections.d.ts +77 -0
  125. package/dist/social/socialSections.d.ts.map +1 -0
  126. package/dist/social/socialSections.js +393 -0
  127. package/dist/social/spinner.d.ts +3 -0
  128. package/dist/social/spinner.d.ts.map +1 -0
  129. package/dist/social/spinner.js +13 -0
  130. package/dist/social/triage.d.ts +17 -0
  131. package/dist/social/triage.d.ts.map +1 -0
  132. package/dist/social/triage.js +79 -0
  133. package/dist/solid-panes.js +49544 -18164
  134. package/dist/solid-panes.js.map +1 -1
  135. package/dist/solid-panes.min.js +3884 -240
  136. package/dist/solid-panes.min.js.map +1 -1
  137. package/dist/{style → styles}/tabbedtab.css +0 -281
  138. package/dist/styles/utilities.css +5 -0
  139. package/dist/tabbed/tabbedPane.d.ts.map +1 -1
  140. package/dist/tabbed/tabbedPane.js +2 -0
  141. package/dist/versionInfo.js +14 -14
  142. package/package.json +37 -31
  143. package/dist/socialPane.js +0 -430
@@ -0,0 +1,77 @@
1
+ import { DataBrowserContext } from 'pane-registry';
2
+ import { Statement, NamedNode } from 'rdflib';
3
+ import { FriendshipTriage } from './triage';
4
+ export type ViewerMode = 'owner' | 'authenticated' | 'anonymous';
5
+ export type HeaderControls = {
6
+ canEdit: boolean;
7
+ viewerMode: ViewerMode;
8
+ showAddFriendAction?: boolean;
9
+ };
10
+ export type SocialHeaderElement = HTMLElement & {
11
+ refreshSocialHeader?: (controls: HeaderControls) => void;
12
+ };
13
+ export type HeaderStats = {
14
+ friendCount: number;
15
+ mutualFriendCount: number | null;
16
+ onSelectFriends?: () => void;
17
+ onSelectMutual?: () => void;
18
+ };
19
+ export type HeaderProfileData = {
20
+ imageUrl?: string;
21
+ name?: string;
22
+ jobTitle?: string;
23
+ organization?: string;
24
+ location?: string | null;
25
+ } | null;
26
+ export type FriendRowRenderers = {
27
+ renderSupportingInfo: (target: NamedNode, renderDom: HTMLDocument) => HTMLElement | null;
28
+ renderNameSuffix: (target: NamedNode, renderDom: HTMLDocument) => HTMLElement | null;
29
+ };
30
+ export declare function createHeaderSection(context: DataBrowserContext, subject: NamedNode, controls: HeaderControls, stats: HeaderStats, getProfileData: () => HeaderProfileData): SocialHeaderElement;
31
+ export declare function createMutualSection(options: {
32
+ dom: HTMLDocument;
33
+ subject: NamedNode;
34
+ familiar: string;
35
+ me: NamedNode;
36
+ meUri: string | null;
37
+ incoming: boolean | NamedNode[] | undefined;
38
+ outgoing: boolean | NamedNode[] | undefined;
39
+ editable: boolean;
40
+ profile: NamedNode | null;
41
+ knows: NamedNode;
42
+ mutualConnections: NamedNode[];
43
+ link: (contents: Node, uri: string | null | undefined) => Node;
44
+ text: (value: string) => Text;
45
+ buildCheckboxForm: (label: string | Node, statement: Statement, state: boolean, options?: {
46
+ disabled?: boolean;
47
+ disabledTitle?: string;
48
+ }) => HTMLElement;
49
+ renderSupportingInfo: FriendRowRenderers['renderSupportingInfo'];
50
+ renderNameSuffix: FriendRowRenderers['renderNameSuffix'];
51
+ }): {
52
+ section: HTMLElement;
53
+ content: HTMLElement;
54
+ refreshMutualFriends: () => void;
55
+ };
56
+ export declare function createAllFriendsSection(options: {
57
+ dom: HTMLDocument;
58
+ subject: NamedNode;
59
+ profile: NamedNode | null;
60
+ editable: boolean;
61
+ renderSupportingInfo: (target: NamedNode, renderDom: HTMLDocument) => HTMLElement | null;
62
+ renderNameSuffix: (target: NamedNode, renderDom: HTMLDocument) => HTMLElement | null;
63
+ }): {
64
+ section: HTMLElement;
65
+ mainTable: HTMLTableElement;
66
+ friendsList: HTMLElement;
67
+ };
68
+ export declare function createRequestsSection(options: {
69
+ dom: HTMLDocument;
70
+ triage: FriendshipTriage;
71
+ renderSupportingInfo: (target: NamedNode, renderDom: HTMLDocument) => HTMLElement | null;
72
+ renderNameSuffix: (target: NamedNode, renderDom: HTMLDocument) => HTMLElement | null;
73
+ }): {
74
+ section: HTMLElement;
75
+ refreshRequests: (triage: FriendshipTriage) => void;
76
+ };
77
+ //# sourceMappingURL=socialSections.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"socialSections.d.ts","sourceRoot":"","sources":["../../src/social/socialSections.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,393 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.createAllFriendsSection = createAllFriendsSection;
7
+ exports.createHeaderSection = createHeaderSection;
8
+ exports.createMutualSection = createMutualSection;
9
+ exports.createRequestsSection = createRequestsSection;
10
+ var _profilePane = require("profile-pane");
11
+ var _rdflib = require("rdflib");
12
+ var _solidUi = require("solid-ui");
13
+ var _editProfileDetails = require("./editProfileDetails");
14
+ var _icons = require("./icons");
15
+ function createHeaderSection(context, subject, controls, stats, getProfileData) {
16
+ const dom = context.dom;
17
+ const kb = context.session.store;
18
+ const header = dom.createElement('header');
19
+ header.className = 'social-pane__header';
20
+ let headerControls = controls;
21
+ const renderHeader = function () {
22
+ header.replaceChildren();
23
+ const profileData = getProfileData();
24
+ const headerContent = dom.createElement('div');
25
+ headerContent.className = 'social-pane__header-content';
26
+ const headerActions = dom.createElement('div');
27
+ headerActions.className = 'social-pane__header-actions profile__actions profile__heading-actions';
28
+ if (headerControls.canEdit) {
29
+ // Hidden for now because the social pane header edit control does not match the new design.
30
+ // Revisit later if we decide to restore profile-link editing here.
31
+ } else if (headerControls.viewerMode === 'authenticated' && headerControls.showAddFriendAction) {
32
+ headerActions.classList.add('social-pane__header-actions--friend');
33
+ const addToFriendsButton = (0, _profilePane.createAddMeToYourFriendsButton)(subject, context);
34
+ addToFriendsButton.classList.add('flex-center');
35
+ headerActions.appendChild(addToFriendsButton);
36
+ }
37
+ header.appendChild(headerContent);
38
+ const headerMedia = dom.createElement('div');
39
+ headerMedia.className = 'social-pane__header-media';
40
+ headerContent.appendChild(headerMedia);
41
+ if (profileData) {
42
+ headerMedia.appendChild(createImage(dom, profileData.imageUrl, profileData.name));
43
+ }
44
+ const headerDetails = dom.createElement('div');
45
+ headerDetails.className = 'social-pane__header-details';
46
+ headerContent.appendChild(headerDetails);
47
+ if (headerActions.childNodes.length > 0) {
48
+ headerContent.appendChild(headerActions);
49
+ }
50
+ const headerSummary = dom.createElement('div');
51
+ headerSummary.classList.add('social-pane__header-summary', 'flex-column');
52
+ headerDetails.appendChild(headerSummary);
53
+ const name = profileData?.name || _solidUi.utils.label(subject);
54
+ const h1 = dom.createElement('h1');
55
+ h1.classList.add('social-pane__header-name');
56
+ h1.appendChild(dom.createTextNode(name));
57
+ headerSummary.appendChild(h1);
58
+ const jobAndOrganization = [profileData?.jobTitle, profileData?.organization].filter(Boolean).join(' | ');
59
+ if (jobAndOrganization) {
60
+ const jobLine = dom.createElement('div');
61
+ jobLine.className = 'social-pane__header-job-org';
62
+ jobLine.textContent = jobAndOrganization;
63
+ headerSummary.appendChild(jobLine);
64
+ }
65
+ if (profileData?.location) {
66
+ const locationLine = dom.createElement('div');
67
+ locationLine.className = 'social-pane__header-location';
68
+ const locationIconSpan = dom.createElement('span');
69
+ locationIconSpan.classList.add('social-pane__header-location-icon', 'inline-flex-row');
70
+ locationIconSpan.innerHTML = _icons.locationIcon;
71
+ locationLine.appendChild(locationIconSpan);
72
+ locationLine.appendChild(dom.createTextNode(profileData.location));
73
+ headerSummary.appendChild(locationLine);
74
+ }
75
+ const statsRow = dom.createElement('div');
76
+ statsRow.className = 'social-pane__header-stats';
77
+ const friendCount = dom.createElement('button');
78
+ friendCount.className = 'social-pane__header-stat';
79
+ friendCount.type = 'button';
80
+ const friendCountLabel = dom.createElement('span');
81
+ friendCountLabel.className = 'social-pane__header-stat-label';
82
+ friendCountLabel.textContent = `${stats.friendCount} friend${stats.friendCount === 1 ? '' : 's'}`;
83
+ friendCount.appendChild(friendCountLabel);
84
+ if (stats.onSelectFriends) {
85
+ friendCount.addEventListener('click', stats.onSelectFriends);
86
+ }
87
+ statsRow.appendChild(friendCount);
88
+ if (typeof stats.mutualFriendCount === 'number') {
89
+ const mutualCount = dom.createElement('button');
90
+ mutualCount.className = 'social-pane__header-stat';
91
+ mutualCount.type = 'button';
92
+ const mutualCountLabel = dom.createElement('span');
93
+ mutualCountLabel.className = 'social-pane__header-stat-label';
94
+ mutualCountLabel.textContent = `${stats.mutualFriendCount} mutual friend${stats.mutualFriendCount === 1 ? '' : 's'}`;
95
+ mutualCount.appendChild(mutualCountLabel);
96
+ if (stats.onSelectMutual) {
97
+ mutualCount.addEventListener('click', stats.onSelectMutual);
98
+ }
99
+ statsRow.appendChild(mutualCount);
100
+ }
101
+ headerSummary.appendChild(statsRow);
102
+ (0, _editProfileDetails.appendProfileLinks)(headerMedia, dom, kb, subject);
103
+ };
104
+ header.refreshSocialHeader = function (nextControls) {
105
+ headerControls = nextControls;
106
+ renderHeader();
107
+ };
108
+ renderHeader();
109
+ return header;
110
+ }
111
+ function createImage(dom, src, alt = '') {
112
+ if (src) {
113
+ const img = dom.createElement('img');
114
+ img.className = 'social-pane__header-hero';
115
+ img.src = src;
116
+ img.alt = alt;
117
+ img.width = 180;
118
+ img.height = 180;
119
+ img.loading = 'eager';
120
+ return img;
121
+ }
122
+ const fallback = dom.createElement('div');
123
+ fallback.className = 'social-pane__header-hero-alt flex-center';
124
+ fallback.setAttribute('role', 'img');
125
+ fallback.setAttribute('aria-label', alt);
126
+ fallback.tabIndex = 0;
127
+ const icon = dom.createElement('span');
128
+ icon.classList.add('social-pane__header-hero-icon', 'inline-flex-row');
129
+ icon.innerHTML = _icons.personInCircleIcon;
130
+ fallback.appendChild(icon);
131
+ return fallback;
132
+ }
133
+ function createMutualSection(options) {
134
+ const {
135
+ dom,
136
+ subject,
137
+ familiar,
138
+ me,
139
+ meUri,
140
+ incoming,
141
+ outgoing,
142
+ editable,
143
+ profile,
144
+ knows,
145
+ mutualConnections,
146
+ link,
147
+ text,
148
+ buildCheckboxForm,
149
+ renderSupportingInfo,
150
+ renderNameSuffix
151
+ } = options;
152
+ let refreshMutualFriends = function () {};
153
+ const mutualSection = dom.createElement('section');
154
+ mutualSection.className = 'social-pane__mutual-friends social-primary__panel';
155
+ mutualSection.id = 'social-panel-mutual';
156
+ mutualSection.setAttribute('role', 'tabpanel');
157
+ mutualSection.setAttribute('aria-labelledby', 'social-tab-mutual');
158
+ const mutualContent = mutualSection.appendChild(dom.createElement('div'));
159
+ mutualContent.classList.add('social-main', 'social-main--mutual', 'flex-column');
160
+ const relationshipSummary = dom.createElement('div');
161
+ relationshipSummary.classList.add('social-mutual-summary', 'flex-column');
162
+ mutualContent.appendChild(relationshipSummary);
163
+ const createRelationshipLine = function () {
164
+ const line = relationshipSummary.appendChild(dom.createElement('div'));
165
+ line.classList.add('social-mutual-summary-line');
166
+ return line;
167
+ };
168
+ const youAndThem = function () {
169
+ const line = createRelationshipLine();
170
+ line.appendChild(link(text('You'), meUri));
171
+ line.appendChild(text(' and '));
172
+ line.appendChild(link(text(familiar), subject.uri));
173
+ return line;
174
+ };
175
+ if (!incoming) {
176
+ if (!outgoing) {
177
+ const line = youAndThem();
178
+ line.appendChild(text(' have not said you know each other.'));
179
+ } else {
180
+ const line = createRelationshipLine();
181
+ line.appendChild(link(text('You'), meUri));
182
+ line.appendChild(text(' know '));
183
+ line.appendChild(link(text(familiar), subject.uri));
184
+ line.appendChild(text(' (unconfirmed).'));
185
+ }
186
+ } else if (!outgoing) {
187
+ relationshipSummary.classList.add('social-mutual-summary--confirm');
188
+ const incomingLine = relationshipSummary.appendChild(dom.createElement('div'));
189
+ incomingLine.classList.add('social-mutual-summary-line');
190
+ incomingLine.appendChild(link(text(familiar), subject.uri));
191
+ incomingLine.appendChild(text(' knows '));
192
+ incomingLine.appendChild(link(text('you'), meUri));
193
+ incomingLine.appendChild(text(' (unconfirmed).'));
194
+ } else {
195
+ const line = youAndThem();
196
+ line.appendChild(text(' say you know each other.'));
197
+ }
198
+ const shouldShowCheckboxPreview = Boolean(incoming) && !outgoing;
199
+ if (shouldShowCheckboxPreview) {
200
+ const confirmLabel = dom.createElement('span');
201
+ confirmLabel.className = 'social-mutual-confirm-prompt';
202
+ confirmLabel.appendChild(text('Confirm you know '));
203
+ confirmLabel.appendChild(link(text(familiar), subject.uri));
204
+ const relationshipForm = buildCheckboxForm(confirmLabel, new _rdflib.Statement(me, knows, subject, profile ?? undefined), Boolean(outgoing), {
205
+ disabled: !editable,
206
+ disabledTitle: !editable ? 'Your profile is not editable' : undefined
207
+ });
208
+ relationshipForm.classList.add('social-mutual-checkbox-form');
209
+ mutualContent.appendChild(relationshipForm);
210
+ }
211
+ if (mutualConnections.length) {
212
+ const mutualFriendsTable = mutualContent.appendChild(dom.createElement('table'));
213
+ mutualFriendsTable.className = 'social-main social-friends-list social-friends-grid';
214
+ const createMutualRow = function (target) {
215
+ return _solidUi.widgets.personTR(dom, _solidUi.ns.foaf('knows'), target, {
216
+ renderSupportingInfo,
217
+ renderNameSuffix
218
+ });
219
+ };
220
+ refreshMutualFriends = function () {
221
+ const sortedMutualConnections = [...mutualConnections].sort((left, right) => {
222
+ const leftLabel = _solidUi.utils.label(left) || left.value;
223
+ const rightLabel = _solidUi.utils.label(right) || right.value;
224
+ return leftLabel.localeCompare(rightLabel);
225
+ });
226
+ _solidUi.utils.syncTableToArray(mutualFriendsTable, sortedMutualConnections, createMutualRow, function (row, thing) {
227
+ const replacement = createMutualRow(thing);
228
+ return replacement;
229
+ });
230
+ };
231
+ refreshMutualFriends();
232
+ }
233
+ return {
234
+ section: mutualSection,
235
+ content: mutualContent,
236
+ refreshMutualFriends
237
+ };
238
+ }
239
+ function createAllFriendsSection(options) {
240
+ const {
241
+ dom,
242
+ subject,
243
+ profile,
244
+ editable,
245
+ renderSupportingInfo,
246
+ renderNameSuffix
247
+ } = options;
248
+ const allFriends = dom.createElement('section');
249
+ allFriends.className = 'social-pane__all-friends social-primary__panel';
250
+ allFriends.id = 'social-panel-all-friends';
251
+ allFriends.setAttribute('role', 'tabpanel');
252
+ allFriends.setAttribute('aria-labelledby', 'social-tab-all-friends');
253
+ const mainTable = allFriends.appendChild(dom.createElement('table'));
254
+ mainTable.className = 'social-main';
255
+ const friendsList = _solidUi.widgets.attachmentList(dom, subject, mainTable, {
256
+ doc: profile,
257
+ modify: !!editable,
258
+ predicate: _solidUi.ns.foaf('knows'),
259
+ noun: 'friend',
260
+ // Social pane already owns the async refresh cycle for friend data in
261
+ // socialPane.ts. Leave attachmentList's generic follow-up rerender disabled
262
+ // here or each fetched friend profile will trigger an extra whole-table refresh
263
+ // on top of the pane's own batched updates.
264
+ refreshOnDocumentLoad: false,
265
+ renderSupportingInfo,
266
+ renderNameSuffix
267
+ });
268
+ friendsList.classList.add('social-friends-list');
269
+ friendsList.style.marginTop = '0';
270
+ const friendsListRow = friendsList.querySelector('tr');
271
+ const friendsListPromptCell = friendsListRow?.children?.[0];
272
+ const friendsListRightCell = friendsListRow?.children?.[1];
273
+ const friendsHeaderActions = dom.createElement('div');
274
+ friendsHeaderActions.classList.add('social-friends-header-actions');
275
+ if (friendsListPromptCell instanceof HTMLTableCellElement) {
276
+ friendsListPromptCell.classList.add('social-friends-header-dropzone-cell');
277
+ const friendsHeaderDropzone = dom.createElement('table');
278
+ friendsHeaderDropzone.className = 'social-friends-header-dropzone';
279
+ friendsHeaderDropzone.setAttribute('role', 'presentation');
280
+ const friendsHeaderDropzoneBody = friendsHeaderDropzone.appendChild(dom.createElement('tbody'));
281
+ const friendsHeaderDropzoneRow = friendsHeaderDropzoneBody.appendChild(dom.createElement('tr'));
282
+ friendsHeaderDropzoneRow.appendChild(friendsListPromptCell);
283
+ friendsHeaderActions.appendChild(friendsHeaderDropzone);
284
+ } else if (friendsListPromptCell instanceof HTMLElement) {
285
+ while (friendsListPromptCell.firstChild) {
286
+ friendsHeaderActions.appendChild(friendsListPromptCell.firstChild);
287
+ }
288
+ friendsListPromptCell.remove();
289
+ }
290
+ const friendDropButtons = friendsHeaderActions.querySelectorAll('button');
291
+ if (editable && friendDropButtons.length > 0) {
292
+ friendDropButtons.forEach(button => {
293
+ button.setAttribute('title', 'Drop friend here');
294
+ button.setAttribute('aria-label', 'Drop friend here');
295
+ const buttonImages = button.querySelectorAll('img');
296
+ buttonImages.forEach(image => {
297
+ image.setAttribute('title', 'Drop friend here');
298
+ image.setAttribute('alt', 'Drop friend here');
299
+ });
300
+ });
301
+ const friendsActionsRow = dom.createElement('div');
302
+ friendsActionsRow.className = 'social-friends-header-actions social-friends-header-actions--standalone';
303
+ friendsActionsRow.appendChild(friendsHeaderActions);
304
+ const dropHint = dom.createElement('span');
305
+ dropHint.className = 'social-friends-header-hint';
306
+ dropHint.textContent = 'Drag a WebId on the target to add a friend.';
307
+ friendsActionsRow.appendChild(dropHint);
308
+ allFriends.prepend(friendsActionsRow);
309
+ }
310
+ if (friendsListRightCell instanceof HTMLTableCellElement) {
311
+ friendsListRightCell.colSpan = 2;
312
+ }
313
+ const friendsItemsTable = friendsList.querySelector('td table');
314
+ if (friendsItemsTable instanceof HTMLTableElement) {
315
+ friendsItemsTable.classList.add('social-friends-grid');
316
+ }
317
+ return {
318
+ section: allFriends,
319
+ mainTable,
320
+ friendsList
321
+ };
322
+ }
323
+ function createRequestsSection(options) {
324
+ const {
325
+ dom,
326
+ triage,
327
+ renderSupportingInfo,
328
+ renderNameSuffix
329
+ } = options;
330
+ const requestsSection = dom.createElement('section');
331
+ requestsSection.className = 'social-pane__requests social-primary__panel';
332
+ requestsSection.id = 'social-panel-requests';
333
+ requestsSection.setAttribute('role', 'tabpanel');
334
+ requestsSection.setAttribute('aria-labelledby', 'social-tab-requests');
335
+ const requestsContent = requestsSection.appendChild(dom.createElement('div'));
336
+ requestsContent.classList.add('social-main', 'social-main--requests', 'flex-column');
337
+ const note = requestsContent.appendChild(dom.createElement('p'));
338
+ note.className = 'social-requests__note';
339
+ note.textContent = 'Best effort preview: this view only reflects profile documents that are currently loaded. A dedicated inbox would still be needed for complete request discovery.';
340
+ const createRequestGroup = function (title, description, emptyText) {
341
+ const group = requestsContent.appendChild(dom.createElement('section'));
342
+ group.classList.add('social-requests__group', 'flex-column');
343
+ const header = group.appendChild(dom.createElement('div'));
344
+ header.className = 'social-requests__group-header';
345
+ const heading = header.appendChild(dom.createElement('h2'));
346
+ heading.className = 'social-requests__group-title';
347
+ heading.textContent = title;
348
+ const blurb = header.appendChild(dom.createElement('p'));
349
+ blurb.className = 'social-requests__group-description';
350
+ blurb.textContent = description;
351
+ const empty = group.appendChild(dom.createElement('p'));
352
+ empty.className = 'social-requests__empty';
353
+ empty.textContent = emptyText;
354
+ const table = group.appendChild(dom.createElement('table'));
355
+ table.className = 'social-main social-friends-list social-friends-grid social-requests__table';
356
+ return {
357
+ empty,
358
+ table
359
+ };
360
+ };
361
+ const incomingGroup = createRequestGroup('Incoming requests', 'People who say they know this profile, but have not been confirmed yet.', 'No incoming requests are visible from the currently loaded data.');
362
+ const pendingGroup = createRequestGroup('Awaiting confirmation', 'People this profile knows who have not linked back yet.', 'Nothing is waiting for confirmation right now.');
363
+ const createPersonRow = function (target) {
364
+ return _solidUi.widgets.personTR(dom, _solidUi.ns.foaf('knows'), target, {
365
+ renderSupportingInfo,
366
+ renderNameSuffix
367
+ });
368
+ };
369
+ const sortNodes = function (nodes) {
370
+ return [...nodes].sort((left, right) => {
371
+ const leftLabel = _solidUi.utils.label(left) || left.value;
372
+ const rightLabel = _solidUi.utils.label(right) || right.value;
373
+ return leftLabel.localeCompare(rightLabel);
374
+ });
375
+ };
376
+ const syncRequestGroup = function (group, nodes) {
377
+ const sortedNodes = sortNodes(nodes);
378
+ group.empty.hidden = sortedNodes.length > 0;
379
+ group.table.hidden = sortedNodes.length === 0;
380
+ _solidUi.utils.syncTableToArray(group.table, sortedNodes, createPersonRow, function (_row, thing) {
381
+ return createPersonRow(thing);
382
+ });
383
+ };
384
+ const refreshRequests = function (nextTriage) {
385
+ syncRequestGroup(incomingGroup, nextTriage.requests);
386
+ syncRequestGroup(pendingGroup, nextTriage.unconfirmed);
387
+ };
388
+ refreshRequests(triage);
389
+ return {
390
+ section: requestsSection,
391
+ refreshRequests
392
+ };
393
+ }
@@ -0,0 +1,3 @@
1
+ import { TemplateResult } from 'lit-html';
2
+ export declare function createSpinner(): TemplateResult;
3
+ //# sourceMappingURL=spinner.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"spinner.d.ts","sourceRoot":"","sources":["../../src/social/spinner.ts"],"names":[],"mappings":"AAAA,OAAO,EAAQ,cAAc,EAAE,MAAM,UAAU,CAAA;AAE/C,wBAAgB,aAAa,IAAK,cAAc,CAK/C"}
@@ -0,0 +1,13 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.createSpinner = createSpinner;
7
+ var _litHtml = require("lit-html");
8
+ function createSpinner() {
9
+ return (0, _litHtml.html)`
10
+ <span class="loading-spinner" aria-hidden="true"></span>
11
+ <span class="sr-only">Saving...</span>
12
+ `;
13
+ }
@@ -0,0 +1,17 @@
1
+ import { DataBrowserContext } from 'pane-registry';
2
+ import { IndexedFormula, NamedNode } from 'rdflib';
3
+ type FriendshipStore = IndexedFormula & {
4
+ fetcher?: {
5
+ load?: (target: NamedNode | string) => Promise<unknown>;
6
+ };
7
+ };
8
+ export interface FriendshipTriage {
9
+ acquaintances: NamedNode[];
10
+ confirmed: NamedNode[];
11
+ unconfirmed: NamedNode[];
12
+ requests: NamedNode[];
13
+ }
14
+ export declare function triageFriends(store: FriendshipStore, subject: NamedNode): FriendshipTriage;
15
+ export declare function loadFriendshipTriage(context: DataBrowserContext, subject: NamedNode): Promise<FriendshipTriage>;
16
+ export {};
17
+ //# sourceMappingURL=triage.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"triage.d.ts","sourceRoot":"","sources":["../../src/social/triage.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,kBAAkB,EAAE,MAAM,eAAe,CAAA;AAClD,OAAO,EAAE,cAAc,EAAE,SAAS,EAAE,MAAM,QAAQ,CAAA;AAGlD,KAAK,eAAe,GAAG,cAAc,GAAG;IACtC,OAAO,CAAC,EAAE;QACR,IAAI,CAAC,EAAE,CAAC,MAAM,EAAE,SAAS,GAAG,MAAM,KAAK,OAAO,CAAC,OAAO,CAAC,CAAA;KACxD,CAAA;CACF,CAAA;AAED,MAAM,WAAW,gBAAgB;IAC/B,aAAa,EAAE,SAAS,EAAE,CAAA;IAC1B,SAAS,EAAE,SAAS,EAAE,CAAA;IACtB,WAAW,EAAE,SAAS,EAAE,CAAA;IACxB,QAAQ,EAAE,SAAS,EAAE,CAAA;CACtB;AAgBD,wBAAgB,aAAa,CAAE,KAAK,EAAE,eAAe,EAAE,OAAO,EAAE,SAAS,GAAG,gBAAgB,CA+B3F;AAED,wBAAsB,oBAAoB,CACxC,OAAO,EAAE,kBAAkB,EAC3B,OAAO,EAAE,SAAS,GACjB,OAAO,CAAC,gBAAgB,CAAC,CAwC3B"}
@@ -0,0 +1,79 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.loadFriendshipTriage = loadFriendshipTriage;
7
+ exports.triageFriends = triageFriends;
8
+ var _solidUi = require("solid-ui");
9
+ function uniqueNamedNodes(nodes) {
10
+ const seen = new Set();
11
+ const unique = [];
12
+ for (const node of nodes) {
13
+ const key = node?.value;
14
+ if (!key || seen.has(key)) continue;
15
+ seen.add(key);
16
+ unique.push(node);
17
+ }
18
+ return unique;
19
+ }
20
+ function triageFriends(store, subject) {
21
+ const outgoingFriends = uniqueNamedNodes(store.each(subject, _solidUi.ns.foaf('knows')));
22
+ const incomingFriends = uniqueNamedNodes(store.each(undefined, _solidUi.ns.foaf('knows'), subject));
23
+ const confirmed = [];
24
+ const unconfirmed = [];
25
+ const requests = [];
26
+ for (const friend of outgoingFriends) {
27
+ const isConfirmed = incomingFriends.some(incomingFriend => incomingFriend.sameTerm(friend));
28
+ if (isConfirmed) {
29
+ confirmed.push(friend);
30
+ } else {
31
+ unconfirmed.push(friend);
32
+ }
33
+ }
34
+ for (const friend of incomingFriends) {
35
+ const isAlreadyOutgoing = outgoingFriends.some(outgoingFriend => outgoingFriend.sameTerm(friend));
36
+ if (!isAlreadyOutgoing) {
37
+ requests.push(friend);
38
+ }
39
+ }
40
+ return {
41
+ acquaintances: outgoingFriends,
42
+ confirmed,
43
+ unconfirmed,
44
+ requests
45
+ };
46
+ }
47
+ async function loadFriendshipTriage(context, subject) {
48
+ const store = context.session.store;
49
+ const fetcher = store.fetcher;
50
+ if (!fetcher || typeof fetcher.load !== 'function') {
51
+ return triageFriends(store, subject);
52
+ }
53
+ try {
54
+ await fetcher.load(subject.doc());
55
+ } catch {
56
+ // Continue with the store snapshot we already have.
57
+ }
58
+ const initialTriage = triageFriends(store, subject);
59
+ const documentsToLoad = new Map();
60
+ documentsToLoad.set(subject.doc().value, subject.doc());
61
+ for (const friend of [...initialTriage.acquaintances, ...initialTriage.requests]) {
62
+ documentsToLoad.set(friend.doc().value, friend.doc());
63
+ }
64
+ const incomingStatements = store.statementsMatching(undefined, _solidUi.ns.foaf('knows'), subject);
65
+ for (const statement of incomingStatements) {
66
+ const why = statement.why;
67
+ if (why?.value) {
68
+ documentsToLoad.set(why.value, why);
69
+ }
70
+ }
71
+ for (const target of documentsToLoad.values()) {
72
+ try {
73
+ await fetcher.load(target);
74
+ } catch {
75
+ // Keep partial results when one profile document fails to load.
76
+ }
77
+ }
78
+ return triageFriends(store, subject);
79
+ }