diff --git a/classes/output/mobile.php b/classes/output/mobile.php new file mode 100644 index 0000000..82325c0 --- /dev/null +++ b/classes/output/mobile.php @@ -0,0 +1,55 @@ +. + +/** + * Mobile output class for qtype_aitext + * + * @package qtype_aitext + * @copyright 2018 Marcus Green + * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later + */ + +namespace qtype_aitext\output; + +/** + * Mobile output class for aitext question type + * + * @package qtype_aitext + * @copyright 2018 Marcus Green + * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later + */ +class mobile { + + /** + * Returns the aite question type for the quiz in the mobile app. + * @param array $args + * @return array + */ + public static function mobile_get_aitext($args) { + global $CFG; + $args = (object) $args; + $templatepath = $CFG->dirroot."/question/type/aitext/mobile/qtype_aitext.html"; + return [ + 'templates' => [ + [ + 'id' => 'main', + 'html' => file_get_contents($templatepath) + ] + ], + 'javascript' => file_get_contents($CFG->dirroot . '/question/type/aitext/mobile/mobile.js') + ]; + } +} \ No newline at end of file diff --git a/db/mobile.php b/db/mobile.php new file mode 100644 index 0000000..a7c2903 --- /dev/null +++ b/db/mobile.php @@ -0,0 +1,53 @@ +. + +/** + * Mobile app areas for AI Text + * + * Documentation: {@link https://moodledev.io/general/app/development/plugins-development-guide} + * + * @package qtype_aitext + * @copyright 2024 Marcus Green + * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later + */ + +defined('MOODLE_INTERNAL') || die(); + +$addons = [ + "qtype_aitext" => [ + "handlers" => [ // Different places where the add-on will display content. + 'aitext' => [ // Handler unique name (can be anything). + 'displaydata' => [ + 'title' => 'AIText question', + 'icon' => '/question/type/aitext/pix/icon.gif', + 'class' => '', + ], + 'delegate' => 'CoreQuestionDelegate', // Delegate (where to display the link to the add-on). + 'method' => 'mobile_get_aitext', + 'offlinefunctions' => [ + 'mobile_get_aitext' => [], // Function in classes/output/mobile.php. + ], // Function needs caching for offline. + 'styles' => [ + 'url' => $CFG->wwwroot.'/question/type/aitext/mobile/qtype_aitext_app.css', + 'version' => '0.1' + ] + ] + ], + 'lang' => [ + ['pluginname', 'qtype_aitext'], // Matching value in lang/en/qtype_gapfill. + ], + ] +]; \ No newline at end of file diff --git a/mobile/mobile.js b/mobile/mobile.js new file mode 100644 index 0000000..8ef5209 --- /dev/null +++ b/mobile/mobile.js @@ -0,0 +1,153 @@ +// This file is part of Moodle - http://moodle.org/ +// +// Moodle is free software: you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// Moodle is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with Moodle. If not, see . + +/** + * PHP calls this from within + * classes/output/mobile.php + */ +/* eslint-disable no-console */ +/* eslint-env es6 */ +var that = this; +var result = { + componentInit: function() { + + // Check that "this.question" was provided. + if (! this.question) { + return that.CoreQuestionHelperProvider.showComponentError(that.onAbort); + } + + // Create a temporary div to ease extraction of parts of the provided html. + var div = document.createElement('div'); + div.innerHTML = this.question.html; + + // Replace Moodle's correct/incorrect classes, feedback and icons with mobile versions. + that.CoreQuestionHelperProvider.replaceCorrectnessClasses(div); + that.CoreQuestionHelperProvider.replaceFeedbackClasses(div); + that.CoreQuestionHelperProvider.treatCorrectnessIcons(div); + + // Get useful parts of the data provided in the question's html. + var text = div.querySelector('.qtext'); + if (text) { + this.question.text = text.innerHTML; + } + + var textarea = div.querySelector('.answer textarea'); + if (textarea === null) { + // review or check + textarea = div.querySelector('.answer .qtype_aitext_response'); + } + if (textarea) { + textarea.style.borderRadius = '4px'; + textarea.style.padding = '6px 12px'; + if (textarea.matches('.readonly')) { + textarea.style.border = '2px #b8dce2 solid'; // light blue + textarea.style.backgroundColor = '#e7f3f5'; // lighter blue + } else { + textarea.style.backgroundColor = '#edf6f7'; // lightest blue + } + this.question.textarea = textarea.outerHTML; + } + + var itemcount = div.querySelector('.itemcount'); + if (itemcount) { + + // Replace bootstrap styles with inline styles because + // adding styles to 'mobile/styles_app.css' doesn't seem to be effective :-( + that.replaceBootstrapClasses(itemcount); + + itemcount.querySelectorAll('p').forEach(function(p){ + that.replaceBootstrapClasses(p); + }); + + // Fix background and text color on "wordswarning" span. + var warning = itemcount.querySelector(".warning"); + if (warning) { + that.replaceBootstrapClasses(warning); + } + + this.question.itemcount = itemcount.outerHTML; + } + + /** + * questionRendered + */ + this.questionRendered = function(){ + + var textarea = this.componentContainer.querySelector('textarea'); + var itemcount = this.componentContainer.querySelector('.itemcount'); + if (textarea && itemcount) { + + // Maybe "this.CoreLangProvider" has a method for fetching a string + // but I can't find it, so we use our own method, thus: + var minwordswarning = that.getPluginString("qtype_aitext", "minwordswarning"); + var maxwordswarning = that.getPluginString("qtype_aitext", "maxwordswarning"); + + var countitems = itemcount.querySelector(".countitems"); + var value = countitems.querySelector(".value"); + var warning = countitems.querySelector(".warning"); + + var itemtype = itemcount.dataset.itemtype; + var minitems = parseInt(itemcount.dataset.minitems); + var maxitems = parseInt(itemcount.dataset.maxitems); + + var itemsplit = ''; + switch (itemtype) { + case "chars": itemsplit = ""; break; + case "words": itemsplit = "[\\sā€”ā€“]+"; break; + case "sentences": itemsplit = "[\\.?!]+"; break; + case "paragraphs": itemsplit = "[\\r\\n]+"; break; + } + + if (itemsplit) { + itemsplit = new RegExp(itemsplit); + textarea.addEventListener("keyup", function() { + var text = textarea.value; + var warningtext = ""; + var count = 0; + if (text) { + count = text.split(itemsplit).filter(function(item) { + return (item !== ""); + }).length; + if (minitems && (count < minitems)) { + warningtext = minwordswarning; + } + if (maxitems && (count > maxitems)) { + warningtext = maxwordswarning; + } + } + value.innerText = count; + if (warning) { + warning.innerText = warningtext; + if (warningtext == "") { + warning.style.display = "none"; + } else { + warning.style.display = "inline"; + } + } + }); + } + } + }; + + if (text && textarea) { + return true; + } + + // Oops, the expected elements, text and textarea, were not found !! + return that.CoreQuestionHelperProvider.showComponentError(that.onAbort); + } +}; +/* eslint-disable-next-line */ +result; \ No newline at end of file diff --git a/mobile/qtype_aitext.html b/mobile/qtype_aitext.html new file mode 100644 index 0000000..48664e8 --- /dev/null +++ b/mobile/qtype_aitext.html @@ -0,0 +1,16 @@ +
+ +

+ +

+
+ + + + + + + \ No newline at end of file diff --git a/mobile/qtype_aitext_app.css b/mobile/qtype_aitext_app.css new file mode 100644 index 0000000..4c02250 --- /dev/null +++ b/mobile/qtype_aitext_app.css @@ -0,0 +1,9 @@ +.qtype-aitext .countwords { + margin-top: 0.5rem; /* mt-2 */ + margin-bottom: 0; +} +.qtype-aitext .minwords, +.qtype-aitext .minwords { + margin-top: 0; + margin-bottom: 0; +} \ No newline at end of file