/*eslint no-var: "error"*/ /*eslint no-console: "off"*/ /*eslint-disable no-trailing-spaces */ /*eslint-env es6*/ // Put this file in path/to/plugin/amd/src import LeaderLine from './leaderline'; import {get_strings} from 'core/str'; import {load_strings} from './string-helper'; import {call} from 'core/ajax'; import notification from 'core/notification'; import {svgarcpath} from './svgarc'; //import {fixLineWrappers} from './studyplan-processor'; import Debugger from './debugger'; // Make π available as a constant const π = Math.PI; export default { install(Vue/*,options*/){ let debug = new Debugger("treestudyplan-viewer"); debug.enable(); let strings = load_strings({ invalid: { error: 'error', }, grading: { ungraded: "ungraded", graded: "graded", allgraded: "allgraded", unsubmitted: "unsubmitted", nogrades: "nogrades", unknown: "unknown", }, completion: { completed: "completion_completed", incomplete: "completion_incomplete", completed_pass: "completion_passed", completed_fail: "completion_failed", ungraded: "ungraded", aggregation_all: "aggregation_all", aggregation_any: "aggregation_any", aggregation_overall_all: "aggregation_overall_all", aggregation_overall_any: "aggregation_overall_any", completion_not_configured: "completion_not_configured", configure_completion: "configure_completion", }, badge: { share_badge: "share_badge", dateissued: "dateissued", dateexpire: "dateexpire", badgeinfo: "badgeinfo", badgeissuedstats: "badgeissuedstats", }, course: { completion_incomplete: "completion_incomplete", completion_failed: "completion_failed", completion_pending: "completion_pending", completion_progress: "completion_progress", completion_completed: "completion_completed", completion_good: "completion_good", completion_excellent: "completion_excellent", view_feedback: "view_feedback", coursetiming_past: "coursetiming_past", coursetiming_present: "coursetiming_present", coursetiming_future: "coursetiming_future", required_goal: "required_goal", }, }); /************************************ * * * Treestudyplan Viewer components * * * ************************************/ /** * Check if element is visible * @param {Object} elem The element to check * @returns {boolean} True if visible */ function isVisible(elem){ return !!( elem.offsetWidth || elem.offsetHeight || elem.getClientRects().length ); } Vue.component('r-progress-circle',{ props: { value: { type: Number, }, max: { type: Number, default: 100, }, min: { type: Number, default: 0, }, stroke: { type: Number, default: 0.2, }, bgopacity: { type: Number, default: 0.2, }, title: { type: String, default: "", }, }, data() { return { selectedstudyplan: null, }; }, computed: { range() { return this.max - this.min; }, fraction(){ if(this.max - this.min == 0){ return 0; // 0 size is always empty :) } else { return (this.value - this.min)/(this.max - this.min); } }, radius() { return 50 - (50*this.stroke); }, arcpath() { let fraction = 0; const r = 50 - (50*this.stroke); if(this.max - this.min != 0){ fraction = (this.value - this.min)/(this.max - this.min); } const Δ = fraction * 2*π; return svgarcpath([50,50],[r,r],[0,Δ], 1.5*π); }, }, methods: { }, template: ` {{title}} `, }); Vue.component('r-report', { props: { value: { type: Array, }, guestmode: { type: Boolean, default: false, }, teachermode: { type: Boolean, default: false, } }, data() { return { selectedstudyplan: null, }; }, computed: { displayedstudyplan(){ if(this.selectedstudyplan){ return this.selectedstudyplan; } else if(this.value && this.value.length > 0){ return this.value[0]; } else { return null; } } }, methods: { }, template: `
{{studyplan.name}}
`, }); Vue.component('r-studyplan', { props: { value: { type: Object, }, guestmode: { type: Boolean, default: false, }, teachermode: { type: Boolean, default: false, } }, data() { return { }; }, updated(){ this.$root.$emit('redrawLines'); }, mounted(){ this.$root.$emit('redrawLines'); }, computed: { }, methods: { }, template: `
`, }); /* * R-STUDYLINE */ Vue.component('r-studyline', { props: ['color','name','code', 'slots','sequence','numlines','guestmode','teachermode'], data() { return { }; }, computed: { }, methods: { }, template: `
{{ code }}
`, }); Vue.component('r-studyline-slot', { props: { type : { type: String, default: 'competency', }, slotindex : { type: Number, default: 0, }, lineid : { type: Number, default: 0, }, value: { type: Array, default(){ return [];}, }, plan: { type: Object, default(){ return null;} }, guestmode: { type: Boolean, default: false, }, teachermode: { type: Boolean, default: false, } }, computed: { sorted(){ let copy = [...this.value]; copy.sort(function(a,b){ return a.layer - b.layer; }); return copy; } }, data() { return { }; }, methods: { }, template: `
`, }); Vue.component('r-item', { props: { value :{ type: Object, default: function(){ return null;}, }, plan: { type: Object, default(){ return null;} }, guestmode: { type: Boolean, default: false, }, teachermode: { type: Boolean, default: false, } }, data() { return { lines: [], }; }, methods: { lineColor(){ if(this.teachermode){ return "#aaa"; } else{ switch(this.value.completion){ default: // "incomplete" return "#777"; case "failed": return "#933"; case "progress": return "#da3"; case "completed": return "#383"; case "good": return "#398"; case "excellent": return "#36f"; } } }, redrawLine(conn){ let lineColor = this.lineColor(); // prepare lineinfo link or delete old line let lineinfo = this.lines[conn.to_id]; if(lineinfo){ if(lineinfo.line){ if(lineinfo.lineElm ){ lineinfo.lineElm.parentNode.removeChild(lineinfo.lineElm); lineinfo.lineElm = undefined; } else { lineinfo.line.remove(); } lineinfo.line = undefined; } } else { lineinfo = {}; this.lines[conn.to_id] = lineinfo; } // draw new line... let start = document.getElementById('studyitem-'+conn.from_id); let end= document.getElementById('studyitem-'+conn.to_id); LeaderLine.positionByWindowResize = false; if(start !== null && end !== null && isVisible(start) && isVisible(end)){ lineinfo.line = new LeaderLine(start,end,{ color: lineColor, startSocket: 'right', endSocket: 'left', startSocketGravity: 75, endSocketGravity: 75, }); let elmWrapper = (this.plan.id >=0)?document.getElementById('studyplan-linewrapper-'+this.plan.id):null; if(elmWrapper !== null){ let elmLine = document.querySelector('body > .leader-line:last-child'); elmWrapper.appendChild(elmLine); lineinfo.lineElm = elmLine; // store line element so it can more easily be removed from the dom } setTimeout(function(){ if(lineinfo.line){ lineinfo.line.position(); } },1); } }, redrawLines(){ for(let i in this.value.connections.out){ let conn = this.value.connections.out[i]; this.redrawLine(conn); } }, onWindowResize(){ this.redrawLines(); } }, computed: { hasConnectionsOut() { return !(["finish",].includes(this.value.type)); }, hasConnectionsIn() { return !(["start",].includes(this.value.type)); }, hasContext() { return ['start','junction','finish'].includes(this.value.type); } }, created(){ }, mounted(){ // Initialize connection lines when mounting this.redrawLines(); setTimeout(()=>{ this.redrawLines(); },50); // Add resize event listener window.addEventListener('resize',this.onWindowResize); }, beforeDestroy(){ for(let i in this.value.connections.out){ let conn = this.value.connections.out[i]; let lineinfo = this.lines[conn.to_id]; if(lineinfo){ if(lineinfo.line){ if(lineinfo.lineElm ){ lineinfo.lineElm.parentNode.removeChild(lineinfo.lineElm); lineinfo.lineElm = undefined; } else { lineinfo.line.remove(); } lineinfo.line = undefined; } } } // Remove resize event listener window.removeEventListener('resize',this.onWindowResize); }, beforeUpdate(){ }, updated(){ if(!this.dummy) { this.redrawLines(); } }, template: `
`, }); Vue.component('r-item-invalid', { props: { 'value' :{ type: Object, default: function(){ return null;}, }, }, data() { return { text: strings.invalid, }; }, methods: { }, template: ` {{ text.error }} `, }); //TAG: Item Course Vue.component('r-item-course', { props: { value :{ type: Object, default(){ return null;}, }, guestmode: { type: Boolean, default(){ return false;} }, teachermode: { type: Boolean, default(){ return false;} }, plan: { type: Object, default(){ return null;} } }, data() { return { text: strings.course, }; }, computed: { }, created(){ const self = this; // Get text strings for condition settings let stringkeys = []; for(const key in this.text){ stringkeys.push({ key: key, component: 'local_treestudyplan'}); } get_strings(stringkeys).then(function(strings){ let i = 0; for(const key in self.text){ self.text[key] = strings[i]; i++; } }); }, methods: { completion_icon(completion) { switch(completion){ default: // case "incomplete" return "circle-o"; case "pending": return "question-circle"; case "failed": return "times-circle"; case "progress": return "exclamation-circle"; case "completed": return "check-circle"; case "good": return "check-circle"; case "excellent": return "check-circle"; } }, }, template: ` {{ value.course.displayname }} `, }); //TAG: Selected activities dispaly Vue.component('r-item-studentgrades',{ props: { value : { type: Object, default: function(){ return {};}, }, guestmode: { type: Boolean, default: false, }, }, data() { return { text: strings.course, }; }, computed: { pendingsubmission(){ let result = false; for(const ix in this.value.course.grades){ const g = this.value.course.grades[ix]; if(g.pendingsubmission){ result = true; break; } } return result; }, useRequiredGrades() { if(this.plan && this.plan.aggregation_info && this.plan.aggregation_info.useRequiredGrades !== undefined){ return this.plan.aggregation_info.useRequiredGrades; } else { return false; } }, }, methods: { completion_icon(completion) { switch(completion){ default: // case "incomplete" return "circle-o"; case "pending": return "question-circle"; case "failed": return "times-circle"; case "progress": return "exclamation-circle"; case "completed": return "check-circle"; case "good": return "check-circle"; case "excellent": return "check-circle"; } }, }, template: `
{{g.name}} {{g.grade}} {{ text["view_feedback"]}}
`, }); //TAG: Core completion version of student course info Vue.component('r-item-studentcompletion',{ props: { value : { type: Object, default: function(){ return {};}, }, guestmode: { type: Boolean, default: false, }, course: { type: Object, default: function(){ return {};}, }, }, data() { return { text: { completion_incomplete: "completion_incomplete", completion_failed: "completion_failed", completion_pending: "completion_pending", completion_progress: "completion_progress", completion_completed: "completion_completed", completion_good: "completion_good", completion_excellent: "completion_excellent", view_feedback: "view_feedback", coursetiming_past: "coursetiming_past", coursetiming_present: "coursetiming_present", coursetiming_future: "coursetiming_future", required_goal: "required_goal", }, }; }, created(){ const self = this; // Get text strings for condition settings let stringkeys = []; for(const key in this.text){ stringkeys.push({ key: key, component: 'local_treestudyplan'}); } get_strings(stringkeys).then(function(strings){ let i = 0; for(const key in self.text){ self.text[key] = strings[i]; i++; } }); }, computed: { }, methods: { completion_icon(completion) { switch(completion){ case "progress": return "exclamation-circle"; case "complete": return "check-circle"; case "complete-pass": return "check-circle"; case "complete-fail": return "times-circle"; default: // case "incomplete" return "circle-o"; } }, completion_tag(cgroup){ return cgroup.completion?'completed':'incomplete'; } }, template: `
`, }); //TAG: Teacher course Vue.component('r-item-teachercourse', { props: { value :{ type: Object, default(){ return null;} }, guestmode: { type: Boolean, default(){ return false;} }, teachermode: { type: Boolean, default(){ return false;} }, plan: { type: Object, default(){ return null;} } }, data() { return { text: { select_conditions: "select_conditions", select_grades: "select_grades", coursetiming_past: "coursetiming_past", coursetiming_present: "coursetiming_present", coursetiming_future: "coursetiming_future", grade_include: "grade_include", grade_require: "grade_require", required_goal: "required_goal", }, txt: { grading: strings.grading, } }; }, computed: { course_grading_needed(){ return this.course_grading_state(); }, course_grading_icon(){ return this.determine_grading_icon(this.course_grading_state()); }, filtered_grades(){ return this.value.course.grades.filter(g => g.selected); }, useRequiredGrades() { if(this.plan && this.plan.aggregation_info && this.plan.aggregation_info.useRequiredGrades !== undefined){ return this.plan.aggregation_info.useRequiredGrades; } else { return false; } }, isCompletable() { let completable = false; if(this.value.course.completion){ if(this.value.course.completion.conditions.length > 0){ completable = true; } } else if (this.value.course.grades){ if(this.value.course.grades.length > 0){ completable = true; } } return completable; }, progress_circle() { //INFO: const status = { students: 0, completed: 0, completed_pass: 0, completed_fail: 0, ungraded: 0, }; if(this.value.course.completion){ for(const cond of this.value.course.completion.conditions){ for(const itm of cond.items){ if(itm.progress){ status.students += itm.progress.students; status.completed += itm.progress.completed; status.completed_pass += itm.progress.completed_pass; status.completed_fail += itm.progress.completed_fail; status.ungraded += itm.progress.completed; } } } } else if (this.value.course.grades){ for( const g of this.value.course.grades){ if(g.grading){ status.students += g.grading.students; status.completed += g.grading.completed; status.completed_pass += g.grading.completed_pass; status.completed_fail += g.grading.completed_fail; status.ungraded += g.grading.ungraded; } } } return status; } }, created(){ const self = this; // Get text strings for condition settings let stringkeys = []; for(const key in this.text){ stringkeys.push({ key: key, component: 'local_treestudyplan'}); } get_strings(stringkeys).then(function(strings){ let i = 0; for(const key in self.text){ self.text[key] = strings[i]; i++; } }); }, methods: { course_grading_state(){ let ungraded = 0; let unknown = 0; let graded = 0; let allgraded = 0; const grades = this.filtered_grades; if(!Array.isArray(grades) || grades == 0){ return 'nogrades'; } for(const ix in grades){ const grade = grades[ix]; if(grade.grading){ if(Number(grade.grading.ungraded) > 0){ ungraded++; } else if(Number(grade.grading.graded) > 0) { if(Number(grade.grading.graded) == Number(grade.grading.students)){ allgraded++; } else { graded++; } } } else { unknown = true; } } if(ungraded > 0){ return 'ungraded'; } else if(unknown) { return 'unknown'; } else if(graded){ return 'graded'; } else if(allgraded){ return 'allgraded'; } else { return 'unsubmitted'; } }, determine_grading_icon(gradingstate){ switch(gradingstate){ default: // "nogrades": return "circle-o"; case "ungraded": return "exclamation-circle"; case "unknown": return "question-circle-o"; case "graded": return "check"; case "allgraded": return "check"; case "unsubmitted": return "dot-circle-o"; } }, }, template: ` {{ value.course.displayname }} `, }); //TAG: Select activities to use in grade overview Vue.component('r-item-teacher-gradepicker', { props: { value : { type: Object, default: function(){ return {};}, }, useRequiredGrades: { type: Boolean, default(){ return null;} } }, data() { return { }; }, computed: { }, methods: { }, template: ` `, }); //TAG: Selected activities dispaly Vue.component('r-item-teachergrades',{ props: { value : { type: Object, default: function(){ return {};}, }, useRequiredGrades: { type: Boolean, default: false, }, }, data() { return { txt: { grading: strings.grading, } }; }, computed: { pendingsubmission(){ let result = false; for(const ix in this.value.grades){ const g = this.value.grades[ix]; if(g.pendingsubmission){ result = true; break; } } return result; }, filtered_grades(){ return this.value.grades.filter(g => g.selected); }, }, methods: { determine_grading_icon(gradingstate){ switch(gradingstate){ default: // "nogrades": return "circle-o"; case "ungraded": return "exclamation-circle"; case "unknown": return "question-circle-o"; case "graded": return "check"; case "allgraded": return "check"; case "unsubmitted": return "dot-circle-o"; } }, grading_icon(grade){ return this.determine_grading_icon(this.is_grading_needed(grade)); }, is_grading_needed(grade){ debug.info("Grade: ", grade.name); debug.info(grade.grading); if(grade.grading){ debug.info("Ping"); if(grade.grading.ungraded){ return 'ungraded'; } else if(grade.grading.completed_pass || grade.grading.completed || grade.grading.completed_fail) { if(Number(grade.grading.completed) + Number(grade.grading.completed_pass) + Number(grade.grading.completed_fail) == Number(grade.grading.students)){ return 'allgraded'; } else { return 'graded'; } } else { return 'unsubmitted'; } } else { return 'unknown'; } }, includeChanged(newValue,g){ call([{ methodname: 'local_treestudyplan_include_grade', args: { 'grade_id': g.id, 'item_id': this.value.id, 'include': newValue, 'required': g.required, } }])[0].fail(notification.exception); }, requiredChanged(newValue,g){ call([{ methodname: 'local_treestudyplan_include_grade', args: { 'grade_id': g.id, 'item_id': this.value.id, 'include': g.selected, 'required': newValue, } }])[0].fail(notification.exception); }, }, template: `
{{g.name}}
`, }); //TODO: Core completion version of student course info Vue.component('r-item-teachercompletion',{ props: { value : { type: Object, default: function(){ return {};}, }, guestmode: { type: Boolean, default: false, }, course: { type: Object, default: function(){ return {};}, }, }, data() { return { text: strings.completion, }; }, created(){ const self = this; // Get text strings for condition settings let stringkeys = []; for(const key in this.text){ stringkeys.push({ key: key, component: 'local_treestudyplan'}); } get_strings(stringkeys).then(function(strings){ let i = 0; for(const key in self.text){ self.text[key] = strings[i]; i++; } }); }, computed: { }, methods: { hasCompletions() { if(this.value.conditions) { for(const cgroup of this.value.conditions){ if(cgroup.items && cgroup.items.length > 0){ return true; } } } return false; }, }, template: `
{{ text.aggregation_overall_all}}{{ text.aggregation_overall_any}}
{{text.completion_not_configured}}!
{{text.configure_completion}}
`, }); Vue.component('r-grading-bar',{ props: { value : { type: Object, default: function(){ return {};}, }, width: { type: Number, default: 150, }, height: { type: Number, default: 15, } }, data() { return { text: strings.grading, }; }, computed: { width_unsubmitted() { return this.width * this.fraction_unsubmitted(); }, width_graded() { return this.width * this.fraction_graded(); }, width_ungraded() { return this.width * this.fraction_ungraded(); }, count_unsubmitted(){ return (this.value.students - this.value.graded - this.value.ungraded); } }, methods: { fraction_unsubmitted() { if(this.value.students > 0){ return 1 - ((this.value.graded + this.value.ungraded) / this.value.students); } else { return 1; } }, fraction_graded() { if(this.value.students > 0){ return this.value.graded / this.value.students; } else { return 0; } }, fraction_ungraded() { if(this.value.students > 0){ return this.value.ungraded / this.value.students; } else { return 0; } }, }, template: ` `, }); Vue.component('r-completion-bar',{ props: { value : { type: Object, default: function(){ return { students: 0, completed: 0, completed_pass: 0, completed_fail: 0, ungraded: 0, };}, }, width: { type: Number, default: 150, }, height: { type: Number, default: 15, } }, data() { return { text: strings.completion, }; }, computed: { width_incomplete() { return this.width * this.fraction_incomplete(); }, width_completed() { return this.width * this.fraction_completed(); }, width_completed_pass() { return this.width * this.fraction_completed_pass(); }, width_completed_fail() { return this.width * this.fraction_completed_fail(); }, width_ungraded() { return this.width * this.fraction_ungraded(); }, count_incomplete(){ return (this.value.students - this.value.completed - this.value.completed_pass - this.value.completed_fail - this.value.ungraded); } }, methods: { fraction_incomplete() { if(this.value.students > 0){ return 1 - ( (this.value.completed + this.value.completed_pass + this.value.completed_fail + this.value.ungraded) / this.value.students); } else { return 1; } }, fraction_completed() { if(this.value.students > 0){ return this.value.completed / this.value.students; } else { return 0; } }, fraction_completed_pass() { if(this.value.students > 0){ return this.value.completed_pass / this.value.students; } else { return 0; } }, fraction_completed_fail() { if(this.value.students > 0){ return this.value.completed_fail / this.value.students; } else { return 0; } }, fraction_ungraded() { if(this.value.students > 0){ return this.value.ungraded / this.value.students; } else { return 0; } }, }, template: ` `, }); Vue.component('r-completion-circle',{ props: { value : { type: Object, default: function(){ return { students: 10, completed: 2, completed_pass: 2, completed_fail: 2, ungraded: 2, };}, }, stroke: { type: Number, default: 0.2, }, disabled: { type: Boolean, default: false, }, title: { type: String, default: "", } }, computed: { radius() { return 50 - (50*this.stroke); }, arcpath_ungraded() { const begin = 0; return this.arcpath(begin,this.fraction_ungraded()); }, arcpath_completed() { const begin = this.fraction_ungraded(); return this.arcpath(begin,this.fraction_completed()); }, arcpath_completed_pass() { const begin = this.fraction_ungraded() + this.fraction_completed(); return this.arcpath(begin,this.fraction_completed_pass()); }, arcpath_completed_fail() { const begin = this.fraction_ungraded() + this.fraction_completed() + this.fraction_completed_pass(); return this.arcpath(begin,this.fraction_completed_fail()); }, arcpath_incomplete() { const begin = this.fraction_ungraded() + this.fraction_completed() + this.fraction_completed_pass() + this.fraction_completed_fail(); return this.arcpath(begin,this.fraction_incomplete()); }, }, methods: { arcpath(start, end) { const r = 50 - (50*this.stroke); const t1 = start * 2*π; const Δ = end * 2*π; return svgarcpath([50,50],[r,r],[t1,Δ], 1.5*π); }, fraction_incomplete() { if(this.value.students > 0){ return 1 - ( (this.value.completed + this.value.completed_pass + this.value.completed_fail + this.value.ungraded) / this.value.students); } else { return 1; } }, fraction_completed() { if(this.value.students > 0){ return this.value.completed / this.value.students; } else { return 0; } }, fraction_completed_pass() { if(this.value.students > 0){ return this.value.completed_pass / this.value.students; } else { return 0; } }, fraction_completed_fail() { if(this.value.students > 0){ return this.value.completed_fail / this.value.students; } else { return 0; } }, fraction_ungraded() { if(this.value.students > 0){ return this.value.ungraded / this.value.students; } else { return 0; } }, }, template: ` {{title}} `, }); Vue.component('r-item-junction',{ props: { value : { type: Object, default: function(){ return {};}, }, guestmode: { type: Boolean, default: false, }, teachermode: { type: Boolean, default: false, } }, data() { return { }; }, computed: { completion(){ if(this.value.completion){ return this.value.completion; } else { return "incomplete"; } } }, methods: { }, template: `
`, }); Vue.component('r-item-finish',{ props: { value : { type: Object, default: function(){ return {};}, }, guestmode: { type: Boolean, default: false, }, teachermode: { type: Boolean, default: false, } }, data() { return { }; }, computed: { completion(){ if(this.value.completion){ return this.value.completion; } else { return "incomplete"; } } }, methods: { }, template: `
`, }); Vue.component('r-item-start',{ props: { value : { type: Object, default: function(){ return {};}, }, guestmode: { type: Boolean, default: false, }, teachermode: { type: Boolean, default: false, } }, data() { return { }; }, computed: { completion(){ if(this.value.completion){ return this.value.completion; } else { return "incomplete"; } } }, created(){ }, methods: { }, template: `
`, }); Vue.component('r-item-badge',{ props: { value : { type: Object, default: function(){ return {};}, }, guestmode: { type: Boolean, default: false, }, teachermode: { type: Boolean, default: false, } }, data() { return { txt: strings }; }, computed: { completion() { return this.value.badge.issued?"completed":"incomplete"; }, issued_icon(){ switch(this.value.badge.issued){ default: // "nogrades": return "circle-o"; case true: return "check"; } }, issuestats(){ // so the r-completion-bar can be used to show issuing stats return { students: (this.value.badge.studentcount)?this.value.badge.studentcount:0, completed: (this.value.badge.issuedcount)?this.value.badge.issuedcount:0, completed_pass: 0, completed_fail: 0, ungraded: 0, }; }, arcpath_issued(){ if(this.value.badge.studentcount){ const fraction = this.value.badge.issuedcount/this.value.badge.studentcount; return this.arcpath(0,fraction); } else { return ""; // no path } } }, methods: { arcpath(start, end) { const r = 44; const t1 = start * 2*π; const Δ = end * 2*π; return svgarcpath([50,50],[r,r],[t1,Δ], 1.5*π); }, }, template: `
{{value.badge.name}}

{{value.badge.description}}

  • {{txt.badge.dateissued}}: {{ value.badge.dateissued }}
  • {{txt.badge.dateexpired}}: {{ value.badge.dateexpired }}
  • {{txt.badge.share_badge}}

{{ txt.badge.badgeinfo }}

{{txt.badge.badgeissuedstats}}:

`, }); }, };