Content inside supertokenslib
(The raw file follows this syntax highlighted file.)
import "minjs"
const ssw=(s)=>"{{.Cfg.SuperTokensWebLibBase}}"+s
addjs(ssw("/website.js"));addjs(ssw("/supertokens.js"));addjs(ssw("/session.js"));addjs(ssw("/passwordless.js"));addjs(ssw("/emailverification.js"))
const stWrapperClass=addcss({boxSizing:"border-box",position:"absolute",top:"12px",right:"12px",zIndex:999,
display:"flex",alignItems:"flex-end",flexDirection:"column"})
const stMenuWrapperClass=addcss({backgroundColor:"white",minWidth:"10em",display:"flex",flexDirection:"column",
border:"black solid 1px",paddingTop:"4px",div:{minHeight:"10px",marginBottom:"4px"}})
const stPlainObj={padding:"4px",cursor:"pointer",marginLeft:"12px",fontVariant:"small-caps"}
const stPlainClass=addcss(stPlainObj)
const stPlainMenuClass=addcss(R.mergeRight(stPlainObj,
{marginLeft:"0px",paddingLeft:"12px",display:"inline-block",width:"calc(100% - 16px)","&:hover":{backgroundColor:"#e6e6e6"}}))
const stButtonClass=addcss({width:"32px",height:"32px",padding:"4px",border:0,cursor:"pointer",borderRadius:"16px",marginLeft:"12px"})
const stImgClass=addcss({width:"24px",height:"24px"})
const stFormClass=addcss({display:"flex",flexDirection:"column",padding:"4px",backgroundColor:"white","input": {marginBottom:"10px"}})
const stInputCentered=addcss({textAlign:"center"});const stOpacity0=addcss({opacity:"0%"});const stMarginTop10=addcss({marginTop:"10px"})
const flexItemsCenter={display:"flex",alignItems:"center"};const stFlexItemsCenter=addcss(flexItemsCenter)
const stNonPlainEmailWrapper=addcss(R.mergeRight(flexItemsCenter,{width:"32px",height:"32px",justifyContent:"center"}))
const truncated=R.replace(/(.)[^@]*(.)@(.).*(.\.[^.]+)/,"$1-$2@$3-$4")
window.onload=()=>{ // TODO - pull out as many funcs as possible
const plain=R.equals("{{.Cfg.PlainUI}}", "true");const prefixedPath=(s)=>"{{.Cfg.APIBasePath}}"+s;
const isLoggedIn=()=>R.equals("{{.LoggedIn}}","true");const userEmail=()=>"{{.UserEmail}}";const userID=()=>"{{.UserID}}"
const uuid="st"+crypto.randomUUID();const emailUUID=uuid+"email";const otpUUID=uuid+"otp"
const callSendOTP=()=>{globalThis.stAuth.sendOTP(valueByID(emailUUID));s.mod(disableID(emailUUID)(disableID(uuid+"emailButton")({})))}
const callVerifyOTP=()=>{globalThis.stAuth.verifyOTP(valueByID(otpUUID));disableID(otpUUID)({});disableID(uuid+"otpButton")({})}
const handleClose=()=>{s.mod({stState:"init",displayState:c.init});h.state("loggedOut")}
const c={init:()=>null,
enterEmail:()=>e("div", {className:stFormClass},
e("input", disableable(s)({id:emailUUID,type:"email",
placeholder:"your.email@address.com",onkeyup:onEnter(callSendOTP),className:stInputCentered})),
e("button", disableable(s)({id:uuid+"emailButton",onclick:callSendOTP}), "Request one-time code"),
e("button",{className:stMarginTop10,onclick:handleClose},"Close")),
enterOTP:()=>e("div", {className:stFormClass},
e("input", disableable(s)({id:otpUUID,type:"text",
placeholder:"One-time code in email",onkeyup:onEnter(callVerifyOTP),className:stInputCentered})),
e("button", disableable(s)({id:uuid+"otpButton",onclick:callVerifyOTP}), "Verify code"),
e("button",{className:stMarginTop10,onclick:handleClose},"Close")),
showEmail:()=>(plain
? null
: e("div", {className:stNonPlainEmailWrapper},
e("div",{id:uuid+"userEmail",className:stOpacity0},userEmail()),
e("button",{className:stButtonClass, onclick:fadeInOut("#"+uuid+"userEmail")},
e("img",{className:stImgClass,src:(isLoggedIn() ? prefixedPath("/person_outline.svg") : null)}))))}
const pre=(o)=>({
"init":R.identity, // TODO - see if we can live without this...
"loggedOut":R.identity,
"checkEmail":R.mergeLeft({displayState:c.enterOTP}),
"wrongOTP":R.identity, "oldOTP":R.identity, "failedOTP":R.identity,
"signUpSuccess":R.tap(()=>window.location.reload()),
"signInSuccess":R.tap(()=>window.location.reload()),
"loggedIn":R.mergeLeft({displayState:c.showEmail}),
"logoutCompleted":R.tap(()=>window.location.reload()),
}[R.propOr("init", "stState", o)](o))
const s=store({init:true,label:"...",stState:"init",showMenu:false,menuItems:[],displayState:R.always(null)},pre)
const handleMenu=()=>{if(!isLoggedIn()){handleClose()}s.mod({showMenu:!s.state.showMenu})}
const handleMenuMouseLeave=()=>s.mod({showMenu:false})
const handle=()=>{handleMenuMouseLeave();if(eon(globalThis.authLogoutCheck)||globalThis.authLogoutCheck()){
if(s.state.stState==="loggedOut"){ s.mod({displayState:c.enterEmail})
}else if(s.state.stState==="loggedIn"){globalThis.stAuth.logout()}}}
let etc=null
const checkForMenuClose=(e)=>{if(neon(etc)&&!etc.contains(e.target)){handleMenuMouseLeave()}}
const removeClickOffTracker=(el)=>{etc=el;document.removeEventListener("click",checkForMenuClose,true)}
const addClickOffTracker=(el)=>{etc=el;removeClickOffTracker(el);document.addEventListener("click",checkForMenuClose,true)}
const setMenuItems=(mis)=>s.mod({menuItems:R.map(
([label,fn])=>e("div",{},e("a",{href:"javascript:",className:stPlainMenuClass,onclick:()=>{handleMenuMouseLeave();fn()}},label)),
R.reject(eon,mis))})
const aui=()=>e("div",{id:uuid,className:stWrapperClass,
style:"opacity:"+(s.state.stState=="init" ? 0: 100)+"%;background:rgba(0,0,0,0)",
onmouseleave:handleMenuMouseLeave,lifecycle:{added:addClickOffTracker,removed:removeClickOffTracker}},
e("button",{style:"width:32px;height:32px;padding:0;border:0;background-color:white;cursor:pointer",onclick:handleMenu},
e("img",{style:"width:32px;height:32px",src:prefixedPath("/menu.svg")})),
s.state.displayState(),
(!s.state.showMenu ? null
: e("div",{className:stMenuWrapperClass},
(isLoggedIn() ? null : e("div",{},e("a",{href:"#",className:stPlainMenuClass,onclick:handle},"login"))),
(!isLoggedIn() ? null : e("div",{className:stPlainMenuClass,style:"cursor:default;font-variant:normal"},
truncated(userEmail()))),
...s.state.menuItems,
(!isLoggedIn() ? null : e("div",{},e("a",{href:"#",className:stPlainMenuClass,onclick:handle},"logout"))),
)),
)
s.subscribe(()=>{ge(uuid).replaceWith(aui());focusIfVisible(emailUUID);focusIfVisible(otpUUID)})
dbac(aui())
const h={catch:(msg,err)=>{/*TODO showError(msg,err)*/},state:(state,data)=>s.mod({stState:state,stData:data})}
const hge=(msg,err)=>h.catch(msg+(err.isSuperTokensGeneralError ? " (some other failure)" : ""),err)
// TODO - check if supertokens is defined...
supertokens.init({appInfo:{apiDomain:"{{.Cfg.APIDomain}}",apiBasePath:"{{.Cfg.APIBasePath}}",appName:"{{.Cfg.AppName}}"},
recipeList:[supertokensEmailVerification.init(),supertokensSession.init(),supertokensPasswordless.init()]});
const sendOTP=async(email)=>{try{
const resp=await supertokensPasswordless.createCode({email})
if(resp.status != "SIGN_IN_UP_NOT_ALLOWED"){h.state("checkEmail")}
}catch(err){hge("sendOTP",err)}}
const verifyOTP=async(otp)=>{try{
const resp=await supertokensPasswordless.consumeCode({userInputCode:otp});
if(resp.status === "OK"){
if(resp.createdNewRecipeUser && resp.user.loginMethods.length === 1){h.state("signUpSuccess",resp.user.id)
}else{h.state("signInSuccess",resp.user.id)}
}else if(resp.status === "INCORRECT_USER_INPUT_CODE_ERROR"){h.state("wrongOTP")
}else if(resp.status === "EXPIRED_USER_INPUT_CODE_ERROR"){h.state("oldOTP")
}else{h.state("failedOTP")}
}catch(err){hge("handleOTP",err)}}
const logout=async()=>{await supertokensSession.signOut();h.state("logoutCompleted")}
globalThis.stAuth={sendOTP,verifyOTP,logout,isLoggedIn,userEmail,userID,setMenuItems}
if(neon(globalThis.authHooks)){globalThis.authHooks.forEach(([l,f])=>{const p=h[l];h[l]=(a,b)=>{p(a,b);f(a,b)}})}
if(neon(globalThis.menuItems)){setMenuItems(globalThis.menuItems)}
if(isLoggedIn()){h.state("loggedIn",userEmail())}else{h.state("loggedOut")}
anime({targets:"#"+uuid,opacity:"100%",duration:1000,easing:"easeInOutQuad",complete:()=>anime.remove("#"+uuid)})
}
import "minjs" const ssw=(s)=>"{{.Cfg.SuperTokensWebLibBase}}"+s addjs(ssw("/website.js"));addjs(ssw("/supertokens.js"));addjs(ssw("/session.js"));addjs(ssw("/passwordless.js"));addjs(ssw("/emailverification.js")) const stWrapperClass=addcss({boxSizing:"border-box",position:"absolute",top:"12px",right:"12px",zIndex:999, display:"flex",alignItems:"flex-end",flexDirection:"column"}) const stMenuWrapperClass=addcss({backgroundColor:"white",minWidth:"10em",display:"flex",flexDirection:"column", border:"black solid 1px",paddingTop:"4px",div:{minHeight:"10px",marginBottom:"4px"}}) const stPlainObj={padding:"4px",cursor:"pointer",marginLeft:"12px",fontVariant:"small-caps"} const stPlainClass=addcss(stPlainObj) const stPlainMenuClass=addcss(R.mergeRight(stPlainObj, {marginLeft:"0px",paddingLeft:"12px",display:"inline-block",width:"calc(100% - 16px)","&:hover":{backgroundColor:"#e6e6e6"}})) const stButtonClass=addcss({width:"32px",height:"32px",padding:"4px",border:0,cursor:"pointer",borderRadius:"16px",marginLeft:"12px"}) const stImgClass=addcss({width:"24px",height:"24px"}) const stFormClass=addcss({display:"flex",flexDirection:"column",padding:"4px",backgroundColor:"white","input": {marginBottom:"10px"}}) const stInputCentered=addcss({textAlign:"center"});const stOpacity0=addcss({opacity:"0%"});const stMarginTop10=addcss({marginTop:"10px"}) const flexItemsCenter={display:"flex",alignItems:"center"};const stFlexItemsCenter=addcss(flexItemsCenter) const stNonPlainEmailWrapper=addcss(R.mergeRight(flexItemsCenter,{width:"32px",height:"32px",justifyContent:"center"})) const truncated=R.replace(/(.)[^@]*(.)@(.).*(.\.[^.]+)/,"$1-$2@$3-$4") window.onload=()=>{ // TODO - pull out as many funcs as possible const plain=R.equals("{{.Cfg.PlainUI}}", "true");const prefixedPath=(s)=>"{{.Cfg.APIBasePath}}"+s; const isLoggedIn=()=>R.equals("{{.LoggedIn}}","true");const userEmail=()=>"{{.UserEmail}}";const userID=()=>"{{.UserID}}" const uuid="st"+crypto.randomUUID();const emailUUID=uuid+"email";const otpUUID=uuid+"otp" const callSendOTP=()=>{globalThis.stAuth.sendOTP(valueByID(emailUUID));s.mod(disableID(emailUUID)(disableID(uuid+"emailButton")({})))} const callVerifyOTP=()=>{globalThis.stAuth.verifyOTP(valueByID(otpUUID));disableID(otpUUID)({});disableID(uuid+"otpButton")({})} const handleClose=()=>{s.mod({stState:"init",displayState:c.init});h.state("loggedOut")} const c={init:()=>null, enterEmail:()=>e("div", {className:stFormClass}, e("input", disableable(s)({id:emailUUID,type:"email", placeholder:"your.email@address.com",onkeyup:onEnter(callSendOTP),className:stInputCentered})), e("button", disableable(s)({id:uuid+"emailButton",onclick:callSendOTP}), "Request one-time code"), e("button",{className:stMarginTop10,onclick:handleClose},"Close")), enterOTP:()=>e("div", {className:stFormClass}, e("input", disableable(s)({id:otpUUID,type:"text", placeholder:"One-time code in email",onkeyup:onEnter(callVerifyOTP),className:stInputCentered})), e("button", disableable(s)({id:uuid+"otpButton",onclick:callVerifyOTP}), "Verify code"), e("button",{className:stMarginTop10,onclick:handleClose},"Close")), showEmail:()=>(plain ? null : e("div", {className:stNonPlainEmailWrapper}, e("div",{id:uuid+"userEmail",className:stOpacity0},userEmail()), e("button",{className:stButtonClass, onclick:fadeInOut("#"+uuid+"userEmail")}, e("img",{className:stImgClass,src:(isLoggedIn() ? prefixedPath("/person_outline.svg") : null)}))))} const pre=(o)=>({ "init":R.identity, // TODO - see if we can live without this... "loggedOut":R.identity, "checkEmail":R.mergeLeft({displayState:c.enterOTP}), "wrongOTP":R.identity, "oldOTP":R.identity, "failedOTP":R.identity, "signUpSuccess":R.tap(()=>window.location.reload()), "signInSuccess":R.tap(()=>window.location.reload()), "loggedIn":R.mergeLeft({displayState:c.showEmail}), "logoutCompleted":R.tap(()=>window.location.reload()), }[R.propOr("init", "stState", o)](o)) const s=store({init:true,label:"...",stState:"init",showMenu:false,menuItems:[],displayState:R.always(null)},pre) const handleMenu=()=>{if(!isLoggedIn()){handleClose()}s.mod({showMenu:!s.state.showMenu})} const handleMenuMouseLeave=()=>s.mod({showMenu:false}) const handle=()=>{handleMenuMouseLeave();if(eon(globalThis.authLogoutCheck)||globalThis.authLogoutCheck()){ if(s.state.stState==="loggedOut"){ s.mod({displayState:c.enterEmail}) }else if(s.state.stState==="loggedIn"){globalThis.stAuth.logout()}}} let etc=null const checkForMenuClose=(e)=>{if(neon(etc)&&!etc.contains(e.target)){handleMenuMouseLeave()}} const removeClickOffTracker=(el)=>{etc=el;document.removeEventListener("click",checkForMenuClose,true)} const addClickOffTracker=(el)=>{etc=el;removeClickOffTracker(el);document.addEventListener("click",checkForMenuClose,true)} const setMenuItems=(mis)=>s.mod({menuItems:R.map( ([label,fn])=>e("div",{},e("a",{href:"javascript:",className:stPlainMenuClass,onclick:()=>{handleMenuMouseLeave();fn()}},label)), R.reject(eon,mis))}) const aui=()=>e("div",{id:uuid,className:stWrapperClass, style:"opacity:"+(s.state.stState=="init" ? 0: 100)+"%;background:rgba(0,0,0,0)", onmouseleave:handleMenuMouseLeave,lifecycle:{added:addClickOffTracker,removed:removeClickOffTracker}}, e("button",{style:"width:32px;height:32px;padding:0;border:0;background-color:white;cursor:pointer",onclick:handleMenu}, e("img",{style:"width:32px;height:32px",src:prefixedPath("/menu.svg")})), s.state.displayState(), (!s.state.showMenu ? null : e("div",{className:stMenuWrapperClass}, (isLoggedIn() ? null : e("div",{},e("a",{href:"#",className:stPlainMenuClass,onclick:handle},"login"))), (!isLoggedIn() ? null : e("div",{className:stPlainMenuClass,style:"cursor:default;font-variant:normal"}, truncated(userEmail()))), ...s.state.menuItems, (!isLoggedIn() ? null : e("div",{},e("a",{href:"#",className:stPlainMenuClass,onclick:handle},"logout"))), )), ) s.subscribe(()=>{ge(uuid).replaceWith(aui());focusIfVisible(emailUUID);focusIfVisible(otpUUID)}) dbac(aui()) const h={catch:(msg,err)=>{/*TODO showError(msg,err)*/},state:(state,data)=>s.mod({stState:state,stData:data})} const hge=(msg,err)=>h.catch(msg+(err.isSuperTokensGeneralError ? " (some other failure)" : ""),err) // TODO - check if supertokens is defined... supertokens.init({appInfo:{apiDomain:"{{.Cfg.APIDomain}}",apiBasePath:"{{.Cfg.APIBasePath}}",appName:"{{.Cfg.AppName}}"}, recipeList:[supertokensEmailVerification.init(),supertokensSession.init(),supertokensPasswordless.init()]}); const sendOTP=async(email)=>{try{ const resp=await supertokensPasswordless.createCode({email}) if(resp.status != "SIGN_IN_UP_NOT_ALLOWED"){h.state("checkEmail")} }catch(err){hge("sendOTP",err)}} const verifyOTP=async(otp)=>{try{ const resp=await supertokensPasswordless.consumeCode({userInputCode:otp}); if(resp.status === "OK"){ if(resp.createdNewRecipeUser && resp.user.loginMethods.length === 1){h.state("signUpSuccess",resp.user.id) }else{h.state("signInSuccess",resp.user.id)} }else if(resp.status === "INCORRECT_USER_INPUT_CODE_ERROR"){h.state("wrongOTP") }else if(resp.status === "EXPIRED_USER_INPUT_CODE_ERROR"){h.state("oldOTP") }else{h.state("failedOTP")} }catch(err){hge("handleOTP",err)}} const logout=async()=>{await supertokensSession.signOut();h.state("logoutCompleted")} globalThis.stAuth={sendOTP,verifyOTP,logout,isLoggedIn,userEmail,userID,setMenuItems} if(neon(globalThis.authHooks)){globalThis.authHooks.forEach(([l,f])=>{const p=h[l];h[l]=(a,b)=>{p(a,b);f(a,b)}})} if(neon(globalThis.menuItems)){setMenuItems(globalThis.menuItems)} if(isLoggedIn()){h.state("loggedIn",userEmail())}else{h.state("loggedOut")} anime({targets:"#"+uuid,opacity:"100%",duration:1000,easing:"easeInOutQuad",complete:()=>anime.remove("#"+uuid)}) }