CDC Booking Script

Auto book the time slot in CDC

// ==UserScript==
// @name         CDC Booking Script
// @namespace    http://tampermonkey.net/
// @version      0.2
// @description  Auto book the time slot in CDC
// @author       Liu Chao
// @include      https://www.cdc.com.sg:8080/NewPortal/Booking/BookingPL.aspx
// @icon         https://www.google.com/s2/favicons?domain=undefined.
// @grant        none
// @license      MIT

// ==/UserScript==

/* jshint esversion: 8 */

const isTesting = false;

// settings
// const preferredTimeSlots = [
//   {
//     day: 'WED',
//     slot: '16:25 - 18:05'
//   }
// ]

const getPreferredTimeSlots = () => {
  return JSON.parse(localStorage.getItem('preferredTimeSlots') || '[]');
}

const isLoading = () => {
  const loadingSpinner = document.querySelector('#ctl00_ContentPlaceHolder1_UpdateProgress1');
  return loadingSpinner && loadingSpinner.style.display !== 'none';
}

const toggleSelector = async () => {
  const s = document.querySelector('#ctl00_ContentPlaceHolder1_ddlCourse')
  const toBeSelected = s.selectedIndex === 0 ? 1 : 0;
  s.selectedIndex = toBeSelected;
  s.onchange()
  await sleep(1000);
  while(isLoading()) {
    await sleep(1000);
  }
  return Promise.resolve(toBeSelected);
}

const isTimeSlotPreferred = ({date= '', day ='', slot = ''}) => {
  if (!date) {
    return false;
  }
  const preferredTimeSlots = getPreferredTimeSlots();
  const matched = preferredTimeSlots.find((timeSlot) => timeSlot.day === day && timeSlot.slot === slot);
  return !!matched;
}

const getAvailableSlots = () => {
  const table = document.querySelector('#ctl00_ContentPlaceHolder1_gvLatestav');
  if (!table) {
    return [];
  }
  const rows = table.querySelectorAll('tr');
  const timeSlotRow = rows[0];

  let allSlots = [];
  rows.forEach((row, index) => {
    if (index !== 0) {
      const slots = []
      const cells = row.querySelectorAll('td');
      const timeSlotCells = timeSlotRow.querySelectorAll('th');
      cells.forEach((cell, index) => {
        const slotInput = cell.querySelector('input');
        if (slotInput) {
          const src = slotInput.src || '';
          if (src.includes('Images1')) {
            const date = cells[0].textContent;
            const day = cells[1].textContent;
            const slot = timeSlotCells[index].textContent.slice(1)
            slots.push({
              slotInput, date, day, slot
            })
          }
        }
      });
      allSlots = allSlots.concat(slots);
    }
  });
  return allSlots;
}

const reserve = (availableSlots) => {
  const timeAvailableSlot = availableSlots.find((slot) => isTimeSlotPreferred(slot));
  if (timeAvailableSlot) {
    const slotInput = timeAvailableSlot.slotInput;
    if (slotInput) {
      slotInput.click();
      return timeAvailableSlot;
    }
  }
  return false;
}

const run = async () => {
  try {
    let selectedIndex;
    if (isTesting) {
      selectedIndex = 1;
    } else {
      selectedIndex = await toggleSelector();
    }
    if (selectedIndex === 0) {
      await toggleSelector();
    }
    const availableSlots = getAvailableSlots();
    const hasResult = !!availableSlots.length;
    if (hasResult) {
      const reservedSlot = reserve(availableSlots);
      let msg;
      if (reservedSlot) {
        msg = `Slot is reserved\n${reservedSlot.date} ${reservedSlot.day} ${reservedSlot.slot}`;
      } else {
        const slotsMsg = `${availableSlots.map(({date, day, slot}) => `${date} ${day} ${slot}`).join('\n')}`;
        msg = `Slots are available\n${slotsMsg}`;
      }
      sendMessage(`[${new Date().toLocaleString('en-ca')}] ${msg}`);
    }
    console.log(`Checked again with result: ${hasResult}`);
    return Promise.resolve(hasResult);
  } catch(e) {
    return Promise.reject(`error: ${e}`)
  };
}

const getRandomTime = (time) => {
  const seed = Math.random();
  return parseInt(time * (0.5 + seed));
}

const sleep = (interval, random=false) => {
  return new Promise((resolve) => {
    window.setTimeout(resolve, random ? getRandomTime(interval) : interval);
  });
}

// https://simpleguics2pygame.readthedocs.io/en/latest/_static/links/snd_links.html
const audio = new Audio('http://codeskulptor-demos.commondatastorage.googleapis.com/descent/gotitem.mp3');

const sendMessage = (msg) => {
  // alert(msg);
  console.log(msg);
  audio.play();
}

let startButton;
let settingButton;
let settingPopup;

const start = async () => {
  const hasResult = await run() ;
  if (!isTesting) {
    if (hasResult) {
      await sleep(3 * 1000, true);
    } else {
      await sleep(10 * 1000, true);
    }
    start();
  }
}

const insertStartButton = () => {
  startButton = document.createElement('button');
  startButton.className = 'start-button';
  startButton.textContent = 'start';

  const startButtonStyle = document.createElement('style');
  startButtonStyle.innerHTML = `
    .start-button {
      position: fixed;
      bottom: 70px;
      right: 50px;
      width: 100px;
      padding: 5px;
      line-height: 1;
      background-color: #ff9933;
      border: none;
      border-radius: 10px;
      font-size: 18px;
      cursor: pointer;
    }
  `;
  document.head.appendChild(startButtonStyle);
  document.body.appendChild(startButton);

  let hasStarted = false;
  startButton.addEventListener('click', async () => {
    try {
      if (!hasStarted) {
        start();
        hasStarted = true;
        startButton.textContent = 'running...'
      }
    } catch(e) {
      startButton.textContent = 'error'
      console.log('error', e);
    }
  });
}


const insertSettingButton = () => {
  settingButton = document.createElement('button');
  settingButton.className = 'setting-button';
  settingButton.textContent = 'setting';

  const settingButtonStyle = document.createElement('style');
  settingButtonStyle.innerHTML = `
    .setting-button {
      position: fixed;
      bottom: 30px;
      right: 50px;
      width: 100px;
      padding: 5px;
      line-height: 1;
      background-color: #ff9933;
      border: none;
      border-radius: 10px;
      font-size: 18px;
      cursor: pointer;
    }
  `;
  document.head.appendChild(settingButtonStyle);
  document.body.appendChild(settingButton);
  settingButton.addEventListener('click', async () => {
    toggleSettingPopup();
  });
}

const toggleSettingPopup = () => {
  if (window.getComputedStyle(settingPopup).display === 'none') {
    settingPopup.style.display = 'initial';
  } else {
    settingPopup.style.display = 'none';
  }
}

const insertSettingPopup = () => {
  settingPopup = document.createElement('div');
  settingPopup.className = 'setting-popup';

  const settingPopupStyle = document.createElement('style');
  settingPopupStyle.innerHTML = `
    .setting-popup {
      display: none;
      position: fixed;
      bottom: 65px;
      right: 50px;
      width: 900px;
      padding: 10px;
      background-color: #ff9933;
      border: none;
      border-radius: 10px;
      font-size: 22px;
    }

    .setting-popup label {
      margin-bottom: 10px;
      display: block;
    }

    .setting-popup table {
      width: 100%;
      table-layout: auto;
      border-collapse: collapse;
    }

    .setting-popup table th,
    .setting-popup table td {
      text-align: center;
      border: 1px solid black;
    }

    .setting-popup table th {
      padding: 5px;
    }

    .setting-popup table td {
      height: 35px;
    }

    .setting-popup table img {
      width: 30px;
    }

    .setting-popup table .selectable {
      cursor: pointer;
    }

    .setting-buttons {
      display: flex;
      flex-wrap: wrap;
    }
    .setting-buttons .toggle-button {
      border: none;
      padding: 5px;
      border-radius: 5px;
      margin-right: 5px;
      margin-bottom: 5px;
      cursor: pointer;
    }

    .setting-buttons .toggle-button.active {
      background-color: red;
    }
  `;
  document.head.appendChild(settingPopupStyle);
  document.body.appendChild(settingPopup);

  renderSettingTable();
}

const toggleTimeSlot = (day, slot) => {
  const savedPreferredTimeSlots = getPreferredTimeSlots()
  const matched = savedPreferredTimeSlots.find((timeSlot) => timeSlot.day === day && timeSlot.slot === slot);
  if (matched) {
    savedPreferredTimeSlots.splice(savedPreferredTimeSlots.indexOf(matched), 1);
  } else {
    savedPreferredTimeSlots.push({
      day,
      slot
    })
  }
  localStorage.setItem('preferredTimeSlots', JSON.stringify(savedPreferredTimeSlots))
  renderSettingTable();
}

const renderSettingTable = () => {
  const slots = ['08:30 - 10:10', '10:20 - 12:00', '12:45 - 14:25', '14:35 - 16:15', '16:25 - 18:05', '18:50 - 20:30', '20:40 - 22:20'];
  const days = ['MON', 'TUE', 'WED', 'THU', 'FRI', 'SAT', 'SUN'];

  window.toggleTimeSlot= toggleTimeSlot;
  const savedPreferredTimeSlots = getPreferredTimeSlots();

  const table = `
    <label>Preferred Time Slots</label>
    <table>
      <thead>
        <tr>
          <th>Day</th>
          ${slots.map(slot => `<th>${slot}</th>`).join('\n')}
        </tr>
      </thead>
      ${days.map(day => {
        return `
          <tr>
            <td>${day}</td>
            ${slots.map((slot) => {
              const isSelected = !!savedPreferredTimeSlots.find((timeSlot) => timeSlot.day === day && timeSlot.slot === slot);
              return `
                <td class="selectable" onclick="toggleTimeSlot('${day}', '${slot}', '${isSelected}')">
                  ${isSelected ? '<img src="" />' : ''}
                </td>
              `;
            }).join('\n')}
          </tr>`;
      }).join('\n')}
    </table>
  `;

  settingPopup.innerHTML = table;
}

const main = async () => {
  insertStartButton();
  insertSettingButton();
  insertSettingPopup();
}

(function() {
  'use strict';
  main();
})();