diff --git a/src/@types/react-navigation.d.ts b/src/@types/react-navigation.d.ts index 6d02bc1a..ce295b28 100644 --- a/src/@types/react-navigation.d.ts +++ b/src/@types/react-navigation.d.ts @@ -118,6 +118,7 @@ declare namespace Nav { title: Mastodon.List['title'] } 'Tab-Me-Settings': undefined + 'Tab-Me-Settings-Fontsize': undefined 'Tab-Me-Settings-Push': undefined 'Tab-Me-Switch': undefined } & TabSharedStackParamList diff --git a/src/components/Instance/Info.tsx b/src/components/Instance/Info.tsx index bb5dcd5d..b651e90f 100644 --- a/src/components/Instance/Info.tsx +++ b/src/components/Instance/Info.tsx @@ -1,8 +1,6 @@ -import { ParseHTML } from '@components/Parse' import { StyleConstants } from '@utils/styles/constants' import { useTheme } from '@utils/styles/ThemeManager' import React from 'react' -import { useTranslation } from 'react-i18next' import { StyleSheet, Text, View, ViewStyle } from 'react-native' import { PlaceholderLine } from 'rn-placeholder' @@ -15,19 +13,15 @@ export interface Props { const InstanceInfo = React.memo( ({ style, header, content, potentialWidth }: Props) => { - const { t } = useTranslation('componentInstance') const { theme } = useTheme() return ( {header} {content ? ( - + + {content} + ) : ( { + ({ + content, + emojis, + size = 'M', + adaptiveSize = false, + fontBold = false + }: Props) => { + const adaptiveFontsize = useSelector(getSettingsFontsize) + const adaptedFontsize = adaptiveScale( + StyleConstants.Font.Size[size], + adaptiveSize ? adaptiveFontsize : 0 + ) + const adaptedLineheight = adaptiveScale( + StyleConstants.Font.LineHeight[size], + adaptiveSize ? adaptiveFontsize : 0 + ) + const { mode, theme } = useTheme() const styles = useMemo(() => { return StyleSheet.create({ text: { color: theme.primary, - ...StyleConstants.FontStyle[size], + fontSize: adaptedFontsize, + lineHeight: adaptedLineheight, ...(fontBold && { fontWeight: StyleConstants.Font.Weight.Bold }) }, image: { - width: StyleConstants.Font.Size[size], - height: StyleConstants.Font.Size[size], - transform: [{ translateY: size === 'L' ? -3 : -1 }] + width: adaptedFontsize, + height: adaptedFontsize, + transform: [{ translateY: -2 }] } }) - }, [mode]) + }, [mode, adaptiveFontsize]) return ( @@ -50,6 +70,7 @@ const ParseEmojis = React.memo( {/* When emoji starts a paragraph, lineHeight will break */} {i === 0 ? : null} diff --git a/src/components/Parse/HTML.tsx b/src/components/Parse/HTML.tsx index 25fa4c02..328b9612 100644 --- a/src/components/Parse/HTML.tsx +++ b/src/components/Parse/HTML.tsx @@ -4,7 +4,8 @@ import openLink from '@components/openLink' import ParseEmojis from '@components/Parse/Emojis' import { useNavigation, useRoute } from '@react-navigation/native' import { StackNavigationProp } from '@react-navigation/stack' -import { StyleConstants } from '@utils/styles/constants' +import { getSettingsFontsize } from '@utils/slices/settingsSlice' +import { adaptiveScale, StyleConstants } from '@utils/styles/constants' import layoutAnimation from '@utils/styles/layoutAnimation' import { useTheme } from '@utils/styles/ThemeManager' import { LinearGradient } from 'expo-linear-gradient' @@ -12,6 +13,7 @@ import React, { useCallback, useState } from 'react' import { useTranslation } from 'react-i18next' import { Platform, Pressable, Text, View } from 'react-native' import HTMLView from 'react-native-htmlview' +import { useSelector } from 'react-redux' // Prevent going to the same hashtag multiple times const renderNode = ({ @@ -19,7 +21,8 @@ const renderNode = ({ theme, node, index, - size, + adaptedFontsize, + adaptedLineheight, navigation, mentions, tags, @@ -30,7 +33,8 @@ const renderNode = ({ theme: any node: any index: number - size: 'S' | 'M' | 'L' + adaptedFontsize: number + adaptedLineheight: number navigation: StackNavigationProp mentions?: Mastodon.Mention[] tags?: Mastodon.Tag[] @@ -52,7 +56,8 @@ const renderNode = ({ key={index} style={{ color: theme.blue, - ...StyleConstants.FontStyle[size] + fontSize: adaptedFontsize, + lineHeight: adaptedLineheight }} onPress={() => { analytics('status_hashtag_press') @@ -79,7 +84,8 @@ const renderNode = ({ key={index} style={{ color: accountIndex !== -1 ? theme.blue : undefined, - ...StyleConstants.FontStyle[size] + fontSize: adaptedFontsize, + lineHeight: adaptedLineheight }} onPress={() => { analytics('status_mention_press') @@ -108,8 +114,9 @@ const renderNode = ({ key={index} style={{ color: theme.blue, - ...StyleConstants.FontStyle[size], - alignItems: 'center' + alignItems: 'center', + fontSize: adaptedFontsize, + lineHeight: adaptedLineheight }} onPress={async () => { analytics('status_link_press') @@ -125,9 +132,9 @@ const renderNode = ({ ) : null} @@ -146,6 +153,7 @@ const renderNode = ({ export interface Props { content: string size?: 'S' | 'M' | 'L' + adaptiveSize?: boolean emojis?: Mastodon.Emoji[] mentions?: Mastodon.Mention[] tags?: Mastodon.Tag[] @@ -158,6 +166,7 @@ export interface Props { const ParseHTML: React.FC = ({ content, size = 'M', + adaptiveSize = false, emojis, mentions, tags, @@ -166,6 +175,16 @@ const ParseHTML: React.FC = ({ expandHint, disableDetails = false }) => { + const adaptiveFontsize = useSelector(getSettingsFontsize) + const adaptedFontsize = adaptiveScale( + StyleConstants.Font.Size[size], + adaptiveSize ? adaptiveFontsize : 0 + ) + const adaptedLineheight = adaptiveScale( + StyleConstants.Font.LineHeight[size], + adaptiveSize ? adaptiveFontsize : 0 + ) + const navigation = useNavigation< StackNavigationProp >() @@ -183,7 +202,8 @@ const ParseHTML: React.FC = ({ theme, node, index, - size, + adaptedFontsize, + adaptedLineheight, navigation, mentions, tags, @@ -199,6 +219,7 @@ const ParseHTML: React.FC = ({ content={children.toString()} emojis={emojis} size={size} + adaptiveSize={adaptiveSize} /> ) } else { @@ -208,7 +229,6 @@ const ParseHTML: React.FC = ({ const rootComponent = useCallback( ({ children }) => { const { t } = useTranslation('componentParse') - const lineHeight = StyleConstants.Font.LineHeight[size] const [expandAllow, setExpandAllow] = useState(false) const [expanded, setExpanded] = useState(false) @@ -234,10 +254,6 @@ const ParseHTML: React.FC = ({ children={children} onTextLayout={onTextLayout} numberOfLines={expanded ? 999 : numberOfLines + 1} - style={{ - ...StyleConstants.FontStyle[size], - color: theme.primary - }} /> {expandAllow ? ( = ({ style={{ marginTop: expanded ? 0 - : -lineHeight * (numberOfLines === 0 ? 1 : 2) + : -adaptedLineheight * (numberOfLines === 0 ? 1 : 2) }} > = ({ theme.backgroundGradientStart, theme.backgroundGradientEnd ]} - locations={[ - 0, - lineHeight / (StyleConstants.Font.Size[size] * 5) - ]} + locations={[0, adaptedLineheight / (adaptedFontsize * 5)]} style={{ paddingTop: StyleConstants.Font.Size.S * 2, paddingBottom: StyleConstants.Font.Size.S diff --git a/src/components/Timeline/Shared/Content.tsx b/src/components/Timeline/Shared/Content.tsx index f40f7ca3..f32bff36 100644 --- a/src/components/Timeline/Shared/Content.tsx +++ b/src/components/Timeline/Shared/Content.tsx @@ -28,6 +28,7 @@ const TimelineContent = React.memo( () @@ -115,6 +116,19 @@ const TabMe = React.memo( headerLeft: () => navigation.pop(1)} /> })} /> + ({ + headerTitle: t('meSettingsFontsize:heading'), + ...(Platform.OS === 'android' && { + headerCenter: () => ( + + ) + }), + headerLeft: () => navigation.pop(1)} /> + })} + /> { + switch (size) { + case -1: + return 'S' + case 0: + return 'M' + case 1: + return 'L' + case 2: + return 'XL' + case 3: + return 'XXL' + } +} + +const ScreenMeSettingsFontsize: React.FC> = () => { + const { mode, theme } = useTheme() + const { t } = useTranslation('meSettingsFontsize') + const initialSize = useSelector(getSettingsFontsize) + const dispatch = useDispatch() + + const item = { + id: 'demo', + uri: 'https://tooot.app', + created_at: new Date(), + sensitive: false, + visibility: 'public', + replies_count: 0, + reblogs_count: 0, + favourites_count: 0, + favourited: true, + reblogged: false, + muted: false, + bookmarked: false, + content: + '

- tooot supports multiple accounts
- tooot supports browsing external instance
- tooot aims to support multiple languages

', + reblog: null, + application: { + name: 'tooot', + website: 'https://tooot.app' + }, + account: { + id: 'demo', + url: 'https://tooot.app', + username: 'tooot📱', + acct: 'tooot@xmflsct.com', + display_name: 'tooot📱', + avatar_static: 'https://avatars.githubusercontent.com/u/77554750?s=100' + }, + media_attachments: [], + mentions: [] + } + + const sizesDemo = useMemo(() => { + return ( + <> + {([-1, 0, 1, 2, 3] as [-1, 0, 1, 2, 3]).map(size => ( + + {t(`content.sizes.${mapFontsizeToName(size)}`)} + + ))} + + ) + }, [mode, initialSize]) + + return ( + + + {t('content.showcase')} + + + + + + + + {t('content.availableSizes')} + + {sizesDemo} + +