mirror of
https://github.com/GSA/notifications-admin.git
synced 2026-02-07 20:03:33 -05:00
224 lines
8.3 KiB
JavaScript
224 lines
8.3 KiB
JavaScript
const path = require('path');
|
|
const fs = require('fs');
|
|
|
|
// Load the D3 script content
|
|
const d3ScriptContent = fs.readFileSync(path.resolve(__dirname, '../javascripts/support/d3.min.js'), 'utf-8');
|
|
|
|
// Helper function to dynamically load a script
|
|
function loadScript(scriptContent) {
|
|
const script = document.createElement('script');
|
|
script.textContent = scriptContent;
|
|
document.head.appendChild(script);
|
|
}
|
|
|
|
// Mocking `clientWidth`
|
|
Object.defineProperty(HTMLElement.prototype, 'clientWidth', {
|
|
value: 600,
|
|
writable: true,
|
|
});
|
|
|
|
// beforeAll hook to set up the DOM and load D3.js script
|
|
beforeEach(done => {
|
|
// Set up the DOM with the D3 script included
|
|
document.body.innerHTML = `
|
|
<div id="activityChartContainer">
|
|
<form class="usa-form">
|
|
<label class="usa-label" for="options">Account</label>
|
|
<select class="usa-select margin-bottom-2" name="options" id="options">
|
|
<option value="">- Select -</option>
|
|
<option value="service" selected>Service Name</option>
|
|
<option value="individual">User Name</option>
|
|
</select>
|
|
</form>
|
|
<div id="activityChart">
|
|
<div class="chart-header">
|
|
<div class="chart-subtitle">Service Name - Last 7 Days</div>
|
|
<div class="chart-legend" aria-label="Legend"></div>
|
|
</div>
|
|
<div class="chart-container" id="weeklyChart" data-service-id="12345" style="width: 600px;"></div>
|
|
<table id="weeklyTable" class="usa-sr-only usa-table"></table>
|
|
</div>
|
|
</div>
|
|
<div id="aria-live-account" class="usa-sr-only" aria-live="polite"></div>
|
|
<div id="table1" class="table-overflow-x-auto hidden"></div>
|
|
<div id="table2" class="table-overflow-x-auto visible"></div>
|
|
`;
|
|
|
|
// Re-bind event listeners in case they are lost between tests
|
|
const dropdown = document.getElementById('options');
|
|
dropdown.addEventListener('change', window.handleDropdownChange);
|
|
|
|
// Load the D3 script dynamically
|
|
loadScript(d3ScriptContent);
|
|
|
|
// Wait a bit to ensure the script is executed
|
|
setTimeout(() => {
|
|
// Require the actual JavaScript file you are testing
|
|
require('../../app/assets/javascripts/activityChart.js');
|
|
done();
|
|
}, 100);
|
|
}, 10000); // Increased timeout to 10 seconds
|
|
|
|
afterEach(() => {
|
|
// Clean up DOM or restore mocks if needed
|
|
jest.restoreAllMocks(); // Restores all mocks, including console logs
|
|
});
|
|
|
|
test('D3 is loaded correctly', () => {
|
|
// Check if D3 is loaded by verifying the existence of the d3 object
|
|
expect(window.d3).toBeDefined();
|
|
expect(typeof window.d3.version).toBe('string');
|
|
});
|
|
|
|
test('Populates the accessible table for activity chart correctly', () => {
|
|
const sampleData = {
|
|
'2024-07-01': { sms: { delivered: 50, failed: 5 } },
|
|
'2024-07-02': { sms: { delivered: 60, failed: 2 } },
|
|
'2024-07-03': { sms: { delivered: 70, failed: 1 } },
|
|
'2024-07-04': { sms: { delivered: 80, failed: 0 } },
|
|
'2024-07-05': { sms: { delivered: 90, failed: 3 } },
|
|
'2024-07-06': { sms: { delivered: 100, failed: 4 } },
|
|
'2024-07-07': { sms: { delivered: 110, failed: 2 } },
|
|
};
|
|
|
|
const labels = Object.keys(sampleData).map(dateString => {
|
|
const dateParts = dateString.split('-');
|
|
return `${dateParts[1]}/${dateParts[2]}/${dateParts[0].slice(2)}`; // Format to MM/DD/YY
|
|
});
|
|
const deliveredData = Object.values(sampleData).map(d => d.sms.delivered);
|
|
const failedData = Object.values(sampleData).map(d => d.sms.failed);
|
|
|
|
window.createTable('weeklyTable', 'activityChart', labels, deliveredData, failedData);
|
|
|
|
const table = document.getElementById('weeklyTable');
|
|
expect(table).toBeDefined();
|
|
|
|
const rows = table.getElementsByTagName('tr');
|
|
expect(rows.length).toBe(8); // Header + 7 data rows
|
|
|
|
const headers = rows[0].getElementsByTagName('th');
|
|
expect(headers[0].textContent).toBe('Day');
|
|
expect(headers[1].textContent).toBe('Delivered');
|
|
expect(headers[2].textContent).toBe('Failed');
|
|
|
|
const firstRowCells = rows[1].getElementsByTagName('td');
|
|
expect(firstRowCells[0].textContent).toBe('07/01/24');
|
|
expect(firstRowCells[1].textContent).toBe('50');
|
|
expect(firstRowCells[2].textContent).toBe('5');
|
|
});
|
|
|
|
test('SVG element is correctly set up', () => {
|
|
window.createChart('#weeklyChart', ['07/01/24', '07/02/24', '07/03/24', '07/04/24', '07/05/24', '07/06/24', '07/07/24'], [50, 60, 70, 80, 90, 100, 110], [5, 2, 1, 0, 3, 4, 2]);
|
|
|
|
const svg = document.getElementById('weeklyChart').querySelector('svg');
|
|
expect(svg).not.toBeNull();
|
|
expect(svg.getAttribute('width')).toBe('0');
|
|
expect(svg.getAttribute('height')).toBe('400');
|
|
});
|
|
|
|
test('Check HTML content after chart creation', () => {
|
|
// Create sample data for the chart
|
|
const labels = ['07/01/24', '07/02/24', '07/03/24', '07/04/24', '07/05/24', '07/06/24', '07/07/24'];
|
|
const deliveredData = [50, 60, 70, 80, 90, 100, 110];
|
|
const failedData = [5, 2, 1, 0, 3, 4, 2];
|
|
|
|
// Ensure the container has the correct width
|
|
const container = document.getElementById('weeklyChart');
|
|
container.style.width = '600px'; // Force a specific width
|
|
const containerWidth = container.clientWidth;
|
|
expect(containerWidth).toBeGreaterThan(0);
|
|
|
|
// Call the function to create the chart
|
|
window.createChart('#weeklyChart', labels, deliveredData, failedData);
|
|
|
|
// Optionally, you can add assertions to check for specific elements
|
|
expect(container.querySelector('svg')).not.toBeNull();
|
|
expect(container.querySelectorAll('rect').length).toBeGreaterThan(0);
|
|
});
|
|
|
|
// Add toHaveTextContent matcher manually
|
|
expect.extend({
|
|
toHaveTextContent(received, expected) {
|
|
const actualTextContent = received.textContent;
|
|
const pass = actualTextContent.includes(expected);
|
|
if (pass) {
|
|
return {
|
|
message: () =>
|
|
`expected element not to have text content '${expected}', but it was found`,
|
|
pass: true,
|
|
};
|
|
} else {
|
|
return {
|
|
message: () =>
|
|
`expected element to have text content '${expected}', but got '${actualTextContent}'`,
|
|
pass: false,
|
|
};
|
|
}
|
|
},
|
|
});
|
|
|
|
test('Initial dropdown state is correct', () => {
|
|
const dropdown = document.getElementById('options');
|
|
|
|
// Check the initial value of the dropdown
|
|
expect(dropdown.value).toBe('service');
|
|
|
|
// Check the initial subtitle
|
|
const subTitle = document.querySelector('.chart-subtitle');
|
|
expect(subTitle).toHaveTextContent('Service Name - Last 7 Days');
|
|
|
|
// Check the initial ARIA live region text
|
|
const liveRegion = document.getElementById('aria-live-account');
|
|
expect(liveRegion.textContent).toBe(''); // Assuming it starts empty
|
|
});
|
|
|
|
test('handleDropdownChange updates subtitle and fetches individual data', () => {
|
|
// Get the dropdown and set its value to 'individual'
|
|
const dropdown = document.getElementById('options');
|
|
dropdown.value = 'individual';
|
|
|
|
// Create and dispatch a change event
|
|
const changeEvent = new Event('change', { bubbles: true });
|
|
dropdown.dispatchEvent(changeEvent);
|
|
|
|
// Check if subtitle text content is updated correctly
|
|
const subTitle = document.querySelector('.chart-subtitle');
|
|
expect(subTitle).toHaveTextContent('User Name - Last 7 Days');
|
|
});
|
|
|
|
test('handleDropdownChange shows "My Activity" table and hides "All Activity" table', () => {
|
|
// Get the dropdown and set its value to 'individual'
|
|
const dropdown = document.getElementById('options');
|
|
dropdown.value = 'individual';
|
|
|
|
// Create and dispatch a change event
|
|
const changeEvent = new Event('change', { bubbles: true });
|
|
dropdown.dispatchEvent(changeEvent);
|
|
|
|
// Check visibility of the tables
|
|
const table1 = document.getElementById('table1');
|
|
const table2 = document.getElementById('table2');
|
|
|
|
// Expect table1 to be visible and table2 to be hidden
|
|
expect(table1.classList).not.toContain('hidden');
|
|
expect(table1.classList).toContain('visible');
|
|
expect(table2.classList).toContain('hidden');
|
|
expect(table2.classList).not.toContain('visible');
|
|
});
|
|
|
|
test('Dropdown change event triggers handleDropdownChange with logging', () => {
|
|
// Spy on console.log
|
|
const consoleSpy = jest.spyOn(console, 'log').mockImplementation(() => {});
|
|
|
|
// Get the dropdown and set its value
|
|
const dropdown = document.getElementById('options');
|
|
dropdown.value = 'individual';
|
|
|
|
// Create and dispatch a change event
|
|
const changeEvent = new Event('change', { bubbles: true });
|
|
dropdown.dispatchEvent(changeEvent);
|
|
|
|
// Restore the console method
|
|
consoleSpy.mockRestore();
|
|
});
|