Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Staging #301

Merged
merged 7 commits into from
Sep 19, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions src/js/main.js
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ import tinyMCEEditor from './modules/tinyMCEEditor';
import uob from './3party/uob7.js';
import util from './modules/util.js';
import utilRoot from './utilRoot.js';
import uucheck from './modules/uucheck.js';

jQuery(function($) {
//KURSP-469 Support embedding of KPAS LTI tool. In general our design should not load in iframes.
Expand Down Expand Up @@ -212,6 +213,8 @@ jQuery(function($) {
coursesettings.addListUsersButton();
coursesettings.addListGroupsButton();
coursesettings.addListAssignmentsButton();
uucheck.addUUButton();

});

routes.addRouteForPath(/\/profile\/settings$/, function() {
Expand Down
6 changes: 4 additions & 2 deletions src/js/modules/fknr.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
utgaatteKommuneNr = [
var fknr = { utgaatteKommuneNr : [
1,
2,
4,
Expand Down Expand Up @@ -425,4 +425,6 @@ utgaatteKommuneNr = [
2121,
2131,
2111
]
]};

export default fknr;
2 changes: 1 addition & 1 deletion src/js/modules/menu.js
Original file line number Diff line number Diff line change
Expand Up @@ -297,7 +297,7 @@ export default (function() {
if(moduleItemSequence && moduleItemSequence.items.length && moduleItemSequence.items[0].next) {
var nextItem = moduleItemSequence.items[0].next;
if(nextItem.indent) {
id = nextItem.id;
var id = nextItem.id;
api.getModuleItemSequence(courseId, id, handleNextModuleItem);
} else {
var nextButton = $(".module-sequence-footer-button--next");
Expand Down
6 changes: 4 additions & 2 deletions src/js/modules/util.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import { hrefAmpQueryString, hrefQueryString } from "../settingsRoot";

import { CourseOptions } from "../utilities/course-options";
import api from '../api/api.js'
import fknr from './fknr.js';
import pages from './pages.js'
import settings from "../settings";

Expand Down Expand Up @@ -322,6 +323,7 @@ export default (function () {
});
},
isMemberOfExpiredCommunity(course, callback) {
let self = this
if(!course) {
return;
}
Expand All @@ -330,9 +332,9 @@ export default (function () {
if (groups.length) {
for (var i = 0; i < groups.length; i++) {
var group = groups[i];
var countyOrCommunityNumber = getCountyOrCommunityNumber(group.description);
var countyOrCommunityNumber = self.getCountyOrCommunityNumber(group.description);
if (countyOrCommunityNumber) {
if (utgaatteKommuneNr.indexOf(countyOrCommunityNumber) > -1) {
if (fknr.utgaatteKommuneNr.indexOf(countyOrCommunityNumber) > -1) {
memberOfUtgaattKommune = true;
break;
}
Expand Down
210 changes: 210 additions & 0 deletions src/js/modules/uucheck.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,210 @@
export default (function () {
class Translated {
constructor (words, defaultlang='nb'){
this.words = words;
this.defaultlang = defaultlang;
}
toString(){
return this.words[ENV.LOCALE]?this.words[ENV.LOCALE]:this.words[this.defaultlang];
}
use(...values){
let returnstring = this.words[ENV.LOCALE]?this.words[ENV.LOCALE]:this.words[this.defaultlang];
for (let i=0; i<values.length; i++){
returnstring = returnstring.replace(`[${i}]`, values[i]);
}
return returnstring;
}
}


function graphql(querydata){
//get csrf_token from the pages cookie
let csrf_token = decodeURIComponent(document.cookie.split(';').find((item) => item.trim().startsWith('_csrf_token')).replace(/\s*_csrf_token\s*\=\s*(.*)$/,"$1"));
if (typeof(querydata)=="string") querydata = {'query': querydata};
//create the graphql query using the querydata input
return fetch(location.origin+"/api/graphql",{
method: 'POST',
mode: 'cors',
credentials: 'same-origin',
headers: {
'Accept': 'application/json+canvas-string-ids, application/json, text/plain, */*',
'Content-Type': 'application/json',
'x-csrf-token': csrf_token
},
body: JSON.stringify(querydata)
})

}
function Get (last_part_of_url){
//Does a Canvas API call and returns a JS-object.
return fetch(location.origin+'/api/v1/'+last_part_of_url);
}


let error = new Translated({en: 'Error', nb: 'Feil', nn: 'Feil'})
let warning = new Translated({en: 'Warning', nb: 'Advarsel', nn: 'Åtvaring'})
let imgerror = new Translated({en: "must be marked as decorative or get an alt text which isn't the filename.", nb:'må markeres som dekorativt eller få alternativ tekst som ikke er filnavnet.', nn:'må markerast som dekorativt eller få alternativ tekst som ikkje er filnamnet.'})
let imgwarning = new Translated({en: "An image is marked as decorative in the old Canvas unstandared way. Open the Image Options and choose Done to rectify.", nb: "Et bilde er merket som dekorativt på den gamle ustandardiserte måten. Åpne Bilde-alternativer og klikk Ferdig for å rette opp.", nn: "Eit bilde er merka som dekorativt på den gamle ustandardiserte måten. Åpne Bilde-alternativ og klikk Ferdig for å retta opp."})
let tablecaptionerror = new Translated({en: 'Table is missing caption.', nb:'Tabell mangler overskrift.', nn:'Tabell manglar overskrift.'})
let tableheadererror = new Translated({en: 'Table is missing headers for rows and/or columns.', nb:'Tabell mangler titler på rader og/eller kolonner.', nn:'Tabell manglar titlar på rader og/eller kolonnar.'})
let herror = new Translated({en: 'Error in the heading hierarchy.', nb:'Feil i overskriftshierarkiet.', nn:'Feil i overskriftshierarkiet.'})
let stylewarning = new Translated({en:'Uses style - check if it is used to make a headline, destroys contrast or leads to other problems.', nb:'Bruker style - sjekk om det brukes for å lage overskrift, ødelegger kontrast eller fører til andre problemer.', nn:'Brukar style - sjekk om det blir brukt for å laga overskrift, øydelegg kontrast eller fører til andre problem.'})
let summary = new Translated({en:'[0] errors, [1] warnings. Remember that this test does not check files, and only does a limited check of pages.', nb:'<hr><p>[0] feil, [1] advarsler. Husk at denne testen ikke sjekker filer, og bare gjør en begrenset sjekk av sider.</p>', nn:'[0] feil, [1] Åtvaringar. Hugs at denne testen ikkje sjekkar filer, og berre gjer ein avgrensa sjekk av sider.'})
let nothing = new Translated({en: "Can't find anything to check.", nb:'Finner ingenting å sjekke.', nn: 'Finn ikkje noko å sjekka.'})
let buttonname = new Translated({en: 'Mini accessibility check', nb:'Mini UU-sjekk', nn:'Mini UU-sjekk'})
let assignment = new Translated({en: 'Assignment', nb: 'Oppgave', nn: 'Oppgåve'})
let discussion = new Translated({en: 'Discussion', nb: 'Diskusjon', nn: 'Diskusjon'})
let title = new Translated({en:'Mini accessibility check of', nb:'Mini-UUsjekk av', nn:'Mini-UUsjekk av'})
let moduleword = new Translated({en:'Module', nb:'Modul', nn:'Modul'})
let frontpage = new Translated({en:'Front page', nb:'Framside', nn:'Framside'})




let content
let myReport = document.getElementById('content')
let warningimg = `<img src="" alt="${warning}" style="display:inline-block" height="20px" width="20px">`
let warningcount = 0
let errorimage = `<img src="" alt="${error}" style="display:inline-block" height="20px" width="20px">`
let errorcount = 0

function addToReport (htmlstring){
let div=document.createElement('div')
div.innerHTML = htmlstring
myReport.append(div)
}

function HTMLcheck(html){
let div = document.createElement('div')
div.innerHTML = html
let HTMLoutput = ''
//imagecheck
let imgtags = div.querySelectorAll('img')
for (const img of imgtags){
if (img.getAttribute('role') == 'presentation') continue
if (img.getAttribute('data-decorative')){
HTMLoutput += `<p>${warningimg} <strong>${img.alt}</strong> ${imgwarning}</p>`
warningcount++
continue
}
if (!['.jpg','.jpeg','.gif','.png','.svg'].includes(img.alt.substring(img.alt.lastIndexOf('.')))) continue
HTMLoutput += `<p>${errorimage} <strong>${img.alt}</strong> ${imgerror}</p>`
errorcount++
}
let tables = div.querySelectorAll('table')
for (const table of tables){
if (!table.querySelector('caption')){
HTMLoutput += `<p>${errorimage} ${tablecaptionerror}</p>`
errorcount++
}
if (!table.querySelector('th')){
HTMLoutput += `<p>${errorimage} ${tableheadererror}</p>`
errorcount++
}
}
let headers = div.querySelectorAll("h2, h3, h4, h5, h6");
let startlevel = 1;
for (const header of headers){
let newlevel = Number(header.tagName.substring(1))
if (newlevel > startlevel+1){
HTMLoutput += `<p>${errorimage} ${herror}</p>`
errorcount++
break
}
startlevel = newlevel
}
if (html.includes('style=')){
HTMLoutput += `<p>${warningimg} ${stylewarning}</p>`
warningcount++
}

return HTMLoutput
}

function assignmentcheck(item){
let log = HTMLcheck(item.description)
if (log)
addToReport(`<h3>${assignment} <a href="${location.origin}/courses/${ENV.COURSE_ID}/assignments/${item._id}/edit" target="_blank">${item.name}</a></h3>${log}`)
}
function discussioncheck(item){
let log = HTMLcheck(item.message)
if (log)
addToReport(`<h3>${discussion} <a href="${location.origin}/courses/${ENV.COURSE_ID}/discussion_topics/${item._id}/edit" target="_blank">${item.title}</a></h3>${log}`)
}
function pagecheck(item){
if (!item._id) item._id = item.page_id
if (!item.body)
item = Get(`courses/${ENV.COURSE_ID}/pages/${item._id}`)
if (!item.body) return
console.log(item.body)
let log = HTMLcheck(item.body)
if (log){
addToReport(`<h3><a href="${location.origin}/courses/${ENV.COURSE_ID}/pages/${item.page_id}/edit" target="_blank">${item.title}</a></h3>${log}`)
}
}
function UUsjekk(){
myReport.innerHTML = ''
addToReport(`<h1>${title} ${content.data.course.name}</h1>`)
let modules = content.data.course.modulesConnection.nodes
if (modules.length){
//Course with modules
Get(`courses/${ENV.COURSE_ID}`)
.then( coursedata => {
if (coursedata.default_view == 'wiki'){
//has front page which might not show up in modules
let pages = Get(`/courses/${ENV.COURSE_ID}/pages?per_page=100`)
for (const page of pages){
if (page.front_page){
addToReport(`<h2>${frontpage}:</h2>`)
console.log ("frontpage")
pagecheck(page)
break
}
}
}
for (const module of modules){
addToReport(`<h2>${moduleword} ${module.name}</h2>`)
for (const item of module.moduleItems){
if (!item.content._id) continue
if (item.content.description) { assignmentcheck(item.content); continue}
if (item.content.message) { discussioncheck(item.content); continue}
pagecheck(item.content)
}
}
})
addToReport(summary.use(errorcount,warningcount))
}else{
//Course without modules
console.log('kurs uten moduler')
Get(`/courses/${ENV.COURSE_ID}/pages?per_page=100`)
.then(pages => {
console.log (pages.length)
if (pages.length){
for (const page of pages){
pagecheck(page)
}
addToReport(summary.use(errorcount,warningcount))
}else{
addToReport(`<p>${nothing}</p>`)
}
})
}


}
return {
addUUButton: function (){
graphql(`query MyQuery {course(id: "${ENV.COURSE_ID}") {name modulesConnection {nodes {name moduleItems {content {... on Page {_id} ... on Assignment {_id name description state} ... on Discussion {_id message title}}}}}}}`)
.then(res => res.json())
.then(res => {
let a = document.createElement('a');
document.querySelector("#right-side table.summary").before(a);
a.outerHTML = `<a class='Button Button--link Button--link--has-divider Button--course-settings' id="miniuusjekk"><i class='icon-off' /> ${buttonname}</a>`;
content = res
document.getElementById('miniuusjekk').onclick = UUsjekk

}
)
}
}
})();