diff --git a/js/SaveInterface.js b/js/SaveInterface.js index 1f47ade66f..347ba48ce1 100644 --- a/js/SaveInterface.js +++ b/js/SaveInterface.js @@ -757,7 +757,7 @@ class SaveInterface { this.activity.logo.runLogoCommands(); } - /** + /** * Perform actions after saving an MXML file. * * This method handles post-processing steps after saving an MXML file. @@ -775,3 +775,8 @@ class SaveInterface { this.activity.logo.runningMxml = false; } } + + +if (typeof module !== 'undefined' && module.exports) { + module.exports = { SaveInterface }; +} diff --git a/js/__tests__/SaveInterface.test.js b/js/__tests__/SaveInterface.test.js new file mode 100644 index 0000000000..4a577da3d9 --- /dev/null +++ b/js/__tests__/SaveInterface.test.js @@ -0,0 +1,799 @@ +const { Midi } = require('@tonejs/midi'); +// Mocking the @tonejs/midi library +jest.mock('@tonejs/midi', () => { + return { + Midi: jest.fn().mockImplementation(() => ({ + header: { ticksPerBeat: 480 }, + addTrack: jest.fn(() => ({ + addNote: jest.fn(), + name: '', + instrument: { number: 0 }, + channel: 0 + })), + toArray: jest.fn(() => new Uint8Array([1, 2, 3])) + })) + }; +}) +global.Midi = Midi; +global.jQuery = jest.fn(() => ({ + on: jest.fn(), + trigger: jest.fn(), +})); +global.jQuery.noConflict = jest.fn(() => global.jQuery); +global._ = jest.fn((str) => str); +global._THIS_IS_TURTLE_BLOCKS_ = true; +global.TITLESTRING = "Music Blocks"; +global.GUIDEURL = "../guide/guide.html"; +global.fileExt = jest.fn((file) => { + if (!file) { // This covers both null and undefined + return ""; + } + const parts = file.split("."); + if (parts.length === 1 || (parts[0] === "" && parts.length === 2)) { + return ""; + } + return parts.pop(); +}); +global.window = { + isElectron: false, + prompt: jest.fn(), +}; +global.document = { + createElement: jest.fn(() => ({ + setAttribute: jest.fn(), + click: jest.fn(), + })), + body: { + appendChild: jest.fn(), + removeChild: jest.fn(), + }, +}; +global.docById = jest.fn((id) => document.getElementById(id));; +global.docByClass = jest.fn((classname) => document.getElementsByClassName(classname)); +global.mockRunLogoCommands = jest.fn(); +global.mockDownload = jest.fn(); + +const { SaveInterface } = require('../SaveInterface'); +const { LILYPONDHEADER } = require('../lilypond'); +global.LILYPONDHEADER = LILYPONDHEADER; +global.instance = new SaveInterface(); + +describe("SaveInterface", () => { + let mockActivity; + let saveInterface; + + beforeEach(() => { + mockActivity = { + beginnerMode: false, + PlanetInterface: { + getCurrentProjectName: jest.fn(() => "My Project"), + }, + }; + saveInterface = new SaveInterface(mockActivity); + }); + + it("should initialize with the correct default values", () => { + expect(saveInterface.activity).toBe(mockActivity); + expect(saveInterface.filename).toBeNull(); + expect(saveInterface.notationConvert).toBe(""); + expect(saveInterface.timeLastSaved).toBe(-100); + }); + + it("should store activity instance correctly", () => { + const newActivity = { beginnerMode: true }; + const instance = new SaveInterface(newActivity); + expect(instance.activity).toBe(newActivity); + }); +}); + +describe("download", () => { + let instance; + + beforeEach(() => { + const mockActivity = { + beginnerMode: false, + PlanetInterface: { + getCurrentProjectName: jest.fn(() => "My Project"), + } + }; + instance = new SaveInterface(mockActivity); + document.body.innerHTML = ""; + mockDownloadURL = jest.spyOn(instance, 'downloadURL'); // Spy on downloadURL + Object.defineProperty(window, 'prompt', { + writable: true, + value: jest.fn(() => "My Project.abc"), + }); + }); + + it("should set correct filename and extension when defaultfilename is provided", () => { + instance.download("abc", "data", "custom"); + expect(mockDownloadURL).toHaveBeenCalledWith("custom.abc", "data"); + }); + + it("should default filename to 'My Project.abc' if defaultfilename is null", () => { + instance.download("abc", "data", null); + expect(mockDownloadURL).toHaveBeenCalledWith("My Project.abc", "data"); + }); + + it("should append the extension if not present in the filename", () => { + instance.download("wav", "data", "My Project"); + expect(mockDownloadURL).toHaveBeenCalledWith("My Project.wav", "data"); + }); + + it("should not append the extension if already present in the filename", () => { + instance.download("xml", "data", "My Project"); + expect(mockDownloadURL).toHaveBeenCalledWith("My Project.xml", "data"); + }); + + instance = new SaveInterface(); + + it('should create an anchor tag and trigger a download', () => { + const filename = 'test.txt'; + const dataurl = 'data:text/plain;base64,SGVsbG8gd29ybGQ='; + document.body.appendChild = jest.fn(); + document.body.removeChild = jest.fn(); + const clickMock = jest.fn(); + + jest.spyOn(document, 'createElement').mockImplementation(() => { + return { + setAttribute: jest.fn(), + click: clickMock, + }; + }); + + instance.downloadURL(filename, dataurl); + + expect(document.createElement).toHaveBeenCalledWith('a'); + expect(clickMock).toHaveBeenCalled(); + expect(document.body.appendChild).toHaveBeenCalled(); + expect(document.body.removeChild).toHaveBeenCalled(); + }); + afterEach(() => { + jest.restoreAllMocks(); + }); +}); + +describe("save HTML methods", () => { + let activity; + + beforeEach(() => { + activity = { + htmlSaveTemplate: "

{{ project_name }}

{{ project_description }}

{{ data }}
", + prepareExport: jest.fn(() => "Mock Exported Data"), + PlanetInterface: { + getCurrentProjectDescription: jest.fn(() => "Mock Description"), + getCurrentProjectName: jest.fn(() => "Mock Project"), + getCurrentProjectImage: jest.fn(() => "mock-image.png"), + }, + }; + }); + + it("should replace placeholders with actual project data", () => { + let file = activity.htmlSaveTemplate; + file = file + .replace(/{{ project_description }}/g, activity.PlanetInterface.getCurrentProjectDescription()) + .replace(/{{ project_name }}/g, activity.PlanetInterface.getCurrentProjectName()) + .replace(/{{ data }}/g, activity.prepareExport()) + .replace(/{{ project_image }}/g, activity.PlanetInterface.getCurrentProjectImage()); + + expect(file).toContain("

Mock Project

"); + expect(file).toContain("

Mock Description

"); + expect(file).toContain(""); + expect(file).toContain("
Mock Exported Data
"); + }); + + it('should call prepareHTML and download the file', () => { + const mockPrepareHTML = jest.fn(() => 'Mock HTML'); + + const activity = { + save: { + prepareHTML: mockPrepareHTML, + download: mockDownload, + } + }; + + instance.saveHTML(activity); + + expect(mockPrepareHTML).toHaveBeenCalled(); + expect(mockDownload).toHaveBeenCalledWith("html", "data:text/plain;charset=utf-8,%3Chtml%3EMock%20HTML%3C%2Fhtml%3E", null); + }); + + jest.useFakeTimers(); + + it('should call prepareHTML and download the file with the correct filename', () => { + const mockPrepareHTML = jest.fn(() => 'Mock HTML'); + const mockGetProjectName = jest.fn(() => 'MockProject'); + const activity = { + save: { + prepareHTML: mockPrepareHTML, + downloadURL: mockDownloadURL, + }, + PlanetInterface: { + getCurrentProjectName: mockGetProjectName, + } + }; + + instance.saveHTMLNoPrompt(activity); + jest.runAllTimers(); + + expect(mockPrepareHTML).toHaveBeenCalled(); + expect(mockGetProjectName).toHaveBeenCalled(); + expect(mockDownloadURL).toHaveBeenCalledWith("MockProject.html", "data:text/plain;charset=utf-8,%3Chtml%3EMock%20HTML%3C%2Fhtml%3E"); + }); +}); + +describe('saveMIDI Method', () => { + let activity, mockLogo; + + beforeEach(() => { + mockLogo = { + runningMIDI: false, + runLogoCommands: jest.fn() + }; + activity = { + logo: mockLogo + }; + + }); + + it('should set runningMIDI to true and run logo commands', () => { + instance.saveMIDI(activity); + expect(activity.logo.runningMIDI).toBe(true); + expect(activity.logo.runLogoCommands).toHaveBeenCalled(); + expect(document.body.style.cursor).toBe("wait"); + }); +}); + +describe('afterSaveMIDI', () => { + + beforeEach(() => { + global.activity = { + logo: { + _midiData: { + "0": [ + { + note: ["G4"], + duration: 4, + bpm: 90, + instrument: "guitar" + }, + { + note: ["E4"], + duration: 4, + bpm: 90, + instrument: "guitar" + } + ] + } + }, + save: { + download: jest.fn() + } + }; + + global.URL.createObjectURL = jest.fn(() => 'mockURL'); + jest.useFakeTimers(); + + global.getMidiInstrument = jest.fn(() => ({ + default: 0, + guitar: 25 + })); + + global.getMidiDrum = jest.fn(() => ({ + "snare drum": 38, + "kick drum": 36 + })); + + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + it('should generate MIDI and trigger download', () => { + instance.afterSaveMIDI(); + jest.runAllTimers(); + + expect(Midi).toHaveBeenCalled(); + expect(activity.save.download).toHaveBeenCalledWith('midi', 'mockURL', null); + expect(activity.logo._midiData).toEqual({}); + expect(document.body.style.cursor).toBe('default'); + }); + + it('should create instrument tracks and add notes correctly', () => { + instance.afterSaveMIDI(); + jest.runAllTimers(); + + const tracks = Midi.mock.results[0].value.addTrack.mock.results.map(res => res.value); + const instrumentTrack = tracks.find(track => track.name === 'Track 1 - guitar'); + expect(instrumentTrack.instrument.number).toBe(25); + expect(instrumentTrack.addNote).toHaveBeenNthCalledWith(1, { + name: 'G4', + time: 0, + duration: 0.6666666666666666, + velocity: 0.8 + }); + expect(instrumentTrack.addNote).toHaveBeenNthCalledWith(2, { + name: 'E4', + time: 0.6666666666666666, + duration: 0.6666666666666666, + velocity: 0.8 + }); + }); +}); + +describe('save artwork methods', () => { + beforeEach(() => { + document.body.innerHTML = ` + + `; + + docById.mockImplementation((id) => document.getElementById(id)); + }); + + it('should call doSVG and download the SVG file', () => { + const mockDoSVG = jest.fn(() => 'Mock SVG'); + global.doSVG = mockDoSVG; + const activity = { + save: { + download: mockDownload, + }, + canvas: { width: 500, height: 500 }, + logo: 'mockLogo', + turtles: 'mockTurtles' + }; + + instance.saveSVG(activity); + expect(mockDoSVG).toHaveBeenCalledWith( + activity.canvas, + activity.logo, + activity.turtles, + activity.canvas.width, + activity.canvas.height, + 1.0 + ); + expect(mockDownload).toHaveBeenCalledWith("svg", "data:image/svg+xml;utf8,Mock SVG", null); + }); + + it('should call toDataURL and download the PNG file', () => { + const mockCanvas = { toDataURL: jest.fn(() => '') }; + global.docById = jest.fn(() => mockCanvas); + const activity = { + save: { + download: mockDownload, + } + }; + instance.savePNG(activity); + + expect(mockCanvas.toDataURL).toHaveBeenCalledWith("image/png"); + expect(mockDownload).toHaveBeenCalledWith("png", "", null); + }); + + it('should call printBlockSVG and download the SVG file', () => { + const mockPrintBlockSVG = jest.fn(() => 'Mock SVG'); + const activity = { + save: { + download: mockDownload, + }, + printBlockSVG: mockPrintBlockSVG + }; + instance.saveBlockArtwork(activity); + + expect(mockPrintBlockSVG).toHaveBeenCalled(); + expect(mockDownload).toHaveBeenCalledWith("svg", "data:image/svg+xml;utf8,Mock SVG", null); + }); + + it('should call printBlockPNG and download the PNG file', async () => { + const mockPrintBlockPNG = jest.fn(() => Promise.resolve('')); + const activity = { + save: { + download: mockDownload, + }, + printBlockPNG: mockPrintBlockPNG + }; + await instance.saveBlockArtworkPNG(activity); + + expect(mockPrintBlockPNG).toHaveBeenCalled(); + expect(mockDownload).toHaveBeenCalledWith("png", "", null); + }); +}); + +describe('saveWAV & saveABC methods', () => { + beforeEach(() => { + global._ = jest.fn((key) => key); + global.ABCHEADER = "X:1\nT:Music Blocks composition\nC:Mr. Mouse\nL:1/16\nM:C\n"; + }); + + it('should start audio recording and update UI', () => { + const mockSetupRecorder = jest.fn(); + const mockStartRecording = jest.fn(); + const mockTextMsg = jest.fn(); + + const activity = { + logo: { + recording: false, + synth: { + setupRecorder: mockSetupRecorder, + recorder: { start: mockStartRecording }, + }, + runLogoCommands: mockRunLogoCommands, + }, + textMsg: mockTextMsg, + }; + + instance.saveWAV(activity); + + expect(document.body.style.cursor).toBe("wait"); + expect(activity.logo.recording).toBe(true); + expect(mockSetupRecorder).toHaveBeenCalled(); + expect(mockStartRecording).toHaveBeenCalled(); + expect(mockRunLogoCommands).toHaveBeenCalled(); + expect(mockTextMsg).toHaveBeenCalledWith("Your recording is in progress."); + }); + + it('should prepare and run ABC notation commands', () => { + + const activity = { + logo: { + runningAbc: false, + notationOutput: "", + notationNotes: {}, + notation: { + notationStaging: {}, + notationDrumStaging: {} + }, + runLogoCommands: mockRunLogoCommands + }, + turtles: { + turtleList: [{ painter: { doClear: jest.fn() } }], + getTurtleCount: jest.fn(() => 1), + getTurtle: function(index) { + return this.turtleList[index]; + } + } + }; + + instance.saveAbc(activity); + + expect(document.body.style.cursor).toBe("wait"); + expect(activity.logo.runningAbc).toBe(true); + expect(mockRunLogoCommands).toHaveBeenCalled(); + }); + + it('should encode and download ABC notation output', () => { + const mockSaveAbcOutput = jest.fn(() => "mock_abc_data"); + + global.saveAbcOutput = mockSaveAbcOutput; + + const activity = { + save: { + download: mockDownload + } + }; + + instance.afterSaveAbc.call({ activity }); + + expect(mockSaveAbcOutput).toHaveBeenCalledWith(activity); + expect(mockDownload).toHaveBeenCalledWith("abc", "data:text;utf8,mock_abc_data", null); + }); +}); + +describe('saveLilypond Methods', () => { + let activity, saveInterface, mockActivity, mockDocById; + + beforeEach(() => { + // Set up the DOM structure + document.body.innerHTML = ` + + `; + + // Mock document functions + global.docById = jest.fn((id) => document.getElementById(id)); + jest.spyOn(document, "getElementById").mockImplementation((id) => document.querySelector(`#${id}`)); + document.execCommand = jest.fn(); + global.platform = { FF: false }; + global.jQuery = jest.fn((selector) => { + if (selector === window) { + return { on: jest.fn() }; + } + return { + appendTo: jest.fn().mockReturnThis(), + val: jest.fn().mockReturnThis(), + select: jest.fn(), + remove: jest.fn(), + on: jest.fn(), + }; + }); + global.jQuery.noConflict = jest.fn().mockImplementation(() => global.jQuery); + + // Mock activity objects + activity = { + PlanetInterface: { + getCurrentProjectName: jest.fn(() => "Test Project"), + }, + storage: { + getItem: jest.fn(() => JSON.stringify("Custom Author")), + setItem: jest.fn(), + }, + save: { + saveLYFile: jest.fn(), + download: jest.fn(), + }, + logo: { + runningLilypond: false, + MIDIOutput: '', + guitarOutputHead: '', + guitarOutputEnd: '', + notationOutput: '', + notationNotes: {}, + notation: { + notationStaging: [], + notationDrumStaging: [], + }, + runLogoCommands: jest.fn(), + }, + textMsg: jest.fn(), + download: jest.fn(), + }; + + // Mock secondary activity object (for different test scenarios) + mockActivity = { + ...activity, + turtles: { + turtleList: [{ painter: { doClear: jest.fn() } }], + getTurtleCount: jest.fn(() => 1), + getTurtle: function(index) { + return this.turtleList[index]; + } + } + }; + + instance = new SaveInterface(); + instance.activity = activity; + instance.notationConvert = "ly"; + instance.afterSaveLilypondLY = jest.fn(); + saveInterface = new SaveInterface(mockActivity); + + mockSaveLilypondOutput = jest.fn(() => "Lilypond Data"); + global.saveLilypondOutput = mockSaveLilypondOutput; + mockDocById = jest.fn(); + + window.Converter = { + ly2pdf: jest.fn(), + }; + + lydata = "Lilypond Data"; + filename = "TestProject.pdf"; + dataurl = "data:application/pdf;base64,abc123"; + + document.body.style.cursor = ""; + }); + + it('should open the Lilypond modal and populate fields', () => { + + instance.saveLilypond(activity); + + expect(docById("lilypondModal").style.display).toBe("block"); + expect(docById("fileName").value).toBe("Test Project.ly"); + expect(docById("title").value).toBe("Test Project"); + expect(docById("author").value).toBe("Custom Author"); + }); + + it('should close the modal when close button is clicked', () => { + + instance.saveLilypond(activity); + + const closeButton = docByClass("close")[0]; + closeButton.click(); + + expect(activity.logo.runningLilypond).toBe(false); + expect(docById("lilypondModal").style.display).toBe("none"); + }); + + it('should call saveLYFile when save button is clicked', () => { + + instance.saveLilypond(activity); + + const saveButton = docById("submitLilypond"); + saveButton.click(); + + expect(activity.save.saveLYFile).toHaveBeenCalledWith(false); + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + it('should save a Lilypond file with default settings', () => { + saveInterface.saveLYFile(); + expect(global.docById).toHaveBeenCalledWith('fileName'); + }); + + it('should save a Lilypond file with PDF conversion', () => { + saveInterface.saveLYFile(true); + expect(saveInterface.notationConvert).toBe('pdf'); + }); + + it('should handle MIDI and guitar output settings correctly', () => { + // Simulate MIDI and guitar checkboxes being checked + mockDocById.mockImplementation((id) => { + const mockElements = { + fileName: { value: 'testFile' }, + title: { value: 'My Project' }, + author: { value: 'Mr. Mouse' }, + MIDICheck: { checked: true }, + guitarCheck: { checked: true }, + lilypondModal: { style: { display: 'block' } }, + }; + return mockElements[id]; + }); + + saveInterface.saveLYFile(); + // Define the expected MIDI output as a string + const expectedMIDIOutput = `% MIDI SECTION +% Delete the %{ and %} below to include MIDI output. +%{ +\\midi { + \\tempo 4=90 +} +%} + +}`; + expect(mockActivity.logo.MIDIOutput).toContain(expectedMIDIOutput); + }); + + it('should call saveLilypondOutput and afterSaveLilypondLY', () => { + instance.afterSaveLilypond("ignored.ly"); + expect(mockSaveLilypondOutput).toHaveBeenCalledWith(instance.activity); + expect(instance.afterSaveLilypondLY).toHaveBeenCalledWith("Lilypond Data", "TestProject.ly"); + expect(instance.notationConvert).toBe(""); + }); + + it('should set cursor to "wait" and call ly2pdf with correct arguments', () => { + instance.afterSaveLilypondPDF(lydata, filename); + expect(document.body.style.cursor).toBe("wait"); + expect(window.Converter.ly2pdf).toHaveBeenCalledWith(lydata, expect.any(Function)); + }); + + it('should reset cursor to "default" and call activity.save.download on success', () => { + // Mock ly2pdf to call the callback with success + window.Converter.ly2pdf.mockImplementation((lydata, callback) => { + callback(true, dataurl); + }); + + instance.afterSaveLilypondPDF(lydata, filename); + expect(document.body.style.cursor).toBe("default"); + expect(activity.save.download).toHaveBeenCalledWith("pdf", dataurl, filename); + }); + + it('should reset cursor to "default" and log an error on failure', () => { + const errorMessage = "Conversion failed"; + + window.Converter.ly2pdf.mockImplementation((lydata, callback) => { + callback(false, errorMessage); + }); + + // Mock console.debug to verify the error message + console.debug = jest.fn(); + + instance.afterSaveLilypondPDF(lydata, filename); + + expect(document.body.style.cursor).toBe("default"); + expect(console.debug).toHaveBeenCalledWith("Error: " + errorMessage); + // Verify activity.save.download is not called + expect(activity.save.download).not.toHaveBeenCalled(); + }); +}); + +describe('MXML Methods', () => { + let instance, activity, mockLogo, mockTurtles; + + beforeEach(() => { + // Mock turtleList with painters + const mockPainter1 = { doClear: jest.fn() }; + const mockPainter2 = { doClear: jest.fn() }; + + mockTurtles = { + turtleList: [ + { painter: mockPainter1 }, + { painter: mockPainter2 } + ], + getTurtleCount: function() { + return this.turtleList.length; + }, + getTurtle: function(index) { + return this.turtleList[index]; + } + }; + + // Mock logo object + mockLogo = { + runningMxml: false, + notation: { + notationStaging: {}, + notationDrumStaging: {} + }, + runLogoCommands: jest.fn() + }; + + activity = { + logo: mockLogo, + turtles: mockTurtles + }; + + // Create SaveInterface instance + instance = new SaveInterface(); + instance.activity = activity; + instance.download = jest.fn(); + + // Mock saveMxmlOutput + global.saveMxmlOutput = jest.fn().mockReturnValue("Mock MXML Data"); + }); + + it('should initialize MXML state and clear turtle canvases', () => { + instance.saveMxml("test.mxml"); + + // Verify runningMxml flag + expect(activity.logo.runningMxml).toBe(true); + + // Verify notation staging initialization + expect(activity.logo.notation.notationStaging[0]).toEqual([]); + expect(activity.logo.notation.notationStaging[1]).toEqual([]); + expect(activity.logo.notation.notationDrumStaging[0]).toEqual([]); + expect(activity.logo.notation.notationDrumStaging[1]).toEqual([]); + expect(mockTurtles.turtleList[0].painter.doClear) + .toHaveBeenCalledWith(true, true, true); + expect(mockTurtles.turtleList[1].painter.doClear) + .toHaveBeenCalledWith(true, true, true); + + // Verify logo commands run + expect(activity.logo.runLogoCommands).toHaveBeenCalled(); + }); + + it('should generate XML and trigger download', () => { + const filename = "TestScore.xml"; + instance.afterSaveMxml(filename); + + expect(global.saveMxmlOutput).toHaveBeenCalledWith(activity.logo); + + const expectedData = "data:text;utf8," + encodeURIComponent("Mock MXML Data"); + expect(instance.download).toHaveBeenCalledWith( + "xml", + expectedData, + filename + ); + + expect(activity.logo.runningMxml).toBe(false); + }); + + it('should handle empty data gracefully', () => { + global.saveMxmlOutput.mockReturnValue(""); + instance.afterSaveMxml("empty.xml"); + expect(instance.download).toHaveBeenCalledWith( + "xml", + "data:text;utf8," + encodeURIComponent(""), + "empty.xml" + ); + }); + +}); diff --git a/js/lilypond.js b/js/lilypond.js index 583caaf04f..71f6b4c310 100644 --- a/js/lilypond.js +++ b/js/lilypond.js @@ -984,3 +984,7 @@ const saveLilypondOutput = function (activity) { activity.logo.notationOutput += "\n%}\n\n"; return activity.logo.notationOutput; }; + +if (typeof module !== 'undefined' && module.exports) { + module.exports = { LILYPONDHEADER }; +} \ No newline at end of file