Files
anything-llm/server/utils/agents/aibitat/plugins/create-files/pptx/create-presentation.js
Timothy Carambat 7aaea7f514 File creation agent skills (#5280)
* Powerpoint File Creation (#5278)

* wip

* download card

* UI for downloading

* move to fs system with endpoint to pull files

* refactor UI

* final-pass

* remove save-file-browser skill and refactor

* remove fileDownload event

* reset

* reset file

* reset timeout

* persist toggle

* Txt creation (#5279)

* wip

* download card

* UI for downloading

* move to fs system with endpoint to pull files

* refactor UI

* final-pass

* remove save-file-browser skill and refactor

* remove fileDownload event

* reset

* reset file

* reset timeout

* wip

* persist toggle

* add arbitrary text creation file

* Add PDF document generation with markdown formatting (#5283)

add support for branding in bottom right corner
refactor core utils and frontend rendering

* Xlsx document creation (#5284)

add Excel doc & sheet creation

* Basic docx creation (#5285)

* Basic docx creation

* add test theme support + styling and title pages

* simplify skill selection

* handle TG attachments

* send documents over tg

* lazy import

* pin deps

* fix lock

* i18n for file creation (#5286)

i18n for file-creation
connect #5280

* theme overhaul

* Add PPTX subagent for better results

* forgot files

* Add PPTX subagent for better results (#5287)

* Add PPTX subagent for better results

* forgot files

* make sub-agent use proper tool calling if it can and better UI hints
2026-03-30 15:13:39 -07:00

343 lines
13 KiB
JavaScript

const createFilesLib = require("../lib.js");
const { getTheme, getAvailableThemes } = require("./themes.js");
const {
renderTitleSlide,
renderSectionSlide,
renderContentSlide,
renderBlankSlide,
} = require("./utils.js");
const { runSectionAgent } = require("./section-agent.js");
/**
* Extracts recent conversation history from the parent AIbitat's chat log
* to provide context to each section sub-agent.
* @param {Array} chats - The parent AIbitat's _chats array
* @param {number} [maxMessages=10] - Maximum messages to include
* @returns {string} Formatted conversation context
*/
function extractConversationContext(chats, maxMessages = 10) {
if (!Array.isArray(chats) || chats.length === 0) return "";
const recent = chats
.filter((c) => c.state === "success" && c.content)
.slice(-maxMessages);
if (recent.length === 0) return "";
return recent
.map((c) => {
const content =
typeof c.content === "string" ? c.content.substring(0, 500) : "";
return `${c.from}: ${content}`;
})
.join("\n");
}
module.exports.CreatePptxPresentation = {
name: "create-pptx-presentation",
plugin: function () {
return {
name: "create-pptx-presentation",
setup(aibitat) {
aibitat.function({
super: aibitat,
name: this.name,
description:
"Create a professional PowerPoint presentation (PPTX). " +
"Provide a title, theme, and section outlines with key points. " +
"Each section is independently researched and built by a focused sub-agent " +
"that can use web search and web scraping to gather data.",
examples: [
{
prompt: "Create a presentation about project updates",
call: JSON.stringify({
filename: "project-updates.pptx",
title: "Q1 Project Updates",
theme: "corporate",
sections: [
{
title: "Overview",
keyPoints: [
"Project on track for Q1 delivery",
"Team expanded by 2 new members",
"Budget within expectations",
],
},
{
title: "Key Achievements",
keyPoints: [
"Launched new feature X",
"Reduced bug count by 40%",
"Improved performance by 25%",
],
instructions:
"Include specific metrics and quarter-over-quarter comparisons",
},
],
}),
},
{
prompt: "Create a dark themed presentation about AI trends",
call: JSON.stringify({
filename: "ai-trends.pptx",
title: "AI Trends 2025",
theme: "dark",
sections: [
{
title: "Large Language Models",
keyPoints: [
"Model scaling trends",
"Open vs closed source landscape",
],
instructions:
"Research the latest developments and include recent data",
},
{
title: "AI in Enterprise",
keyPoints: ["Adoption rates", "Top use cases", "ROI data"],
},
],
}),
},
],
parameters: {
$schema: "http://json-schema.org/draft-07/schema#",
type: "object",
properties: {
filename: {
type: "string",
description:
"The filename for the presentation (should end with .pptx).",
},
title: {
type: "string",
description:
"The title of the presentation (shown on title slide).",
},
author: {
type: "string",
description:
"Optional author name for the presentation metadata.",
},
theme: {
type: "string",
enum: getAvailableThemes(),
description:
"Color theme for the presentation. Options: " +
getAvailableThemes().join(", "),
},
sections: {
type: "array",
description:
"Section outlines for the presentation. Each section is independently researched and built by a focused sub-agent.",
items: {
type: "object",
properties: {
title: {
type: "string",
description: "The section title.",
},
keyPoints: {
type: "array",
items: { type: "string" },
description:
"Key points this section should cover. The sub-agent will expand these into detailed slides.",
},
instructions: {
type: "string",
description:
"Optional guidance for the section builder (e.g. 'research recent statistics', 'compare with competitors', 'include a data table').",
},
},
required: ["title"],
},
},
},
required: ["filename", "title", "sections"],
additionalProperties: false,
},
handler: async function ({
filename = "presentation.pptx",
title = "Untitled Presentation",
author = "",
theme: themeName = "default",
sections = [],
}) {
try {
this.super.handlerProps.log(
`Using the create-pptx-presentation tool.`
);
if (!filename.toLowerCase().endsWith(".pptx"))
filename += ".pptx";
const theme = getTheme(themeName);
const totalSections = sections.length;
this.super.introspect(
`${this.caller}: Planning presentation "${title}" — ${totalSections} section${totalSections !== 1 ? "s" : ""}, ${theme.name} theme`
);
// Ask for approval BEFORE kicking off the expensive sub-agent work
if (this.super.requestToolApproval) {
const approval = await this.super.requestToolApproval({
skillName: this.name,
payload: {
filename,
title,
sectionCount: totalSections,
sectionTitles: sections.map((s) => s.title),
},
description: `Create PowerPoint presentation "${title}" with ${totalSections} sections`,
});
if (!approval.approved) {
this.super.introspect(
`${this.caller}: User rejected the ${this.name} request.`
);
return approval.message;
}
}
const conversationContext = extractConversationContext(
this.super._chats
);
// Run a focused sub-agent for each section sequentially.
// Sequential execution is intentional — local models typically serve
// one request at a time, and it keeps introspection events ordered.
const allSlides = [];
const allCitations = [];
for (let i = 0; i < sections.length; i++) {
const section = sections[i];
this.super.introspect(
`${this.caller}: [${i + 1}/${totalSections}] Building section "${section.title}"…`
);
const sectionResult = await runSectionAgent({
parentAibitat: this.super,
section,
presentationTitle: title,
conversationContext,
sectionPrefix: `${i + 1}/${totalSections}`,
});
const slideCount = sectionResult.slides?.length || 0;
allSlides.push(...(sectionResult.slides || []));
if (sectionResult.citations?.length > 0)
allCitations.push(...sectionResult.citations);
this.super.introspect(
`${this.caller}: [${i + 1}/${totalSections}] Section "${section.title}" complete — ${slideCount} slide${slideCount !== 1 ? "s" : ""}`
);
}
// Roll up all citations from sub-agents to the parent so they
// appear as sources on the final assistant message.
if (allCitations.length > 0) this.super.addCitation(allCitations);
// Assemble the final PPTX from all section outputs
this.super.introspect(
`${this.caller}: Assembling final deck — ${allSlides.length} slides total`
);
const PptxGenJS = require("pptxgenjs");
const pptx = new PptxGenJS();
pptx.title = title;
if (author) pptx.author = author;
pptx.company = "AnythingLLM";
const totalSlideCount = allSlides.length;
// Title slide
const titleSlide = pptx.addSlide();
renderTitleSlide(titleSlide, pptx, { title, author }, theme);
// Render every slide produced by the section agents
allSlides.forEach((slideData, index) => {
const slide = pptx.addSlide();
const slideNumber = index + 1;
const layout = slideData.layout || "content";
switch (layout) {
case "title":
case "section":
renderSectionSlide(
slide,
pptx,
slideData,
theme,
slideNumber,
totalSlideCount
);
break;
case "blank":
renderBlankSlide(
slide,
pptx,
theme,
slideNumber,
totalSlideCount
);
break;
default:
renderContentSlide(
slide,
pptx,
slideData,
theme,
slideNumber,
totalSlideCount
);
break;
}
});
const buffer = await pptx.write({ outputType: "nodebuffer" });
const bufferSizeKB = (buffer.length / 1024).toFixed(2);
const bufferSizeMB = (buffer.length / (1024 * 1024)).toFixed(2);
this.super.handlerProps.log(
`create-pptx-presentation: Generated buffer - size: ${bufferSizeKB}KB (${bufferSizeMB}MB), slides: ${totalSlideCount}, theme: ${theme.name}`
);
const displayFilename = filename.split("/").pop();
const savedFile = await createFilesLib.saveGeneratedFile({
fileType: "pptx",
extension: "pptx",
buffer,
displayFilename,
});
this.super.socket.send("fileDownloadCard", {
filename: savedFile.displayFilename,
storageFilename: savedFile.filename,
fileSize: savedFile.fileSize,
});
createFilesLib.registerOutput(this.super, "PptxFileDownload", {
filename: savedFile.displayFilename,
storageFilename: savedFile.filename,
fileSize: savedFile.fileSize,
});
this.super.introspect(
`${this.caller}: Successfully created presentation "${title}"`
);
return `Successfully created presentation "${title}" with ${totalSlideCount} slides across ${totalSections} sections using the ${theme.name} theme.`;
} catch (e) {
this.super.handlerProps.log(
`create-pptx-presentation error: ${e.message}`
);
this.super.introspect(`Error: ${e.message}`);
return `Error creating presentation: ${e.message}`;
}
},
});
},
};
},
};