DOM FAQ
- FOUC - flash of unstyled content
- 应用 style 时
 
- 参考
- Comparison of Event Targets
size
- width, height
- clientWidth, clientHeight
- offsetWidth, offsetHeight
- size+border+padding
 
- scrollWidth, scrollHeight
- scrollbarWidth = offsetWidth - clientWidth - getComputedStyle().borderLeftWidth - getComputedStyle().borderRightWidth
- scrollbarWidth = getComputedStyle().width + getComputedStyle().paddingLeft + getComputedStyle().paddingRight - clientWidth- 在 chrome 下可能不准确
 
 
- element.getBoundingClientRect()
- naturalWidth, naturalHeight
- 图片的原始大小
 
ShadowRoot rem & font-size
- 无法覆盖
- 导致 rem 受外部影响 - html font-size
- 可考虑使用 em
- 参考
ShadowRoot.mode
- open
- 记录 element.shadowRoot
- 受外部 style 影响
 
- closed
- 不会记录 root
- 如有需要需要自己使用 WeakMap 跟踪引用
 
shadow dom 重置 host 样式
:host {
  all: initial;
}
初始化 style
// 新的方式 - 2019, Chrome 73+
var sheet = new CSSStyleSheet();
sheet.replaceSync(`.color { color: pink }`);
host.shadowRoot.adoptedStyleSheets = [sheet];
// 旧的方式
let style = document.createElement('style');
style.textContent = css;
container.appendChild(style);
a 的 download 不生效
- 如果 HTTP 有 Content-Disposition 头 则优先
- 非 same-origin download 属性无效
可以选择预先下载 base64 然后下载
<!-- same orgin -->
<a href="/wp-content/uploads/file.mp4" download="file.mp4">
  <!-- pre-download -->
  <a download href="data:application/octet-stream;base64,PD94ANDSOON">Download Me</a></a
>
key vs code
- key - 实际输入内容
- 用于文本输入
- 受键盘布局影响
 
- code - 输入的物理布局
- 用于操作控制
- 不受键盘布局影响
 
- 参考
| key | code | 
|---|---|
| a | KeyA | 
| A | KeyA | 
| 1 | Digit1 | 
| ! | Digit1 | 
| - | Minus | 
| Enter | Enter | 
| Shift | ShiftLeft | 
字体检测
document.fonts.check('12px ui-serif');
- Safari 因为隐私原因,不支持,返回错误结果
监听 URL 变化
- 目前无法可靠的检测 url 变化
- patch history 的方式 https://github.com/streamich/react-use/blob/master/src/useLocation.ts
- observe 任何修改然后检测
- Chrome 102+ window.navigation 接口
let last = document.location.href;
const observer = new MutationObserver(function (mutations) {
  mutations.forEach(function (mutation) {
    if (last !== document.location.href) {
      last = document.location.href;
      /* Changed */
    }
  });
});
observer.observe(document.querySelector('body'), {
  childList: true,
  subtree: true,
});
// Chrome 102+
navigation.addEventListener('navigate', (e) => {
  console.log(`navigate ->`, e.destination.url);
});
The target origin provided does not match the recipient window's origin
idle
- IdleDetector
- API
- 需要权限
 
- Idle Detection
- Chromium 94+
- https://stackoverflow.com/a/10126042/1870054
- 通过事件检测
 
tabIndex
- tabIndex=0
- Tabbable and focusable
 
- tabIndex=-1
- Not tabbable, but focusable
 
- :focus-visible
- :focus-within
- document.activeElement
- 当前获得焦点的元素
 
document.addEventListener('focusin', (event) => {
    console.log('focusin:', event.target);
});
document.addEventListener('focusout', (event) => {
    console.log('focusout:', event.target);
});
HTML attributes vs DOM properties
- HTML attributes
- 可以序列化 - 在 HTML 里能表现出来
- 所有类型都是 string - 因为序列化
- 大小写无关
- 访问方式不同
- div.getAttributeNames()
- div.getAttribute('id')
 
- Reflection - property 可能会映射为 attribute
- crossOrigin->- crossorigin
- ariaLabel->- aria-label
- className->- class
- htmlFor->- for
- ⚠️ 注意 defaultValue->value- valueproperty 没有对应的 attribute
 
 
 
- DOM properties
- 如果一个 property 反映一个 attribute,那么 attribute 为 property 的 source 值
- 也就是说以 attribute 为准
 
- 有初始值和校验逻辑
 
- 如果一个 property 反映一个 attribute,那么 attribute 为 property 的 source 值
IME
- 事件
- compositionstart
- compositionupdate
- compositionend