🏠 Home 

Adventure + Scenario Exporter

Export any adventure or scenario to a local file.

// ==UserScript==
// @version      1.0.0
// @name         Adventure + Scenario Exporter
// @description  Export any adventure or scenario to a local file.
// @author       Magic <[email protected]>
// @supportURL   https://github.com/magicoflolis/userscriptrepo/issues
// @namespace    https://github.com/magicoflolis/userscriptrepo/tree/master/userscripts/AIDungeon
// @homepageURL  https://github.com/magicoflolis/userscriptrepo/tree/master/userscripts/AIDungeon
// @icon         
// @license      MIT
// @compatible     chrome
// @compatible     firefox
// @compatible     edge
// @compatible     opera
// @compatible     safari
// @connect     play.aidungeon.com
// @grant     unsafeWindow
// @match     https://play.aidungeon.com/*
// @noframes
// @run-at     document-start
// ==/UserScript==
(() => {
'use strict';
/******************************************************************************/
const inIframe = () => {
try {
return window.self !== window.top;
} catch (e) {
return true;
}
}
if (inIframe()) {
return;
}
let userjs = self.userjs;
/**
* Skip text/plain documents, based on uBlock Origin `vapi.js` file
*
* [Source Code](https://github.com/gorhill/uBlock/blob/master/platform/common/vapi.js)
*/
if (
(document instanceof Document ||
(document instanceof XMLDocument && document.createElement('div') instanceof HTMLDivElement)) &&
/^image\/|^text\/plain/.test(document.contentType || '') === false &&
(self.userjs instanceof Object === false || userjs.UserJS !== true)
) {
userjs = self.userjs = { UserJS: true };
}
if (!(typeof userjs === 'object' && userjs.UserJS)) {
return;
}
/******************************************************************************/
// #region Console
const err = (...msg) => {
console.error(
'[%cAI Dungeon Script%c] %cERROR',
'color: rgb(69, 91, 106);',
'',
'color: rgb(249, 24, 128);',
...msg
);
const a = typeof alert !== 'undefined' && alert;
for (const ex of msg) {
if (typeof ex === 'object' && 'cause' in ex && a) {
a(`[AI Dungeon Script] (${ex.cause}) ${ex.message}`);
}
}
};
// #endregion
function getUAData() {
if (userjs.isMobile !== undefined) {
return userjs.isMobile;
}
try {
if (navigator) {
const { userAgent, userAgentData } = navigator;
const { platform, mobile } = userAgentData ? Object(userAgentData) : {};
userjs.isMobile =
/Mobile|Tablet/.test(userAgent ? String(userAgent) : '') ||
Boolean(mobile) ||
/Android|Apple/.test(platform ? String(platform) : '');
} else {
userjs.isMobile = false;
}
} catch (ex) {
userjs.isMobile = false;
ex.cause = 'getUAData';
err(ex);
}
return userjs.isMobile;
}
const isMobile = getUAData();
const win = unsafeWindow ?? window;
// #region Utilities
/**
* @type { import("../typings/types.d.ts").qs }
*/
const qs = (selector, root) => {
try {
return (root || document).querySelector(selector);
} catch (ex) {
err(ex);
}
return null;
};
/**
* @type { import("../typings/types.d.ts").objToStr }
*/
const objToStr = (obj) => Object.prototype.toString.call(obj);
/**
* @type { import("../typings/types.d.ts").isElem }
*/
const isElem = (obj) => {
const s = objToStr(obj);
return s.includes('Element');
};
/**
* @type { import("../typings/types.d.ts").isObj }
*/
const isObj = (obj) => {
const s = objToStr(obj);
return s.includes('Object');
};
/**
* @type { import("../typings/types.d.ts").isFN }
*/
const isFN = (obj) => {
const s = objToStr(obj);
return s.includes('Function');
};
/**
* @type { import("../typings/types.d.ts").isNull }
*/
const isNull = (obj) => {
return Object.is(obj, null) || Object.is(obj, undefined);
};
/**
* @type { import("../typings/types.d.ts").isBlank }
*/
const isBlank = (obj) => {
return (
(typeof obj === 'string' && Object.is(obj.trim(), '')) ||
((obj instanceof Set || obj instanceof Map) && Object.is(obj.size, 0)) ||
(Array.isArray(obj) && Object.is(obj.length, 0)) ||
(isObj(obj) && Object.is(Object.keys(obj).length, 0))
);
};
/**
* @type { import("../typings/types.d.ts").isEmpty }
*/
const isEmpty = (obj) => {
return isNull(obj) || isBlank(obj);
};
/**
* @type { import("../typings/types.d.ts").normalizeTarget }
*/
const normalizeTarget = (target, toQuery = true, root) => {
if (Object.is(target, null) || Object.is(target, undefined)) {
return [];
}
if (Array.isArray(target)) {
return target;
}
if (typeof target === 'string') {
return toQuery ? Array.from((root || document).querySelectorAll(target)) : [target];
}
if (isElem(target)) {
return [target];
}
return Array.from(target);
};
/**
* @type { import("../typings/UserJS.d.ts").observe }
*/
const observe = (element, listener, options = { subtree: true, childList: true }) => {
const observer = new MutationObserver(listener);
observer.observe(element, options);
listener.call(element, [], observer);
return observer;
};
/**
* @type { import("../typings/types.d.ts").ael }
*/
const ael = (el, type, listener, options = {}) => {
try {
for (const elem of normalizeTarget(el)) {
if (!elem) {
continue;
}
if (isMobile && type === 'click') {
elem.addEventListener('touchstart', listener, options);
continue;
}
elem.addEventListener(type, listener, options);
}
} catch (ex) {
ex.cause = 'ael';
err(ex);
}
};
/**
* @type { import("../typings/types.d.ts").make }
*/
const make = (tagName, cname, attrs) => {
let el;
try {
/**
* @type { import("../typings/types.d.ts").formAttrs }
*/
const formAttrs = (elem, attr = {}) => {
if (!elem) {
return elem;
}
for (const key in attr) {
if (typeof attr[key] === 'object') {
formAttrs(elem[key], attr[key]);
} else if (isFN(attr[key])) {
if (/^on/.test(key)) {
elem[key] = attr[key];
continue;
}
ael(elem, key, attr[key]);
} else if (key === 'class') {
elem.className = attr[key];
} else {
elem[key] = attr[key];
}
}
return elem;
};
el = document.createElement(tagName);
if (!isEmpty(cname)) {
if (typeof cname === 'string') {
el.className = cname;
} else if (isObj(cname)) {
formAttrs(el, cname);
}
}
if (!isEmpty(attrs)) {
if (typeof attrs === 'string') {
el.textContent = attrs;
} else if (isObj(attrs)) {
formAttrs(el, attrs);
}
}
} catch (ex) {
ex.cause = 'make';
err(ex);
}
return el;
};
//#endregion
/**
* @type { import("../typings/UserJS.d.ts").Network }
*/
const Network = {
async req(url, method = 'GET', responseType = 'json', data) {
if (isEmpty(url)) {
throw new Error('"url" parameter is empty');
}
data = Object.assign({}, data);
method = this.bscStr(method, false);
responseType = this.bscStr(responseType);
const params = {
method,
...data
};
return new Promise((resolve, reject) => {
fetch(url, params)
.then((response_1) => {
if (!response_1.ok) reject(response_1);
const check = (str_2 = 'text') => {
return isFN(response_1[str_2]) ? response_1[str_2]() : response_1;
};
if (responseType.match(/buffer/)) {
resolve(check('arrayBuffer'));
} else if (responseType.match(/json/)) {
resolve(check('json'));
} else if (responseType.match(/text/)) {
resolve(check('text'));
} else if (responseType.match(/blob/)) {
resolve(check('blob'));
} else if (responseType.match(/formdata/)) {
resolve(check('formData'));
} else if (responseType.match(/clone/)) {
resolve(check('clone'));
} else if (responseType.match(/document/)) {
const respTxt = check('text');
const domParser = new DOMParser();
if (respTxt instanceof Promise) {
respTxt.then((txt) => {
const doc = domParser.parseFromString(txt, 'text/html');
resolve(doc);
});
} else {
const doc = domParser.parseFromString(respTxt, 'text/html');
resolve(doc);
}
} else {
resolve(response_1);
}
})
.catch(reject);
});
},
bscStr(str = '', lowerCase = true) {
const txt = str[lowerCase ? 'toLowerCase' : 'toUpperCase']();
return txt.replaceAll(/\W/g, '');
}
};
const getToken = () => {
return new Promise((resolve, reject) => {
if (userjs.accessToken !== undefined) {
resolve(userjs.accessToken);
}
const dbReq = win.indexedDB.open('firebaseLocalStorageDb');
dbReq.onerror = reject;
dbReq.onsuccess = (event) => {
const transaction = event.target.r###lt.transaction(['firebaseLocalStorage'], 'readwrite');
const objectStore = transaction.objectStore('firebaseLocalStorage');
const allKeys = objectStore.getAllKeys();
allKeys.onerror = reject;
allKeys.onsuccess = (evt) => {
const key = evt.target.r###lt.find((r) => r.includes('firebase:authUser:'));
objectStore.get(key).onsuccess = (evt) => {
const { value } = evt.target.r###lt;
userjs.accessToken = value.stsTokenManager.accessToken;
resolve(userjs.accessToken);
};
};
};
});
};
/**
* @param {string} pathname
*/
const getAdventure = async (pathname) => {
try {
const parts = /\/(adventure|scenario)\/([\w-]+)\/.+(\/)?/.exec(pathname);
if (!parts) {
return;
}
const resp = {
data: {}
};
const shortId = parts[2];
const type = {
adventure: {
headers: {
'x-gql-operation-name': 'GetGameplayAdventure'
},
body: {
operationName: 'GetGameplayAdventure',
variables: { shortId, limit: 100, desc: true },
query:
'query GetGameplayAdventure($shortId: String, $limit: Int, $offset: Int, $desc: Boolean) {\n  adventure(shortId: $shortId) {\n    id\n    publicId\n    shortId\n    scenarioId\n    instructions\n    title\n    description\n    tags\n    nsfw\n    isOwner\n    userJoined\n    gameState\n    actionCount\n    contentType\n    createdAt\n    showComments\n    commentCount\n    allowComments\n    voteCount\n    userVote\n    editedAt\n    published\n    unlisted\n    deletedAt\n    saveCount\n    isSaved\n    user {\n      id\n      isCurrentUser\n      isMember\n      profile {\n        id\n        title\n        thumbImageUrl\n        __typename\n      }\n      __typename\n    }\n    shortCode\n    thirdPerson\n    imageStyle\n    memory\n    authorsNote\n    image\n    actionWindow(limit: $limit, offset: $offset, desc: $desc) {\n      id\n      imageText\n      ...ActionSubscriptionAction\n      __typename\n    }\n    allPlayers {\n      ...PlayerSubscriptionPlayer\n      __typename\n    }\n    storyCards {\n      id\n      ...StoryCard\n      __typename\n    }\n    __typename\n  }\n}\n\nfragment ActionSubscriptionAction on Action {\n  id\n  text\n  type\n  imageUrl\n  shareUrl\n  imageText\n  adventureId\n  decisionId\n  undoneAt\n  deletedAt\n  createdAt\n  logId\n  __typename\n}\n\nfragment PlayerSubscriptionPlayer on Player {\n  id\n  userId\n  characterName\n  isTypingAt\n  user {\n    id\n    isMember\n    profile {\n      id\n      title\n      thumbImageUrl\n      __typename\n    }\n    __typename\n  }\n  createdAt\n  deletedAt\n  blockedAt\n  __typename\n}\n\nfragment StoryCard on StoryCard {\n  id\n  type\n  keys\n  value\n  title\n  useForCharacterCreation\n  description\n  updatedAt\n  deletedAt\n  __typename\n}'
}
},
scenario: {
headers: {
'x-gql-operation-name': 'GetScenario'
},
body: {
operationName: 'GetScenario',
variables: { shortId },
query:
'query GetScenario($shortId: String) {\n  scenario(shortId: $shortId) {\n    id\n    contentType\n    createdAt\n    editedAt\n    publicId\n    shortId\n    title\n    description\n    prompt\n    memory\n    authorsNote\n    image\n    isOwner\n    published\n    unlisted\n    allowComments\n    showComments\n    commentCount\n    voteCount\n    userVote\n    saveCount\n    storyCardCount\n    isSaved\n    tags\n    adventuresPlayed\n    thirdPerson\n    nsfw\n    contentRating\n    contentRatingLockedAt\n    contentRatingLockedMessage\n    tags\n    type\n    details\n    parentScenario {\n      id\n      shortId\n      title\n      __typename\n    }\n    user {\n      isCurrentUser\n      isMember\n      profile {\n        title\n        thumbImageUrl\n        __typename\n      }\n      __typename\n    }\n    options {\n      id\n      userId\n      shortId\n      title\n      prompt\n      parentScenarioId\n      deletedAt\n      __typename\n    }\n    storyCards {\n      id\n      ...StoryCard\n      __typename\n    }\n    ...CardSearchable\n    __typename\n  }\n}\n\nfragment CardSearchable on Searchable {\n  id\n  contentType\n  publicId\n  shortId\n  title\n  description\n  image\n  tags\n  userVote\n  voteCount\n  published\n  unlisted\n  publishedAt\n  createdAt\n  isOwner\n  editedAt\n  deletedAt\n  blockedAt\n  isSaved\n  saveCount\n  commentCount\n  userId\n  contentRating\n  user {\n    id\n    isMember\n    profile {\n      id\n      title\n      thumbImageUrl\n      __typename\n    }\n    __typename\n  }\n  ... on Adventure {\n    actionCount\n    userJoined\n    playPublicId\n    unlisted\n    playerCount\n    __typename\n  }\n  ... on Scenario {\n    adventuresPlayed\n    __typename\n  }\n  __typename\n}\n\nfragment StoryCard on StoryCard {\n  id\n  type\n  keys\n  value\n  title\n  useForCharacterCreation\n  description\n  updatedAt\n  deletedAt\n  __typename\n}'
}
}
};
const accessToken = await getToken();
const adventure = await Network.req('https://api.aidungeon.com/graphql', 'POST', 'json', {
headers: {
authorization: `firebase ${accessToken}`,
'content-type': 'application/json',
'Sec-GPC': '1',
'Sec-Fetch-Dest': 'empty',
'Sec-Fetch-Mode': 'cors',
'Sec-Fetch-Site': 'same-site',
Priority: 'u=4',
...type[parts[1]].headers
},
referrer: 'https://play.aidungeon.com/',
body: JSON.stringify(type[parts[1]].body)
});
if (parts[1] === 'adventure') {
const state = await Network.req('https://api.aidungeon.com/graphql', 'POST', 'json', {
headers: {
authorization: `firebase ${accessToken}`,
'content-type': 'application/json',
'x-gql-operation-name': 'GetGameplayAdventure',
'Sec-GPC': '1',
'Sec-Fetch-Dest': 'empty',
'Sec-Fetch-Mode': 'cors',
'Sec-Fetch-Site': 'same-site',
Priority: 'u=4'
},
referrer: 'https://play.aidungeon.com/',
body: JSON.stringify({
operationName: 'GetAdventureDetails',
variables: { shortId },
query:
'query GetAdventureDetails($shortId: String) {\n  adventureState(shortId: $shortId) {\n    id\n    details\n    __typename\n  }\n}'
})
});
Object.assign(resp.data, {...adventure.data, ...state.data});
} else {
Object.assign(resp, adventure);
}
return resp;
} catch (ex) {
err(ex);
}
return {};
};
/**
* @param {HTMLElement} parent
* @param {string} type
*/
const inject = (parent, type = 'play') => {
if (!parent) {
return;
}
if (qs('.mujs-btn')) {
return;
}
const parts = /\/(adventure|scenario)\/([\w-]+)\/.+(\/)?/.exec(location.pathname);
const rootType = parts && parts[1];
const cl = {
play: 'mujs-btn is_Button _bg-0hover-513675900 _btc-0hover-1394778429 _brc-0hover-1394778429 _###-0hover-1394778429 _blc-0hover-1394778429 _bxsh-0hover-448821143 _bg-0active-744986709 _btc-0active-1163467620 _brc-0active-1163467620 _###-0active-1163467620 _blc-0active-1163467620 _bxsh-0active-680131952 _bg-0focus-455866976 _btc-0focus-1452587353 _brc-0focus-1452587353 _###-0focus-1452587353 _blc-0focus-1452587353 _bxsh-0focus-391012219 _dsp-flex _fb-auto _bxs-border-box _pos-relative _mih-0px _miw-0px _fs-1 _cur-pointer _ox-hidden _oy-hidden _jc-center _ai-center _h-606181790 _btlr-1307609905 _btrr-1307609905 _bbrr-1307609905 _bblr-1307609905 _pr-1481558338 _pl-1481558338 _fd-row _bg-1633501478 _btc-2122800589 _brc-2122800589 _###-2122800589 _blc-2122800589 _btw-1px _brw-1px _bbw-1px _blw-1px _gap-1481558369 _outlineColor-43811550 _fg-1 _bbs-solid _bts-solid _bls-solid _brs-solid _bxsh-1445571361',
preview:
'mujs-btn is_Row _dsp-flex _fb-auto _bxs-border-box _pos-relative _mih-0px _miw-0px _fs-1 _fd-row _ai-center _gap-1481558369 _w-5037 _pt-1481558338 _pb-1481558338 _fg-1'
};
const btn = make('div', cl[type], {
id: 'game-blur-button'
});
const txt = make(
'span',
'is_ButtonText font_body _ff-2###67014 _dsp-inline _bxs-border-box _ww-break-word _whiteSpace-pre-wrap _mt-0px _mr-0px _mb-0px _ml-0px _col-675002279 _fos-229441189 _lh-222976573 _tt-uppercase _mah-606181790 _pe-none _zi-1',
{
textContent: `Export ${rootType} (JSON)`
}
);
const ico = make(
'p',
'is_Paragraph font_icons _dsp-inline _bxs-border-box _ww-break-word _mt-0px _mr-0px _mb-0px _ml-0px _col-675002279 _ff-2###67014 _fow-233016109 _ls-167744028 _fos-229441158 _lh-222976511 _ussel-auto _whiteSpace-1357640891 _pe-none _pt-1316335105 _pb-1316335105',
{
textContent: 'w_export'
}
);
let span;
ico.importantforaccessibility = 'no';
ico['aria-hidden'] = true;
btn.append(ico, txt);
ael(btn, 'click', async (evt) => {
evt.preventDefault();
const obj = await getAdventure(location.pathname);
const root = obj.data.adventure ?? obj.data.scenario;
const str = JSON.stringify(obj, null, ' ');
const bytes = new TextEncoder().encode(str);
const blob = new Blob([bytes], { type: 'application/json;charset=utf-8' });
const e = make('a', 'mujs-exporter', {
href: URL.createObjectURL(blob),
download: `${root.title}_${root.shortId}.${rootType}.json`
});
e.click();
URL.revokeObjectURL(e.href);
});
if (type === 'play') {
span = make('span', 't_sub_theme t_coreA1 _dsp_contents is_Theme', {
style: 'color: var(--color);'
});
} else if (type === 'preview') {
span = make(
'div',
'is_Row _dsp-flex _fb-auto _bxs-border-box _pos-relative _miw-0px _fs-0 _fd-row _pe-auto _jc-441309761 _ai-center _gap-1481558307 _btw-1px _btc-43811426 _mt--1px _mih-606181883 _bts-solid '
);
btn.style = 'cursor: pointer;';
}
span.append(btn);
parent.appendChild(span);
};
/**
* @template { Function } F
* @param { (this: F, doc: Document) => * } onDomReady
*/
const loadDOM = (onDomReady) => {
if (isFN(onDomReady)) {
if (document.readyState === 'interactive' || document.readyState === 'complete') {
onDomReady(document);
} else {
document.addEventListener('DOMContentLoaded', (evt) => onDomReady(evt.target), {
once: true
});
}
}
};
loadDOM((doc) => {
try {
if (window.location === null) {
throw new Error('"window.location" is null, reload the webpage or use a different one', {
cause: 'loadDOM'
});
}
if (doc === null) {
throw new Error('"doc" is null, reload the webpage or use a different one', {
cause: 'loadDOM'
});
}
const ignoreTags = new Set(['br', 'head', 'link', 'meta', 'script', 'style']);
observe(doc, (mutations) => {
try {
for (const mutation of mutations) {
for (const node of mutation.addedNodes) {
if (node.nodeType !== 1) {
continue;
}
if (ignoreTags.has(node.localName)) {
continue;
}
if (node.parentElement === null) {
continue;
}
if (!(node instanceof HTMLElement)) {
continue;
}
if (qs('div._pt-1481558307._btrr-1881205710', node)) {
inject(qs('div._pt-1481558307._btrr-1881205710', node), 'play');
}
if (qs('div.is_Column._pt-1481558400[role="list"]', node)) {
inject(qs('div.is_Column._pt-1481558400[role="list"]', node), 'preview');
}
}
}
} catch (ex) {
err(ex);
}
});
} catch (ex) {
err(ex);
}
});
})();