More progress

This commit is contained in:
octospacc 2024-01-03 00:31:15 +01:00
parent 0dbe29c536
commit a9a061cd6d
1 changed files with 194 additions and 160 deletions

View File

@ -10,10 +10,85 @@
<script src="https://googlechrome.github.io/dialog-polyfill/dist/dialog-polyfill.js"></script> <script src="https://googlechrome.github.io/dialog-polyfill/dist/dialog-polyfill.js"></script>
<link rel="stylesheet" href="./paper.min.css"/> <link rel="stylesheet" href="./paper.min.css"/>
<div id="Main" hidden="true">
<div id="LayoutAccountSelect"></div>
<div id="LayoutAccountLogin">
<label>
Homeserver
<input type="text"/>
</label>
<label>
Username
<input type="text"/>
</label>
<label>
Password
<input type="password"/>
</label>
<label>
Add account via session token instead
<input type="checkbox"/>
</label>
<label>
<input type="button" value="Add"/>
</label>
</div>
<div id="LayoutCollectionActions">
<button name="back">🔙️ Go Back</button>
<button name="commit" disabled="true">📝️ Commit Changes</button>
</div>
<div id="LayoutPacksList"></div>
<div id="LayoutPackGrid"></div>
<div id="LayoutInfo"></div>
<details class="col border margin" id="LayoutCollectionOptions">
<summary>
<p>Options</p>
</summary>
<p>
Reinitializing sticker data for your account will simply override
<!-- the <code>stickerpicker</code> subfield in --> its <code>m.widgets</code> field.
</p>
<button name="reinit">💡️ Reinitialize sticker data as new</button>
<details class="col border margin">
<summary>
<p>Advanced</p>
</summary>
<label>
Sticker selector app URL
<input name="pickerUrl" type="text"/>
</label>
</details>
</details>
</div>
<hr/>
<p>
🃏️ [Matrix] Sticker Helper <a name="version" href="javascript:;">WIP</a>,
created with ☕️ by <a href="https://hub.octt.eu.org">OctoSpacc</a>.
<br/>
Made possible by <a href="https://github.com/maunium/stickerpicker">Maunium sticker picker</a>,
brought to paper thanks to <a href="https://www.getpapercss.com">PaperCSS</a>.
</p>
<details class="col border margin">
<summary>
<p>Help</p>
</summary>
<p>
There is no one around to help.
</p>
</details>
<script module="Meta">({ <script module="Meta">({
Name: "🃏️ [Matrix] Sticker Helper", Name: "🃏️ [Matrix] Sticker Helper",
})</script> })</script>
<!-- <script module="Main"> -->
<script module="Main" type="module"> <script module="Main" type="module">
//import { html, render } from 'https://esm.sh/htm/preact/standalone'; //import { html, render } from 'https://esm.sh/htm/preact/standalone';
//import axios from 'https://cdn.skypack.dev/axios'; //import axios from 'https://cdn.skypack.dev/axios';
@ -26,8 +101,9 @@
appIdentity: "org.eu.octt.MatrixStickerHelper", appIdentity: "org.eu.octt.MatrixStickerHelper",
appInterface: "v1", appInterface: "v1",
Strings: { Strings: {
mustInitMessage: `Your account must first be initialized to handle stickers.`, mConfirmCommit: "Confirm committing account data?",
notManagedMessage: ` mMustInit: "Your account must first be initialized to handle stickers.",
mNotManaged: `
Your account is set-up with a sticker picker, but it's not marked as being managed by this app. Your account is set-up with a sticker picker, but it's not marked as being managed by this app.
This could mean that you are currently using an incompatible sticker picker. This could mean that you are currently using an incompatible sticker picker.
You can try to continue anyway if you think it should work, otherwise you should reinitialize sticker data. You can try to continue anyway if you think it should work, otherwise you should reinitialize sticker data.
@ -190,37 +266,6 @@
document.querySelector('input[name="reinit"]').disabled = false; document.querySelector('input[name="reinit"]').disabled = false;
} }
function AccountChoice (props) {
let accountsActions = [];
for (let i=0; i<Config.accounts.length; i++) {
let account = Config.accounts[i];
accountsActions.push({
label: `🐱️ ${GetMatrixUserTag(account)}`,
onclick: () => {
State.account = account;
Render(CollectionScreen);
},
options: [ {
dataIndex: i,
label: '❌️',
onclick: () => {
Config.accounts.pop(this.dataIndex);
Config.Save();
Render();
},
} ]
})
}
return html`
<${ActionForm} actionEntries=${[...accountsActions,
{ label: '🆕️ Add new account', onclick: () => Render(AccountLogin) },
]}/>
`
}
function AccountLogin (props) { function AccountLogin (props) {
return html`<Fragment> return html`<Fragment>
<p> <p>
@ -257,16 +302,6 @@
} }
function CollectionList (props) { function CollectionList (props) {
let packElems = [];
for (const pack of (props.extras?.dataPacks?.packs || [])) {
packElems.push(html`
<img
src="${props.extras?.dataPacks?.homeserver_url}/_matrix/media/r0/thumbnail/${pack.stickers[0].info.thumbnail_url.split('mxc://')[1]}?&height=128&width=128&method=scale"
onclick=${null}
/>
`);
}
return html`<${CollectionViewMenu} optionsOpen=${false}> return html`<${CollectionViewMenu} optionsOpen=${false}>
<div id="CollectionList"><div> <div id="CollectionList"><div>
<button style=${{ verticalAlign: 'top', height: '64px' }}> <button style=${{ verticalAlign: 'top', height: '64px' }}>
@ -365,7 +400,7 @@
const GetMatrixUserTag = (account) => `@${account.username}:${account.homeserver.split('://')[1]}`; const GetMatrixUserTag = (account) => `@${account.username}:${account.homeserver.split('://')[1]}`;
const GetMatrixMediaUrl = (mxcId, homeserverUrl) => `${homeserverUrl}/_matrix/media/r0/thumbnail/${mxcId?.split('mxc://')[1]}`; const GetMatrixMediaUrl = (mxcId, homeserverUrl) => (mxcId ? `${homeserverUrl || `https://${mxcId.split('mxc://')[1].split('/')[0]}`}/_matrix/media/r0/thumbnail/${mxcId.split('mxc://')[1]}` : undefined);
function ResetLayouts () { function ResetLayouts () {
$`#Main`.hidden = false; $`#Main`.hidden = false;
@ -400,9 +435,9 @@
} }
async function RequestUploadFile (account, fileData, fileMime) { async function RequestUploadFile (account, fileData, fileMime) {
const request = await fetch(`${account.homeserver}/_matrix/client/v3/upload`, { const request = await fetch(`${account.homeserver}/_matrix/media/v3/upload`, {
method: "POST", method: "POST",
headers: { Authorization: `Bearer ${account.token}`, "Content-Type": (fileData.type || fileMime) }, headers: { Authorization: `Bearer ${account.token}`, ...(fileMime ? { "Content-Type": fileMime } : {}) },
body: fileData, body: fileData,
}); });
const result = await request.json(); const result = await request.json();
@ -414,18 +449,14 @@
} }
async function PreparePacksEditor (account) { async function PreparePacksEditor (account) {
const userTag = GetMatrixUserTag(account);
let widgetsData = {};
ResetLayouts(); ResetLayouts();
State.packsCount = 0;
$`#LayoutCollectionActions`.hidden = false; $`#LayoutCollectionActions`.hidden = false;
$`#LayoutCollectionOptions`.hidden = false; $`#LayoutCollectionOptions`.hidden = false;
$`#LayoutCollectionActions button[name="commit"]`.onclick = () => Spacc.ShowModal({ label: 'Confirm committing account data?', action: async () => { $`#LayoutCollectionOptions button[name="reinit"]`.onclick = () => Spacc.ShowModal({ label: Defaults.Strings.mConfirmCommit, action: async () => {
// ... we must read updated widget data that is stored somewhere accessible widgetsData = {
//if (await RequestAccountWidgetsData(account, {})) {
// `#LayoutCollectionActions button[name="commit"]`.disabled = true;
//}
} });
$`#LayoutCollectionOptions button[name="reinit"]`.onclick = () => Spacc.ShowModal({ label: 'Confirm committing account data?', action: async () => {
const userTag = GetMatrixUserTag(account);
await RequestAccountWidgetsData(account, {
stickerpicker: { stickerpicker: {
content: { content: {
type: "m.stickerpicker", type: "m.stickerpicker",
@ -442,79 +473,103 @@
type: "m.widget", type: "m.widget",
id: "stickerpicker", id: "stickerpicker",
}, },
}); };
DisplayPacksEditor(account); await RequestAccountWidgetsData(account, widgetsData);
DisplayPacksEditor(account, widgetsData);
} }); } });
$`#LayoutInfo`.innerHTML = `<p> $`#LayoutInfo`.innerHTML = `<p>
Fetching account data... Fetching account data...
</p>`; </p>`;
const widgetsData = await RequestAccountWidgetsData(account); widgetsData = await RequestAccountWidgetsData(account);
if (widgetsData) { if (widgetsData) {
const isManaged = widgetsData?.stickerpicker?.content?.managedBy?.includes(`${Defaults.appIdentity}/${Defaults.appInterface}`); const isManaged = widgetsData.stickerpicker?.content?.managedBy?.includes(`${Defaults.appIdentity}/${Defaults.appInterface}`);
const packsUrl = (new URLSearchParams(widgetsData?.stickerpicker?.content?.url?.split('?')[1])).get('config'); const packsUrl = (new URLSearchParams(widgetsData.stickerpicker?.content?.url?.split('?')[1])).get('config');
if (!isManaged || !widgetsData?.stickerpicker) { if (!isManaged || !widgetsData.stickerpicker) {
$`#LayoutCollectionOptions`.open = true; $`#LayoutCollectionOptions`.open = true;
if (!widgetsData?.stickerpicker) { if (!widgetsData.stickerpicker) {
$`#LayoutInfo`.innerHTML = `<p>${Defaults.Strings.mustInitMessage}</p>`; $`#LayoutInfo`.innerHTML = `<p>${Defaults.Strings.mMustInit}</p>`;
} else } else
if (!isManaged) { if (!isManaged) {
$`#LayoutInfo`.innerHTML = ` $`#LayoutInfo`.innerHTML = `
<p>${Defaults.Strings.notManagedMessage}</p> <p>${Defaults.Strings.mNotManaged}</p>
<button name="continue">⏭️ Continue</button> <button name="continue">⏭️ Continue</button>
`; `;
$`#LayoutInfo > button[name="continue"]`.onclick = () => { $`#LayoutInfo > button[name="continue"]`.onclick = () => {
$`#LayoutCollectionOptions`.open = false; $`#LayoutCollectionOptions`.open = false;
DisplayPacksEditor(account, packsUrl); DisplayPacksEditor(account, widgetsData, packsUrl);
}; };
} }
} else { } else {
DisplayPacksEditor(account, packsUrl); DisplayPacksEditor(account, widgetsData, packsUrl);
} }
} }
} }
async function DisplayPacksEditor (account, packsUrl) { async function DisplayPacksEditor (account, widgetsData, packsUrl) {
let packsData = { homeserver_url: account.homeserver, packs: [] };
let editedPacks = {};
ResetLayouts(); ResetLayouts();
$`#LayoutCollectionActions`.hidden = false; $`#LayoutCollectionActions`.hidden = false;
$`#LayoutCollectionOptions`.hidden = false; $`#LayoutCollectionOptions`.hidden = false;
let packsData; $`#LayoutCollectionActions button[name="commit"]`.onclick = () => Spacc.ShowModal({
label: Defaults.Strings.mConfirmCommit,
action: () => CommitNewAccountData(account, widgetsData, packsData, editedPacks),
});
if (packsUrl) { if (packsUrl) {
try {
const request = await fetch(packsUrl); const request = await fetch(packsUrl);
if (request.status === 200) {
packsData = await request.json(); packsData = await request.json();
} }
} catch(err) {
SpaccDotWeb.ShowModal(`${err} ${packsUrl}`);
}
}
const addButton = $().createElement('button'); const addButton = $().createElement('button');
addButton.name = 'add'; addButton.name = 'add';
addButton.innerHTML = ' Create New Pack'; addButton.innerHTML = ' Create New Pack';
addButton.onclick = (ev) => { addButton.onclick = (event) => Spacc.ShowModal({ label: 'Optionally include the URL of a sticker pack in JSON format to import it. Otherwise, leave the field empty to create a brand-new pack.', extraHTML: `
for (const elem of ev.srcElement.parentElement.querySelectorAll('button')) { <input name="packUrl" type="text"/>
`, action: async (event, modalButton) => {
// TODO error handling?
let packData = { stickers: [] };
const packUrl = modalButton.parentElement.querySelector('input[name="packUrl"]').value;
if (packUrl) {
const request = await fetch(packUrl);
packData = await request.json();
}
for (const elem of event.srcElement.parentElement.querySelectorAll('button')) {
elem.disabled = false; elem.disabled = false;
} }
const packButton = MakeStickerPackButton(account, packsData); const packButton = MakeStickerPackButton(State.packsCount, account, packsData, packData);
$`#LayoutPacksList`.insertBefore(packButton, $`#LayoutPacksList > button[name="add"]`.nextElementSibling); $`#LayoutPacksList`.insertBefore(packButton, $`#LayoutPacksList > button[name="add"]`.nextElementSibling);
$`#LayoutCollectionActions button[name="commit"]`.disabled = false; $`#LayoutCollectionActions button[name="commit"]`.disabled = false;
packButton.click(); packButton.click();
}; State.packsCount++;
} });
$`#LayoutPacksList`.appendChild(addButton); $`#LayoutPacksList`.appendChild(addButton);
for (const pack of (packsData?.packs || [])) { for (const pack of packsData?.packs) {
const request = await fetch(pack); const request = await fetch(pack);
const packData = await request.json(); const packData = await request.json();
$`#LayoutPacksList`.appendChild(MakeStickerPackButton(account, packsData, packData)); $`#LayoutPacksList`.appendChild(MakeStickerPackButton(State.packsCount, account, packsData, packData));
State.packsCount++;
} }
} }
function MakeStickerPackButton (account, packsData, packData) { function MakeStickerPackButton (index, account, packsData, packData) {
const packButton = $().createElement('button'); const packButton = $().createElement('button');
packButton.innerHTML = `<img src="${GetMatrixMediaUrl(packData?.stickers[0]?.info?.thumbnail_url, packsData?.homeserver_url)}?&height=64&width=64&method=scale"/>`; packButton.dataset.index = index;
packButton.onclick = (ev) => { packButton.innerHTML = `<img src="${GetMatrixMediaUrl(packData.stickers[0]?.info?.thumbnail_url, packsData?.homeserver_url) || ''}?&height=64&width=64&method=scale"/>`;
const thisElem = (ev.srcElement.tagName.toLowerCase() === 'button' ? ev.srcElement : ev.srcElement.parentElement); packButton.onclick = (event) => {
const thisElem = (event.srcElement.tagName.toLowerCase() === 'button' ? event.srcElement : event.srcElement.parentElement);
for (const elem of thisElem.parentElement.querySelectorAll('button')) { for (const elem of thisElem.parentElement.querySelectorAll('button')) {
elem.disabled = false; elem.disabled = false;
} }
thisElem.disabled = true; thisElem.disabled = true;
$`#LayoutPackGrid`.innerHTML = ''; $`#LayoutPackGrid`.innerHTML = '';
for (const sticker of (packData?.stickers || [])) { for (const sticker of (packData.stickers || [])) {
const stickerElem = $().createElement('button'); const stickerElem = $().createElement('button');
stickerElem.innerHTML = `<img src="${GetMatrixMediaUrl(sticker.info.thumbnail_url, packsData?.homeserver_url)}?&height=128&width=128&method=scale"/>`; stickerElem.innerHTML = `<img src="${GetMatrixMediaUrl(sticker.info.thumbnail_url, packsData?.homeserver_url) || ''}?&height=128&width=128&method=scale"/>`;
$`#LayoutPackGrid`.appendChild(stickerElem); $`#LayoutPackGrid`.appendChild(stickerElem);
} }
const addButton = $().createElement('button'); const addButton = $().createElement('button');
@ -522,31 +577,73 @@
Upload New Sticker(s) Upload New Sticker(s)
<input type=file hidden="true" multiple="true" accept="image/jpeg,image/gif,image/png,image/webp"/> <input type=file hidden="true" multiple="true" accept="image/jpeg,image/gif,image/png,image/webp"/>
`; `;
addButton.querySelector('input[type="file"]').onchange = async (ev) => { addButton.querySelector('input[type="file"]').onchange = async (event) => {
for (const file in ev.target.files) { let newPackStickers = [];
for (const file of event.target.files) {
const result = await RequestUploadFile(account, file); const result = await RequestUploadFile(account, file);
if (result) {
newPackStickers.push({
id: result.content_uri,
url: result.content_uri,
msgtype: "m.sticker",
body: file.name,
info: {
// ... TODO read real image dimensions
w: 256,
h: 256,
size: file.size,
mimetype: file.type,
thumbnail_url: result.content_uri,
thumbnail_info: {
// ... TODO read real image dimensions
w: 256,
h: 256,
size: file.size,
mimetype: file.type,
},
},
});
// ... we must use the result.content_uri to add the new sticker to the screen, add it somewhere keeping the current pack state, and the packs state, ready for committing // ... we must use the result.content_uri to add the new sticker to the screen, add it somewhere keeping the current pack state, and the packs state, ready for committing
if (!result) { } else {
switch (Spacc.ShowModal({ label: 'File upload failed. What to do?', action: () => 'continue', actionCancel: () => 'cancel', labelSecondary: '🔄️ Retry', actionSecondary: () => 'retry' })) { const answer = Spacc.ShowModal({
case 'cancel': label: 'File upload failed. What to do?',
action: () => 'continue',
actionCancel: () => 'cancel',
labelSecondary: '🔄️ Retry', actionSecondary: () => 'retry',
});
if (answer === 'cancel') {
newPackStickers = [];
break; break;
break; } else if (answer === 'retry') {
case 'continue':
continue;
break;
case 'retry':
// ... find out how to handle this // ... find out how to handle this
break; } else if (answer === 'continue') {
continue;
} }
} }
} }
// ... we must handle new stickers to add them to the initial object
packData.stickers = [...packData.stickers, newPackStickers];
}; };
addButton.onclick = (ev) => ev.srcElement.querySelector('input[type="file"]')?.click(); addButton.onclick = (event) => event.srcElement.querySelector('input[type="file"]')?.click();
$`#LayoutPackGrid`.appendChild(addButton); $`#LayoutPackGrid`.appendChild(addButton);
}; };
return packButton; return packButton;
} }
async function CommitNewAccountData (account, widgetsData, packsData, editedPacks) {
for (const packIndex in editedPacks) {
const pack = editedPacks[pack];
const packUrlNew = GetMatrixMediaUrl(await RequestUploadFile(account, JSON.stringify(pack.data), 'application/json').content_uri);
packsData.packs.remove(pack.url);
packsData.packs.push(packUrlNew);
}
const packsUrlNew = GetMatrixMediaUrl(await RequestUploadFile(account, JSON.stringify(packsData), 'application/json').content_uri);
widgetsData.stickerpicker.content.url = `${$`#LayoutCollectionOptions input[name="pickerUrl"]`.value}?&config=${packsUrlNew}&theme=$theme`;
if (await RequestAccountWidgetsData(account, widgetsData)) {
$`#LayoutCollectionActions button[name="commit"]`.disabled = true;
}
}
function DisplayAccountSelect () { function DisplayAccountSelect () {
ResetLayouts(); ResetLayouts();
for (const account of Config.accounts) { for (const account of Config.accounts) {
@ -583,69 +680,6 @@
DisplayAccountSelectLogin(); DisplayAccountSelectLogin();
</script> </script>
<div id="Main" hidden="true">
<div id="LayoutAccountSelect"></div>
<div id="LayoutAccountLogin">
<label>
Homeserver
<input type="text"/>
</label>
<label>
Username
<input type="text"/>
</label>
<label>
Password
<input type="password"/>
</label>
<label>
Add account via session token instead
<input type="checkbox"/>
</label>
<label>
<input type="button" value="Add"/>
</label>
</div>
<div id="LayoutCollectionActions">
<button name="back">🔙️ Go Back</button>
<button name="commit" disabled="true">📝️ Commit Changes</button>
</div>
<div id="LayoutPacksList"></div>
<div id="LayoutPackGrid"></div>
<div id="LayoutInfo"></div>
<details class="col border margin" id="LayoutCollectionOptions">
<summary>
<p>Options</p>
</summary>
<p>
Reinitializing sticker data for your account will simply override
<!-- the <code>stickerpicker</code> subfield in --> its <code>m.widgets</code> field.
</p>
<button name="reinit">💡️ Reinitialize sticker data as new</button>
<details class="col border margin">
<summary>
<p>Advanced</p>
</summary>
<label>
Sticker selector app URL
<input name="pickerUrl" type="text"/>
</label>
</details>
</details>
</div>
<p>
🃏️ [Matrix] Sticker Helper <a name="version" href="javascript:;">WIP</a>,
created with ☕️ by <a href="https://hub.octt.eu.org">OctoSpacc</a>.
<br/>
Made possible by <a href="https://github.com/maunium/stickerpicker">Maunium sticker picker</a>,
brought to paper thanks to <a href="https://www.getpapercss.com">PaperCSS</a>.
</p>
<style> <style>
:root { :root {
--margin: 8px; --margin: 8px;