ReactNative字體大小不隨系統(tǒng)字體大小變化而變化

引言

在開(kāi)發(fā)react-nativeApp時(shí),相信大家都應(yīng)該遇到過(guò)這樣的問(wèn)題:用戶(hù)設(shè)置了系統(tǒng)的字體大小之后,導(dǎo)致自己的APP布局紊亂,甚至有些內(nèi)容會(huì)被切掉/隱藏,這對(duì)于用戶(hù)來(lái)講,是非常不好的用戶(hù)體驗(yàn)。

那為什么會(huì)出現(xiàn)這種情況呢呢?原因是我們?cè)陂_(kāi)發(fā)的時(shí)候,布局的前提是系統(tǒng)的字體大小設(shè)置為默認(rèn)大小,所以只能保證在系統(tǒng)字體大小正常的情況下,我們的布局是友好的,

那么,我們應(yīng)該如何解決這個(gè)問(wèn)題呢?今天這篇文章,就給大家介紹幾種解決方案。

Text和TextInput

react-native中用來(lái)顯示文字的,一般會(huì)用到兩個(gè)組件:TextTextInput。所以,我們只要針對(duì)這兩個(gè)組件做好工作,那就基本上解決了字體大小適配的問(wèn)題

TextTextInput它們有一個(gè)共同屬性:

allowFontScaling

這個(gè)屬性在react-native官方文檔中解釋如下:

Specifies whether fonts should scale to respect Text Size accessibility settings. The default is true.

意思是:

是否隨系統(tǒng)指定的文字大小變化而變化。默認(rèn)值為true

這給我提供了解決方案,只要我們給TextTextInput的屬性allowFontScaling設(shè)置值為false,那么文字大小就不會(huì)隨系統(tǒng)字體大小的變化而變化。

設(shè)置allowFontScaling

我們有幾種方式來(lái)設(shè)置TextTextInputallowFontScaling。第一種:

1. 給TextTextInput組件設(shè)置allowFontScaling = false

<Text allowFontScaling={false}/>
<TextInput allowFontScaling={false}/> 

這種方案效率很低,需要在每個(gè)使用到這兩個(gè)組件的地方都加上這個(gè)屬性。但是一般這兩個(gè)組件的使用率還是很高的,所以這是一個(gè)龐大的工作量,而且在開(kāi)發(fā)過(guò)程當(dāng)中,我們也很容易忘記設(shè)置它

那么有沒(méi)有更好實(shí)現(xiàn)方式呢?當(dāng)然有,這就是下面講的第二種:

2. 自定義MyText/MyTextInput組件

我們可以自定義一個(gè)組件MyText, 然后統(tǒng)一設(shè)置allowFontScaling = false屬性,然后在其他需要調(diào)用的時(shí)候,直接用MyText代替Text

MyText.js

import React from 'react'
import {Text} from 'react-native'

export default class MyText extends React.Component {

    render() {
        return (
            <Text
                allowFontScaling={false}
                {...this.props}>
                {this.props.children}
            </Text>
        )
    }
}

這個(gè)方案足夠好了,但是,你仍然可能在某段代碼里,忘記使用MyText而是直接使用Text,這個(gè)時(shí)候,問(wèn)題依然會(huì)出現(xiàn)。

那么,就沒(méi)有一種萬(wàn)無(wú)一失的方案嗎?當(dāng)然有啦,第三種:

3. 重寫(xiě)Text的render()

是的,我們可以重寫(xiě)Textrender()方法,讓Text在渲染的時(shí)候,設(shè)置allowFontScaling = false。這里,我們需要用到lodashwrap()方法:

0.56(不包括)版本之前

Text.prototype.render = _.wrap(Text.prototype.render, function (func, ...args) {
    let originText = func.apply(this, args)
    return React.cloneElement(originText, {allowFontScaling: false})
})

注意1:
react-native版本0.56之前,Text是通過(guò)React的createReactClass方式來(lái)創(chuàng)建類(lèi)的,也就是說(shuō),是通過(guò)javascriptprototype的方式來(lái)創(chuàng)建類(lèi)。所以重寫(xiě)render方法時(shí),需要通過(guò)Text.prototype.render來(lái)引用

而在0.56版本,Text改為了es6extends的實(shí)現(xiàn)方式來(lái)創(chuàng)建類(lèi),所以,需要如下方式來(lái)重寫(xiě)render

0.56(包括)版本之后

Text.render = _.wrap(Text.render, function (func, ...args) {
    let originText = func.apply(this, args)
    return React.cloneElement(originText, {allowFontScaling: false})
})

大家可以查看源碼,或者查看0.56change-log

注意2:
這段代碼最好放在你app整個(gè)組件執(zhí)行調(diào)用之前,比如在我的項(xiàng)目中,我放的位置:

import React from 'react'
import {AppRegistry, Text, DeviceEventEmitter, YellowBox} from 'react-native'
import {Provider} from 'react-redux'
import App from './src/App'
import _ from 'lodash'


//字體不隨系統(tǒng)字體變化
Text.render = _.wrap(Text.render, function (func, ...args) {
    let originText = func.apply(this, args)
    return React.cloneElement(originText, {allowFontScaling: false})
})

...
...

class MyApp extends React.Component {
    render() {
        return (
            <Provider store={store}>
                <App/>
            </Provider>
        )
    }
}

AppRegistry.registerComponent("xxx", () => MyApp);

注意3:
但是很遺憾的是,這個(gè)只適用于TextTextInput不能用于此方案。

那么,有沒(méi)有一種方案,能夠同時(shí)兼容TextTextInput并且做到一勞永逸呢?當(dāng)然有了,終極方案:

4. 完美方案:修改defaultProps

首先我們來(lái)看各種組件的源碼.

TextInput.js

...
  getDefaultProps(): Object {
    return {
      allowFontScaling: true,
      underlineColorAndroid: 'transparent',
    };
  },
...

Text.js

...
  static defaultProps = {
    accessible: true,
    allowFontScaling: true,
    ellipsizeMode: 'tail',
  };
...

通過(guò)這兩個(gè)代碼片段可以知道,在定義TextTextInput時(shí),都有給組件設(shè)置默認(rèn)屬性的操作.

所以我們可以:

TextInput.defaultProps = Object.assign({}, TextInput.defaultProps, {defaultProps: false})
Text.defaultProps = Object.assign({}, Text.defaultProps, {allowFontScaling: false})

來(lái)直接設(shè)置TextTextInputallowFontScaling屬性默認(rèn)值為false,真正實(shí)現(xiàn)了一勞永逸。

確保react-navigation兼容

通過(guò)設(shè)置defaultProps的方式來(lái)修改allowFontScaling的值為false,會(huì)有一個(gè)問(wèn)題。

大家在使用react-native時(shí),最常用到的navigator應(yīng)該是react-navigation。你需要單獨(dú)設(shè)置headertitleallowfontscalingallowFontScaling來(lái)確保react-navigationtabTitleheaderTitle沒(méi)有問(wèn)題。

結(jié)語(yǔ)

好了,到此,我們就完美解決了react-native開(kāi)發(fā)中,字體大小不隨系統(tǒng)字體大小變化而變化的問(wèn)題。

我們總結(jié)一下:

  1. react-native中使用TextTextInput負(fù)責(zé)顯示文字信息
  2. TextTextInput中設(shè)置allowFontScaling=false可以讓字體大小不隨系統(tǒng)設(shè)置而變化
  3. 可以通過(guò)單個(gè)組件設(shè)置、自定義組件、重寫(xiě)render()、設(shè)置defaultProps默認(rèn)值這四種方式來(lái)設(shè)置allowFontScaling的值為false
  4. 對(duì)于重寫(xiě)render()、設(shè)置defaultProps默認(rèn)值這兩種方式,需要把設(shè)置代碼放在app組件初始化之前。
  5. 確保react-navigation兼容
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,316評(píng)論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,481評(píng)論 3 415
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 176,241評(píng)論 0 374
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 62,939評(píng)論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,697評(píng)論 6 409
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 55,182評(píng)論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,247評(píng)論 3 441
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 42,406評(píng)論 0 288
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,933評(píng)論 1 334
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,772評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,973評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,516評(píng)論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,209評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 34,638評(píng)論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 35,866評(píng)論 1 285
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,644評(píng)論 3 391
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,953評(píng)論 2 373

推薦閱讀更多精彩內(nèi)容