2022-12-10 12:14:48 +03:00
import './settings.css' ;
2023-07-09 11:51:05 +03:00
import { useEffect , useRef , useState } from 'preact/hooks' ;
2023-01-14 14:42:04 +03:00
import { useSnapshot } from 'valtio' ;
2022-12-10 12:14:48 +03:00
2023-01-25 19:54:30 +03:00
import logo from '../assets/logo.svg' ;
2023-10-03 10:07:47 +03:00
2023-04-20 11:10:57 +03:00
import Icon from '../components/icon' ;
2023-09-01 10:40:00 +03:00
import Link from '../components/link' ;
2023-01-05 05:50:27 +03:00
import RelativeTime from '../components/relative-time' ;
2023-03-07 17:38:06 +03:00
import targetLanguages from '../data/lingva-target-languages' ;
2023-07-09 11:32:09 +03:00
import { api } from '../utils/api' ;
2023-03-07 17:38:06 +03:00
import getTranslateTargetLanguage from '../utils/get-translate-target-language' ;
import localeCode2Text from '../utils/localeCode2Text' ;
2023-09-01 10:40:00 +03:00
import {
initSubscription ,
isPushSupported ,
removeSubscription ,
updateSubscription ,
} from '../utils/push-notifications' ;
2023-10-23 03:43:27 +03:00
import showToast from '../utils/show-toast' ;
2022-12-25 18:31:50 +03:00
import states from '../utils/states' ;
2022-12-10 12:14:48 +03:00
import store from '../utils/store' ;
2023-03-08 12:17:23 +03:00
const DEFAULT _TEXT _SIZE = 16 ;
const TEXT _SIZES = [ 16 , 17 , 18 , 19 , 20 ] ;
2022-12-16 08:27:04 +03:00
function Settings ( { onClose } ) {
2023-01-14 14:42:04 +03:00
const snapStates = useSnapshot ( states ) ;
2022-12-10 12:14:48 +03:00
const currentTheme = store . local . get ( 'theme' ) || 'auto' ;
const themeFormRef = useRef ( ) ;
2023-03-07 17:38:06 +03:00
const targetLanguage =
snapStates . settings . contentTranslationTargetLanguage || null ;
const systemTargetLanguage = getTranslateTargetLanguage ( ) ;
const systemTargetLanguageText = localeCode2Text ( systemTargetLanguage ) ;
2023-03-08 12:17:23 +03:00
const currentTextSize = store . local . get ( 'textSize' ) || DEFAULT _TEXT _SIZE ;
2023-03-07 17:38:06 +03:00
2023-07-09 11:32:09 +03:00
const [ prefs , setPrefs ] = useState ( store . account . get ( 'preferences' ) || { } ) ;
2023-10-02 16:13:56 +03:00
const { masto , authenticated , instance } = api ( ) ;
2023-07-09 11:32:09 +03:00
// Get preferences every time Settings is opened
// NOTE: Disabled for now because I don't expect this to change often. Also for some reason, the /api/v1/preferences endpoint is cached for a while and return old prefs if refresh immediately after changing them.
// useEffect(() => {
// const { masto } = api();
// (async () => {
// try {
// const preferences = await masto.v1.preferences.fetch();
// setPrefs(preferences);
// store.account.set('preferences', preferences);
// } catch (e) {
// // Silently fail
// console.error(e);
// }
// })();
// }, []);
2022-12-10 12:14:48 +03:00
return (
2022-12-29 11:11:58 +03:00
< div id = "settings-container" class = "sheet" tabIndex = "-1" >
2023-04-20 11:10:57 +03:00
{ ! ! onClose && (
< button type = "button" class = "sheet-close" onClick = { onClose } >
< Icon icon = "x" / >
< / button >
) }
2023-03-07 19:32:33 +03:00
< header >
2023-01-17 12:58:04 +03:00
< h2 > Settings < / h2 >
2023-03-07 19:32:33 +03:00
< / header >
< main >
2023-02-28 12:12:17 +03:00
< section >
< ul >
< li >
< div >
< label > Appearance < / label >
< / div >
< div >
< form
ref = { themeFormRef }
onInput = { ( e ) => {
console . log ( e ) ;
e . preventDefault ( ) ;
const formData = new FormData ( themeFormRef . current ) ;
const theme = formData . get ( 'theme' ) ;
const html = document . documentElement ;
2022-12-10 12:14:48 +03:00
2023-02-28 12:12:17 +03:00
if ( theme === 'auto' ) {
html . classList . remove ( 'is-light' , 'is-dark' ) ;
} else {
html . classList . toggle ( 'is-light' , theme === 'light' ) ;
html . classList . toggle ( 'is-dark' , theme === 'dark' ) ;
}
document
. querySelector ( 'meta[name="color-scheme"]' )
. setAttribute (
'content' ,
theme === 'auto' ? 'dark light' : theme ,
) ;
2022-12-10 12:14:48 +03:00
2023-02-28 12:12:17 +03:00
if ( theme === 'auto' ) {
store . local . del ( 'theme' ) ;
} else {
store . local . set ( 'theme' , theme ) ;
}
} }
>
< div class = "radio-group" >
< label >
< input
type = "radio"
name = "theme"
value = "light"
defaultChecked = { currentTheme === 'light' }
/ >
< span > Light < / span >
< / label >
< label >
< input
type = "radio"
name = "theme"
value = "dark"
defaultChecked = { currentTheme === 'dark' }
/ >
< span > Dark < / span >
< / label >
< label >
< input
type = "radio"
name = "theme"
value = "auto"
defaultChecked = {
currentTheme !== 'light' && currentTheme !== 'dark'
}
/ >
< span > Auto < / span >
< / label >
< / div >
< / form >
< / div >
< / li >
2023-03-08 12:17:23 +03:00
< li >
< div >
< label > Text size < / label >
< / div >
< div class = "range-group" >
< span style = { { fontSize : TEXT _SIZES [ 0 ] } } > A < / span > { ' ' }
< input
type = "range"
min = { TEXT _SIZES [ 0 ] }
max = { TEXT _SIZES [ TEXT _SIZES . length - 1 ] }
step = "1"
value = { currentTextSize }
list = "sizes"
onChange = { ( e ) => {
const value = parseInt ( e . target . value , 10 ) ;
const html = document . documentElement ;
// set CSS variable
html . style . setProperty ( '--text-size' , ` ${ value } px ` ) ;
// save to local storage
if ( value === DEFAULT _TEXT _SIZE ) {
store . local . del ( 'textSize' ) ;
} else {
store . local . set ( 'textSize' , e . target . value ) ;
}
} }
/ > { ' ' }
< span style = { { fontSize : TEXT _SIZES [ TEXT _SIZES . length - 1 ] } } >
A
< / span >
< datalist id = "sizes" >
{ TEXT _SIZES . map ( ( size ) => (
< option value = { size } / >
) ) }
< / datalist >
< / div >
< / li >
2023-03-07 19:32:33 +03:00
< / ul >
< / section >
2023-09-14 19:28:20 +03:00
{ authenticated && (
< >
< h3 > Posting < / h3 >
< section >
< ul >
< li >
< div >
< label for = "posting-privacy-field" >
2023-10-02 16:13:56 +03:00
Default visibility { ' ' }
< Icon icon = "cloud" alt = "Synced" class = "synced-icon" / >
2023-09-14 19:28:20 +03:00
< / label >
< / div >
< div >
< select
id = "posting-privacy-field"
value = { prefs [ 'posting:default:visibility' ] || 'public' }
onChange = { ( e ) => {
const { value } = e . target ;
( async ( ) => {
try {
await masto . v1 . accounts . updateCredentials ( {
source : {
privacy : value ,
} ,
} ) ;
setPrefs ( {
... prefs ,
'posting:default:visibility' : value ,
} ) ;
store . account . set ( 'preferences' , {
... prefs ,
'posting:default:visibility' : value ,
} ) ;
} catch ( e ) {
alert ( 'Failed to update posting privacy' ) ;
console . error ( e ) ;
}
} ) ( ) ;
} }
>
< option value = "public" > Public < / option >
< option value = "unlisted" > Unlisted < / option >
< option value = "private" > Followers only < / option >
< / select >
< / div >
< / li >
< / ul >
< / section >
2023-10-02 16:13:56 +03:00
< p class = "section-postnote" >
< Icon icon = "cloud" alt = "Synced" class = "synced-icon" / > { ' ' }
< small >
Synced to your instance server 's settings.{' ' }
< a
href = { ` https:// ${ instance } / ` }
target = "_blank"
rel = "noopener noreferrer"
>
Go to your instance ( { instance } ) for more settings .
< / a >
< / small >
< / p >
2023-09-14 19:28:20 +03:00
< / >
) }
2023-03-07 19:32:33 +03:00
< h3 > Experiments < / h3 >
< section >
< ul >
2023-05-05 12:53:16 +03:00
< li >
< label >
< input
type = "checkbox"
checked = { snapStates . settings . autoRefresh }
onChange = { ( e ) => {
states . settings . autoRefresh = e . target . checked ;
} }
/ > { ' ' }
Auto refresh timeline posts
< / label >
< / li >
2023-02-28 12:12:17 +03:00
< li >
< label >
< input
type = "checkbox"
checked = { snapStates . settings . boostsCarousel }
onChange = { ( e ) => {
states . settings . boostsCarousel = e . target . checked ;
} }
/ > { ' ' }
2023-03-07 19:32:33 +03:00
Boosts carousel
2023-02-28 12:12:17 +03:00
< / label >
< / li >
2023-03-07 17:38:06 +03:00
< li >
< label >
< input
type = "checkbox"
checked = { snapStates . settings . contentTranslation }
onChange = { ( e ) => {
2023-03-09 08:20:01 +03:00
const { checked } = e . target ;
states . settings . contentTranslation = checked ;
if ( ! checked ) {
states . settings . contentTranslationTargetLanguage = null ;
}
2023-03-07 17:38:06 +03:00
} }
/ > { ' ' }
2023-03-07 19:32:33 +03:00
Post translation
2023-03-07 17:38:06 +03:00
< / label >
2023-03-09 08:20:01 +03:00
< div
class = { ` sub-section ${
! snapStates . settings . contentTranslation
? 'more-insignificant'
: ''
} ` }
>
2023-07-22 15:59:07 +03:00
< div >
< label >
Translate to { ' ' }
< select
value = { targetLanguage || '' }
disabled = { ! snapStates . settings . contentTranslation }
onChange = { ( e ) => {
states . settings . contentTranslationTargetLanguage =
e . target . value || null ;
} }
>
< option value = "" >
System language ( { systemTargetLanguageText } )
< / option >
< option disabled > ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ < / option >
{ targetLanguages . map ( ( lang ) => (
< option value = { lang . code } > { lang . name } < / option >
) ) }
< / select >
< / label >
< / div >
< hr / >
2023-03-28 14:04:52 +03:00
< p class = "checkbox-fieldset" >
2023-07-22 15:59:07 +03:00
Hide "Translate" button for
{ snapStates . settings . contentTranslationHideLanguages . length >
0 && (
< >
{ ' ' }
(
{
snapStates . settings . contentTranslationHideLanguages
. length
}
)
< / >
) }
:
2023-03-28 14:04:52 +03:00
< div class = "checkbox-fields" >
{ targetLanguages . map ( ( lang ) => (
< label >
< input
type = "checkbox"
checked = { snapStates . settings . contentTranslationHideLanguages . includes (
lang . code ,
) }
onChange = { ( e ) => {
const { checked } = e . target ;
if ( checked ) {
states . settings . contentTranslationHideLanguages . push (
lang . code ,
) ;
} else {
states . settings . contentTranslationHideLanguages =
snapStates . settings . contentTranslationHideLanguages . filter (
( code ) => code !== lang . code ,
) ;
}
} }
/ > { ' ' }
{ lang . name }
< / label >
) ) }
< / div >
< / p >
2023-07-22 15:59:07 +03:00
< p class = "insignificant" >
2023-03-09 08:20:01 +03:00
< small >
Note : This feature uses an external API to translate ,
powered by { ' ' }
< a
href = "https://github.com/thedaviddelta/lingva-translate"
target = "_blank"
2023-10-02 10:58:59 +03:00
rel = "noopener noreferrer"
2023-03-07 17:38:06 +03:00
>
2023-03-09 08:20:01 +03:00
Lingva Translate
< / a >
.
< / small >
< / p >
2023-07-22 15:59:07 +03:00
< hr / >
< div >
< label >
< input
type = "checkbox"
checked = { snapStates . settings . contentTranslationAutoInline }
disabled = { ! snapStates . settings . contentTranslation }
onChange = { ( e ) => {
states . settings . contentTranslationAutoInline =
e . target . checked ;
} }
/ > { ' ' }
Auto inline translation
< / label >
< p class = "insignificant" >
< small >
Automatically show translation for posts in timeline . Only
works for < b > short < / b > posts without content warning ,
media and poll .
< / small >
< / p >
< / div >
2023-03-09 08:20:01 +03:00
< / div >
2023-03-07 17:38:06 +03:00
< / li >
2023-04-23 07:08:41 +03:00
< li >
< label >
< input
type = "checkbox"
checked = { snapStates . settings . cloakMode }
onChange = { ( e ) => {
states . settings . cloakMode = e . target . checked ;
} }
/ > { ' ' }
2023-04-23 14:47:49 +03:00
Cloak mode { ' ' }
< span class = "insignificant" >
( < samp > Text < / samp > → < samp > █ █ █ █ < / samp > )
< / span >
2023-04-23 07:08:41 +03:00
< / label >
2023-04-23 14:47:49 +03:00
< div class = "sub-section insignificant" >
< small >
Replace text as blocks , useful when taking screenshots , for
privacy reasons .
< / small >
< / div >
2023-04-23 07:08:41 +03:00
< / li >
2023-09-14 19:28:20 +03:00
{ authenticated && (
< li >
< button
type = "button"
class = "light"
onClick = { ( ) => {
states . showDrafts = true ;
states . showSettings = false ;
} }
>
Unsent drafts
< / button >
< / li >
) }
2023-02-28 12:12:17 +03:00
< / ul >
< / section >
2023-09-14 19:28:20 +03:00
{ authenticated && < PushNotificationsSection onClose = { onClose } / > }
2023-03-07 19:32:33 +03:00
< h3 > About < / h3 >
2023-01-17 12:58:04 +03:00
< section >
2023-03-09 06:23:07 +03:00
< div
style = { {
display : 'flex' ,
gap : 8 ,
lineHeight : 1.25 ,
alignItems : 'center' ,
marginTop : 8 ,
} }
>
2023-01-25 19:54:30 +03:00
< img
src = { logo }
alt = ""
2023-03-09 06:23:07 +03:00
width = "64"
height = "64"
2023-01-25 19:54:30 +03:00
style = { {
aspectRatio : '1/1' ,
verticalAlign : 'middle' ,
2023-03-09 06:23:07 +03:00
background : '#b7cdf9' ,
borderRadius : 12 ,
2023-01-25 19:54:30 +03:00
} }
2023-03-09 06:23:07 +03:00
/ >
< div >
< b > Phanpy < / b > { ' ' }
< a
href = "https://hachyderm.io/@phanpy"
// target="_blank"
2023-10-02 10:58:59 +03:00
rel = "noopener noreferrer"
2023-03-09 06:23:07 +03:00
onClick = { ( e ) => {
e . preventDefault ( ) ;
states . showAccount = 'phanpy@hachyderm.io' ;
} }
>
@ phanpy
< / a >
< br / >
2023-09-26 05:55:36 +03:00
< a
href = "https://github.com/cheeaun/phanpy"
target = "_blank"
rel = "noopener noreferrer"
>
2023-03-09 06:23:07 +03:00
Built
< / a > { ' ' }
by { ' ' }
< a
href = "https://mastodon.social/@cheeaun"
// target="_blank"
2023-10-02 10:58:59 +03:00
rel = "noopener noreferrer"
2023-03-09 06:23:07 +03:00
onClick = { ( e ) => {
e . preventDefault ( ) ;
states . showAccount = 'cheeaun@mastodon.social' ;
} }
>
@ cheeaun
< / a >
< / div >
< / div >
2023-01-30 18:16:00 +03:00
< p >
2023-10-03 10:07:47 +03:00
< a
href = "https://github.com/sponsors/cheeaun"
target = "_blank"
rel = "noopener noreferrer"
>
Sponsor
< / a > { ' ' }
& middot ; { ' ' }
< a
href = "https://www.buymeacoffee.com/cheeaun"
target = "_blank"
rel = "noopener noreferrer"
>
Donate
< / a > { ' ' }
& middot ; { ' ' }
2023-01-30 18:16:00 +03:00
< a
href = "https://github.com/cheeaun/phanpy/blob/main/PRIVACY.MD"
target = "_blank"
2023-10-02 10:58:59 +03:00
rel = "noopener noreferrer"
2023-01-30 18:16:00 +03:00
>
Privacy Policy
< / a >
2022-12-25 13:01:01 +03:00
< / p >
2023-01-17 12:58:04 +03:00
{ _ _BUILD _TIME _ _ && (
< p >
2023-10-23 03:43:27 +03:00
Version : { ' ' }
< input
type = "text"
class = "version-string"
readOnly
size = "18" // Manually calculated here
value = { ` ${ _ _BUILD _TIME _ _ . slice ( 0 , 10 ) . replace ( /-/g , '.' ) } ${
_ _COMMIT _HASH _ _ ? ` . ${ _ _COMMIT _HASH _ _ } ` : ''
} ` }
onClick = { ( e ) => {
e . target . select ( ) ;
// Copy to clipboard
try {
navigator . clipboard . writeText ( e . target . value ) ;
showToast ( 'Version string copied' ) ;
} catch ( e ) {
console . warn ( e ) ;
showToast ( 'Unable to copy version string' ) ;
}
} }
/ > { ' ' }
< span class = "ib insignificant" >
(
< a
href = { ` https://github.com/cheeaun/phanpy/commit/ ${ _ _COMMIT _HASH _ _ } ` }
target = "_blank"
rel = "noopener noreferrer"
>
< RelativeTime datetime = { new Date ( _ _BUILD _TIME _ _ ) } / >
< / a >
)
< / span >
2023-01-17 12:58:04 +03:00
< / p >
) }
< / section >
2022-12-25 13:01:01 +03:00
< / main >
2022-12-10 12:14:48 +03:00
< / div >
) ;
2022-12-16 08:27:04 +03:00
}
2023-09-01 10:40:00 +03:00
function PushNotificationsSection ( { onClose } ) {
if ( ! isPushSupported ( ) ) return null ;
const { instance } = api ( ) ;
const [ uiState , setUIState ] = useState ( 'default' ) ;
const pushFormRef = useRef ( ) ;
const [ allowNofitications , setAllowNotifications ] = useState ( false ) ;
const [ needRelogin , setNeedRelogin ] = useState ( false ) ;
const previousPolicyRef = useRef ( ) ;
useEffect ( ( ) => {
( async ( ) => {
setUIState ( 'loading' ) ;
try {
const { subscription , backendSubscription } = await initSubscription ( ) ;
if (
backendSubscription ? . policy &&
backendSubscription . policy !== 'none'
) {
setAllowNotifications ( true ) ;
const { alerts , policy } = backendSubscription ;
previousPolicyRef . current = policy ;
const { elements } = pushFormRef . current ;
const policyEl = elements . namedItem ( policy ) ;
if ( policyEl ) policyEl . value = policy ;
// alerts is {}, iterate it
Object . keys ( alerts ) . forEach ( ( alert ) => {
const el = elements . namedItem ( alert ) ;
if ( el ? . type === 'checkbox' ) {
el . checked = true ;
}
} ) ;
}
setUIState ( 'default' ) ;
} catch ( err ) {
console . warn ( err ) ;
if ( /outside.*authorized/i . test ( err . message ) ) {
setNeedRelogin ( true ) ;
} else {
alert ( err ? . message || err ) ;
}
setUIState ( 'error' ) ;
}
} ) ( ) ;
} , [ ] ) ;
const isLoading = uiState === 'loading' ;
return (
< form
ref = { pushFormRef }
onChange = { ( ) => {
const values = Object . fromEntries ( new FormData ( pushFormRef . current ) ) ;
const allowNofitications = ! ! values [ 'policy-allow' ] ;
const params = {
policy : values . policy ,
data : {
alerts : {
mention : ! ! values . mention ,
favourite : ! ! values . favourite ,
reblog : ! ! values . reblog ,
follow : ! ! values . follow ,
follow _request : ! ! values . followRequest ,
poll : ! ! values . poll ,
update : ! ! values . update ,
status : ! ! values . status ,
} ,
} ,
} ;
let alertsCount = 0 ;
// Remove false values from data.alerts
// API defaults to false anyway
Object . keys ( params . data . alerts ) . forEach ( ( key ) => {
if ( ! params . data . alerts [ key ] ) {
delete params . data . alerts [ key ] ;
} else {
alertsCount ++ ;
}
} ) ;
const policyChanged = previousPolicyRef . current !== params . policy ;
console . log ( 'PN Form' , { values , allowNofitications , params } ) ;
if ( allowNofitications && alertsCount > 0 ) {
if ( policyChanged ) {
console . debug ( 'Policy changed.' ) ;
removeSubscription ( )
. then ( ( ) => {
updateSubscription ( params ) ;
} )
. catch ( ( err ) => {
console . warn ( err ) ;
alert ( 'Failed to update subscription. Please try again.' ) ;
} ) ;
} else {
updateSubscription ( params ) . catch ( ( err ) => {
console . warn ( err ) ;
alert ( 'Failed to update subscription. Please try again.' ) ;
} ) ;
}
} else {
removeSubscription ( ) . catch ( ( err ) => {
console . warn ( err ) ;
alert ( 'Failed to remove subscription. Please try again.' ) ;
} ) ;
}
} }
>
< h3 > Push Notifications ( beta ) < / h3 >
< section >
< ul >
< li >
< label >
< input
type = "checkbox"
disabled = { isLoading || needRelogin }
name = "policy-allow"
checked = { allowNofitications }
onChange = { async ( e ) => {
const { checked } = e . target ;
if ( checked ) {
// Request permission
const permission = await Notification . requestPermission ( ) ;
if ( permission === 'granted' ) {
setAllowNotifications ( true ) ;
} else {
setAllowNotifications ( false ) ;
if ( permission === 'denied' ) {
alert (
'Push notifications are blocked. Please enable them in your browser settings.' ,
) ;
}
}
} else {
setAllowNotifications ( false ) ;
}
} }
/ > { ' ' }
Allow from { ' ' }
< select
name = "policy"
disabled = { isLoading || needRelogin || ! allowNofitications }
>
{ [
{
value : 'all' ,
label : 'anyone' ,
} ,
{
value : 'followed' ,
label : 'people I follow' ,
} ,
{
value : 'follower' ,
label : 'followers' ,
} ,
] . map ( ( type ) => (
< option value = { type . value } > { type . label } < / option >
) ) }
< / select >
< / label >
< div
class = "shazam-container no-animation"
style = { {
width : '100%' ,
} }
hidden = { ! allowNofitications }
>
< div class = "shazam-container-inner" >
< div class = "sub-section" >
< ul >
{ [
{
value : 'mention' ,
label : 'Mentions' ,
} ,
{
value : 'favourite' ,
label : 'Favourites' ,
} ,
{
value : 'reblog' ,
label : 'Boosts' ,
} ,
{
value : 'follow' ,
label : 'Follows' ,
} ,
{
value : 'followRequest' ,
label : 'Follow requests' ,
} ,
{
value : 'poll' ,
label : 'Polls' ,
} ,
{
value : 'update' ,
label : 'Post edits' ,
} ,
{
value : 'status' ,
label : 'New posts' ,
} ,
] . map ( ( alert ) => (
< li >
< label >
< input type = "checkbox" name = { alert . value } / > { ' ' }
{ alert . label }
< / label >
< / li >
) ) }
< / ul >
< / div >
< / div >
< / div >
{ needRelogin && (
< div class = "sub-section" >
< p >
Push permission was not granted since your last login . You ' ll
need to { ' ' }
< Link to = { ` /login?instance= ${ instance } ` } onClick = { onClose } >
< b > log in < / b > again to grant push permission
< / Link >
.
< / p >
< / div >
) }
< / li >
< / ul >
< / section >
< p class = "section-postnote" >
< small >
2023-09-05 08:26:30 +03:00
NOTE : Push notifications only work for < b > one account < / b > .
2023-09-01 10:40:00 +03:00
< / small >
< / p >
< / form >
) ;
}
2022-12-16 08:27:04 +03:00
export default Settings ;