Merge branch 'develop' into feat-12909
|
@ -2,3 +2,4 @@
|
||||||
POSTGRES_PASSWORD=example-misskey-pass
|
POSTGRES_PASSWORD=example-misskey-pass
|
||||||
POSTGRES_USER=example-misskey-user
|
POSTGRES_USER=example-misskey-user
|
||||||
POSTGRES_DB=misskey
|
POSTGRES_DB=misskey
|
||||||
|
DATABASE_URL="postgres://${POSTGRES_USER}:${POSTGRES_PASSWORD}@db:5432/${POSTGRES_DB}"
|
||||||
|
|
|
@ -14,7 +14,11 @@
|
||||||
|
|
||||||
## 202x.x.x (Unreleased)
|
## 202x.x.x (Unreleased)
|
||||||
|
|
||||||
|
### General
|
||||||
|
- Feat: [mCaptcha](https://github.com/mCaptcha/mCaptcha)のサポートを追加
|
||||||
|
|
||||||
### Client
|
### Client
|
||||||
|
- Feat: 新しいゲームを追加
|
||||||
- Enhance: ハッシュタグ入力時に、本文の末尾の行に何も書かれていない場合は新たにスペースを追加しないように
|
- Enhance: ハッシュタグ入力時に、本文の末尾の行に何も書かれていない場合は新たにスペースを追加しないように
|
||||||
- Fix: v2023.12.0で追加された「モデレーターがユーザーのアイコンもしくはバナー画像を未設定状態にできる機能」が管理画面上で正しく表示されていない問題を修正
|
- Fix: v2023.12.0で追加された「モデレーターがユーザーのアイコンもしくはバナー画像を未設定状態にできる機能」が管理画面上で正しく表示されていない問題を修正
|
||||||
- Enhance: チャンネルノートのピン留めをノートのメニューからできるよ
|
- Enhance: チャンネルノートのピン留めをノートのメニューからできるよ
|
||||||
|
|
|
@ -7,6 +7,7 @@ services:
|
||||||
links:
|
links:
|
||||||
- db
|
- db
|
||||||
- redis
|
- redis
|
||||||
|
# - mcaptcha
|
||||||
# - meilisearch
|
# - meilisearch
|
||||||
depends_on:
|
depends_on:
|
||||||
db:
|
db:
|
||||||
|
@ -48,6 +49,36 @@ services:
|
||||||
interval: 5s
|
interval: 5s
|
||||||
retries: 20
|
retries: 20
|
||||||
|
|
||||||
|
# mcaptcha:
|
||||||
|
# restart: always
|
||||||
|
# image: mcaptcha/mcaptcha:latest
|
||||||
|
# networks:
|
||||||
|
# internal_network:
|
||||||
|
# external_network:
|
||||||
|
# aliases:
|
||||||
|
# - localhost
|
||||||
|
# ports:
|
||||||
|
# - 7493:7493
|
||||||
|
# env_file:
|
||||||
|
# - .config/docker.env
|
||||||
|
# environment:
|
||||||
|
# PORT: 7493
|
||||||
|
# MCAPTCHA_redis_URL: "redis://mcaptcha_redis/"
|
||||||
|
# depends_on:
|
||||||
|
# db:
|
||||||
|
# condition: service_healthy
|
||||||
|
# mcaptcha_redis:
|
||||||
|
# condition: service_healthy
|
||||||
|
#
|
||||||
|
# mcaptcha_redis:
|
||||||
|
# image: mcaptcha/cache:latest
|
||||||
|
# networks:
|
||||||
|
# - internal_network
|
||||||
|
# healthcheck:
|
||||||
|
# test: "redis-cli ping"
|
||||||
|
# interval: 5s
|
||||||
|
# retries: 20
|
||||||
|
|
||||||
# meilisearch:
|
# meilisearch:
|
||||||
# restart: always
|
# restart: always
|
||||||
# image: getmeili/meilisearch:v1.3.4
|
# image: getmeili/meilisearch:v1.3.4
|
||||||
|
|
|
@ -382,6 +382,11 @@ export interface Locale {
|
||||||
"enableHcaptcha": string;
|
"enableHcaptcha": string;
|
||||||
"hcaptchaSiteKey": string;
|
"hcaptchaSiteKey": string;
|
||||||
"hcaptchaSecretKey": string;
|
"hcaptchaSecretKey": string;
|
||||||
|
"mcaptcha": string;
|
||||||
|
"enableMcaptcha": string;
|
||||||
|
"mcaptchaSiteKey": string;
|
||||||
|
"mcaptchaSecretKey": string;
|
||||||
|
"mcaptchaInstanceUrl": string;
|
||||||
"recaptcha": string;
|
"recaptcha": string;
|
||||||
"enableRecaptcha": string;
|
"enableRecaptcha": string;
|
||||||
"recaptchaSiteKey": string;
|
"recaptchaSiteKey": string;
|
||||||
|
|
|
@ -379,6 +379,11 @@ hcaptcha: "hCaptcha"
|
||||||
enableHcaptcha: "hCaptchaを有効にする"
|
enableHcaptcha: "hCaptchaを有効にする"
|
||||||
hcaptchaSiteKey: "サイトキー"
|
hcaptchaSiteKey: "サイトキー"
|
||||||
hcaptchaSecretKey: "シークレットキー"
|
hcaptchaSecretKey: "シークレットキー"
|
||||||
|
mcaptcha: "mCaptcha"
|
||||||
|
enableMcaptcha: "mCaptchaを有効にする"
|
||||||
|
mcaptchaSiteKey: "サイトキー"
|
||||||
|
mcaptchaSecretKey: "シークレットキー"
|
||||||
|
mcaptchaInstanceUrl: "mCaptchaのインスタンスのURL"
|
||||||
recaptcha: "reCAPTCHA"
|
recaptcha: "reCAPTCHA"
|
||||||
enableRecaptcha: "reCAPTCHAを有効にする"
|
enableRecaptcha: "reCAPTCHAを有効にする"
|
||||||
recaptchaSiteKey: "サイトキー"
|
recaptchaSiteKey: "サイトキー"
|
||||||
|
|
|
@ -0,0 +1,22 @@
|
||||||
|
/*
|
||||||
|
* SPDX-FileCopyrightText: syuilo and other misskey contributors
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
export class SupportMcaptcha1704373210054 {
|
||||||
|
name = 'SupportMcaptcha1704373210054'
|
||||||
|
|
||||||
|
async up(queryRunner) {
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" ADD "enableMcaptcha" boolean NOT NULL DEFAULT false`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" ADD "mcaptchaSitekey" character varying(1024)`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" ADD "mcaptchaSecretKey" character varying(1024)`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" ADD "mcaptchaInstanceUrl" character varying(1024)`);
|
||||||
|
}
|
||||||
|
|
||||||
|
async down(queryRunner) {
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "mcaptchaInstanceUrl"`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "mcaptchaSecretKey"`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "mcaptchaSitekey"`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "enableMcaptcha"`);
|
||||||
|
}
|
||||||
|
}
|
|
@ -73,6 +73,37 @@ export class CaptchaService {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// https://codeberg.org/Gusted/mCaptcha/src/branch/main/mcaptcha.go
|
||||||
|
@bindThis
|
||||||
|
public async verifyMcaptcha(secret: string, siteKey: string, instanceHost: string, response: string | null | undefined): Promise<void> {
|
||||||
|
if (response == null) {
|
||||||
|
throw new Error('mcaptcha-failed: no response provided');
|
||||||
|
}
|
||||||
|
|
||||||
|
const endpointUrl = new URL('/api/v1/pow/siteverify', instanceHost);
|
||||||
|
const result = await this.httpRequestService.send(endpointUrl.toString(), {
|
||||||
|
method: 'POST',
|
||||||
|
body: JSON.stringify({
|
||||||
|
key: siteKey,
|
||||||
|
secret: secret,
|
||||||
|
token: response,
|
||||||
|
}),
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (result.status !== 200) {
|
||||||
|
throw new Error('mcaptcha-failed: mcaptcha didn\'t return 200 OK');
|
||||||
|
}
|
||||||
|
|
||||||
|
const resp = (await result.json()) as { valid: boolean };
|
||||||
|
|
||||||
|
if (!resp.valid) {
|
||||||
|
throw new Error('mcaptcha-request-failed');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@bindThis
|
@bindThis
|
||||||
public async verifyTurnstile(secret: string, response: string | null | undefined): Promise<void> {
|
public async verifyTurnstile(secret: string, response: string | null | undefined): Promise<void> {
|
||||||
if (response == null) {
|
if (response == null) {
|
||||||
|
|
|
@ -191,6 +191,29 @@ export class MiMeta {
|
||||||
})
|
})
|
||||||
public hcaptchaSecretKey: string | null;
|
public hcaptchaSecretKey: string | null;
|
||||||
|
|
||||||
|
@Column('boolean', {
|
||||||
|
default: false,
|
||||||
|
})
|
||||||
|
public enableMcaptcha: boolean;
|
||||||
|
|
||||||
|
@Column('varchar', {
|
||||||
|
length: 1024,
|
||||||
|
nullable: true,
|
||||||
|
})
|
||||||
|
public mcaptchaSitekey: string | null;
|
||||||
|
|
||||||
|
@Column('varchar', {
|
||||||
|
length: 1024,
|
||||||
|
nullable: true,
|
||||||
|
})
|
||||||
|
public mcaptchaSecretKey: string | null;
|
||||||
|
|
||||||
|
@Column('varchar', {
|
||||||
|
length: 1024,
|
||||||
|
nullable: true,
|
||||||
|
})
|
||||||
|
public mcaptchaInstanceUrl: string | null;
|
||||||
|
|
||||||
@Column('boolean', {
|
@Column('boolean', {
|
||||||
default: false,
|
default: false,
|
||||||
})
|
})
|
||||||
|
@ -467,7 +490,7 @@ export class MiMeta {
|
||||||
nullable: true,
|
nullable: true,
|
||||||
})
|
})
|
||||||
public truemailInstance: string | null;
|
public truemailInstance: string | null;
|
||||||
|
|
||||||
@Column('varchar', {
|
@Column('varchar', {
|
||||||
length: 1024,
|
length: 1024,
|
||||||
nullable: true,
|
nullable: true,
|
||||||
|
|
|
@ -65,6 +65,7 @@ export class SignupApiService {
|
||||||
'hcaptcha-response'?: string;
|
'hcaptcha-response'?: string;
|
||||||
'g-recaptcha-response'?: string;
|
'g-recaptcha-response'?: string;
|
||||||
'turnstile-response'?: string;
|
'turnstile-response'?: string;
|
||||||
|
'm-captcha-response'?: string;
|
||||||
}
|
}
|
||||||
}>,
|
}>,
|
||||||
reply: FastifyReply,
|
reply: FastifyReply,
|
||||||
|
@ -82,6 +83,12 @@ export class SignupApiService {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (instance.enableMcaptcha && instance.mcaptchaSecretKey && instance.mcaptchaSitekey && instance.mcaptchaInstanceUrl) {
|
||||||
|
await this.captchaService.verifyMcaptcha(instance.mcaptchaSecretKey, instance.mcaptchaSitekey, instance.mcaptchaInstanceUrl, body['m-captcha-response']).catch(err => {
|
||||||
|
throw new FastifyReplyError(400, err);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
if (instance.enableRecaptcha && instance.recaptchaSecretKey) {
|
if (instance.enableRecaptcha && instance.recaptchaSecretKey) {
|
||||||
await this.captchaService.verifyRecaptcha(instance.recaptchaSecretKey, body['g-recaptcha-response']).catch(err => {
|
await this.captchaService.verifyRecaptcha(instance.recaptchaSecretKey, body['g-recaptcha-response']).catch(err => {
|
||||||
throw new FastifyReplyError(400, err);
|
throw new FastifyReplyError(400, err);
|
||||||
|
|
|
@ -41,6 +41,18 @@ export const meta = {
|
||||||
type: 'string',
|
type: 'string',
|
||||||
optional: false, nullable: true,
|
optional: false, nullable: true,
|
||||||
},
|
},
|
||||||
|
enableMcaptcha: {
|
||||||
|
type: 'boolean',
|
||||||
|
optional: false, nullable: false,
|
||||||
|
},
|
||||||
|
mcaptchaSiteKey: {
|
||||||
|
type: 'string',
|
||||||
|
optional: false, nullable: true,
|
||||||
|
},
|
||||||
|
mcaptchaInstanceUrl: {
|
||||||
|
type: 'string',
|
||||||
|
optional: false, nullable: true,
|
||||||
|
},
|
||||||
enableRecaptcha: {
|
enableRecaptcha: {
|
||||||
type: 'boolean',
|
type: 'boolean',
|
||||||
optional: false, nullable: false,
|
optional: false, nullable: false,
|
||||||
|
@ -163,6 +175,10 @@ export const meta = {
|
||||||
type: 'string',
|
type: 'string',
|
||||||
optional: false, nullable: true,
|
optional: false, nullable: true,
|
||||||
},
|
},
|
||||||
|
mcaptchaSecretKey: {
|
||||||
|
type: 'string',
|
||||||
|
optional: false, nullable: true,
|
||||||
|
},
|
||||||
recaptchaSecretKey: {
|
recaptchaSecretKey: {
|
||||||
type: 'string',
|
type: 'string',
|
||||||
optional: false, nullable: true,
|
optional: false, nullable: true,
|
||||||
|
@ -468,6 +484,9 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
|
||||||
emailRequiredForSignup: instance.emailRequiredForSignup,
|
emailRequiredForSignup: instance.emailRequiredForSignup,
|
||||||
enableHcaptcha: instance.enableHcaptcha,
|
enableHcaptcha: instance.enableHcaptcha,
|
||||||
hcaptchaSiteKey: instance.hcaptchaSiteKey,
|
hcaptchaSiteKey: instance.hcaptchaSiteKey,
|
||||||
|
enableMcaptcha: instance.enableMcaptcha,
|
||||||
|
mcaptchaSiteKey: instance.mcaptchaSitekey,
|
||||||
|
mcaptchaInstanceUrl: instance.mcaptchaInstanceUrl,
|
||||||
enableRecaptcha: instance.enableRecaptcha,
|
enableRecaptcha: instance.enableRecaptcha,
|
||||||
recaptchaSiteKey: instance.recaptchaSiteKey,
|
recaptchaSiteKey: instance.recaptchaSiteKey,
|
||||||
enableTurnstile: instance.enableTurnstile,
|
enableTurnstile: instance.enableTurnstile,
|
||||||
|
@ -498,6 +517,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
|
||||||
sensitiveWords: instance.sensitiveWords,
|
sensitiveWords: instance.sensitiveWords,
|
||||||
preservedUsernames: instance.preservedUsernames,
|
preservedUsernames: instance.preservedUsernames,
|
||||||
hcaptchaSecretKey: instance.hcaptchaSecretKey,
|
hcaptchaSecretKey: instance.hcaptchaSecretKey,
|
||||||
|
mcaptchaSecretKey: instance.mcaptchaSecretKey,
|
||||||
recaptchaSecretKey: instance.recaptchaSecretKey,
|
recaptchaSecretKey: instance.recaptchaSecretKey,
|
||||||
turnstileSecretKey: instance.turnstileSecretKey,
|
turnstileSecretKey: instance.turnstileSecretKey,
|
||||||
sensitiveMediaDetection: instance.sensitiveMediaDetection,
|
sensitiveMediaDetection: instance.sensitiveMediaDetection,
|
||||||
|
|
|
@ -63,6 +63,10 @@ export const paramDef = {
|
||||||
enableHcaptcha: { type: 'boolean' },
|
enableHcaptcha: { type: 'boolean' },
|
||||||
hcaptchaSiteKey: { type: 'string', nullable: true },
|
hcaptchaSiteKey: { type: 'string', nullable: true },
|
||||||
hcaptchaSecretKey: { type: 'string', nullable: true },
|
hcaptchaSecretKey: { type: 'string', nullable: true },
|
||||||
|
enableMcaptcha: { type: 'boolean' },
|
||||||
|
mcaptchaSiteKey: { type: 'string', nullable: true },
|
||||||
|
mcaptchaInstanceUrl: { type: 'string', nullable: true },
|
||||||
|
mcaptchaSecretKey: { type: 'string', nullable: true },
|
||||||
enableRecaptcha: { type: 'boolean' },
|
enableRecaptcha: { type: 'boolean' },
|
||||||
recaptchaSiteKey: { type: 'string', nullable: true },
|
recaptchaSiteKey: { type: 'string', nullable: true },
|
||||||
recaptchaSecretKey: { type: 'string', nullable: true },
|
recaptchaSecretKey: { type: 'string', nullable: true },
|
||||||
|
@ -269,6 +273,22 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
|
||||||
set.hcaptchaSecretKey = ps.hcaptchaSecretKey;
|
set.hcaptchaSecretKey = ps.hcaptchaSecretKey;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (ps.enableMcaptcha !== undefined) {
|
||||||
|
set.enableMcaptcha = ps.enableMcaptcha;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ps.mcaptchaSiteKey !== undefined) {
|
||||||
|
set.mcaptchaSitekey = ps.mcaptchaSiteKey;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ps.mcaptchaInstanceUrl !== undefined) {
|
||||||
|
set.mcaptchaInstanceUrl = ps.mcaptchaInstanceUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ps.mcaptchaSecretKey !== undefined) {
|
||||||
|
set.mcaptchaSecretKey = ps.mcaptchaSecretKey;
|
||||||
|
}
|
||||||
|
|
||||||
if (ps.enableRecaptcha !== undefined) {
|
if (ps.enableRecaptcha !== undefined) {
|
||||||
set.enableRecaptcha = ps.enableRecaptcha;
|
set.enableRecaptcha = ps.enableRecaptcha;
|
||||||
}
|
}
|
||||||
|
@ -472,7 +492,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
|
||||||
set.verifymailAuthKey = ps.verifymailAuthKey;
|
set.verifymailAuthKey = ps.verifymailAuthKey;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ps.enableTruemailApi !== undefined) {
|
if (ps.enableTruemailApi !== undefined) {
|
||||||
set.enableTruemailApi = ps.enableTruemailApi;
|
set.enableTruemailApi = ps.enableTruemailApi;
|
||||||
}
|
}
|
||||||
|
|
|
@ -108,6 +108,18 @@ export const meta = {
|
||||||
type: 'string',
|
type: 'string',
|
||||||
optional: false, nullable: true,
|
optional: false, nullable: true,
|
||||||
},
|
},
|
||||||
|
enableMcaptcha: {
|
||||||
|
type: 'boolean',
|
||||||
|
optional: false, nullable: false,
|
||||||
|
},
|
||||||
|
mcaptchaSiteKey: {
|
||||||
|
type: 'string',
|
||||||
|
optional: false, nullable: true,
|
||||||
|
},
|
||||||
|
mcaptchaInstanceUrl: {
|
||||||
|
type: 'string',
|
||||||
|
optional: false, nullable: true,
|
||||||
|
},
|
||||||
enableRecaptcha: {
|
enableRecaptcha: {
|
||||||
type: 'boolean',
|
type: 'boolean',
|
||||||
optional: false, nullable: false,
|
optional: false, nullable: false,
|
||||||
|
@ -351,6 +363,9 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
|
||||||
emailRequiredForSignup: instance.emailRequiredForSignup,
|
emailRequiredForSignup: instance.emailRequiredForSignup,
|
||||||
enableHcaptcha: instance.enableHcaptcha,
|
enableHcaptcha: instance.enableHcaptcha,
|
||||||
hcaptchaSiteKey: instance.hcaptchaSiteKey,
|
hcaptchaSiteKey: instance.hcaptchaSiteKey,
|
||||||
|
enableMcaptcha: instance.enableMcaptcha,
|
||||||
|
mcaptchaSiteKey: instance.mcaptchaSitekey,
|
||||||
|
mcaptchaInstanceUrl: instance.mcaptchaInstanceUrl,
|
||||||
enableRecaptcha: instance.enableRecaptcha,
|
enableRecaptcha: instance.enableRecaptcha,
|
||||||
recaptchaSiteKey: instance.recaptchaSiteKey,
|
recaptchaSiteKey: instance.recaptchaSiteKey,
|
||||||
enableTurnstile: instance.enableTurnstile,
|
enableTurnstile: instance.enableTurnstile,
|
||||||
|
|
After Width: | Height: | Size: 40 KiB |
After Width: | Height: | Size: 646 B |
After Width: | Height: | Size: 46 KiB |
After Width: | Height: | Size: 36 KiB |
After Width: | Height: | Size: 39 KiB |
After Width: | Height: | Size: 68 KiB |
After Width: | Height: | Size: 40 KiB |
After Width: | Height: | Size: 22 KiB |
After Width: | Height: | Size: 43 KiB |
After Width: | Height: | Size: 51 KiB |
After Width: | Height: | Size: 47 KiB |
After Width: | Height: | Size: 44 KiB |
|
@ -19,6 +19,7 @@
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@discordapp/twemoji": "15.0.2",
|
"@discordapp/twemoji": "15.0.2",
|
||||||
"@github/webauthn-json": "2.1.1",
|
"@github/webauthn-json": "2.1.1",
|
||||||
|
"@mcaptcha/vanilla-glue": "0.1.0-alpha-3",
|
||||||
"@misskey-dev/browser-image-resizer": "2.2.1-misskey.10",
|
"@misskey-dev/browser-image-resizer": "2.2.1-misskey.10",
|
||||||
"@rollup/plugin-json": "6.1.0",
|
"@rollup/plugin-json": "6.1.0",
|
||||||
"@rollup/plugin-replace": "5.0.5",
|
"@rollup/plugin-replace": "5.0.5",
|
||||||
|
|
|
@ -6,12 +6,16 @@ SPDX-License-Identifier: AGPL-3.0-only
|
||||||
<template>
|
<template>
|
||||||
<div>
|
<div>
|
||||||
<span v-if="!available">{{ i18n.ts.waiting }}<MkEllipsis/></span>
|
<span v-if="!available">{{ i18n.ts.waiting }}<MkEllipsis/></span>
|
||||||
<div ref="captchaEl"></div>
|
<div v-if="props.provider == 'mcaptcha'">
|
||||||
|
<div id="mcaptcha__widget-container" class="m-captcha-style"></div>
|
||||||
|
<div ref="captchaEl"></div>
|
||||||
|
</div>
|
||||||
|
<div v-else ref="captchaEl"></div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
import { ref, shallowRef, computed, onMounted, onBeforeUnmount, watch } from 'vue';
|
import { ref, shallowRef, computed, onMounted, onBeforeUnmount, watch, onUnmounted } from 'vue';
|
||||||
import { defaultStore } from '@/store.js';
|
import { defaultStore } from '@/store.js';
|
||||||
import { i18n } from '@/i18n.js';
|
import { i18n } from '@/i18n.js';
|
||||||
|
|
||||||
|
@ -26,7 +30,7 @@ export type Captcha = {
|
||||||
getResponse(id: string): string;
|
getResponse(id: string): string;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type CaptchaProvider = 'hcaptcha' | 'recaptcha' | 'turnstile';
|
export type CaptchaProvider = 'hcaptcha' | 'recaptcha' | 'turnstile' | 'mcaptcha';
|
||||||
|
|
||||||
type CaptchaContainer = {
|
type CaptchaContainer = {
|
||||||
readonly [_ in CaptchaProvider]?: Captcha;
|
readonly [_ in CaptchaProvider]?: Captcha;
|
||||||
|
@ -39,6 +43,7 @@ declare global {
|
||||||
const props = defineProps<{
|
const props = defineProps<{
|
||||||
provider: CaptchaProvider;
|
provider: CaptchaProvider;
|
||||||
sitekey: string | null; // null will show error on request
|
sitekey: string | null; // null will show error on request
|
||||||
|
instanceUrl?: string | null;
|
||||||
modelValue?: string | null;
|
modelValue?: string | null;
|
||||||
}>();
|
}>();
|
||||||
|
|
||||||
|
@ -55,6 +60,7 @@ const variable = computed(() => {
|
||||||
case 'hcaptcha': return 'hcaptcha';
|
case 'hcaptcha': return 'hcaptcha';
|
||||||
case 'recaptcha': return 'grecaptcha';
|
case 'recaptcha': return 'grecaptcha';
|
||||||
case 'turnstile': return 'turnstile';
|
case 'turnstile': return 'turnstile';
|
||||||
|
case 'mcaptcha': return 'mcaptcha';
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -65,6 +71,7 @@ const src = computed(() => {
|
||||||
case 'hcaptcha': return 'https://js.hcaptcha.com/1/api.js?render=explicit&recaptchacompat=off';
|
case 'hcaptcha': return 'https://js.hcaptcha.com/1/api.js?render=explicit&recaptchacompat=off';
|
||||||
case 'recaptcha': return 'https://www.recaptcha.net/recaptcha/api.js?render=explicit';
|
case 'recaptcha': return 'https://www.recaptcha.net/recaptcha/api.js?render=explicit';
|
||||||
case 'turnstile': return 'https://challenges.cloudflare.com/turnstile/v0/api.js?render=explicit';
|
case 'turnstile': return 'https://challenges.cloudflare.com/turnstile/v0/api.js?render=explicit';
|
||||||
|
case 'mcaptcha': return null;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -72,9 +79,9 @@ const scriptId = computed(() => `script-${props.provider}`);
|
||||||
|
|
||||||
const captcha = computed<Captcha>(() => window[variable.value] || {} as unknown as Captcha);
|
const captcha = computed<Captcha>(() => window[variable.value] || {} as unknown as Captcha);
|
||||||
|
|
||||||
if (loaded) {
|
if (loaded || props.provider === 'mcaptcha') {
|
||||||
available.value = true;
|
available.value = true;
|
||||||
} else {
|
} else if (src.value !== null) {
|
||||||
(document.getElementById(scriptId.value) ?? document.head.appendChild(Object.assign(document.createElement('script'), {
|
(document.getElementById(scriptId.value) ?? document.head.appendChild(Object.assign(document.createElement('script'), {
|
||||||
async: true,
|
async: true,
|
||||||
id: scriptId.value,
|
id: scriptId.value,
|
||||||
|
@ -87,7 +94,7 @@ function reset() {
|
||||||
if (captcha.value.reset) captcha.value.reset();
|
if (captcha.value.reset) captcha.value.reset();
|
||||||
}
|
}
|
||||||
|
|
||||||
function requestRender() {
|
async function requestRender() {
|
||||||
if (captcha.value.render && captchaEl.value instanceof Element) {
|
if (captcha.value.render && captchaEl.value instanceof Element) {
|
||||||
captcha.value.render(captchaEl.value, {
|
captcha.value.render(captchaEl.value, {
|
||||||
sitekey: props.sitekey,
|
sitekey: props.sitekey,
|
||||||
|
@ -96,6 +103,15 @@ function requestRender() {
|
||||||
'expired-callback': callback,
|
'expired-callback': callback,
|
||||||
'error-callback': callback,
|
'error-callback': callback,
|
||||||
});
|
});
|
||||||
|
} else if (props.provider === 'mcaptcha' && props.instanceUrl && props.sitekey) {
|
||||||
|
const { default: Widget } = await import('@mcaptcha/vanilla-glue');
|
||||||
|
// @ts-expect-error avoid typecheck error
|
||||||
|
new Widget({
|
||||||
|
siteKey: {
|
||||||
|
instanceUrl: new URL(props.instanceUrl),
|
||||||
|
key: props.sitekey,
|
||||||
|
},
|
||||||
|
});
|
||||||
} else {
|
} else {
|
||||||
window.setTimeout(requestRender, 1);
|
window.setTimeout(requestRender, 1);
|
||||||
}
|
}
|
||||||
|
@ -105,14 +121,27 @@ function callback(response?: string) {
|
||||||
emit('update:modelValue', typeof response === 'string' ? response : null);
|
emit('update:modelValue', typeof response === 'string' ? response : null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function onReceivedMessage(message: MessageEvent) {
|
||||||
|
if (message.data.token) {
|
||||||
|
if (props.instanceUrl && new URL(message.origin).host === new URL(props.instanceUrl).host) {
|
||||||
|
callback(<string>message.data.token);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
if (available.value) {
|
if (available.value) {
|
||||||
|
window.addEventListener('message', onReceivedMessage);
|
||||||
requestRender();
|
requestRender();
|
||||||
} else {
|
} else {
|
||||||
watch(available, requestRender);
|
watch(available, requestRender);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
onUnmounted(() => {
|
||||||
|
window.removeEventListener('message', onReceivedMessage);
|
||||||
|
});
|
||||||
|
|
||||||
onBeforeUnmount(() => {
|
onBeforeUnmount(() => {
|
||||||
reset();
|
reset();
|
||||||
});
|
});
|
||||||
|
|
|
@ -5,7 +5,7 @@ SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div :class="$style.root" :style="{ zIndex, top: `${y - 64}px`, left: `${x - 64}px` }">
|
<div :class="$style.root" :style="{ zIndex, top: `${y - 64}px`, left: `${x - 64}px` }">
|
||||||
<span class="text" :class="{ up }">+1</span>
|
<span class="text" :class="{ up }">+{{ value }}</span>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
@ -16,7 +16,9 @@ import * as os from '@/os.js';
|
||||||
const props = withDefaults(defineProps<{
|
const props = withDefaults(defineProps<{
|
||||||
x: number;
|
x: number;
|
||||||
y: number;
|
y: number;
|
||||||
|
value?: number;
|
||||||
}>(), {
|
}>(), {
|
||||||
|
value: 1,
|
||||||
});
|
});
|
||||||
|
|
||||||
const emit = defineEmits<{
|
const emit = defineEmits<{
|
||||||
|
|
|
@ -63,6 +63,7 @@ SPDX-License-Identifier: AGPL-3.0-only
|
||||||
</template>
|
</template>
|
||||||
</MkInput>
|
</MkInput>
|
||||||
<MkCaptcha v-if="instance.enableHcaptcha" ref="hcaptcha" v-model="hCaptchaResponse" :class="$style.captcha" provider="hcaptcha" :sitekey="instance.hcaptchaSiteKey"/>
|
<MkCaptcha v-if="instance.enableHcaptcha" ref="hcaptcha" v-model="hCaptchaResponse" :class="$style.captcha" provider="hcaptcha" :sitekey="instance.hcaptchaSiteKey"/>
|
||||||
|
<MkCaptcha v-if="instance.enableMcaptcha" ref="mcaptcha" v-model="mCaptchaResponse" :class="$style.captcha" provider="mcaptcha" :sitekey="instance.mcaptchaSiteKey" :instanceUrl="instance.mcaptchaInstanceUrl"/>
|
||||||
<MkCaptcha v-if="instance.enableRecaptcha" ref="recaptcha" v-model="reCaptchaResponse" :class="$style.captcha" provider="recaptcha" :sitekey="instance.recaptchaSiteKey"/>
|
<MkCaptcha v-if="instance.enableRecaptcha" ref="recaptcha" v-model="reCaptchaResponse" :class="$style.captcha" provider="recaptcha" :sitekey="instance.recaptchaSiteKey"/>
|
||||||
<MkCaptcha v-if="instance.enableTurnstile" ref="turnstile" v-model="turnstileResponse" :class="$style.captcha" provider="turnstile" :sitekey="instance.turnstileSiteKey"/>
|
<MkCaptcha v-if="instance.enableTurnstile" ref="turnstile" v-model="turnstileResponse" :class="$style.captcha" provider="turnstile" :sitekey="instance.turnstileSiteKey"/>
|
||||||
<MkButton type="submit" :disabled="shouldDisableSubmitting" large gradate rounded data-cy-signup-submit style="margin: 0 auto;">
|
<MkButton type="submit" :disabled="shouldDisableSubmitting" large gradate rounded data-cy-signup-submit style="margin: 0 auto;">
|
||||||
|
@ -117,6 +118,7 @@ const passwordStrength = ref<'' | 'low' | 'medium' | 'high'>('');
|
||||||
const passwordRetypeState = ref<null | 'match' | 'not-match'>(null);
|
const passwordRetypeState = ref<null | 'match' | 'not-match'>(null);
|
||||||
const submitting = ref<boolean>(false);
|
const submitting = ref<boolean>(false);
|
||||||
const hCaptchaResponse = ref<string | null>(null);
|
const hCaptchaResponse = ref<string | null>(null);
|
||||||
|
const mCaptchaResponse = ref<string | null>(null);
|
||||||
const reCaptchaResponse = ref<string | null>(null);
|
const reCaptchaResponse = ref<string | null>(null);
|
||||||
const turnstileResponse = ref<string | null>(null);
|
const turnstileResponse = ref<string | null>(null);
|
||||||
const usernameAbortController = ref<null | AbortController>(null);
|
const usernameAbortController = ref<null | AbortController>(null);
|
||||||
|
@ -125,6 +127,7 @@ const emailAbortController = ref<null | AbortController>(null);
|
||||||
const shouldDisableSubmitting = computed((): boolean => {
|
const shouldDisableSubmitting = computed((): boolean => {
|
||||||
return submitting.value ||
|
return submitting.value ||
|
||||||
instance.enableHcaptcha && !hCaptchaResponse.value ||
|
instance.enableHcaptcha && !hCaptchaResponse.value ||
|
||||||
|
instance.enableMcaptcha && !mCaptchaResponse.value ||
|
||||||
instance.enableRecaptcha && !reCaptchaResponse.value ||
|
instance.enableRecaptcha && !reCaptchaResponse.value ||
|
||||||
instance.enableTurnstile && !turnstileResponse.value ||
|
instance.enableTurnstile && !turnstileResponse.value ||
|
||||||
instance.emailRequiredForSignup && emailState.value !== 'ok' ||
|
instance.emailRequiredForSignup && emailState.value !== 'ok' ||
|
||||||
|
@ -252,6 +255,7 @@ async function onSubmit(): Promise<void> {
|
||||||
emailAddress: email.value,
|
emailAddress: email.value,
|
||||||
invitationCode: invitationCode.value,
|
invitationCode: invitationCode.value,
|
||||||
'hcaptcha-response': hCaptchaResponse.value,
|
'hcaptcha-response': hCaptchaResponse.value,
|
||||||
|
'm-captcha-response': mCaptchaResponse.value,
|
||||||
'g-recaptcha-response': reCaptchaResponse.value,
|
'g-recaptcha-response': reCaptchaResponse.value,
|
||||||
'turnstile-response': turnstileResponse.value,
|
'turnstile-response': turnstileResponse.value,
|
||||||
});
|
});
|
||||||
|
|
|
@ -16,13 +16,13 @@
|
||||||
<!-- https://developer.mozilla.org/en-US/docs/Web/HTTP/CSP -->
|
<!-- https://developer.mozilla.org/en-US/docs/Web/HTTP/CSP -->
|
||||||
<meta
|
<meta
|
||||||
http-equiv="Content-Security-Policy"
|
http-equiv="Content-Security-Policy"
|
||||||
content="default-src 'self';
|
content="default-src 'self' https://newassets.hcaptcha.com/ https://challenges.cloudflare.com/ http://localhost:7493/;
|
||||||
worker-src 'self';
|
worker-src 'self';
|
||||||
script-src 'self' 'unsafe-eval';
|
script-src 'self' 'unsafe-eval' https://*.hcaptcha.com https://challenges.cloudflare.com;
|
||||||
style-src 'self' 'unsafe-inline';
|
style-src 'self' 'unsafe-inline';
|
||||||
img-src 'self' data: www.google.com xn--931a.moe localhost:3000 localhost:5173 127.0.0.1:5173 127.0.0.1:3000;
|
img-src 'self' data: www.google.com xn--931a.moe localhost:3000 localhost:5173 127.0.0.1:5173 127.0.0.1:3000;
|
||||||
media-src 'self' localhost:3000 localhost:5173 127.0.0.1:5173 127.0.0.1:3000;
|
media-src 'self' localhost:3000 localhost:5173 127.0.0.1:5173 127.0.0.1:3000;
|
||||||
connect-src 'self' localhost:3000 localhost:5173 127.0.0.1:5173 127.0.0.1:3000;"
|
connect-src 'self' localhost:3000 localhost:5173 127.0.0.1:5173 127.0.0.1:3000 https://newassets.hcaptcha.com;"
|
||||||
/>
|
/>
|
||||||
<meta property="og:site_name" content="[DEV BUILD] Misskey" />
|
<meta property="og:site_name" content="[DEV BUILD] Misskey" />
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||||
|
|
|
@ -10,6 +10,7 @@ SPDX-License-Identifier: AGPL-3.0-only
|
||||||
<MkRadios v-model="provider">
|
<MkRadios v-model="provider">
|
||||||
<option :value="null">{{ i18n.ts.none }} ({{ i18n.ts.notRecommended }})</option>
|
<option :value="null">{{ i18n.ts.none }} ({{ i18n.ts.notRecommended }})</option>
|
||||||
<option value="hcaptcha">hCaptcha</option>
|
<option value="hcaptcha">hCaptcha</option>
|
||||||
|
<option value="mcaptcha">mCaptcha</option>
|
||||||
<option value="recaptcha">reCAPTCHA</option>
|
<option value="recaptcha">reCAPTCHA</option>
|
||||||
<option value="turnstile">Turnstile</option>
|
<option value="turnstile">Turnstile</option>
|
||||||
</MkRadios>
|
</MkRadios>
|
||||||
|
@ -28,6 +29,24 @@ SPDX-License-Identifier: AGPL-3.0-only
|
||||||
<MkCaptcha provider="hcaptcha" :sitekey="hcaptchaSiteKey || '10000000-ffff-ffff-ffff-000000000001'"/>
|
<MkCaptcha provider="hcaptcha" :sitekey="hcaptchaSiteKey || '10000000-ffff-ffff-ffff-000000000001'"/>
|
||||||
</FormSlot>
|
</FormSlot>
|
||||||
</template>
|
</template>
|
||||||
|
<template v-else-if="provider === 'mcaptcha'">
|
||||||
|
<MkInput v-model="mcaptchaSiteKey">
|
||||||
|
<template #prefix><i class="ti ti-key"></i></template>
|
||||||
|
<template #label>{{ i18n.ts.mcaptchaSiteKey }}</template>
|
||||||
|
</MkInput>
|
||||||
|
<MkInput v-model="mcaptchaSecretKey">
|
||||||
|
<template #prefix><i class="ti ti-key"></i></template>
|
||||||
|
<template #label>{{ i18n.ts.mcaptchaSecretKey }}</template>
|
||||||
|
</MkInput>
|
||||||
|
<MkInput v-model="mcaptchaInstanceUrl">
|
||||||
|
<template #prefix><i class="ti ti-link"></i></template>
|
||||||
|
<template #label>{{ i18n.ts.mcaptchaInstanceUrl }}</template>
|
||||||
|
</MkInput>
|
||||||
|
<FormSlot v-if="mcaptchaSiteKey && mcaptchaInstanceUrl">
|
||||||
|
<template #label>{{ i18n.ts.preview }}</template>
|
||||||
|
<MkCaptcha provider="mcaptcha" :sitekey="mcaptchaSiteKey" :instanceUrl="mcaptchaInstanceUrl"/>
|
||||||
|
</FormSlot>
|
||||||
|
</template>
|
||||||
<template v-else-if="provider === 'recaptcha'">
|
<template v-else-if="provider === 'recaptcha'">
|
||||||
<MkInput v-model="recaptchaSiteKey">
|
<MkInput v-model="recaptchaSiteKey">
|
||||||
<template #prefix><i class="ti ti-key"></i></template>
|
<template #prefix><i class="ti ti-key"></i></template>
|
||||||
|
@ -81,6 +100,9 @@ const MkCaptcha = defineAsyncComponent(() => import('@/components/MkCaptcha.vue'
|
||||||
const provider = ref<CaptchaProvider | null>(null);
|
const provider = ref<CaptchaProvider | null>(null);
|
||||||
const hcaptchaSiteKey = ref<string | null>(null);
|
const hcaptchaSiteKey = ref<string | null>(null);
|
||||||
const hcaptchaSecretKey = ref<string | null>(null);
|
const hcaptchaSecretKey = ref<string | null>(null);
|
||||||
|
const mcaptchaSiteKey = ref<string | null>(null);
|
||||||
|
const mcaptchaSecretKey = ref<string | null>(null);
|
||||||
|
const mcaptchaInstanceUrl = ref<string | null>(null);
|
||||||
const recaptchaSiteKey = ref<string | null>(null);
|
const recaptchaSiteKey = ref<string | null>(null);
|
||||||
const recaptchaSecretKey = ref<string | null>(null);
|
const recaptchaSecretKey = ref<string | null>(null);
|
||||||
const turnstileSiteKey = ref<string | null>(null);
|
const turnstileSiteKey = ref<string | null>(null);
|
||||||
|
@ -90,12 +112,18 @@ async function init() {
|
||||||
const meta = await misskeyApi('admin/meta');
|
const meta = await misskeyApi('admin/meta');
|
||||||
hcaptchaSiteKey.value = meta.hcaptchaSiteKey;
|
hcaptchaSiteKey.value = meta.hcaptchaSiteKey;
|
||||||
hcaptchaSecretKey.value = meta.hcaptchaSecretKey;
|
hcaptchaSecretKey.value = meta.hcaptchaSecretKey;
|
||||||
|
mcaptchaSiteKey.value = meta.mcaptchaSiteKey;
|
||||||
|
mcaptchaSecretKey.value = meta.mcaptchaSecretKey;
|
||||||
|
mcaptchaInstanceUrl.value = meta.mcaptchaInstanceUrl;
|
||||||
recaptchaSiteKey.value = meta.recaptchaSiteKey;
|
recaptchaSiteKey.value = meta.recaptchaSiteKey;
|
||||||
recaptchaSecretKey.value = meta.recaptchaSecretKey;
|
recaptchaSecretKey.value = meta.recaptchaSecretKey;
|
||||||
turnstileSiteKey.value = meta.turnstileSiteKey;
|
turnstileSiteKey.value = meta.turnstileSiteKey;
|
||||||
turnstileSecretKey.value = meta.turnstileSecretKey;
|
turnstileSecretKey.value = meta.turnstileSecretKey;
|
||||||
|
|
||||||
provider.value = meta.enableHcaptcha ? 'hcaptcha' : meta.enableRecaptcha ? 'recaptcha' : meta.enableTurnstile ? 'turnstile' : null;
|
provider.value = meta.enableHcaptcha ? 'hcaptcha' :
|
||||||
|
meta.enableRecaptcha ? 'recaptcha' :
|
||||||
|
meta.enableTurnstile ? 'turnstile' :
|
||||||
|
meta.enableMcaptcha ? 'mcaptcha' : null;
|
||||||
}
|
}
|
||||||
|
|
||||||
function save() {
|
function save() {
|
||||||
|
@ -103,6 +131,10 @@ function save() {
|
||||||
enableHcaptcha: provider.value === 'hcaptcha',
|
enableHcaptcha: provider.value === 'hcaptcha',
|
||||||
hcaptchaSiteKey: hcaptchaSiteKey.value,
|
hcaptchaSiteKey: hcaptchaSiteKey.value,
|
||||||
hcaptchaSecretKey: hcaptchaSecretKey.value,
|
hcaptchaSecretKey: hcaptchaSecretKey.value,
|
||||||
|
enableMcaptcha: provider.value === 'mcaptcha',
|
||||||
|
mcaptchaSiteKey: mcaptchaSiteKey.value,
|
||||||
|
mcaptchaSecretKey: mcaptchaSecretKey.value,
|
||||||
|
mcaptchaInstanceUrl: mcaptchaInstanceUrl.value,
|
||||||
enableRecaptcha: provider.value === 'recaptcha',
|
enableRecaptcha: provider.value === 'recaptcha',
|
||||||
recaptchaSiteKey: recaptchaSiteKey.value,
|
recaptchaSiteKey: recaptchaSiteKey.value,
|
||||||
recaptchaSecretKey: recaptchaSecretKey.value,
|
recaptchaSecretKey: recaptchaSecretKey.value,
|
||||||
|
|
|
@ -13,6 +13,7 @@ SPDX-License-Identifier: AGPL-3.0-only
|
||||||
<template #icon><i class="ti ti-shield"></i></template>
|
<template #icon><i class="ti ti-shield"></i></template>
|
||||||
<template #label>{{ i18n.ts.botProtection }}</template>
|
<template #label>{{ i18n.ts.botProtection }}</template>
|
||||||
<template v-if="enableHcaptcha" #suffix>hCaptcha</template>
|
<template v-if="enableHcaptcha" #suffix>hCaptcha</template>
|
||||||
|
<template v-else-if="enableMcaptcha" #suffix>mCaptcha</template>
|
||||||
<template v-else-if="enableRecaptcha" #suffix>reCAPTCHA</template>
|
<template v-else-if="enableRecaptcha" #suffix>reCAPTCHA</template>
|
||||||
<template v-else-if="enableTurnstile" #suffix>Turnstile</template>
|
<template v-else-if="enableTurnstile" #suffix>Turnstile</template>
|
||||||
<template v-else #suffix>{{ i18n.ts.none }} ({{ i18n.ts.notRecommended }})</template>
|
<template v-else #suffix>{{ i18n.ts.none }} ({{ i18n.ts.notRecommended }})</template>
|
||||||
|
@ -155,6 +156,7 @@ import { definePageMetadata } from '@/scripts/page-metadata.js';
|
||||||
|
|
||||||
const summalyProxy = ref<string>('');
|
const summalyProxy = ref<string>('');
|
||||||
const enableHcaptcha = ref<boolean>(false);
|
const enableHcaptcha = ref<boolean>(false);
|
||||||
|
const enableMcaptcha = ref<boolean>(false);
|
||||||
const enableRecaptcha = ref<boolean>(false);
|
const enableRecaptcha = ref<boolean>(false);
|
||||||
const enableTurnstile = ref<boolean>(false);
|
const enableTurnstile = ref<boolean>(false);
|
||||||
const sensitiveMediaDetection = ref<string>('none');
|
const sensitiveMediaDetection = ref<string>('none');
|
||||||
|
@ -174,6 +176,7 @@ async function init() {
|
||||||
const meta = await misskeyApi('admin/meta');
|
const meta = await misskeyApi('admin/meta');
|
||||||
summalyProxy.value = meta.summalyProxy;
|
summalyProxy.value = meta.summalyProxy;
|
||||||
enableHcaptcha.value = meta.enableHcaptcha;
|
enableHcaptcha.value = meta.enableHcaptcha;
|
||||||
|
enableMcaptcha.value = meta.enableMcaptcha;
|
||||||
enableRecaptcha.value = meta.enableRecaptcha;
|
enableRecaptcha.value = meta.enableRecaptcha;
|
||||||
enableTurnstile.value = meta.enableTurnstile;
|
enableTurnstile.value = meta.enableTurnstile;
|
||||||
sensitiveMediaDetection.value = meta.sensitiveMediaDetection;
|
sensitiveMediaDetection.value = meta.sensitiveMediaDetection;
|
||||||
|
|
|
@ -0,0 +1,761 @@
|
||||||
|
<!--
|
||||||
|
SPDX-FileCopyrightText: syuilo and other misskey contributors
|
||||||
|
SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
-->
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<MkStickyContainer>
|
||||||
|
<template #header><MkPageHeader/></template>
|
||||||
|
<MkSpacer :contentMax="800">
|
||||||
|
<div class="_gaps_s" :class="$style.root" style="margin: 0 auto;" :style="{ maxWidth: GAME_WIDTH + 'px' }">
|
||||||
|
<div style="display: flex;">
|
||||||
|
<div :class="$style.frame" style="flex: 1; margin-right: 10px;">
|
||||||
|
<div :class="$style.frameInner">
|
||||||
|
SCORE: <b><MkNumber :value="score"/></b>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div :class="[$style.frame, $style.stock]" style="margin-left: auto;">
|
||||||
|
<div :class="$style.frameInner" style="text-align: center;">
|
||||||
|
NEXT >>>
|
||||||
|
<TransitionGroup
|
||||||
|
:enterActiveClass="$style.transition_stock_enterActive"
|
||||||
|
:leaveActiveClass="$style.transition_stock_leaveActive"
|
||||||
|
:enterFromClass="$style.transition_stock_enterFrom"
|
||||||
|
:leaveToClass="$style.transition_stock_leaveTo"
|
||||||
|
:moveClass="$style.transition_stock_move"
|
||||||
|
>
|
||||||
|
<div v-for="x in stock" :key="x.id" style="display: inline-block;">
|
||||||
|
<img :src="x.fruit.img" style="width: 32px;"/>
|
||||||
|
</div>
|
||||||
|
</TransitionGroup>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div :class="$style.main">
|
||||||
|
<div ref="containerEl" :class="[$style.container, { [$style.gameOver]: gameOver }]" @click.stop.prevent="onClick" @touchmove="onTouchmove" @touchend="onTouchend" @mousemove="onMousemove">
|
||||||
|
<img src="/client-assets/drop-and-fusion/frame.svg" :class="$style.mainFrameImg"/>
|
||||||
|
<canvas ref="canvasEl" :class="$style.canvas"/>
|
||||||
|
<Transition
|
||||||
|
:enterActiveClass="$style.transition_combo_enterActive"
|
||||||
|
:leaveActiveClass="$style.transition_combo_leaveActive"
|
||||||
|
:enterFromClass="$style.transition_combo_enterFrom"
|
||||||
|
:leaveToClass="$style.transition_combo_leaveTo"
|
||||||
|
:moveClass="$style.transition_combo_move"
|
||||||
|
>
|
||||||
|
<div v-show="combo > 1" :class="$style.combo" :style="{ fontSize: `${100 + ((comboPrev - 2) * 15)}%` }">{{ comboPrev }} Chain!</div>
|
||||||
|
</Transition>
|
||||||
|
<Transition
|
||||||
|
:enterActiveClass="$style.transition_picked_enterActive"
|
||||||
|
:leaveActiveClass="$style.transition_picked_leaveActive"
|
||||||
|
:enterFromClass="$style.transition_picked_enterFrom"
|
||||||
|
:leaveToClass="$style.transition_picked_leaveTo"
|
||||||
|
:moveClass="$style.transition_picked_move"
|
||||||
|
mode="out-in"
|
||||||
|
>
|
||||||
|
<img v-if="currentPick" :key="currentPick.id" :src="currentPick?.fruit.img" :class="$style.currentFruit" :style="{ top: -(currentPick?.fruit.size / 2) + 'px', left: (mouseX - (currentPick?.fruit.size / 2)) + 'px', width: `${currentPick?.fruit.size}px` }"/>
|
||||||
|
</Transition>
|
||||||
|
<template v-if="dropReady">
|
||||||
|
<img src="/client-assets/drop-and-fusion/drop-arrow.svg" :class="$style.currentFruitArrow" :style="{ top: (currentPick?.fruit.size / 2) + 10 + 'px', left: (mouseX - 10) + 'px', width: `20px` }"/>
|
||||||
|
<div :class="$style.dropGuide" :style="{ left: (mouseX - 2) + 'px' }"/>
|
||||||
|
</template>
|
||||||
|
<div v-if="gameOver" :class="$style.gameOverLabel">
|
||||||
|
<div>GAME OVER!</div>
|
||||||
|
<div>SCORE: <MkNumber :value="score"/></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<MkButton @click="restart">Restart</MkButton>
|
||||||
|
</div>
|
||||||
|
</MkSpacer>
|
||||||
|
</MkStickyContainer>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import * as Matter from 'matter-js';
|
||||||
|
import { Ref, onMounted, ref, shallowRef } from 'vue';
|
||||||
|
import { EventEmitter } from 'eventemitter3';
|
||||||
|
import { definePageMetadata } from '@/scripts/page-metadata.js';
|
||||||
|
import * as sound from '@/scripts/sound.js';
|
||||||
|
import MkRippleEffect from '@/components/MkRippleEffect.vue';
|
||||||
|
import * as os from '@/os.js';
|
||||||
|
import MkNumber from '@/components/MkNumber.vue';
|
||||||
|
import MkPlusOneEffect from '@/components/MkPlusOneEffect.vue';
|
||||||
|
import MkButton from '@/components/MkButton.vue';
|
||||||
|
|
||||||
|
const containerEl = shallowRef<HTMLElement>();
|
||||||
|
const canvasEl = shallowRef<HTMLCanvasElement>();
|
||||||
|
const mouseX = ref(0);
|
||||||
|
|
||||||
|
const BASE_SIZE = 30;
|
||||||
|
const FRUITS = [{
|
||||||
|
id: '9377076d-c980-4d83-bdaf-175bc58275b7',
|
||||||
|
level: 10,
|
||||||
|
size: BASE_SIZE * 1.25 * 1.25 * 1.25 * 1.25 * 1.25 * 1.25 * 1.25 * 1.25 * 1.25,
|
||||||
|
score: 512,
|
||||||
|
available: false,
|
||||||
|
sfxPitch: 0.25,
|
||||||
|
img: '/client-assets/drop-and-fusion/exploding_head.png',
|
||||||
|
imgSize: 256,
|
||||||
|
spriteScale: 1.12,
|
||||||
|
}, {
|
||||||
|
id: 'be9f38d2-b267-4b1a-b420-904e22e80568',
|
||||||
|
level: 9,
|
||||||
|
size: BASE_SIZE * 1.25 * 1.25 * 1.25 * 1.25 * 1.25 * 1.25 * 1.25 * 1.25,
|
||||||
|
score: 256,
|
||||||
|
available: false,
|
||||||
|
sfxPitch: 0.5,
|
||||||
|
img: '/client-assets/drop-and-fusion/face_with_symbols_on_mouth.png',
|
||||||
|
imgSize: 256,
|
||||||
|
spriteScale: 1.12,
|
||||||
|
}, {
|
||||||
|
id: 'beb30459-b064-4888-926b-f572e4e72e0c',
|
||||||
|
level: 8,
|
||||||
|
size: BASE_SIZE * 1.25 * 1.25 * 1.25 * 1.25 * 1.25 * 1.25 * 1.25,
|
||||||
|
score: 128,
|
||||||
|
available: false,
|
||||||
|
sfxPitch: 0.75,
|
||||||
|
img: '/client-assets/drop-and-fusion/cold_face.png',
|
||||||
|
imgSize: 256,
|
||||||
|
spriteScale: 1.12,
|
||||||
|
}, {
|
||||||
|
id: 'feab6426-d9d8-49ae-849c-048cdbb6cdf0',
|
||||||
|
level: 7,
|
||||||
|
size: BASE_SIZE * 1.25 * 1.25 * 1.25 * 1.25 * 1.25 * 1.25,
|
||||||
|
score: 64,
|
||||||
|
available: false,
|
||||||
|
sfxPitch: 1,
|
||||||
|
img: '/client-assets/drop-and-fusion/zany_face.png',
|
||||||
|
imgSize: 256,
|
||||||
|
spriteScale: 1.12,
|
||||||
|
}, {
|
||||||
|
id: 'd6d8fed6-6d18-4726-81a1-6cf2c974df8a',
|
||||||
|
level: 6,
|
||||||
|
size: BASE_SIZE * 1.25 * 1.25 * 1.25 * 1.25 * 1.25,
|
||||||
|
score: 32,
|
||||||
|
available: false,
|
||||||
|
sfxPitch: 1.5,
|
||||||
|
img: '/client-assets/drop-and-fusion/pleading_face.png',
|
||||||
|
imgSize: 256,
|
||||||
|
spriteScale: 1.12,
|
||||||
|
}, {
|
||||||
|
id: '249c728e-230f-4332-bbbf-281c271c75b2',
|
||||||
|
level: 5,
|
||||||
|
size: BASE_SIZE * 1.25 * 1.25 * 1.25 * 1.25,
|
||||||
|
score: 16,
|
||||||
|
available: true,
|
||||||
|
sfxPitch: 2,
|
||||||
|
img: '/client-assets/drop-and-fusion/face_with_open_mouth.png',
|
||||||
|
imgSize: 256,
|
||||||
|
spriteScale: 1.12,
|
||||||
|
}, {
|
||||||
|
id: '23d67613-d484-4a93-b71e-3e81b19d6186',
|
||||||
|
level: 4,
|
||||||
|
size: BASE_SIZE * 1.25 * 1.25 * 1.25,
|
||||||
|
score: 8,
|
||||||
|
available: true,
|
||||||
|
sfxPitch: 2.5,
|
||||||
|
img: '/client-assets/drop-and-fusion/smiling_face_with_sunglasses.png',
|
||||||
|
imgSize: 256,
|
||||||
|
spriteScale: 1.12,
|
||||||
|
}, {
|
||||||
|
id: '3cbd0add-ad7d-4685-bad0-29f6dddc0b99',
|
||||||
|
level: 3,
|
||||||
|
size: BASE_SIZE * 1.25 * 1.25,
|
||||||
|
score: 4,
|
||||||
|
available: true,
|
||||||
|
sfxPitch: 3,
|
||||||
|
img: '/client-assets/drop-and-fusion/grinning_squinting_face.png',
|
||||||
|
imgSize: 256,
|
||||||
|
spriteScale: 1.12,
|
||||||
|
}, {
|
||||||
|
id: '8f86d4f4-ee02-41bf-ad38-1ce0ae457fb5',
|
||||||
|
level: 2,
|
||||||
|
size: BASE_SIZE * 1.25,
|
||||||
|
score: 2,
|
||||||
|
available: true,
|
||||||
|
sfxPitch: 3.5,
|
||||||
|
img: '/client-assets/drop-and-fusion/smiling_face_with_hearts.png',
|
||||||
|
imgSize: 256,
|
||||||
|
spriteScale: 1.12,
|
||||||
|
}, {
|
||||||
|
id: '64ec4add-ce39-42b4-96cb-33908f3f118d',
|
||||||
|
level: 1,
|
||||||
|
size: BASE_SIZE,
|
||||||
|
score: 1,
|
||||||
|
available: true,
|
||||||
|
sfxPitch: 4,
|
||||||
|
img: '/client-assets/drop-and-fusion/heart_suit.png',
|
||||||
|
imgSize: 256,
|
||||||
|
spriteScale: 1.12,
|
||||||
|
}] as const;
|
||||||
|
|
||||||
|
const GAME_WIDTH = 450;
|
||||||
|
const GAME_HEIGHT = 600;
|
||||||
|
const PHYSICS_QUALITY_FACTOR = 32; // 低いほどパフォーマンスが高いがガタガタして安定しなくなる
|
||||||
|
|
||||||
|
let viewScaleX = 1;
|
||||||
|
let viewScaleY = 1;
|
||||||
|
const currentPick = shallowRef<{ id: string; fruit: typeof FRUITS[number] } | null>(null);
|
||||||
|
const stock = shallowRef<{ id: string; fruit: typeof FRUITS[number] }[]>([]);
|
||||||
|
const score = ref(0);
|
||||||
|
const combo = ref(0);
|
||||||
|
const comboPrev = ref(0);
|
||||||
|
const dropReady = ref(true);
|
||||||
|
const gameOver = ref(false);
|
||||||
|
const gameStarted = ref(false);
|
||||||
|
|
||||||
|
class Game extends EventEmitter<{
|
||||||
|
changeScore: (score: number) => void;
|
||||||
|
changeCombo: (combo: number) => void;
|
||||||
|
changeStock: (stock: { id: string; fruit: typeof FRUITS[number] }[]) => void;
|
||||||
|
dropped: () => void;
|
||||||
|
fusioned: (x: number, y: number, score: number) => void;
|
||||||
|
gameOver: () => void;
|
||||||
|
}> {
|
||||||
|
private COMBO_INTERVAL = 1000;
|
||||||
|
public readonly DROP_INTERVAL = 500;
|
||||||
|
private PLAYAREA_MARGIN = 25;
|
||||||
|
private engine: Matter.Engine;
|
||||||
|
private render: Matter.Render;
|
||||||
|
private runner: Matter.Runner;
|
||||||
|
private detector: Matter.Detector;
|
||||||
|
private overflowCollider: Matter.Body;
|
||||||
|
private isGameOver = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* フィールドに出ていて、かつ合体の対象となるアイテム
|
||||||
|
*/
|
||||||
|
private activeBodyIds: Matter.Body['id'][] = [];
|
||||||
|
|
||||||
|
private latestDroppedBodyId: Matter.Body['id'] | null = null;
|
||||||
|
|
||||||
|
private latestDroppedAt = 0;
|
||||||
|
private latestFusionedAt = 0;
|
||||||
|
private stock: { id: string; fruit: typeof FRUITS[number] }[] = [];
|
||||||
|
|
||||||
|
private _combo = 0;
|
||||||
|
private get combo() {
|
||||||
|
return this._combo;
|
||||||
|
}
|
||||||
|
private set combo(value: number) {
|
||||||
|
this._combo = value;
|
||||||
|
this.emit('changeCombo', value);
|
||||||
|
}
|
||||||
|
|
||||||
|
private _score = 0;
|
||||||
|
private get score() {
|
||||||
|
return this._score;
|
||||||
|
}
|
||||||
|
private set score(value: number) {
|
||||||
|
this._score = value;
|
||||||
|
this.emit('changeScore', value);
|
||||||
|
}
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
super();
|
||||||
|
|
||||||
|
this.engine = Matter.Engine.create({
|
||||||
|
constraintIterations: 2 * PHYSICS_QUALITY_FACTOR,
|
||||||
|
positionIterations: 6 * PHYSICS_QUALITY_FACTOR,
|
||||||
|
velocityIterations: 4 * PHYSICS_QUALITY_FACTOR,
|
||||||
|
gravity: {
|
||||||
|
x: 0,
|
||||||
|
y: 1,
|
||||||
|
},
|
||||||
|
timing: {
|
||||||
|
timeScale: 2,
|
||||||
|
},
|
||||||
|
enableSleeping: false,
|
||||||
|
});
|
||||||
|
|
||||||
|
this.render = Matter.Render.create({
|
||||||
|
engine: this.engine,
|
||||||
|
canvas: canvasEl.value,
|
||||||
|
options: {
|
||||||
|
width: GAME_WIDTH,
|
||||||
|
height: GAME_HEIGHT,
|
||||||
|
background: 'transparent', // transparent to hide
|
||||||
|
wireframeBackground: 'transparent', // transparent to hide
|
||||||
|
wireframes: false,
|
||||||
|
showSleeping: false,
|
||||||
|
pixelRatio: window.devicePixelRatio,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
Matter.Render.run(this.render);
|
||||||
|
|
||||||
|
this.runner = Matter.Runner.create();
|
||||||
|
Matter.Runner.run(this.runner, this.engine);
|
||||||
|
|
||||||
|
this.detector = Matter.Detector.create();
|
||||||
|
|
||||||
|
this.engine.world.bodies = [];
|
||||||
|
|
||||||
|
//#region walls
|
||||||
|
const WALL_OPTIONS: Matter.IChamferableBodyDefinition = {
|
||||||
|
isStatic: true,
|
||||||
|
render: {
|
||||||
|
strokeStyle: 'transparent',
|
||||||
|
fillStyle: 'transparent',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
const thickness = 100;
|
||||||
|
Matter.Composite.add(this.engine.world, [
|
||||||
|
Matter.Bodies.rectangle(GAME_WIDTH / 2, GAME_HEIGHT + (thickness / 2) - this.PLAYAREA_MARGIN, GAME_WIDTH, thickness, WALL_OPTIONS),
|
||||||
|
Matter.Bodies.rectangle(GAME_WIDTH + (thickness / 2) - this.PLAYAREA_MARGIN, GAME_HEIGHT / 2, thickness, GAME_HEIGHT, WALL_OPTIONS),
|
||||||
|
Matter.Bodies.rectangle(-((thickness / 2) - this.PLAYAREA_MARGIN), GAME_HEIGHT / 2, thickness, GAME_HEIGHT, WALL_OPTIONS),
|
||||||
|
]);
|
||||||
|
//#endregion
|
||||||
|
|
||||||
|
this.overflowCollider = Matter.Bodies.rectangle(GAME_WIDTH / 2, 0, GAME_WIDTH, 125, {
|
||||||
|
isStatic: true,
|
||||||
|
isSensor: true,
|
||||||
|
render: {
|
||||||
|
strokeStyle: 'transparent',
|
||||||
|
fillStyle: 'transparent',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
Matter.Composite.add(this.engine.world, this.overflowCollider);
|
||||||
|
|
||||||
|
// fit the render viewport to the scene
|
||||||
|
Matter.Render.lookAt(this.render, {
|
||||||
|
min: { x: 0, y: 0 },
|
||||||
|
max: { x: GAME_WIDTH, y: GAME_HEIGHT },
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private createBody(fruit: typeof FRUITS[number], x: number, y: number) {
|
||||||
|
return Matter.Bodies.circle(x, y, fruit.size / 2, {
|
||||||
|
label: fruit.id,
|
||||||
|
density: 0.0005,
|
||||||
|
frictionAir: 0.01,
|
||||||
|
restitution: 0.4,
|
||||||
|
friction: 0.5,
|
||||||
|
frictionStatic: 5,
|
||||||
|
//mass: 0,
|
||||||
|
render: {
|
||||||
|
sprite: {
|
||||||
|
texture: fruit.img,
|
||||||
|
xScale: (fruit.size / fruit.imgSize) * fruit.spriteScale,
|
||||||
|
yScale: (fruit.size / fruit.imgSize) * fruit.spriteScale,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private fusion(bodyA: Matter.Body, bodyB: Matter.Body) {
|
||||||
|
const now = Date.now();
|
||||||
|
if (this.latestFusionedAt > now - this.COMBO_INTERVAL) {
|
||||||
|
this.combo++;
|
||||||
|
} else {
|
||||||
|
this.combo = 1;
|
||||||
|
}
|
||||||
|
this.latestFusionedAt = now;
|
||||||
|
|
||||||
|
// TODO: 単に位置だけでなくそれぞれの動きベクトルも融合する
|
||||||
|
const newX = (bodyA.position.x + bodyB.position.x) / 2;
|
||||||
|
const newY = (bodyA.position.y + bodyB.position.y) / 2;
|
||||||
|
|
||||||
|
Matter.Composite.remove(this.engine.world, [bodyA, bodyB]);
|
||||||
|
this.activeBodyIds = this.activeBodyIds.filter(x => x !== bodyA.id && x !== bodyB.id);
|
||||||
|
|
||||||
|
const currentFruit = FRUITS.find(y => y.id === bodyA.label)!;
|
||||||
|
const nextFruit = FRUITS.find(x => x.level === currentFruit.level + 1);
|
||||||
|
|
||||||
|
if (nextFruit) {
|
||||||
|
const body = this.createBody(nextFruit, newX, newY);
|
||||||
|
Matter.Composite.add(this.engine.world, body);
|
||||||
|
|
||||||
|
// 連鎖してfusionした場合の分かりやすさのため少し間を置いてからfusion対象になるようにする
|
||||||
|
window.setTimeout(() => {
|
||||||
|
this.activeBodyIds.push(body.id);
|
||||||
|
}, 100);
|
||||||
|
|
||||||
|
const additionalScore = Math.round(currentFruit.score * (1 + (this.combo / 3)));
|
||||||
|
this.score += additionalScore;
|
||||||
|
|
||||||
|
const pan = ((newX / GAME_WIDTH) - 0.5) * 2;
|
||||||
|
sound.playRaw('syuilo/bubble2', 1, pan, nextFruit.sfxPitch);
|
||||||
|
|
||||||
|
this.emit('fusioned', newX, newY, additionalScore);
|
||||||
|
} else {
|
||||||
|
//const VELOCITY = 30;
|
||||||
|
//for (let i = 0; i < 10; i++) {
|
||||||
|
// const body = createBody(FRUITS.find(x => x.level === (1 + Math.floor(Math.random() * 3)))!, x + ((Math.random() * VELOCITY) - (VELOCITY / 2)), y + ((Math.random() * VELOCITY) - (VELOCITY / 2)));
|
||||||
|
// Matter.Composite.add(world, body);
|
||||||
|
// bodies.push(body);
|
||||||
|
//}
|
||||||
|
//sound.playRaw({
|
||||||
|
// type: 'syuilo/bubble2',
|
||||||
|
// volume: 1,
|
||||||
|
//});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private gameOver() {
|
||||||
|
this.isGameOver = true;
|
||||||
|
Matter.Runner.stop(this.runner);
|
||||||
|
this.emit('gameOver');
|
||||||
|
}
|
||||||
|
|
||||||
|
public start() {
|
||||||
|
for (let i = 0; i < 4; i++) {
|
||||||
|
this.stock.push({
|
||||||
|
id: Math.random().toString(),
|
||||||
|
fruit: FRUITS.filter(x => x.available)[Math.floor(Math.random() * FRUITS.filter(x => x.available).length)],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
this.emit('changeStock', this.stock);
|
||||||
|
|
||||||
|
// TODO: fusion予約状態のアイテムは光らせるなどの演出をすると楽しそう
|
||||||
|
let fusionReservedPairs: { bodyA: Matter.Body; bodyB: Matter.Body }[] = [];
|
||||||
|
|
||||||
|
const minCollisionDepthForSound = 2.5;
|
||||||
|
const maxCollisionDepthForSound = 9;
|
||||||
|
const soundPitchMax = 4;
|
||||||
|
const soundPitchMin = 0.5;
|
||||||
|
|
||||||
|
Matter.Events.on(this.engine, 'collisionStart', (event) => {
|
||||||
|
for (const pairs of event.pairs) {
|
||||||
|
const { bodyA, bodyB } = pairs;
|
||||||
|
if (bodyA.id === this.overflowCollider.id || bodyB.id === this.overflowCollider.id) {
|
||||||
|
if (bodyA.id === this.latestDroppedBodyId || bodyB.id === this.latestDroppedBodyId) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
this.gameOver();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
const shouldFusion = (bodyA.label === bodyB.label) && !fusionReservedPairs.some(x => x.bodyA.id === bodyA.id || x.bodyA.id === bodyB.id || x.bodyB.id === bodyA.id || x.bodyB.id === bodyB.id);
|
||||||
|
if (shouldFusion) {
|
||||||
|
if (this.activeBodyIds.includes(bodyA.id) && this.activeBodyIds.includes(bodyB.id)) {
|
||||||
|
this.fusion(bodyA, bodyB);
|
||||||
|
} else {
|
||||||
|
fusionReservedPairs.push({ bodyA, bodyB });
|
||||||
|
window.setTimeout(() => {
|
||||||
|
fusionReservedPairs = fusionReservedPairs.filter(x => x.bodyA.id !== bodyA.id && x.bodyB.id !== bodyB.id);
|
||||||
|
this.fusion(bodyA, bodyB);
|
||||||
|
}, 100);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
const energy = pairs.collision.depth;
|
||||||
|
if (energy > minCollisionDepthForSound) {
|
||||||
|
const vol = (Math.min(maxCollisionDepthForSound, energy - minCollisionDepthForSound) / maxCollisionDepthForSound) / 4;
|
||||||
|
const pan = ((((bodyA.position.x + bodyB.position.x) / 2) / GAME_WIDTH) - 0.5) * 2;
|
||||||
|
const pitch = soundPitchMin + ((soundPitchMax - soundPitchMin) * (1 - (Math.min(10, energy) / 10)));
|
||||||
|
sound.playRaw('syuilo/poi1', vol, pan, pitch);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
window.setInterval(() => {
|
||||||
|
if (this.latestFusionedAt < Date.now() - this.COMBO_INTERVAL) {
|
||||||
|
this.combo = 0;
|
||||||
|
}
|
||||||
|
}, 500);
|
||||||
|
}
|
||||||
|
|
||||||
|
public drop(_x: number) {
|
||||||
|
if (this.isGameOver) return;
|
||||||
|
if (Date.now() - this.latestDroppedAt < this.DROP_INTERVAL) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const st = this.stock.shift()!;
|
||||||
|
this.stock.push({
|
||||||
|
id: Math.random().toString(),
|
||||||
|
fruit: FRUITS.filter(x => x.available)[Math.floor(Math.random() * FRUITS.filter(x => x.available).length)],
|
||||||
|
});
|
||||||
|
this.emit('changeStock', this.stock);
|
||||||
|
|
||||||
|
const x = Math.min(GAME_WIDTH - this.PLAYAREA_MARGIN - (st.fruit.size / 2), Math.max(this.PLAYAREA_MARGIN + (st.fruit.size / 2), _x));
|
||||||
|
const body = this.createBody(st.fruit, x, st.fruit.size / 2);
|
||||||
|
Matter.Composite.add(this.engine.world, body);
|
||||||
|
this.activeBodyIds.push(body.id);
|
||||||
|
this.latestDroppedBodyId = body.id;
|
||||||
|
this.latestDroppedAt = Date.now();
|
||||||
|
this.emit('dropped');
|
||||||
|
const pan = ((x / GAME_WIDTH) - 0.5) * 2;
|
||||||
|
sound.playRaw('syuilo/poi2', 1, pan);
|
||||||
|
}
|
||||||
|
|
||||||
|
public dispose() {
|
||||||
|
Matter.Render.stop(this.render);
|
||||||
|
Matter.Runner.stop(this.runner);
|
||||||
|
Matter.World.clear(this.engine.world, false);
|
||||||
|
Matter.Engine.clear(this.engine);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let game: Game;
|
||||||
|
|
||||||
|
function onClick(ev: MouseEvent) {
|
||||||
|
const rect = containerEl.value.getBoundingClientRect();
|
||||||
|
|
||||||
|
const x = (ev.clientX - rect.left) / viewScaleX;
|
||||||
|
|
||||||
|
game.drop(x);
|
||||||
|
}
|
||||||
|
|
||||||
|
function onTouchend(ev: TouchEvent) {
|
||||||
|
const rect = containerEl.value.getBoundingClientRect();
|
||||||
|
|
||||||
|
const x = (ev.changedTouches[0].clientX - rect.left) / viewScaleX;
|
||||||
|
|
||||||
|
game.drop(x);
|
||||||
|
}
|
||||||
|
|
||||||
|
function onMousemove(ev: MouseEvent) {
|
||||||
|
mouseX.value = ev.clientX - containerEl.value.getBoundingClientRect().left;
|
||||||
|
}
|
||||||
|
|
||||||
|
function onTouchmove(ev: TouchEvent) {
|
||||||
|
mouseX.value = ev.touches[0].clientX - containerEl.value.getBoundingClientRect().left;
|
||||||
|
}
|
||||||
|
|
||||||
|
function restart() {
|
||||||
|
game.dispose();
|
||||||
|
gameOver.value = false;
|
||||||
|
currentPick.value = null;
|
||||||
|
dropReady.value = true;
|
||||||
|
stock.value = [];
|
||||||
|
score.value = 0;
|
||||||
|
combo.value = 0;
|
||||||
|
comboPrev.value = 0;
|
||||||
|
game = new Game();
|
||||||
|
attachGame();
|
||||||
|
game.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
function attachGame() {
|
||||||
|
game.addListener('changeScore', value => {
|
||||||
|
score.value = value;
|
||||||
|
});
|
||||||
|
|
||||||
|
game.addListener('changeCombo', value => {
|
||||||
|
if (value === 0) {
|
||||||
|
comboPrev.value = combo.value;
|
||||||
|
} else {
|
||||||
|
comboPrev.value = value;
|
||||||
|
}
|
||||||
|
combo.value = value;
|
||||||
|
});
|
||||||
|
|
||||||
|
game.addListener('changeStock', value => {
|
||||||
|
currentPick.value = JSON.parse(JSON.stringify(value[0]));
|
||||||
|
stock.value = JSON.parse(JSON.stringify(value.slice(1)));
|
||||||
|
});
|
||||||
|
|
||||||
|
game.addListener('dropped', () => {
|
||||||
|
dropReady.value = false;
|
||||||
|
window.setTimeout(() => {
|
||||||
|
if (!gameOver.value) {
|
||||||
|
dropReady.value = true;
|
||||||
|
}
|
||||||
|
}, game.DROP_INTERVAL);
|
||||||
|
});
|
||||||
|
|
||||||
|
game.addListener('fusioned', (x, y, score) => {
|
||||||
|
const rect = canvasEl.value.getBoundingClientRect();
|
||||||
|
const domX = rect.left + (x * viewScaleX);
|
||||||
|
const domY = rect.top + (y * viewScaleY);
|
||||||
|
os.popup(MkRippleEffect, { x: domX, y: domY }, {}, 'end');
|
||||||
|
os.popup(MkPlusOneEffect, { x: domX, y: domY, value: score }, {}, 'end');
|
||||||
|
});
|
||||||
|
|
||||||
|
game.addListener('gameOver', () => {
|
||||||
|
currentPick.value = null;
|
||||||
|
dropReady.value = false;
|
||||||
|
gameOver.value = true;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
game = new Game();
|
||||||
|
|
||||||
|
attachGame();
|
||||||
|
|
||||||
|
game.start();
|
||||||
|
|
||||||
|
const actualCanvasWidth = canvasEl.value.getBoundingClientRect().width;
|
||||||
|
const actualCanvasHeight = canvasEl.value.getBoundingClientRect().height;
|
||||||
|
viewScaleX = actualCanvasWidth / GAME_WIDTH;
|
||||||
|
viewScaleY = actualCanvasHeight / GAME_HEIGHT;
|
||||||
|
});
|
||||||
|
|
||||||
|
definePageMetadata({
|
||||||
|
title: 'Drop & Fusion',
|
||||||
|
icon: 'ti ti-apple',
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" module>
|
||||||
|
.transition_stock_move,
|
||||||
|
.transition_stock_enterActive,
|
||||||
|
.transition_stock_leaveActive {
|
||||||
|
transition: opacity 0.4s cubic-bezier(0,.5,.5,1), transform 0.4s cubic-bezier(0,.5,.5,1) !important;
|
||||||
|
}
|
||||||
|
.transition_stock_enterFrom,
|
||||||
|
.transition_stock_leaveTo {
|
||||||
|
opacity: 0;
|
||||||
|
transform: scale(0.7);
|
||||||
|
}
|
||||||
|
.transition_stock_leaveActive {
|
||||||
|
position: absolute;
|
||||||
|
}
|
||||||
|
|
||||||
|
.transition_picked_move,
|
||||||
|
.transition_picked_enterActive {
|
||||||
|
transition: opacity 0.5s cubic-bezier(0,.5,.5,1), transform 0.5s cubic-bezier(0,.5,.5,1) !important;
|
||||||
|
}
|
||||||
|
.transition_picked_leaveActive {
|
||||||
|
transition: all 0s !important;
|
||||||
|
}
|
||||||
|
.transition_picked_enterFrom,
|
||||||
|
.transition_picked_leaveTo {
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateY(-50px);
|
||||||
|
}
|
||||||
|
.transition_picked_leaveActive {
|
||||||
|
position: absolute;
|
||||||
|
}
|
||||||
|
|
||||||
|
.transition_combo_move,
|
||||||
|
.transition_combo_enterActive {
|
||||||
|
transition: all 0s !important;
|
||||||
|
}
|
||||||
|
.transition_combo_leaveActive {
|
||||||
|
transition: opacity 0.4s cubic-bezier(0,.5,.5,1), transform 0.4s cubic-bezier(0,.5,.5,1) !important;
|
||||||
|
}
|
||||||
|
.transition_combo_enterFrom,
|
||||||
|
.transition_combo_leaveTo {
|
||||||
|
opacity: 0;
|
||||||
|
transform: scale(0.7);
|
||||||
|
}
|
||||||
|
.transition_combo_leaveActive {
|
||||||
|
position: absolute;
|
||||||
|
}
|
||||||
|
|
||||||
|
.root {
|
||||||
|
user-select: none;
|
||||||
|
|
||||||
|
* {
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.frame {
|
||||||
|
padding: 7px;
|
||||||
|
background: #8C4F26;
|
||||||
|
box-shadow: 0 6px 16px #0007, 0 0 1px 1px #693410, inset 0 0 2px 1px #ce8a5c;
|
||||||
|
border-radius: 10px;
|
||||||
|
}
|
||||||
|
.frameInner {
|
||||||
|
padding: 4px 8px;
|
||||||
|
background: #F1E8DC;
|
||||||
|
box-shadow: 0 0 2px 1px #ce8a5c, inset 0 0 1px 1px #693410;
|
||||||
|
border-radius: 6px;
|
||||||
|
color: #693410;
|
||||||
|
}
|
||||||
|
|
||||||
|
.main {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mainFrameImg {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
filter: drop-shadow(0 6px 16px #0007);
|
||||||
|
pointer-events: none;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.canvas {
|
||||||
|
position: relative;
|
||||||
|
display: block;
|
||||||
|
z-index: 1;
|
||||||
|
margin-top: -50px;
|
||||||
|
max-width: 100%;
|
||||||
|
pointer-events: none;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.container {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.stock {
|
||||||
|
pointer-events: none;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.combo {
|
||||||
|
position: absolute;
|
||||||
|
z-index: 3;
|
||||||
|
top: 50%;
|
||||||
|
width: 100%;
|
||||||
|
text-align: center;
|
||||||
|
font-weight: bold;
|
||||||
|
font-style: oblique;
|
||||||
|
pointer-events: none;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.currentFruit {
|
||||||
|
position: absolute;
|
||||||
|
margin-top: 20px;
|
||||||
|
z-index: 2;
|
||||||
|
filter: drop-shadow(0 6px 16px #0007);
|
||||||
|
pointer-events: none;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.currentFruitArrow {
|
||||||
|
position: absolute;
|
||||||
|
margin-top: 20px;
|
||||||
|
z-index: 3;
|
||||||
|
animation: currentFruitArrow 2s ease infinite;
|
||||||
|
pointer-events: none;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropGuide {
|
||||||
|
position: absolute;
|
||||||
|
top: 50px;
|
||||||
|
z-index: 3;
|
||||||
|
width: 3px;
|
||||||
|
height: calc(100% - 50px);
|
||||||
|
background: #f002;
|
||||||
|
pointer-events: none;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.gameOverLabel {
|
||||||
|
position: absolute;
|
||||||
|
z-index: 10;
|
||||||
|
top: 50%;
|
||||||
|
width: 100%;
|
||||||
|
padding: 16px;
|
||||||
|
box-sizing: border-box;
|
||||||
|
background: #0007;
|
||||||
|
color: #fff;
|
||||||
|
text-align: center;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
.gameOver {
|
||||||
|
.canvas {
|
||||||
|
filter: grayscale(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes currentFruitArrow {
|
||||||
|
0% { transform: translateY(0); }
|
||||||
|
25% { transform: translateY(-8px); }
|
||||||
|
50% { transform: translateY(0); }
|
||||||
|
75% { transform: translateY(-8px); }
|
||||||
|
100% { transform: translateY(0); }
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -527,6 +527,10 @@ export const routes = [{
|
||||||
path: '/clicker',
|
path: '/clicker',
|
||||||
component: page(() => import('./pages/clicker.vue')),
|
component: page(() => import('./pages/clicker.vue')),
|
||||||
loginRequired: true,
|
loginRequired: true,
|
||||||
|
}, {
|
||||||
|
path: '/drop-and-fusion',
|
||||||
|
component: page(() => import('./pages/drop-and-fusion.vue')),
|
||||||
|
loginRequired: true,
|
||||||
}, {
|
}, {
|
||||||
path: '/timeline',
|
path: '/timeline',
|
||||||
component: page(() => import('./pages/timeline.vue')),
|
component: page(() => import('./pages/timeline.vue')),
|
||||||
|
|
|
@ -92,7 +92,13 @@ export type OperationType = typeof operationTypes[number];
|
||||||
* @param soundStore サウンド設定
|
* @param soundStore サウンド設定
|
||||||
* @param options `useCache`: デフォルトは`true` 一度再生した音声はキャッシュする
|
* @param options `useCache`: デフォルトは`true` 一度再生した音声はキャッシュする
|
||||||
*/
|
*/
|
||||||
export async function loadAudio(soundStore: SoundStore, options?: { useCache?: boolean; }) {
|
export async function loadAudio(soundStore: {
|
||||||
|
type: Exclude<SoundType, '_driveFile_'>;
|
||||||
|
} | {
|
||||||
|
type: '_driveFile_';
|
||||||
|
fileId: string;
|
||||||
|
fileUrl: string;
|
||||||
|
}, options?: { useCache?: boolean; }) {
|
||||||
if (_DEV_) console.log('loading audio. opts:', options);
|
if (_DEV_) console.log('loading audio. opts:', options);
|
||||||
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
|
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
|
||||||
if (soundStore.type === null || (soundStore.type === '_driveFile_' && !soundStore.fileUrl)) {
|
if (soundStore.type === null || (soundStore.type === '_driveFile_' && !soundStore.fileUrl)) {
|
||||||
|
@ -179,18 +185,31 @@ export async function playFile(soundStore: SoundStore) {
|
||||||
createSourceNode(buffer, soundStore.volume)?.start();
|
createSourceNode(buffer, soundStore.volume)?.start();
|
||||||
}
|
}
|
||||||
|
|
||||||
export function createSourceNode(buffer: AudioBuffer, volume: number) : AudioBufferSourceNode | null {
|
export async function playRaw(type: Exclude<SoundType, '_driveFile_'>, volume = 1, pan = 0, playbackRate = 1) {
|
||||||
|
const buffer = await loadAudio({ type });
|
||||||
|
if (!buffer) return;
|
||||||
|
createSourceNode(buffer, volume, pan, playbackRate)?.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
export function createSourceNode(buffer: AudioBuffer, volume: number, pan = 0, playbackRate = 1) : AudioBufferSourceNode | null {
|
||||||
const masterVolume = defaultStore.state.sound_masterVolume;
|
const masterVolume = defaultStore.state.sound_masterVolume;
|
||||||
if (isMute() || masterVolume === 0 || volume === 0) {
|
if (isMute() || masterVolume === 0 || volume === 0) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const panNode = ctx.createStereoPanner();
|
||||||
|
panNode.pan.value = pan;
|
||||||
|
|
||||||
const gainNode = ctx.createGain();
|
const gainNode = ctx.createGain();
|
||||||
gainNode.gain.value = masterVolume * volume;
|
gainNode.gain.value = masterVolume * volume;
|
||||||
|
|
||||||
const soundSource = ctx.createBufferSource();
|
const soundSource = ctx.createBufferSource();
|
||||||
soundSource.buffer = buffer;
|
soundSource.buffer = buffer;
|
||||||
soundSource.connect(gainNode).connect(ctx.destination);
|
soundSource.playbackRate.value = playbackRate;
|
||||||
|
soundSource
|
||||||
|
.connect(panNode)
|
||||||
|
.connect(gainNode)
|
||||||
|
.connect(ctx.destination);
|
||||||
|
|
||||||
return soundSource;
|
return soundSource;
|
||||||
}
|
}
|
||||||
|
|
|
@ -27,6 +27,11 @@ function toolsMenuItems(): MenuItem[] {
|
||||||
to: '/clicker',
|
to: '/clicker',
|
||||||
text: '🍪👈',
|
text: '🍪👈',
|
||||||
icon: 'ti ti-cookie',
|
icon: 'ti ti-cookie',
|
||||||
|
}, {
|
||||||
|
type: 'link',
|
||||||
|
to: '/drop-and-fusion',
|
||||||
|
text: 'Drop & Fusion',
|
||||||
|
icon: 'ti ti-apple',
|
||||||
}, ($i && ($i.isAdmin || $i.policies.canManageCustomEmojis)) ? {
|
}, ($i && ($i.isAdmin || $i.policies.canManageCustomEmojis)) ? {
|
||||||
type: 'link',
|
type: 'link',
|
||||||
to: '/custom-emojis-manager',
|
to: '/custom-emojis-manager',
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
/*
|
/*
|
||||||
* version: 2023.12.2
|
* version: 2023.12.2
|
||||||
* generatedAt: 2024-01-02T08:53:57.449Z
|
* generatedAt: 2024-01-04T18:10:15.096Z
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import type { SwitchCaseResponseType } from '../api.js';
|
import type { SwitchCaseResponseType } from '../api.js';
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
/*
|
/*
|
||||||
* version: 2023.12.2
|
* version: 2023.12.2
|
||||||
* generatedAt: 2024-01-02T08:53:57.445Z
|
* generatedAt: 2024-01-04T18:10:15.094Z
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import type {
|
import type {
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
/*
|
/*
|
||||||
* version: 2023.12.2
|
* version: 2023.12.2
|
||||||
* generatedAt: 2024-01-02T08:53:57.443Z
|
* generatedAt: 2024-01-04T18:10:15.093Z
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { operations } from './types.js';
|
import { operations } from './types.js';
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
/*
|
/*
|
||||||
* version: 2023.12.2
|
* version: 2023.12.2
|
||||||
* generatedAt: 2024-01-02T08:53:57.441Z
|
* generatedAt: 2024-01-04T18:10:15.091Z
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { components } from './types.js';
|
import { components } from './types.js';
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* version: 2023.12.2
|
* version: 2023.12.2
|
||||||
* generatedAt: 2024-01-02T08:53:56.447Z
|
* generatedAt: 2024-01-04T18:10:15.023Z
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -4400,6 +4400,9 @@ export type operations = {
|
||||||
emailRequiredForSignup: boolean;
|
emailRequiredForSignup: boolean;
|
||||||
enableHcaptcha: boolean;
|
enableHcaptcha: boolean;
|
||||||
hcaptchaSiteKey: string | null;
|
hcaptchaSiteKey: string | null;
|
||||||
|
enableMcaptcha: boolean;
|
||||||
|
mcaptchaSiteKey: string | null;
|
||||||
|
mcaptchaInstanceUrl: string | null;
|
||||||
enableRecaptcha: boolean;
|
enableRecaptcha: boolean;
|
||||||
recaptchaSiteKey: string | null;
|
recaptchaSiteKey: string | null;
|
||||||
enableTurnstile: boolean;
|
enableTurnstile: boolean;
|
||||||
|
@ -4425,6 +4428,7 @@ export type operations = {
|
||||||
bannedEmailDomains?: string[];
|
bannedEmailDomains?: string[];
|
||||||
preservedUsernames: string[];
|
preservedUsernames: string[];
|
||||||
hcaptchaSecretKey: string | null;
|
hcaptchaSecretKey: string | null;
|
||||||
|
mcaptchaSecretKey: string | null;
|
||||||
recaptchaSecretKey: string | null;
|
recaptchaSecretKey: string | null;
|
||||||
turnstileSecretKey: string | null;
|
turnstileSecretKey: string | null;
|
||||||
sensitiveMediaDetection: string;
|
sensitiveMediaDetection: string;
|
||||||
|
@ -8197,6 +8201,10 @@ export type operations = {
|
||||||
enableHcaptcha?: boolean;
|
enableHcaptcha?: boolean;
|
||||||
hcaptchaSiteKey?: string | null;
|
hcaptchaSiteKey?: string | null;
|
||||||
hcaptchaSecretKey?: string | null;
|
hcaptchaSecretKey?: string | null;
|
||||||
|
enableMcaptcha?: boolean;
|
||||||
|
mcaptchaSiteKey?: string | null;
|
||||||
|
mcaptchaInstanceUrl?: string | null;
|
||||||
|
mcaptchaSecretKey?: string | null;
|
||||||
enableRecaptcha?: boolean;
|
enableRecaptcha?: boolean;
|
||||||
recaptchaSiteKey?: string | null;
|
recaptchaSiteKey?: string | null;
|
||||||
recaptchaSecretKey?: string | null;
|
recaptchaSecretKey?: string | null;
|
||||||
|
@ -18704,6 +18712,9 @@ export type operations = {
|
||||||
emailRequiredForSignup: boolean;
|
emailRequiredForSignup: boolean;
|
||||||
enableHcaptcha: boolean;
|
enableHcaptcha: boolean;
|
||||||
hcaptchaSiteKey: string | null;
|
hcaptchaSiteKey: string | null;
|
||||||
|
enableMcaptcha: boolean;
|
||||||
|
mcaptchaSiteKey: string | null;
|
||||||
|
mcaptchaInstanceUrl: string | null;
|
||||||
enableRecaptcha: boolean;
|
enableRecaptcha: boolean;
|
||||||
recaptchaSiteKey: string | null;
|
recaptchaSiteKey: string | null;
|
||||||
enableTurnstile: boolean;
|
enableTurnstile: boolean;
|
||||||
|
|
101
pnpm-lock.yaml
|
@ -661,6 +661,9 @@ importers:
|
||||||
'@github/webauthn-json':
|
'@github/webauthn-json':
|
||||||
specifier: 2.1.1
|
specifier: 2.1.1
|
||||||
version: 2.1.1
|
version: 2.1.1
|
||||||
|
'@mcaptcha/vanilla-glue':
|
||||||
|
specifier: 0.1.0-alpha-3
|
||||||
|
version: 0.1.0-alpha-3
|
||||||
'@misskey-dev/browser-image-resizer':
|
'@misskey-dev/browser-image-resizer':
|
||||||
specifier: 2.2.1-misskey.10
|
specifier: 2.2.1-misskey.10
|
||||||
version: 2.2.1-misskey.10
|
version: 2.2.1-misskey.10
|
||||||
|
@ -1820,7 +1823,7 @@ packages:
|
||||||
'@babel/traverse': 7.22.11
|
'@babel/traverse': 7.22.11
|
||||||
'@babel/types': 7.22.17
|
'@babel/types': 7.22.17
|
||||||
convert-source-map: 1.9.0
|
convert-source-map: 1.9.0
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
gensync: 1.0.0-beta.2
|
gensync: 1.0.0-beta.2
|
||||||
json5: 2.2.3
|
json5: 2.2.3
|
||||||
semver: 6.3.1
|
semver: 6.3.1
|
||||||
|
@ -1843,7 +1846,7 @@ packages:
|
||||||
'@babel/traverse': 7.23.5
|
'@babel/traverse': 7.23.5
|
||||||
'@babel/types': 7.23.5
|
'@babel/types': 7.23.5
|
||||||
convert-source-map: 2.0.0
|
convert-source-map: 2.0.0
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
gensync: 1.0.0-beta.2
|
gensync: 1.0.0-beta.2
|
||||||
json5: 2.2.3
|
json5: 2.2.3
|
||||||
semver: 6.3.1
|
semver: 6.3.1
|
||||||
|
@ -1945,7 +1948,7 @@ packages:
|
||||||
'@babel/core': 7.23.5
|
'@babel/core': 7.23.5
|
||||||
'@babel/helper-compilation-targets': 7.22.15
|
'@babel/helper-compilation-targets': 7.22.15
|
||||||
'@babel/helper-plugin-utils': 7.22.5
|
'@babel/helper-plugin-utils': 7.22.5
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
lodash.debounce: 4.0.8
|
lodash.debounce: 4.0.8
|
||||||
resolve: 1.22.8
|
resolve: 1.22.8
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
|
@ -3345,7 +3348,7 @@ packages:
|
||||||
'@babel/helper-split-export-declaration': 7.22.6
|
'@babel/helper-split-export-declaration': 7.22.6
|
||||||
'@babel/parser': 7.23.5
|
'@babel/parser': 7.23.5
|
||||||
'@babel/types': 7.22.17
|
'@babel/types': 7.22.17
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
globals: 11.12.0
|
globals: 11.12.0
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
|
@ -3363,7 +3366,7 @@ packages:
|
||||||
'@babel/helper-split-export-declaration': 7.22.6
|
'@babel/helper-split-export-declaration': 7.22.6
|
||||||
'@babel/parser': 7.23.5
|
'@babel/parser': 7.23.5
|
||||||
'@babel/types': 7.23.5
|
'@babel/types': 7.23.5
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
globals: 11.12.0
|
globals: 11.12.0
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
|
@ -4242,7 +4245,7 @@ packages:
|
||||||
engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
|
engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
|
||||||
dependencies:
|
dependencies:
|
||||||
ajv: 6.12.6
|
ajv: 6.12.6
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
espree: 9.6.1
|
espree: 9.6.1
|
||||||
globals: 13.19.0
|
globals: 13.19.0
|
||||||
ignore: 5.2.4
|
ignore: 5.2.4
|
||||||
|
@ -4259,7 +4262,7 @@ packages:
|
||||||
engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
|
engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
|
||||||
dependencies:
|
dependencies:
|
||||||
ajv: 6.12.6
|
ajv: 6.12.6
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
espree: 9.6.1
|
espree: 9.6.1
|
||||||
globals: 13.19.0
|
globals: 13.19.0
|
||||||
ignore: 5.2.4
|
ignore: 5.2.4
|
||||||
|
@ -4524,7 +4527,7 @@ packages:
|
||||||
engines: {node: '>=10.10.0'}
|
engines: {node: '>=10.10.0'}
|
||||||
dependencies:
|
dependencies:
|
||||||
'@humanwhocodes/object-schema': 2.0.1
|
'@humanwhocodes/object-schema': 2.0.1
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
minimatch: 3.1.2
|
minimatch: 3.1.2
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
|
@ -4902,6 +4905,16 @@ packages:
|
||||||
dev: false
|
dev: false
|
||||||
optional: true
|
optional: true
|
||||||
|
|
||||||
|
/@mcaptcha/core-glue@0.1.0-alpha-5:
|
||||||
|
resolution: {integrity: sha512-16qWm5O5X0Y9LXULULaAks8Vf9FNlUUBcR5KDt49aWhFhG5++JzxNmCwQM9EJSHNU7y0U+FdyAWcGmjfKlkRLA==}
|
||||||
|
dev: false
|
||||||
|
|
||||||
|
/@mcaptcha/vanilla-glue@0.1.0-alpha-3:
|
||||||
|
resolution: {integrity: sha512-GT6TJBgmViGXcXiT5VOr+h/6iOnThSlZuCoOWncubyTZU9R3cgU5vWPkF7G6Ob6ee2CBe3yqBxxk24CFVGTVXw==}
|
||||||
|
dependencies:
|
||||||
|
'@mcaptcha/core-glue': 0.1.0-alpha-5
|
||||||
|
dev: false
|
||||||
|
|
||||||
/@mdx-js/react@2.3.0(react@18.2.0):
|
/@mdx-js/react@2.3.0(react@18.2.0):
|
||||||
resolution: {integrity: sha512-zQH//gdOmuu7nt2oJR29vFhDv88oGPmVw6BggmrHeMI+xgEkp1B2dX9/bMBSYtK0dyLX/aOmesKS09g222K1/g==}
|
resolution: {integrity: sha512-zQH//gdOmuu7nt2oJR29vFhDv88oGPmVw6BggmrHeMI+xgEkp1B2dX9/bMBSYtK0dyLX/aOmesKS09g222K1/g==}
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
|
@ -5084,7 +5097,7 @@ packages:
|
||||||
'@open-draft/until': 1.0.3
|
'@open-draft/until': 1.0.3
|
||||||
'@types/debug': 4.1.7
|
'@types/debug': 4.1.7
|
||||||
'@xmldom/xmldom': 0.8.6
|
'@xmldom/xmldom': 0.8.6
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
headers-polyfill: 3.2.5
|
headers-polyfill: 3.2.5
|
||||||
outvariant: 1.4.0
|
outvariant: 1.4.0
|
||||||
strict-event-emitter: 0.2.8
|
strict-event-emitter: 0.2.8
|
||||||
|
@ -7365,7 +7378,7 @@ packages:
|
||||||
hasBin: true
|
hasBin: true
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
'@swc/core': ^1.2.66
|
'@swc/core': ^1.2.66
|
||||||
chokidar: 3.5.3
|
chokidar: ^3.5.1
|
||||||
peerDependenciesMeta:
|
peerDependenciesMeta:
|
||||||
chokidar:
|
chokidar:
|
||||||
optional: true
|
optional: true
|
||||||
|
@ -8493,7 +8506,7 @@ packages:
|
||||||
'@typescript-eslint/type-utils': 6.11.0(eslint@8.53.0)(typescript@5.3.3)
|
'@typescript-eslint/type-utils': 6.11.0(eslint@8.53.0)(typescript@5.3.3)
|
||||||
'@typescript-eslint/utils': 6.11.0(eslint@8.53.0)(typescript@5.3.3)
|
'@typescript-eslint/utils': 6.11.0(eslint@8.53.0)(typescript@5.3.3)
|
||||||
'@typescript-eslint/visitor-keys': 6.11.0
|
'@typescript-eslint/visitor-keys': 6.11.0
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
eslint: 8.53.0
|
eslint: 8.53.0
|
||||||
graphemer: 1.4.0
|
graphemer: 1.4.0
|
||||||
ignore: 5.2.4
|
ignore: 5.2.4
|
||||||
|
@ -8522,7 +8535,7 @@ packages:
|
||||||
'@typescript-eslint/type-utils': 6.14.0(eslint@8.56.0)(typescript@5.3.3)
|
'@typescript-eslint/type-utils': 6.14.0(eslint@8.56.0)(typescript@5.3.3)
|
||||||
'@typescript-eslint/utils': 6.14.0(eslint@8.56.0)(typescript@5.3.3)
|
'@typescript-eslint/utils': 6.14.0(eslint@8.56.0)(typescript@5.3.3)
|
||||||
'@typescript-eslint/visitor-keys': 6.14.0
|
'@typescript-eslint/visitor-keys': 6.14.0
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
eslint: 8.56.0
|
eslint: 8.56.0
|
||||||
graphemer: 1.4.0
|
graphemer: 1.4.0
|
||||||
ignore: 5.2.4
|
ignore: 5.2.4
|
||||||
|
@ -8548,7 +8561,7 @@ packages:
|
||||||
'@typescript-eslint/types': 6.11.0
|
'@typescript-eslint/types': 6.11.0
|
||||||
'@typescript-eslint/typescript-estree': 6.11.0(typescript@5.3.3)
|
'@typescript-eslint/typescript-estree': 6.11.0(typescript@5.3.3)
|
||||||
'@typescript-eslint/visitor-keys': 6.11.0
|
'@typescript-eslint/visitor-keys': 6.11.0
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
eslint: 8.53.0
|
eslint: 8.53.0
|
||||||
typescript: 5.3.3
|
typescript: 5.3.3
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
|
@ -8569,7 +8582,7 @@ packages:
|
||||||
'@typescript-eslint/types': 6.14.0
|
'@typescript-eslint/types': 6.14.0
|
||||||
'@typescript-eslint/typescript-estree': 6.14.0(typescript@5.3.3)
|
'@typescript-eslint/typescript-estree': 6.14.0(typescript@5.3.3)
|
||||||
'@typescript-eslint/visitor-keys': 6.14.0
|
'@typescript-eslint/visitor-keys': 6.14.0
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
eslint: 8.56.0
|
eslint: 8.56.0
|
||||||
typescript: 5.3.3
|
typescript: 5.3.3
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
|
@ -8604,7 +8617,7 @@ packages:
|
||||||
dependencies:
|
dependencies:
|
||||||
'@typescript-eslint/typescript-estree': 6.11.0(typescript@5.3.3)
|
'@typescript-eslint/typescript-estree': 6.11.0(typescript@5.3.3)
|
||||||
'@typescript-eslint/utils': 6.11.0(eslint@8.53.0)(typescript@5.3.3)
|
'@typescript-eslint/utils': 6.11.0(eslint@8.53.0)(typescript@5.3.3)
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
eslint: 8.53.0
|
eslint: 8.53.0
|
||||||
ts-api-utils: 1.0.1(typescript@5.3.3)
|
ts-api-utils: 1.0.1(typescript@5.3.3)
|
||||||
typescript: 5.3.3
|
typescript: 5.3.3
|
||||||
|
@ -8624,7 +8637,7 @@ packages:
|
||||||
dependencies:
|
dependencies:
|
||||||
'@typescript-eslint/typescript-estree': 6.14.0(typescript@5.3.3)
|
'@typescript-eslint/typescript-estree': 6.14.0(typescript@5.3.3)
|
||||||
'@typescript-eslint/utils': 6.14.0(eslint@8.56.0)(typescript@5.3.3)
|
'@typescript-eslint/utils': 6.14.0(eslint@8.56.0)(typescript@5.3.3)
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
eslint: 8.56.0
|
eslint: 8.56.0
|
||||||
ts-api-utils: 1.0.1(typescript@5.3.3)
|
ts-api-utils: 1.0.1(typescript@5.3.3)
|
||||||
typescript: 5.3.3
|
typescript: 5.3.3
|
||||||
|
@ -8653,7 +8666,7 @@ packages:
|
||||||
dependencies:
|
dependencies:
|
||||||
'@typescript-eslint/types': 6.11.0
|
'@typescript-eslint/types': 6.11.0
|
||||||
'@typescript-eslint/visitor-keys': 6.11.0
|
'@typescript-eslint/visitor-keys': 6.11.0
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
globby: 11.1.0
|
globby: 11.1.0
|
||||||
is-glob: 4.0.3
|
is-glob: 4.0.3
|
||||||
semver: 7.5.4
|
semver: 7.5.4
|
||||||
|
@ -8674,7 +8687,7 @@ packages:
|
||||||
dependencies:
|
dependencies:
|
||||||
'@typescript-eslint/types': 6.14.0
|
'@typescript-eslint/types': 6.14.0
|
||||||
'@typescript-eslint/visitor-keys': 6.14.0
|
'@typescript-eslint/visitor-keys': 6.14.0
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
globby: 11.1.0
|
globby: 11.1.0
|
||||||
is-glob: 4.0.3
|
is-glob: 4.0.3
|
||||||
semver: 7.5.4
|
semver: 7.5.4
|
||||||
|
@ -9131,7 +9144,7 @@ packages:
|
||||||
engines: {node: '>= 6.0.0'}
|
engines: {node: '>= 6.0.0'}
|
||||||
requiresBuild: true
|
requiresBuild: true
|
||||||
dependencies:
|
dependencies:
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
|
|
||||||
|
@ -9139,7 +9152,7 @@ packages:
|
||||||
resolution: {integrity: sha512-o/zjMZRhJxny7OyEF+Op8X+efiELC7k7yOjMzgfzVqOzXqkBkWI79YoTdOtsuWd5BWhAGAuOY/Xa6xpiaWXiNg==}
|
resolution: {integrity: sha512-o/zjMZRhJxny7OyEF+Op8X+efiELC7k7yOjMzgfzVqOzXqkBkWI79YoTdOtsuWd5BWhAGAuOY/Xa6xpiaWXiNg==}
|
||||||
engines: {node: '>= 14'}
|
engines: {node: '>= 14'}
|
||||||
dependencies:
|
dependencies:
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
dev: false
|
dev: false
|
||||||
|
@ -9514,7 +9527,7 @@ packages:
|
||||||
resolution: {integrity: sha512-TAlMYvOuwGyLK3PfBb5WKBXZmXz2fVCgv23d6zZFdle/q3gPjmxBaeuC0pY0Dzs5PWMSgfqqEZkrye19GlDTgw==}
|
resolution: {integrity: sha512-TAlMYvOuwGyLK3PfBb5WKBXZmXz2fVCgv23d6zZFdle/q3gPjmxBaeuC0pY0Dzs5PWMSgfqqEZkrye19GlDTgw==}
|
||||||
dependencies:
|
dependencies:
|
||||||
archy: 1.0.0
|
archy: 1.0.0
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
fastq: 1.15.0
|
fastq: 1.15.0
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
|
@ -10948,7 +10961,6 @@ packages:
|
||||||
dependencies:
|
dependencies:
|
||||||
ms: 2.1.2
|
ms: 2.1.2
|
||||||
supports-color: 5.5.0
|
supports-color: 5.5.0
|
||||||
dev: true
|
|
||||||
|
|
||||||
/debug@4.3.4(supports-color@8.1.1):
|
/debug@4.3.4(supports-color@8.1.1):
|
||||||
resolution: {integrity: sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==}
|
resolution: {integrity: sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==}
|
||||||
|
@ -10961,6 +10973,7 @@ packages:
|
||||||
dependencies:
|
dependencies:
|
||||||
ms: 2.1.2
|
ms: 2.1.2
|
||||||
supports-color: 8.1.1
|
supports-color: 8.1.1
|
||||||
|
dev: true
|
||||||
|
|
||||||
/decamelize-keys@1.1.1:
|
/decamelize-keys@1.1.1:
|
||||||
resolution: {integrity: sha512-WiPxgEirIV0/eIOMcnFBA3/IJZAZqKnwAwWyvvdi4lsr1WCN22nhdf/3db3DoZcUjTV2SqfzIwNyp6y2xs3nmg==}
|
resolution: {integrity: sha512-WiPxgEirIV0/eIOMcnFBA3/IJZAZqKnwAwWyvvdi4lsr1WCN22nhdf/3db3DoZcUjTV2SqfzIwNyp6y2xs3nmg==}
|
||||||
|
@ -11177,7 +11190,7 @@ packages:
|
||||||
hasBin: true
|
hasBin: true
|
||||||
dependencies:
|
dependencies:
|
||||||
address: 1.2.2
|
address: 1.2.2
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
dev: true
|
dev: true
|
||||||
|
@ -11501,7 +11514,7 @@ packages:
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
esbuild: '>=0.12 <1'
|
esbuild: '>=0.12 <1'
|
||||||
dependencies:
|
dependencies:
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
esbuild: 0.18.20
|
esbuild: 0.18.20
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
|
@ -11840,7 +11853,7 @@ packages:
|
||||||
ajv: 6.12.6
|
ajv: 6.12.6
|
||||||
chalk: 4.1.2
|
chalk: 4.1.2
|
||||||
cross-spawn: 7.0.3
|
cross-spawn: 7.0.3
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
doctrine: 3.0.0
|
doctrine: 3.0.0
|
||||||
escape-string-regexp: 4.0.0
|
escape-string-regexp: 4.0.0
|
||||||
eslint-scope: 7.2.2
|
eslint-scope: 7.2.2
|
||||||
|
@ -11887,7 +11900,7 @@ packages:
|
||||||
ajv: 6.12.6
|
ajv: 6.12.6
|
||||||
chalk: 4.1.2
|
chalk: 4.1.2
|
||||||
cross-spawn: 7.0.3
|
cross-spawn: 7.0.3
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
doctrine: 3.0.0
|
doctrine: 3.0.0
|
||||||
escape-string-regexp: 4.0.0
|
escape-string-regexp: 4.0.0
|
||||||
eslint-scope: 7.2.2
|
eslint-scope: 7.2.2
|
||||||
|
@ -12491,7 +12504,7 @@ packages:
|
||||||
debug:
|
debug:
|
||||||
optional: true
|
optional: true
|
||||||
dependencies:
|
dependencies:
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
|
|
||||||
/for-each@0.3.3:
|
/for-each@0.3.3:
|
||||||
resolution: {integrity: sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==}
|
resolution: {integrity: sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==}
|
||||||
|
@ -13043,7 +13056,6 @@ packages:
|
||||||
/has-flag@3.0.0:
|
/has-flag@3.0.0:
|
||||||
resolution: {integrity: sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==}
|
resolution: {integrity: sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==}
|
||||||
engines: {node: '>=4'}
|
engines: {node: '>=4'}
|
||||||
dev: true
|
|
||||||
|
|
||||||
/has-flag@4.0.0:
|
/has-flag@4.0.0:
|
||||||
resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==}
|
resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==}
|
||||||
|
@ -13181,7 +13193,7 @@ packages:
|
||||||
engines: {node: '>= 14'}
|
engines: {node: '>= 14'}
|
||||||
dependencies:
|
dependencies:
|
||||||
agent-base: 7.1.0
|
agent-base: 7.1.0
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
dev: false
|
dev: false
|
||||||
|
@ -13243,7 +13255,7 @@ packages:
|
||||||
engines: {node: '>= 6.0.0'}
|
engines: {node: '>= 6.0.0'}
|
||||||
dependencies:
|
dependencies:
|
||||||
agent-base: 5.1.1
|
agent-base: 5.1.1
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
dev: true
|
dev: true
|
||||||
|
@ -13253,7 +13265,7 @@ packages:
|
||||||
engines: {node: '>= 6'}
|
engines: {node: '>= 6'}
|
||||||
dependencies:
|
dependencies:
|
||||||
agent-base: 6.0.2
|
agent-base: 6.0.2
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
|
|
||||||
|
@ -13262,7 +13274,7 @@ packages:
|
||||||
engines: {node: '>= 14'}
|
engines: {node: '>= 14'}
|
||||||
dependencies:
|
dependencies:
|
||||||
agent-base: 7.1.0
|
agent-base: 7.1.0
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
dev: false
|
dev: false
|
||||||
|
@ -13272,7 +13284,7 @@ packages:
|
||||||
engines: {node: '>= 14'}
|
engines: {node: '>= 14'}
|
||||||
dependencies:
|
dependencies:
|
||||||
agent-base: 7.1.0
|
agent-base: 7.1.0
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
dev: false
|
dev: false
|
||||||
|
@ -13422,7 +13434,7 @@ packages:
|
||||||
dependencies:
|
dependencies:
|
||||||
'@ioredis/commands': 1.2.0
|
'@ioredis/commands': 1.2.0
|
||||||
cluster-key-slot: 1.1.2
|
cluster-key-slot: 1.1.2
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
denque: 2.1.0
|
denque: 2.1.0
|
||||||
lodash.defaults: 4.2.0
|
lodash.defaults: 4.2.0
|
||||||
lodash.isarguments: 3.1.0
|
lodash.isarguments: 3.1.0
|
||||||
|
@ -13863,7 +13875,7 @@ packages:
|
||||||
resolution: {integrity: sha512-n3s8EwkdFIJCG3BPKBYvskgXGoy88ARzvegkitk60NxRdwltLOTaH7CUiMRXvwYorl0Q712iEjcWB+fK/MrWVw==}
|
resolution: {integrity: sha512-n3s8EwkdFIJCG3BPKBYvskgXGoy88ARzvegkitk60NxRdwltLOTaH7CUiMRXvwYorl0Q712iEjcWB+fK/MrWVw==}
|
||||||
engines: {node: '>=10'}
|
engines: {node: '>=10'}
|
||||||
dependencies:
|
dependencies:
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
istanbul-lib-coverage: 3.2.0
|
istanbul-lib-coverage: 3.2.0
|
||||||
source-map: 0.6.1
|
source-map: 0.6.1
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
|
@ -14541,7 +14553,7 @@ packages:
|
||||||
resolution: {integrity: sha512-pJ4XLQP4Q9HTxl6RVDLJ8Cyh1uitSs0CzDBAz1uoJ4sRD/Bk7cFSXL1FUXDW3zJ7YnfliJx6eu8Jn283bpZ4Yg==}
|
resolution: {integrity: sha512-pJ4XLQP4Q9HTxl6RVDLJ8Cyh1uitSs0CzDBAz1uoJ4sRD/Bk7cFSXL1FUXDW3zJ7YnfliJx6eu8Jn283bpZ4Yg==}
|
||||||
engines: {node: '>=10'}
|
engines: {node: '>=10'}
|
||||||
dependencies:
|
dependencies:
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
rfdc: 1.3.0
|
rfdc: 1.3.0
|
||||||
uri-js: 4.4.1
|
uri-js: 4.4.1
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
|
@ -17109,7 +17121,7 @@ packages:
|
||||||
engines: {node: '>=8.16.0'}
|
engines: {node: '>=8.16.0'}
|
||||||
dependencies:
|
dependencies:
|
||||||
'@types/mime-types': 2.1.4
|
'@types/mime-types': 2.1.4
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
extract-zip: 1.7.0
|
extract-zip: 1.7.0
|
||||||
https-proxy-agent: 4.0.0
|
https-proxy-agent: 4.0.0
|
||||||
mime: 2.6.0
|
mime: 2.6.0
|
||||||
|
@ -18108,7 +18120,7 @@ packages:
|
||||||
dependencies:
|
dependencies:
|
||||||
'@hapi/hoek': 10.0.1
|
'@hapi/hoek': 10.0.1
|
||||||
'@hapi/wreck': 18.0.1
|
'@hapi/wreck': 18.0.1
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
joi: 17.7.0
|
joi: 17.7.0
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
|
@ -18308,7 +18320,7 @@ packages:
|
||||||
engines: {node: '>= 14'}
|
engines: {node: '>= 14'}
|
||||||
dependencies:
|
dependencies:
|
||||||
agent-base: 7.1.0
|
agent-base: 7.1.0
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
socks: 2.7.1
|
socks: 2.7.1
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
|
@ -18461,7 +18473,7 @@ packages:
|
||||||
arg: 5.0.2
|
arg: 5.0.2
|
||||||
bluebird: 3.7.2
|
bluebird: 3.7.2
|
||||||
check-more-types: 2.24.0
|
check-more-types: 2.24.0
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
execa: 5.1.1
|
execa: 5.1.1
|
||||||
lazy-ass: 1.6.0
|
lazy-ass: 1.6.0
|
||||||
ps-tree: 1.2.0
|
ps-tree: 1.2.0
|
||||||
|
@ -18726,7 +18738,6 @@ packages:
|
||||||
engines: {node: '>=4'}
|
engines: {node: '>=4'}
|
||||||
dependencies:
|
dependencies:
|
||||||
has-flag: 3.0.0
|
has-flag: 3.0.0
|
||||||
dev: true
|
|
||||||
|
|
||||||
/supports-color@7.2.0:
|
/supports-color@7.2.0:
|
||||||
resolution: {integrity: sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==}
|
resolution: {integrity: sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==}
|
||||||
|
@ -19343,7 +19354,7 @@ packages:
|
||||||
chalk: 4.1.2
|
chalk: 4.1.2
|
||||||
cli-highlight: 2.1.11
|
cli-highlight: 2.1.11
|
||||||
date-fns: 2.30.0
|
date-fns: 2.30.0
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
dotenv: 16.0.3
|
dotenv: 16.0.3
|
||||||
glob: 8.1.0
|
glob: 8.1.0
|
||||||
ioredis: 5.3.2
|
ioredis: 5.3.2
|
||||||
|
@ -19701,7 +19712,7 @@ packages:
|
||||||
hasBin: true
|
hasBin: true
|
||||||
dependencies:
|
dependencies:
|
||||||
cac: 6.7.14
|
cac: 6.7.14
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
mlly: 1.4.0
|
mlly: 1.4.0
|
||||||
pathe: 1.1.1
|
pathe: 1.1.1
|
||||||
picocolors: 1.0.0
|
picocolors: 1.0.0
|
||||||
|
@ -19813,7 +19824,7 @@ packages:
|
||||||
acorn-walk: 8.2.0
|
acorn-walk: 8.2.0
|
||||||
cac: 6.7.14
|
cac: 6.7.14
|
||||||
chai: 4.3.10
|
chai: 4.3.10
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
happy-dom: 10.0.3
|
happy-dom: 10.0.3
|
||||||
local-pkg: 0.4.3
|
local-pkg: 0.4.3
|
||||||
magic-string: 0.30.3
|
magic-string: 0.30.3
|
||||||
|
@ -19895,7 +19906,7 @@ packages:
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
eslint: '>=6.0.0'
|
eslint: '>=6.0.0'
|
||||||
dependencies:
|
dependencies:
|
||||||
debug: 4.3.4(supports-color@8.1.1)
|
debug: 4.3.4(supports-color@5.5.0)
|
||||||
eslint: 8.56.0
|
eslint: 8.56.0
|
||||||
eslint-scope: 7.2.2
|
eslint-scope: 7.2.2
|
||||||
eslint-visitor-keys: 3.4.3
|
eslint-visitor-keys: 3.4.3
|
||||||
|
|