Plugin: OmniExport for Amplenote

linkOmniExport for Amplenote Plugin

A plugin that enables you to export your notes in a variety of formats.

Watch the Overview Video

linkFeatures

Supported Formats: PDF, Word, ePub, Single File ePub, LaTeX, CSV, Markdown

ePub: Includes custom cover with note title. You can also bulk export your notes as a single epub book.

GIF support: Will auto-render gifs in your notes to still PNGs for formats that don't support GIF (PDF and LaTeX)

Bulk Export: You can export your notes across different tags, name queries or even your whole notebook. Exports Images and Attachments!

Warning: This feature works best on the desktop app because the export can be streamed. On browsers it has to load all the exports into memory before downloading them, which could crash Amplenote. I am currently working with the Amplenote team to try and get the necessary plugin api support to make this work efficiently across all platforms.

Export Current Graph or Search: Filter your graph or do a note search and then open the bulk export wizard, as easy as that.

Table Export: You ever just want to export a single table as CSV? With this plugin you can select a part of your note containing a table and do just that.

Lazy Loaded Dependencies: The plugin only loads the external libraries on demand so no need to worry about plugin bloat here.

linkUsage

Bulk Export

Use the Cmd/Ctrl-O Bulk Export menu to open the bulk export wizard.

You can query by name and tags, or export the whole notebook

The wizard also includes a shortcut button to export every note tagged with "export"

Choose whether to export attachments or not.

Note: Mac OSX Archive Utility is very bug prone and sometimes refuses to open certain zip exports very randomly, especially epub exports. Please use a different archive utility if you get a "It is in an unsupported format" error. This is NOT an issue with the plugin or zip creation.

Export By Temporary Export Tag

You can tag individual notes or groups of notes (through graph, autotagger plugin, etc) with a export tag (default: system/export) and then hit the "Export by Export Tag" button in the bulk exporter. Note: This will ignore the query and just export those notes.

The plugin will automatically remove the export tag from those notes after export unless disabled.

Export Current Graph or Search

Search and filter in graph mode or note serach mode until you get the set of notes you want to export then Cmd/Ctrl-O Bulk Export and it will be prefiiled with your query.

Export As

From the Note Menu you can export in to all the supported formats. This does not currently export attachments, for that you have to use the bulk exporter.

Table Export: Note: Because of Amplenotes interface you must select a text block that contains a table to do a table export. If you just select the table itself it won't give you the plugin menu.

linkRoadmap

Export as Single PDF

More export customization options, styling and fonts.

More bulk export query options

Export Interlinking

Create your own fully fledged epub with interlinked chapters/notes

linkAuthor

☕ If the plugin is helpful to you, you can buy me a coffee!


Check out my other amplenote plugin! Dynamic Templater

Published by: lapluviosilla

Date: November 1st, 2024

Last Updated: November 1st, 2024

linkFeedback

If you have any questions, issues, or feedback, please feel free to reach out!

linkOmniExport (1.0.0)

linkTable - Plugin Parameters:

name

OmniExport

description

A plugin that enables you to export your notes in a variety of formats.

icon

output

instructions

Use the Cmd/Ctrl-O Bulk Export menu to open the bulk export wizard. Use the Note option menu to export as various formats.
View the README on Github for detailed usage instructions

Watch the Overview Video


☕ If the plugin is helpful to you, you can buy me a coffee!

- Supported Formats: PDF, Word, ePub, Single File ePub, LaTeX, CSV, Markdown

- ePub: Includes custom cover with note title. You can also bulk export your notes as a single epub book.

- GIF support: Will auto-render gifs in your notes to still PNGs for formats that don't support GIF (PDF and LaTeX)

- Bulk Export: You can export your notes across different tags, name queries or even your whole notebook. Exports Images and Attachments!

- Warning: This feature works best on the desktop app because the export can be streamed. On browsers it has to load all the exports into memory before downloading them, which could crash Amplenote. I am currently working with the Amplenote team to try and get the necessary plugin api support to make this work efficiently across all platforms.

- Export Current Graph or Search: Filter your graph or do a note search and then open the bulk export wizard, as easy as that.

- Table Export: You ever just want to export a single table as CSV? With this plugin you can select a part of your note containing a table and do just that.

- Lazy Loaded Dependencies: The plugin only loads the external libraries on demand so no need to worry about plugin bloat here.

setting

Export Tag

setting

Keep Export Tag after Export? (set to true to disable removal of the tag)

linkCode Base:

Source Repo: https://github.com/lapluviosilla/amplenote-omniexport

(()=>{var k=null;function P(A){if(typeof A!="object"||A===null)throw new TypeError("App interface must be a non-null object.");k=A}function T(){if(!k)throw new Error("App interface has not been set yet.");return k}var X=new Map;async function V(A,e){if(X.has(A)){const r=X.get(A);if(r.module)return r.module;if(r.promise)return r.promise}const n=new Promise((r,t)=>{const a=document.createElement("script");a.type="text/javascript",a.src=A,a.async=!0,a.onload=()=>{const o=e.split(".");let i=window[o.shift()];o.forEach(s=>{i=i?i[s]:null}),i?(X.set(A,{module:i,promise:null}),r(i)):t(new Error(`Module "${e}" not found on window after loading ${A}`))},a.onerror=()=>{X.delete(A),t(new Error(`Failed to load script: ${A}`))},document.body.appendChild(a)});return X.set(A,{module:null,promise:n}),n}function uA(){return V("https://cdn.jsdelivr.net/npm/markdown-it@14.1.0/dist/markdown-it.min.js","markdownit")}function pA(){return V("https://cdnjs.cloudflare.com/ajax/libs/pdfmake/0.2.4/pdfmake.min.js","pdfMake")}function mA(){return V("https://cdnjs.cloudflare.com/ajax/libs/pdfmake/0.2.4/vfs_fonts.js","pdfMake.vfs")}function BA(){return V("https://cdn.jsdelivr.net/npm/html-docx-js/dist/html-docx.js","htmlDocx")}function $(){return V("https://unpkg.com/epub-gen-memory@1.1.2/dist/bundle.min.js","epubGen")}function N(){return V("https://cdn.jsdelivr.net/npm/@transcend-io/conflux@4.1/dist/conflux.umd.min.js","conflux")}var F=`body {
font-family: Arial, sans-serif;
line-height: 1.6;
padding: 20px;
max-width: 800px;
margin: auto;
/* background-color: #f9f9f9; */
color: #333;
}
 
h1, h2, h3, h4, h5, h6 {
color: #2c3e50;
line-height: 1.2;
margin-top: 1.5em;
margin-bottom: 0.5em;
}
 
h1 {
font-size: 2.2em;
border-bottom: 2px solid #e2e2e2;
padding-bottom: 0.3em;
}
 
h2 {
font-size: 1.8em;
border-bottom: 1px solid #e2e2e2;
padding-bottom: 0.2em;
}
 
h3 {
font-size: 1.5em;
}
 
h4 {
font-size: 1.3em;
}
 
h5 {
font-size: 1.1em;
}
 
h6 {
font-size: 1em;
color: #555;
}
 
p {
margin-bottom: 1em;
}
 
ul, ol {
margin: 1em 0;
padding-left: 2em;
}
 
li {
margin-bottom: 0.5em;
}
 
blockquote {
border-left: 4px solid #ccc;
padding-left: 1em;
color: #666;
font-style: italic;
margin: 1.5em 0;
background: #f0f0f0;
}
 
a {
color: #3498db;
text-decoration: none;
}
 
a:hover {
text-decoration: underline;
}
 
pre {
background: #f4f4f4;
padding: 15px;
border-radius: 5px;
overflow-x: auto;
font-family: 'Courier New', Courier, monospace;
}
 
code {
background: #f4f4f4;
padding: 3px 5px;
border-radius: 3px;
font-family: 'Courier New', Courier, monospace;
}
 
table {
width: 100%;
border-collapse: collapse;
margin: 1.5em 0;
}
 
table, th, td {
border: 1px solid #ddd;
}
 
th, td {
padding: 10px;
text-align: left;
}
 
th {
background-color: #f2f2f2;
}
 
img {
max-width: 100%;
height: auto;
display: block;
margin: 1.5em auto;
}
 
hr {
border: 0;
height: 1px;
background: #ddd;
margin: 2em 0;
}
 
strong {
font-weight: bold;
}
 
em {
font-style: italic;
}`,AA="";/*! streamsaver. MIT License. Jimmy Wärting <https://jimmy.warting.se/opensource> *//*! With modifications by Paul Strong (2024)
 
/* global chrome location ReadableStream define MessageChannel TransformStream */var I=typeof window=="object"?window:void 0;I.HTMLElement||console.warn("streamsaver is meant to run on browsers main thread");var C=null,U=!1,vA=A=>{try{A()}catch{}},bA=I.WebStreamsPolyfill||{},L=I.isSecureContext,E=/constructor/i.test(I.HTMLElement)||!!I.safari||!!I.WebKitPoint,j=L||"MozAppearance"in document.documentElement.style?"iframe":"navigate",Q={createWriteStream:EA,WritableStream:I.WritableStream||bA.WritableStream,supported:!0,version:{full:"2.0.5",major:2,minor:0,dot:5},mitm:"https://jimmywarting.github.io/StreamSaver.js/mitm.html?version=2.0.0"};function W(A){if(!A)throw new Error("meh");const e=document.createElement("iframe");return e.hidden=!0,e.src=A,e.loaded=!1,e.name="iframe",e.isIframe=!0,e.postMessage=(...n)=>e.contentWindow.postMessage(...n),e.addEventListener("load",()=>{e.loaded=!0},{once:!0}),document.body.appendChild(e),e}function CA(A){const e="width=200,height=100",n=document.createDocumentFragment(),r={frame:I.open(A,"popup",e),loaded:!1,isIframe:!1,isPopup:!0,remove(){r.frame.close()},addEventListener(...a){n.addEventListener(...a)},dispatchEvent(...a){n.dispatchEvent(...a)},removeEventListener(...a){n.removeEventListener(...a)},postMessage(...a){r.frame.postMessage(...a)}},t=a=>{a.source===r.frame&&(r.loaded=!0,I.removeEventListener("message",t),r.dispatchEvent(new Event("load")))};return I.addEventListener("message",t),r}try{new Response(new ReadableStream),L&&!("serviceWorker"in navigator)&&(E=!0)}catch{E=!0}vA(()=>{const{readable:A}=new TransformStream,e=new MessageChannel;e.port1.postMessage(A,[A]),e.port1.close(),e.port2.close(),U=!0,Object.defineProperty(Q,"TransformStream",{configurable:!1,writable:!1,value:TransformStream})});function IA(){C||(C=L?W(Q.mitm):CA(Q.mitm))}function EA(A,e,n){let r={size:null,pathname:null,writableStrategy:void 0,readableStrategy:void 0,useBlob:!1},t=0,a=null,o=null,i=null;if(Number.isFinite(e)?([n,e]=[e,n],console.warn("[StreamSaver] Deprecated pass an object as 2nd argument when creating a write stream"),r.size=n,r.writableStrategy=e):e&&e.highWaterMark?(console.warn("[StreamSaver] Deprecated pass an object as 2nd argument when creating a write stream"),r.size=n,r.writableStrategy=e):r=e||{},E=E||r.useBlob,!E){IA(),o=new MessageChannel,A=encodeURIComponent(A.replace(/\//g,":")).replace(/['()]/g,escape).replace(/\*/g,"%2A");const l={transferringReadable:U,pathname:r.pathname||Math.random().toString().slice(-6)+"/"+A,headers:{"Content-Type":"application/octet-stream; charset=utf-8","Content-Disposition":"attachment; filename*=UTF-8''"+A}};r.size&&(l.headers["Content-Length"]=r.size);const g=[l,"*",[o.port2]];if(U){const c=j==="iframe"?void 0:{transform(u,m){if(!(u instanceof Uint8Array))throw new TypeError("Can only write Uint8Arrays");t+=u.length,m.enqueue(u),a&&(location.href=a,a=null)},flush(){a&&(location.href=a)}};i=new Q.TransformStream(c,r.writableStrategy,r.readableStrategy);const d=i.readable;o.port1.postMessage({readableStream:d},[d])}o.port1.onmessage=c=>{c.data.download?j==="navigate"?(C.remove(),C=null,t?location.href=c.data.download:a=c.data.download):(C.isPopup&&(C.remove(),C=null,j==="iframe"&&W(Q.mitm)),W(c.data.download)):c.data.abort&&(s=[],o.port1.postMessage("abort"),o.port1.onmessage=null,o.port1.close(),o.port2.close(),o=null)},C.loaded?C.postMessage(...g):C.addEventListener("load",()=>{C.postMessage(...g)},{once:!0})}let s=[];return!E&&i&&i.writable||new Q.WritableStream({write(l){if(!(l instanceof Uint8Array))throw new TypeError("Can only write Uint8Arrays");if(E){s.push(l);return}o.port1.postMessage(l),t+=l.length,a&&(location.href=a,a=null)},close(){if(E){const l=new Blob(s,{type:"application/octet-stream; charset=utf-8"});T().saveFile(l,A)}else o.port1.postMessage("end")},abort(){s=[],o.port1.postMessage("abort"),o.port1.onmessage=null,o.port1.close(),o.port2.close(),o=null}},r.writableStrategy)}var eA=Q,Z;async function tA(A="export.zip"){Z||({Writer:Z}=await N());const e=eA.createWriteStream("export.zip",{useBlob:navigator.userAgent.indexOf("Electron")<0}),{readable:n,writable:r}=new Z;let t;if(window.WritableStream&&n.pipeTo)t=n.pipeTo(e).then(()=>{console.log("Export completed")});else{const a=e.getWriter(),o=n.getReader(),i=()=>o.read().then(s=>s.done?a.close():a.write(s.value).then(i));i()}return{writable:r,pipePromise:t}}var q="";async function hA(A,e=!1){try{const n=await QA(A);if(!n)return null;if(e&&n.type==="image/gif"){const r=URL.createObjectURL(n),t=await nA(r,"dataurl");return URL.revokeObjectURL(r),t}else{const r=new FileReader;return await new Promise((a,o)=>{r.onloadend=()=>{a(r.result)},r.onerror=o,r.readAsDataURL(n)})}}catch(n){return console.warn(`Failed to load image: ${n.message}`),null}}function nA(A,e="dataurl"){return new Promise((n,r)=>{const t=new Image;t.onload=()=>{const a=document.createElement("canvas");if(a.width=t.width,a.height=t.height,a.getContext("2d").drawImage(t,0,0,a.width,a.height),e==="blob")sA(a,"image/png").then(i=>n(i));else if(e==="dataurl"){const i=a.toDataURL("image/png");n(i)}},t.onerror=()=>{r(new Error(`Failed to load image: ${A}`))},t.src=A})}async function wA(A,e="blob"){const n=URL.createObjectURL(A),r=await nA(n,e);return URL.revokeObjectURL(n),r}async function QA(A){const e=await rA(A);return e?await e.blob():null}async function rA(A){try{const e=oA(A);if(!e)return null;const n=await fetch(e,{mode:"cors"});if(!n.ok)return console.warn(`Bulk Export - Failed to fetch image: ${n.status} ${n.statusText}`),null;const r=n.headers.get("Content-Type");return!r||!r.startsWith("image/")?(console.warn(`Bulk Export - URL does not point to a valid image: ${A}`),null):n}catch(e){return console.warn(`Bulk Export - Failed to load image: ${e.message}`),null}}function xA(A){return new Promise((e,n)=>{const r=new Image;r.onload=()=>{e(r)},r.onerror=()=>{n(new Error("Failed to load image from Dataurl."))},r.src=A})}function VA(A){return R(A)?aA(A):oA(A)}async function aA(A){if(R(A)){const e=A.split("/").pop(),n=await T().getAttachmentURL(e);if(!n)return A;const r=new URL("https://plugins.amplenote.com/cors-proxy");return r.searchParams.set("apiurl",n),r.toString()}else return A}function oA(A){if(/^data:image\/(png|jpeg|jpg|gif|webp);base64,/.test(A)||/^(?:\.\/|\.\.\/|[^\/\\]+\w+\/?)*\w+\.\w+$/.test(A))return A;let r;try{r=new URL(A)}catch{return console.warn(`Bulk Export - Invalid URL provided: ${A}`),q}if(!["http:","https:"].includes(r.protocol))return console.warn(`Unsupported URL protocol (${r.protocol}): ${A}`),q;if(r.hostname==="images.amplenote.com"){const t=new URL("https://plugins.amplenote.com/cors-proxy");return t.searchParams.set("apiurl",A),t.toString()}else return A}async function yA(A){try{const e=await aA(A);if(!e)return null;const n=await fetch(e,{mode:"cors"});return n.ok?n:(console.warn(`Bulk Export - Failed to fetch attachment: ${n.status} ${n.statusText}`),null)}catch(e){return console.warn(`Bulk Export - Failed to load attachment: ${e.message}`),null}}function R(A){return A&&A.startsWith("attachment://")}function PA(A){let e;try{e=new URL(A)}catch{return!1}return e.protocol==="http:"||e.protocol==="https:"}function sA(A,e="image/png",n=1){return new Promise((r,t)=>{A.toBlob(a=>{a?r(a):t(new Error("Canvas is empty or conversion failed."))},e,n)})}function iA(A,e,n){const r=e.split(" "),t=[];let a="";for(let o of r){const i=a+o+" ";A.measureText(i).width>n&&a!==""?(t.push(a.trim()),a=o+" "):a=i}return t.push(a.trim()),t}function XA(A,e,n){const{x:r,y:t,width:a,height:o}=e,i='"Segoe UI", "Helvetica Neue", "Helvetica", "Arial", sans-serif';let s=200;A.fillStyle="white",A.textAlign="center",A.textBaseline="top",A.font=`${s}px ${i}`;let l=iA(A,n,a-40);for(;l.length*(s*1.2)>o&&s>10;)s-=2,A.font=`${s}px ${i}`,l=iA(A,n,a-40);const g=l.length*(s*1.2);let c=t+(o-g)/2;const d=a,u=g;A.fillStyle="rgb(49, 58, 71)",A.fillRect(r,t,a,o),A.fillStyle="white";for(let m of l)A.fillText(m,r+a/2,c),c+=s*1.2}async function lA(A,e){const n=await xA(A),r=document.createElement("canvas"),t=r.getContext("2d");return r.width=1600,r.height=2600,t.drawImage(n,0,0,r.width,r.height),e&&XA(t,{x:285,y:1548,width:1030,height:800},e),await sA(r,"image/png")}async function h(A,e){return T().saveFile(A,e)}function qA(A){return/\|(?:[^\n|]*\|)+\n\|(?:[\s\-:]*\|)+\n(\|(?:[^\n|]*\|)+\n)*/g.test(A)}function HA(A){const e=/(\|(?:[^\n|]*\|)+\n\|(?:[\s\-:]*\|)+\n(?:\|(?:[^\n|]*\|)+\n)*)\\\n/g;return A.replace(e,`$1
\\
`)}function DA(A){const e=/\\{1}$/gm;return A.replace(e,`
`)}function cA(A){const{name:e,uuid:n,created:r,updated:t,tags:a}=A,o=a&&a.length>0?`tags:
${a.map(i=>` - ${i}`).join(`
`)}
`:"";return`---
title: '${e}'
uuid: ${n}
created: '${r}'
updated: '${t}'
${o}---`}function x(A){var e=A.split(/\D+/);return new Date(Date.UTC(e[0],--e[1],e[2],e[3],e[4],e[5],e[6]))}function zA(A){return'"'+(A&&A.replaceAll(/\"/g,'""')||"")+'"'}function D(A){try{return encodeURI(A)}catch{return console.error("Invalid URL or path provided:",A),A}}function z(A,e,n={}){const r=[];let t=0;for(;t<A.length;){const a=A[t];switch(a.type){case"heading_open":{const o=parseInt(a.tag.substring(1),10);t++;const i=A[t],s=H(i.children,e,n);r.push({text:s,style:o>3?"h3":a.tag}),t++}break;case"paragraph_open":{t++;const o=A[t];let i;if(o.type==="inline")i=H(o.children,e,n),t++;else{const l=[];for(;A[t].type!=="paragraph_close";)l.push(A[t]),t++;i=z(l,e,n)}kA(i).forEach(l=>r.push(l)),t++}break;case"bullet_list_open":case"ordered_list_open":{const o=a.type==="bullet_list_open"?"ul":"ol",i=[];t++;let s=1;for(;s>0&&t<A.length;){const l=A[t];if(l.type==="bullet_list_open"||l.type==="ordered_list_open")s++;else if((l.type==="bullet_list_close"||l.type==="ordered_list_close")&&(s--,s===0)){t++;break}if(l.type==="list_item_open"){const g=[];let c=1;for(t++;c>0&&t<A.length;){const u=A[t];if(u.type==="list_item_open")c++;else if(u.type==="list_item_close"&&(c--,c===0)){t++;break}g.push(u),t++}const d=z(g,e,n);d.length===1?i.push(d[0]):i.push({stack:d})}else t++}r.push({[o]:i,style:"listItem"})}break;case"blockquote_open":{const o=[];t++;let i=1;for(;i>0&&t<A.length;){const l=A[t];if(l.type==="blockquote_open")i++;else if(l.type==="blockquote_close"&&(i--,i===0)){t++;break}o.push(l),t++}const s=z(o,e,n);r.push({stack:s,style:"quote"})}break;case"code_block":case"fence":{const o=a.content;r.push(TA(o)),t++}break;case"hr":r.push({canvas:[{type:"line",x1:0,y1:0,x2:515,y2:0,lineWidth:1,lineColor:"#000000"}],margin:[0,10,0,10]}),t++;break;case"inline":{const o=H(a.children,e,n);r.push({text:o,margin:[0,5,0,5]}),t++}break;case"table_open":{const o=[];let i=1;for(t++;i>0&&t<A.length;){const l=A[t];if(l.type==="table_open")i++;else if(l.type==="table_close"&&(i--,i===0)){t++;break}o.push(l),t++}const s=OA(o,e,n);r.push(s)}break;default:t++;break}}return r}function OA(A,e,n){const r=[];let t=0,a=[],o=[];for(;t<A.length;){const i=A[t];switch(i.type){case"thead_open":{t++;const s=[];for(;A[t].type!=="thead_close";)if(A[t].type==="tr_open"){for(t++;A[t].type!=="tr_close";)if(A[t].type==="th_open"){const l=i.attrs&&i.attrs[0]?i.attrs[0][1]:"left";o.push(l),t++;const g=A[t],c=H(g.children,e,n);s.push({text:c,style:"tableHeader",alignment:l}),t++,t++}else t++;t++}else t++;a=s,r.push(a),t++}break;case"tbody_open":{for(t++;A[t].type!=="tbody_close";)if(A[t].type==="tr_open"){t++;const s=[];let l=0;for(;A[t].type!=="tr_close";)if(A[t].type==="td_open"){const g=o[l]||"left";t++;const c=A[t],d=H(c.children,e,n);s.push({text:d,alignment:g}),t++,t++,l++}else t++;r.push(s),t++}else t++;t++}break;default:t++;break}}return{table:{headerRows:a.length>0?1:0,widths:Array(a.length).fill("*"),body:r},layout:"lightHorizontalLines",margin:[0,5,0,15]}}function H(A,e,n){const r=[];let t={},a=null;return A.forEach(o=>{switch(o.type){case"text":{const c={text:o.content==="\\"?" ":o.content};Object.keys(t).length>0&&Object.assign(c,t),a&&(c.link=a,c.color="blue",c.decoration="underline"),r.push(c)}break;case"strong_open":t.bold=!0;break;case"strong_close":delete t.bold;break;case"em_open":t.italics=!0;break;case"em_close":delete t.italics;break;case"s_open":t.decoration="lineThrough";break;case"s_close":delete t.decoration;break;case"link_open":a=o.attrs.find(g=>g[0]==="href")[1];break;case"link_close":a=null;break;case"html_inline":const l=new DOMParser().parseFromString(o.content,"text/html").querySelector("mark");if(l){const g=l.getAttribute("style");g&&g.split(";").forEach(c=>{const[d,u]=c.split(":").map(m=>m.trim());d==="background-color"?t.background=u:d==="color"&&(t.color=u)})}else o.content.includes("</mark>")&&(delete t.background,delete t.color);break;case"code_inline":{const g={text:o.content,style:"codeInline"};r.push(g)}break;case"image":{const g=o.attrs.find(p=>p[0]==="src")[1],c=o.attrs.find(p=>p[0]==="alt"),d=o.attrs.find(p=>p[0]==="width"),u=d?d[1]:250,m=c?c[1]:"";let B;const b=u>e?e:u;(PA(g)||/^data:image\/(png|jpeg|jpg|gif|webp);base64,/.test(g))&&(B=Object.keys(n).find(p=>n[p]===g),B||(B=`image_${Object.keys(n).length}`,n[B]=g)),r.push({image:B||g,width:b,alt:m})}break;case"hardbreak":r.push({text:`
`,...t});break;default:break}}),r}function kA(A,e){const n=[];if(A.some(i=>i.image)){if(A.length===1)return[A]}else return A.length>1?[{text:A}]:[A];let t=[],a=[];const o=function(){a.length===1?t.push(a[0]):t.push({text:a}),a=[]};return A.forEach(i=>{i.image?(a.length>0&&o(),t.push(i)):a.push(i)}),a.length>0&&o(),t.forEach((i,s)=>{if(i.text){const l=s===t.length-1;i.width=l?"*":"auto"}}),[{columns:t}]}function TA(A){return{table:{widths:["*"],body:[[{text:A,margin:[10,10,10,10],fontSize:11}]]},layout:{fillColor:"#f0f0f0",hLineWidth:()=>0,vLineWidth:()=>0,paddingLeft:()=>0,paddingRight:()=>0,paddingTop:()=>0,paddingBottom:()=>0},margin:[0,0,0,20]}}var NA=z;function FA(A){A.heading_open=(e,n)=>{const r=e[n].markup.length;return`${{1:"\\section",2:"\\subsection",3:"\\subsubsection",4:"\\paragraph",5:"\\subparagraph",6:"\\textbf"}[r]}{`},A.heading_close=()=>`}
`,A.paragraph_open=()=>"",A.paragraph_close=()=>`
 
`,A.em_open=()=>"\\emph{",A.em_close=()=>"}",A.strong_open=()=>"\\textbf{",A.strong_close=()=>"}",A.s_open=()=>"\\sout{",A.s_close=()=>"}",A.code_inline=(e,n)=>{const r=gA(e[n].content),t=LA(r);return`\\lstinline${t}${r}${t}`},A.fence=(e,n)=>`\\begin{lstlisting}
${e[n].content}\\end{lstlisting}
`,A.blockquote_open=()=>`\\begin{quote}
`,A.blockquote_close=()=>`\\end{quote}
`,A.bullet_list_open=()=>`\\begin{itemize}
`,A.bullet_list_close=()=>`\\end{itemize}
`,A.ordered_list_open=()=>`\\begin{enumerate}
`,A.ordered_list_close=()=>`\\end{enumerate}
`,A.list_item_open=()=>"\\item ",A.list_item_close=()=>`
`,A.link_open=(e,n)=>{const r=e[n].attrGet("href");return`\\href{${O(r)}}{`},A.link_close=()=>"}",A.image=(e,n)=>{const r=e[n],t=r.attrs.find(b=>b[0]==="src"),a=t?t[1]:"",o=r.attrs.find(b=>b[0]==="alt"),i=o?o[1]:"",s=r.attrs.find(b=>b[0]==="title"),l=s?s[1]:"",g=r.content.split("|"),c=g[0].trim(),d=g.length>1?parseInt(g[1],10):250;let m=[`width=${jA(d).toFixed(2)}cm`];const B=m.length?`[${m.join(",")}]`:"";return l?`
\\begin{figure}[h]
\\centering
\\includegraphics${B}{${a}}
\\caption{${O(l)}}
\\label{fig:${O(i||"image")}}
\\end{figure}
`:`\\includegraphics${B}{${a}}
`},A.hardbreak=()=>`\\\\
`,A.softbreak=()=>`\\\\
`,A.html_inline=(e,n)=>{const r=e[n].content.trim();if(r.startsWith("<!--")&&r.endsWith("-->"))return"";if(r==="<br>"||r==="<br/>")return`\\\\
`;if(r.startsWith("<mark")){let t=function(l){return l=l.replace("#",""),l};const a=r.match(/style="([^"]*)"/);let o="",i="";a&&a[1].split(";").map(c=>c.trim()).forEach(c=>{const[d,u]=c.split(":").map(m=>m.trim());d==="background-color"&&(o=u),d==="color"&&(i=u)});let s="";if(o){const l=t(o);s+=`\\colorbox[HTML]{${l}}{`}if(i){const l=t(i);s+=`\\textcolor[HTML]{${l}}{`}return!o&&!i&&(s+="\\hl{"),e[n].markStyleStack={bgColor:o,textColor:i},s}if(r==="</mark>"){let t="";const a=e[n-1],o=a&&a.markStyleStack;return o?(o.textColor&&(t+="}"),o.bgColor&&(t+="}")):t+="}",t}return""},A.text=(e,n)=>{let r=e[n].content;return r=gA(r),r=UA(r),r=O(r),r},A.hr=()=>`\\hrule
`,A.table_open=(e,n,r,t,a)=>{let o=0;for(let s=n+1;s<e.length;s++)if(e[s].type==="tr_open"){for(let l=s+1;l<e.length;l++)if(e[l].type==="th_open"||e[l].type==="td_open")o++;else if(e[l].type==="tr_close")break;break}return t.tableColumnCount=o,`\\begin{tabularx}{\\textwidth}{${"|"+"X|".repeat(o)}}
\\hline
`},A.table_close=()=>`\\end{tabularx}
`,A.thead_open=(e,n,r,t,a)=>(t.inHeader=!0,""),A.thead_close=(e,n,r,t,a)=>(t.inHeader=!1,""),A.tbody_open=()=>"",A.tbody_close=()=>"",A.tr_open=(e,n,r,t,a)=>(t.tableCellIndex=0,""),A.tr_close=(e,n,r,t,a)=>` \\\\\\hline
`,A.th_open=(e,n,r,t,a)=>"",A.th_close=(e,n,r,t,a)=>(t.tableCellIndex++,t.tableCellIndex<t.tableColumnCount?" & ":""),A.td_open=(e,n,r,t,a)=>"",A.td_close=(e,n,r,t,a)=>(t.tableCellIndex++,t.tableCellIndex<t.tableColumnCount?" & ":"")}function O(A){const e={"\\":"\\textbackslash{}","{":"\\{","}":"\\}",$:"\\$","&":"\\&","%":"\\%","#":"\\#",_:"\\_","^":"\\^{}","~":"\\~{}"};return A.replace(/([\\{}$&%#_^~])/g,n=>e[n])}function gA(A){return A.replace(/\\([\[\]\|])/g,"$1")}function UA(A){const e={"&quot;":'"',"&amp;":"\\&","&lt;":"<","&gt;":">","&nbsp;":" "};return A.replace(/&[a-z]+;/g,n=>e[n]||n)}function LA(A){const e=["|","/","@","%","#","!","~","=","+","-","*",":",";","?"];for(const n of e)if(!A.includes(n))return n;return"|"}function jA(A){return A*2.54/96}function WA(A){A.renderAsync=async function(e,n={},r){const t=A.parse(e,n);return typeof r=="function"&&await r(t,n),A.renderer.render(t,A.options,n)}}var dA=class{constructor(A={}){this.options={imageStrategy:"dataurl",attachmentStrategy:"silent",imageFolder:"images",attachmentFolder:"attachments",assetFilenamePrefix:null,exportAttachments:null,replaceBrokenImages:!0,renderGifAsStill:!1,...A},this.options.exportAttachments&&(this.options.attachmentStrategy="local"),this.pathCache=[],this.urlCache={},this.collectedImages={},this.collectedAttachments={},this.idCounters={},this.asset_plugin=this.asset_plugin.bind(this)}initialize(){}setAssetFilenamePrefix(A){this.options.assetFilenamePrefix=A}resetAssetFilenamePrefix(){this.options.assetFilenamePrefix=null}setAssetStrategy(A,e){this.options.imageStrategy=A,this.options.attachmentStrategy=e}asset_plugin(A){if(A.renderAsync||A.use(WA),!A.renderWithAssets){const e=this.processAssets.bind(this);A.renderWithAssets=function(n,r={}){return this.renderAsync(n,r,e)}}A.core.ruler.push("collect_assets",e=>{this._collectAssets(e.tokens,e.env)})}_collectAssets(A,e){const n=this;function r(t,a){let o=a;for(let i=0;i<t.length;i++){const s=t[i];if(o&&s.type!=="link_close"){s.hidden=!0,s.children&&s.children.length>0&&r(s.children,o);continue}if(s.type==="image")n.options.imageStrategy==="silent"?s.hidden=!0:n._collectImageToken(s);else if(s.type==="link_open"){const l=s.attrGet("href");if(R(l))if(n.options.exportAttachments===!1||n.options.attachmentStrategy==="silent")o=!0,s.hidden=!0;else{let g="attachment",c=t[i+1];c&&c.type==="text"&&(g=c.content),n._collectAttachmentToken(s,l,g)}}else s.type==="link_close"&&o&&(o=!1,s.hidden=!0);s.children&&s.children.length>0&&r(s.children,o)}}r(A,!1)}_collectImageToken(A){const e=A.attrGet("src");this.options.imageStrategy!=="ignore"&&(this.collectedImages[e]||(this.collectedImages[e]={tokens:[],type:"image",src:e}),this.collectedImages[e].tokens.push(A))}_collectAttachmentToken(A,e,n){this.options.attachmentStrategy!=="ignore"&&(this.collectedAttachments[e]||(this.collectedAttachments[e]={tokens:[],filenames:[],type:"attachment",src:e}),this.collectedAttachments[e].tokens.push(A),this.collectedAttachments[e].filenames.push(n))}async processAssets(A,e){await Promise.all(Object.keys(this.collectedImages).map(async n=>{const r=this.collectedImages[n],t=this.options.imageStrategy;let a=n;if(!this.collectedImages[n].processed){if(t==="dataurl"){let o=await hA(n,this.options.renderGifAsStill);!o&&this.options.replaceBrokenImages&&(o=q),a=o||n,this.urlCache[n]=a}else if(t==="local"){let o=n;const i=o.split(".");this.options.renderGifAsStill&&i.pop()==="gif"&&(i.push("png"),o=i.join("."));const s=this._getUniqueFilePath(o,"image");this.urlCache[n]=D(s),this.collectedImages[n].filePath=s,this.collectedImages[n].type="image"}else if(t==="uniqueId"){const o=this._getUniqueId("image");this.urlCache[n]=o,this.collectedImages[n].uniqueId=o,this.collectedImages[n].type="image"}else if(t==="proxify"){let o=VA(n);!o&&this.options.replaceBrokenImages&&(o=q),a=o||n,this.urlCache[n]=a}this.collectedImages[n].strategy=t,this.collectedImages[n].processed=!0}r.tokens.forEach(o=>{if(o.meta&&o.meta.assetProcessed)return;o.attrSet("src",this.urlCache[n]);const i=o.content.split("|"),s=i.length>1?parseInt(i[i.length-1],10):null;if(s){o.attrSet("width",s);const l=i.slice(0,i.length-1).join("|");o.content=o.children[0].content=l}o.meta||(o.meta={}),o.meta.assetProcessed=!0})})),await Promise.all(Object.keys(this.collectedAttachments).map(async n=>{const r=this.collectedAttachments[n],t=this.options.attachmentStrategy;if(!r.processed){if(t==="local"){let a=r.filenames[0]||"attachment";const o=this._getUniqueFilePath(a,"attachment");this.urlCache[n]=D(o),r.filePath=o}else if(t==="uniqueId"){const a=this._getUniqueId("attachment");this.urlCache[n]=a,r.filePath=a}r.strategy=t,r.processed=!0}r.tokens.forEach(a=>{a.meta&&a.meta.assetProcessed||(a.attrSet("href",this.urlCache[n]),a.meta||(a.meta={}),a.meta.assetProcessed=!0)})}))}async getLocalImages(){return this._collectFromStream(this.streamLocalImages())}async getLocalAttachments(){return this._collectFromStream(this.streamLocalAttachments())}async getLocalAssets(){return this._collectFromStream(this.streamLocalAssets())}async _collectFromStream(A){const e=[],n=A.getReader();let r;for(;!(r=await n.read()).done;)e.push(r.value);return e}streamLocalImages(){const A=Object.values(this.collectedImages).filter(e=>e.strategy==="local")[Symbol.iterator]();return this._createAssetStreamFromIterator(A)}streamLocalAttachments(){const A=Object.values(this.collectedAttachments).filter(e=>e.strategy==="local")[Symbol.iterator]();return this._createAssetStreamFromIterator(A)}streamLocalAssets(){const e=[...Object.values(this.collectedImages),...Object.values(this.collectedAttachments)].filter(n=>n.strategy==="local")[Symbol.iterator]();return this._createAssetStreamFromIterator(e)}_createAssetStreamFromIterator(A){const e=this;return new ReadableStream({async pull(n){let r;do{if(r=A.next(),r.done){n.close();return}const t=r.value,a=t.filePath||t.uniqueId;let o;if(t.type==="image"){if(o=await rA(t.src),o&&o.headers.get("content-type")==="image/gif"&&e.options.renderGifAsStill){const i=await o.blob(),s=await wA(i);o=new Response(s,{headers:{"Content-Type":"image/png"}})}!o&&e.options.replaceBrokenImages&&(o=await fetch(q))}else t.type==="attachment"&&(o=await yA(t.src));if(o){const i=o.headers.get("content-type");n.enqueue({name:a,mimeType:i,blob:async()=>await o.blob(),stream:()=>o.body});return}else console.info("Could not find asset: "+a)}while(!0)}})}_getUniqueFilePath(A,e){const{assetFilenamePrefix:n,imageFolder:r,attachmentFolder:t}=this.options;let a=e==="image"?r:t,o=ZA(A);n&&(a=`${a}/${n}`);let s=`${a}/${o}`,l=1;for(;this.pathCache.includes(s);)s=`${a}/${RA(o,"-"+l)}`,l++;return this.pathCache.push(s),s}_getUniqueId(A){return this.idCounters[A]?this.idCounters[A]++:this.idCounters[A]=1,`${A}${this.idCounters[A]}`}};function ZA(A){const e=A.match(/[^/\\&\?]+\.\w{3,4}(?=([\?&].*$|$))/);return e?e[0]:A}function RA(A,e){const n=A.lastIndexOf(".");return n===-1?A+e:A.substring(0,n)+e+A.substring(n)}var M,S,K,J,fA,y=class{constructor(A,e,n=new dA({exportAttachments:!1})){this.reset(A,e),this.assetExporter=n}async initialize(){await this._ensureBaseDependenciesLoaded(),this.md=M({html:!0,linkify:!0,typographer:!0}),this.md.use(this.assetExporter.asset_plugin),this.latexMd=M({html:!0}),this.latexMd.use(this.assetExporter.asset_plugin),FA(this.latexMd.renderer.rules)}reset(A,e){this.title=A,this.markdownText=e?this._preprocessMarkdown(e):null,this.htmlContent=null}_preprocessMarkdown(A){return DA(HA(A))}async _loadJsPDF(){S||(S=await pA()),await mA()}async _loadHtmlDocx(){K||(K=await BA())}async _loadEpubGen(){J||(J=await $())}async _loadConflux(){fA||({Writer:fA}=await N())}async _ensureBaseDependenciesLoaded(){[M]=await Promise.all([uA()])}async _getHtmlContent(){return this.htmlContent||(this.htmlContent=await this.md.renderWithAssets(this.markdownText),this._removeEmptyTableHeaders()),this.htmlContent}_removeEmptyTableHeaders(){const e=new DOMParser().parseFromString(this.htmlContent,"text/html");e.querySelectorAll("thead").forEach(r=>{r.textContent.trim()||r.remove()}),this.htmlContent=e.body.innerHTML}async toMarkdown(){this.assetExporter.setAssetStrategy("local","local"),this.assetExporter.options.renderGifAsStill=!1,this.md.parse(this.markdownText,{}),await this.assetExporter.processAssets();let A=this.markdownText;for(const e of Object.values(this.assetExporter.collectedImages))A=A.replaceAll(e.src,D(e.filePath));for(const e of Object.values(this.assetExporter.collectedAttachments))A=A.replaceAll(e.src,D(e.filePath));return A}async toHTML(A=!0){const e=await this._getHtmlContent();return A?`
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>${this.title}</title>
<style>${F}</style>
</head>
<body>
${e}
</body>
</html>
`:e}async saveHTML(A="output.html"){const e=await this.toHTML(),n=new Blob([e],{type:"text/html;charset=utf-8"});await h(n,A)}async toLaTeX(){this.assetExporter.setAssetStrategy("local","local"),this.assetExporter.options.renderGifAsStill=!0;const A=await this.latexMd.renderWithAssets(this.markdownText),e=`
\\documentclass{article}
\\usepackage[utf8]{inputenc}
\\usepackage[T1]{fontenc}
\\usepackage{hyperref}
\\usepackage{graphicx}
\\usepackage{xcolor}
\\usepackage{ulem}
\\usepackage{soul}
\\usepackage{listings}
\\usepackage{emoji}
\\usepackage{tabularx}
\\lstset{
breaklines=true,
breakatwhitespace=true,
columns=fullflexible,
keywordstyle=color{blue},
commentstyle=color{gray},
stringstyle=color{orange},
}
\\begin{document}
`,n=`
\\end{document}
`;return e+A+n}async saveLaTeX(A="output.tex"){const e=await this.toLaTeX(),n=new Blob([e],{type:"text/plain;charset=utf-8"});if(Object.keys(this.assetExporter.collectedImages).length===0)h(n,A);else{const{writable:r,pipePromise:t}=await tA("export.zip"),a=r.getWriter();await a.write({name:A,stream:()=>n.stream()}),a.releaseLock(),await this.assetExporter.streamLocalImages().pipeTo(r),t&&await t}}async toDOCX(){await this._loadHtmlDocx(),this.assetExporter.setAssetStrategy("dataurl","local");const A=await this.toHTML();return K.asBlob(A)}async saveDOCX(A="output.docx"){const e=await this.toDOCX();h(e,A)}async toPDF(){await this._loadJsPDF(),this.assetExporter.setAssetStrategy("dataurl","local"),this.assetExporter.options.renderGifAsStill=!0;const A=this.md.parse(this.markdownText,{});await this.assetExporter.processAssets();const e={},n=595.28,r=[40,60,40,60],t=n-r[0]-r[2],a=NA(A,t,e),o={pageSize:"A4",pageMargins:r,content:[{text:this.title,style:"h2"},{canvas:[{type:"line",x1:r[0],y1:0,x2:n-r[2],y2:0,lineWidth:.5,lineColor:"#F3F4F4"}]},...a],images:e,styles:{h1:{fontSize:18.02,bold:!0,margin:[0,20,0,10]},h2:{fontSize:15.76,bold:!0,margin:[0,10,0,5]},h3:{fontSize:13.51,bold:!0,margin:[0,10,0,5]},quote:{italics:!0,color:"gray",margin:[0,10,0,10]},codeBlock:{fontSize:10,margin:[0,5,0,5],background:"#f0f0f0",fillColor:"111111",color:"#333333"},codeInline:{fontSize:10,background:"#f0f0f0",color:"#333333"},listItem:{margin:[0,5,0,5]},tableHeader:{bold:!0,fillColor:"#eeeeee"}}};return new Promise((i,s)=>{S.createPdf(o).getBlob(l=>{l?i(l):s(console.error("Failed to create PDF Blob"))})})}async savePDF(A="output.pdf"){const e=await this.toPDF();h(e,A)}async toEPUB(){await this._loadEpubGen(),this.assetExporter.setAssetStrategy("proxify","silent");const A=await lA(AA,this.title);return await new J.EPub({title:this.title,css:F,cover:new File([A],"cover.png",{type:"image/png"}),ignoreFailedDownloads:!0},[{title:this.title,content:await this.toHTML(!1)}]).genEpub()}async saveEPUB(A="output.epub"){const e=await this.toEPUB();h(e,A)}async tableToCSV(){const A=await this._getHtmlContent(),r=new DOMParser().parseFromString(A,"text/html").querySelector("table");if(r){let t="";return r.querySelectorAll("tr").forEach((o,i)=>{const s=o.querySelectorAll("th, td"),l=Array.from(s).map(g=>{let c=g.textContent.trim();return(c.includes('"')||c.includes(",")||c.includes(`
`))&&(c='"'+c.replace(/"/g,'""')+'"'),c}).join(",");t+=l+`
`}),t}return""}async saveTableCSV(A="table.csv"){const e=new Blob([await this.tableToCSV()],{type:"text/csv;charset=utf-8"});h(e,A)}async saveTableCSVIfTable(A="table.csv"){const e=await this.tableToCSV();e!==""&&h(new Blob([e],{type:"text/csv;charset=utf-8"}),A)}},MA,Y,SA=class{constructor(A,e,n,r={}){this._app=A,this.exportNoteList=e,this.format=n,this.options={directory:"tag",includeAttachments:!0,...r}}async initialize(){await this._ensureBaseDependenciesLoaded()}async _ensureBaseDependenciesLoaded(){({Writer:MA}=await N())}_logProgress(A,e){this.options.progressCallback&&this.options.progressCallback(A,e)}async _loadEpubGen(){Y||(Y=await $())}async export(){if(this.format==="csv"){await this.exportCSV();return}if(this.format==="epub-single"){await this.exportEpubNotebook();return}const A=new dA({exportAttachments:this.options.includeAttachments}),e=new y(null,null,A);await e.initialize(),this._logProgress(0,`Starting Bulk Export of ${this.exportNoteList.length} notes`);const{writable:n,pipePromise:r}=await tA(),t=n.getWriter();let a=0;const o=[];for(const s of this.exportNoteList){const l=await this._app.getNoteContent({uuid:s.uuid});let g=s.name||s.uuid;e.reset(s.name,l);let c=g.replace(/[<>:"/\\|?*]+/g,"_"),d;switch(o.includes(c)?(d=c+`(${s.uuid})`,this._logProgress(a,`${c}.${this.format} already exists! Renaming to ${d}.${this.format}.`)):d=c,o.push(d),this.format){case"pdf":const u=await e.toPDF();await t.write({name:d+".pdf",lastModified:x(s.updated),stream:()=>u.stream()});break;case"docx":const m=await e.toDOCX();await t.write({name:d+".docx",lastModified:x(s.updated),stream:()=>m.stream()});break;case"epub":const B=await e.toEPUB();await t.write({name:d+".epub",lastModified:x(s.updated),stream:()=>B.stream()});break;case"latex":const b=await e.toLaTeX();await t.write({name:d+".tex",lastModified:x(s.updated),stream:()=>new Response(b,{headers:{"Content-Type":"application/x-tex"}}).body});break;case"html":e.assetExporter.setAssetStrategy("local","local");const p=await e.toHTML();await t.write({name:d+".html",lastModified:x(s.updated),stream:()=>new Response(p,{headers:{"Content-Type":"text/html"}}).body});break;case"md":e.assetExporter.setAssetStrategy("local","local");const v=await e.toMarkdown(),G=cA(s)+`
 
`+v;await t.write({name:d+".md",lastModified:x(s.updated),stream:()=>new Response(G,{headers:{"Content-Type":"text/markdown"}}).body});break}a++,this._logProgress(a,`${d}.${this.format}`)}t.releaseLock();let i;this.options.includeAttachments?i=e.assetExporter.streamLocalAssets():i=e.assetExporter.streamLocalImages(),i.pipeTo(n),this._logProgress(a,"Export Complete"),r?await r:console.log("Export completed")}async exportEpubNotebook(){await this._loadEpubGen();const A=await this._app.prompt("EPub Options",{inputs:[{label:"Book Title (will be embedded in cover)",type:"string"}]});if(!A)return;this._logProgress(0,`Starting ePub Export of ${this.exportNoteList.length} notes`);const e=await lA(AA,A),n=new y;await n.initialize(),n.assetExporter.setAssetStrategy("proxify","silent");let r=0;const t=this.exportNoteList.map(async s=>(n.reset(s.name||"",await this._app.getNoteContent({uuid:s.uuid})),r++,this._logProgress(r,s.name||""),{title:s.name||"",content:await n.toHTML(!1)})),a=await Promise.all(t);this._logProgress(r,"Building eBook");const i=await new Y.EPub({title:A,css:F,cover:new File([e],"cover.png",{type:"image/png"}),ignoreFailedDownloads:!0},a).genEpub();h(i,A.replace(/[^A-Za-z0-9 ]/g,"_")+".epub"),this._logProgress(r,"Export Complete")}async exportCSV(){const A=`UUID,Title,Tags,Content
`,e=eA.createWriteStream("export.csv",{useBlob:navigator.userAgent.indexOf("Electron")<0}),n=this.exportNoteList.values(),r=new TextEncoder;this._logProgress(0,`Starting CSV Export of ${this.exportNoteList.length} notes`);const t=this._app;let a=0;const o=(s,l)=>this._logProgress(s,l);await new ReadableStream({start(s){s.enqueue(r.encode(A))},async pull(s){const{done:l,value:g}=n.next();if(l)return s.close();const c=await t.getNoteContent({uuid:g.uuid}),d=g.tags?g.tags.join():"",m=[g.uuid,g.name,d,c].map(B=>zA(B)).join()+`
`;return a++,o(a,g.name),s.enqueue(r.encode(m))}}).pipeTo(e),this._logProgress(a,"Export Complete")}},KA=`<style>
/* Define CSS variables for light and dark themes */
:root {
--background-color: #ffffff;
--text-color: #333333;
--progress-bar-bg: #e0e0e0;
--progress-bar-fill: #4caf50;
--log-bg: #f9f9f9;
--log-border: #ddd;
--delete-button-bg: #ff4d4d;
--cancel-button-bg: #ff9800;
}
 
/* Override variables for dark mode */
@media (prefers-color-scheme: dark) {
:root {
--background-color: #1e1e1e;
--text-color: #f1f1f1;
--progress-bar-bg: #3a3a3a;
--progress-bar-fill: #81c784;
--log-bg: #2e2e2e;
--log-border: #555555;
--delete-button-bg: #e57373;
--cancel-button-bg: #ffb74d;
}
}
 
/* Apply the variables */
.export-container {
font-family: Arial, sans-serif;
padding: 20px;
background-color: var(--background-color);
color: var(--text-color);
transition: background-color 0.3s, color 0.3s;
}
.export-header {
text-align: center;
}
.progress-bar-container {
background-color: var(--progress-bar-bg);
border-radius: 10px;
overflow: hidden;
height: 30px;
margin-bottom: 15px;
}
.progress-bar {
width: 0%;
height: 100%;
background-color: var(--progress-bar-fill);
transition: width 0.5s, background-color 0.3s;
}
.progress-text {
text-align: center;
margin-top: 5px;
}
.log-container {
height: 200px;
overflow-y: auto;
background-color: var(--log-bg);
padding: 10px;
border: 1px solid var(--log-border);
border-radius: 5px;
transition: background-color 0.3s, border-color 0.3s;
}
.delete-button, .cancel-button {
display: none;
padding: 10px 20px;
margin: 5px 3px;
color: #ffffff;
border: none;
border-radius: 5px;
cursor: pointer;
transition: background-color 0.3s;
}
.delete-button {
background-color: var(--delete-button-bg);
margin-right: 10px;
}
.cancel-button {
background-color: var(--cancel-button-bg);
}
/* Show buttons when needed */
.show {
display: inline-block;
}
</style>
 
<div class="export-container">
<h2 class="export-header">Exporting...</h2>
 
<!-- Progress Bar -->
<div class="progress-bar-container">
<div id="progressBar" class="progress-bar"></div>
</div>
<p id="progressText">0/0 notes exported</p>
 
<!-- Log -->
<div id="logContainer" class="log-container">
<p>Starting export...</p>
</div>
 
<!-- Delete Button -->
<button id="deleteButton" class="delete-button">Delete Export Log</button>
<button id="logButton" class="cancel-button">Save Log</button>
 
<!-- Embed JavaScript -->
<!-- Embed JavaScript -->
<script type="text/javascript">
/**
* Updates the progress bar and progress text.
*
* @param {number} progress - Progress percentage.
* @param {number} completed - Number of notes exported.
* @param {number} total - Total number of notes to export.
*/
function updateProgress(completed, total) {
const progressBar = document.getElementById('progressBar');
const progressText = document.getElementById('progressText');
const progress = Math.round((completed / total) * 100);
 
progressBar.style.width = progress + '%';
progressText.innerText = completed + '/' + total + ' notes exported';
}
 
/**
* Appends new logs to the log container.
*
* @param {Array} newLogs - Array of new log messages.
*/
function appendLogs(newLogs) {
const logContainer = document.getElementById('logContainer');
newLogs.forEach(log => {
const p = document.createElement('p');
p.innerText = log;
logContainer.appendChild(p);
logContainer.scrollTop = logContainer.scrollHeight; // Auto-scroll to bottom
});
}
 
/**
* Shows the delete button.
*/
function showDeleteButton() {
const deleteButton = document.getElementById('deleteButton');
deleteButton.style.display = 'inline-block';
const logButton = document.getElementById('logButton');
logButton.style.display = 'inline-block';
}
 
/**
* Polls the plugin for progress and new logs.
*/
function pollPlugin() {
window.callAmplenotePlugin().then(data => {
const completed = data.completed;
const total = data.total;
const newLogs = data.logs;
const progress = Math.round((completed/total) * 100);
 
updateProgress(completed, total);
if (newLogs && newLogs.length > 0) {
appendLogs(newLogs);
}
 
if (progress < 100) {
setTimeout(pollPlugin, 200); // Poll every 200ms
} else {
showDeleteButton();
}
}).catch(error => {
console.error("Error polling plugin:", error);
});
}
 
/**
* Handles the deletion of the temporary note.
*/
document.getElementById('deleteButton').addEventListener('click', function() {
window.callAmplenotePlugin('deleteNote');
});
document.getElementById('logButton').addEventListener('click', function() {
window.callAmplenotePlugin('saveLog');
});
 
// Initiate Polling
pollPlugin();
<\/script>
</div>`;/*! OmniExport Amplenote Plugin. MIT License. Paul Strong(lapluviosilla) */async function JA(A){const e=new Date().toISOString().slice(0,16).replace(/-/g,"/").replace("T"," "),n=await A.notes.create("Bulk Export on "+e),r=`<object data="plugin://${A.context.pluginUUID}" data-aspect-ratio="1.2" />`;await n.insertContent(r),A.navigate(await n.url())}var f={_exportStatus:{completed:0,total:0,active:!1,log:[],logBuffer:[]},constants:{DEFAULT_EXPORT_TAG:"system/export",SETTING_DISABLE_EXPORT_TAG_REMOVAL:"Keep Export Tag after Export? (set to true to disable removal of the tag)",EXPORT_OPTIONS:[{label:"Microsoft Word (*.docx)",value:"docx"},{label:"PDF (*.pdf)",value:"pdf"},{label:"Web Page (*.html)",value:"html"},{label:"ePub (*.epub)",value:"epub"},{label:"LaTeX (*.tex)",value:"latex"},{label:"Markdown (*.md)",value:"md"}],BULK_EXPORT_OPTIONS:[{label:"Microsoft Word (*.docx)",value:"docx"},{label:"PDF (*.pdf)",value:"pdf"},{label:"Web Page (*.html)",value:"html"},{label:"ePub (*.epub)",value:"epub"},{label:"ePub Single Book (*.epub)",value:"epub-single"},{label:"LaTeX (*.tex)",value:"latex"},{label:"Markdown (*.md)",value:"md"},{label:"CSV (*.csv)",value:"csv"}]},renderEmbed(A,...e){return KA},async onEmbedCall(A,...e){if(e.length>0&&e[0]==="deleteNote"){if(A.context.noteUUID){const r=await A.deleteNote({uuid:A.context.noteUUID});return A.navigate("https://www.amplenote.com/notes"),r}return!1}if(e.length>0&&e[0]==="saveLog"){A.context.noteUUID&&A.replaceNoteContent({uuid:A.context.noteUUID},f._exportStatus.log.join(`
`));return}const n={completed:f._exportStatus.completed,total:f._exportStatus.total,logs:f._exportStatus.logBuffer.slice()};return f._exportStatus.logBuffer=[],n},appOption:{"Bulk Export":{check(A){return!f._exportStatus.active},async run(A){P(A);let e,n=null,r=null,t=null,a=null,o=null,i=null,s=null;try{if(e=new URL(A.context.url),n=e.searchParams.get("group"),r=e.searchParams.get("query"),t=e.searchParams.get("tag"),a=e.searchParams.get("updatedAfterDays"),o=e.searchParams.get("updatedAfterEnd"),a){const v=o||Date.now()/1e3;s=new Date(v*1e3),i=new Date((v-86400*a)*1e3)}}catch{console.warn("Invalid URL provided:",A.context.url)}let l="Bulk Export Wizard";if(i){const v=i.toDateString().slice(4,10),w=s.toDateString().slice(4,10);l+=`
* Filtering notes between `+v+" and "+w+" because of graph filter."}navigator.userAgent.indexOf("Electron")<0&&(l+=`
 
Warning: Due to limitations of the plugin api and browser restrictions this works best on the desktop app.
Using this in the browser could consume too much memory and crash amplenote, and there will be a delay before it downloads.
If you would like to use this on browser then please upvote the plugin and petition amplenote for the plugin api support.`);const[g,c,d,u,m,B]=await A.prompt(l,{inputs:[{label:"Name Search (not a full-text content search)",type:"string",value:r||""},{label:"Tags",type:"tags",value:t||""},{label:"Group",type:"string",value:n||""},{label:"Export As (required)",type:"select",options:f.constants.BULK_EXPORT_OPTIONS},{label:"Include Attachments?",type:"checkbox",value:!0}],actions:[{label:"Export Whole Notebook (warning: this could take a while)",icon:"cloud_download",value:"full-export"},{label:"Export By Export Tag ("+(A.settings["Export Tag"]||f.constants.DEFAULT_EXPORT_TAG)+")",icon:"book",value:"export-by-tag"}]});if(!B||u===null)return;const b={...g?{query:g}:{},...d?{group:d}:{},...c?{tag:c}:{}};let p;if(B==="full-export")p=await A.notes.filter();else if(B==="export-by-tag")p=await A.notes.filter({tag:A.settings["Export Tag"]||f.constants.DEFAULT_EXPORT_TAG});else if(B===-1){if(Object.keys(b).length===0){A.alert("You didn't specify a query! If you meant to export your whole notebook then please use the 'Export Whole Notebook' button");return}p=await A.notes.filter(b),i&&(p=p.filter(v=>{const w=x(v.updated);return w>=i&&w<=s}))}if(!p||p.length===0){A.alert("No notes found with that query!");return}f._exportStatus.completed=0,f._exportStatus.total=p.length,f._exportStatus.log=[],f._exportStatus.logBuffer=[],f._exportStatus.active=!0;try{let v=function(G,_){f._exportStatus.completed=G,_&&(f._exportStatus.log.push(_),f._exportStatus.logBuffer.push(_))};await JA(A);const w=new SA(A,p,u,{progressCallback:v,includeAttachments:m});await w.initialize(),await w.export(),f._exportStatus.active=!1}catch(v){A.alert("Export failed: "+v),f._exportStatus.active=!1}if(B==="export-by-tag"&&A.settings[f.constants.SETTING_DISABLE_EXPORT_TAG_REMOVAL]!=="true")for(const v of p)v.removeTag(A.settings["Export Tag"]||f.constants.DEFAULT_EXPORT_TAG)}}},noteOption:{"Export as...":{check(A,e){return"Export as..."},async run(A,e){P(A);const n=await A.prompt("Export Options",{inputs:[{type:"select",options:f.constants.EXPORT_OPTIONS}]});if(n){const r=await A.notes.find({uuid:e}),t=new y(r.name,await r.content());switch(await t.initialize(),n){case"docx":t.saveDOCX(r.name+".docx");break;case"pdf":t.savePDF(r.name+".pdf");break;case"html":t.saveHTML(r.name+".html");break;case"epub":t.saveEPUB(r.name+".epub");break;case"latex":t.saveLaTeX(r.name+".tex");break;case"md":const a=await t.toMarkdown(),i=cA(r)+`
 
`+a,s=new Blob([i],{type:"text/plain;charset=utf-8"});A.saveFile(s,r.name+".md");break;case"csv":}}}},"Export as PDF":{run:async function(A,e){P(A);const n=await A.notes.find({uuid:e});if(n){const r=new y(n.name,await n.content());await r.initialize(),r.savePDF(n.name+".pdf")}}},"Export as DOCX":{run:async function(A,e){P(A);const n=await A.notes.find({uuid:e});if(n){const r=new y(n.name,await n.content());await r.initialize(),r.saveDOCX(n.name+".docx")}}}},replaceText:{"Export Table as CSV":{check(A,e){return qA(A.context.selectionContent)},async run(A,e){P(A);const n=new y("Table",A.context.selectionContent);return await n.initialize(),await n.saveTableCSVIfTable(),!1}}}},YA=f;return f})()