1064 lines
42 KiB
Svelte
1064 lines
42 KiB
Svelte
<script lang='ts'>
|
|
import CreateNewPost from '$lib/components/CreateNewPost.svelte';
|
|
import PostSection from '$lib/components/PostSection.svelte';
|
|
import SkeletonLoading from '$lib/components/SkeletonLoading.svelte';
|
|
import Input from '$lib/components/Input.svelte';
|
|
import { serialize } from 'object-to-formdata';
|
|
|
|
import { onMount, onDestroy } from 'svelte';
|
|
import {getImageURL } from '$lib/utils';
|
|
import {screenWidth, userRegion, setCache, getCache, newAvatar, clientSideCache, numberOfUnreadNotification, postIdDeleted } from '$lib/store';
|
|
|
|
import toast from 'svelte-french-toast';
|
|
import InfiniteLoading from '$lib/components/InfiniteLoading.svelte';
|
|
|
|
import { pb } from '$lib/pocketbase';
|
|
import { z } from 'zod';
|
|
import { updatePersonalDataSchema, updatePasswordSchema} from '$lib/schemas';
|
|
import communityBanner from '$lib/images/community_banner.jpg';
|
|
import { enhance } from '$app/forms';
|
|
|
|
export let data;
|
|
export let form;
|
|
|
|
const usRegion = ['cle1','iad1','pdx1','sfo1'];
|
|
|
|
let apiURL;
|
|
let fastifyURL;
|
|
|
|
userRegion.subscribe(value => {
|
|
if (usRegion.includes(value)) {
|
|
apiURL = import.meta.env.VITE_USEAST_API_URL;
|
|
fastifyURL = import.meta.env.VITE_USEAST_FASTIFY_URL;
|
|
} else {
|
|
apiURL = import.meta.env.VITE_EU_API_URL;
|
|
fastifyURL = import.meta.env.VITE_EU_FASTIFY_URL;
|
|
}
|
|
});
|
|
|
|
|
|
let zodErrors = [];
|
|
let moderators;
|
|
let numberOfPosts = '-';
|
|
let loading = true;
|
|
let isLoaded = false;
|
|
let errorAvatar;
|
|
let errorUsername = '';
|
|
let errorOldPassword = '';
|
|
let errorPassword = '';
|
|
let errorPasswordConfirm = '';
|
|
let subscriptionData = data?.getSubscriptionData;
|
|
let isClicked = false;
|
|
let userStats = {'numberOfPosts': 0, 'numberOfComments': 0}
|
|
|
|
const showPreview = (event) => {
|
|
const target = event.target;
|
|
const files = target.files;
|
|
|
|
if (files.length > 0) {
|
|
const src = URL.createObjectURL(files[0]);
|
|
const preview = document.getElementById('avatar-preview');
|
|
preview.src = src;
|
|
|
|
document.getElementById('submit-btn').click();
|
|
|
|
}
|
|
};
|
|
|
|
|
|
|
|
const submitCancellation = () => {
|
|
return async ({ result, update}) => {
|
|
switch (result.type) {
|
|
case 'success':
|
|
toast.success('Subscription Cancelled successfully!', {
|
|
style: 'border-radius: 200px; background: #333; color: #fff;'});
|
|
await update();
|
|
break;
|
|
case 'redirect':
|
|
toast.success('Subscription Cancelled successfully!', {
|
|
style: 'border-radius: 200px; background: #333; color: #fff;'});
|
|
await update();
|
|
break;
|
|
case 'failure':
|
|
toast.error('Something went wrong.', {
|
|
style: 'border-radius: 200px; background: #333; color: #fff;'});
|
|
await update();
|
|
break;
|
|
case 'error':
|
|
toast.error(result.error.message, {
|
|
style: 'border-radius: 200px; background: #333; color: #fff;'});
|
|
break;
|
|
default:
|
|
await update();
|
|
}
|
|
|
|
setTimeout(() => {
|
|
if (result.type === 'redirect') {
|
|
const anchor = document.createElement('a');
|
|
anchor.href = '/community/profile';
|
|
anchor.dataset.sveltekitReload = true;
|
|
document.body.appendChild(anchor);
|
|
anchor.dispatchEvent(new MouseEvent('click'));
|
|
}
|
|
}, 1500);
|
|
|
|
}
|
|
}
|
|
|
|
const submitReactivate = () => {
|
|
return async ({ result, update}) => {
|
|
switch (result.type) {
|
|
case 'success':
|
|
toast.success('Subscription Reactivate successfully!', {
|
|
style: 'border-radius: 200px; background: #333; color: #fff;'});
|
|
await update();
|
|
break;
|
|
case 'redirect':
|
|
toast.success('Subscription Reactivate successfully!', {
|
|
style: 'border-radius: 200px; background: #333; color: #fff;'});
|
|
await update();
|
|
break;
|
|
case 'failure':
|
|
toast.error('Something went wrong.', {
|
|
style: 'border-radius: 200px; background: #333; color: #fff;'});
|
|
await update();
|
|
break;
|
|
case 'error':
|
|
toast.error(result.error.message, {
|
|
style: 'border-radius: 200px; background: #333; color: #fff;'});
|
|
break;
|
|
default:
|
|
await update();
|
|
}
|
|
|
|
setTimeout(() => {
|
|
if (result.type === 'redirect') {
|
|
const anchor = document.createElement('a');
|
|
anchor.href = '/community/profile';
|
|
anchor.dataset.sveltekitReload = true;
|
|
document.body.appendChild(anchor);
|
|
anchor.dispatchEvent(new MouseEvent('click'));
|
|
}
|
|
}, 1500);
|
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
async function handleReactivateSubscription() {
|
|
|
|
// make the POST request to the endpoint
|
|
const response = await fetch('/api/reactivate-subscription', {
|
|
method: 'GET',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
},
|
|
});
|
|
|
|
const output = await response.json();
|
|
|
|
}
|
|
|
|
|
|
async function updateAvatar(event)
|
|
{
|
|
|
|
event.preventDefault(); // prevent the default form submission behavior
|
|
const formData = new FormData(event.target); // create a FormData object from the form
|
|
const postData = {}
|
|
|
|
for (const [key, value] of formData.entries()) {
|
|
postData[key] = value;
|
|
}
|
|
try {
|
|
//To-do
|
|
//const cleanedData = updateAvatarSchema.parse({postData});
|
|
|
|
const { avatar } = await pb.collection('users').update(data?.user?.id, serialize(postData));
|
|
data.user.avatar = avatar;
|
|
$newAvatar = avatar;
|
|
/*
|
|
toast.success('Avatar updated successfully!', {
|
|
style: 'border-radius: 200px; background: #333; color: #fff;'});
|
|
*/
|
|
}
|
|
catch(error) {
|
|
|
|
if (error instanceof z.ZodError) {
|
|
// Handle Zod validation errors
|
|
zodErrors = error.errors;
|
|
|
|
errorAvatar = zodErrors.find((err) => err.path[0] === 'avatar')?.message ?? '';
|
|
|
|
|
|
toast.error(errorAvatar, {
|
|
style: 'border-radius: 200px; background: #333; color: #fff;',
|
|
});
|
|
}
|
|
else {
|
|
toast.error('Something went wrong. Please try again!', {
|
|
style: 'border-radius: 200px; background: #333; color: #fff;',
|
|
});
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
async function handleAvatar(event) {
|
|
toast.promise(
|
|
updateAvatar(event),
|
|
{
|
|
loading: 'Saving...',
|
|
success: 'Avatar updated successfully!',
|
|
error: 'Could not save.',
|
|
},
|
|
{
|
|
style: 'border-radius: 200px; background: #333; color: #fff;',
|
|
}
|
|
);
|
|
|
|
}
|
|
|
|
async function updatePersonalData(event) {
|
|
event.preventDefault(); // prevent the default form submission behavior
|
|
errorUsername = '';
|
|
const formData = new FormData(event.target); // create a FormData object from the form
|
|
const postData = {}
|
|
|
|
for (const [key, value] of formData.entries()) {
|
|
postData[key] = value;
|
|
}
|
|
|
|
try {
|
|
// Use Zod validation
|
|
const cleanedData = updatePersonalDataSchema.parse(postData);
|
|
const { username } = await pb.collection('users').update(data?.user?.id, cleanedData);
|
|
toast.success('Username updated!', {
|
|
style: 'border-radius: 200px; background: #333; color: #fff;',
|
|
});
|
|
data.user.username = username;
|
|
}
|
|
catch (error) {
|
|
if (error instanceof z.ZodError) {
|
|
// Handle Zod validation errors
|
|
zodErrors = error.errors;
|
|
|
|
errorUsername = zodErrors?.find((err) => err.path[0] === 'username')?.message ?? '';
|
|
|
|
toast.error('Invalid credentials', {
|
|
style: 'border-radius: 200px; background: #333; color: #fff;',
|
|
});
|
|
}
|
|
else {
|
|
// Handle other errors
|
|
console.error('Unexpected error during registration:', error);
|
|
|
|
toast.error('An unexpected error occurred', {
|
|
style: 'border-radius: 200px; background: #333; color: #fff;',
|
|
});
|
|
}
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
async function updatePassword(event) {
|
|
|
|
event.preventDefault(); // prevent the default form submission behavior
|
|
errorOldPassword = '';
|
|
errorPassword = '';
|
|
errorPasswordConfirm = '';
|
|
|
|
|
|
const formData = new FormData(event.target); // create a FormData object from the form
|
|
const postData = {}
|
|
|
|
for (const [key, value] of formData.entries()) {
|
|
postData[key] = value;
|
|
}
|
|
|
|
try {
|
|
// Use Zod validation
|
|
const cleanedData = updatePasswordSchema.parse(postData);
|
|
await pb.collection('users').update(data?.user?.id, cleanedData);
|
|
toast.success('Password updated!', {
|
|
style: 'border-radius: 200px; background: #333; color: #fff;',
|
|
});
|
|
}
|
|
catch (error) {
|
|
if (error instanceof z.ZodError) {
|
|
// Handle Zod validation errors
|
|
zodErrors = error.errors;
|
|
|
|
errorOldPassword = zodErrors?.find((err) => err.path[0] === 'oldPassword')?.message ?? '';
|
|
errorPassword = zodErrors?.find((err) => err.path[0] === 'password')?.message ?? '';
|
|
errorPasswordConfirm = zodErrors?.find((err) => err.path[0] === 'passwordConfirm')?.message ?? '';
|
|
|
|
toast.error('Invalid credentials', {
|
|
style: 'border-radius: 200px; background: #333; color: #fff;',
|
|
});
|
|
}
|
|
else {
|
|
// Handle other errors
|
|
console.error('Unexpected error during registration:', error);
|
|
|
|
toast.error('An unexpected error occurred', {
|
|
style: 'border-radius: 200px; background: #333; color: #fff;',
|
|
});
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const getModerators = async () => {
|
|
let output;
|
|
|
|
// Get cached data for the specific tickerID
|
|
const cachedData = getCache('', 'getModerators');
|
|
if (cachedData) {
|
|
output = cachedData;
|
|
} else {
|
|
|
|
// make the POST request to the endpoint
|
|
const response = await fetch(fastifyURL + '/get-moderators', {
|
|
method: 'GET',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
},
|
|
});
|
|
|
|
output = (await response.json())?.items;
|
|
|
|
setCache('', output, 'getModerators');
|
|
}
|
|
|
|
return output;
|
|
};
|
|
|
|
|
|
|
|
function isModerator(data, moderators) {
|
|
return moderators?.some(moderator => data?.user?.id === moderator?.expand?.user?.id);
|
|
}
|
|
|
|
|
|
const getUserStats = async () => {
|
|
let output;
|
|
|
|
// Get cached data for the specific tickerID
|
|
const cachedData = getCache(data?.user?.id, 'getUserStats');
|
|
if (cachedData) {
|
|
output = cachedData;
|
|
} else {
|
|
|
|
const postData = {'userId': data?.user?.id};
|
|
|
|
// make the POST request to the endpoint
|
|
const response = await fetch(fastifyURL + '/get-user-stats', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
},
|
|
body: JSON.stringify(postData)
|
|
});
|
|
|
|
output = (await response.json())?.items;
|
|
|
|
|
|
setCache(data?.user?.id, output, 'getUserStats');
|
|
}
|
|
|
|
return output
|
|
};
|
|
|
|
|
|
|
|
|
|
let posts: any[] = [];
|
|
|
|
|
|
|
|
let currentPage = 1;
|
|
let postLoading = false;
|
|
let seenPostId = [];
|
|
|
|
let noPostMore = false;
|
|
|
|
|
|
async function infiniteHandler({ detail: { loaded, complete } })
|
|
{
|
|
|
|
// console.log("Page position:", window.pageYOffset);
|
|
seenPostId = posts.map(obj => obj.id);
|
|
|
|
if (!postLoading && !noPostMore) {
|
|
postLoading = true;
|
|
|
|
const newPosts = await getPost();
|
|
if (newPosts.length === 0) {
|
|
noPostMore = true;
|
|
complete();
|
|
} else {
|
|
// Remove new posts with duplicate IDs
|
|
const filteredNewPosts = newPosts.filter((newPost) => !posts.find((post) => post.id === newPost.id));
|
|
posts = [...posts, ...filteredNewPosts];
|
|
currentPage++;
|
|
loaded();
|
|
}
|
|
postLoading = false;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
async function getPost() {
|
|
|
|
|
|
const postData = {
|
|
startPage: currentPage,
|
|
seenPostId: seenPostId.length === 0 ? [] : seenPostId,
|
|
userId: data?.user?.id,
|
|
};
|
|
|
|
// Make the POST request to the endpoint
|
|
const response = await fetch(fastifyURL+'/get-post', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
},
|
|
body: JSON.stringify(postData),
|
|
});
|
|
|
|
const output = (await response.json())?.items;
|
|
|
|
|
|
return output
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
let showTab = 'post';
|
|
|
|
let settingsTab = 'personalData';
|
|
|
|
const changeTab = (state) => {
|
|
switch (state) {
|
|
case 'post':
|
|
showTab = 'post';
|
|
break;
|
|
case 'settings':
|
|
showTab = 'settings';
|
|
break;
|
|
case 'subscription':
|
|
showTab = 'subscription';
|
|
break;
|
|
case 'personalData':
|
|
settingsTab = 'personalData';
|
|
break;
|
|
case 'changePassword':
|
|
settingsTab = 'changePassword';
|
|
break;
|
|
|
|
}
|
|
};
|
|
|
|
|
|
onMount(async () => {
|
|
|
|
window.scrollTo(0, 0);
|
|
|
|
[posts, moderators, userStats] = await Promise.all([
|
|
getPost(),
|
|
getModerators(),
|
|
getUserStats(),
|
|
]);
|
|
|
|
loading = false;
|
|
isLoaded = true;
|
|
|
|
});
|
|
|
|
|
|
onDestroy(async () => {
|
|
$postIdDeleted ='';
|
|
|
|
});
|
|
|
|
|
|
$: {
|
|
if($postIdDeleted.length !== 0)
|
|
{
|
|
posts = posts?.filter(item => item?.id !== $postIdDeleted);
|
|
|
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
<svelte:head>
|
|
<title> {$numberOfUnreadNotification > 0 ? `(${$numberOfUnreadNotification})` : ''} {data?.user?.username} · stocknear</title>
|
|
<meta charset="utf-8" />
|
|
<meta name="viewport" content="width=device-width" />
|
|
|
|
<meta name="description" content="Explore {data?.user?.username}'s latest posts, comments, and notebooks on stocknear. Discover new insights and connect with other users in the stocknear community.">
|
|
<!-- Other meta tags -->
|
|
<meta property="og:title" content="{data?.user?.username} · stocknear"/>
|
|
<meta property="og:description" content="Explore {data?.user?.username}'s latest posts, comments, and notebooks on stocknear. Discover new insights and connect with other users in the stocknear community.">
|
|
<meta property="og:image" content="https://stocknear-pocketbase.s3.amazonaws.com/logo/meta_logo.jpg"/>
|
|
<meta property="og:type" content="website"/>
|
|
<!-- Add more Open Graph meta tags as needed -->
|
|
|
|
<!-- Twitter specific meta tags -->
|
|
<meta name="twitter:card" content="summary_large_image"/>
|
|
<meta name="twitter:title" content="{data?.user?.username} · stocknear"/>
|
|
<meta name="twitter:description" content="Explore {data?.user?.username}'s latest posts, comments, and notebooks on stocknear. Discover new insights and connect with other users in the stocknear community.">
|
|
<meta name="twitter:image" content="https://stocknear-pocketbase.s3.amazonaws.com/logo/meta_logo.jpg"/>
|
|
<!-- Add more Twitter meta tags as needed -->
|
|
</svelte:head>
|
|
|
|
<section>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<body class="bg-[#0F0F0F] text-slate-200 overflow-hidden mt-10 sm:mt-5">
|
|
|
|
<!-- Page wrapper -->
|
|
<div class="flex flex-col min-h-screen overflow-hidden">
|
|
|
|
|
|
<main class="grow">
|
|
|
|
|
|
|
|
<!-- Page content -->
|
|
<section>
|
|
<div class="w-full max-w-6xl m-auto sm:px-20 ml-auto ">
|
|
|
|
|
|
<div class="w-full flex flex-row">
|
|
<!--Start Profile Pic-->
|
|
<div class="flex items-center justify-start mb-5 w-screen sm:w-full bg-[#202020] h-48 sm:rounded-xl border border-gray-700 sm:hover:border-gray-600">
|
|
<form
|
|
on:submit={handleAvatar}
|
|
class="ml-5"
|
|
>
|
|
<input
|
|
type="file"
|
|
name="avatar"
|
|
id="avatar"
|
|
value=""
|
|
accept="image/*"
|
|
hidden
|
|
on:change={showPreview}
|
|
/>
|
|
<label for="avatar" class="avatar w-20 h-20 sm:w-24 sm:h-24 rounded-full hover:cursor-pointer">
|
|
<label for="avatar" class="absolute z-10 -bottom-0.5 -right-0.5 hover:cursor-pointer">
|
|
<span class="btn btn-circle btn-sm btn-secondary">
|
|
<svg xmlns="http://www.w3.org/2000/svg" class="w-5 h-5 sm:w-6 sm:h-6" viewBox="0 0 24 24"><path fill="white" d="m14.06 9l.94.94L5.92 19H5v-.92L14.06 9m3.6-6c-.25 0-.51.1-.7.29l-1.83 1.83l3.75 3.75l1.83-1.83c.39-.39.39-1.04 0-1.41l-2.34-2.34c-.2-.2-.45-.29-.71-.29m-3.6 3.19L3 17.25V21h3.75L17.81 9.94l-3.75-3.75Z"/></svg>
|
|
</span>
|
|
</label>
|
|
<img style="clip-path: circle(50%);" class="w-24 bg-slate-300 border border-slate-400 rounded-full inline-block "
|
|
src={data?.user?.avatar
|
|
? getImageURL(data?.user?.collectionId, data?.user?.id, data?.user?.avatar)
|
|
: `https://api.dicebear.com/7.x/thumbs/svg?scale=200`}
|
|
alt="User avatar"
|
|
id="avatar-preview"
|
|
/>
|
|
|
|
|
|
</label>
|
|
<button id="submit-btn" class="hidden" type="submit"></button>
|
|
|
|
</form>
|
|
<div class="mt-5 ml-5 p-2">
|
|
<p class="text-sm sm:text-xl text-gray-200 font-semibold">
|
|
@{data?.user?.username}
|
|
{#if isModerator(data, moderators)}
|
|
<svg class="inline-block w-5 h-5 -ml-0.5 mr-0.5 mb-0.5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><path fill="#75d377" d="M256 32C174 69.06 121.38 86.46 32 96c0 77.59 5.27 133.36 25.29 184.51a348.86 348.86 0 0 0 71.43 112.41c49.6 52.66 104.17 80.4 127.28 87.08c23.11-6.68 77.68-34.42 127.28-87.08a348.86 348.86 0 0 0 71.43-112.41C474.73 229.36 480 173.59 480 96c-89.38-9.54-142-26.94-224-64Z"/></svg>
|
|
{/if}
|
|
</p>
|
|
<span class="text-sm text-gray-200">
|
|
Joined on {new Date(data?.user?.created ?? null)?.toLocaleString('en-US', { month: 'short', day: 'numeric', year: 'numeric', daySuffix: '2-digit' })}
|
|
</span>
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
<!-- Sidebar -->
|
|
<aside class="hidden lg:inline-block h-sh lg:w-1/2 lg:pr-2 xl:pr-0">
|
|
<div class="lg:pl-5 z-20 h-full">
|
|
|
|
<!-- Sidebar content -->
|
|
|
|
<!--Start User Profile -->
|
|
<div class="space-y-6 ml-4">
|
|
<div class="rounded-xl bg-[#202020] h-48 w-full border border-gray-700 font-mono">
|
|
<!--Start Header-->
|
|
<div class="ml-2 w-full p-3">
|
|
<span class="text-white text-lg font-medium ml-0.5">User Profile</span>
|
|
</div>
|
|
<hr class="border-b border-gray-700"/>
|
|
<!--End Header-->
|
|
<!--Start Content-->
|
|
<div class="w-full p-2">
|
|
|
|
<table class="font-semibold table table-compact bg-[#202020] text-start flex justify-start items-center w-full px-3 m-auto">
|
|
<tbody class="bg-[#202020]">
|
|
<!-- row 1 -->
|
|
<tr class="text-gray-300">
|
|
<td class="bg-[#202020] border-b border-[#202020]">Karma: {data?.user?.karma}</td>
|
|
<td class="bg-[#202020 border-b border-[#202020]">Posts: {userStats?.numberOfPosts}</td>
|
|
</tr>
|
|
<!-- row 2 -->
|
|
<tr class="text-gray-300">
|
|
<td class="bg-[#202020]">Comments: {userStats?.numberOfComments}</td>
|
|
<td class="bg-[#202020]"></td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
<!--End User Profile -->
|
|
|
|
{#if data?.user?.tier === 'Pro'}
|
|
<!--Start Badge-->
|
|
<div class="rounded-xl bg-[#202020] h-48 w-full border border-gray-700 mt-14">
|
|
<!--Start Header-->
|
|
<div class="ml-2 w-full p-3">
|
|
<span class="text-white text-lg font-medium ml-0.5 ">Badge</span>
|
|
</div>
|
|
<hr class="border-b border-gray-700"/>
|
|
<!--End Header-->
|
|
<!--Start Content-->
|
|
<div class="w-full p-2 flex flex-col items-start">
|
|
<div class="ml-2 mt-3 rounded-full border border-gray-500 w-16 h-16 relative bg-[#20202E] flex items-center justify-center">
|
|
<svg style="clip-path: circle(50%);" class="rounded-full w-10" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><path fill="#F9784E" d="M273.857 21.904c-24.193.012-51.198 5.552-81.1 17.467c143.7 12.608 150.35 129.263 84.032 132.814c-85.27 4.565-53.232-57.217-133.34-103.03C200.445 201.48 94.44 190.33 21.054 59.23c12.805 85.755 24.28 116.942 78.26 153.596C261.996 323.294 94.618 347.8 36.82 245.53c14.568 93.454 68.364 132.803 131.707 139.93c-42.753 24.49-99.452 32.49-143.01 25.556c51.025 42.317 131.606 40.94 193.515 8.576c-37.137 36.123-97.446 70.644-116.803 74.728H276.36C517 405.563 530.305 232.45 454.827 124.492c-2.433 26.21-10.08 49.507-25.545 70.23c-18.48-102.394-69.02-172.86-155.426-172.818zm2.82 184.666l141.384 52.155c.286-3.207.86-6.495 1.747-9.807c5.62-20.973 21.605-34.913 35.705-31.135s20.973 23.842 15.353 44.815c-5.62 20.974-21.603 34.914-35.703 31.136a18 18 0 0 1-2.113-.72l-60.58 49.394l70.637 19.584l-140.023 84.71l65.848-68.866l-31.32-7.006l-150.335 122.58l158.06-196.89l-137.39-41.137l137.006 5.654l-68.275-54.467z"/></svg>
|
|
</div>
|
|
<span class="ml-7 mt-2 text-white font-mono font-medium">Pro</span>
|
|
</div>
|
|
<!--End Content-->
|
|
</div>
|
|
<!--End Badge-->
|
|
{/if}
|
|
|
|
<div class="flex flex-col items-start {data?.user?.tier === 'Pro' ? 'mt-3' : 'mt-12'} ml-2 font-sans">
|
|
<div class="flex flex-row gap-x-3">
|
|
<a href="/about" class="text-sm text-gray-400 hover:text-gray-300 hover:underline">
|
|
About
|
|
</a>
|
|
<a href="/terms-of-use" class="text-sm text-gray-400 hover:text-gray-300 hover:underline">
|
|
Terms
|
|
</a>
|
|
<a href="/privacy-policy" class="text-sm text-gray-400 hover:text-gray-300 hover:underline">
|
|
Privacy
|
|
</a>
|
|
<a href="/contact" class="text-sm text-gray-400 hover:text-gray-300 hover:underline">
|
|
Contact
|
|
</a>
|
|
<a href="/imprint" class="text-sm text-gray-400 hover:text-gray-300 hover:underline">
|
|
Imprint
|
|
</a>
|
|
|
|
</div>
|
|
|
|
<span class="text-sm text-gray-400 mt-1.5">
|
|
© 2024 stocknear
|
|
</span>
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
</div>
|
|
</aside>
|
|
|
|
|
|
</div>
|
|
|
|
<!--End Profile Pic-->
|
|
|
|
|
|
|
|
<div class="w-full" >
|
|
<ul class="w-full font-medium flex flex-row items-center bg-[#0F0F0F] space-x-5 rtl:space-x-reverse py-2">
|
|
<li class="cursor-pointer flex flex-col items-center">
|
|
<label on:click={() => changeTab('post')} class="cursor-pointer px-3 text-sm sm:text-[0.9rem] font-medium text-gray-400 sm:hover:text-white {showTab === 'post' ? 'text-white ' : 'bg-[#0F0F0F]'}" >
|
|
Posts
|
|
</label>
|
|
<div class="{showTab === 'post' ? 'bg-[#75D377]' : 'bg-[#0F0F0F]'} mt-1 h-[3px] rounded-full w-[2.6rem]" />
|
|
</li>
|
|
<li class="cursor-pointer flex flex-col items-center">
|
|
<label on:click={() => changeTab('settings')} class="cursor-pointer px-3 text-sm sm:text-[0.9rem] font-medium text-gray-400 sm:hover:text-white {showTab === 'settings' ? 'text-white ' : 'bg-[#0F0F0F]'}" >
|
|
Settings
|
|
</label>
|
|
<div class="{showTab === 'settings' ? 'bg-[#75D377]' : 'bg-[#0F0F0F]'} mt-1 h-[3px] rounded-full w-[3.5rem]" />
|
|
</li>
|
|
<li class="cursor-pointer flex flex-col items-center">
|
|
<label on:click={() => changeTab('subscription')} class="cursor-pointer px-3 text-sm sm:text-[0.9rem] font-medium text-gray-400 sm:hover:text-white {showTab === 'subscription' ? 'text-white ' : 'bg-[#0F0F0F]'}" >
|
|
Subscription
|
|
</label>
|
|
<div class="{showTab === 'subscription' ? 'bg-[#75D377]' : 'bg-[#0F0F0F]'} mt-1 h-[3px] rounded-full w-[5rem]" />
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
|
|
|
|
<div class="md:flex md:justify-between">
|
|
<!-- Main content -->
|
|
|
|
|
|
<div class="md:grow pb-12 md:pb-20">
|
|
<div class="md:pr-6 lg:pr-10 mt-6">
|
|
{#if showTab === 'post'}
|
|
|
|
<CreateNewPost data={data?.user}/>
|
|
|
|
<div class="{!loading? 'hidden' : ''}">
|
|
{#each Array(5) as _}
|
|
<SkeletonLoading />
|
|
{/each}
|
|
</div>
|
|
{#if posts?.length === 0}
|
|
<div class="flex flex-col justify-center items-center ">
|
|
<p class="text-center text-gray-400 text-md sm:text-xl mt-10">You didn't post anything yet</p>
|
|
<p class="text-center text-gray-400 text-md sm:text-xl mt-3">Contribute to the community and make your first post</p>
|
|
</div>
|
|
{:else}
|
|
{#each posts as post, index}
|
|
<div class="flex items-start w-full">
|
|
<div class="w-full m-auto ">
|
|
<PostSection
|
|
data={data}
|
|
posts= {post}
|
|
moderators={moderators}
|
|
/>
|
|
</div>
|
|
|
|
</div>
|
|
|
|
|
|
{/each}
|
|
<InfiniteLoading on:infinite={infiniteHandler} />
|
|
|
|
{/if}
|
|
|
|
<!--Start Settings Tab-->
|
|
{:else if showTab === 'settings'}
|
|
|
|
|
|
<div class="flex flex-col w-full h-screen p-2">
|
|
<div class="w-full">
|
|
|
|
<div class="tabs pb-10 pt-10 flex flex-row items-center">
|
|
<label on:click={() => changeTab('personalData')} class="tab tab-sm pr-5 text-gray-400 hover:text-white {settingsTab === 'personalData' ? 'rounded text-white bg-[#333333]' : 'bg-[#0F0F0F]'}">Personal Data</label>
|
|
<label on:click={() => changeTab('changePassword')} class="tab tab-sm pl-5 text-gray-400 hover:text-white {settingsTab === 'changePassword' ? 'rounded text-white bg-[#333333]' : 'bg-[#0F0F0F]'}">Change Password</label>
|
|
</div>
|
|
|
|
{#if settingsTab ==='personalData'}
|
|
<form
|
|
on:submit={updatePersonalData}
|
|
class="flex flex-col space-y-2 w-full"
|
|
>
|
|
<Input
|
|
id="username"
|
|
type="text"
|
|
label="Username"
|
|
required={true}
|
|
disabled={loading}
|
|
errors={errorUsername}
|
|
/>
|
|
<!--
|
|
<Input
|
|
id="email"
|
|
type="email"
|
|
label="Email address"
|
|
required={true}
|
|
value={form?.data?.email ?? data?.user?.email}
|
|
disabled={loading}
|
|
errors={form?.errors?.email}
|
|
/>
|
|
-->
|
|
<div class="w-full max-w-lg pt-3">
|
|
<button type="submit" class="btn bg-blue-700 hover:bg-blue-600 text-white w-full max-w-lg normal-case">Update your data</button>
|
|
</div>
|
|
|
|
</form>
|
|
{:else}
|
|
<form
|
|
on:submit={updatePassword}
|
|
class="flex flex-col space-y-2 w-full"
|
|
>
|
|
<Input
|
|
id="oldPassword"
|
|
label="Old Password"
|
|
type="password"
|
|
required
|
|
errors={errorOldPassword}
|
|
disabled={loading}
|
|
/>
|
|
<Input
|
|
id="password"
|
|
label="New Password"
|
|
type="password"
|
|
required
|
|
errors={errorPassword}
|
|
disabled={loading}
|
|
/>
|
|
<Input
|
|
id="passwordConfirm"
|
|
label="Confirm New Password"
|
|
type="password"
|
|
required
|
|
errors={errorPasswordConfirm}
|
|
disabled={loading}
|
|
/>
|
|
|
|
<div class="w-full max-w-lg pt-3">
|
|
<button type="submit" class="btn bg-blue-700 text-white hover:bg-blue-600 w-full max-w-lg normal-case text-md">Update Password</button>
|
|
</div>
|
|
</form>
|
|
{/if}
|
|
</div>
|
|
</div>
|
|
|
|
<!--End Settings Tab-->
|
|
<!--Start Info Tab -->
|
|
{:else if showTab === 'subscription'}
|
|
<div class="flex flex-col sm:flex-row overflow-hidden pl-5 p-0 sm:p-5">
|
|
|
|
|
|
<div class="flex flex-col justify-start items-start w-full text-white">
|
|
|
|
<h2 class="text-xl sm:text-2xl font-bold text-start mt-5 mb-5">
|
|
Manage Subscription
|
|
</h2>
|
|
<div class="flex flex-row items-center">
|
|
<span class="text-white text-[1rem] sm:text-lg">
|
|
Status:
|
|
</span>
|
|
<div class="ml-2 flex flex-row items-center">
|
|
|
|
<span class="relative flex h-2 w-2 ">
|
|
<span class="animate-ping absolute inline-flex h-full w-full rounded-full {subscriptionData?.status_formatted === 'Active' || subscriptionData?.status_formatted === 'Paid' || subscriptionData?.status_formatted === 'On Trial' ? 'bg-[#10DB06]' : 'bg-[#FF3131]'} opacity-75"></span>
|
|
<span class="relative inline-flex rounded-full h-2 w-2 {subscriptionData?.status_formatted === 'Active' || subscriptionData?.status_formatted === 'Paid' || subscriptionData?.status_formatted === 'On Trial' ? 'bg-[#10DB06]' : 'bg-[#FF3131]'}"></span>
|
|
</span>
|
|
|
|
<span class="ml-2 text-[1rem] text-slate-200 font-medium">
|
|
{subscriptionData?.status_formatted ?? 'Inactive'}
|
|
</span>
|
|
</div>
|
|
|
|
</div>
|
|
{#if subscriptionData?.status_formatted === 'Active'}
|
|
<span class="text-white text-sm font-medium pr-5">
|
|
Your subscription will automatically renew on {new Date(subscriptionData?.renews_at)?.toLocaleDateString('en-GB', { day: 'numeric', month: 'long', year: 'numeric' })}
|
|
</span>
|
|
{:else if subscriptionData?.status_formatted === 'On Trial'}
|
|
<span class="text-white text-sm font-medium pr-5">
|
|
Your trial will end on {new Date(subscriptionData?.trial_ends_at)?.toLocaleDateString('en-GB', { day: 'numeric', month: 'long', year: 'numeric' })}.
|
|
</span>
|
|
{:else if subscriptionData?.status_formatted === 'Cancelled'}
|
|
<span class="text-white text-sm font-medium">
|
|
Your subscription will remain active until {new Date(subscriptionData?.ends_at)?.toLocaleDateString('en-GB', { day: 'numeric', month: 'long', year: 'numeric' })}
|
|
</span>
|
|
{/if}
|
|
|
|
|
|
<div class="flex flex-col justify-start items-start mt-8">
|
|
<span class="text-white font-medium mr-2 text-lg">
|
|
Current Plan:
|
|
</span>
|
|
<span class="text-[1rem]">
|
|
{['Active', 'Paid', 'On Trial', 'Cancelled']?.includes(subscriptionData?.status_formatted) ? subscriptionData?.product_name : 'Free Subscription'}
|
|
</span>
|
|
<span class="text-sm text-white {subscriptionData?.status_formatted !== 'Active' ? 'hidden' : ''}">
|
|
{subscriptionData?.product_name?.includes('Monthly') ? '$9.99 billed every month' : '$90 billed every year'}
|
|
</span>
|
|
</div>
|
|
|
|
{#if subscriptionData?.status_formatted === 'Active' || subscriptionData?.status_formatted === 'On Trial'}
|
|
<label for="cancelSubscriptionModal" class="cursor-pointer text-white bg-[#FF3131] hover:bg-red-600 bg-opacity-[0.5] text-sm sm:text-[1rem] px-4 py-2 rounded-lg mt-5">
|
|
Cancel Subscription
|
|
</label>
|
|
{:else if subscriptionData?.status_formatted === 'Cancelled'}
|
|
<label for="reactivateSubscriptionModal" class="cursor-pointer text-white bg-[#75D377] bg-opacity-[0.5] text-sm sm:text-[1rem] px-4 py-2 rounded-lg mt-5">
|
|
Reactivate Subscription
|
|
</label>
|
|
{:else if subscriptionData?.status_formatted === 'Paid'}
|
|
<span class="text-white mt-5">
|
|
Please wait a moment; you will be updated to Pro in a second.
|
|
</span>
|
|
{:else}
|
|
<a href="/pricing" class="text-blue-400 mt-5">
|
|
Get Full Access with Pro Subscription.
|
|
</a>
|
|
{/if}
|
|
|
|
|
|
</div>
|
|
|
|
</div>
|
|
<!--End Info Tab-->
|
|
{/if}
|
|
</div>
|
|
</div>
|
|
|
|
|
|
<!-- Sidebar -->
|
|
<aside class="hidden {showTab === 'post' ? 'lg:inline-block' : 'hidden'} h-sh w-[300px] pt-[1.5rem]">
|
|
<div class="lg:pl-5 z-20 h-full invisible">
|
|
|
|
<!-- Sidebar content -->
|
|
|
|
<!--Start User Profile -->
|
|
<div class="space-y-6">
|
|
<div class="shadow-lg rounded-md bg-[#202020] h-auto w-full md:w-80 border border-gray-700">
|
|
<!--Start Header-->
|
|
<div class="bg-[#202020] w-full p-3 ">
|
|
<svg style="clip-path: circle(50%);" class="flex-shrink-0 w-10 h-10 inline-block" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 48 48"><g fill="white"><path d="M32 20a8 8 0 1 1-16 0a8 8 0 0 1 16 0"/><path fill-rule="evenodd" d="M23.184 43.984C12.517 43.556 4 34.772 4 24C4 12.954 12.954 4 24 4s20 8.954 20 20s-8.954 20-20 20h-.274q-.272 0-.542-.016M11.166 36.62a3.028 3.028 0 0 1 2.523-4.005c7.796-.863 12.874-.785 20.632.018a2.99 2.99 0 0 1 2.498 4.002A17.94 17.94 0 0 0 42 24c0-9.941-8.059-18-18-18S6 14.059 6 24c0 4.916 1.971 9.373 5.166 12.621" clip-rule="evenodd"/></g></svg>
|
|
<span class="text-white text-md ml-0.5 ">User Profile</span>
|
|
</div>
|
|
<!--End Header-->
|
|
<!--Start Content-->
|
|
<div class="w-full p-2 flex-1 flex flex-wrap">
|
|
|
|
<table class="table table-compact bg-[#202020] text-start flex justify-start items-center w-full px-3 m-auto">
|
|
<tbody class="bg-[#202020]">
|
|
<!-- row 1 -->
|
|
<tr class="text-gray-300">
|
|
<td class="bg-[#202020] border-b border-[#202020]">Karma: {data?.user?.karma}</td>
|
|
<td class="bg-[#202020 border-b border-[#202020]">Posts: {userStats?.numberOfPosts}</td>
|
|
</tr>
|
|
<!-- row 2 -->
|
|
<tr class="text-gray-300">
|
|
<td class="bg-[#202020]">Comments: {userStats?.numberOfComments}</td>
|
|
<td class="bg-[#202020]"></td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
<!--End User Profile -->
|
|
|
|
|
|
</div>
|
|
</div>
|
|
</aside>
|
|
|
|
|
|
</div>
|
|
|
|
</div>
|
|
</section>
|
|
|
|
</main>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
</body>
|
|
|
|
|
|
|
|
</section>
|
|
|
|
|
|
|
|
|
|
<!-- Start Cancel Subscription Modal -->
|
|
<input type="checkbox" id="cancelSubscriptionModal" class="modal-toggle" />
|
|
|
|
<dialog id="cancelSubscriptionModal" class="modal modal-bottom sm:modal-middle">
|
|
|
|
|
|
<label for="cancelSubscriptionModal" class="cursor-pointer modal-backdrop bg-[#000] bg-opacity-[0.5]"></label>
|
|
|
|
|
|
<!-- Desktop modal content -->
|
|
<form method="POST" action="?/cancelSubscription" use:enhance={submitCancellation} class="modal-box w-full bg-[#202020] flex flex-col items-center">
|
|
<div class="mx-auto mb-8 h-1.5 w-20 flex-shrink-0 rounded-full bg-[#404040]" />
|
|
<div class="text-white mb-5 text-center">
|
|
<h3 class="font-bold text-2xl mb-5">Are you sure?</h3>
|
|
<span class="text-white text-[1rem] font-normal">
|
|
You will no longer be charged for this subscription, and at the end of the billing period, your account will transfer to the Free Plan.
|
|
</span>
|
|
</div>
|
|
|
|
<button on:click={() => isClicked = !isClicked} class="{!isClicked ? '' : 'hidden'} cursor-pointer px-7 py-2 mb-5 rounded-full bg-red-600 text-center text-white text-[1rem] font-normal">
|
|
Cancel Subscription
|
|
<input class="hidden" name='subscriptionId' value={subscriptionData?.first_subscription_item?.subscription_id}/>
|
|
</button>
|
|
{#if isClicked === true}
|
|
<label class="cursor-pointer px-7 py-2 mb-5 rounded-full bg-red-600 text-center text-white text-[1rem] font-normal">
|
|
<div class="flex flex-row m-auto">
|
|
<span class="loading loading-infinity"></span>
|
|
<span class="text-white ml-2">Proceeding</span>
|
|
</div>
|
|
</label>
|
|
{/if}
|
|
|
|
</form>
|
|
</dialog>
|
|
<!-- End Cancel Subscription Modal -->
|
|
|
|
|
|
|
|
<!-- Start Reactivate Subscription Modal -->
|
|
<input type="checkbox" id="reactivateSubscriptionModal" class="modal-toggle" />
|
|
|
|
<dialog id="reactivateSubscriptionModal" class="modal modal-bottom sm:modal-middle">
|
|
|
|
|
|
<label for="reactivateSubscriptionModal" class="cursor-pointer modal-backdrop bg-[#000] bg-opacity-[0.5]"></label>
|
|
|
|
|
|
<!-- Desktop modal content -->
|
|
<form method="POST" action="?/reactivateSubscription" use:enhance={submitReactivate} class="modal-box w-full bg-[#202020] flex flex-col items-center">
|
|
<div class="mx-auto mb-8 h-1.5 w-20 flex-shrink-0 rounded-full bg-[#404040]" />
|
|
<div class="text-white mb-5 text-center">
|
|
<h3 class="font-bold text-2xl mb-5">Reactivate Subscription</h3>
|
|
<span class="text-white text-[1rem] font-normal">
|
|
Reactivate your Pro Subscription now to unlock unlimited features and gain the edge over the competition.
|
|
</span>
|
|
</div>
|
|
|
|
|
|
<button on:click={() => isClicked = !isClicked} class="{!isClicked ? '' : 'hidden'} cursor-pointer px-7 py-2 mb-5 rounded-full bg-[#417143] text-center text-white text-[1rem] font-normal">
|
|
Reactivate Subscription
|
|
<input class="hidden" name='subscriptionId' value={subscriptionData?.first_subscription_item?.subscription_id}/>
|
|
</button>
|
|
{#if isClicked === true}
|
|
<label class="cursor-pointer px-7 py-2 mb-5 rounded-full bg-[#417143] text-center text-white text-[1rem] font-normal">
|
|
<div class="flex flex-row m-auto">
|
|
<span class="loading loading-infinity"></span>
|
|
<span class="text-white ml-2">Proceeding</span>
|
|
</div>
|
|
</label>
|
|
{/if}
|
|
|
|
</form>
|
|
</dialog>
|
|
<!-- End Reactivate Subscription Modal -->
|
|
|