2024-06-25 20:18:14 +09:00
|
|
|
|
/*
|
|
|
|
|
|
* SPDX-FileCopyrightText: syuilo and misskey-project
|
|
|
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
*/
|
|
|
|
|
|
import { v4 as uuid } from 'uuid';
|
|
|
|
|
|
import { url } from '@/config.js';
|
|
|
|
|
|
import { MOBILE_THRESHOLD } from '@/const.js';
|
|
|
|
|
|
import * as os from '@/os.js';
|
|
|
|
|
|
import copy from '@/scripts/copy-to-clipboard.js';
|
|
|
|
|
|
import MkEmbedCodeGenDialog from '@/components/MkEmbedCodeGenDialog.vue';
|
|
|
|
|
|
|
|
|
|
|
|
// 埋め込みの対象となるエンティティ(/embed/xxx の xxx の部分と対応させる)
|
|
|
|
|
|
const embeddableEntities = [
|
|
|
|
|
|
'notes',
|
|
|
|
|
|
'user-timeline',
|
2024-06-26 00:32:26 +09:00
|
|
|
|
'clips',
|
|
|
|
|
|
'tags',
|
2024-06-25 20:18:14 +09:00
|
|
|
|
] as const;
|
|
|
|
|
|
|
|
|
|
|
|
export type EmbeddableEntity = typeof embeddableEntities[number];
|
|
|
|
|
|
|
|
|
|
|
|
// 内部でスクロールがあるページ
|
|
|
|
|
|
export const embedRouteWithScrollbar: EmbeddableEntity[] = [
|
2024-06-26 00:32:26 +09:00
|
|
|
|
'clips',
|
|
|
|
|
|
'tags',
|
2024-06-25 20:18:14 +09:00
|
|
|
|
'user-timeline'
|
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
|
|
export type EmbedParams = {
|
|
|
|
|
|
maxHeight?: number;
|
|
|
|
|
|
colorMode?: 'light' | 'dark';
|
|
|
|
|
|
rounded?: boolean;
|
|
|
|
|
|
border?: boolean;
|
|
|
|
|
|
autoload?: boolean;
|
|
|
|
|
|
header?: boolean;
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
export function normalizeEmbedParams(params: EmbedParams): Record<string, string> {
|
|
|
|
|
|
// paramsのvalueをすべてstringに変換。undefinedやnullはプロパティごと消す
|
|
|
|
|
|
const normalizedParams: Record<string, string> = {};
|
|
|
|
|
|
for (const key in params) {
|
|
|
|
|
|
if (params[key] == null) {
|
|
|
|
|
|
continue;
|
|
|
|
|
|
}
|
|
|
|
|
|
switch (typeof params[key]) {
|
|
|
|
|
|
case 'number':
|
|
|
|
|
|
normalizedParams[key] = params[key].toString();
|
|
|
|
|
|
break;
|
|
|
|
|
|
case 'boolean':
|
|
|
|
|
|
normalizedParams[key] = params[key] ? 'true' : 'false';
|
|
|
|
|
|
break;
|
|
|
|
|
|
default:
|
|
|
|
|
|
normalizedParams[key] = params[key];
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return normalizedParams;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* 埋め込みコードを生成(iframe IDの発番もやる)
|
|
|
|
|
|
*/
|
|
|
|
|
|
export function getEmbedCode(path: string, params?: EmbedParams): string {
|
|
|
|
|
|
const iframeId = 'v1_' + uuid(); // 将来embed.jsのバージョンが上がったとき用にv1_を付けておく
|
|
|
|
|
|
|
|
|
|
|
|
let paramString = '';
|
|
|
|
|
|
if (params) {
|
|
|
|
|
|
const searchParams = new URLSearchParams(normalizeEmbedParams(params));
|
|
|
|
|
|
paramString = '?' + searchParams.toString();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const iframeCode = [
|
|
|
|
|
|
`<iframe src="${url + path + paramString}" data-misskey-embed-id="${iframeId}" style="border: none; width: 100%; max-width: 500px; height: 300px; color-scheme: light dark;"></iframe>`,
|
|
|
|
|
|
`<script defer src="${url}/embed.js"></script>`,
|
|
|
|
|
|
];
|
|
|
|
|
|
return iframeCode.join('\n');
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* 埋め込みコードを生成してコピーする(カスタマイズ機能つき)
|
|
|
|
|
|
*
|
|
|
|
|
|
* カスタマイズ機能がいらない場合(事前にパラメータを指定する場合)は getEmbedCode を直接使ってください
|
|
|
|
|
|
*/
|
|
|
|
|
|
export function copyEmbedCode(entity: EmbeddableEntity, idOrUsername: string, params?: EmbedParams) {
|
|
|
|
|
|
const _params = { ...params };
|
|
|
|
|
|
|
|
|
|
|
|
if (embedRouteWithScrollbar.includes(entity) && _params.maxHeight == null) {
|
|
|
|
|
|
_params.maxHeight = 700;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// PCじゃない場合はコードカスタマイズ画面を出さずにそのままコピー
|
|
|
|
|
|
if (window.innerWidth < MOBILE_THRESHOLD) {
|
|
|
|
|
|
const _idOrUsername = entity === 'user-timeline' ? `@${idOrUsername}` : idOrUsername;
|
|
|
|
|
|
copy(getEmbedCode(`/embed/${entity}/${_idOrUsername}`, _params));
|
|
|
|
|
|
os.success();
|
|
|
|
|
|
} else {
|
|
|
|
|
|
os.popup(MkEmbedCodeGenDialog, {
|
|
|
|
|
|
entity,
|
|
|
|
|
|
idOrUsername,
|
|
|
|
|
|
params: _params,
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|