feat(dashboard): randomize member and group lists
This commit is contained in:
		| @@ -15,6 +15,7 @@ import DiscordLogin from "./pages/DiscordLogin.svelte"; | |||||||
|   import { onMount } from 'svelte'; |   import { onMount } from 'svelte'; | ||||||
| import BulkGroupPrivacy from "./pages/BulkGroupPrivacy.svelte"; | import BulkGroupPrivacy from "./pages/BulkGroupPrivacy.svelte"; | ||||||
| import BulkMemberPrivacy from "./pages/BulkMemberPrivacy.svelte"; | import BulkMemberPrivacy from "./pages/BulkMemberPrivacy.svelte"; | ||||||
|  |   import Random from './pages/Random.svelte'; | ||||||
|    |    | ||||||
|   // theme cdns (I might make some myself too) |   // theme cdns (I might make some myself too) | ||||||
|   // if there's a style already set, retrieve it |   // if there's a style already set, retrieve it | ||||||
| @@ -38,8 +39,6 @@ import BulkMemberPrivacy from "./pages/BulkMemberPrivacy.svelte"; | |||||||
|     }; |     }; | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   let falseBool = false; |  | ||||||
|  |  | ||||||
|   onMount(() => { |   onMount(() => { | ||||||
|     let settings = JSON.parse(localStorage.getItem("pk-settings")); |     let settings = JSON.parse(localStorage.getItem("pk-settings")); | ||||||
|  |  | ||||||
| @@ -55,21 +54,29 @@ import BulkMemberPrivacy from "./pages/BulkMemberPrivacy.svelte"; | |||||||
|     <Route path="/"><Home /></Route> |     <Route path="/"><Home /></Route> | ||||||
|     <Route path="/login/discord"><DiscordLogin /></Route> |     <Route path="/login/discord"><DiscordLogin /></Route> | ||||||
|     <Route path="dash"><Dash /></Route> |     <Route path="dash"><Dash /></Route> | ||||||
|     <Route path="dash/m/:id"><Member isPublic={falseBool}/></Route> |     <Route path="dash/m/:id"><Member isPublic={false}/></Route> | ||||||
|     <Route path = "dash/g/:id"><Group isPublic={falseBool}/></Route> |     <Route path = "dash/g/:id"><Group isPublic={false}/></Route> | ||||||
|  |     <Route path="dash/random"><Random isPublic={false} type={"member"}/></Route> | ||||||
|  |     <Route path="dash/random/m"><Random isPublic={false} type={"member"}/></Route> | ||||||
|  |     <Route path="dash/random/g"><Random isPublic={false} type={"group"}/></Route> | ||||||
|  |     <Route path="dash/g/:groupId/random"><Random isPublic={false} type={"member"} pickFromGroup={true}/></Route> | ||||||
|     <Route path="dash/bulk-member-privacy"><BulkMemberPrivacy/></Route> |     <Route path="dash/bulk-member-privacy"><BulkMemberPrivacy/></Route> | ||||||
|     <Route path="dash/bulk-group-privacy"><BulkGroupPrivacy/></Route> |     <Route path="dash/bulk-group-privacy"><BulkGroupPrivacy/></Route> | ||||||
|     <Route path="settings"><Settings /></Route> |     <Route path="settings"><Settings /></Route> | ||||||
|     <Route path="profile"><Public /></Route> |     <Route path="profile"><Public /></Route> | ||||||
|     <Route path = "profile/s/:id"><Main /></Route> |     <Route path = "profile/s/:id"><Main /></Route> | ||||||
|     <Route path = "s"> |     <Route path = "profile/s"> | ||||||
|       <Alert color="danger">Please provide a system ID in the URL.</Alert> |       <Alert color="danger">Please provide a system ID in the URL.</Alert> | ||||||
|     </Route> |     </Route> | ||||||
|  |     <Route path="profile/s/:id/random"><Random isPublic={true} type={"member"}/></Route> | ||||||
|  |     <Route path="profile/s/:id/random/m"><Random isPublic={true} type={"member"}/></Route> | ||||||
|  |     <Route path="profile/s/:id/random/g"><Random isPublic={true} type={"group"}/></Route> | ||||||
|     <Route path = "profile/m/:id"><Member/></Route> |     <Route path = "profile/m/:id"><Member/></Route> | ||||||
|     <Route path = "profile/m"> |     <Route path = "profile/m"> | ||||||
|       <Alert color="danger">Please provide a member ID in the URL.</Alert> |       <Alert color="danger">Please provide a member ID in the URL.</Alert> | ||||||
|     </Route> |     </Route> | ||||||
|     <Route path = "profile/g/:id"><Group/></Route> |     <Route path = "profile/g/:id"><Group/></Route> | ||||||
|  |     <Route path="profile/g/:groupId/random"><Random isPublic={true} type={"member"} pickFromGroup={true}/></Route> | ||||||
|     <Route path = "profile/g"> |     <Route path = "profile/g"> | ||||||
|       <Alert color="danger">Please provide a group ID in the URL.</Alert> |       <Alert color="danger">Please provide a group ID in the URL.</Alert> | ||||||
|     </Route> |     </Route> | ||||||
|   | |||||||
| @@ -104,25 +104,50 @@ | |||||||
|     </Col> |     </Col> | ||||||
|     {/if} |     {/if} | ||||||
| </Row> | </Row> | ||||||
| <div class="my-2 description" bind:this={descriptionElement}> | <div class="mt-2 mb-3 description" bind:this={descriptionElement}> | ||||||
|     <b>Description:</b><br /> |     <b>Description:</b><br /> | ||||||
|     {@html htmlDescription && htmlDescription} |     {@html htmlDescription && htmlDescription} | ||||||
| </div> | </div> | ||||||
| {#if (group.banner && ((settings && settings.appearance.banner_bottom) || !settings))} | {#if (group.banner && ((settings && settings.appearance.banner_bottom) || !settings))} | ||||||
| <img src={group.banner} alt="group banner" class="w-100 mb-3 rounded" style="max-height: 12em; object-fit: cover"/> | <img src={group.banner} alt="group banner" class="w-100 mb-3 rounded" style="max-height: 12em; object-fit: cover"/> | ||||||
| {/if} | {/if} | ||||||
|  |  | ||||||
| {#if !isPublic} | {#if !isPublic} | ||||||
| <Button style="flex: 0" color="primary" on:click={() => editMode = true} aria-label="edit group information">Edit</Button>  | <Button style="flex: 0" class="link-button" color="primary" on:click={() => editMode = true} aria-label="edit group information">Edit</Button>  | ||||||
| {#if isMainDash}<Button style="flex: 0" color="secondary" on:click={() => memberMode = true} aria-label="edit group members">Members</Button>{/if} |     {#if isMainDash} | ||||||
| {/if} |     <Button class="link-button" style="flex: 0" color="secondary" on:click={() => memberMode = true} aria-label="edit group members">Members</Button> | ||||||
| {#if !isPage} |  | ||||||
|     <Link to={isPublic ? `/profile/g/${group.id}` : `/dash/g/${group.id}`}><Button style="flex: 0; {!isPublic && "float: right;"}" color="primary" tabindex={-1} aria-label="view group page">View page</Button></Link> |  | ||||||
|     {:else if !isPublic} |  | ||||||
|     <Link to="/dash?tab=groups"><Button style="flex: 0; {!isPublic && "float: right;"}" color="primary" tabindex={-1} aria-label="view group system">View system</Button></Link> |  | ||||||
|     {/if} |     {/if} | ||||||
|  | {/if} | ||||||
|  |  | ||||||
|  | {#if !isPage} | ||||||
|  |     <Link to={isPublic ? `/profile/g/${group.id}` : `/dash/g/${group.id}`}><button class="link-button button-right btn btn-primary" tabindex={-1} aria-label="view group page">View page</button></Link> | ||||||
|  |     {:else if !isPublic} | ||||||
|  |     <Link to="/dash?tab=groups"><button class="link-button button-right btn btn-primary" tabindex={-1} aria-label="view group system">View system</button></Link> | ||||||
|  | {/if} | ||||||
|  | <Link to={isPublic ? `/profile/g/${group.id}/random` : `/dash/g/${group.id}/random`}><button class="link-button button-right btn btn-secondary" style={isPublic ? "float: none !important; margin-left: 0;" : ""} tabindex={-1} aria-label="randomize group members">Randomize group</button></Link> | ||||||
|  |  | ||||||
| {:else if editMode} | {:else if editMode} | ||||||
| <Edit on:deletion bind:group bind:editMode /> | <Edit on:deletion bind:group bind:editMode /> | ||||||
| {:else if memberMode} | {:else if memberMode} | ||||||
|     <MemberEdit bind:group bind:memberMode bind:members /> |     <MemberEdit bind:group bind:memberMode bind:members /> | ||||||
| {/if} | {/if} | ||||||
| </CardBody> | </CardBody> | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  |     .link-button { | ||||||
|  |         width: 100%; | ||||||
|  |         margin-bottom: 0.2em; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @media (min-width: 992px) { | ||||||
|  |         .link-button { | ||||||
|  |             width: max-content; | ||||||
|  |             margin-bottom: 0; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         .button-right { | ||||||
|  |             float: right; | ||||||
|  |             margin-left: 0.25em; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | </style> | ||||||
| @@ -3,6 +3,7 @@ import { Card, CardHeader, CardBody, CardTitle, Alert, Accordion, AccordionItem, | |||||||
| import FaSearch from 'svelte-icons/fa/FaSearch.svelte' | import FaSearch from 'svelte-icons/fa/FaSearch.svelte' | ||||||
| import Svelecte, { addFormatter } from 'svelecte'; | import Svelecte, { addFormatter } from 'svelecte'; | ||||||
| import { Member, Group } from '../../api/types'; | import { Member, Group } from '../../api/types'; | ||||||
|  | import { Link, useParams } from 'svelte-navigator'; | ||||||
|  |  | ||||||
| export let list: Member[] | Group[] = []; | export let list: Member[] | Group[] = []; | ||||||
|  |  | ||||||
| @@ -24,6 +25,9 @@ let selectedGroups = []; | |||||||
| export let currentPage: number; | export let currentPage: number; | ||||||
| export let isPublic: boolean; | export let isPublic: boolean; | ||||||
|  |  | ||||||
|  | let params = useParams(); | ||||||
|  | $: systemId = $params.id; | ||||||
|  |  | ||||||
| $: {searchValue; privacyFilter; currentPage = 1}; | $: {searchValue; privacyFilter; currentPage = 1}; | ||||||
|  |  | ||||||
| // converting list to any[] avoids a "this expression is not calleable" error | // converting list to any[] avoids a "this expression is not calleable" error | ||||||
| @@ -143,6 +147,19 @@ function memberListRenderer(item: any) { | |||||||
| addFormatter({ | addFormatter({ | ||||||
|     'member-list': memberListRenderer |     'member-list': memberListRenderer | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  | function getRandomizerUrl(): string { | ||||||
|  |     let str: string; | ||||||
|  |     if (isPublic) str = `/profile/s/${systemId}/random` | ||||||
|  |     else str = "/dash/random"; | ||||||
|  |      | ||||||
|  |     if (itemType === "group") str += "/g"; | ||||||
|  |     return str; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function capitalizeFirstLetter(string: string) { | ||||||
|  |     return string.charAt(0).toUpperCase() + string.slice(1); | ||||||
|  | } | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
| <Card class="mb-3"> | <Card class="mb-3"> | ||||||
| @@ -210,6 +227,9 @@ addFormatter({ | |||||||
|             </InputGroup> |             </InputGroup> | ||||||
|         </Col> |         </Col> | ||||||
|         {/if} |         {/if} | ||||||
|  |         <Col xs={12} lg={3} class="mb-2"> | ||||||
|  |             <Link to={getRandomizerUrl()}><Button class="w-100" color="secondary" tabindex={-1} aria-label={`randomize ${itemType}s`}>Random {capitalizeFirstLetter(itemType)}</Button></Link> | ||||||
|  |         </Col> | ||||||
|     </Row> |     </Row> | ||||||
|     {#if !isPublic} |     {#if !isPublic} | ||||||
|     <hr/> |     <hr/> | ||||||
| @@ -220,10 +240,12 @@ addFormatter({ | |||||||
|     <Svelecte disableHighlight renderer="member-list" valueAsObject bind:value={selectedGroups} options={memberList} multiple style="margin-bottom: 0.5rem" /> |     <Svelecte disableHighlight renderer="member-list" valueAsObject bind:value={selectedGroups} options={memberList} multiple style="margin-bottom: 0.5rem" /> | ||||||
|     {/if} |     {/if} | ||||||
|  |  | ||||||
|     <span style="cursor: pointer" id="m-include" on:click={() => groupSearchMode = "include"} on:keyup={e => e.key === "Enter" ? groupSearchMode = "include" : ""} tabindex={0}>{@html groupSearchMode === "include" ? "<b>include</b>" : "include"}</span> |     <div class="filter-mode-group"> | ||||||
|      | <span style="cursor: pointer" id="m-exclude" on:click={() => groupSearchMode = "exclude"} on:keyup={e => e.key === "Enter" ? groupSearchMode = "exclude" : ""} tabindex={0}>{@html groupSearchMode === "exclude" ? "<b>exclude</b>" : "exclude"}</span>  |     <span class="filter-mode-label" id="m-include" on:click={() => groupSearchMode = "include"} on:keyup={e => e.key === "Enter" ? groupSearchMode = "include" : ""} tabindex={0}>{@html groupSearchMode === "include" ? "<b>include</b>" : "include"}</span> | ||||||
|      | <span style="cursor: pointer" id="m-match" on:click={() => groupSearchMode = "match"} on:keyup={e => e.key === "Enter" ? groupSearchMode = "match" : ""} tabindex={0}>{@html groupSearchMode === "match" ? "<b>exact match</b>" : "exact match"}</span> |      | <span class="filter-mode-label" id="m-exclude" on:click={() => groupSearchMode = "exclude"} on:keyup={e => e.key === "Enter" ? groupSearchMode = "exclude" : ""} tabindex={0}>{@html groupSearchMode === "exclude" ? "<b>exclude</b>" : "exclude"}</span>  | ||||||
|      | <span style="cursor: pointer" id="m-none" on:click={() => groupSearchMode = "none"} on:keyup={e => e.key === "Enter" ? groupSearchMode = "none" : ""} tabindex={0}>{@html groupSearchMode === "none" ? "<b>none</b>" : "none"}</span> |      | <span class="filter-mode-label" id="m-match" on:click={() => groupSearchMode = "match"} on:keyup={e => e.key === "Enter" ? groupSearchMode = "match" : ""} tabindex={0}>{@html groupSearchMode === "match" ? "<b>exact match</b>" : "exact match"}</span> | ||||||
|  |      | <span class="filter-mode-label" id="m-none" on:click={() => groupSearchMode = "none"} on:keyup={e => e.key === "Enter" ? groupSearchMode = "none" : ""} tabindex={0}>{@html groupSearchMode === "none" ? "<b>none</b>" : "none"}</span> | ||||||
|  |     </div> | ||||||
|      <Tooltip placement="bottom" target="m-include">Includes every member who's a part of any of the groups.</Tooltip> |      <Tooltip placement="bottom" target="m-include">Includes every member who's a part of any of the groups.</Tooltip> | ||||||
|     <Tooltip placement="bottom" target="m-exclude">Excludes every member who's a part of any of the groups, the opposite of include.</Tooltip> |     <Tooltip placement="bottom" target="m-exclude">Excludes every member who's a part of any of the groups, the opposite of include.</Tooltip> | ||||||
|     <Tooltip placement="bottom" target="m-match">Only includes members who are a part of every group.</Tooltip> |     <Tooltip placement="bottom" target="m-match">Only includes members who are a part of every group.</Tooltip> | ||||||
| @@ -231,3 +253,16 @@ addFormatter({ | |||||||
|     {/if} |     {/if} | ||||||
| </CardBody> | </CardBody> | ||||||
| </Card> | </Card> | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  |     .filter-mode-label { | ||||||
|  |         cursor: pointer; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .filter-mode-group { | ||||||
|  |         line-height: 1.5em; | ||||||
|  |         padding:0.375rem 0; | ||||||
|  |         display: inline-block; | ||||||
|  |         margin-bottom: 0.25em; | ||||||
|  |     } | ||||||
|  | </style> | ||||||
							
								
								
									
										291
									
								
								dashboard/src/pages/Random.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										291
									
								
								dashboard/src/pages/Random.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,291 @@ | |||||||
|  | <script lang="ts"> | ||||||
|  |     import { onMount } from 'svelte'; | ||||||
|  |     import { Link, useLocation, useParams, navigate } from 'svelte-navigator'; | ||||||
|  |     import { Alert, Col, Container, Row, Card, CardBody, CardHeader, CardTitle, Input, Label, Button, Accordion, AccordionHeader, AccordionItem } from 'sveltestrap'; | ||||||
|  |     import FaRandom from 'svelte-icons/fa/FaRandom.svelte' | ||||||
|  |      | ||||||
|  |     import MemberBody from '../lib/member/Body.svelte'; | ||||||
|  |     import GroupBody from '../lib/group/Body.svelte'; | ||||||
|  |     import CardsHeader from '../lib/CardsHeader.svelte'; | ||||||
|  |     import api from '../api'; | ||||||
|  |     import { Group, Member } from '../api/types'; | ||||||
|  |     import FaUserCircle from 'svelte-icons/fa/FaUserCircle.svelte'; | ||||||
|  |     import FaUsers from 'svelte-icons/fa/FaUsers.svelte'; | ||||||
|  |     import FaLock from 'svelte-icons/fa/FaLock.svelte'; | ||||||
|  |  | ||||||
|  |     export let isPublic: boolean = false; | ||||||
|  |     export let type: string = "member"; | ||||||
|  |     export let pickFromGroup: boolean = false; | ||||||
|  |  | ||||||
|  |     let list: Member[]|Group[] = []; | ||||||
|  |     let randomList: Member[]|Group[] = []; | ||||||
|  |      | ||||||
|  |     let loading = true; | ||||||
|  |     let err: string; | ||||||
|  |  | ||||||
|  |     let params = useParams(); | ||||||
|  |     $: id = $params.id; | ||||||
|  |     $: groupId = $params.groupId; | ||||||
|  |  | ||||||
|  |     let location = useLocation(); | ||||||
|  |     let searchParams = $location.search && new URLSearchParams($location.search); | ||||||
|  |  | ||||||
|  |     let path = $location.pathname; | ||||||
|  |  | ||||||
|  |     let amount: number = 1; | ||||||
|  |  | ||||||
|  |     if (searchParams && searchParams.get("amount")) { | ||||||
|  |         amount = parseInt(searchParams.get("amount")); | ||||||
|  |         if (amount === NaN) amount = 1; | ||||||
|  |         else if (amount > 5) amount = 5; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     let usePrivateMembers = false; | ||||||
|  |  | ||||||
|  |     if (searchParams && searchParams.get("all") === "true") usePrivateMembers = true; | ||||||
|  |  | ||||||
|  |     let allowDoubles = false; | ||||||
|  |     if (searchParams && searchParams.get("doubles") === "true") allowDoubles = true; | ||||||
|  |  | ||||||
|  |     // just a hidden option to expand the cards by default regardless of your global settings | ||||||
|  |     let openByDefault = false; | ||||||
|  |     if (searchParams && searchParams.get("open") === "true") openByDefault = true; | ||||||
|  |  | ||||||
|  |     let settings = JSON.parse(localStorage.getItem("pk-settings")); | ||||||
|  |      | ||||||
|  |     onMount(async () => { | ||||||
|  |         await fetchList(amount, usePrivateMembers); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     async function fetchList(amount: number, usePrivateMembers?: boolean|string) { | ||||||
|  |         err = ""; | ||||||
|  |         loading = true; | ||||||
|  |         try { | ||||||
|  |             if (type === "member") {  | ||||||
|  |                 if (pickFromGroup) { | ||||||
|  |                     const res: Member[] = await api().groups(groupId).members.get({auth: !isPublic}); | ||||||
|  |                     list = res; | ||||||
|  |                 } else { | ||||||
|  |                     const res: Member[] = await api().systems(isPublic ? id : "@me").members.get({ auth: !isPublic }); | ||||||
|  |                     list = res; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             else if (type === "group") { | ||||||
|  |                 const res: Group[] = await api().systems(isPublic ? id : "@me").groups.get({ auth: !isPublic }); | ||||||
|  |                 list = res; | ||||||
|  |             } | ||||||
|  |             else throw new Error(`Unknown list type ${type}`); | ||||||
|  |             randomList = randomizeList(amount, usePrivateMembers, allowDoubles); | ||||||
|  |         } catch (error) { | ||||||
|  |             console.log(error); | ||||||
|  |             err = error.message; | ||||||
|  |         } | ||||||
|  |         loading = false; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function randomizeList(amount: number, usePrivateMembers?: boolean|string, allowDoubles?: boolean|string) { | ||||||
|  |         err = ""; | ||||||
|  |         let filteredList = [...list]; | ||||||
|  |         if (!isPublic && (!usePrivateMembers || usePrivateMembers === "false")) filteredList = (list as Member[]).filter(item => item.privacy && item.privacy.visibility === "public" ? true : false); | ||||||
|  |  | ||||||
|  |         let cappedAmount = amount; | ||||||
|  |         if (amount > filteredList.length) cappedAmount = filteredList.length; | ||||||
|  |  | ||||||
|  |         if (cappedAmount === 0) err = `No valid ${type}s could be randomized. ${!isPublic ? `If every ${type} is privated, roll again with private ${type}s included.` : ""}`; | ||||||
|  |  | ||||||
|  |         let tempList = []; | ||||||
|  |         for (let i = 0; i < cappedAmount; i++) { | ||||||
|  |             let index = Math.floor(Math.random() * filteredList.length); | ||||||
|  |             tempList.push(filteredList[index]); | ||||||
|  |             | ||||||
|  |             if (!allowDoubles || allowDoubles === "false") { | ||||||
|  |                 filteredList.splice(index, 1); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return tempList; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function rerollList() { | ||||||
|  |         let amount = parseInt(optionAmount); | ||||||
|  |         let paramArray = []; | ||||||
|  |         if (amount > 1) paramArray.push(`amount=${amount}`); | ||||||
|  |         if (optionAllowDoubles === "true") paramArray.push("doubles=true"); | ||||||
|  |         if (optionUsePrivateMembers === "true") paramArray.push("all=true"); | ||||||
|  |         if (openByDefault === true) paramArray.push("open=true"); | ||||||
|  |          | ||||||
|  |         randomList = randomizeList(parseInt(optionAmount), optionUsePrivateMembers, optionAllowDoubles); | ||||||
|  |         navigate(`${path}${paramArray.length > 0 ? `?${paramArray.join('&')}` : ""}`); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function capitalizeFirstLetter(string: string) { | ||||||
|  |         return string.charAt(0).toUpperCase() + string.slice(1); | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     let optionAmount = amount.toString(); | ||||||
|  |      | ||||||
|  |     let optionUsePrivateMembers = "false"; | ||||||
|  |     if (usePrivateMembers === true) optionUsePrivateMembers = "true"; | ||||||
|  |  | ||||||
|  |     let optionAllowDoubles = "false"; | ||||||
|  |     if (allowDoubles === true) optionAllowDoubles = "true"; | ||||||
|  |  | ||||||
|  |     function getItemLink(item: Member | Group): string { | ||||||
|  |         let url: string; | ||||||
|  |  | ||||||
|  |         if (isPublic) url = "/dash/"; | ||||||
|  |         else url = "/profile/"; | ||||||
|  |          | ||||||
|  |         if (type === "member") url += "m/"; | ||||||
|  |         else if (type === "group") url += "g/"; | ||||||
|  |  | ||||||
|  |         url += item.id; | ||||||
|  |  | ||||||
|  |         return url; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function getBackUrl() { | ||||||
|  |         let str: string; | ||||||
|  |         if (isPublic)  {  | ||||||
|  |             str = "/profile"; | ||||||
|  |             if (!pickFromGroup) str += `/s/${id}`; | ||||||
|  |         } else str = "/dash" | ||||||
|  |          | ||||||
|  |         if (pickFromGroup) str += `/g/${groupId}`; | ||||||
|  |  | ||||||
|  |         return str; | ||||||
|  |     } | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <Container> | ||||||
|  |     <Row> | ||||||
|  |         <Col class="mx-auto" xs={12} lg={11} xl={10}> | ||||||
|  |             <Card class="mb-4"> | ||||||
|  |                 <CardHeader> | ||||||
|  |                     <CardTitle style="margin-top: 8px; outline: none;"> | ||||||
|  |                     <div class="icon d-inline-block"> | ||||||
|  |                         <FaRandom /> | ||||||
|  |                     </div>Randomize {capitalizeFirstLetter(type)}s {isPublic && id ? `(${id})` : pickFromGroup ? `(${groupId})` : ""}</CardTitle> | ||||||
|  |                 </CardHeader> | ||||||
|  |                 <CardBody> | ||||||
|  |                     <Row> | ||||||
|  |                         <Col xs={12} lg={4} class="mb-2"> | ||||||
|  |                             <Label>Amount:</Label> | ||||||
|  |                             <Input bind:value={optionAmount} type="select" aria-label="amount"> | ||||||
|  |                                 <option default={amount === 1}>1</option> | ||||||
|  |                                 <option default={amount === 2}>2</option> | ||||||
|  |                                 <option default={amount === 3}>3</option> | ||||||
|  |                                 <option default={amount === 4}>4</option> | ||||||
|  |                                 <option default={amount === 5}>5</option> | ||||||
|  |                             </Input> | ||||||
|  |                         </Col> | ||||||
|  |                         <Col xs={12} lg={4} class="mb-2"> | ||||||
|  |                             <Label>Allow duplicates:</Label> | ||||||
|  |                             <Input bind:value={optionAllowDoubles} type="select" aria-label="allow duplicates"> | ||||||
|  |                                 <option value="false" default={allowDoubles === false}>no</option> | ||||||
|  |                                 <option value="true" default={allowDoubles === true}>yes</option> | ||||||
|  |                             </Input> | ||||||
|  |                         </Col> | ||||||
|  |                         {#if !isPublic} | ||||||
|  |                         <Col xs={12} lg={4} class="mb-2"> | ||||||
|  |                             <Label>Use all members:</Label> | ||||||
|  |                             <Input bind:value={optionUsePrivateMembers} type="select" aria-label="include private members"> | ||||||
|  |                                 <option value="false" default={usePrivateMembers === false}>no (only public members)</option> | ||||||
|  |                                 <option value="true" default={usePrivateMembers === true}>yes (include private members)</option> | ||||||
|  |                             </Input> | ||||||
|  |                         </Col> | ||||||
|  |                         {/if} | ||||||
|  |                     </Row> | ||||||
|  |                     <Button color="primary" on:click={() => {rerollList()}}> | ||||||
|  |                         Reroll list | ||||||
|  |                     </Button> | ||||||
|  |                     <Link to={getBackUrl()}> | ||||||
|  |                         <Button color="secondary" tabindex={-1} aria-label={`back to ${pickFromGroup ? "group" : "system"}`}> | ||||||
|  |                             Back to {pickFromGroup ? "group" : "system"} | ||||||
|  |                         </Button> | ||||||
|  |                     </Link> | ||||||
|  |                 </CardBody> | ||||||
|  |             </Card> | ||||||
|  |             {#if loading} | ||||||
|  |                 <span>loading...</span> | ||||||
|  |             {:else if err} | ||||||
|  |                 <Alert color="danger">{err}</Alert> | ||||||
|  |             {:else} | ||||||
|  |             {#if !openByDefault && (settings && settings.accessibility ? (!settings.accessibility.expandedcards && !settings.accessibility.pagelinks) : true)} | ||||||
|  |             <Accordion class="my-3" stayOpen> | ||||||
|  |                 {#each randomList as item, index (item.id + index)} | ||||||
|  |                 <AccordionItem> | ||||||
|  |                     <CardsHeader {item} slot="header"> | ||||||
|  |                         <div slot="icon"> | ||||||
|  |                             {#if isPublic || item.privacy.visibility === "public"} | ||||||
|  |                             {#if type === "member"} | ||||||
|  |                             <FaUserCircle /> | ||||||
|  |                             {:else if type === "group"} | ||||||
|  |                             <FaUsers /> | ||||||
|  |                             {/if} | ||||||
|  |                             {:else} | ||||||
|  |                             <FaLock /> | ||||||
|  |                             {/if} | ||||||
|  |                         </div> | ||||||
|  |                     </CardsHeader> | ||||||
|  |                     {#if type === "member"} | ||||||
|  |                     <MemberBody isPublic={true} bind:member={item} isMainDash={false} /> | ||||||
|  |                     {:else if type === "group"} | ||||||
|  |                     <GroupBody isPublic={true} bind:group={item} isMainDash={false} /> | ||||||
|  |                     {/if} | ||||||
|  |                 </AccordionItem> | ||||||
|  |                 {/each} | ||||||
|  |             </Accordion> | ||||||
|  |             {:else if openByDefault || settings.accessibility.expandedcards} | ||||||
|  |                 {#each randomList as item, index (item.id + index)} | ||||||
|  |                 <Card class="mb-3"> | ||||||
|  |                     <CardHeader> | ||||||
|  |                         <CardsHeader {item}> | ||||||
|  |                             <div slot="icon"> | ||||||
|  |                                 {#if isPublic || item.privacy.visibility === "public"} | ||||||
|  |                                 {#if type === "member"} | ||||||
|  |                                 <FaUserCircle /> | ||||||
|  |                                 {:else if type === "group"} | ||||||
|  |                                 <FaUsers /> | ||||||
|  |                                 {/if} | ||||||
|  |                                 {:else} | ||||||
|  |                                 <FaLock /> | ||||||
|  |                                 {/if} | ||||||
|  |                             </div> | ||||||
|  |                         </CardsHeader> | ||||||
|  |                     </CardHeader> | ||||||
|  |                     <CardBody> | ||||||
|  |                         {#if type === "member"} | ||||||
|  |                         <MemberBody isPublic={true} bind:member={item} isMainDash={false} /> | ||||||
|  |                         {:else if type === "group"} | ||||||
|  |                         <GroupBody isPublic={true} bind:group={item} isMainDash={false} /> | ||||||
|  |                         {/if} | ||||||
|  |                     </CardBody> | ||||||
|  |                 </Card> | ||||||
|  |                 {/each} | ||||||
|  |             {:else} | ||||||
|  |                 <div class="my-3"> | ||||||
|  |                 {#each randomList as item, index (item.id + index)} | ||||||
|  |                 <Card> | ||||||
|  |                     <Link class="accordion-button collapsed" style="text-decoration: none;" to={getItemLink(item)}> | ||||||
|  |                         <CardsHeader {item}> | ||||||
|  |                             <div slot="icon"> | ||||||
|  |                                 {#if isPublic || item.privacy.visibility === "public"} | ||||||
|  |                                 {#if type === "member"} | ||||||
|  |                                 <FaUserCircle /> | ||||||
|  |                                 {:else if type === "group"} | ||||||
|  |                                 <FaUsers /> | ||||||
|  |                                 {/if} | ||||||
|  |                                 {:else} | ||||||
|  |                                 <FaLock /> | ||||||
|  |                                 {/if} | ||||||
|  |                             </div> | ||||||
|  |                         </CardsHeader> | ||||||
|  |                     </Link> | ||||||
|  |                 </Card> | ||||||
|  |                 {/each} | ||||||
|  |                 </div> | ||||||
|  |             {/if} | ||||||
|  |             {/if} | ||||||
|  |         </Col> | ||||||
|  |     </Row> | ||||||
|  | </Container> | ||||||
		Reference in New Issue
	
	Block a user