yumechi-no-kuni/packages/client/src/directives/get-size.ts

56 lines
1.3 KiB
TypeScript
Raw Normal View History

2021-10-08 08:03:06 -05:00
import { Directive } from 'vue';
const mountings = new Map<Element, {
resize: ResizeObserver;
intersection?: IntersectionObserver;
fn: (w: number, h: number) => void;
}>();
function calc(src: Element) {
const info = mountings.get(src);
const height = src.clientHeight;
const width = src.clientWidth;
if (!info) return;
// アクティベート前などでsrcが描画されていない場合
if (!height) {
// IntersectionObserverで表示検出する
if (!info.intersection) {
info.intersection = new IntersectionObserver(entries => {
if (entries.some(entry => entry.isIntersecting)) calc(src);
});
}
info.intersection.observe(src);
return;
}
if (info.intersection) {
info.intersection.disconnect()
delete info.intersection;
};
2021-10-08 08:03:06 -05:00
info.fn(width, height);
};
2021-10-08 08:03:06 -05:00
export default {
mounted(src, binding, vn) {
2021-10-08 08:03:06 -05:00
const resize = new ResizeObserver((entries, observer) => {
calc(src);
2021-10-08 08:03:06 -05:00
});
resize.observe(src);
2021-10-08 08:03:06 -05:00
mountings.set(src, { resize, fn: binding.value, });
calc(src);
2021-10-08 08:03:06 -05:00
},
unmounted(src, binding, vn) {
binding.value(0, 0);
const info = mountings.get(src);
if (!info) return;
info.resize.disconnect();
if (info.intersection) info.intersection.disconnect();
mountings.delete(src);
2021-10-08 08:03:06 -05:00
}
} as Directive<Element, (w: number, h: number) => void>;