mirror of
https://github.com/misskey-dev/misskey.git
synced 2025-01-26 03:20:54 +09:00
ダイアログ内でコードを表示するように
This commit is contained in:
parent
faead03310
commit
48d3b9a7b8
16
locales/index.d.ts
vendored
16
locales/index.d.ts
vendored
@ -10103,6 +10103,10 @@ export interface Locale extends ILocale {
|
|||||||
* 高さの最大値制限が無効(0)になっています。これが意図した変更ではない場合は、高さの最大値を何らかの値に設定してください。
|
* 高さの最大値制限が無効(0)になっています。これが意図した変更ではない場合は、高さの最大値を何らかの値に設定してください。
|
||||||
*/
|
*/
|
||||||
"maxHeightWarn": string;
|
"maxHeightWarn": string;
|
||||||
|
/**
|
||||||
|
* プレビュー画面で表示可能な範囲を超えたため、実際に埋め込んだ際とは表示が異なります。
|
||||||
|
*/
|
||||||
|
"previewIsNotActual": string;
|
||||||
/**
|
/**
|
||||||
* 角丸にする
|
* 角丸にする
|
||||||
*/
|
*/
|
||||||
@ -10116,9 +10120,17 @@ export interface Locale extends ILocale {
|
|||||||
*/
|
*/
|
||||||
"applyToPreview": string;
|
"applyToPreview": string;
|
||||||
/**
|
/**
|
||||||
* プレビュー画面で表示可能な範囲を超えたため、実際に埋め込んだ際とは表示が異なります。
|
* 埋め込みコードを作成
|
||||||
*/
|
*/
|
||||||
"previewIsNotActual": string;
|
"generateCode": string;
|
||||||
|
/**
|
||||||
|
* コードが生成されました
|
||||||
|
*/
|
||||||
|
"codeGenerated": string;
|
||||||
|
/**
|
||||||
|
* 生成されたコードをウェブサイトに貼り付けてご利用ください。
|
||||||
|
*/
|
||||||
|
"codeGeneratedDescription": string;
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
declare const locales: {
|
declare const locales: {
|
||||||
|
@ -2695,7 +2695,10 @@ _embedCodeGen:
|
|||||||
maxHeight: "高さの最大値"
|
maxHeight: "高さの最大値"
|
||||||
maxHeightDescription: "0で最大値の設定が無効になります。ウィジェットが縦に伸び続けるのを防ぐために、何らかの値に指定してください。"
|
maxHeightDescription: "0で最大値の設定が無効になります。ウィジェットが縦に伸び続けるのを防ぐために、何らかの値に指定してください。"
|
||||||
maxHeightWarn: "高さの最大値制限が無効(0)になっています。これが意図した変更ではない場合は、高さの最大値を何らかの値に設定してください。"
|
maxHeightWarn: "高さの最大値制限が無効(0)になっています。これが意図した変更ではない場合は、高さの最大値を何らかの値に設定してください。"
|
||||||
|
previewIsNotActual: "プレビュー画面で表示可能な範囲を超えたため、実際に埋め込んだ際とは表示が異なります。"
|
||||||
rounded: "角丸にする"
|
rounded: "角丸にする"
|
||||||
border: "外枠に枠線をつける"
|
border: "外枠に枠線をつける"
|
||||||
applyToPreview: "プレビューに反映"
|
applyToPreview: "プレビューに反映"
|
||||||
previewIsNotActual: "プレビュー画面で表示可能な範囲を超えたため、実際に埋め込んだ際とは表示が異なります。"
|
generateCode: "埋め込みコードを作成"
|
||||||
|
codeGenerated: "コードが生成されました"
|
||||||
|
codeGeneratedDescription: "生成されたコードをウェブサイトに貼り付けてご利用ください。"
|
||||||
|
@ -5,7 +5,7 @@ SPDX-License-Identifier: AGPL-3.0-only
|
|||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div :class="$style.codeBlockRoot">
|
<div :class="$style.codeBlockRoot">
|
||||||
<button :class="$style.codeBlockCopyButton" class="_button" @click="copy">
|
<button v-if="copyButton" :class="$style.codeBlockCopyButton" class="_button" @click="copy">
|
||||||
<i class="ti ti-copy"></i>
|
<i class="ti ti-copy"></i>
|
||||||
</button>
|
</button>
|
||||||
<Suspense>
|
<Suspense>
|
||||||
@ -32,12 +32,17 @@ import { defaultStore } from '@/store.js';
|
|||||||
import { i18n } from '@/i18n.js';
|
import { i18n } from '@/i18n.js';
|
||||||
import copyToClipboard from '@/scripts/copy-to-clipboard.js';
|
import copyToClipboard from '@/scripts/copy-to-clipboard.js';
|
||||||
|
|
||||||
const props = defineProps<{
|
const props = withDefaults(defineProps<{
|
||||||
code: string;
|
code: string;
|
||||||
|
forceShow?: boolean;
|
||||||
|
copyButton?: boolean;
|
||||||
lang?: string;
|
lang?: string;
|
||||||
}>();
|
}>(), {
|
||||||
|
copyButton: true,
|
||||||
|
forceShow: false,
|
||||||
|
});
|
||||||
|
|
||||||
const show = ref(!defaultStore.state.dataSaver.code);
|
const show = ref(props.forceShow === true ? true : !defaultStore.state.dataSaver.code);
|
||||||
|
|
||||||
const XCode = defineAsyncComponent(() => import('@/components/MkCode.core.vue'));
|
const XCode = defineAsyncComponent(() => import('@/components/MkCode.core.vue'));
|
||||||
|
|
||||||
|
@ -9,15 +9,21 @@ SPDX-License-Identifier: AGPL-3.0-only
|
|||||||
:width="1000"
|
:width="1000"
|
||||||
:height="600"
|
:height="600"
|
||||||
:scroll="false"
|
:scroll="false"
|
||||||
:withOkButton="true"
|
:withOkButton="false"
|
||||||
@close="cancel()"
|
@close="cancel()"
|
||||||
@ok="ok()"
|
|
||||||
@closed="$emit('closed')"
|
@closed="$emit('closed')"
|
||||||
>
|
>
|
||||||
<template #header>{{ i18n.ts._embedCodeGen.title }}</template>
|
<template #header>{{ i18n.ts._embedCodeGen.title }}</template>
|
||||||
|
|
||||||
<div :class="$style.embedCodeGenRoot">
|
<div :class="$style.embedCodeGenRoot">
|
||||||
<div :class="$style.embedCodeGenWrapper">
|
<Transition
|
||||||
|
mode="out-in"
|
||||||
|
:enterActiveClass="$style.transition_x_enterActive"
|
||||||
|
:leaveActiveClass="$style.transition_x_leaveActive"
|
||||||
|
:enterFromClass="$style.transition_x_enterFrom"
|
||||||
|
:leaveToClass="$style.transition_x_leaveTo"
|
||||||
|
>
|
||||||
|
<div v-if="phase === 'input'" key="input" :class="$style.embedCodeGenInputRoot">
|
||||||
<div
|
<div
|
||||||
:class="$style.embedCodeGenPreviewRoot"
|
:class="$style.embedCodeGenPreviewRoot"
|
||||||
>
|
>
|
||||||
@ -60,9 +66,25 @@ SPDX-License-Identifier: AGPL-3.0-only
|
|||||||
<MkInfo v-if="typeof maxHeight === 'number' && (maxHeight <= 0 || maxHeight > 700)">{{ i18n.ts._embedCodeGen.previewIsNotActual }}</MkInfo>
|
<MkInfo v-if="typeof maxHeight === 'number' && (maxHeight <= 0 || maxHeight > 700)">{{ i18n.ts._embedCodeGen.previewIsNotActual }}</MkInfo>
|
||||||
<div class="_buttons">
|
<div class="_buttons">
|
||||||
<MkButton :disabled="iframeLoading" @click="applyToPreview">{{ i18n.ts._embedCodeGen.applyToPreview }}</MkButton>
|
<MkButton :disabled="iframeLoading" @click="applyToPreview">{{ i18n.ts._embedCodeGen.applyToPreview }}</MkButton>
|
||||||
|
<MkButton :disabled="iframeLoading" primary @click="generate">{{ i18n.ts._embedCodeGen.generateCode }} <i class="ti ti-arrow-right"></i></MkButton>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<div v-else-if="phase === 'result'" key="result" :class="$style.embedCodeGenResultRoot">
|
||||||
|
<div :class="$style.embedCodeGenResultWrapper" class="_gaps">
|
||||||
|
<div class="_gaps_s">
|
||||||
|
<div :class="$style.embedCodeGenResultHeadingIcon"><i class="ti ti-check"></i></div>
|
||||||
|
<div :class="$style.embedCodeGenResultHeading">{{ i18n.ts._embedCodeGen.codeGenerated }}</div>
|
||||||
|
<div :class="$style.embedCodeGenResultDescription">{{ i18n.ts._embedCodeGen.codeGeneratedDescription }}</div>
|
||||||
|
</div>
|
||||||
|
<div class="_gaps_s">
|
||||||
|
<MkCode :code="result" lang="html" :forceShow="true" :copyButton="false" :class="$style.embedCodeGenResultCode"/>
|
||||||
|
<MkButton :class="$style.embedCodeGenResultButtons" rounded primary @click="doCopy"><i class="ti ti-copy"></i> {{ i18n.ts.copy }}</MkButton>
|
||||||
|
</div>
|
||||||
|
<MkButton :class="$style.embedCodeGenResultButtons" rounded transparent @click="close">{{ i18n.ts.close }}</MkButton>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Transition>
|
||||||
</div>
|
</div>
|
||||||
</MkModalWindow>
|
</MkModalWindow>
|
||||||
</template>
|
</template>
|
||||||
@ -75,6 +97,8 @@ import MkInput from '@/components/MkInput.vue';
|
|||||||
import MkSelect from '@/components/MkSelect.vue';
|
import MkSelect from '@/components/MkSelect.vue';
|
||||||
import MkSwitch from '@/components/MkSwitch.vue';
|
import MkSwitch from '@/components/MkSwitch.vue';
|
||||||
import MkButton from '@/components/MkButton.vue';
|
import MkButton from '@/components/MkButton.vue';
|
||||||
|
|
||||||
|
import MkCode from '@/components/MkCode.vue';
|
||||||
import MkInfo from '@/components/MkInfo.vue';
|
import MkInfo from '@/components/MkInfo.vue';
|
||||||
|
|
||||||
import * as os from '@/os.js';
|
import * as os from '@/os.js';
|
||||||
@ -86,19 +110,16 @@ import { embedRouteWithScrollbar } from '@/scripts/embed-page.js';
|
|||||||
import type { EmbeddableEntity, EmbedParams } from '@/scripts/embed-page.js';
|
import type { EmbeddableEntity, EmbedParams } from '@/scripts/embed-page.js';
|
||||||
|
|
||||||
const emit = defineEmits<{
|
const emit = defineEmits<{
|
||||||
(ev: 'ok', url: string, code: string): void;
|
(ev: 'ok'): void;
|
||||||
(ev: 'cancel'): void;
|
(ev: 'cancel'): void;
|
||||||
(ev: 'closed'): void;
|
(ev: 'closed'): void;
|
||||||
}>();
|
}>();
|
||||||
|
|
||||||
const props = withDefaults(defineProps<{
|
const props = defineProps<{
|
||||||
entity: EmbeddableEntity;
|
entity: EmbeddableEntity;
|
||||||
idOrUsername: string;
|
idOrUsername: string;
|
||||||
params?: EmbedParams;
|
params?: EmbedParams;
|
||||||
doCopy?: boolean;
|
}>();
|
||||||
}>(), {
|
|
||||||
doCopy: true,
|
|
||||||
});
|
|
||||||
|
|
||||||
//#region Modalの制御
|
//#region Modalの制御
|
||||||
const dialogEl = shallowRef<InstanceType<typeof MkModalWindow>>();
|
const dialogEl = shallowRef<InstanceType<typeof MkModalWindow>>();
|
||||||
@ -108,17 +129,11 @@ function cancel() {
|
|||||||
dialogEl.value?.close();
|
dialogEl.value?.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
function ok() {
|
function close() {
|
||||||
const _idOrUsername = props.entity === 'user-timeline' ? '@' + props.idOrUsername : props.idOrUsername;
|
|
||||||
const generatedUrl = `${url}/embed/${props.entity}/${_idOrUsername}?${new URLSearchParams(normalizeEmbedParams(paramsForUrl.value)).toString()}`;
|
|
||||||
const generatedCode = getEmbedCode(`/embed/${props.entity}/${_idOrUsername}`, paramsForUrl.value);
|
|
||||||
if (props.doCopy) {
|
|
||||||
copy(generatedCode);
|
|
||||||
os.success();
|
|
||||||
}
|
|
||||||
emit('ok', generatedUrl, generatedCode);
|
|
||||||
dialogEl.value?.close();
|
dialogEl.value?.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const phase = ref<'input' | 'result'>('input');
|
||||||
//#endregion
|
//#endregion
|
||||||
|
|
||||||
//#region 埋め込みURL生成・カスタマイズ
|
//#region 埋め込みURL生成・カスタマイズ
|
||||||
@ -143,7 +158,7 @@ const embedPreviewUrl = computed(() => {
|
|||||||
const maxHeight = parseInt(paramClass.get('maxHeight')!);
|
const maxHeight = parseInt(paramClass.get('maxHeight')!);
|
||||||
paramClass.set('maxHeight', maxHeight === 0 ? '500' : Math.min(maxHeight, 700).toString()); // プレビューであまりにも縮小されると見づらいため、700pxまでに制限
|
paramClass.set('maxHeight', maxHeight === 0 ? '500' : Math.min(maxHeight, 700).toString()); // プレビューであまりにも縮小されると見づらいため、700pxまでに制限
|
||||||
}
|
}
|
||||||
return `${url}/embed/${props.entity}/${_idOrUsername}${paramClass.toString() ? '?' + paramClass.toString() : ''}`;
|
return `http://localhost:3000/embed/${props.entity}/${_idOrUsername}${paramClass.toString() ? '?' + paramClass.toString() : ''}`;
|
||||||
});
|
});
|
||||||
|
|
||||||
const isEmbedWithScrollbar = computed(() => embedRouteWithScrollbar.includes(props.entity));
|
const isEmbedWithScrollbar = computed(() => embedRouteWithScrollbar.includes(props.entity));
|
||||||
@ -174,6 +189,18 @@ function applyToPreview() {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const result = ref('');
|
||||||
|
function generate() {
|
||||||
|
const _idOrUsername = props.entity === 'user-timeline' ? '@' + props.idOrUsername : props.idOrUsername;
|
||||||
|
result.value = getEmbedCode(`/embed/${props.entity}/${_idOrUsername}`, paramsForUrl.value);
|
||||||
|
phase.value = 'result';
|
||||||
|
}
|
||||||
|
|
||||||
|
function doCopy() {
|
||||||
|
copy(result.value);
|
||||||
|
os.success();
|
||||||
|
}
|
||||||
//#endregion
|
//#endregion
|
||||||
|
|
||||||
//#region プレビューのリサイズ
|
//#region プレビューのリサイズ
|
||||||
@ -235,25 +262,48 @@ onMounted(() => {
|
|||||||
resizeObserver.observe(resizerRootEl.value);
|
resizeObserver.observe(resizerRootEl.value);
|
||||||
});
|
});
|
||||||
|
|
||||||
onDeactivated(() => {
|
function reset() {
|
||||||
window.removeEventListener('message', windowEventHandler);
|
window.removeEventListener('message', windowEventHandler);
|
||||||
resizeObserver.disconnect();
|
resizeObserver.disconnect();
|
||||||
|
|
||||||
|
// プレビューのリセット
|
||||||
|
iframeHeight.value = 0;
|
||||||
|
iframeScale.value = 1;
|
||||||
|
iframeLoading.value = true;
|
||||||
|
result.value = '';
|
||||||
|
phase.value = 'input';
|
||||||
|
}
|
||||||
|
|
||||||
|
onDeactivated(() => {
|
||||||
|
reset();
|
||||||
});
|
});
|
||||||
|
|
||||||
onUnmounted(() => {
|
onUnmounted(() => {
|
||||||
window.removeEventListener('message', windowEventHandler);
|
reset();
|
||||||
resizeObserver.disconnect();
|
|
||||||
});
|
});
|
||||||
//#endregion
|
//#endregion
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style module>
|
<style module>
|
||||||
|
.transition_x_enterActive,
|
||||||
|
.transition_x_leaveActive {
|
||||||
|
transition: opacity 0.3s cubic-bezier(0,0,.35,1), transform 0.3s cubic-bezier(0,0,.35,1);
|
||||||
|
}
|
||||||
|
.transition_x_enterFrom {
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateX(50px);
|
||||||
|
}
|
||||||
|
.transition_x_leaveTo {
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateX(-50px);
|
||||||
|
}
|
||||||
|
|
||||||
.embedCodeGenRoot {
|
.embedCodeGenRoot {
|
||||||
container-type: inline-size;
|
container-type: inline-size;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.embedCodeGenWrapper {
|
.embedCodeGenInputRoot {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: 1fr 400px;
|
grid-template-columns: 1fr 400px;
|
||||||
@ -319,6 +369,47 @@ onUnmounted(() => {
|
|||||||
overflow-y: scroll;
|
overflow-y: scroll;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.embedCodeGenResultRoot {
|
||||||
|
box-sizing: border-box;
|
||||||
|
padding: 24px;
|
||||||
|
height: 100%;
|
||||||
|
max-width: 700px;
|
||||||
|
margin: 0 auto;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.embedCodeGenResultHeading {
|
||||||
|
text-align: center;
|
||||||
|
font-size: 1.2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.embedCodeGenResultHeadingIcon {
|
||||||
|
margin: 0 auto;
|
||||||
|
background-color: var(--accentedBg);
|
||||||
|
color: var(--accent);
|
||||||
|
text-align: center;
|
||||||
|
height: 64px;
|
||||||
|
width: 64px;
|
||||||
|
font-size: 24px;
|
||||||
|
line-height: 64px;
|
||||||
|
border-radius: 50%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.embedCodeGenResultDescription {
|
||||||
|
text-align: center;
|
||||||
|
white-space: pre-wrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.embedCodeGenResultWrapper,
|
||||||
|
.embedCodeGenResultCode {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.embedCodeGenResultButtons {
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
@container (max-width: 800px) {
|
@container (max-width: 800px) {
|
||||||
.embedCodeGenWrapper {
|
.embedCodeGenWrapper {
|
||||||
grid-template-columns: 1fr;
|
grid-template-columns: 1fr;
|
||||||
|
Loading…
Reference in New Issue
Block a user