Canvas All Info

Place all information on a single page (https://canvas.example.com/all or https://example.instructure.com/all)

// ==UserScript==
// @name         Canvas All Info
// @namespace    https://theusaf.github.io
// @version      1.4.0
// @icon         https://canvas.instructure.com/favicon.ico
// @copyright    2020-2021, Daniel Lau
// @license      MIT
// @description  Place all information on a single page (https://canvas.example.com/all or https://example.instructure.com/all)
// @author       theusaf
// @include      /^https:\/\/canvas\.[a-z0-9]*?\.[a-z]*?\/all\/?(\?.*)?$/
// @include      /^https:\/\/[a-z0-9]*?\.instructure\.com\/all\/?(\?.*)?$/
// @inject-into  page
// @grant        none
// ==/UserScript==

/*
  Note:
  - This userscript uses public APIs accessed by canvas
    - Gets class information
    - Gets assignments
    - Gets basic teacher information
  - This userscript does not store or upload any information gathered by the script
  - This userscript overwrites /all in canvas
  - This userscript was originally developed for Oregon State University
*/

/* Useful Links (for use later?)
/api/v1/conversations?scope=inbox&filter_mode=and&include_private_conversation_enrollments=false
- Canvas mail
/api/v1/conversations/(mailbox_id)?include_participant_contexts=false&include_private_conversation_enrollments=false
- Specific mail
/courses/(class_id)/modules/items/assignment_info
- Module items
*/

/**
 * log - logs info to console
 * @param {*} str
 * @param  {...any} data
 */
const log = (str, ...data) => {
  if (data.length > 0) {
    console.log(`[CANVAS-ALL] ${str}`, data);
  }
  console.log(`[CANVAS-ALL] ${str}`);
};

/**
 * load - Loads everything
 */
async function load() {
  document.title = "Dashboard - All";

  /**
   * mainElement - the main application div
   * iFrameLoader - The div for loading iframes for getting data
   * styles - A style element
   */
  const mainElement = document.getElementById("main"),
    iFrameLoader = document.createElement("div");
  mainElement.innerHTML = `<style>
    #canvas-all-iframe-loader{
      visibility: hidden;
      position: fixed;
      width: 100%;
      height: 100%;
      pointer-events: none;
      left: 0;
    }
    #canvas-all-iframe-loader > iframe{
      width: 100%;
      height: 100%;
      position: absolute;
      left: 0;
      top: 0;
    }
    #main{
      display: flex;
      flex-flow: column;
      padding: 1rem;
    }
    #main>span{
      flex: 0;
      margin-bottom: 1rem;
    }
    #main>div{
      flex: 1;
    }
    #canvasall_main_wrapper{
      display: flex;
    }
    #canvasall_main_wrapper>div{
      flex: 75%;
    }
    #canvasall_class_grades{
      display: flex;
      flex-flow: column;
      background: #fff5e0;
      padding: 0.5rem;
      border-radius: 0.5rem;
    }
    #canvasall_assignments_wrapper{
      padding: 0.5rem;
      background: #fff5e0;
      border-radius: 0.5rem;
    }
    #canvasall_main_wrapper>#canvasall_announcement_wrapper{
      flex: 25%;
      padding: 0.5rem;
    }
    #canvasall_assignment_filter_list_chosen{
      display: inline-block;
    }
    #canvasall_assignment_filter_list_chosen>option{
      display: inline-block;
      background: grey;
      color: white;
      padding: 0.25rem;
      margin: 0.25rem;
      cursor: pointer;
    }
    #canvasall_assignment_filter_list_chosen>option::before{
      content: "x ";
    }
    .canvasall_announcement_wrapper{
      margin-bottom: 0.5rem;
      padding: 0.5rem;
      border-radius: 0.5rem;
    }
    .canvasall_announcement_wrapper:nth-child(2n+1){
      background: #fff5e0;
    }
    .canvasall_announcement_wrapper:nth-child(2n){
      background: #ddd;
    }
    .canvasall_announcement_title{
      font-size: 1.25rem;
      font-weight: bold;
    }
    .canvasall_announcement_class,
    .canvasall_announcement_when{
      font-size: 0.75rem;
    }
    .canvasall_announcement_class>a{
      color: grey;
    }
    .canvasall_class_grade_wrapper:nth-child(2n+1){
      background: white;
    }
    .canvasall_class_grade_wrapper:nth-child(2n){
      background: #eee;
    }
    .canvasall_class_grade_wrapper{
      flex: 1;
      display: flex;
      border-radius: 0.5rem;
    }
    .canvasall_class_grade_wrapper>div{
      flex: 1;
      padding: 0.5rem;
      word-break: break-all;
    }
    .canvasall_assignment_wrapper:nth-child(2n+1){
      background: rgba(255,255,255,0.8);
    }
    .canvasall_assignment_wrapper:nth-child(2n){
      background: rgba(255,255,255,0.4);
    }
    .canvasall_assignment_wrapper{
      display: flex;
      padding: 0.5rem;
      border-radius: 0.5rem;
    }
    .canvasall_assignment_wrapper>div{
      flex: 1;
      padding-right: 0.25rem;
      padding-left: 0.25rem;
    }
    .canvasall_assignment_title{
      display: flex;
      align-items: center;
    }
    .canvasall_assignment_title>img{
      height: 1.5rem;
      width: 1.5rem;
      margin-right: 0.5rem;
    }
    .canvasall_assignment_icon {
      min-height: 1.5rem;
      min-width: 1.5rem;
      margin-right: 0.5rem;
    }
    .canvasall_assignment_type_assignment:before {
      content: url("");
    }
    .canvasall_assignment_type_quiz:before {
      content: url("");
    }
    .canvasall_assignment_type_discussion_topic:before {
      content: url("")
    }
    .canvasall_status_submit,
    .canvasall_status_submit a{
      color: green;
    }
    .canvasall_status_done{
      text-decoration: line-through;
    }
    .canvasall_status_late{
      background: orange !important;
      color: white;
    }
    .canvasall_status_late a{
      color: white;
    }
    .canvasall_status_missing{
      background: red !important;
      color: white;
    }
    .canvasall_status_missing a{
      color: white;
    }
    .canvasall_status_feedback>.canvasall_assignment_title::after{
      content: " (Feedback available)"
    }
  </style>
  <span id="canvasall_fetching_information">Fetching information... please wait...</span>
  <div id="canvasall_main_wrapper">
    <div>
      <div id="canvasall_class_wrapper">
        <!-- Courses, Grades, etc. -->
        <h3>Courses</h3>
        <div id="canvasall_class_grades">
          <div id="canvasall_class_grade_header" class="canvasall_class_grade_wrapper">
            <div><span>Class</span></div>
            <div><span>Grade</span></div>
            <div><span>Professor</span></div>
          </div>
        </div>
      </div>
      <h3>Current Assignments</h3>
      <div>
        <span>Filters:</span>
        <select id="canvasall_assignment_filter_status">
          <option value="">Select</option>
          <option value="submit">Hide Submitted</option>
          <option value="done">Hide Graded</option>
          <option value="late">Hide Late</option>
          <option value="missing">Hide Missing</option>
          <option value="quiz">Hide Quiz</option>
          <option value="assignment">Hide Assignment</option>
        </select>
        <div id="canvasall_assignment_filter_list_chosen">
        </div>
      </div>
      <div id="canvasall_assignments_wrapper">
        <div class="canvasall_assignment_wrapper">
          <div>
            <span>Assignment Name</span>
          </div>
          <div>
            <span>Class</span>
          </div>
          <div>
            <span>Due Date</span>
          </div>
        </div>
      </div>
    </div>
    <div id="canvasall_announcement_wrapper">
      <h3>Announcements</h3>
    </div>
  </div>`;
  iFrameLoader.id = "canvas-all-iframe-loader";
  mainElement.append(iFrameLoader);

  /**
   * courses - Class information
   * classAssignments - Assignments for courses
   * mainFrame - The main iframe
   * ENV - Global variables with useful data
   * currentUserID - The current user id
   */
  log("Getting courses");
  const courses = await getCourses(),
    courseAssignments = [],
    courseGrades = {},
    { ENV } = window,
    { currentUserID } = ENV,
    courseGradeDiv = mainElement.querySelector("#canvasall_class_grades"),
    assignmentsDiv = mainElement.querySelector("#canvasall_assignments_wrapper"),
    announcementsDiv = mainElement.querySelector(
      "#canvasall_announcement_wrapper"
    ),
    statusFilter = mainElement.querySelector(
      "#canvasall_assignment_filter_status"
    ),
    statusFilterList = mainElement.querySelector(
      "#canvasall_assignment_filter_list_chosen"
    ),
    localStorageConfigStr = localStorage.canvasAllConfig ?? "{}",
    localStorageConfig = JSON.parse(localStorageConfigStr);

  function hideType(value) {
    console.log(`Hiding '${value}'`);
    if (value === "") {
      // ignore reset to empty value
      console.log("Ignoring empty value");
      return;
    }
    const elem = statusFilter.querySelector(
        `option[value="${value}"]`
      ),
      temp = document.createElement("template"),
      now = `${Math.random()}`.substring(2);
    if (!elem) {
      console.error("Could not find element");
      return;
    }
    temp.innerHTML = `<style id="canvasall_filter_${now}">
      .canvasall_status_${value}{
        display: none;
      }
    </style>`;
    localStorageConfig[value] = true;
    localStorage.canvasAllConfig = JSON.stringify(localStorageConfig);
    setTimeout(() => {
      document.body.append(temp.content.cloneNode(true));
      statusFilterList.append(elem);
      elem.addEventListener("click", click);
      statusFilter.value = "";
    });
    function click() {
      localStorageConfig[value] = false;
      localStorage.canvasAllConfig = JSON.stringify(localStorageConfig);
      // remove style
      document.querySelector(`#canvasall_filter_${now}`).remove();
      elem.removeEventListener("click", click);
      setTimeout(() => {
        statusFilter.append(elem);
        statusFilter.value = "";
      });
    }
  }

  // Filters
  statusFilter.addEventListener("change", () => {
    hideType(statusFilter.value);
  });

  // Begin writing class information
  for (const [i, course] of Object.entries(courses)) {
    const template = document.createElement("template");
    template.innerHTML = `<div class="canvasall_class_grade_wrapper" canvasall-class-id="${
      course.id
    }">
      <div class="canvasall_class_grade_name">
        <span>${+i + 1}. <a href="/courses/${course.id}">${
      course.name
    }</a></span>
      </div>
      <div class="canvasall_class_grade_score">
        <span>(Loading scores...)</span>
      </div>
      <div class="canvasall_class_grade_instructor">
        <span>(Loading instructors...)</span>
      </div>
    </div>`;
    courseGradeDiv.append(template.content.cloneNode(true));
  }

  log("Getting course assignments and grades");
  let courseCount = 0,
    finishedCollections = 0;
  for (const course of courses) {
    courseCount++;
    getCourseAssignments(course.id, currentUserID).then(assignments => {
      courseAssignments.push.apply(
        courseAssignments,
        assignments
      );
      finishedCollections++;
      if (finishedCollections === courseCount) {
        // Write assignments
        courseAssignments.sort((a, b) => {
          // sort by "due date"
          // also places non-due-date at end. (+1 week)
          const dueA = new Date(
              a.plannable.due_at || b.plannable.created_at || Date.now() + 604800e3
            ),
            dueB = new Date(
              b.plannable.due_at || b.plannable.created_at || Date.now() + 604800e3
            );
          return dueA.getTime() - dueB.getTime();
        });

        log("Compiling assignments");
        for (const assignment of courseAssignments) {
          const template = document.createElement("template");
          if (assignment.plannable_type === "announcement") {
            // Put in announcement thing
            template.innerHTML = `<div class="canvasall_announcement_wrapper">
                <div class="canvasall_announcement_title">
                  <a href="${assignment.html_url}">${assignment.plannable.title}</a>
                </div>
                <div class="canvasall_announcement_class">
                  <a href="/courses/${assignment.course_id}">${
              assignment.context_name
            }</a>
                </div>
                <div class="canvasall_announcement_when">
                  <span>${new Date(assignment.plannable.created_at)
                    .toString()
                    .split(" ")
                    .slice(0, 5)
                    .join(" ")
                    .replace(/:\d{2}$/, "")
                    .replace(/\s(?=\w{3}\s\d{2})/, ", ")
                    .replace(/\d{4}/, "at")}</span>
                </div>
              </div>`;
            announcementsDiv.append(template.content.cloneNode(true));
            continue;
          }
          const divClasses = [],
            { submissions } = assignment;
          if (submissions) {
            const { excused, graded, has_feedback, late, missing, submitted } =
              submissions;
            if (excused || graded) {
              divClasses.push("canvasall_status_done");
            }
            if (submitted) {
              divClasses.push("canvasall_status_submit");
            }
            if (late) {
              divClasses.push("canvasall_status_late");
            }
            if (missing) {
              divClasses.push("canvasall_status_missing");
            }
            if (has_feedback) {
              divClasses.push("canvasall_status_feedback");
            }
          }

          divClasses.push("canvasall_status_" + assignment.plannable_type);
          template.innerHTML = `<div class="canvasall_assignment_wrapper ${divClasses.join(
            " "
          )}" class-id="${assignment.course_id}">
            <div class="canvasall_assignment_title">
              <div class="canvasall_assignment_icon canvasall_assignment_type_${
                assignment.plannable_type
              }"></div>
              <a href="${assignment.html_url}">${assignment.plannable.title}</a>
            </div>
            <div class="canvasall_assignment_class">
              <a href="/courses/${assignment.course_id}">${
            assignment.context_name
          }</a>
            </div>
            <div class="canvasall_assignment_due">
              <span>${
                assignment.plannable.due_at
                  ? new Date(assignment.plannable.due_at)
                      .toString()
                      .split(" ")
                      .slice(0, 5)
                      .join(" ")
                      .replace(/:\d{2}$/, "")
                      .replace(/\s(?=\w{3}\s\d{2})/, ", ")
                      .replace(/\d{4}/, "at")
                  : "No due date"
              }</span>
            </div>
          </div>`;
          assignmentsDiv.append(template.content.cloneNode(true));
        }

        // Restore hide
        for (const [key, value] of Object.entries(localStorageConfig)) {
          if (value && key !== "") {
            hideType(key);
          }
        }

        courseAssignments.splice(0); // Free up memory
        document.querySelector("#canvasall_fetching_information").outerHTML = "";
      }
    });
    // Get grades
    loadFrame(iFrameLoader, `/courses/${course.id}/grades`).then(
      (ClassGradeFrame) => {
        const { document: d, window: w, frame } = ClassGradeFrame;
        let overallGrade = d
            .querySelector(
              "#student-grades-right-content .student_assignment.final_grade > .grade"
            )
            ?.textContent?.replace(/%|\s/g, ""),
          titles = d.querySelectorAll(".title"),
          possiblePoints = d.querySelectorAll(".points_possible"),
          dueDates = d.querySelectorAll(".due");
        courseGrades[course.id] = {
          Grades: overallGrade + "%",
          Titles: titles,
          PossiblePoints: possiblePoints,
          DueDates: dueDates,
        };
        // Write grades
        const gradeDiv = courseGradeDiv.querySelector(
            `[canvasall-class-id="${course.id}"] > .canvasall_class_grade_score`
          ),
          { ENV } = w,
          { grading_scheme } = ENV,
          [letter] =
            (grading_scheme ?? ["N/A", 0]).find((scheme) => {
              try {
                return +overallGrade / 100 >= scheme[1];
              } catch (e) {
                console.error(e);
              }
            }) ?? ["?"];
        gradeDiv.innerHTML =
          overallGrade !== "N/A" && overallGrade
            ? `<span>${letter} (${overallGrade}%)</span>`
            : `<span>N/A</span>`;
        // Done using data from iframe, attempt to clean up memory usage
        overallGrade = titles = possiblePoints = dueDates = null;
        for (const i in courseGrades) {
          delete courseGrades[i];
        }
        w.location = "about:blank";
        setTimeout(() => {
          frame.remove();
        }, 500);
      }
    );
    // Get instructors
    getCourseTeacher(course.id)
      .then((teacher) => {
        const teacherDiv = courseGradeDiv.querySelector(
          `[canvasall-class-id="${course.id}"] > .canvasall_class_grade_instructor`
        );
        teacherDiv.innerHTML = `<span>${teacher.name}</span>`;
      })
      .catch(() => {
        // no teacher, or failed to get teacher
        const teacherDiv = courseGradeDiv.querySelector(
          `[canvasall-class-id="${course.id}"] > .canvasall_class_grade_instructor`
        );
        teacherDiv.innerHTML = "<span>No instructor found.</span>";
      });
  }
}

// /api/v1/users/:user_id/communication_channels
/**
 * getCourses - Gets all the classes of the user
 *
 * @returns {Promise<Array>} A list of class information
 */
function getCourses() {
  return new Promise((res, rej) => {
    const x = new XMLHttpRequest();
    x.open(
      "GET",
      "/api/v1/users/self/favorites/courses?include[]=term&exclude[]=enrollments&sort=nickname"
    );
    x.setRequestHeader("X-Requested-With", "XMLHttpRequest");
    x.setRequestHeader(
      "accept",
      "application/json, text/javascript, application/json+canvas-string-ids, */*; q=0.01"
    );
    x.onload = () => {
      res(JSON.parse(x.responseText));
    };
    x.onerror = () => {
      rej();
    };
    x.send();
  });
}

/**
 * GetClassAssignments - Gets the class assignments
 *
 * @param  {String} courseID The class id
 * @param  {String} userID The user id
 * @returns {Promise<Array>} The list of assignments
 */
function getCourseAssignments(courseID, userID) {
  const x = new XMLHttpRequest(),
    now = new Date(),
    offset = now.getTimezoneOffset() / 60,
    nextUrlRegex = /[^<>]*(?=>; rel="next")/;
  now.addDays(-14); // To get assignments from 2 weeks ago
  now.setHours(8 - offset);
  now.setMinutes(0);
  now.setSeconds(0);
  now.setMilliseconds(0);

  function makeHttpRequest(url) {
    return new Promise((res, rej) => {
      x.open("GET", url);
      x.setRequestHeader("X-Requested-With", "XMLHttpRequest");
      x.setRequestHeader(
        "accept",
        "application/json+canvas-string-ids, application/json+canvas-string-ids, application/json, text/plain, */*"
      );
      x.onerror = () => {rej();}
      x.onload = () => {
        res(x);
      }
      x.send();
    });
  }

  async function gatherRecursiveRequests(url, list = []) {
    const response = await makeHttpRequest(url),
      data = JSON.parse(response.responseText),
      linkHeader = response.getResponseHeader("link");
    list.push(...data);
    if (linkHeader) {
      const lastLink = linkHeader.match(nextUrlRegex);
      if (lastLink) {
        return gatherRecursiveRequests(lastLink[0], list);
      } else {
        return list;
      }
    } else {
      return list;
    }
  }

  return gatherRecursiveRequests(`/api/v1/planner/items?start_date=${now.toISOString()}&order=asc&context_codes[]=course_${courseID}`) //&context_codes[]=user_${userID}
    .catch(() => []);
}

/**
 * getCourseTeacher - Gets the teacher of the course
 *
 * @param  {String} courseID The class id
 * @returns {Object} The teacher
 */
function getCourseTeacher(courseID) {
  return new Promise((res, rej) => {
    const x = new XMLHttpRequest();
    x.open(
      "GET",
      `/api/v1/search/recipients?search=&per_page=20&permissions[]=send_messages_all&messageable_only=true&synthetic_contexts=true&context=course_${courseID}_teachers`
    );
    x.setRequestHeader("X-Requested-With", "XMLHttpRequest");
    x.setRequestHeader(
      "Accept",
      "application/json, text/javascript, application/json+canvas-string-ids, */*; q=0.01"
    );
    x.onload = () => {
      res(JSON.parse(x.responseText)[0]);
    };
    x.onerror = () => {
      rej();
    };
    x.send();
  });
}

/**
 * loadFrame - Loads an iframe
 *
 * @param  {HTMLElement} iFrameLoader The element to append iframes to
 * @param  {String} src The iframe url to load
 * @returns {Promise<Object>} returns an object with the "window" and "document" of the iframe
 */
function loadFrame(iFrameLoader, src) {
  return new Promise((res) => {
    const mainFrame = document.createElement("iframe");
    mainFrame.src = src || "/";
    iFrameLoader.append(mainFrame);
    mainFrame.addEventListener("load", () => {
      const frameContext = {
        document: mainFrame.contentDocument,
        window: mainFrame.contentWindow,
        frame: mainFrame
      };
      res(frameContext);
    });
  });
}

load();