diff --git a/src/assets/Classes.js b/src/assets/Classes.js new file mode 100644 index 0000000..d0bcbc8 --- /dev/null +++ b/src/assets/Classes.js @@ -0,0 +1,110 @@ +/** + * Represents a line item with text, font information, and spacing options. + * @param {String} text The contents of the line + * @param {FontInfo} fontInfo Font details to use for the line + * @param {Boolean} moveDownAfter Provide extra spacing after this line? + * @returns LineItem + */ +class LineItem { + constructor(text, fontInfo, moveDownAfter) { + this.text = text; + this.fontInfo = fontInfo; + this.moveDownAfter = moveDownAfter || false; + + return this; + } + + /** + * Set the contents of the line + * @param {String} text The contents of the line + * @returns LineItem + */ + setText(text) { + if (typeof text !== 'string') { + throw new Error('Text must be a string'); + } + this.text = text; + return this; + } + + /** + * Set the font information for the line + * @param {FontInfo} fontInfo Font details to use for the line + * @returns LineItem + */ + setFontInfo(fontInfo) { + if (!(fontInfo instanceof FontInfo)) { + throw new Error('fontInfo must be an instance of FontInfo'); + } + this.fontInfo = fontInfo; + return this; + } + + /** + * Set whether to provide extra spacing after this line + * @param {Boolean} moveDownAfter Provide extra spacing after this line? + * @returns LineItem + */ + setMoveDownAfter(moveDownAfter) { + if (typeof moveDownAfter !== 'boolean') { + throw new Error('moveDownAfter must be a boolean'); + } + this.moveDownAfter = moveDownAfter; + return this; + } +} + +/** + * Represents font information including name, size, and style. + * @param {String} name The font name + * @param {Number} size The font size + * @param {String} style The font style + * @returns FontInfo + */ +class FontInfo { + constructor(name, size, style) { + this.name = name || 'Helvetica'; + this.style = style || ''; + this.size = size || 12; + return this; + } + + /** + * Set the font name + * @param {String} name The font name + * @returns FontInfo + */ + setName(name) { + if (typeof name !== 'string') { + throw new Error('Font name must be a string'); + } + this.name = name; + return this; + } + + /** + * Set the font size + * @param {Number} size The font size + * @returns FontInfo + */ + setSize(size) { + if (typeof size !== 'number' || size <= 0) { + throw new Error('Font size must be a positive number'); + } + this.size = size; + return this; + } + + /** + * Set the font style + * @param {String} style The font style + * @returns FontInfo + */ + setStyle(style) { + if (typeof style !== 'string') { + throw new Error('Font style must be a string'); + } + this.style = style; + return this; + } +} \ No newline at end of file diff --git a/src/assets/ITThermal.js b/src/assets/ITThermal.js index 88a7d98..a46b322 100644 --- a/src/assets/ITThermal.js +++ b/src/assets/ITThermal.js @@ -1,7 +1,7 @@ /* ITS Thermal Receipt Printer * Developed by Skylar Grant for MCCS ITS */ -const version = "1.2.4"; +const version = "1.2.5"; // ############################################################# // Variables // ############################################################# @@ -21,140 +21,6 @@ const contactInfo = { const headerImageUri = ""; const headerImageType = 'JPEG'; -// Strings for use generating the PDF -const strings = { - global: { - receiptFooter: `${contactInfo.deptName} Contact Details:\n${contactInfo.supportEmail}\nor ${contactInfo.supportPhone}.` - }, - tempPassword: { - main: 'We have set a temporary password on your account, shown below. Next time you log into Microsoft you will be prompted to change your password.', - valid: 'This password will not work until after', - unLabel: 'Email:', - pwLabel: 'Password:', - pwRequirements: 'Password requirements:\n - Does not contain words or names\n - 8+ characters\n - 1+ Uppercase\n - 1+ Lowercase\n - 1+ Special (!/_=.,?)', - }, - deviceTracker: { - date: 'Date Tracked:', - ticketNumber: 'Ticket Number:', - user: 'User:', - serviceType: 'Service Type:', - notes: 'Notes:' - } -} - -// ############################################################# -// Classes -// ############################################################# -/** - * Represents a line item with text, font information, and spacing options. - * @param {String} text The contents of the line - * @param {FontInfo} fontInfo Font details to use for the line - * @param {Boolean} moveDownAfter Provide extra spacing after this line? - * @returns LineItem - */ -class LineItem { - constructor(text, fontInfo, moveDownAfter) { - this.text = text; - this.fontInfo = fontInfo; - this.moveDownAfter = moveDownAfter || false; - - return this; - } - - /** - * Set the contents of the line - * @param {String} text The contents of the line - * @returns LineItem - */ - setText(text) { - if (typeof text !== 'string') { - throw new Error('Text must be a string'); - } - this.text = text; - return this; - } - - /** - * Set the font information for the line - * @param {FontInfo} fontInfo Font details to use for the line - * @returns LineItem - */ - setFontInfo(fontInfo) { - if (!(fontInfo instanceof FontInfo)) { - throw new Error('fontInfo must be an instance of FontInfo'); - } - this.fontInfo = fontInfo; - return this; - } - - /** - * Set whether to provide extra spacing after this line - * @param {Boolean} moveDownAfter Provide extra spacing after this line? - * @returns LineItem - */ - setMoveDownAfter(moveDownAfter) { - if (typeof moveDownAfter !== 'boolean') { - throw new Error('moveDownAfter must be a boolean'); - } - this.moveDownAfter = moveDownAfter; - return this; - } -} - -/** - * Represents font information including name, size, and style. - * @param {String} name The font name - * @param {Number} size The font size - * @param {String} style The font style - * @returns FontInfo - */ -class FontInfo { - constructor(name, size, style) { - this.name = name || 'Helvetica'; - this.style = style || ''; - this.size = size || 12; - return this; - } - - /** - * Set the font name - * @param {String} name The font name - * @returns FontInfo - */ - setName(name) { - if (typeof name !== 'string') { - throw new Error('Font name must be a string'); - } - this.name = name; - return this; - } - - /** - * Set the font size - * @param {Number} size The font size - * @returns FontInfo - */ - setSize(size) { - if (typeof size !== 'number' || size <= 0) { - throw new Error('Font size must be a positive number'); - } - this.size = size; - return this; - } - - /** - * Set the font style - * @param {String} style The font style - * @returns FontInfo - */ - setStyle(style) { - if (typeof style !== 'string') { - throw new Error('Font style must be a string'); - } - this.style = style; - return this; - } -} // ############################################################# // Fonts @@ -166,6 +32,47 @@ const standardBold = new FontInfo('Helvetica', 12, 'Bold'); /** @type {FontInfo} */ const monospace = new FontInfo('Courier', 16, 'Bold'); +// ############################################################# +// Strings to use generating the PDF +// ############################################################# +const strings = { + global: { + receiptFooter: [ + {text: `${contactInfo.deptName} Contact Details:`, fontInfo: standard}, + {text: `${contactInfo.supportEmail}`, fontInfo: standard}, + {text: `or ${contactInfo.supportPhone}.`, fontInfo: standard} + ], + domain: '@mainecc.edu' + }, + tempPassword: { + main: { + text: 'We have set a temporary password on your account, shown below. Next time you log into Microsoft you will be prompted to change your password.', + fontInfo: standard + }, + valid: { + text: 'This password will not work until after', + fontInfo: standard + }, + unLabel: {text: 'Email:', fontInfo: standard}, + pwLabel: {text: 'Password:', fontInfo: standard}, + pwRequirements: [ + {text: 'Password requirements:', fontInfo: standard}, + {text: ' - Does not contain words or names', fontInfo: standardBold}, + {text: ' - 8+ characters', fontInfo: standard}, + {text: ' - 1+ Uppercase', fontInfo: standard}, + {text: ' - 1+ Lowercase', fontInfo: standard}, + {text: ' - 1+ Special (!/_=.,?)', fontInfo: standard} + ], + }, + deviceTracker: { + date: {text: 'Date Tracked:', fontInfo: standard}, + ticketNumber: {text: 'Ticket Number:', fontInfo: standard}, + user: {text: 'User:', fontInfo: standard}, + serviceType: {text: 'Service Type:', fontInfo: standard}, + notes: {text: 'Notes:', fontInfo: standard} + } +} + // ############################################################# // Functions // ############################################################# @@ -236,6 +143,7 @@ function deviceTrackerHandler() { * @returns Array */ function prepTempPassword(password, username) { + const s = strings.tempPassword; const lineArray = []; const validTime = new Date(Date.now() + 10 * 60000); // 10 minutes from now const formattedTime = validTime.toLocaleTimeString('en-US', { @@ -244,18 +152,20 @@ function prepTempPassword(password, username) { hour12: true // Explicitly ensures 12-hour format, though it's the default for 'en-US' }); - lineArray.push(new LineItem(strings.tempPassword.main, standard, true)); - lineArray.push(new LineItem(`${strings.tempPassword.valid} ${formattedTime}.`, standard, true)); + lineArray.push(new LineItem(s.main.text, s.main.fontInfo, true)); + lineArray.push(new LineItem(`${s.valid.text} ${formattedTime}.`, s.valid.fontInfo, true)); if (username) { - if (!(username.toLowerCase().endsWith('@mainecc.edu'))) username += '@mainecc.edu'; - lineArray.push(new LineItem(strings.tempPassword.unLabel, standard, false)); + if (!(username.toLowerCase().endsWith(strings.global.domain))) username += strings.global.domain; + lineArray.push(new LineItem(s.unLabel.text, s.unLabel.fontInfo, false)); lineArray.push(new LineItem(username, monospace, true)); } - lineArray.push(new LineItem(strings.tempPassword.pwLabel, standard, false)); + lineArray.push(new LineItem(s.pwLabel.text, s.pwLabel.fontInfo, false)); lineArray.push(new LineItem(password, monospace, true)); - lineArray.push(new LineItem(strings.tempPassword.pwRequirements, standardBold, true)); + for (const reqLine of s.pwRequirements) { + lineArray.push(new LineItem(reqLine.text, reqLine.fontInfo, false)); + } return lineArray; }; @@ -271,7 +181,6 @@ function prepCustomDetails(content) { }; /** - * * @param {String} dtTicketNumber * @param {String} dtUser * @param {String} dtService @@ -279,18 +188,19 @@ function prepCustomDetails(content) { * @returns Array */ function prepDeviceTracker(dtTicketNumber, dtUser, dtService, dtNotes) { + const s = strings.deviceTracker; const lineArray = []; const now = new Date(); const formattedDate = `${String(now.getMonth() + 1).padStart(2, '0')}-${String(now.getDate()).padStart(2, '0')}-${now.getFullYear()}`; - lineArray.push(new LineItem(strings.deviceTracker.date, standard, false)); + lineArray.push(new LineItem(s.date.text, s.date.fontInfo, false)); lineArray.push(new LineItem(formattedDate, monospace, false)); - lineArray.push(new LineItem(strings.deviceTracker.ticketNumber, standard, false)); + lineArray.push(new LineItem(s.ticketNumber.text, s.ticketNumber.fontInfo, false)); lineArray.push(new LineItem(dtTicketNumber, monospace, false)); - lineArray.push(new LineItem(strings.deviceTracker.user, standard, false)); + lineArray.push(new LineItem(s.user.text, s.user.fontInfo, false)); lineArray.push(new LineItem(dtUser, monospace, false)); - lineArray.push(new LineItem(strings.deviceTracker.serviceType, standard, false)); + lineArray.push(new LineItem(s.serviceType.text, s.serviceType.fontInfo, false)); lineArray.push(new LineItem(dtService, monospace, false)); - lineArray.push(new LineItem(strings.deviceTracker.notes, standard, false)); + lineArray.push(new LineItem(s.notes.text, s.notes.fontInfo, false)); lineArray.push(new LineItem(dtNotes, monospace, false)); return lineArray; }; @@ -320,7 +230,7 @@ function generateFile(lineArray, header, footer) { currentPos += logoHeightMm + ( verticalSpacing * 2); - for (row of lineArray) { + for (const row of lineArray) { doc.setFont(row.fontInfo.name, row.fontInfo.style); doc.setFontSize(row.fontInfo.size); @@ -339,9 +249,16 @@ function generateFile(lineArray, header, footer) { // Print the footer if needed if (footer) { - doc.setFontSize(12); - doc.setFont(standard.name, ''); - doc.text(strings.global.receiptFooter, 0, currentPos); + // doc.setFontSize(12); + // doc.setFont(standard.name, ''); + // doc.text(strings.global.receiptFooter, 0, currentPos); + for (const footerLine of strings.global.receiptFooter) { + doc.setFont(footerLine.fontInfo.name, footerLine.fontInfo.style); + doc.setFontSize(footerLine.fontInfo.size); + const footerStringArray = doc.splitTextToSize(footerLine.text, 68); + doc.text(footerStringArray, 0, currentPos); + currentPos += (verticalSpacing * ( footerStringArray.length - 1 )) + verticalSpacing; + } } // Output the PDF to a blob URL and return it diff --git a/src/index.html b/src/index.html index 5030fce..44e8254 100644 --- a/src/index.html +++ b/src/index.html @@ -157,6 +157,7 @@ +