From 8dc366d6d130782b2acd753a3cf1e29c21b95c53 Mon Sep 17 00:00:00 2001 From: sairaj mote Date: Mon, 26 Feb 2024 02:00:26 +0530 Subject: [PATCH] adding ability for sub admin to view updates --- css/main.css | 21 ++++- css/main.min.css | 2 +- css/main.scss | 26 +++++- scripts/app.js | 220 ++++++++++++++++++++++++++------------------- scripts/app.min.js | 110 ++++++++++++----------- 5 files changed, 233 insertions(+), 146 deletions(-) diff --git a/css/main.css b/css/main.css index 4ca5214..5b84ef5 100644 --- a/css/main.css +++ b/css/main.css @@ -932,6 +932,21 @@ ol { grid-area: task-actions; } +.update-card { + border-radius: 0.5rem; + padding: max(1rem, 1.5vw); + background-color: rgba(var(--text-color), 0.06); +} +.update-card__sender, .update-card__time { + font-size: 0.8rem; + font-weight: 500; + color: rgba(var(--text-color), 0.8); +} +.update-card__message { + margin-top: 0.5rem; + white-space: pre-line; +} + #profile__header h1, #profile__header h4 { line-height: 0.7; @@ -1052,12 +1067,16 @@ ol { #landing #relative_tasks > div:last-of-type { margin-top: 3rem; } - #home > section { + #home > section, +#task > section { padding: 0 8vw; } #task_popup { --width: 36rem; } + #send_update_popup { + --width: 36rem; + } #profile section { gap: 4rem; justify-content: center; diff --git a/css/main.min.css b/css/main.min.css index e7de223..c8f565a 100644 --- a/css/main.min.css +++ b/css/main.min.css @@ -1 +1 @@ -*{padding:0;margin:0;box-sizing:border-box;font-family:"IBM Plex Sans",sans-serif}:root{font-size:clamp(1rem,1.2vmax,1.2rem)}html,body{height:100%}body{--accent-color: #4d77ff;--accent-color-rgb: 77, 119, 255;--text-color: 20, 20, 20;--foreground-color: 255, 255, 255;--background-color: 248, 245, 242;--danger-color: rgb(225, 58, 58);--green: #1cad59;--yellow: rgb(255, 207, 65);scrollbar-width:thin;scrollbar-gutter:stable;color:rgba(var(--text-color), 1);background-color:rgba(var(--background-color), 1);position:relative;display:flex;flex-direction:column;color-scheme:light}body[data-theme=dark]{--accent-color: #a0b6ff;--accent-color-rgb: 160, 182, 255;--text-color: 220, 220, 220;--foreground-color: 27, 28, 29;--background-color: 21, 22, 22;--danger-color: rgb(255, 106, 106);--green: #00e676;--yellow: rgb(255, 213, 5);color-scheme:dark}body[data-theme=dark] sm-popup::part(popup){background-color:rgba(var(--foreground-color), 1)}input[type=date]{display:flex;width:100%;padding:.8rem .6rem;border:none;border-radius:.5rem;font-weight:500;font-family:inherit;font-size:inherit;color:inherit;background-color:rgba(var(--text-color), 0.06)}input[type=date]:focus{outline:none;box-shadow:0 0 0 .1rem var(--accent-color)}p,strong{font-size:.9rem;max-width:65ch;line-height:1.7;color:rgba(var(--text-color), 0.9)}.warning{line-height:normal;padding:1rem;background-color:khaki;border-radius:.5rem;font-weight:500;color:rgba(0,0,0,.7)}a{text-decoration:none;color:var(--accent-color)}a:focus-visible{box-shadow:0 0 0 .1rem rgba(var(--text-color), 1) inset}h1{font-size:2.5rem}h2{font-size:1.5rem}h3{font-size:1.2rem}h4{font-size:1rem}h5{font-size:.9rem}h6{font-size:.8rem}button,.button{-webkit-user-select:none;-moz-user-select:none;user-select:none;position:relative;display:inline-flex;border:none;background-color:rgba(0,0,0,0);overflow:hidden;-webkit-tap-highlight-color:rgba(0,0,0,0);align-items:center;font-size:.9rem;font-weight:700;white-space:nowrap;padding:.5rem 1rem;border-radius:.5rem;justify-content:center;flex-shrink:0}button:focus-visible,.button:focus-visible{outline:var(--accent-color) solid medium}button:not(:disabled),.button:not(:disabled){color:var(--accent-color);cursor:pointer}button .icon,.button .icon{fill:var(--accent-color)}.button{display:inline-flex;background-color:rgba(var(--text-color), 0.06);color:rgba(var(--text-color), 1)}.button--primary,.button--danger{color:rgba(var(--background-color), 1) !important}.button--primary .icon,.button--danger .icon{fill:rgba(var(--background-color), 1)}.button--primary{padding:.8rem 1.2rem;background-color:rgba(var(--text-color), 1)}.button--danger{background-color:var(--danger-color)}.button--small{padding:.4rem .5rem}.button--colored{color:var(--accent-color) !important}.button--colored .icon{fill:var(--accent-color)}.button--outlined{border:solid .12rem rgba(var(--text-color), 1);background-color:rgba(0,0,0,0);color:rgba(var(--text-color), 1) !important}.cta{text-transform:uppercase;font-size:.9rem;font-weight:700;letter-spacing:.05em;padding:.8rem 1rem}.icon{width:1.2rem;height:1.2rem;fill:rgba(var(--text-color), 0.8);flex-shrink:0}.icon-only{padding:.5rem;border-radius:.3rem}button:disabled{opacity:.5}a:-webkit-any-link:focus-visible{outline:rgba(var(--text-color), 1) .1rem solid}a:-moz-any-link:focus-visible{outline:rgba(var(--text-color), 1) .1rem solid}a:any-link:focus-visible{outline:rgba(var(--text-color), 1) .1rem solid}details summary{display:flex;gap:.3rem;-webkit-user-select:none;-moz-user-select:none;user-select:none;cursor:pointer;align-items:center;color:var(--accent-color)}details summary .down-arrow{fill:var(--accent-color)}details[open] summary{margin-bottom:1rem}details[open]>summary .down-arrow{transform:rotate(180deg)}sm-input,sm-textarea{width:100%;--border-radius: 0.5rem;--background-color: rgba(var(--foreground-color), 1)}sm-input button .icon,sm-textarea button .icon{fill:var(--accent-color)}sm-button{--padding: 0.8rem}sm-button[variant=primary] .icon{fill:rgba(var(--background-color), 1)}sm-button[disabled] .icon{fill:rgba(var(--text-color), 0.6)}sm-button.danger{--background: var(--danger-color);color:rgba(var(--background-color), 1)}sm-spinner{--size: 1rem;--stroke-width: 0.1rem}sm-form{--gap: 1rem}sm-select{font-size:.9rem;--padding: 0.6rem 0.3rem 0.6rem 0.6rem}sm-option{font-size:.9rem;--border-radius: 0.3rem}sm-chips{--gap: 0.3rem;background-color:rgba(var(--text-color), 0.06);border-radius:.7rem;padding:.3rem}sm-chip{font-size:.9rem;--border-radius: 0.5rem;-webkit-user-select:none;-moz-user-select:none;user-select:none;font-weight:500}sm-chip[selected]{color:rgba(var(--background-color), 1);--background: var(--accent-color)}sm-notifications{z-index:100000}ul,ol{list-style:none}.overflow-ellipsis{width:100%;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}.wrap-around{overflow-wrap:break-word;word-wrap:break-word;word-break:break-word}.full-bleed{grid-column:1/-1}.uppercase{text-transform:uppercase}.capitalize{text-transform:capitalize}.sticky{position:-webkit-sticky;position:sticky}.top-0{top:0}.flex{display:flex}.flex-column{display:flex;flex-direction:column}.flex-wrap{flex-wrap:wrap}.flex-1{flex:1}.flex-direction-column{flex-direction:column}.grid{display:grid}.flow-column{grid-auto-flow:column}.gap-0-3{gap:.3rem}.gap-0-5{gap:.5rem}.gap-1{gap:1rem}.gap-1-5{gap:1.5rem}.gap-2{gap:2rem}.gap-3{gap:3rem}.text-align-right{text-align:right}.align-content-start{align-content:flex-start}.align-items-start{align-items:flex-start}.align-self-start{align-self:flex-start}.align-center{align-items:center}.align-end{align-items:flex-end}.text-center{text-align:center}.justify-start{justify-items:start}.justify-center{justify-content:center}.justify-right{margin-left:auto}.justify-items-center{justify-items:center}.align-self-center{align-self:center}.align-self-end{align-self:end}.justify-self-center{justify-self:center}.justify-self-start{justify-self:start}.justify-self-end{justify-self:end}.flex-direction-column{flex-direction:column}.space-between{justify-content:space-between}.w-100{width:100%}.h-100{height:100%}.label{font-size:.9rem;color:rgba(var(--text-color), 0.8);font-weight:500;margin-bottom:.3rem}.button--primary .ripple,.button--danger .ripple{background:radial-gradient(circle, rgba(var(--background-color), 0.3) 0%, rgba(0, 0, 0, 0) 50%)}.ripple{height:8rem;width:8rem;position:absolute;border-radius:50%;transform:scale(0);background:radial-gradient(circle, rgba(var(--text-color), 0.3) 0%, rgba(0, 0, 0, 0) 50%);pointer-events:none}.interactive{position:relative;overflow:hidden;cursor:pointer;-webkit-tap-highlight-color:rgba(0,0,0,0);-webkit-user-select:none;-moz-user-select:none;user-select:none}.empty-state{display:grid;width:100%;padding:1.5rem 0}.observe-empty-state:empty{display:none !important}.observe-empty-state:not(:empty)+.empty-state{display:none !important}.bullet-point{display:flex;align-items:center;justify-content:center;margin:0 .8ch}.bullet-point::after{content:"";height:.4ch;width:.4ch;border-radius:.5em;background-color:var(--accent-color)}.margin-right-0-3{margin-right:.3rem}.margin-right-0-5{margin-right:.5rem}.margin-right-auto{margin-right:auto}.margin-left-0-5{margin-left:.5rem}.margin-left-auto{margin-left:auto}.margin-top-1-5{margin-top:1.5rem}.margin-bottom-0-5{margin-bottom:.5rem}.margin-bottom-1{margin-bottom:1rem}.margin-bottom-1-5{margin-bottom:1.5rem}.margin-bottom-2{margin-bottom:2rem}.icon-button{padding:.6rem;border-radius:.8rem;background-color:rgba(var(--text-color), 0.1);height:-webkit-max-content;height:-moz-max-content;height:max-content}.icon-button .icon{fill:var(--accent-color)}#confirmation_popup,#prompt_popup{flex-direction:column}#confirmation_popup h4,#prompt_popup h4{margin-bottom:1.5rem}#confirmation_popup sm-button,#prompt_popup sm-button{margin:0}#confirmation_popup>.flex:last-of-type,#prompt_popup>.flex:last-of-type{padding:0;margin-top:1rem}#confirmation_popup>.flex:last-of-type sm-button:first-of-type,#prompt_popup>.flex:last-of-type sm-button:first-of-type{margin-right:.6rem;margin-left:auto}#prompt_message{margin-bottom:1.5rem}.popup__header{display:grid;gap:.5rem;width:100%;padding:0 1.5rem;align-items:center;justify-items:flex-start}.popup__header__close{padding:.5rem;margin-left:-0.5rem}#loading{display:grid;grid-template-rows:auto 1fr}#loading section{place-content:center;justify-items:center}#loading h4{margin-top:1.5rem;font-weight:500}#loading sm-spinner{--size: 1.5rem}#sign_in,#sign_up{display:grid;height:100%;grid-template-rows:auto 1fr}#sign_in section,#sign_up section{margin:auto;width:min(26rem,100%)}#sign_in sm-form,#sign_up sm-form{margin:2rem 0}#main_header{padding:1rem max(1rem,4vw)}.app-brand{display:flex;gap:.3rem;align-items:center}.app-brand .icon{height:1.7rem;width:1.7rem}.app-name__company{font-size:.8rem;font-weight:500;color:rgba(var(--text-color), 0.8)}.password-field label{display:flex}.password-field label input:checked~.visible{display:none}.password-field label input:not(:checked)~.invisible{display:none}.multi-state-button{display:grid;text-align:center;align-items:center;justify-items:center}.multi-state-button>*{grid-area:1/1/2/2}.multi-state-button button{z-index:1;width:100%}.scrolling-wrapper{overflow-y:auto}#flo_id_warning{padding-bottom:1.5rem;border-bottom:thin solid rgba(var(--text-color), 0.3)}#flo_id_warning .icon{height:4rem;width:4rem;padding:1rem;background-color:#ffc107;border-radius:3rem;fill:rgba(0,0,0,.8);margin-bottom:1.5rem}.generated-id-card{display:grid;gap:1rem}.generated-id-card h5{margin-bottom:.3rem}#app_body{display:grid;grid-template-columns:1rem [main-margin-start] minmax(0, 1fr) [main-margin-end] 1rem;height:100%}#app_body>*{grid-column:main-margin}#main_header{padding:1rem 0;position:-webkit-sticky;position:sticky;top:0;z-index:10;background-color:rgba(var(--background-color), 0.7);-webkit-backdrop-filter:blur(1rem);backdrop-filter:blur(1rem)}#main_header:has(button){display:grid;grid-template-columns:auto minmax(0, 1fr) auto}#main_header #main_logo{height:2.5rem;width:2.5rem;padding:.6rem;border-radius:5rem;background-color:rgba(var(--text-color), 0.1)}#main_header .button{text-transform:capitalize}#main_header .button--primary{padding:.6rem 1.8rem;border:solid .12rem rgba(var(--text-color), 1)}#landing{display:grid;grid-template-rows:auto 1fr;height:100%}#landing #relative_tasks{height:100%}#landing #relative_tasks>div:first-of-type{display:grid;place-content:center;align-self:center}#landing h1 span:first-of-type{font-size:1.2rem}#landing h1 span:last-of-type{font-size:2rem}#category_selection{text-align:center;width:min(56rem,100%);margin-inline:auto;grid-template-rows:auto 1fr auto;padding:1.5rem 0;gap:1.5rem}#category_selection h1{line-height:.9;margin-bottom:1rem}#category_selection p{max-width:100%;margin-bottom:1rem}#category_selection ul{display:grid;align-self:flex-start;gap:1rem;grid-template-columns:repeat(auto-fill, minmax(11rem, 1fr))}#category_selection ul li{display:flex;-webkit-animation:popin .3s ease-out var(--delay) forwards;animation:popin .3s ease-out var(--delay) forwards;opacity:0;transform:scale(0.9)}#category_selection ul li:has(:checked) label{box-shadow:0 0 0 .1rem var(--accent-color);background-color:rgba(0,0,0,0)}#category_selection ul li:has(:checked) label:hover{box-shadow:0 0 0 .2rem var(--accent-color);background-color:rgba(0,0,0,0)}#category_selection ul li label{display:flex;align-items:center;justify-content:center;width:100%;height:100%;text-align:center;background-color:rgba(var(--text-color), 0.06);padding:1rem;border-radius:40%;cursor:pointer;font-weight:500;transition:background-color .3s,box-shadow .3s;aspect-ratio:1/1}#category_selection ul li label:hover{background-color:rgba(var(--text-color), 0.03);box-shadow:0 0 0 .1rem rgba(var(--text-color), 0.3)}#category_selection ul li input{display:none}@-webkit-keyframes popin{to{opacity:1;transform:none}}@keyframes popin{to{opacity:1;transform:none}}#category_selection button{justify-self:center;padding:1rem 3rem;border-radius:5rem;gap:.5rem}#category_selection button span{transform:translateX(1rem);transition:transform .3s}#category_selection button .icon{opacity:0;transform:translateX(-0.5rem);transition:opacity .3s,transform .3s}#category_selection button:hover span{transform:translateX(0)}#category_selection button:hover .icon{opacity:1;transform:translateX(0)}.task-card{display:grid;grid-template-columns:1fr auto;gap:.5rem;align-items:center;grid-template-areas:"task-title task-time" "task-description task-description" "task-requisites task-actions"}.task-card:not(:last-of-type){border-bottom:thin solid rgba(var(--text-color), 0.5);padding-bottom:1.5rem;margin-bottom:1.5rem}.task-card:last-of-type{padding-bottom:3rem}.task-card h4{grid-area:task-title;font-size:1.2rem}.task-card>sl-relative-time{grid-area:task-time;font-size:.8rem;color:rgba(var(--text-color), 0.6);text-align:end}.task-card p{grid-area:task-description;font-size:.9rem;color:rgba(var(--text-color), 0.8)}.task-card ul{grid-area:task-requisites}.task-card ul li{font-size:.9rem;color:rgba(var(--text-color), 0.8);font-weight:500}.task-card ul li:not(:last-of-type)::after{content:"•";margin-left:.3rem}.task-card .button{grid-area:task-actions}#profile__header h1,#profile__header h4{line-height:.7}#profile sm-form{width:min(32rem,100%)}#task_popup sm-chips{background-color:rgba(0,0,0,0);padding:0}#task_popup sm-chips sm-chip{--background: rgba(var(--text-color), 0.06) }#user_popup_button{background-color:rgba(var(--text-color), 0.06);border-radius:2rem;font-size:.8rem;padding:.6rem .8rem;color:currentColor}#user_popup_button .icon{fill:currentColor}@media screen and (max-width: 40rem){#landing #relative_tasks{align-content:flex-start}#landing #relative_tasks>div:first-of-type{padding:3rem 0}#category_selection ul{grid-template-columns:repeat(auto-fill, minmax(9rem, 1fr))}#emblem{display:none}.task-card{grid-template-areas:"task-title task-time" "task-description task-description" "task-requisites task-requisites" "task-actions task-actions"}#profile__header{padding:1rem 0;margin-top:10vh}#user_popup_button{grid-area:1/2/2/3}.hide-on-small{display:none}}@media screen and (min-width: 40rem){h1{font-size:4rem}h2{font-size:2rem}h3{font-size:1.5rem}h4{font-size:1.2rem}h5{font-size:1rem}h6{font-size:.9rem}sm-popup{--width: 24rem}.popup__header{grid-column:1/-1;padding:1rem 1.5rem 0 1.5rem}#app_body{grid-template-columns:4vw [main-margin-start] 4vw [inner-margin-start] minmax(0, 1fr) [inner-margin-end] 4vw [main-margin-end] 4vw}#main_header{padding:2rem 0}#landing #relative_tasks{display:grid;grid-template-columns:1fr 1.2fr;gap:8vw}#landing #relative_tasks>div:first-of-type{align-self:flex-start;position:-webkit-sticky;position:sticky;top:40vh}#landing #relative_tasks>div:first-of-type::after{content:"";position:absolute;height:12rem;width:.1rem;background-color:rgba(var(--text-color), 1);right:0;align-self:center;margin-top:5rem}#landing #relative_tasks>div:first-of-type #emblem{position:absolute;align-self:center;right:-1.45rem;width:3rem;height:3rem;stroke:rgba(var(--text-color), 1);stroke-width:.1rem;margin-top:5rem}#landing #relative_tasks>div:last-of-type{margin-top:3rem}#home>section{padding:0 8vw}#task_popup{--width: 36rem}#profile section{gap:4rem;justify-content:center;align-items:center}}@media(any-hover: hover){html{scrollbar-color:#888 rgba(0,0,0,0)}::-webkit-scrollbar{width:10px;height:10px}::-webkit-scrollbar-thumb{background:currentColor}::-webkit-scrollbar-track{background:rgba(0,0,0,0)}.interactive:not([disabled]){transition:background-color .3s}.interactive:not([disabled]):hover{background-color:rgba(var(--text-color), 0.06)}button:not([disabled]),.button:not([disabled]){transition:background-color .3s,filter .3s}button:not([disabled]):hover,.button:not([disabled]):hover{filter:contrast(2)}}@supports(overflow: overlay){body{overflow:overlay}}.hidden{display:none !important} \ No newline at end of file +*{padding:0;margin:0;box-sizing:border-box;font-family:"IBM Plex Sans",sans-serif}:root{font-size:clamp(1rem,1.2vmax,1.2rem)}html,body{height:100%}body{--accent-color: #4d77ff;--accent-color-rgb: 77, 119, 255;--text-color: 20, 20, 20;--foreground-color: 255, 255, 255;--background-color: 248, 245, 242;--danger-color: rgb(225, 58, 58);--green: #1cad59;--yellow: rgb(255, 207, 65);scrollbar-width:thin;scrollbar-gutter:stable;color:rgba(var(--text-color), 1);background-color:rgba(var(--background-color), 1);position:relative;display:flex;flex-direction:column;color-scheme:light}body[data-theme=dark]{--accent-color: #a0b6ff;--accent-color-rgb: 160, 182, 255;--text-color: 220, 220, 220;--foreground-color: 27, 28, 29;--background-color: 21, 22, 22;--danger-color: rgb(255, 106, 106);--green: #00e676;--yellow: rgb(255, 213, 5);color-scheme:dark}body[data-theme=dark] sm-popup::part(popup){background-color:rgba(var(--foreground-color), 1)}input[type=date]{display:flex;width:100%;padding:.8rem .6rem;border:none;border-radius:.5rem;font-weight:500;font-family:inherit;font-size:inherit;color:inherit;background-color:rgba(var(--text-color), 0.06)}input[type=date]:focus{outline:none;box-shadow:0 0 0 .1rem var(--accent-color)}p,strong{font-size:.9rem;max-width:65ch;line-height:1.7;color:rgba(var(--text-color), 0.9)}.warning{line-height:normal;padding:1rem;background-color:khaki;border-radius:.5rem;font-weight:500;color:rgba(0,0,0,.7)}a{text-decoration:none;color:var(--accent-color)}a:focus-visible{box-shadow:0 0 0 .1rem rgba(var(--text-color), 1) inset}h1{font-size:2.5rem}h2{font-size:1.5rem}h3{font-size:1.2rem}h4{font-size:1rem}h5{font-size:.9rem}h6{font-size:.8rem}button,.button{-webkit-user-select:none;-moz-user-select:none;user-select:none;position:relative;display:inline-flex;border:none;background-color:rgba(0,0,0,0);overflow:hidden;-webkit-tap-highlight-color:rgba(0,0,0,0);align-items:center;font-size:.9rem;font-weight:700;white-space:nowrap;padding:.5rem 1rem;border-radius:.5rem;justify-content:center;flex-shrink:0}button:focus-visible,.button:focus-visible{outline:var(--accent-color) solid medium}button:not(:disabled),.button:not(:disabled){color:var(--accent-color);cursor:pointer}button .icon,.button .icon{fill:var(--accent-color)}.button{display:inline-flex;background-color:rgba(var(--text-color), 0.06);color:rgba(var(--text-color), 1)}.button--primary,.button--danger{color:rgba(var(--background-color), 1) !important}.button--primary .icon,.button--danger .icon{fill:rgba(var(--background-color), 1)}.button--primary{padding:.8rem 1.2rem;background-color:rgba(var(--text-color), 1)}.button--danger{background-color:var(--danger-color)}.button--small{padding:.4rem .5rem}.button--colored{color:var(--accent-color) !important}.button--colored .icon{fill:var(--accent-color)}.button--outlined{border:solid .12rem rgba(var(--text-color), 1);background-color:rgba(0,0,0,0);color:rgba(var(--text-color), 1) !important}.cta{text-transform:uppercase;font-size:.9rem;font-weight:700;letter-spacing:.05em;padding:.8rem 1rem}.icon{width:1.2rem;height:1.2rem;fill:rgba(var(--text-color), 0.8);flex-shrink:0}.icon-only{padding:.5rem;border-radius:.3rem}button:disabled{opacity:.5}a:-webkit-any-link:focus-visible{outline:rgba(var(--text-color), 1) .1rem solid}a:-moz-any-link:focus-visible{outline:rgba(var(--text-color), 1) .1rem solid}a:any-link:focus-visible{outline:rgba(var(--text-color), 1) .1rem solid}details summary{display:flex;gap:.3rem;-webkit-user-select:none;-moz-user-select:none;user-select:none;cursor:pointer;align-items:center;color:var(--accent-color)}details summary .down-arrow{fill:var(--accent-color)}details[open] summary{margin-bottom:1rem}details[open]>summary .down-arrow{transform:rotate(180deg)}sm-input,sm-textarea{width:100%;--border-radius: 0.5rem;--background-color: rgba(var(--foreground-color), 1)}sm-input button .icon,sm-textarea button .icon{fill:var(--accent-color)}sm-button{--padding: 0.8rem}sm-button[variant=primary] .icon{fill:rgba(var(--background-color), 1)}sm-button[disabled] .icon{fill:rgba(var(--text-color), 0.6)}sm-button.danger{--background: var(--danger-color);color:rgba(var(--background-color), 1)}sm-spinner{--size: 1rem;--stroke-width: 0.1rem}sm-form{--gap: 1rem}sm-select{font-size:.9rem;--padding: 0.6rem 0.3rem 0.6rem 0.6rem}sm-option{font-size:.9rem;--border-radius: 0.3rem}sm-chips{--gap: 0.3rem;background-color:rgba(var(--text-color), 0.06);border-radius:.7rem;padding:.3rem}sm-chip{font-size:.9rem;--border-radius: 0.5rem;-webkit-user-select:none;-moz-user-select:none;user-select:none;font-weight:500}sm-chip[selected]{color:rgba(var(--background-color), 1);--background: var(--accent-color)}sm-notifications{z-index:100000}ul,ol{list-style:none}.overflow-ellipsis{width:100%;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}.wrap-around{overflow-wrap:break-word;word-wrap:break-word;word-break:break-word}.full-bleed{grid-column:1/-1}.uppercase{text-transform:uppercase}.capitalize{text-transform:capitalize}.sticky{position:-webkit-sticky;position:sticky}.top-0{top:0}.flex{display:flex}.flex-column{display:flex;flex-direction:column}.flex-wrap{flex-wrap:wrap}.flex-1{flex:1}.flex-direction-column{flex-direction:column}.grid{display:grid}.flow-column{grid-auto-flow:column}.gap-0-3{gap:.3rem}.gap-0-5{gap:.5rem}.gap-1{gap:1rem}.gap-1-5{gap:1.5rem}.gap-2{gap:2rem}.gap-3{gap:3rem}.text-align-right{text-align:right}.align-content-start{align-content:flex-start}.align-items-start{align-items:flex-start}.align-self-start{align-self:flex-start}.align-center{align-items:center}.align-end{align-items:flex-end}.text-center{text-align:center}.justify-start{justify-items:start}.justify-center{justify-content:center}.justify-right{margin-left:auto}.justify-items-center{justify-items:center}.align-self-center{align-self:center}.align-self-end{align-self:end}.justify-self-center{justify-self:center}.justify-self-start{justify-self:start}.justify-self-end{justify-self:end}.flex-direction-column{flex-direction:column}.space-between{justify-content:space-between}.w-100{width:100%}.h-100{height:100%}.label{font-size:.9rem;color:rgba(var(--text-color), 0.8);font-weight:500;margin-bottom:.3rem}.button--primary .ripple,.button--danger .ripple{background:radial-gradient(circle, rgba(var(--background-color), 0.3) 0%, rgba(0, 0, 0, 0) 50%)}.ripple{height:8rem;width:8rem;position:absolute;border-radius:50%;transform:scale(0);background:radial-gradient(circle, rgba(var(--text-color), 0.3) 0%, rgba(0, 0, 0, 0) 50%);pointer-events:none}.interactive{position:relative;overflow:hidden;cursor:pointer;-webkit-tap-highlight-color:rgba(0,0,0,0);-webkit-user-select:none;-moz-user-select:none;user-select:none}.empty-state{display:grid;width:100%;padding:1.5rem 0}.observe-empty-state:empty{display:none !important}.observe-empty-state:not(:empty)+.empty-state{display:none !important}.bullet-point{display:flex;align-items:center;justify-content:center;margin:0 .8ch}.bullet-point::after{content:"";height:.4ch;width:.4ch;border-radius:.5em;background-color:var(--accent-color)}.margin-right-0-3{margin-right:.3rem}.margin-right-0-5{margin-right:.5rem}.margin-right-auto{margin-right:auto}.margin-left-0-5{margin-left:.5rem}.margin-left-auto{margin-left:auto}.margin-top-1-5{margin-top:1.5rem}.margin-bottom-0-5{margin-bottom:.5rem}.margin-bottom-1{margin-bottom:1rem}.margin-bottom-1-5{margin-bottom:1.5rem}.margin-bottom-2{margin-bottom:2rem}.icon-button{padding:.6rem;border-radius:.8rem;background-color:rgba(var(--text-color), 0.1);height:-webkit-max-content;height:-moz-max-content;height:max-content}.icon-button .icon{fill:var(--accent-color)}#confirmation_popup,#prompt_popup{flex-direction:column}#confirmation_popup h4,#prompt_popup h4{margin-bottom:1.5rem}#confirmation_popup sm-button,#prompt_popup sm-button{margin:0}#confirmation_popup>.flex:last-of-type,#prompt_popup>.flex:last-of-type{padding:0;margin-top:1rem}#confirmation_popup>.flex:last-of-type sm-button:first-of-type,#prompt_popup>.flex:last-of-type sm-button:first-of-type{margin-right:.6rem;margin-left:auto}#prompt_message{margin-bottom:1.5rem}.popup__header{display:grid;gap:.5rem;width:100%;padding:0 1.5rem;align-items:center;justify-items:flex-start}.popup__header__close{padding:.5rem;margin-left:-0.5rem}#loading{display:grid;grid-template-rows:auto 1fr}#loading section{place-content:center;justify-items:center}#loading h4{margin-top:1.5rem;font-weight:500}#loading sm-spinner{--size: 1.5rem}#sign_in,#sign_up{display:grid;height:100%;grid-template-rows:auto 1fr}#sign_in section,#sign_up section{margin:auto;width:min(26rem,100%)}#sign_in sm-form,#sign_up sm-form{margin:2rem 0}#main_header{padding:1rem max(1rem,4vw)}.app-brand{display:flex;gap:.3rem;align-items:center}.app-brand .icon{height:1.7rem;width:1.7rem}.app-name__company{font-size:.8rem;font-weight:500;color:rgba(var(--text-color), 0.8)}.password-field label{display:flex}.password-field label input:checked~.visible{display:none}.password-field label input:not(:checked)~.invisible{display:none}.multi-state-button{display:grid;text-align:center;align-items:center;justify-items:center}.multi-state-button>*{grid-area:1/1/2/2}.multi-state-button button{z-index:1;width:100%}.scrolling-wrapper{overflow-y:auto}#flo_id_warning{padding-bottom:1.5rem;border-bottom:thin solid rgba(var(--text-color), 0.3)}#flo_id_warning .icon{height:4rem;width:4rem;padding:1rem;background-color:#ffc107;border-radius:3rem;fill:rgba(0,0,0,.8);margin-bottom:1.5rem}.generated-id-card{display:grid;gap:1rem}.generated-id-card h5{margin-bottom:.3rem}#app_body{display:grid;grid-template-columns:1rem [main-margin-start] minmax(0, 1fr) [main-margin-end] 1rem;height:100%}#app_body>*{grid-column:main-margin}#main_header{padding:1rem 0;position:-webkit-sticky;position:sticky;top:0;z-index:10;background-color:rgba(var(--background-color), 0.7);-webkit-backdrop-filter:blur(1rem);backdrop-filter:blur(1rem)}#main_header:has(button){display:grid;grid-template-columns:auto minmax(0, 1fr) auto}#main_header #main_logo{height:2.5rem;width:2.5rem;padding:.6rem;border-radius:5rem;background-color:rgba(var(--text-color), 0.1)}#main_header .button{text-transform:capitalize}#main_header .button--primary{padding:.6rem 1.8rem;border:solid .12rem rgba(var(--text-color), 1)}#landing{display:grid;grid-template-rows:auto 1fr;height:100%}#landing #relative_tasks{height:100%}#landing #relative_tasks>div:first-of-type{display:grid;place-content:center;align-self:center}#landing h1 span:first-of-type{font-size:1.2rem}#landing h1 span:last-of-type{font-size:2rem}#category_selection{text-align:center;width:min(56rem,100%);margin-inline:auto;grid-template-rows:auto 1fr auto;padding:1.5rem 0;gap:1.5rem}#category_selection h1{line-height:.9;margin-bottom:1rem}#category_selection p{max-width:100%;margin-bottom:1rem}#category_selection ul{display:grid;align-self:flex-start;gap:1rem;grid-template-columns:repeat(auto-fill, minmax(11rem, 1fr))}#category_selection ul li{display:flex;-webkit-animation:popin .3s ease-out var(--delay) forwards;animation:popin .3s ease-out var(--delay) forwards;opacity:0;transform:scale(0.9)}#category_selection ul li:has(:checked) label{box-shadow:0 0 0 .1rem var(--accent-color);background-color:rgba(0,0,0,0)}#category_selection ul li:has(:checked) label:hover{box-shadow:0 0 0 .2rem var(--accent-color);background-color:rgba(0,0,0,0)}#category_selection ul li label{display:flex;align-items:center;justify-content:center;width:100%;height:100%;text-align:center;background-color:rgba(var(--text-color), 0.06);padding:1rem;border-radius:40%;cursor:pointer;font-weight:500;transition:background-color .3s,box-shadow .3s;aspect-ratio:1/1}#category_selection ul li label:hover{background-color:rgba(var(--text-color), 0.03);box-shadow:0 0 0 .1rem rgba(var(--text-color), 0.3)}#category_selection ul li input{display:none}@-webkit-keyframes popin{to{opacity:1;transform:none}}@keyframes popin{to{opacity:1;transform:none}}#category_selection button{justify-self:center;padding:1rem 3rem;border-radius:5rem;gap:.5rem}#category_selection button span{transform:translateX(1rem);transition:transform .3s}#category_selection button .icon{opacity:0;transform:translateX(-0.5rem);transition:opacity .3s,transform .3s}#category_selection button:hover span{transform:translateX(0)}#category_selection button:hover .icon{opacity:1;transform:translateX(0)}.task-card{display:grid;grid-template-columns:1fr auto;gap:.5rem;align-items:center;grid-template-areas:"task-title task-time" "task-description task-description" "task-requisites task-actions"}.task-card:not(:last-of-type){border-bottom:thin solid rgba(var(--text-color), 0.5);padding-bottom:1.5rem;margin-bottom:1.5rem}.task-card:last-of-type{padding-bottom:3rem}.task-card h4{grid-area:task-title;font-size:1.2rem}.task-card>sl-relative-time{grid-area:task-time;font-size:.8rem;color:rgba(var(--text-color), 0.6);text-align:end}.task-card p{grid-area:task-description;font-size:.9rem;color:rgba(var(--text-color), 0.8)}.task-card ul{grid-area:task-requisites}.task-card ul li{font-size:.9rem;color:rgba(var(--text-color), 0.8);font-weight:500}.task-card ul li:not(:last-of-type)::after{content:"•";margin-left:.3rem}.task-card .button{grid-area:task-actions}.update-card{border-radius:.5rem;padding:max(1rem,1.5vw);background-color:rgba(var(--text-color), 0.06)}.update-card__sender,.update-card__time{font-size:.8rem;font-weight:500;color:rgba(var(--text-color), 0.8)}.update-card__message{margin-top:.5rem;white-space:pre-line}#profile__header h1,#profile__header h4{line-height:.7}#profile sm-form{width:min(32rem,100%)}#task_popup sm-chips{background-color:rgba(0,0,0,0);padding:0}#task_popup sm-chips sm-chip{--background: rgba(var(--text-color), 0.06) }#user_popup_button{background-color:rgba(var(--text-color), 0.06);border-radius:2rem;font-size:.8rem;padding:.6rem .8rem;color:currentColor}#user_popup_button .icon{fill:currentColor}@media screen and (max-width: 40rem){#landing #relative_tasks{align-content:flex-start}#landing #relative_tasks>div:first-of-type{padding:3rem 0}#category_selection ul{grid-template-columns:repeat(auto-fill, minmax(9rem, 1fr))}#emblem{display:none}.task-card{grid-template-areas:"task-title task-time" "task-description task-description" "task-requisites task-requisites" "task-actions task-actions"}#profile__header{padding:1rem 0;margin-top:10vh}#user_popup_button{grid-area:1/2/2/3}.hide-on-small{display:none}}@media screen and (min-width: 40rem){h1{font-size:4rem}h2{font-size:2rem}h3{font-size:1.5rem}h4{font-size:1.2rem}h5{font-size:1rem}h6{font-size:.9rem}sm-popup{--width: 24rem}.popup__header{grid-column:1/-1;padding:1rem 1.5rem 0 1.5rem}#app_body{grid-template-columns:4vw [main-margin-start] 4vw [inner-margin-start] minmax(0, 1fr) [inner-margin-end] 4vw [main-margin-end] 4vw}#main_header{padding:2rem 0}#landing #relative_tasks{display:grid;grid-template-columns:1fr 1.2fr;gap:8vw}#landing #relative_tasks>div:first-of-type{align-self:flex-start;position:-webkit-sticky;position:sticky;top:40vh}#landing #relative_tasks>div:first-of-type::after{content:"";position:absolute;height:12rem;width:.1rem;background-color:rgba(var(--text-color), 1);right:0;align-self:center;margin-top:5rem}#landing #relative_tasks>div:first-of-type #emblem{position:absolute;align-self:center;right:-1.45rem;width:3rem;height:3rem;stroke:rgba(var(--text-color), 1);stroke-width:.1rem;margin-top:5rem}#landing #relative_tasks>div:last-of-type{margin-top:3rem}#home>section,#task>section{padding:0 8vw}#task_popup{--width: 36rem}#send_update_popup{--width: 36rem}#profile section{gap:4rem;justify-content:center;align-items:center}}@media(any-hover: hover){html{scrollbar-color:#888 rgba(0,0,0,0)}::-webkit-scrollbar{width:10px;height:10px}::-webkit-scrollbar-thumb{background:currentColor}::-webkit-scrollbar-track{background:rgba(0,0,0,0)}.interactive:not([disabled]){transition:background-color .3s}.interactive:not([disabled]):hover{background-color:rgba(var(--text-color), 0.06)}button:not([disabled]),.button:not([disabled]){transition:background-color .3s,filter .3s}button:not([disabled]):hover,.button:not([disabled]):hover{filter:contrast(2)}}@supports(overflow: overlay){body{overflow:overlay}}.hidden{display:none !important} \ No newline at end of file diff --git a/css/main.scss b/css/main.scss index 0b0f7d2..2a40bd6 100644 --- a/css/main.scss +++ b/css/main.scss @@ -992,6 +992,25 @@ ol { } } +.update-card { + border-radius: 0.5rem; + padding: max(1rem, 1.5vw); + background-color: rgba(var(--text-color), 0.06); + + &__sender, + &__time { + font-size: 0.8rem; + font-weight: 500; + color: rgba(var(--text-color), 0.8); + } + + &__message { + margin-top: 0.5rem; + white-space: pre-line; + + } +} + #profile { &__header { @@ -1153,7 +1172,8 @@ ol { } } - #home { + #home, + #task { &>section { padding: 0 8vw; @@ -1164,6 +1184,10 @@ ol { --width: 36rem; } + #send_update_popup { + --width: 36rem; + } + #profile { section { gap: 4rem; diff --git a/scripts/app.js b/scripts/app.js index d50ad10..e76dd6d 100644 --- a/scripts/app.js +++ b/scripts/app.js @@ -713,8 +713,7 @@ async function initSendingUpdate(id) { const update = { id, type: updateType.value, - message, - date: Date.now() + updateMessage: message } // send update await floCloudAPI.sendGeneralData(update, 'taskUpdates') @@ -775,11 +774,9 @@ async function initSendingUpdate(id) { }) openPopup('send_update_popup') } -getRef('send_update_popup').addEventListener('option-clicked', e => { - console.log(e) -}) -function editTask(id) { - const { title, description, category, deadline } = task; + +function editTask(details) { + const { id, title, description, category, deadline } = details; getRef('task_popup__title_input').value = title; getRef('task_popup__description').value = description; getRef('task_popup__category').value = category; @@ -869,9 +866,9 @@ const render = { if (floGlobals.isUserLoggedIn) { if (floGlobals.isSubAdmin) { actions = html` - + - ${floGlobals.applications[id]?.size || 0} working + See updates ` } else if (!floGlobals.isAdmin) { // check if user has already working on the task @@ -916,24 +913,28 @@ const render = { ` }, availableTasks(options = {}) { - const { type } = options; - let availableTasks = [...floGlobals.interestedCategories || Object.keys(floGlobals.taskCategories)].flatMap(category => - floGlobals.appObjects[category].tasks - ) - if (availableTasks.length === 0) - return renderElem(getRef('available_tasks_list'), html`

No tasks available

`); - if (type) { - if (type === 'active') - availableTasks = availableTasks.filter(task => floGlobals.applications.has(task.id)) - else if (type === 'other') - availableTasks = availableTasks.filter(task => !floGlobals.applications.has(task.id)) + if (floGlobals.isSubAdmin) { + let availableTasks = Object.keys(floGlobals.taskCategories) + .flatMap(category => floGlobals.appObjects[category].tasks) + .map(render.task) + if (availableTasks.length === 0) + return renderElem(getRef('available_tasks_list'), html`

No tasks available

`); + renderElem(getRef('available_tasks_list'), html`${availableTasks}`) + } else if (!floGlobals.isAdmin) { + const { type } = options; + let availableTasks = [...floGlobals.interestedCategories || Object.keys(floGlobals.taskCategories)].flatMap(category => + floGlobals.appObjects[category].tasks + ) + if (availableTasks.length === 0) + return renderElem(getRef('available_tasks_list'), html`

No tasks available

`); + if (type) { + if (type === 'active') + availableTasks = availableTasks.filter(task => floGlobals.applications.has(task.id)) + else if (type === 'other') + availableTasks = availableTasks.filter(task => !floGlobals.applications.has(task.id)) + } + renderElem(getRef('available_tasks_list'), html`${availableTasks.map(render.task)}`) } - console.log(availableTasks) - availableTasks = availableTasks.map(task => { - console.log(render.task(task)) - return render.task(task) - }); - renderElem(getRef('available_tasks_list'), html`${availableTasks}`) } } @@ -959,6 +960,7 @@ const header = () => { const isUserLoggedIn = page === 'loading' || floGlobals.isUserLoggedIn; if (userAddressTimeInterval) clearInterval(userAddressTimeInterval); userAddressTimeInterval = setInterval(() => { + if (!getRef('user_popup_button')) return; if (!isShowingFloID) { renderElem(getRef('user_popup_button'), html` @@ -1185,6 +1187,7 @@ function renderHome(state) {
${header()}
+
Tasks
- - ` - }) - renderElem(getRef('task_applications_list'), html`${applications}`) + `); + const updates = (floGlobals.appObjects[category].tasks.find(task => task.id === id).updates || []) + .map(updateVC => { + const { message: { id, updateMessage, type }, senderID, time } = floDapps.getNextGeneralData('taskUpdates', '0')[updateVC]; + return html` +
  • +
    +
    ${senderID}
    + +
    +
    ${updateMessage}
    +
  • + ` + }).reverse(); + if (updates.length) { + renderElem(getRef('task_update_list'), html`${updates}`) + } else { + renderElem(getRef('task_update_list'), html`
  • No updates yet
  • `) + } } else if (!floGlobals.isAdmin) { } }) router.addRoute('profile', (state) => { - const { } = state; - let name = email = college = course = whatsappNumber = ''; - if (floGlobals.userProfile) { - const userDetails = JSON.parse(floDapps.user.decipher(floGlobals.userProfile)); - name = userDetails.name; - email = userDetails.email; - college = userDetails.college; - course = userDetails.course; - whatsappNumber = userDetails.whatsappNumber; + try { + const { } = state; + let name = email = college = course = whatsappNumber = ''; + if (floGlobals.userProfile) { + const userDetails = JSON.parse(floDapps.user.decipher(floGlobals.userProfile)); + name = userDetails.name; + email = userDetails.email; + college = userDetails.college; + course = userDetails.course; + whatsappNumber = userDetails.whatsappNumber; + } + renderElem(getRef('app_body'), html` +
    + ${header()} +
    +
    +

    Tell us about

    +

    yourself

    +
    + +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + +
    +
    +
    +
    + `) + } + catch (err) { + notify(err, 'error') } - renderElem(getRef('app_body'), html` -
    - ${header()} -
    -
    -

    Tell us about

    -

    yourself

    -
    - -
    - - -
    -
    - - -
    -
    - - -
    -
    - - -
    -
    - - -
    -
    - -
    -
    -
    -
    - `) }) router.addRoute('404', async () => { @@ -1429,11 +1447,12 @@ window.addEventListener("load", () => { floGlobals.myBtcID = getBtcAddress(floGlobals.myFloID) floGlobals.isSubAdmin = floGlobals.subAdmins.includes(floGlobals.myFloID) floGlobals.isAdmin = floGlobals.myFloID === floGlobals.adminID - let showingFloID = true try { if (floGlobals.isSubAdmin) { const promises = [] + await Promise.all(Object.keys(floGlobals.taskCategories).map(category => floCloudAPI.requestObjectData(category))) + for (const category in floGlobals.taskCategories) { if (!floGlobals.appObjects[category]) { console.log('resetting', category) @@ -1444,21 +1463,34 @@ window.addEventListener("load", () => { } } promises.push(floCloudAPI.requestGeneralData('taskApplications')) + promises.push(floCloudAPI.requestGeneralData('taskUpdates')) await Promise.all(promises) - const taskApplications = floDapps.getNextGeneralData('taskApplications', '0'); - floGlobals.applications = {} floGlobals.allAvailableTasks = new Set(); for (const category in floGlobals.taskCategories) { const tasks = floGlobals.appObjects[category].tasks || []; tasks.forEach(task => floGlobals.allAvailableTasks.add(task.id)) } + const taskApplications = floDapps.getNextGeneralData('taskApplications', '0'); + floGlobals.applications = {} for (const application in taskApplications) { const { message: { taskID }, senderID } = taskApplications[application]; if (!floGlobals.applications[taskID]) floGlobals.applications[taskID] = new Set() floGlobals.applications[taskID].add(senderID) } - + const taskUpdates = floDapps.getNextGeneralData('taskUpdates', '0'); + for (const updateVC in taskUpdates) { + const { message: { id } } = taskUpdates[updateVC]; + const [category] = id.split('_'); + const task = floGlobals.appObjects[category].tasks.find(task => task.id === id) + if (!task) { + continue; + } else if (task && !task.updates) { + task.updates = [updateVC] + } else { + floGlobals.appObjects[category].tasks.find(task => task.id === id).updates.push(updateVC) + } + } } else if (floGlobals.isAdmin) { } else { @@ -1469,6 +1501,9 @@ window.addEventListener("load", () => { }), floCloudAPI.requestGeneralData('userProfile', { senderID: [floGlobals.myFloID, floGlobals.myBtcID], + }), + floCloudAPI.requestGeneralData('taskUpdates', { + senderID: [floGlobals.myFloID, floGlobals.myBtcID], }) ] await Promise.all(promises) @@ -1485,6 +1520,11 @@ window.addEventListener("load", () => { floGlobals.applications.add(taskID) } const userProfile = floDapps.getNextGeneralData('userProfile', '0'); + // remove general data which is not from the user + for (const profile in userProfile) { + if (userProfile[profile].senderID !== floGlobals.myFloID) + delete userProfile[profile] + } floGlobals.userProfile = Object.values(userProfile).at(-1)?.message.encryptedData; } if (['#/landing', '#/sign_in', '#/sign_up'].includes(window.location.hash)) { diff --git a/scripts/app.min.js b/scripts/app.min.js index 2381da0..49d139e 100644 --- a/scripts/app.min.js +++ b/scripts/app.min.js @@ -51,13 +51,13 @@ const uiGlobals={},{html:html,svg:svg,render:renderElem}=uhtml,{signal:signal,co ${updateForm}
    - +
    - `)})),openPopup("send_update_popup")}function editTask(id){const{title:title,description:description,category:category,deadline:deadline}=task;getRef("task_popup__title_input").value=title,getRef("task_popup__description").value=description,getRef("task_popup__category").value=category,getRef("task_popup__deadline").value=deadline,getRef("task_popup").dataset.taskId=id,openPopup("task_popup")}async function saveTask(){if(!await getConfirmation("Save task",{message:"Are you sure you want to save this task?",confirmText:"Save"}))return;const title=getRef("task_popup__title_input").value,description=getRef("task_popup__description").value,category=getRef("task_popup__category").value,id=getRef("task_popup").dataset.taskId||category+"_"+Math.random().toString(36).substring(2,9),deadline=getRef("task_popup__deadline").value,task={id:id,title:title,description:description,category:category,deadline:deadline,status:"open"},foundTask=floGlobals.appObjects[category].tasks.find((task=>task.id===id));if(foundTask){let taskDetailsChanged=!1;for(const key in task)task[key]!==foundTask[key]&&(taskDetailsChanged=!0,foundTask[key]=task[key]);if(!taskDetailsChanged)return notify("Please update at least one detail to save the changes","error")}else task.date=Date.now(),floGlobals.appObjects[category].tasks.unshift(task);buttonLoader("task_popup__submit",!0),floCloudAPI.updateObjectData(category).then((response=>{notify("Task saved successfully","success"),render.availableTasks()})).catch((e=>{notify("An error occurred while saving the task","error"),console.error(e)})).finally((()=>{buttonLoader("task_popup__submit",!1),closePopup()}))}async function deleteTask(id){if(!await getConfirmation("Delete task",{message:"Are you sure you want to delete this task?",confirmText:"Delete",danger:!0}))return;const[category]=id.split("_"),taskIndex=floGlobals.appObjects[category].tasks.findIndex((task=>task.id===id));if(taskIndex<0)return notify("Task not found","error");const[cloneOfTaskToBeDeleted]=floGlobals.appObjects[category].tasks.splice(taskIndex,1);floCloudAPI.updateObjectData(category).then((response=>{notify("Task deleted successfully","success")})).catch((e=>{notify("An error occurred while deleting the task","error"),floGlobals.appObjects[category].tasks.splice(taskIndex,0,cloneOfTaskToBeDeleted)})).finally((()=>{closePopup(),render.availableTasks()}))}window.smCompConfig={"sm-input":[{selector:"[data-flo-address]",customValidation:value=>value?{isValid:floCrypto.validateFloID(value),errorText:'Invalid FLO address.
    It usually starts with "F"'}:{isValid:!1,errorText:"Please enter a FLO address"}},{selector:"[data-btc-address]",customValidation:value=>value?{isValid:btcOperator.validateAddress(value),errorText:'Invalid address.
    It usually starts with "1", "3" or "bc1"'}:{isValid:!1,errorText:"Please enter a BTC address"}},{selector:"[data-private-key]",customValidation:(value,inputElem)=>{if(!value)return{isValid:!1,errorText:"Please enter a private key"};if(floCrypto.getPubKeyHex(value)){const forAddress=inputElem.dataset.forAddress;return forAddress?{isValid:btcOperator.verifyKey(forAddress,value),errorText:`This private key does not match the address ${forAddress}`}:{isValid:!0}}return{isValid:!1,errorText:"Invalid private key. Please check and try again."}}},{selector:'[type="email"]',customValidation:(value,target)=>""===value?{isValid:!1,errorText:"Please enter an email address"}:{isValid:/\S+@\S+\.\S+/.test(value),errorText:"Invalid email address"}},{selector:"#profile__whatsapp_number",customValidation:(value,target)=>value.length<10?{isValid:!1,errorText:"Number must be at least 10 digits long"}:value.length>13?{isValid:!1,errorText:"Number must be at most 13 digits long"}:{isValid:!0}}]},getRef("send_update_popup").addEventListener("option-clicked",(e=>{console.log(e)}));const render={task(details={}){const{title:title,description:description,date:date,id:id,status:status,deadline:deadline,category:category}=details;let actions="";if(floGlobals.isUserLoggedIn){if(floGlobals.isSubAdmin)actions=html` - + `)})),openPopup("send_update_popup")}function editTask(details){const{id:id,title:title,description:description,category:category,deadline:deadline}=details;getRef("task_popup__title_input").value=title,getRef("task_popup__description").value=description,getRef("task_popup__category").value=category,getRef("task_popup__deadline").value=deadline,getRef("task_popup").dataset.taskId=id,openPopup("task_popup")}async function saveTask(){if(!await getConfirmation("Save task",{message:"Are you sure you want to save this task?",confirmText:"Save"}))return;const title=getRef("task_popup__title_input").value,description=getRef("task_popup__description").value,category=getRef("task_popup__category").value,id=getRef("task_popup").dataset.taskId||category+"_"+Math.random().toString(36).substring(2,9),deadline=getRef("task_popup__deadline").value,task={id:id,title:title,description:description,category:category,deadline:deadline,status:"open"},foundTask=floGlobals.appObjects[category].tasks.find((task=>task.id===id));if(foundTask){let taskDetailsChanged=!1;for(const key in task)task[key]!==foundTask[key]&&(taskDetailsChanged=!0,foundTask[key]=task[key]);if(!taskDetailsChanged)return notify("Please update at least one detail to save the changes","error")}else task.date=Date.now(),floGlobals.appObjects[category].tasks.unshift(task);buttonLoader("task_popup__submit",!0),floCloudAPI.updateObjectData(category).then((response=>{notify("Task saved successfully","success"),render.availableTasks()})).catch((e=>{notify("An error occurred while saving the task","error"),console.error(e)})).finally((()=>{buttonLoader("task_popup__submit",!1),closePopup()}))}async function deleteTask(id){if(!await getConfirmation("Delete task",{message:"Are you sure you want to delete this task?",confirmText:"Delete",danger:!0}))return;const[category]=id.split("_"),taskIndex=floGlobals.appObjects[category].tasks.findIndex((task=>task.id===id));if(taskIndex<0)return notify("Task not found","error");const[cloneOfTaskToBeDeleted]=floGlobals.appObjects[category].tasks.splice(taskIndex,1);floCloudAPI.updateObjectData(category).then((response=>{notify("Task deleted successfully","success")})).catch((e=>{notify("An error occurred while deleting the task","error"),floGlobals.appObjects[category].tasks.splice(taskIndex,0,cloneOfTaskToBeDeleted)})).finally((()=>{closePopup(),render.availableTasks()}))}window.smCompConfig={"sm-input":[{selector:"[data-flo-address]",customValidation:value=>value?{isValid:floCrypto.validateFloID(value),errorText:'Invalid FLO address.
    It usually starts with "F"'}:{isValid:!1,errorText:"Please enter a FLO address"}},{selector:"[data-btc-address]",customValidation:value=>value?{isValid:btcOperator.validateAddress(value),errorText:'Invalid address.
    It usually starts with "1", "3" or "bc1"'}:{isValid:!1,errorText:"Please enter a BTC address"}},{selector:"[data-private-key]",customValidation:(value,inputElem)=>{if(!value)return{isValid:!1,errorText:"Please enter a private key"};if(floCrypto.getPubKeyHex(value)){const forAddress=inputElem.dataset.forAddress;return forAddress?{isValid:btcOperator.verifyKey(forAddress,value),errorText:`This private key does not match the address ${forAddress}`}:{isValid:!0}}return{isValid:!1,errorText:"Invalid private key. Please check and try again."}}},{selector:'[type="email"]',customValidation:(value,target)=>""===value?{isValid:!1,errorText:"Please enter an email address"}:{isValid:/\S+@\S+\.\S+/.test(value),errorText:"Invalid email address"}},{selector:"#profile__whatsapp_number",customValidation:(value,target)=>value.length<10?{isValid:!1,errorText:"Number must be at least 10 digits long"}:value.length>13?{isValid:!1,errorText:"Number must be at most 13 digits long"}:{isValid:!0}}]};const render={task(details={}){const{title:title,description:description,date:date,id:id,status:status,deadline:deadline,category:category}=details;let actions="";if(floGlobals.isUserLoggedIn){if(floGlobals.isSubAdmin)actions=html` + - ${floGlobals.applications[id]?.size||0} working + See updates `;else if(!floGlobals.isAdmin){const isThisActive=floGlobals.applications.has(id),hasAnyActiveTask=isThisActive||[...floGlobals.applications].some((id=>{const[category]=id.split("_");return floGlobals.appObjects[category].tasks.some((task=>floGlobals.applications.has(task.id)))}));actions=isThisActive?html`
    - `},availableTasks(options={}){const{type:type}=options;let availableTasks=[...floGlobals.interestedCategories||Object.keys(floGlobals.taskCategories)].flatMap((category=>floGlobals.appObjects[category].tasks));if(0===availableTasks.length)return renderElem(getRef("available_tasks_list"),html`

    No tasks available

    `);type&&("active"===type?availableTasks=availableTasks.filter((task=>floGlobals.applications.has(task.id))):"other"===type&&(availableTasks=availableTasks.filter((task=>!floGlobals.applications.has(task.id))))),console.log(availableTasks),availableTasks=availableTasks.map((task=>(console.log(render.task(task)),render.task(task)))),renderElem(getRef("available_tasks_list"),html`${availableTasks}`)}},router=new Router({routingStart(state){"scrollRestoration"in history&&(history.scrollRestoration="manual"),window.scrollTo(0,0)},routingEnd(state){const{page:page,lastPage:lastPage}=state;lastPage!==page&&closePopup()}});let userAddressTimeInterval,isShowingFloID=!1;const header=()=>{const{page:page}=router.state,isUserLoggedIn="loading"===page||floGlobals.isUserLoggedIn;return userAddressTimeInterval&&clearInterval(userAddressTimeInterval),userAddressTimeInterval=setInterval((()=>{isShowingFloID?(renderElem(getRef("user_popup_button"),html` + `},availableTasks(options={}){if(floGlobals.isSubAdmin){let availableTasks=Object.keys(floGlobals.taskCategories).flatMap((category=>floGlobals.appObjects[category].tasks)).map(render.task);if(0===availableTasks.length)return renderElem(getRef("available_tasks_list"),html`

    No tasks available

    `);renderElem(getRef("available_tasks_list"),html`${availableTasks}`)}else if(!floGlobals.isAdmin){const{type:type}=options;let availableTasks=[...floGlobals.interestedCategories||Object.keys(floGlobals.taskCategories)].flatMap((category=>floGlobals.appObjects[category].tasks));if(0===availableTasks.length)return renderElem(getRef("available_tasks_list"),html`

    No tasks available

    `);type&&("active"===type?availableTasks=availableTasks.filter((task=>floGlobals.applications.has(task.id))):"other"===type&&(availableTasks=availableTasks.filter((task=>!floGlobals.applications.has(task.id))))),renderElem(getRef("available_tasks_list"),html`${availableTasks.map(render.task)}`)}}},router=new Router({routingStart(state){"scrollRestoration"in history&&(history.scrollRestoration="manual"),window.scrollTo(0,0)},routingEnd(state){const{page:page,lastPage:lastPage}=state;lastPage!==page&&closePopup()}});let userAddressTimeInterval,isShowingFloID=!1;const header=()=>{const{page:page}=router.state,isUserLoggedIn="loading"===page||floGlobals.isUserLoggedIn;return userAddressTimeInterval&&clearInterval(userAddressTimeInterval),userAddressTimeInterval=setInterval((()=>{getRef("user_popup_button")&&(isShowingFloID?(renderElem(getRef("user_popup_button"),html`
    ${floGlobals.myBtcID}
    `),isShowingFloID=!1):(renderElem(getRef("user_popup_button"),html`
    ${floGlobals.myFloID}
    - `),isShowingFloID=!0)}),1e4),html` + `),isShowingFloID=!0))}),1e4),html`
    @@ -113,6 +113,7 @@ const uiGlobals={},{html:html,svg:svg,render:renderElem}=uhtml,{signal:signal,co
    ${header()}
    +
    Tasks
    - - `));renderElem(getRef("task_applications_list"),html`${applications}`)}else floGlobals.isAdmin})),router.addRoute("profile",(state=>{const{}=state;let name=email=college=course=whatsappNumber="";if(floGlobals.userProfile){const userDetails=JSON.parse(floDapps.user.decipher(floGlobals.userProfile));name=userDetails.name,email=userDetails.email,college=userDetails.college,course=userDetails.course,whatsappNumber=userDetails.whatsappNumber}renderElem(getRef("app_body"),html` -
    - ${header()} -
    -
    -

    Tell us about

    -

    yourself

    -
    - -
    - - + `);const updates=(floGlobals.appObjects[category].tasks.find((task=>task.id===id)).updates||[]).map((updateVC=>{const{message:{id:id,updateMessage:updateMessage,type:type},senderID:senderID,time:time}=floDapps.getNextGeneralData("taskUpdates","0")[updateVC];return html` +
  • +
    +
    ${senderID}
    + +
    +
    ${updateMessage}
    +
  • + `})).reverse();updates.length?renderElem(getRef("task_update_list"),html`${updates}`):renderElem(getRef("task_update_list"),html`
  • No updates yet
  • `)}else floGlobals.isAdmin})),router.addRoute("profile",(state=>{try{const{}=state;let name=email=college=course=whatsappNumber="";if(floGlobals.userProfile){const userDetails=JSON.parse(floDapps.user.decipher(floGlobals.userProfile));name=userDetails.name,email=userDetails.email,college=userDetails.college,course=userDetails.course,whatsappNumber=userDetails.whatsappNumber}renderElem(getRef("app_body"),html` +
    + ${header()} +
    +
    +

    Tell us about

    +

    yourself

    -
    - - -
    -
    - - -
    -
    - - -
    -
    - - -
    -
    - -
    - -
    -
    - `)})),router.addRoute("404",(async()=>{renderElem(getRef("app_body"),html` + +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + +
    +
    +
    +
    + `)}catch(err){notify(err,"error")}})),router.addRoute("404",(async()=>{renderElem(getRef("app_body"),html`

    404

    Page not found

    - `)}));let privKeyResolver=null;function getSignedIn(passwordType){return new Promise(((resolve,reject)=>{privKeyResolver=resolve;try{getPromptInput("Enter password","",{isPassword:!0}).then((password=>{password&&resolve(password)}))}catch(err){floGlobals.isPrivKeySecured="PIN/Password"===passwordType,["#/landing","#/sign_in","#/sign_up"].some((route=>window.location.hash.includes(route)))||(history.replaceState(null,null,"#/landing"),router.routeTo("#/landing"))}}))}function setSecurePassword(){if(!floGlobals.isPrivKeySecured){const password=getRef("secure_pwd_input").value.trim();floDapps.securePrivKey(password).then((()=>{floGlobals.isPrivKeySecured=!0,notify("Password set successfully","success"),closePopup()})).catch((err=>{notify(err,"error")}))}}function signOut(){getConfirmation("Sign out?",{message:"You are about to sign out of the app, continue?",confirmText:"Leave",cancelText:"Stay"}).then((async res=>{res&&(await floDapps.clearCredentials(),location.reload())}))}const btcAddresses={},floAddresses={};function getBtcAddress(floAddress){return btcAddresses[floAddress]||(btcAddresses[floAddress]=btcOperator.convert.legacy2bech(floAddress)),btcAddresses[floAddress]}function getFloAddress(btcAddress){return floAddresses[btcAddress]||(floAddresses[btcAddress]=floCrypto.toFloID(btcAddress)),floAddresses[btcAddress]}router.routeTo("loading"),window.addEventListener("load",(()=>{const[browserName,browserVersion]=detectBrowser().split(" "),supportedVersions={Chrome:85,Firefox:75,Safari:13};browserName in supportedVersions?parseInt(browserVersion){"Escape"===e.key&&closePopup()})),document.addEventListener("copy",(()=>{notify("copied","success")})),document.addEventListener("pointerdown",(e=>{e.target.closest("button:not(:disabled), .interactive:not(:disabled)")&&createRipple(e,e.target.closest("button, .interactive"))}));const interestedCategories=localStorage.getItem("interestedCategories")||"[]";floGlobals.interestedCategories=new Set(JSON.parse(interestedCategories)),floDapps.setMidStartup((()=>new Promise(((resolve,reject)=>{floCloudAPI.requestObjectData("rmInterns").then((()=>{["#/landing","#/sign_in","#/sign_up"].some((route=>window.location.hash.includes(route)))&&router.routeTo(window.location.hash),resolve()})).catch((err=>{console.error(err),reject()}))})))),floDapps.setCustomPrivKeyInput(getSignedIn),floDapps.launchStartUp().then((async result=>{console.log(result),floGlobals.isUserLoggedIn=!0,floGlobals.myFloID=getFloAddress(floDapps.user.id),floGlobals.myBtcID=getBtcAddress(floGlobals.myFloID),floGlobals.isSubAdmin=floGlobals.subAdmins.includes(floGlobals.myFloID),floGlobals.isAdmin=floGlobals.myFloID===floGlobals.adminID;try{if(floGlobals.isSubAdmin){const promises=[];await Promise.all(Object.keys(floGlobals.taskCategories).map((category=>floCloudAPI.requestObjectData(category))));for(const category in floGlobals.taskCategories)floGlobals.appObjects[category]||(console.log("resetting",category),floGlobals.appObjects[category]={tasks:[]},promises.push(floCloudAPI.resetObjectData(category)));promises.push(floCloudAPI.requestGeneralData("taskApplications")),await Promise.all(promises);const taskApplications=floDapps.getNextGeneralData("taskApplications","0");floGlobals.applications={},floGlobals.allAvailableTasks=new Set;for(const category in floGlobals.taskCategories){(floGlobals.appObjects[category].tasks||[]).forEach((task=>floGlobals.allAvailableTasks.add(task.id)))}for(const application in taskApplications){const{message:{taskID:taskID},senderID:senderID}=taskApplications[application];floGlobals.applications[taskID]||(floGlobals.applications[taskID]=new Set),floGlobals.applications[taskID].add(senderID)}}else if(floGlobals.isAdmin);else{floGlobals.applications=new Set;const promises=[floCloudAPI.requestGeneralData("taskApplications",{senderID:[floGlobals.myFloID,floGlobals.myBtcID]}),floCloudAPI.requestGeneralData("userProfile",{senderID:[floGlobals.myFloID,floGlobals.myBtcID]})];await Promise.all(promises);const taskApplications=floDapps.getNextGeneralData("taskApplications","0");floGlobals.allAvailableTasks=new Set;for(const category in floGlobals.taskCategories){(floGlobals.appObjects[category].tasks||[]).forEach((task=>floGlobals.allAvailableTasks.add(task.id)))}for(const application in taskApplications){const{message:{taskID:taskID}}=taskApplications[application];floGlobals.allAvailableTasks.has(taskID)&&floGlobals.applications.add(taskID)}const userProfile=floDapps.getNextGeneralData("userProfile","0");floGlobals.userProfile=Object.values(userProfile).at(-1)?.message.encryptedData}["#/landing","#/sign_in","#/sign_up"].includes(window.location.hash)?(history.replaceState(null,null,"#/home"),router.routeTo("home")):router.routeTo(window.location.hash)}catch(err){console.error(err)}})).catch((error=>console.error(error)))})); \ No newline at end of file + `)}));let privKeyResolver=null;function getSignedIn(passwordType){return new Promise(((resolve,reject)=>{privKeyResolver=resolve;try{getPromptInput("Enter password","",{isPassword:!0}).then((password=>{password&&resolve(password)}))}catch(err){floGlobals.isPrivKeySecured="PIN/Password"===passwordType,["#/landing","#/sign_in","#/sign_up"].some((route=>window.location.hash.includes(route)))||(history.replaceState(null,null,"#/landing"),router.routeTo("#/landing"))}}))}function setSecurePassword(){if(!floGlobals.isPrivKeySecured){const password=getRef("secure_pwd_input").value.trim();floDapps.securePrivKey(password).then((()=>{floGlobals.isPrivKeySecured=!0,notify("Password set successfully","success"),closePopup()})).catch((err=>{notify(err,"error")}))}}function signOut(){getConfirmation("Sign out?",{message:"You are about to sign out of the app, continue?",confirmText:"Leave",cancelText:"Stay"}).then((async res=>{res&&(await floDapps.clearCredentials(),location.reload())}))}const btcAddresses={},floAddresses={};function getBtcAddress(floAddress){return btcAddresses[floAddress]||(btcAddresses[floAddress]=btcOperator.convert.legacy2bech(floAddress)),btcAddresses[floAddress]}function getFloAddress(btcAddress){return floAddresses[btcAddress]||(floAddresses[btcAddress]=floCrypto.toFloID(btcAddress)),floAddresses[btcAddress]}router.routeTo("loading"),window.addEventListener("load",(()=>{const[browserName,browserVersion]=detectBrowser().split(" "),supportedVersions={Chrome:85,Firefox:75,Safari:13};browserName in supportedVersions?parseInt(browserVersion){"Escape"===e.key&&closePopup()})),document.addEventListener("copy",(()=>{notify("copied","success")})),document.addEventListener("pointerdown",(e=>{e.target.closest("button:not(:disabled), .interactive:not(:disabled)")&&createRipple(e,e.target.closest("button, .interactive"))}));const interestedCategories=localStorage.getItem("interestedCategories")||"[]";floGlobals.interestedCategories=new Set(JSON.parse(interestedCategories)),floDapps.setMidStartup((()=>new Promise(((resolve,reject)=>{floCloudAPI.requestObjectData("rmInterns").then((()=>{["#/landing","#/sign_in","#/sign_up"].some((route=>window.location.hash.includes(route)))&&router.routeTo(window.location.hash),resolve()})).catch((err=>{console.error(err),reject()}))})))),floDapps.setCustomPrivKeyInput(getSignedIn),floDapps.launchStartUp().then((async result=>{console.log(result),floGlobals.isUserLoggedIn=!0,floGlobals.myFloID=getFloAddress(floDapps.user.id),floGlobals.myBtcID=getBtcAddress(floGlobals.myFloID),floGlobals.isSubAdmin=floGlobals.subAdmins.includes(floGlobals.myFloID),floGlobals.isAdmin=floGlobals.myFloID===floGlobals.adminID;try{if(floGlobals.isSubAdmin){const promises=[];await Promise.all(Object.keys(floGlobals.taskCategories).map((category=>floCloudAPI.requestObjectData(category))));for(const category in floGlobals.taskCategories)floGlobals.appObjects[category]||(console.log("resetting",category),floGlobals.appObjects[category]={tasks:[]},promises.push(floCloudAPI.resetObjectData(category)));promises.push(floCloudAPI.requestGeneralData("taskApplications")),promises.push(floCloudAPI.requestGeneralData("taskUpdates")),await Promise.all(promises),floGlobals.allAvailableTasks=new Set;for(const category in floGlobals.taskCategories){(floGlobals.appObjects[category].tasks||[]).forEach((task=>floGlobals.allAvailableTasks.add(task.id)))}const taskApplications=floDapps.getNextGeneralData("taskApplications","0");floGlobals.applications={};for(const application in taskApplications){const{message:{taskID:taskID},senderID:senderID}=taskApplications[application];floGlobals.applications[taskID]||(floGlobals.applications[taskID]=new Set),floGlobals.applications[taskID].add(senderID)}const taskUpdates=floDapps.getNextGeneralData("taskUpdates","0");for(const updateVC in taskUpdates){const{message:{id:id}}=taskUpdates[updateVC],[category]=id.split("_"),task=floGlobals.appObjects[category].tasks.find((task=>task.id===id));task&&(task&&!task.updates?task.updates=[updateVC]:floGlobals.appObjects[category].tasks.find((task=>task.id===id)).updates.push(updateVC))}}else if(floGlobals.isAdmin);else{floGlobals.applications=new Set;const promises=[floCloudAPI.requestGeneralData("taskApplications",{senderID:[floGlobals.myFloID,floGlobals.myBtcID]}),floCloudAPI.requestGeneralData("userProfile",{senderID:[floGlobals.myFloID,floGlobals.myBtcID]}),floCloudAPI.requestGeneralData("taskUpdates",{senderID:[floGlobals.myFloID,floGlobals.myBtcID]})];await Promise.all(promises);const taskApplications=floDapps.getNextGeneralData("taskApplications","0");floGlobals.allAvailableTasks=new Set;for(const category in floGlobals.taskCategories){(floGlobals.appObjects[category].tasks||[]).forEach((task=>floGlobals.allAvailableTasks.add(task.id)))}for(const application in taskApplications){const{message:{taskID:taskID}}=taskApplications[application];floGlobals.allAvailableTasks.has(taskID)&&floGlobals.applications.add(taskID)}const userProfile=floDapps.getNextGeneralData("userProfile","0");for(const profile in userProfile)userProfile[profile].senderID!==floGlobals.myFloID&&delete userProfile[profile];floGlobals.userProfile=Object.values(userProfile).at(-1)?.message.encryptedData}["#/landing","#/sign_in","#/sign_up"].includes(window.location.hash)?(history.replaceState(null,null,"#/home"),router.routeTo("home")):router.routeTo(window.location.hash)}catch(err){console.error(err)}})).catch((error=>console.error(error)))})); \ No newline at end of file