666 lines
24 KiB
PHP
666 lines
24 KiB
PHP
<?php
|
|
// This file is part of the Studyplan plugin for Moodle
|
|
//
|
|
// 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 <https://www.gnu.org/licenses/>.
|
|
/**
|
|
* Model class for study items
|
|
* @package local_treestudyplan
|
|
* @copyright 2023 P.M. Kuipers
|
|
* @license https://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
*/
|
|
|
|
namespace local_treestudyplan;
|
|
defined('MOODLE_INTERNAL') || die();
|
|
|
|
require_once($CFG->libdir.'/externallib.php');
|
|
|
|
/**
|
|
* Model class for study items
|
|
*/
|
|
class studyitem {
|
|
|
|
/** @var string */
|
|
public const COURSE = 'course';
|
|
/** @var string */
|
|
public const JUNCTION = 'junction';
|
|
/** @var string */
|
|
public const BADGE = 'badge';
|
|
/** @var string */
|
|
public const FINISH = 'finish';
|
|
/** @var string */
|
|
public const START = 'start';
|
|
/** @var string */
|
|
public const INVALID = 'invalid';
|
|
|
|
/** @var string */
|
|
public const TABLE = "local_treestudyplan_item";
|
|
|
|
/**
|
|
* Cache retrieved studyitems in this session
|
|
* @var array */
|
|
private static $cache = [];
|
|
/**
|
|
* Holds database record
|
|
* @var stdClass
|
|
*/
|
|
private $r;
|
|
/** @var int */
|
|
private $id;
|
|
|
|
/** @var courseinfo */
|
|
private $courseinfo = null;
|
|
/** @var studyline */
|
|
private $studyline;
|
|
/** @var aggregator */
|
|
private $aggregator;
|
|
|
|
|
|
/**
|
|
* Return the context the studyplan is associated to
|
|
*/
|
|
public function context(): \context {
|
|
return $this->studyline->context();
|
|
}
|
|
|
|
/**
|
|
* Return the studyline for this item
|
|
*/
|
|
public function studyline(): studyline {
|
|
return $this->studyline;
|
|
}
|
|
|
|
/**
|
|
* Return the condition string for this item
|
|
*/
|
|
public function conditions() : string {
|
|
if($this->r->type == self::COURSE) {
|
|
return (!empty($this->r->conditions)) ? $this->r->conditions : "";
|
|
} else {
|
|
return (!empty($this->r->conditions)) ? $this->r->conditions : "ALL";
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Find record in database and return management object
|
|
* @param int $id Id of database record
|
|
*/
|
|
public static function find_by_id($id): self {
|
|
if (!array_key_exists($id, self::$cache)) {
|
|
self::$cache[$id] = new self($id);
|
|
}
|
|
return self::$cache[$id];
|
|
}
|
|
|
|
/**
|
|
* Construct a new model based on study item id
|
|
* @param int $id Study item id
|
|
*/
|
|
public function __construct($id) {
|
|
global $DB;
|
|
$this->id = $id;
|
|
$this->r = $DB->get_record(self::TABLE, ['id' => $id], "*", MUST_EXIST);
|
|
|
|
$this->studyline = studyline::find_by_id($this->r->line_id);
|
|
$this->aggregator = $this->studyline()->studyplan()->aggregator();
|
|
}
|
|
|
|
/**
|
|
* Return database identifier
|
|
*/
|
|
public function id() : int {
|
|
return $this->id;
|
|
}
|
|
|
|
/**
|
|
* Return period slot for this item
|
|
*/
|
|
public function slot() : int {
|
|
return $this->r->slot;
|
|
}
|
|
|
|
/**
|
|
* Return layer (order within a line and slot) for this item
|
|
*/
|
|
public function layer() : int {
|
|
return $this->r->layer;
|
|
}
|
|
|
|
/**
|
|
* Return study item type (see constants above)
|
|
*/
|
|
public function type() : string {
|
|
return $this->r->type;
|
|
}
|
|
|
|
/**
|
|
* Return period span
|
|
*/
|
|
public function span() : string {
|
|
return $this->r->span;
|
|
}
|
|
|
|
/**
|
|
* Return id of linked course (only relevant on COURSE types) or 0 if none
|
|
*/
|
|
public function courseid() : int {
|
|
return $this->r->course_id;
|
|
}
|
|
|
|
/**
|
|
* Check if a studyitem with the given id exists
|
|
* @param int $id Id of studyplan
|
|
*/
|
|
public static function exists($id) : bool {
|
|
global $DB;
|
|
return is_numeric($id) && $DB->record_exists(self::TABLE, array('id' => $id));
|
|
}
|
|
|
|
/**
|
|
* Webservice structure for editor info
|
|
* @param int $value Webservice requirement constant
|
|
*/
|
|
public static function editor_structure($value = VALUE_REQUIRED) : \external_description {
|
|
return new \external_single_structure([
|
|
"id" => new \external_value(PARAM_INT, 'id of study item'),
|
|
"type" => new \external_value(PARAM_TEXT, 'shortname of study item'),
|
|
"conditions" => new \external_value(PARAM_TEXT, 'conditions for completion'),
|
|
"slot" => new \external_value(PARAM_INT, 'slot in the study plan'),
|
|
"layer" => new \external_value(PARAM_INT, 'layer in the slot'),
|
|
"span" => new \external_value(PARAM_INT, 'how many periods the item spans'),
|
|
"course" => courseinfo::editor_structure(VALUE_OPTIONAL),
|
|
"badge" => badgeinfo::editor_structure(VALUE_OPTIONAL),
|
|
"continuation_id" => new \external_value(PARAM_INT, 'id of continued item'),
|
|
"connections" => new \external_single_structure([
|
|
'in' => new \external_multiple_structure(studyitemconnection::structure()),
|
|
'out' => new \external_multiple_structure(studyitemconnection::structure()),
|
|
]),
|
|
]);
|
|
|
|
}
|
|
|
|
/**
|
|
* Webservice model for editor info
|
|
* @return array Webservice data model
|
|
*/
|
|
public function editor_model() {
|
|
return $this->generate_model("editor");
|
|
}
|
|
|
|
/**
|
|
* Create a model for the given type of operation
|
|
* @param string $mode One of [ 'editor', 'export']
|
|
*/
|
|
private function generate_model($mode) {
|
|
// Mode parameter is used to geep this function for both editor model and export model.
|
|
// (Export model results in fewer parameters on children, but is otherwise basically the same as this function).
|
|
global $DB;
|
|
|
|
$model = [
|
|
'id' => $this->r->id, // Id is needed in export model because of link references.
|
|
'type' => $this->valid() ? $this->r->type : self::INVALID,
|
|
'conditions' => $this->conditions(),
|
|
'slot' => $this->r->slot,
|
|
'layer' => $this->r->layer,
|
|
'span' => $this->r->span,
|
|
'continuation_id' => $this->r->continuation_id,
|
|
'connections' => [
|
|
"in" => [],
|
|
"out" => [],
|
|
]
|
|
];
|
|
if ($mode == "export") {
|
|
// Remove slot and layer.
|
|
unset($model["slot"]);
|
|
unset($model["layer"]);
|
|
unset($model["continuation_id"]);
|
|
$model["connections"] = []; // In export mode, connections is just an array of outgoing connections.
|
|
}
|
|
|
|
// Add course link if available.
|
|
$ci = $this->getcourseinfo();
|
|
if (isset($ci)) {
|
|
if ($mode == "export") {
|
|
$model['course'] = $ci->shortname();
|
|
} else {
|
|
$model['course'] = $ci->editor_model();
|
|
}
|
|
}
|
|
|
|
// Add badge info if available.
|
|
if (is_numeric($this->r->badge_id) && $DB->record_exists('badge', array('id' => $this->r->badge_id))) {
|
|
$badge = new \core_badges\badge($this->r->badge_id);
|
|
$badgeinfo = new badgeinfo($badge);
|
|
if ($mode == "export") {
|
|
$model['badge'] = $badgeinfo->name();
|
|
} else {
|
|
// Also supply a list of linked users, so the badgeinfo can give stats on .
|
|
// The amount issued, related to this studyplan.
|
|
$studentids = $this->studyline()->studyplan()->find_linked_userids();
|
|
$model['badge'] = $badgeinfo->editor_model($studentids);
|
|
}
|
|
}
|
|
|
|
if ($mode == "export") {
|
|
// Also export gradables.
|
|
$gradables = gradeinfo::list_studyitem_gradables($this);
|
|
if (count($gradables) > 0) {
|
|
$model["gradables"] = [];
|
|
foreach ($gradables as $g) {
|
|
$model["gradables"][] = $g->export_model();;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Add incoming and outgoing connection info.
|
|
$connout = studyitemconnection::find_outgoing($this->id);
|
|
|
|
if ($mode == "export") {
|
|
foreach ($connout as $c) {
|
|
$model["connections"][] = $c->to_id();
|
|
}
|
|
} else {
|
|
foreach ($connout as $c) {
|
|
$model['connections']['out'][$c->to_id()] = $c->model();
|
|
}
|
|
$connin = studyitemconnection::find_incoming($this->id);
|
|
foreach ($connin as $c) {
|
|
$model['connections']['in'][$c->from_id()] = $c->model();
|
|
}
|
|
}
|
|
|
|
return $model;
|
|
|
|
}
|
|
|
|
/**
|
|
* Add a new item
|
|
* @param array $fields Properties for study line ['line_id', 'type', 'layer', 'conditions', 'slot',
|
|
* 'competency_id', 'course_id', 'badge_id', 'continuation_id', 'span']
|
|
* @param bool $import Set tot true if calling on import
|
|
*/
|
|
public static function add($fields, $import = false) : self {
|
|
global $DB;
|
|
$addable = ['line_id', 'type', 'layer', 'conditions', 'slot',
|
|
'competency_id', 'course_id', 'badge_id', 'continuation_id', 'span'];
|
|
$info = [ 'layer' => 0, ];
|
|
foreach ($addable as $f) {
|
|
if (array_key_exists($f, $fields)) {
|
|
$info[$f] = $fields[$f];
|
|
}
|
|
}
|
|
$id = $DB->insert_record(self::TABLE, $info);
|
|
$item = self::find_by_id($id);
|
|
if ($item->type() == self::COURSE) {
|
|
// Signal the studyplan that a course has been added so it can be marked for csync cascading.
|
|
$item->studyline()->studyplan()->mark_csync_changed();
|
|
}
|
|
return $item;
|
|
}
|
|
|
|
/**
|
|
* Edit study item properties
|
|
* @param array $fields Changed roperties for study line ['conditions', 'course_id', 'continuation_id', 'span']
|
|
*/
|
|
public function edit($fields) : self {
|
|
global $DB;
|
|
$editable = ['conditions', 'course_id', 'continuation_id', 'span'];
|
|
|
|
$info = ['id' => $this->id, ];
|
|
foreach ($editable as $f) {
|
|
if (array_key_exists($f, $fields) && isset($fields[$f])) {
|
|
$info[$f] = $fields[$f];
|
|
}
|
|
}
|
|
|
|
$DB->update_record(self::TABLE, $info);
|
|
// Reload record after edit.
|
|
$this->r = $DB->get_record(self::TABLE, ['id' => $this->id], "*", MUST_EXIST);
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Check if references course and badges are still available
|
|
*/
|
|
public function valid() : bool {
|
|
// Check if referenced courses and/or badges still exist.
|
|
if ($this->r->type == static::COURSE) {
|
|
return courseinfo::exists($this->r->course_id);
|
|
} else if ($this->r->type == static::BADGE) {
|
|
return badgeinfo::exists($this->r->badge_id);
|
|
} else {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Delete studyitem
|
|
* @param bool $force Force deletion even if item is referenced
|
|
*/
|
|
public function delete($force = false) {
|
|
global $DB;
|
|
|
|
// Check if this item is referenced in a START item.
|
|
if ($force) {
|
|
// Clear continuation id from any references to this item.
|
|
$records = $DB->get_records(self::TABLE, ['continuation_id' => $this->id]);
|
|
foreach ($records as $r) {
|
|
$r->continuation_id = 0;
|
|
$DB->update_record(self::TABLE, $r);
|
|
}
|
|
}
|
|
|
|
if ($DB->count_records(self::TABLE, ['continuation_id' => $this->id]) > 0) {
|
|
return success::fail('Cannot remove: item is referenced by another item');
|
|
} else {
|
|
// Delete al related connections to this item.
|
|
studyitemconnection::clear($this->id);
|
|
// Delete all grade inclusion references to this item.
|
|
$DB->delete_records("local_treestudyplan_gradeinc", ['studyitem_id' => $this->id]);
|
|
// Delete the item itself.
|
|
$DB->delete_records(self::TABLE, ['id' => $this->id]);
|
|
|
|
return success::success();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Reposition study items in line, layer and/or slot
|
|
* @param mixed $resequence Array of item info [id, line_id, slot, layer]
|
|
*/
|
|
public static function reorder($resequence) : success {
|
|
global $DB;
|
|
|
|
foreach ($resequence as $sq) {
|
|
// Only change line_id if new line is within the same studyplan page.
|
|
if ( self::find_by_id($sq['id'])->studyline()->page()->id() ==
|
|
studyline::find_by_id($sq['line_id'])->page()->id() ) {
|
|
$DB->update_record(self::TABLE, [
|
|
'id' => $sq['id'],
|
|
'line_id' => $sq['line_id'],
|
|
'slot' => $sq['slot'],
|
|
'layer' => $sq['layer'],
|
|
]);
|
|
}
|
|
}
|
|
|
|
return success::success();
|
|
}
|
|
|
|
/**
|
|
* Find all studyitems associated with a studyline
|
|
* @param studyline $line The studyline to search for
|
|
* @return studyitem[]
|
|
*/
|
|
public static function find_studyline_children(studyline $line) : array {
|
|
global $DB;
|
|
$list = [];
|
|
$ids = $DB->get_fieldset_select(self::TABLE, "id", "line_id = :line_id ORDER BY layer", ['line_id' => $line->id()]);
|
|
foreach ($ids as $id) {
|
|
$item = self::find_by_id($id, $line);
|
|
$list[] = $item;
|
|
}
|
|
return $list;
|
|
}
|
|
|
|
/**
|
|
* Webservice structure for linking between plans
|
|
* @param int $value Webservice requirement constant
|
|
*/
|
|
private static function link_structure($value = VALUE_REQUIRED) : \external_description {
|
|
return new \external_single_structure([
|
|
"id" => new \external_value(PARAM_INT, 'id of study item'),
|
|
"type" => new \external_value(PARAM_TEXT, 'type of study item'),
|
|
"completion" => completion::structure(),
|
|
"studyline" => new \external_value(PARAM_TEXT, 'reference label of studyline'),
|
|
"studyplan" => new \external_value(PARAM_TEXT, 'reference label of studyplan'),
|
|
], 'basic info of referenced studyitem', $value);
|
|
}
|
|
|
|
/**
|
|
* Webservice model for linking between plans
|
|
* @param int $userid ID for user to links completion from
|
|
*/
|
|
private function link_model($userid) : array {
|
|
global $DB;
|
|
$line = $DB->get_record(studyline::TABLE, ['id' => $this->r->line_id]);
|
|
$plan = $DB->get_record(studyplan::TABLE, ['id' => $line->studyplan_id]);
|
|
|
|
return [
|
|
"id" => $this->r->id,
|
|
"type" => $this->r->type,
|
|
"completion" => $this->completion($userid),
|
|
"studyline" => $line->name(),
|
|
"studyplan" => $plan->name(),
|
|
];
|
|
}
|
|
|
|
/**
|
|
* Webservice structure for userinfo
|
|
* @param int $value Webservice requirement constant
|
|
*/
|
|
public static function user_structure($value = VALUE_REQUIRED) : \external_description {
|
|
return new \external_single_structure([
|
|
"id" => new \external_value(PARAM_INT, 'id of study item'),
|
|
"type" => new \external_value(PARAM_TEXT, 'type of study item'),
|
|
"completion" => new \external_value(PARAM_TEXT, 'completion state (incomplete|progress|completed|excellent)'),
|
|
"slot" => new \external_value(PARAM_INT, 'slot in the study plan'),
|
|
"layer" => new \external_value(PARAM_INT, 'layer in the slot'),
|
|
"span" => new \external_value(PARAM_INT, 'how many periods the item spans'),
|
|
"course" => courseinfo::user_structure(VALUE_OPTIONAL),
|
|
"badge" => badgeinfo::user_structure(VALUE_OPTIONAL),
|
|
"continuation" => self::link_structure(VALUE_OPTIONAL),
|
|
"connections" => new \external_single_structure([
|
|
'in' => new \external_multiple_structure(studyitemconnection::structure()),
|
|
'out' => new \external_multiple_structure(studyitemconnection::structure()),
|
|
]),
|
|
], 'Study item info', $value);
|
|
|
|
}
|
|
|
|
/**
|
|
* Webservice model for user info
|
|
* @param int $userid ID of user to check specific info for
|
|
* @return array Webservice data model
|
|
*/
|
|
public function user_model($userid) {
|
|
global $CFG, $DB;
|
|
|
|
$model = [
|
|
'id' => $this->r->id,
|
|
'type' => $this->r->type,
|
|
'completion' => completion::label($this->completion($userid)),
|
|
'slot' => $this->r->slot,
|
|
'layer' => $this->r->layer,
|
|
'span' => $this->r->span,
|
|
'connections' => [
|
|
"in" => [],
|
|
"out" => [],
|
|
]
|
|
];
|
|
|
|
// Add badge info if available.
|
|
if (badgeinfo::exists($this->r->badge_id)) {
|
|
$badge = new \core_badges\badge($this->r->badge_id);
|
|
$badgeinfo = new badgeinfo($badge);
|
|
$model['badge'] = $badgeinfo->user_model($userid);
|
|
}
|
|
|
|
// Add continuation_info if available.
|
|
if (self::exists($this->r->continuation_id)) {
|
|
$citem = self::find_by_id($this->r->continuation_id);
|
|
$model['continuation'] = $citem->link_model($userid);
|
|
}
|
|
|
|
// Add course if available.
|
|
if (courseinfo::exists($this->r->course_id)) {
|
|
$cinfo = $this->getcourseinfo();
|
|
$model['course'] = $cinfo->user_model($userid);
|
|
}
|
|
|
|
// Add incoming and outgoing connection info.
|
|
$connout = studyitemconnection::find_outgoing($this->id);
|
|
foreach ($connout as $c) {
|
|
$model['connections']['out'][$c->to_id()] = $c->model();
|
|
}
|
|
$connin = studyitemconnection::find_incoming($this->id);
|
|
foreach ($connin as $c) {
|
|
$model['connections']['in'][$c->from_id()] = $c->model();
|
|
}
|
|
|
|
return $model;
|
|
|
|
}
|
|
|
|
/**
|
|
* Get courseinfo for studyitem if it references a course
|
|
*/
|
|
public function getcourseinfo() : ?courseinfo {
|
|
if (empty($this->courseinfo) && courseinfo::exists($this->r->course_id)) {
|
|
$this->courseinfo = new courseinfo($this->r->course_id, $this);
|
|
}
|
|
return $this->courseinfo;
|
|
}
|
|
|
|
/**
|
|
* Determine completion for a particular user
|
|
* @param int $userid User id
|
|
* @return int completion:: constant
|
|
*/
|
|
public function completion($userid) : int {
|
|
global $DB;
|
|
|
|
if ($this->valid()) {
|
|
if (strtolower($this->r->type) == 'course') {
|
|
// Determine competency by competency completion.
|
|
$courseinfo = $this->getcourseinfo();
|
|
return $this->aggregator->aggregate_course($courseinfo, $this, $userid);
|
|
} else if (strtolower($this->r->type) == 'start') {
|
|
// Does not need to use aggregator.
|
|
// Either true, or the completion of the reference.
|
|
if (self::exists($this->r->continuation_id)) {
|
|
$citem = self::find_by_id($this->r->continuation_id);
|
|
return $citem->completion($userid);
|
|
} else {
|
|
return completion::COMPLETED;
|
|
}
|
|
} else if (in_array(strtolower($this->r->type), ['junction', 'finish'])) {
|
|
// Completion of the linked items, according to the rule.
|
|
$incomingcompletions = [];
|
|
// Retrieve incoming connections.
|
|
$incoming = $DB->get_records(studyitemconnection::TABLE, ['to_id' => $this->r->id]);
|
|
foreach ($incoming as $conn) {
|
|
$item = self::find_by_id($conn->from_id);
|
|
$incomingcompletions[] = $item->completion($userid);
|
|
}
|
|
return $this->aggregator->aggregate_junction($incomingcompletions, $this, $userid);
|
|
} else if (strtolower($this->r->type) == 'badge') {
|
|
global $DB;
|
|
// Badge awarded.
|
|
if (badgeinfo::exists($this->r->badge_id)) {
|
|
$badge = new \core_badges\badge($this->r->badge_id);
|
|
if ($badge->is_issued($userid)) {
|
|
if ($badge->can_expire()) {
|
|
// Get the issued badges and check if any of them have not expired yet.
|
|
$badgesissued = $DB->get_records("badge_issued",
|
|
["badge_id" => $this->r->badge_id,
|
|
"user_id" => $userid]);
|
|
$notexpired = false;
|
|
$now = time();
|
|
foreach ($badgesissued as $bi) {
|
|
if ($bi->dateexpire == null || $bi->dateexpire > $now) {
|
|
$notexpired = true;
|
|
break;
|
|
}
|
|
}
|
|
return ($notexpired) ? completion::COMPLETED : completion::INCOMPLETE;
|
|
} else {
|
|
return completion::COMPLETED;
|
|
}
|
|
} else {
|
|
return completion::INCOMPLETE;
|
|
}
|
|
} else {
|
|
return completion::INCOMPLETE;
|
|
}
|
|
} else {
|
|
// Return incomplete for other types.
|
|
return completion::INCOMPLETE;
|
|
}
|
|
} else {
|
|
// Return incomplete for other types.
|
|
return completion::INCOMPLETE;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Duplicate this studyitem
|
|
* @param studyline $newline Study line to add duplicate to
|
|
*/
|
|
public function duplicate($newline) : self {
|
|
global $DB;
|
|
// Clone the database fields.
|
|
$fields = clone $this->r;
|
|
// Set new line id.
|
|
unset($fields->id);
|
|
$fields->line_id = $newline->id();
|
|
// Create new record with the new data.
|
|
$id = $DB->insert_record(self::TABLE, (array)$fields);
|
|
$new = self::find_by_id($id, $newline);
|
|
|
|
// Copy the grading info if relevant.
|
|
$gradables = gradeinfo::list_studyitem_gradables($this);
|
|
foreach ($gradables as $g) {
|
|
gradeinfo::include_grade($g->get_gradeitem()->id, $new->id(), true);
|
|
}
|
|
return $new;
|
|
}
|
|
|
|
/**
|
|
* Export essential information for export
|
|
* @return array information model
|
|
*/
|
|
public function export_model() {
|
|
return $this->generate_model("export");
|
|
}
|
|
|
|
|
|
/**
|
|
* Import studyitems from model
|
|
* @param array $model Decoded array
|
|
*/
|
|
public static function import_item($model) : self {
|
|
unset($model["course_id"]);
|
|
unset($model["competency_id"]);
|
|
unset($model["badge_id"]);
|
|
unset($model["continuation_id"]);
|
|
if (isset($model["course"])) {
|
|
$model["course_id"] = courseinfo::id_from_shortname(($model["course"]));
|
|
}
|
|
if (isset($model["badge"])) {
|
|
$model["badge_id"] = badgeinfo::id_from_name(($model["badge"]));
|
|
}
|
|
|
|
$item = self::add($model, true);
|
|
|
|
if (isset($model["course_id"])) {
|
|
// Attempt to import the gradables.
|
|
foreach ($model["gradables"] as $gradable) {
|
|
gradeinfo::import($item, $gradable);
|
|
}
|
|
}
|
|
|
|
return $item;
|
|
}
|
|
|
|
}
|