- // ==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();