什么是 dp (密度无关像素)与 CSS 单位?

对于 Android,人们建议对 UI 元素使用 dp (密度独立像素)度量,并且存在一些约定,比如对按钮高度使用 48dp,等等。

我正在开发一个网络应用程序,我收到了很多关于 UI 设计的批评,说它不符合 Android 的设计标准。显然,我的应用程序看起来会有所不同,因为它使用的是 CSS 和 HTML 而不是 Android Holo 主题,但我仍然希望尽可能地使它符合规范。然而 CSS 不允许密度独立的测量。

当我用不同的分辨率和像素密度测试我的应用程序时,它看起来并不好,有时候,它的比例过大,甚至不能正常工作。CSS 不像 Android 本地开发那样拥有 dp 单元,但我想知道有哪些替代方案。

我是否可以用 Javascript 获得像素密度,然后适当地手动缩放所有东西?什么是最好的方法来制作一个网络应用程序,外观和工作良好的所有分辨率/密度?

82386 次浏览

http://www.w3.org/TR/css3-values/#lengths

The closest unit available in CSS are the viewport-percentage units.

  • vw - Equal to 1% of the width of the initial containing block.
  • vh - Equal to 1% of the height of the initial containing block.
  • vmin - Equal to the smaller of vw or vh.
  • vmax - Equal to the larger of vw or vh.

The only mobile browser to be aware of that doesn't support these units is Opera. http://caniuse.com/#feat=viewport-units

What about a interface based on rem units?

I'm not experienced enough on the matter to confirm, but I think you could make some math based on the viewport size to apply a font-size in the root element and the entire interface would adjust accordingly. This stuff is a bit of witchcraft to me yet.

In CSS3 it may be more accurate to say that the web doesn't have Android's px. The spec for CSS3's px says this:

pixels; 1px is equal to 1/96th of 1in

px might be the measurement you want, in the future.

I disagree with the currently accepted answer. As uber5001 suggests, a px is a fixed unit, and in a similar spirit to the efforts of the Android-specifc dp.

Per Material's spec:

When writing CSS, use px wherever dp or sp is stated. Dp only needs to be used in developing for Android.

Additionally

When designing for the web, replace dp with px (for pixel).

Use rem.

It is the font size of the root element and a very good base unit for the size of other UI elements.

If one used the same absolute size (in centimeter), the text and other elements would be much too big on mobile or much too small on desktop.

If one used the same amount of pixels, the text and other elements would be much too small on mobile or much too big on desktop.

The rem unit is on spot because it says "Hey, this is how big normal text should be." Basing the size of other UI elements on this is a pretty reasonable choice.

Why not use points, it is a consistent size across devices. And is relative to the screen size. Most are not familiar with it since it comes from traditional publishing.

As of CSS3, there are no CSS units that are truly device-independent. See the W3C spec on absolute lengths. In particular, the absolute units might not match their physical measurements. From the spec:

Note: If the anchor unit is the pixel unit, the physical units might not match their physical measurements. Alternatively if the anchor unit is a physical unit, the pixel unit might not map to a whole number of device pixels.

Note: This definition of the pixel unit and the physical units differs from previous versions of CSS. In particular, in previous versions of CSS the pixel unit and the physical units were not related by a fixed ratio: the physical units were always tied to their physical measurements while the pixel unit would vary to most closely match the reference pixel. (This change was made because too much existing content relies on the assumption of 96dpi, and breaking that assumption broke the content.)

If physical units were true to their purpose, you could use something like points; points are close enough to dps:

1 in = 72 pt
1 in = 160 dp


1 dp = 72 / 160 pt

If you use SCSS, you can write a function to return in pts:

@function dp($_dp) {
@return (72 / 160) * $_dp + pt;
}

And use it:

.shadow-2 {
height: dp(2);
}

What about use some mix between vw ( screen's width for any device) and em? Here font-size is changing dynamically, depending your device screen width. Use in your main CSS file next rule:

font-size: calc(100vw * 10 / 375);

(width for iPhone 6/7/8 is 375px, change it to 320px (iPhone5) and etc)

In all cases it is going work perfect. There is only one minus. If your goal is "pixel perfect", then you need to use big numbers after dot.

Example: your goal is font-size h5: 18px on all screens.

Then your perfect "em" will be:

h5 {
font-size: 1.79904em;
}

without perfection use 18 / 10:

h5 {
font-size: 1.8em;
}

According to Google Chrome you get 18.0096px;