Understand the real 『REM』 mobile screen adaptation

Understand the real 『REM』 mobile screen adaptation

As a low-key unit of length, rem has been reused in screen adaptation due to the rise of mobile web pages. Using rem, front-end developers can easily achieve the required effect of the design drawing by proportional scaling on various screen sizes.

The official definition of rem is "The font size of the root element.", which means that the font size of the root node is used as the reference value for length calculation. It is generally believed that the root node of a web page is the html element, so the method used is to set the font-size of the html element to adapt to the screen, but is the actual situation really that simple?

First, let's take a look at the common solutions for using rem to achieve mobile screen adaptation.

The width of the design draft is 640px, that is: designWidth = 640, and at the same time, 1rem=100px is set under 640px screen width, that is: rem2px = 100.

The advantage of setting 1rem=100px is self-evident. When front-end developers are cutting pictures and reconstructing pages, they can convert the px value measured in the UI picture into the corresponding rem value by directly shifting the decimal point, which is convenient and quick.

In addition, in the head we also set: <meta name="viewport" content="width=device-width, initial-scale=1.0, minimum-scale=1.0, maximum-scale=1.0" />

The role of viewport is very important, but it is not the focus of this article so I will not elaborate on it. Students who are interested can search for it on their own.

Let’s take a look at the specific plan first:

The following four solutions are shared by colleagues. The principle is to use proportional scaling - get the ratio of the target screen width to the design draft width, use it as the base value of rem (scaling factor), and set it as the font size of the HTML tag. The only difference is the performance trade-off and writing habits.

Solution 1

  1. @media screen and ( min -width: 320px) {html{font- size :50px;}}
  2. @media screen and ( min -width: 360px) {html{font- size :56.25px;}}
  3. @media screen and ( min -width: 375px) {html{font- size :58.59375px;}}
  4. @media screen and ( min -width: 400px) {html{font- size :62.5px;}}
  5. @media screen and ( min -width: 414px) {html{font- size :64.6875px;}}
  6. @media screen and ( min -width: 440px) {html{font- size :68.75px;}}
  7. @media screen and ( min -width: 480px) {html{font- size :75px;}}
  8. @media screen and ( min -width: 520px) {html{font- size :81.25px;}}
  9. @media screen and ( min -width: 560px) {html{font- size :87.5px;}}
  10. @media screen and ( min -width: 600px) {html{font- size :93.75px;}}
  11. @media screen and ( min -width: 640px) {html{font- size :100px;}}
  12. @media screen and ( min -width: 680px) {html{font- size :106.25px;}}
  13. @media screen and ( min -width: 720px) {html{font- size :112.5px;}}
  14. @media screen and ( min -width: 760px) {html{font- size :118.75px;}}
  15. @media screen and ( min -width: 800px) {html{font- size :125px;}}
  16. @media screen and ( min -width: 960px) {html{font- size :150px;}}

Solution 2

  1. @media screen and ( min -width: 320px) {html{font- size :312.5%;}}
  2. @media screen and ( min -width: 360px) {html{font- size :351.5625%;}}
  3. @media screen and ( min -width: 375px) {html{font- size :366.211%;}}
  4. @media screen and ( min -width: 400px) {html{font- size :390.625%;}}
  5. @media screen and ( min -width: 414px) {html{font- size :404.2969%;}}
  6. @media screen and ( min -width: 440px) {html{font- size :429.6875%;}}
  7. @media screen and ( min -width: 480px) {html{font- size :468.75%;}}
  8. @media screen and ( min -width: 520px) {html{font- size :507.8125%;}}
  9. @media screen and ( min -width: 560px) {html{font- size :546.875%;}}
  10. @media screen and ( min -width: 600px) {html{font- size :585.9375%;}}
  11. @media screen and ( min -width: 640px) {html{font- size :625%;}}
  12. @media screen and ( min -width: 680px) {html{font- size :664.0625%;}}
  13. @media screen and ( min -width: 720px) {html{font- size :703.125%;}}
  14. @media screen and ( min -width: 760px) {html{font- size :742.1875%;}}
  15. @media screen and ( min -width: 800px) {html{font- size :781.25%;}}
  16. @media screen and ( min -width: 960px) {html{font- size :937.5%;}}

Solution 3

  1. var designWidth = 640, rem2px = 100;
  2. document.documentElement.style.fontSize =
  3. ((window.innerWidth / designWidth) * rem2px) + 'px' ;

Solution 4

  1. var designWidth = 640, rem2px = 100;
  2. document.documentElement.style.fontSize =
  3. ((((window.innerWidth / designWidth) * rem2px) / 16) * 100) + '%' ;

In order to avoid confusion in understanding, I added () in the above js code, which is not necessary in the actual code.

After a detailed analysis, the direct conversion formula between rem and px can be written as:

  1. 1rem = 1 * htmlFontSize

htmlFontSize is the font size of the html element.

First, let’s look at the settings in Solution 1 when the screen width is 640px:

  1. @media screen and ( min -width: 640px) {html{font- size :100px;}}

This can be clearly shown as 1rem = 1 * 100px, which is the same as our initial setting. So what should we do to get the htmlFontSize value for other screen sizes? It is very simple, as in solution 3, because we use proportional scaling to adapt, so we can calculate the ratio of the target screen width to the width of the design draft:

  1. window.innerWidth / designWidth * rem2px + 'px'  

Since the default browser font size is 16px, when we use a percentage as the font size of the root node html, that is, the font-size value of the html element is set to a percentage value, the calculation method of rem will be changed to:

  1. defaultFontSize = 16px
  2. 1rem = 1 * htmlFontSize * defaultFontSize

As in solution 2, the settings when the screen width is 640px are:

  1. @media screen and ( min -width: 640px) {html{font- size :625%;}}

Applying the formula above:

  1. 1rem = 1 * 625% * 16px

Where: 625% * 16 = 6.25 * 16 = 100

So: 1rem = 1 * 100px

Similarly, we can get the calculation formula of the font-size value of HTML under all screen sizes, which is Solution 4:

  1. window.innerWidth / designWidth * rem2px / 16 * 100 + '%'  

Through the formulas of Solution 3 and Solution 4, the CSS in Solution 1 and Solution 2 can be easily generated.

Only the corresponding verification pages of Scheme 3 and Scheme 4 are given here (Scheme 1 and Scheme 2 are their variations): scheme3.html(http://htmlpreview.github.io/?https://github.com/hbxeagle/rem/blob/master/scheme3.html), scheme4.html(http://htmlpreview.github.io/?https://github.com/hbxeagle/rem/blob/master/scheme4.html)

The following two pictures show the effect when the screen width is 360px. The target is calculated as: 1rem = 56.25px. The setting value of solution 3 is: 56.25px, and the setting value of solution 4 is: 351.5625%

So far, it seems that the problem has been solved perfectly, but the actual situation is of course unexpected. On some Android phones, the default font of the browser or webview changes with the font set by the system. This will cause the default font to be larger or smaller than 16px.

After modifying the default font size, let's look at options 3 and 4.

Similarly, when the screen width is 360px, we increase the system font size, as shown below

The calculated value of the font size of the html element before setting is 18px, and the calculated value after setting is 65px. Since the screen width has not changed, our target value, that is, the font-size value we set on the html element, has not changed and is still 56.25px, but the final calculated value has deviated.

Before analyzing the deviation, let’s first look at the calculation process of Scheme 3 and Scheme 4 at a screen width of 360px:

Scenario 3:

  1. document.documentElement.style.fontSize = 56.25px
  2. htmlFontSize = 56.25px
  3. 1rem = 1 * htmlFontSize = 56.25px

The actual value is:

1rem = 64.6875px

Solution 4:

  1. document.documentElement.style.fontSize = 351.5625%
  2. htmlFontSize = 351.5625%
  3. defaultFontSize = 18px
  4. 1rem = 1 * htmlFontSize * defaultFontSize = 351.5625% * 18px = 63.28125px
  5. 351.5625% * 18 = 63.28125

The actual value is:

1rem = 64.6875px

It seems that the calculation result of Scheme 4 is very close to the actual effect, while the deviation of Scheme 3 is large. Let's compare the calculation formulas of Scheme 3 and Scheme 4:

  1. // Solution 3
  2. document.documentElement.style.fontSize =
  3. window.innerWidth / designWidth * rem2px + 'px' ;
  4.   
  5. // Solution 4
  6. document.documentElement.style.fontSize =
  7. window.innerWidth / designWidth * rem2px / 16 * 100 + '%' ;

Compared with Solution 3, Solution 4 actually has an extra 16. It can be inferred that when the browser calculates the specific value of rem, if the font-size set in html is a px value, it will first divide it by 16 and then multiply it by htmlFontSize.

  1. 1rem = 1 * (56.25px / 16) * 18
  2. 1 * (56.25 / 16) * 18 = 63.28125

There is a problem with solution 4 because the system's default font size has been changed to 18px, but when we calculate the percentage, we still use 16px as the base value, so there is a deviation (there is a slight deviation between the calculated value and the actual value, which will be mentioned later).

In solution 3, we actually do not consider the browser's default font size, but in actual use, the browser still uses 16, and the default font size is 18px. The calculation formula for rem is as follows when the HTML fontSize is set to px:

  1. 1rem = 1 * (htmlFontSize / 16) * defaultFontSize

When the system font size changes, defaultFontSize will change accordingly, but 16 will not change. So although Solution 3 does not consider the change of the default font size on the surface and only focuses on the width ratio between the screen and the design draft, the default font size is still used in the actual calculation, and there is also an unchanging 16 at work, causing Solution 3 to fail.

The so-called "root element" is actually not what you imagine. One is 16 and the other is 18. What they actually take is the font size of the root element.

Ok, when calculating rem, the px method will have a 16 that does not change with the system font size, so we use the percentage solution to circumvent this problem.

Solution 4, which uses percentages, uses a default font size of 16px because it is hardcoded in the calculation. So its deviation is that it fails to dynamically obtain the default font size. Update as follows:

Solution 4.1

  1. var designWidth = 640, rem2px = 100;
  2. var h = document.getElementsByTagName( 'html' )[0];
  3. var htmlFontSize = parseFloat(window.getComputedStyle(h, null )
  4. .getPropertyValue( 'font-size' ));
  5.   
  6. document.documentElement.style.fontSize =
  7. window.innerWidth / designWidth * rem2px / htmlFontSize * 100 + '%' ;

The effect is as follows:

In the 16px image, the font-size of the HTML is different from the actual value of 1rem, and the calculated value of 6.4rem is also different. By looking at the code, it is found that the font-size of HTML uses getPropertyValue('font-size') while 1rem uses getPropertyValue('width'). The deviation is that the browser rounds off when calculating font-size. Another element in the rem definition, "font size", cannot be used literally.

The deviation in 18px and the deviation between the actual value and the calculated value in Solution 4 above are the same problem. So the baseline value needs to be modified.

In the updated version, solution 4.2:

  1. var designWidth = 640, rem2px = 100;
  2. var d = window.document.createElement( 'div' );
  3. d.style.width = '1rem' ;
  4. d.style.display = "none" ;
  5. var head = window.document.getElementsByTagName( 'head' )[0];
  6. head.appendChild(d);
  7. var defaultFontSize = parseFloat(window.getComputedStyle(d, null ).getPropertyValue( 'width' ));
  8. d.remove();
  9. document.documentElement.style.fontSize =
  10. window.innerWidth / designWidth * rem2px / defaultFontSize * 100 + '%' ;

The effect is as follows:

So far, the problem of rem when the default font is not 16px has been solved. Considering the design of screen rotation, the final solution for mobile phones is:

  1. function adapt(designWidth, rem2px){
  2. var d = window.document.createElement( 'div' );
  3. d.style.width = '1rem' ;
  4. d.style.display = "none" ;
  5. var head = window.document.getElementsByTagName( 'head' )[0];
  6. head.appendChild(d);
  7. var defaultFontSize = parseFloat(window.getComputedStyle(d, null ).getPropertyValue( 'width' ));
  8. d.remove();
  9. document.documentElement.style.fontSize = window.innerWidth / designWidth * rem2px / defaultFontSize * 100 + '%' ;
  10. var st = document.createElement( 'style' );
  11. var portrait = "@media screen and (min-width: " +window.innerWidth+ "px) {html{font-size:" + ((window.innerWidth/(designWidth/rem2px)/defaultFontSize)*100) + "%;}}" ;
  12. var landscape = "@media screen and (min-width: " +window.innerHeight+ "px) {html{font-size:" + ((window.innerHeight/(designWidth/rem2px)/defaultFontSize)*100) + "%;}}"  
  13. st.innerHTML = portrait + landscape;
  14. head.appendChild(st);
  15. return defaultFontSize
  16. };
  17. var defaultFontSize = adapt(640, 100);

Let’s look back at the definition of rem, “The font size of the root element.” The root element we thought of as html actually has a shadow at work, and the font-size we thought of is actually an approximate value.

<<:  Mobile development weekly reading list: iOS multithreading safety, building Android MVVM application framework

>>:  Writing testable JavaScript code

Recommend

Practical Tips丨How to conduct a successful fission campaign?

The article starts with the wool party and discus...

Xiao Fei Da Xia Training Camp 6th

Resource introduction of the 6th Xiaofei Hero Tra...

Content operation, do you know the strategies of these four important links?

This is an era of information overload. Even very...

I have 5 creative techniques for operators to use their imagination

Creativity: New abstract thinking and behavioral ...

Advanced Software Test-System Architect Elite Class 2022 5th

Course Catalog: ├──1–Understanding and cognition ...

What does iOS 8 mean for app design?

Although the interaction design of iOS 8 is simil...

Event promotion plan, execution and implementation!

How to independently design and execute the strat...

Nanping SEO training: The website articles have been updated but not received

If the content of the article is in this situatio...