CSSでモーダルダイアログの背景をスクロールさせないようにできるかもしれない ― 2022年12月01日 01時41分
この記事はCSS Advent Calendar 2022の1日目の分です。
HTMLのdialog
要素を使うとモーダルダイアログを表現できます(使い方によってはモードレスダイアログも表現できます)。ただし、そのままだとモーダルダイアログを開いているときに、マウスホイールなどによってダイアログの背景(文書全体)までスクロールしてしまいます。
モーダルダイアログの背景をスクロールさせたくない場合、これを書いている現在のCSS仕様草案によれば、以下の記述で実現できるはずです(デモ)。
dialog {
overscroll-behavior: contain;
}
しかしながら、この方法はChrome canary 110では期待通り動作しますがマウスホイールによるスクロールは防げますが、矢印キーやPageUp/PageDownキーによるスクロールは防げず、Firefox nightly 109では動作しません。
このあたりの事情はちょっと複雑で、
- CSS仕様草案によれば、
overflow: auto
な要素においてはスクロール可能な領域がないときもoverscroll-behavior
プロパティが適用されるはず。- CSS Overscroll Behaviorモジュール草案によれば、スクロールコンテナに対して
overscroll-behavior
プロパティが適用される。 - CSS Overflowモジュール草案によれば、
overflow: auto
な要素(およびoverflow: hidden
な要素)はスクロール可能な領域がないときもスクロールコンテナである。
- CSS Overscroll Behaviorモジュール草案によれば、スクロールコンテナに対して
- 各ブラウザ(少なくともChromeの場合とFirefoxの場合)は、あえて仕様草案を無視し、スクロール可能な領域がないときに
overscroll-behavior
プロパティを適用しない。- しかし、Chromeはなぜか、
dialog
要素においてはスクロール可能な領域がないときもマウスホイールによるスクロールに関してはoverscroll-behavior
プロパティを適用する模様(そうなっている経緯は未調査)。
- しかし、Chromeはなぜか、
- CSS仕様草案に対して、スクロール可能な領域がないときは
overscroll-behavior
プロパティが適用されないようにしよう(仕様側を現状のブラウザ実装に合わせよう)という提案がなされている。
となっています。上述のCSSコードが将来的にも機能するかどうかは不透明です。
overflow: hidden
を使ってスクロールを防止する方法でも、:has
疑似クラスと組み合わせることで、CSSのみでモーダルダイアログの背景のスクロールを防止できるそうです。
Perlで配列の先頭何要素かを抜き出す ― 2022年12月02日 02時16分
この記事はPerl Advent Calendar 2022の2日目の分です。
Perlで配列の先頭n要素を抜き出したいとき、最近はList::Util
モジュールのhead
関数を使えます。
use List::Util qw(head);
my @array1 = qw(a b c d e);
my @array2 = head 3, @array1;
# @array2の内容は('a', 'b', 'c')
List::Util
モジュールはコアモジュール(Perl本体と一緒にインストールされるモジュール)であり、Perl 5.28以降なら追加のモジュールインストールなしにhead
関数を使えます。それより古いPerlでは、List::Util
の新しいバージョン(1.50以降)をインストールする必要があります。
以前からある方法
配列スライスを使うこともできますが、抜き出す要素数から1引いた値を指定することになって、ちょっと紛らわしいです。
my @array1 = qw(a b c d e);
my @array2 = @array1[0 .. 2];
# @array2の内容は('a', 'b', 'c')
また、元の配列の要素数が抜き出す要素数より少ないときは、不足分がundef
で埋められてしまいます。
my @array1 = qw(a);
my @array2 = @array1[0 .. 2];
# @array2の内容は('a', undef, undef)
undef
で埋められたくなければ、min
関数を使うなどひと工夫する必要があります。
use List::Util qw(min);
my @array1 = qw(a);
my @array2 = @array1[0 .. min(2, $#array1)];
# @array2の内容は('a')
splice
関数を使うこともできますが、元の配列も変更されてしまいます。
my @array1 = qw(a b c d e);
my @array2 = splice @array1, 0, 3;
# @array1の内容は('d', 'e')
# @array2の内容は('a', 'b', 'c')
TypeScriptでイベントをPromise化する関数の型を定義したい ― 2022年12月03日 10時41分
この記事はTypeScript Advent Calendar 2022の3日目の分です。
「addEventListenerでリッスンしているイベントをPromise化する」という記事で、イベントをPromise
で受け取る関数が紹介されています。Node.jsのevents
モジュールのevents.once
メソッドと同じ機能を実現するものですね。Webブラウザ組み込みのDOMでも同じ機能を提供しようという提案もなされています。
最初の記事では余談として今回紹介したeventPromisifyはTypeScriptで書こうとすると型の定義が難しいなと思いました
と書かれています。例えばeventPromisify(document, 'click')
と呼び出したら返り値の型がPromise<MouseEvent>
になってほしいのですが、そのような型定義を記述できるでしょうか? (以下、TypeScript 4.9を想定しています。)
イベントが発生する対象(target
)とイベント名(type
)を決め打ちできるのなら、
type EventForDocumentClick =
typeof document.addEventListener<'click'> extends
(type: 'click', listener: (event: infer E) => void) => void
? E : never;
// = MouseEvent
のように、document
と'click'
からMouseEvent
を導出できます。しかし、変数document
ではなくDocument
型だけが与えられているとき、Document['addEventListener']<'click'>
のように型引数を指定することはできません。
またDocument
型におけるaddEventListener
メソッドの定義は、イベント名がDocument
固有(keyof DocumentEventMap
型)のものとイベント名が文字列全般(string
型)のものがオーバーロードされています。型引数が絡んでいなければ、「オーバーロードされた関数型から引数の型や返り値の型を取り出す方法」に書かれているように型を取り出せます。しかし、型引数を持つメソッドがオーバーロードされているときに、意図した型引数が指定された場合の引数や返り値の型を取り出せるのかどうか、私にはわかりませんでした。
結局私にできたのは以下の状態までです(TypeScript Playgroundで確認)。
type EventTypeFor<Target extends EventTarget> = Target['addEventListener'] extends {
(type: infer T, listener: (e: Event) => void): void;
(type: string, listener: EventListenerOrEventListenerObject): void;
} ? T : never;
type EventFor<Target extends EventTarget> = Target['addEventListener'] extends {
(type: string, listener: (e: infer E) => void): void;
(type: string, listener: EventListenerOrEventListenerObject): void;
} ? E : never;
const eventPromisify = <T extends EventTarget>(
target: T,
type: EventTypeFor<T>
): Promise<EventFor<T>> => {
throw new Error('Not implemented');
}
const p = eventPromisify(document, 'click');
// p: Promise<Event | MouseEvent | UIEvent | ClipboardEvent | AnimationEvent | InputEvent | FocusEvent | ... 11 more ... | WheelEvent>
第2引数をDocument
固有のイベント名に限定することはできています。しかしながら、返り値のPromise
の値の型をMouseEvent
に限定することはできず、Document
固有のイベントすべての共用体型となってしまいます。
Perlで配列の先頭何要素か以外を抜き出す ― 2022年12月05日 01時47分
この記事はPerl Advent Calendar 2022の5日目の分です。
Perlで配列の先頭n要素以外を抜き出したい——例えば配列('a', 'b', 'c', 'd', 'e')
から先頭2要素以外を抜き出して配列('c', 'd', 'e')
を得たい——とき、最近はList::Util
モジュールのtail
関数を使えます。
tail
関数は配列の末尾n要素を抜き出す関数ですが、抜き出す要素数として負数-mを指定すると、先頭m要素以外の要素を返します。
use List::Util qw(tail);
my @array1 = qw(a b c d e);
my @array2 = tail -2, @array1;
# @array2の内容は('c', 'd', 'e')
List::Util
モジュールはコアモジュール(Perl本体と一緒にインストールされるモジュール)であり、Perl 5.28以降なら追加のモジュールインストールなしにtail
関数を使えます。それより古いPerlでは、List::Util
の新しいバージョン(1.50以降)をインストールする必要があります。
以前からある方法として、配列スライスを使うこともできます。
my @array1 = qw(a b c d e);
my @array2 = @array1[2 .. $#array1];
# @array2の内容は('c', 'd', 'e')
CSSでスクロールバーの有無によるがたつきをなくす ― 2022年12月09日 09時42分
この記事はCSS Advent Calendar 2022の9日目の分です。
CSSのボックスモデルにおいては、ボーダーの内辺とパディングの外辺の間にスクロールバーが配置されます。最近はどのOSでもオーバーレイスクロールバー(スクロールバーが内容の前面に覆いかぶさるようなもの)が主流となり、スクロールバーが存在してもしなくても内容の幅が変わらないようになっています。一方、クラシックスクロールバー(スクロールバーが常に表示されるようなもの)が使われる環境では、overflow: auto
な要素において内容がはみ出すときとはみ出さないときで内容の幅が変わってきます。
クラシックスクロールバーが使われる環境でも内容の幅を一定に保ちたいという場合は、scrollbar-gutter
プロパティを使います。scrollbar-gutter: stable
を指定すれば、スクロールバーが表示されないときでもスクロールバーと同じだけの領域が確保され、内容の幅はその分狭くなります。左右中央に配置したいのにスクロールバーの領域の分だけずれて困るというときは、scrollbar-gutter: stable both-edges
を指定することで、左右どちらにもスクロールバーと同じだけの領域が確保されます。
scrollbar-gutter
プロパティはoverflow: hidden
な要素にも適用されますが、overflow: visible
な要素には適用されません。また、scrollbar-gutter
プロパティの効果は、縦スクロールバー(縦書きなら横スクロールバー)にのみ影響します。
オーバーレイスクロールバーが使われる環境では、scrollbar-gutter: stable
の効果はありません。
使用事例
使用事例としては、@about_hiroppyさんの紹介している「HTMLのdialog
要素によるモーダルダイアログが開いているときに、背景の文書をスクロールさせない」というのが考えられます(scrollbar-gutter
プロパティを指定した例と指定しなかった例)。
html {
scrollbar-gutter: stable;
}
html:has(dialog:modal) {
overflow: hidden;
}
クラシックスクロールバーが使われる環境では、文書をスクロールさせないためにoverflow: hidden
を指定すると、スクロールバーが消えて文書の内容ががたついてしまいます。scrollbar-gutter: stable
をあらかじめ指定しておけば、スクロールバーが消えてもその分の領域が確保され続けるので、文書の内容ががたつきません。
最近のコメント