[feat] multiple cards import
This commit is contained in:
@@ -54,7 +54,7 @@
|
|||||||
}
|
}
|
||||||
async function createImport() {
|
async function createImport() {
|
||||||
reseter();
|
reseter();
|
||||||
const cid = await importCharacter();
|
await importCharacter();
|
||||||
selectedCharID.set(-1);
|
selectedCharID.set(-1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { get } from "svelte/store"
|
import { get } from "svelte/store"
|
||||||
import { alertConfirm, alertError, alertNormal, alertSelect, alertStore } from "./alert"
|
import { alertConfirm, alertError, alertNormal, alertSelect, alertStore } from "./alert"
|
||||||
import { DataBase, defaultSdDataFunc, type character, setDatabase, type customscript, type loreSettings, type loreBook } from "./database"
|
import { DataBase, defaultSdDataFunc, type character, setDatabase, type customscript, type loreSettings, type loreBook } from "./database"
|
||||||
import { checkNullish, selectSingleFile, sleep } from "./util"
|
import { checkNullish, selectMultipleFile, selectSingleFile, sleep } from "./util"
|
||||||
import { language } from "src/lang"
|
import { language } from "src/lang"
|
||||||
import { encode as encodeMsgpack, decode as decodeMsgpack } from "@msgpack/msgpack";
|
import { encode as encodeMsgpack, decode as decodeMsgpack } from "@msgpack/msgpack";
|
||||||
import { v4 as uuidv4 } from 'uuid';
|
import { v4 as uuidv4 } from 'uuid';
|
||||||
@@ -14,96 +14,13 @@ import { cloneDeep } from "lodash"
|
|||||||
|
|
||||||
export async function importCharacter() {
|
export async function importCharacter() {
|
||||||
try {
|
try {
|
||||||
const f = await selectSingleFile(['png', 'json'])
|
const files = await selectMultipleFile(['png', 'json'])
|
||||||
if(!f){
|
if(!files){
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if(f.name.endsWith('json')){
|
|
||||||
const da = JSON.parse(Buffer.from(f.data).toString('utf-8'))
|
|
||||||
if(await importSpecv2(da)){
|
|
||||||
let db = get(DataBase)
|
|
||||||
return db.characters.length - 1
|
|
||||||
}
|
|
||||||
if((da.char_name || da.name) && (da.char_persona || da.description) && (da.char_greeting || da.first_mes)){
|
|
||||||
let db = get(DataBase)
|
|
||||||
db.characters.push(convertOldTavernAndJSON(da))
|
|
||||||
DataBase.set(db)
|
|
||||||
alertNormal(language.importedCharacter)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
else{
|
|
||||||
alertError(language.errors.noData)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
alertStore.set({
|
|
||||||
type: 'wait',
|
|
||||||
msg: 'Loading... (Reading)'
|
|
||||||
})
|
|
||||||
await sleep(10)
|
|
||||||
const img = f.data
|
|
||||||
const readed = (await exifr.parse(img, true))
|
|
||||||
if(readed.chara){
|
|
||||||
// standard spec v2 imports
|
|
||||||
const charaData:CharacterCardV2 = JSON.parse(Buffer.from(readed.chara, 'base64').toString('utf-8'))
|
|
||||||
if(await importSpecv2(charaData, img)){
|
|
||||||
let db = get(DataBase)
|
|
||||||
return db.characters.length - 1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if(readed.risuai){
|
|
||||||
// old risu imports
|
|
||||||
await sleep(10)
|
|
||||||
const va = decodeMsgpack(Buffer.from(readed.risuai, 'base64')) as any
|
|
||||||
if(va.type !== 101){
|
|
||||||
alertError(language.errors.noData)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
let char:character = va.data
|
for(const f of files){
|
||||||
let db = get(DataBase)
|
await importCharacterProcess(f)
|
||||||
if(char.emotionImages && char.emotionImages.length > 0){
|
|
||||||
for(let i=0;i<char.emotionImages.length;i++){
|
|
||||||
alertStore.set({
|
|
||||||
type: 'wait',
|
|
||||||
msg: `Loading... (Getting Emotions ${i} / ${char.emotionImages.length})`
|
|
||||||
})
|
|
||||||
await sleep(10)
|
|
||||||
const imgp = await saveAsset(char.emotionImages[i][1] as any)
|
|
||||||
char.emotionImages[i][1] = imgp
|
|
||||||
}
|
|
||||||
}
|
|
||||||
char.chats = [{
|
|
||||||
message: [],
|
|
||||||
note: '',
|
|
||||||
name: 'Chat 1',
|
|
||||||
localLore: []
|
|
||||||
}]
|
|
||||||
|
|
||||||
if(checkNullish(char.sdData)){
|
|
||||||
char.sdData = defaultSdDataFunc()
|
|
||||||
}
|
|
||||||
|
|
||||||
char.chatPage = 0
|
|
||||||
char.image = await saveAsset(PngMetadata.filter(img))
|
|
||||||
db.characters.push(characterFormatUpdate(char))
|
|
||||||
char.chaId = uuidv4()
|
|
||||||
setDatabase(db)
|
|
||||||
alertNormal(language.importedCharacter)
|
|
||||||
return db.characters.length - 1
|
|
||||||
}
|
|
||||||
else if(readed.chara){
|
|
||||||
const charaData:OldTavernChar = JSON.parse(Buffer.from(readed.chara, 'base64').toString('utf-8'))
|
|
||||||
const imgp = await saveAsset(PngMetadata.filter(img))
|
|
||||||
let db = get(DataBase)
|
|
||||||
db.characters.push(convertOldTavernAndJSON(charaData, imgp))
|
|
||||||
DataBase.set(db)
|
|
||||||
alertNormal(language.importedCharacter)
|
|
||||||
return db.characters.length - 1
|
|
||||||
}
|
|
||||||
else{
|
|
||||||
alertError(language.errors.noData)
|
|
||||||
return null
|
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
alertError(`${error}`)
|
alertError(`${error}`)
|
||||||
@@ -111,6 +28,99 @@ export async function importCharacter() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function importCharacterProcess(f:{
|
||||||
|
name: string;
|
||||||
|
data: Uint8Array;
|
||||||
|
}) {
|
||||||
|
if(f.name.endsWith('json')){
|
||||||
|
const da = JSON.parse(Buffer.from(f.data).toString('utf-8'))
|
||||||
|
if(await importSpecv2(da)){
|
||||||
|
let db = get(DataBase)
|
||||||
|
return db.characters.length - 1
|
||||||
|
}
|
||||||
|
if((da.char_name || da.name) && (da.char_persona || da.description) && (da.char_greeting || da.first_mes)){
|
||||||
|
let db = get(DataBase)
|
||||||
|
db.characters.push(convertOldTavernAndJSON(da))
|
||||||
|
DataBase.set(db)
|
||||||
|
alertNormal(language.importedCharacter)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
else{
|
||||||
|
alertError(language.errors.noData)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
alertStore.set({
|
||||||
|
type: 'wait',
|
||||||
|
msg: 'Loading... (Reading)'
|
||||||
|
})
|
||||||
|
await sleep(10)
|
||||||
|
const img = f.data
|
||||||
|
const readed = (await exifr.parse(img, true))
|
||||||
|
if(readed.chara){
|
||||||
|
// standard spec v2 imports
|
||||||
|
const charaData:CharacterCardV2 = JSON.parse(Buffer.from(readed.chara, 'base64').toString('utf-8'))
|
||||||
|
if(await importSpecv2(charaData, img)){
|
||||||
|
let db = get(DataBase)
|
||||||
|
return db.characters.length - 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(readed.risuai){
|
||||||
|
// old risu imports
|
||||||
|
await sleep(10)
|
||||||
|
const va = decodeMsgpack(Buffer.from(readed.risuai, 'base64')) as any
|
||||||
|
if(va.type !== 101){
|
||||||
|
alertError(language.errors.noData)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
let char:character = va.data
|
||||||
|
let db = get(DataBase)
|
||||||
|
if(char.emotionImages && char.emotionImages.length > 0){
|
||||||
|
for(let i=0;i<char.emotionImages.length;i++){
|
||||||
|
alertStore.set({
|
||||||
|
type: 'wait',
|
||||||
|
msg: `Loading... (Getting Emotions ${i} / ${char.emotionImages.length})`
|
||||||
|
})
|
||||||
|
await sleep(10)
|
||||||
|
const imgp = await saveAsset(char.emotionImages[i][1] as any)
|
||||||
|
char.emotionImages[i][1] = imgp
|
||||||
|
}
|
||||||
|
}
|
||||||
|
char.chats = [{
|
||||||
|
message: [],
|
||||||
|
note: '',
|
||||||
|
name: 'Chat 1',
|
||||||
|
localLore: []
|
||||||
|
}]
|
||||||
|
|
||||||
|
if(checkNullish(char.sdData)){
|
||||||
|
char.sdData = defaultSdDataFunc()
|
||||||
|
}
|
||||||
|
|
||||||
|
char.chatPage = 0
|
||||||
|
char.image = await saveAsset(PngMetadata.filter(img))
|
||||||
|
db.characters.push(characterFormatUpdate(char))
|
||||||
|
char.chaId = uuidv4()
|
||||||
|
setDatabase(db)
|
||||||
|
alertNormal(language.importedCharacter)
|
||||||
|
return db.characters.length - 1
|
||||||
|
}
|
||||||
|
else if(readed.chara){
|
||||||
|
const charaData:OldTavernChar = JSON.parse(Buffer.from(readed.chara, 'base64').toString('utf-8'))
|
||||||
|
const imgp = await saveAsset(PngMetadata.filter(img))
|
||||||
|
let db = get(DataBase)
|
||||||
|
db.characters.push(convertOldTavernAndJSON(charaData, imgp))
|
||||||
|
DataBase.set(db)
|
||||||
|
alertNormal(language.importedCharacter)
|
||||||
|
return db.characters.length - 1
|
||||||
|
}
|
||||||
|
else{
|
||||||
|
alertError(language.errors.noData)
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export async function characterHubImport() {
|
export async function characterHubImport() {
|
||||||
const charPath = (new URLSearchParams(location.search)).get('charahub')
|
const charPath = (new URLSearchParams(location.search)).get('charahub')
|
||||||
try {
|
try {
|
||||||
|
|||||||
Reference in New Issue
Block a user