リンクをたどって到達できるページを全部まとめるJS
code:script.js
function fetchAllPageData(projectName, title) {
title
)}`;
return fetch(url).then((res) => res.json());
}
async function fetchPage({ projectName, title }) {
title
)}/text`;
const pageRes = await fetch(url);
if (pageRes.ok) {
const text = await pageRes.text();
return "## " + title + "\n" + text;
} else {
return "";
}
}
function ensureDialogExists() {
let dialog = document.getElementById("resultDialog");
if (!dialog) {
dialog = document.createElement("dialog");
dialog.id = "resultDialog";
// チェックボックス
// Create label and checkbox for current page
const labelCurrent = document.createElement("label");
const cbCurrent = document.createElement("input");
cbCurrent.type = "checkbox";
cbCurrent.id = "cbCurrent";
cbCurrent.checked = true;
labelCurrent.appendChild(cbCurrent);
labelCurrent.appendChild(document.createTextNode("current"));
dialog.appendChild(labelCurrent);
dialog.appendChild(document.createTextNode(" "));
const label1hop = document.createElement("label");
const cb1hop = document.createElement("input");
cb1hop.type = "checkbox";
cb1hop.id = "cb1hop";
cb1hop.checked = true;
label1hop.appendChild(cb1hop);
label1hop.appendChild(document.createTextNode("1hop"));
dialog.appendChild(label1hop);
dialog.appendChild(document.createTextNode(" "));
const label2hop = document.createElement("label");
const cb2hop = document.createElement("input");
cb2hop.type = "checkbox";
cb2hop.id = "cb2hop";
cb2hop.checked = true;
label2hop.appendChild(cb2hop);
label2hop.appendChild(document.createTextNode("2hop"));
dialog.appendChild(label2hop);
dialog.appendChild(document.createTextNode(" "));
const labelProj = document.createElement("label");
const cbProj = document.createElement("input");
cbProj.type = "checkbox";
cbProj.id = "cbProj";
cbProj.checked = true;
labelProj.appendChild(cbProj);
labelProj.appendChild(document.createTextNode("proj"));
dialog.appendChild(labelProj);
dialog.appendChild(document.createTextNode(" "));
dialog.appendChild(document.createElement("br"));
const textarea = document.createElement("textarea");
textarea.id = "resultTextarea";
textarea.style.width = "600px";
textarea.style.height = "400px";
dialog.appendChild(textarea);
dialog.appendChild(document.createElement("br"));
const closeBtn = document.createElement("button");
closeBtn.id = "closeBtn";
closeBtn.textContent = "閉じる";
closeBtn.addEventListener("click", () => dialog.close());
dialog.appendChild(closeBtn);
document.body.appendChild(dialog);
}
return dialog;
}
// グローバル変数としてキャッシュとリンクセット
let links1hop = [];
let links2hop = [];
let projLinks = [];
let cache = null; // 初回フェッチ後、 {links1hop: {...}, links2hop: {...}, projLinks: {...}} を格納
let initDone = false; // 初期化済みフラグ
async function initAndShowDialog() {
if (!initDone) {
const projectName = scrapbox.Project.name;
const data = await fetchAllPageData(projectName, scrapbox.Page.title);
const relatedPages = data.relatedPages;
links1hop = relatedPages.links1hop.map(({ title }) => ({
projectName,
title,
}));
links2hop = relatedPages.links2hop.map(({ title }) => ({
projectName,
title,
}));
projLinks = data.relatedPages.projectLinks1hop.map(
({ projectName, title }) => ({
projectName,
title,
})
);
// 全てのリンク先ページをfetchしてキャッシュ
cache = {
links1hop: {},
links2hop: {},
projLinks: {},
};
await Promise.all([
...links1hop.map(async (link) => {
cache.links1hoplink.title = await fetchPage(link); }),
...links2hop.map(async (link) => {
cache.links2hoplink.title = await fetchPage(link); }),
...projLinks.map(async (link) => {
cache.projLinkslink.title = await fetchPage(link); }),
]);
// Fetch the current page content
const currentPageContent = await fetchPage({ projectName, title: scrapbox.Page.title });
// Store in cache with a separate key
cache.currentPage = currentPageContent;
ensureDialogExists();
// イベントリスナー登録
document
.getElementById("cb1hop")
.addEventListener("change", updateTextareaContent);
document
.getElementById("cb2hop")
.addEventListener("change", updateTextareaContent);
document
.getElementById("cbProj")
.addEventListener("change", updateTextareaContent);
document
.getElementById("cbCurrent")
.addEventListener("change", updateTextareaContent);
initDone = true;
}
// 初回または再表示時に更新
updateTextareaContent();
ensureDialogExists().showModal();
}
function updateTextareaContent() {
const cb1hop = document.getElementById("cb1hop").checked;
const cb2hop = document.getElementById("cb2hop").checked;
const cbProj = document.getElementById("cbProj").checked;
const cbCurrent = document.getElementById("cbCurrent").checked;
const resultPages = [];
if (cb1hop) {
resultPages.push(...Object.values(cache.links1hop));
}
if (cb2hop) {
resultPages.push(...Object.values(cache.links2hop));
}
if (cbProj) {
resultPages.push(...Object.values(cache.projLinks));
}
// Add current page content if checked
if (cbCurrent && cache.currentPage) {
resultPages.push(cache.currentPage);
}
const content = resultPages.join("\n\n");
document.getElementById("resultTextarea").value = content;
// Copy to clipboard after updating
navigator.clipboard.writeText(content)
.catch((err) => console.error("Clipboard error: ", err));
}
// ScrapboxのカスタムメニューにconcatPagesという項目を追加
scrapbox.PageMenu.addItem({
title: "concatPages",
onClick: () => initAndShowDialog(),
});