 d30ddd4c2e
			
		
	
	d30ddd4c2e
	
	
	
		
			
			* wip
* wip
* wip
* test
* wip rollup pluginでsearchIndexの情報生成
* wip
* SPDX
* wip: markerIdを自動付与
* rollupでビルド時・devモード時に毎回uuidを生成するように
* 開発サーバーでだけ必要な挙動は開発サーバーのみで
* 条件が逆
* wip: childrenの生成
* update comment
* update comment
* rename auto generated file
* hashをパスと行数から決定
* Update privacy.vue
* Update privacy.vue
* wip
* Update general.vue
* Update general.vue
* wip
* wip
* Update SearchMarker.vue
* wip
* Update profile.vue
* Update mute-block.vue
* Update mute-block.vue
* Update general.vue
* Update general.vue
* childrenがduplicate key errorを吐く問題をいったん解決
* マーカーの形を成形
* loggerを置きかえ
* とりあえず省略記法に対応
* Refactor and Format codes
* wip
* Update settings-search-index.ts
* wip
* wip
* とりあえず不確定要因の仮置きidを削除
* hashの生成を正規化(絶対パスになっていたのを緩和)
* pathの入力を省略可能に
* adminでもパス生成できるように
* Update settings-search-index.ts
* Update privacy.vue
* wip
* build searchIndex
* wip
* build
* Update general.vue
* build
* Update sounds.vue
* build
* build
* Update sounds.vue
* 🎨
* 🎨
* Update privacy.vue
* Update privacy.vue
* Update security.vue
* create-search-indexを多少改善
* build
* Update 2fa.vue
* wip
* 必ずtransformCodeCacheを利用するように, キャッシュの明確な受け渡しを定義
* キャッシュはdevServerでなくても更新
* Revert "wip"
This reverts commit 41bffd3a13.
* inlining
* wip
* Update theme.vue
* 🎨
* wip normalize
* Update theme.vue
* キャッシュのパス変換
* build
* wip
* wip
* Update SearchMarker.vue
* i18n.ts['key'] の形式が取り出せない問題のFix
* build
* 仮でpath入れ
* 必ず絶対パスが使われるように
* wip
* 🎨
* storybookビルド時はcreateSearchIndexをしない
* inliningの構造化
* format code
* Update index.vue
* wip
* wip
* 🎨
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* clean up
* wip
* wip
* wip
* Update rollup-plugin-unwind-css-module-class-name.test.ts
* Update navbar.vue
* clean up
* wip
* wip
* wip
* wip
* wip
* Update preferences-backups.vue
* Update common.ts
* Update preferences.ts
* wip
* wip
* wip
* wip
* Update MkPreferenceContainer.vue
* Update MkPreferenceContainer.vue
* Update MkPreferenceContainer.vue
* enhance: 検索で上下矢印を使用することで検索結果を移動できるように
* Update main-boot.ts
* refactor
* wip
* Update sounds.vue
* fix(frontend): PageWindowでSearchMarkerが動作するように
* enhance(frontend): SearchMarkerの点滅を一定時間で止める
* wip
* lint fix
* fix: 子要素監視が抜けていたのを修正
* アニメーションの回数はCSSで制御するように
* refactor
* enhance(frontend): 検索インデックス作成時のログを削減
* revert
* fix
* fix
* Update preferences.ts
* Update preferences.ts
* wip
* Update preferences.ts
* wip
* 🎨
* wip
* Update MkPreferenceContainer.vue
* wip
* Update preferences.ts
* wip
* Update preferences.ts
* Update preferences.ts
* wip
* wip
* Update preferences.ts
* wip
* wip
* Update preferences.ts
* Update CHANGELOG.md
* Update preferences.ts
* Update deck-store.ts
* deckStoreをdefaultStoreに統合
* wip
* defaultStore -> store
* Update profile.ts
* wip
* refactor
* wip: plugin
* plugin
* plugin
* plugin
* Update plugin.ts
* wip
* Update plugin.vue
* Update preferences.ts
* Update main-boot.ts
* wip
* fix test
* Update plugin.vue
* Update plugin.vue
* Update utility.ts
* wip
* wip
* Update utility.ts
* wip
* wip
* clean up
* Update utility.ts
---------
Co-authored-by: tai-cha <dev@taichan.site>
Co-authored-by: taichan <40626578+tai-cha@users.noreply.github.com>
Co-authored-by: kakkokari-gtyih <67428053+kakkokari-gtyih@users.noreply.github.com>
		
	
		
			
				
	
	
		
			600 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			600 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| /*
 | |
|  * SPDX-FileCopyrightText: syuilo and misskey-project
 | |
|  * SPDX-License-Identifier: AGPL-3.0-only
 | |
|  */
 | |
| 
 | |
| import { parse } from 'acorn';
 | |
| import { generate } from 'astring';
 | |
| import { describe, expect, it } from 'vitest';
 | |
| import { normalizeClass, unwindCssModuleClassName } from './rollup-plugin-unwind-css-module-class-name.js';
 | |
| import type * as estree from 'estree';
 | |
| 
 | |
| function parseExpression(code: string): estree.Expression {
 | |
| 	const program = parse(code, { ecmaVersion: 'latest', sourceType: 'module' }) as unknown as estree.Program;
 | |
| 	const statement = program.body[0] as estree.ExpressionStatement;
 | |
| 	return statement.expression;
 | |
| }
 | |
| 
 | |
| describe(normalizeClass.name, () => {
 | |
| 	it('should normalize string', () => {
 | |
| 		expect(normalizeClass(parseExpression('"a b c"'))).toBe('a b c');
 | |
| 	});
 | |
| 	it('should trim redundant spaces', () => {
 | |
| 		expect(normalizeClass(parseExpression('" a b  c "'))).toBe('a b c');
 | |
| 	});
 | |
| 	it('should ignore undefined', () => {
 | |
| 		expect(normalizeClass(parseExpression('undefined'))).toBe('');
 | |
| 	});
 | |
| 	it('should ignore non string literals', () => {
 | |
| 		expect(normalizeClass(parseExpression('0'))).toBe('');
 | |
| 		expect(normalizeClass(parseExpression('true'))).toBe('');
 | |
| 		expect(normalizeClass(parseExpression('null'))).toBe('');
 | |
| 		expect(normalizeClass(parseExpression('/I.D/'))).toBe('');
 | |
| 	});
 | |
| 	it('should not normalize identifiers', () => {
 | |
| 		expect(normalizeClass(parseExpression('EScape'))).toBeNull();
 | |
| 	});
 | |
| 	it('should normalize recursively array', () => {
 | |
| 		expect(normalizeClass(parseExpression('["from", ...["Utopia"]]'))).toBe('from Utopia');
 | |
| 		expect(normalizeClass(parseExpression('["from", ...[Utopia]]'))).toBeNull();
 | |
| 	});
 | |
| 	it('should normalize recursively template literal', () => {
 | |
| 		expect(normalizeClass(parseExpression('`name ${"shiho"} code ${33}`'))).toBe('name shiho code');
 | |
| 		expect(normalizeClass(parseExpression('`name ${shiho.name} code ${33}`'))).toBeNull();
 | |
| 	});
 | |
| 	it('should normalize recursively binary expression', () => {
 | |
| 		expect(normalizeClass(parseExpression('"mirage" + "mirror"'))).toBe('miragemirror');
 | |
| 		expect(normalizeClass(parseExpression('"mirage" + mirror'))).toBeNull();
 | |
| 	});
 | |
| 	it('should normalize recursively object expression', () => {
 | |
| 		expect(normalizeClass(parseExpression('({ a: true, b: "c" })'))).toBe('a b');
 | |
| 		expect(normalizeClass(parseExpression('({ a: false, b: "c" })'))).toBe('b');
 | |
| 		expect(normalizeClass(parseExpression('({ a: true, b: c })'))).toBeNull();
 | |
| 		expect(normalizeClass(parseExpression('({ a: true, b: "c", ...({ d: true }) })'))).toBe('a b d');
 | |
| 		expect(normalizeClass(parseExpression('({ a: true, [b]: "c" })'))).toBeNull();
 | |
| 		expect(normalizeClass(parseExpression('({ a: true, b: false, c: !false, d: !!0 })'))).toBe('a c');
 | |
| 	});
 | |
| });
 | |
| 
 | |
| it('Composition API (standard)', () => {
 | |
| 	const ast = parse(`
 | |
| import { c as api, d as store, i as i18n, aD as notePage, bN as ImgWithBlurhash, bY as getStaticImageUrl, _ as _export_sfc } from './app-!~{001}~.js';
 | |
| import { M as MkContainer } from './MkContainer-!~{03M}~.js';
 | |
| import { b as defineComponent, a as ref, e as onMounted, z as resolveComponent, g as openBlock, h as createBlock, i as withCtx, K as createTextVNode, E as toDisplayString, u as unref, l as createBaseVNode, q as normalizeClass, B as createCommentVNode, k as createElementBlock, F as Fragment, C as renderList, A as createVNode } from './vue-!~{002}~.js';
 | |
| import './photoswipe-!~{003}~.js';
 | |
| 
 | |
| const _hoisted_1 = /* @__PURE__ */ createBaseVNode("i", { class: "ti ti-photo" }, null, -1);
 | |
| const _sfc_main = /* @__PURE__ */ defineComponent({
 | |
|   __name: "index.photos",
 | |
|   props: {
 | |
|     user: {}
 | |
|   },
 | |
|   setup(__props) {
 | |
|     const props = __props;
 | |
|     let fetching = ref(true);
 | |
|     let images = ref([]);
 | |
|     function thumbnail(image) {
 | |
|       return store.state.disableShowingAnimatedImages ? getStaticImageUrl(image.url) : image.thumbnailUrl;
 | |
|     }
 | |
|     onMounted(() => {
 | |
|       const image = [
 | |
|         "image/jpeg",
 | |
|         "image/webp",
 | |
|         "image/avif",
 | |
|         "image/png",
 | |
|         "image/gif",
 | |
|         "image/apng",
 | |
|         "image/vnd.mozilla.apng"
 | |
|       ];
 | |
|       api("users/notes", {
 | |
|         userId: props.user.id,
 | |
|         fileType: image,
 | |
|         limit: 10
 | |
|       }).then((notes) => {
 | |
|         for (const note of notes) {
 | |
|           for (const file of note.files) {
 | |
|             images.value.push({
 | |
|               note,
 | |
|               file
 | |
|             });
 | |
|           }
 | |
|         }
 | |
|         fetching.value = false;
 | |
|       });
 | |
|     });
 | |
|     return (_ctx, _cache) => {
 | |
|       const _component_MkLoading = resolveComponent("MkLoading");
 | |
|       const _component_MkA = resolveComponent("MkA");
 | |
|       return openBlock(), createBlock(MkContainer, {
 | |
|         "max-height": 300,
 | |
|         foldable: true
 | |
|       }, {
 | |
|         icon: withCtx(() => [
 | |
|           _hoisted_1
 | |
|         ]),
 | |
|         header: withCtx(() => [
 | |
|           createTextVNode(toDisplayString(unref(i18n).ts.images), 1)
 | |
|         ]),
 | |
|         default: withCtx(() => [
 | |
|           createBaseVNode("div", {
 | |
|             class: normalizeClass(_ctx.$style.root)
 | |
|           }, [
 | |
|             unref(fetching) ? (openBlock(), createBlock(_component_MkLoading, { key: 0 })) : createCommentVNode("", true),
 | |
|             !unref(fetching) && unref(images).length > 0 ? (openBlock(), createElementBlock("div", {
 | |
|               key: 1,
 | |
|               class: normalizeClass(_ctx.$style.stream)
 | |
|             }, [
 | |
|               (openBlock(true), createElementBlock(Fragment, null, renderList(unref(images), (image) => {
 | |
|                 return openBlock(), createBlock(_component_MkA, {
 | |
|                   key: image.note.id + image.file.id,
 | |
|                   class: normalizeClass(_ctx.$style.img),
 | |
|                   to: unref(notePage)(image.note)
 | |
|                 }, {
 | |
|                   default: withCtx(() => [
 | |
|                     createVNode(ImgWithBlurhash, {
 | |
|                       hash: image.file.blurhash,
 | |
|                       src: thumbnail(image.file),
 | |
|                       title: image.file.name
 | |
|                     }, null, 8, ["hash", "src", "title"])
 | |
|                   ]),
 | |
|                   _: 2
 | |
|                 }, 1032, ["class", "to"]);
 | |
|               }), 128))
 | |
|             ], 2)) : createCommentVNode("", true),
 | |
|             !unref(fetching) && unref(images).length == 0 ? (openBlock(), createElementBlock("p", {
 | |
|               key: 2,
 | |
|               class: normalizeClass(_ctx.$style.empty)
 | |
|             }, toDisplayString(unref(i18n).ts.nothing), 3)) : createCommentVNode("", true)
 | |
|           ], 2)
 | |
|         ]),
 | |
|         _: 1
 | |
|       });
 | |
|     };
 | |
|   }
 | |
| });
 | |
| 
 | |
| const root = "xenMW";
 | |
| const stream = "xaZzf";
 | |
| const img = "xtA8t";
 | |
| const empty = "xhYKj";
 | |
| const style0 = {
 | |
|         root: root,
 | |
|         stream: stream,
 | |
|         img: img,
 | |
|         empty: empty
 | |
| };
 | |
| 
 | |
| const cssModules = {
 | |
|   "$style": style0
 | |
| };
 | |
| const index_photos = /* @__PURE__ */ _export_sfc(_sfc_main, [["__cssModules", cssModules]]);
 | |
| 
 | |
| export { index_photos as default };
 | |
| `.slice(1), { ecmaVersion: 'latest', sourceType: 'module' });
 | |
| 	unwindCssModuleClassName(ast);
 | |
| 	expect(generate(ast)).toBe(`
 | |
| import {c as api, d as store, i as i18n, aD as notePage, bN as ImgWithBlurhash, bY as getStaticImageUrl, _ as _export_sfc} from './app-!~{001}~.js';
 | |
| import {M as MkContainer} from './MkContainer-!~{03M}~.js';
 | |
| import {b as defineComponent, a as ref, e as onMounted, z as resolveComponent, g as openBlock, h as createBlock, i as withCtx, K as createTextVNode, E as toDisplayString, u as unref, l as createBaseVNode, q as normalizeClass, B as createCommentVNode, k as createElementBlock, F as Fragment, C as renderList, A as createVNode} from './vue-!~{002}~.js';
 | |
| import './photoswipe-!~{003}~.js';
 | |
| const _hoisted_1 = createBaseVNode("i", {
 | |
|   class: "ti ti-photo"
 | |
| }, null, -1);
 | |
| const index_photos = defineComponent({
 | |
|   __name: "index.photos",
 | |
|   props: {
 | |
|     user: {}
 | |
|   },
 | |
|   setup(__props) {
 | |
|     const props = __props;
 | |
|     let fetching = ref(true);
 | |
|     let images = ref([]);
 | |
|     function thumbnail(image) {
 | |
|       return store.state.disableShowingAnimatedImages ? getStaticImageUrl(image.url) : image.thumbnailUrl;
 | |
|     }
 | |
|     onMounted(() => {
 | |
|       const image = ["image/jpeg", "image/webp", "image/avif", "image/png", "image/gif", "image/apng", "image/vnd.mozilla.apng"];
 | |
|       api("users/notes", {
 | |
|         userId: props.user.id,
 | |
|         fileType: image,
 | |
|         limit: 10
 | |
|       }).then(notes => {
 | |
|         for (const note of notes) {
 | |
|           for (const file of note.files) {
 | |
|             images.value.push({
 | |
|               note,
 | |
|               file
 | |
|             });
 | |
|           }
 | |
|         }
 | |
|         fetching.value = false;
 | |
|       });
 | |
|     });
 | |
|     return (_ctx, _cache) => {
 | |
|       const _component_MkLoading = resolveComponent("MkLoading");
 | |
|       const _component_MkA = resolveComponent("MkA");
 | |
|       return (openBlock(), createBlock(MkContainer, {
 | |
|         "max-height": 300,
 | |
|         foldable: true
 | |
|       }, {
 | |
|         icon: withCtx(() => [_hoisted_1]),
 | |
|         header: withCtx(() => [createTextVNode(toDisplayString(unref(i18n).ts.images), 1)]),
 | |
|         default: withCtx(() => [createBaseVNode("div", {
 | |
|           class: "xenMW"
 | |
|         }, [unref(fetching) ? (openBlock(), createBlock(_component_MkLoading, {
 | |
|           key: 0
 | |
|         })) : createCommentVNode("", true), !unref(fetching) && unref(images).length > 0 ? (openBlock(), createElementBlock("div", {
 | |
|           key: 1,
 | |
|           class: "xaZzf"
 | |
|         }, [(openBlock(true), createElementBlock(Fragment, null, renderList(unref(images), image => {
 | |
|           return (openBlock(), createBlock(_component_MkA, {
 | |
|             key: image.note.id + image.file.id,
 | |
|             class: "xtA8t",
 | |
|             to: unref(notePage)(image.note)
 | |
|           }, {
 | |
|             default: withCtx(() => [createVNode(ImgWithBlurhash, {
 | |
|               hash: image.file.blurhash,
 | |
|               src: thumbnail(image.file),
 | |
|               title: image.file.name
 | |
|             }, null, 8, ["hash", "src", "title"])]),
 | |
|             _: 2
 | |
|           }, 1032, ["class", "to"]));
 | |
|         }), 128))], 2)) : createCommentVNode("", true), !unref(fetching) && unref(images).length == 0 ? (openBlock(), createElementBlock("p", {
 | |
|           key: 2,
 | |
|           class: "xhYKj"
 | |
|         }, toDisplayString(unref(i18n).ts.nothing), 3)) : createCommentVNode("", true)], 2)]),
 | |
|         _: 1
 | |
|       }));
 | |
|     };
 | |
|   }
 | |
| });
 | |
| const root = "xenMW";
 | |
| const stream = "xaZzf";
 | |
| const img = "xtA8t";
 | |
| const empty = "xhYKj";
 | |
| const style0 = {
 | |
|   root: root,
 | |
|   stream: stream,
 | |
|   img: img,
 | |
|   empty: empty
 | |
| };
 | |
| const cssModules = {
 | |
|   "$style": style0
 | |
| };
 | |
| export {index_photos as default};
 | |
| `.slice(1));
 | |
| });
 | |
| 
 | |
| it('Composition API (with `useCssModule()`)', () => {
 | |
| 	const ast = parse(`
 | |
| import { a7 as getCurrentInstance, b as defineComponent, G as useCssModule, a1 as h, H as TransitionGroup } from './!~{002}~.js';
 | |
| import { d as store, aK as toast, b5 as MkAd, i as i18n, _ as _export_sfc } from './app-!~{001}~.js';
 | |
| 
 | |
| function isDebuggerEnabled(id) {
 | |
|   try {
 | |
|     return localStorage.getItem(\`DEBUG_\${id}\`) !== null;
 | |
|   } catch {
 | |
|     return false;
 | |
|   }
 | |
| }
 | |
| function stackTraceInstances() {
 | |
|   let instance = getCurrentInstance();
 | |
|   const stack = [];
 | |
|   while (instance) {
 | |
|     stack.push(instance);
 | |
|     instance = instance.parent;
 | |
|   }
 | |
|   return stack;
 | |
| }
 | |
| 
 | |
| const _sfc_main = defineComponent({
 | |
|   props: {
 | |
|     items: {
 | |
|       type: Array,
 | |
|       required: true
 | |
|     },
 | |
|     direction: {
 | |
|       type: String,
 | |
|       required: false,
 | |
|       default: "down"
 | |
|     },
 | |
|     reversed: {
 | |
|       type: Boolean,
 | |
|       required: false,
 | |
|       default: false
 | |
|     },
 | |
|     noGap: {
 | |
|       type: Boolean,
 | |
|       required: false,
 | |
|       default: false
 | |
|     },
 | |
|     ad: {
 | |
|       type: Boolean,
 | |
|       required: false,
 | |
|       default: false
 | |
|     }
 | |
|   },
 | |
|   setup(props, { slots, expose }) {
 | |
|     const $style = useCssModule();
 | |
|     function getDateText(time) {
 | |
|       const date = new Date(time).getDate();
 | |
|       const month = new Date(time).getMonth() + 1;
 | |
|       return i18n.t("monthAndDay", {
 | |
|         month: month.toString(),
 | |
|         day: date.toString()
 | |
|       });
 | |
|     }
 | |
|     if (props.items.length === 0)
 | |
|       return;
 | |
|     const renderChildrenImpl = () => props.items.map((item, i) => {
 | |
|       if (!slots || !slots.default)
 | |
|         return;
 | |
|       const el = slots.default({
 | |
|         item
 | |
|       })[0];
 | |
|       if (el.key == null && item.id)
 | |
|         el.key = item.id;
 | |
|       if (i !== props.items.length - 1 && new Date(item.createdAt).getDate() !== new Date(props.items[i + 1].createdAt).getDate()) {
 | |
|         const separator = h("div", {
 | |
|           class: $style["separator"],
 | |
|           key: item.id + ":separator"
 | |
|         }, h("p", {
 | |
|           class: $style["date"]
 | |
|         }, [
 | |
|           h("span", {
 | |
|             class: $style["date-1"]
 | |
|           }, [
 | |
|             h("i", {
 | |
|               class: \`ti ti-chevron-up \${$style["date-1-icon"]}\`
 | |
|             }),
 | |
|             getDateText(item.createdAt)
 | |
|           ]),
 | |
|           h("span", {
 | |
|             class: $style["date-2"]
 | |
|           }, [
 | |
|             getDateText(props.items[i + 1].createdAt),
 | |
|             h("i", {
 | |
|               class: \`ti ti-chevron-down \${$style["date-2-icon"]}\`
 | |
|             })
 | |
|           ])
 | |
|         ]));
 | |
|         return [el, separator];
 | |
|       } else {
 | |
|         if (props.ad && item._shouldInsertAd_) {
 | |
|           return [h(MkAd, {
 | |
|             key: item.id + ":ad",
 | |
|             prefer: ["horizontal", "horizontal-big"]
 | |
|           }), el];
 | |
|         } else {
 | |
|           return el;
 | |
|         }
 | |
|       }
 | |
|     });
 | |
|     const renderChildren = () => {
 | |
|       const children = renderChildrenImpl();
 | |
|       if (isDebuggerEnabled(6864)) {
 | |
|         const nodes = children.flatMap((node) => node ?? []);
 | |
|         const keys = new Set(nodes.map((node) => node.key));
 | |
|         if (keys.size !== nodes.length) {
 | |
|           const id = crypto.randomUUID();
 | |
|           const instances = stackTraceInstances();
 | |
|           toast(instances.reduce((a, c) => \`\${a} at \${c.type.name}\`, \`[DEBUG_6864 (\${id})]: \${nodes.length - keys.size} duplicated keys found\`));
 | |
|           console.warn({ id, debugId: 6864, stack: instances });
 | |
|         }
 | |
|       }
 | |
|       return children;
 | |
|     };
 | |
|     function onBeforeLeave(el) {
 | |
|       el.style.top = \`\${el.offsetTop}px\`;
 | |
|       el.style.left = \`\${el.offsetLeft}px\`;
 | |
|     }
 | |
|     function onLeaveCanceled(el) {
 | |
|       el.style.top = "";
 | |
|       el.style.left = "";
 | |
|     }
 | |
|     return () => h(
 | |
|       prefer.s.animation ? TransitionGroup : "div",
 | |
|       {
 | |
|         class: {
 | |
|           [$style["date-separated-list"]]: true,
 | |
|           [$style["date-separated-list-nogap"]]: props.noGap,
 | |
|           [$style["reversed"]]: props.reversed,
 | |
|           [$style["direction-down"]]: props.direction === "down",
 | |
|           [$style["direction-up"]]: props.direction === "up"
 | |
|         },
 | |
|         ...prefer.s.animation ? {
 | |
|           name: "list",
 | |
|           tag: "div",
 | |
|           onBeforeLeave,
 | |
|           onLeaveCanceled
 | |
|         } : {}
 | |
|       },
 | |
|       { default: renderChildren }
 | |
|     );
 | |
|   }
 | |
| });
 | |
| 
 | |
| const reversed = "xxiZh";
 | |
| const separator = "xxeDx";
 | |
| const date = "xxawD";
 | |
| const style0 = {
 | |
|         "date-separated-list": "xfKPa",
 | |
|         "date-separated-list-nogap": "xf9zr",
 | |
|         "direction-up": "x7AeO",
 | |
|         "direction-down": "xBIqc",
 | |
|         reversed: reversed,
 | |
|         separator: separator,
 | |
|         date: date,
 | |
|         "date-1": "xwtmh",
 | |
|         "date-1-icon": "xsNPa",
 | |
|         "date-2": "x1xvw",
 | |
|         "date-2-icon": "x9ZiG"
 | |
| };
 | |
| 
 | |
| const cssModules = {
 | |
|   "$style": style0
 | |
| };
 | |
| const MkDateSeparatedList = /* @__PURE__ */ _export_sfc(_sfc_main, [["__cssModules", cssModules]]);
 | |
| 
 | |
| export { MkDateSeparatedList as M };
 | |
| `.slice(1), { ecmaVersion: 'latest', sourceType: 'module' });
 | |
| 	unwindCssModuleClassName(ast);
 | |
| 	expect(generate(ast)).toBe(`
 | |
| import {a7 as getCurrentInstance, b as defineComponent, G as useCssModule, a1 as h, H as TransitionGroup} from './!~{002}~.js';
 | |
| import {d as store, aK as toast, b5 as MkAd, i as i18n, _ as _export_sfc} from './app-!~{001}~.js';
 | |
| function isDebuggerEnabled(id) {
 | |
|   try {
 | |
|     return localStorage.getItem(\`DEBUG_\${id}\`) !== null;
 | |
|   } catch {
 | |
|     return false;
 | |
|   }
 | |
| }
 | |
| function stackTraceInstances() {
 | |
|   let instance = getCurrentInstance();
 | |
|   const stack = [];
 | |
|   while (instance) {
 | |
|     stack.push(instance);
 | |
|     instance = instance.parent;
 | |
|   }
 | |
|   return stack;
 | |
| }
 | |
| const _sfc_main = defineComponent({
 | |
|   props: {
 | |
|     items: {
 | |
|       type: Array,
 | |
|       required: true
 | |
|     },
 | |
|     direction: {
 | |
|       type: String,
 | |
|       required: false,
 | |
|       default: "down"
 | |
|     },
 | |
|     reversed: {
 | |
|       type: Boolean,
 | |
|       required: false,
 | |
|       default: false
 | |
|     },
 | |
|     noGap: {
 | |
|       type: Boolean,
 | |
|       required: false,
 | |
|       default: false
 | |
|     },
 | |
|     ad: {
 | |
|       type: Boolean,
 | |
|       required: false,
 | |
|       default: false
 | |
|     }
 | |
|   },
 | |
|   setup(props, {slots, expose}) {
 | |
|     const $style = useCssModule();
 | |
|     function getDateText(time) {
 | |
|       const date = new Date(time).getDate();
 | |
|       const month = new Date(time).getMonth() + 1;
 | |
|       return i18n.t("monthAndDay", {
 | |
|         month: month.toString(),
 | |
|         day: date.toString()
 | |
|       });
 | |
|     }
 | |
|     if (props.items.length === 0) return;
 | |
|     const renderChildrenImpl = () => props.items.map((item, i) => {
 | |
|       if (!slots || !slots.default) return;
 | |
|       const el = slots.default({
 | |
|         item
 | |
|       })[0];
 | |
|       if (el.key == null && item.id) el.key = item.id;
 | |
|       if (i !== props.items.length - 1 && new Date(item.createdAt).getDate() !== new Date(props.items[i + 1].createdAt).getDate()) {
 | |
|         const separator = h("div", {
 | |
|           class: $style["separator"],
 | |
|           key: item.id + ":separator"
 | |
|         }, h("p", {
 | |
|           class: $style["date"]
 | |
|         }, [h("span", {
 | |
|           class: $style["date-1"]
 | |
|         }, [h("i", {
 | |
|           class: \`ti ti-chevron-up \${$style["date-1-icon"]}\`
 | |
|         }), getDateText(item.createdAt)]), h("span", {
 | |
|           class: $style["date-2"]
 | |
|         }, [getDateText(props.items[i + 1].createdAt), h("i", {
 | |
|           class: \`ti ti-chevron-down \${$style["date-2-icon"]}\`
 | |
|         })])]));
 | |
|         return [el, separator];
 | |
|       } else {
 | |
|         if (props.ad && item._shouldInsertAd_) {
 | |
|           return [h(MkAd, {
 | |
|             key: item.id + ":ad",
 | |
|             prefer: ["horizontal", "horizontal-big"]
 | |
|           }), el];
 | |
|         } else {
 | |
|           return el;
 | |
|         }
 | |
|       }
 | |
|     });
 | |
|     const renderChildren = () => {
 | |
|       const children = renderChildrenImpl();
 | |
|       if (isDebuggerEnabled(6864)) {
 | |
|         const nodes = children.flatMap(node => node ?? []);
 | |
|         const keys = new Set(nodes.map(node => node.key));
 | |
|         if (keys.size !== nodes.length) {
 | |
|           const id = crypto.randomUUID();
 | |
|           const instances = stackTraceInstances();
 | |
|           toast(instances.reduce((a, c) => \`\${a} at \${c.type.name}\`, \`[DEBUG_6864 (\${id})]: \${nodes.length - keys.size} duplicated keys found\`));
 | |
|           console.warn({
 | |
|             id,
 | |
|             debugId: 6864,
 | |
|             stack: instances
 | |
|           });
 | |
|         }
 | |
|       }
 | |
|       return children;
 | |
|     };
 | |
|     function onBeforeLeave(el) {
 | |
|       el.style.top = \`\${el.offsetTop}px\`;
 | |
|       el.style.left = \`\${el.offsetLeft}px\`;
 | |
|     }
 | |
|     function onLeaveCanceled(el) {
 | |
|       el.style.top = "";
 | |
|       el.style.left = "";
 | |
|     }
 | |
|     return () => h(prefer.s.animation ? TransitionGroup : "div", {
 | |
|       class: {
 | |
|         [$style["date-separated-list"]]: true,
 | |
|         [$style["date-separated-list-nogap"]]: props.noGap,
 | |
|         [$style["reversed"]]: props.reversed,
 | |
|         [$style["direction-down"]]: props.direction === "down",
 | |
|         [$style["direction-up"]]: props.direction === "up"
 | |
|       },
 | |
|       ...prefer.s.animation ? {
 | |
|         name: "list",
 | |
|         tag: "div",
 | |
|         onBeforeLeave,
 | |
|         onLeaveCanceled
 | |
|       } : {}
 | |
|     }, {
 | |
|       default: renderChildren
 | |
|     });
 | |
|   }
 | |
| });
 | |
| const reversed = "xxiZh";
 | |
| const separator = "xxeDx";
 | |
| const date = "xxawD";
 | |
| const style0 = {
 | |
|   "date-separated-list": "xfKPa",
 | |
|   "date-separated-list-nogap": "xf9zr",
 | |
|   "direction-up": "x7AeO",
 | |
|   "direction-down": "xBIqc",
 | |
|   reversed: reversed,
 | |
|   separator: separator,
 | |
|   date: date,
 | |
|   "date-1": "xwtmh",
 | |
|   "date-1-icon": "xsNPa",
 | |
|   "date-2": "x1xvw",
 | |
|   "date-2-icon": "x9ZiG"
 | |
| };
 | |
| const cssModules = {
 | |
|   "$style": style0
 | |
| };
 | |
| const MkDateSeparatedList = _export_sfc(_sfc_main, [["__cssModules", cssModules]]);
 | |
| export {MkDateSeparatedList as M};
 | |
| `.slice(1));
 | |
| });
 |