Skip to content

Instantly share code, notes, and snippets.

@jamespwright
Created February 9, 2024 01:55
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save jamespwright/6dab2525d36c9d5a31f64dee35058bf2 to your computer and use it in GitHub Desktop.
Save jamespwright/6dab2525d36c9d5a31f64dee35058bf2 to your computer and use it in GitHub Desktop.
name: Bunpilot Word
description: ''
host: WORD
api_set: {}
script:
content: |
// Constants for API configuration
const OPENAI_API_URL = "https://api.openai.com/v1/chat/completions";
//TODO: Keyvault access for better security
const OPENAI_API_TOKEN = "YOURAPIKEY";
const OPENAI_API_MODEL = "gpt-4";
const promptsize = 15000;
// Set up event listeners for buttons
$("#btnimprove").click(improve);
$("#btnsummarise").click(summarise);
$("#btncoach").click(coach);
$(".close").click(function() {
$("#myModal").hide();
});
function improve() {
$(".loader").show();
// Get selected text from the email
Office.context.document.getSelectedDataAsync(Office.CoercionType.Text, function(result) {
const text = result.value;
// Check if any text is selected
if (text && text.trim() !== "") {
//console.log(text);
callOpenAI(text, "improve");
} else {
// Inform the user that no text is selected.
$(".loader").hide();
openmodal();
document.getElementById("modalText").innerHTML = "No text is selected for improvement.";
}
});
}
async function summarise() {
$(".loader").show();
await Word.run(async (context) => {
var body = context.document.body;
body.load("text");
await context.sync();
var text = body.text;
//console.log(text);
callOpenAI(text, "summarise");
});
}
async function coach() {
$(".loader").show();
await Word.run(async (context) => {
var body = context.document.body;
body.load("text");
await context.sync();
var text = body.text;
//console.log(text);
callOpenAI(text, "coach");
});
}
function callOpenAI(text, mode) {
//Reduce prompt size so we dont exceed max openai tokens
let prompttext = text.slice(0, promptsize);
//set prompt length
if ($("#short").prop("checked")) {
let promptlength = "short length sentences";
} else if ($("#medium").prop("checked")) {
let promptlength = "medium length sentences";
} else if ($("#long").prop("checked")) {
let promptlength = "long length sentenses";
}
//Set prompt tone
if ($("#direct").prop("checked")) {
let prompttone = "direct";
} else if ($("#formal").prop("checked")) {
let prompttone = "formal";
} else if ($("#casual").prop("checked")) {
let prompttone = "casual";
} else if ($("#neutral").prop("checked")) {
let prompttone = "neutral";
}
// Modify prompt text based on the selected mode
if (mode === "improve") {
prompttext = `Rewrite the following text in ${promptlength} using a ${prompttone} tone. ${prompttext}`;
} else if (mode === "summarise") {
prompttext = `Summarise the following text into the five dot points using ${promptlength}. ${prompttext}`;
} else if (mode === "coach") {
prompttext = `Five dot points with recommendations how to improve this text using ${promptlength}. ${prompttext}`;
}
// Define the conversation messages
const messages = [
{
role: "system",
content:
"You are a helpful assistant that writes documents in a corporate environment. Do not include the subject line. Use British English spelling."
},
{ role: "user", content: prompttext }
];
// Set up and send the API request
const xhr = new XMLHttpRequest();
xhr.open("POST", OPENAI_API_URL);
xhr.setRequestHeader("Content-Type", "application/json");
xhr.setRequestHeader("Authorization", "Bearer " + OPENAI_API_TOKEN);
xhr.onload = function(e) {
const jsonResponse = JSON.parse(this.response);
let outputContent = jsonResponse.choices[0].message.content;
// Replace the selected text with the improved or replied text
if (mode === "improve") {
$(".loader").hide();
//console.log(outputContent);
replaceSelection(outputContent);
}
$(".loader").hide();
// Log the summarized content to the console
if (mode === "summarise" || mode === "coach") {
//console.log(outputContent);
outputContent = outputContent.replace(/\n/g, "<br><br>");
$(".loader").hide();
openmodal();
document.getElementById("modalText").innerHTML = outputContent;
}
};
const requestData = {
model: OPENAI_API_MODEL,
messages: messages
};
//Call the openai api
xhr.send(JSON.stringify(requestData));
}
function replaceSelection(text) {
// Replace selected text
Office.context.document.setSelectedDataAsync(text, function(asyncResult) {
if (asyncResult.status === Office.AsyncResultStatus.Succeeded) {
//console.log(`Replaced selected text with "${text}".`);
} else {
handleAsyncError(asyncResult.error, "Failed to replace selected text.");
}
});
}
// Function to handle asynchronous errors
function handleAsyncError(error, errorMessage) {
// Display error message)
openmodal();
document.getElementById("modalText").innerHTML = errorMessage;
}
function openmodal() {
var modal = document.getElementById("myModal");
modal.style.display = "block";
}
language: typescript
template:
content: "<section>\n\t<img class=\"banner\" src=\"\">\n</section>\n\t<div class=\"center\">\n\t\t<div class=\"loader\"></div>\n\t\t<section class=\"samples ms-font-m\">\n\t\t\t<p class=\"ms-font-m\">\n\t\t\t\t<h1>Bunpilot uses generative AI to help you write documents.</H1>\n\t\t\t</p>\n\t\t\t<p>Highlight text then click the improve button to improve grammar and wording.</p>\n\t\t\t<button id=\"btnimprove\" class=\"ms-Button\">\n\t\t\t<span class=\"ms-Button-label\">Improve</span>\n\t\t\t</button>\n\t\t\t<p>Click coach to generate coaching recommendations about your document.</p>\n\t\t\t<button id=\"btncoach\" class=\"ms-Button\">\n\t\t\t\t\t<span class=\"ms-Button-label\">Coach</span>\n\t\t\t</button>\n\t\t\t<p>Click summarise to view a summary of long documents.</p>\n\t\t\t<button id=\"btnsummarise\" class=\"ms-Button\">\n\t\t\t\t\t\t\t<span class=\"ms-Button-label\">Summarise</span>\n\t\t\t</button>\n\t\t\t<p>Select response tone.</p>\n\t\t\t<input type=\"radio\" id=\"neutral\" name=\"prompttone\" checked/>\n\t\t\t<label>Neutral</label>\n\t\t\t<input type=\"radio\" id=\"direct\" name=\"prompttone\"/>\n\t\t\t<label>Direct</label>\n\t\t\t<input type=\"radio\" id=\"formal\" name=\"prompttone\"/>\n\t\t\t<label>Formal</label>\n\t\t\t<input type=\"radio\" id=\"casual\" name=\"prompttone\"/>\n\t\t\t<label>Casual</label>\n\t\t\t<p>Select response length.</p>\n\t\t\t<input type=\"radio\" id=\"short\" name=\"promptlength\" checked/>\n\t\t\t<label>Short</label>\n\t\t\t<input type=\"radio\" id=\"medium\" name=\"promptlength\"/>\n\t\t\t<label>Medium</label>\n\t\t\t<input type=\"radio\" id=\"long\" name=\"promptlength\"/>\n\t\t\t<label>Long</label>\n\t\t</section>\n\n\t\t<div id=\"myModal\" class=\"modal\">\n\t\t\t<div class=\"modal-content\">\n\t\t\t\t<span class=\"close\">&times;</span>\n\t\t\t\t<p id=\"modalText\"></p>\n\t\t\t</div>\n\t\t</div>\n\t</div>"
language: html
style:
content: |+
section.samples {
margin-top: 20px;
margin-left: 10px;
margin-right: 10px;
}
section.samples .ms-Button, section.setup .ms-Button {
display: block;
margin-bottom: 5px;
margin-left: auto;
margin-right: auto;
min-width: 80px;
max-width: 200px;
border-radius:10px;
width: 90%;
}
body {
background-color: #0E5358;
background-repeat: no-repeat;
margin-left: auto;
margin-right: auto;
}
.center {
display: block;
margin-left: auto;
margin-right: auto;
width: 90%;
color: white;
}
.banner {
display: block;
margin-left: auto;
margin-right: auto;
width: 250px;
}
/* The Modal (background) */
.modal {
display: none; /* Hidden by default */
position: fixed; /* Stay in place */
z-index: 1; /* Sit on top */
left: 0;
top: 0;
width: 100%; /* Full width */
height: 100%; /* Full height */
overflow: auto; /* Enable scroll if needed */
background-color: rgb(0,0,0); /* Fallback color */
background-color: rgba(0,0,0,0.4); /* Black w/ opacity */
}
.modal-content {
background-color: whitesmoke;
color: black;
margin: 15% auto;
padding: 20px;
border: 1px solid #888;
width: 80%;
text-align: left;
font-family: "Segoe UI", Tahoma, Geneva, Verdana, sans-serif;
}
/* The Close Button */
.close {
color: #aaa;
float: right;
font-size: 28px;
font-weight: bold;
}
.close:hover,
.close:focus {
color: black;
text-decoration: none;
cursor: pointer;
}
.loader {
display: none;
position: absolute;
left: 45%;
top: 183px;
transform: translate(-50%, -50%);
z-index: 1;
width: 20px;
height: 20px;
border: 16px solid #f3f3f3;
border-radius: 50%;
border-top: 16px solid #3498db;
-webkit-animation: spin 2s linear infinite;
animation: spin 2s linear infinite;
}
@keyframes spin {
0% { transform: rotate(0deg); }
100% { transform: rotate(360deg); }
}
language: css
libraries: |
https://appsforoffice.microsoft.com/lib/1/hosted/office.js
@types/office-js
office-ui-fabric-js@1.4.0/dist/css/fabric.min.css
office-ui-fabric-js@1.4.0/dist/css/fabric.components.min.css
core-js@2.4.1/client/core.min.js
@types/core-js
jquery@3.1.1
@types/jquery@3.3.1
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment