You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
			
				
					
					
						
							312 lines
						
					
					
						
							8.9 KiB
						
					
					
				
			
		
		
		
			
			
			
				
					
				
				
					
				
			
		
		
	
	
							312 lines
						
					
					
						
							8.9 KiB
						
					
					
				| <?php | |
| // 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 <http://www.gnu.org/licenses/>. | |
|  | |
| defined('MOODLE_INTERNAL') || die(); | |
| 
 | |
| /** | |
|  * The class CryptexDB loads/save the cryptex from/to database. | |
|  * | |
|  * @package    mod_game | |
|  * @copyright  2007 Vasilis Daloukas | |
|  * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later | |
|  */ | |
| class CryptexDB extends CrossDB { | |
|     /** | |
|      * Save cryptex. | |
|      * | |
|      * @param stdClass $game | |
|      * @param array &$crossm | |
|      * @param array &$crossd | |
|      * @param int id | |
|      * @param $letters | |
|      * | |
|      * @return the saved record | |
|      */ | |
|     public function savecryptex( $game, &$crossm, $crossd, $id, $letters) { | |
|         global $USER; | |
| 
 | |
|         CrossDB::delete_records( $id); | |
| 
 | |
|         if ((CrossDB::savecross( $game, $crossm, $crossd, $id)) == false) { | |
|             return false; | |
|         } | |
| 
 | |
|         $crossm->id = $id; | |
| 
 | |
|         $newrec = new stdClass(); | |
|         $newrec->id = $id; | |
|         $newrec->letters = $letters; | |
| 
 | |
|         if (!($cryptexid = game_insert_record( "game_cryptex", $newrec))) { | |
|             print_error( 'Insert page: new page game_cryptex not inserted'); | |
|         } | |
| 
 | |
|         return $newrec; | |
|     } | |
| 
 | |
|     /** | |
|      * Compute letters. | |
|      * | |
|      * @param array &$crossm | |
|      * @param array &$crossd | |
|      * | |
|      * @return the letters. | |
|      */ | |
|     public function computeletters( $crossm, $crossd) { | |
|         $letters = ''; | |
|         $cols = $crossm->cols + 1; | |
|         $letters = str_repeat('.', $crossm->cols).'#'; | |
|         $letters = str_repeat($letters, $crossm->rows); | |
| 
 | |
|         $freqs1 = array(); | |
|         $count1 = $count2 = 0; | |
|         foreach ($crossd as $rec) { | |
|             $pos = $rec->col - 1 + ($rec->row - 1) * $cols; | |
|             $s = $rec->answertext; | |
|             $len = game_strlen( $s); | |
| 
 | |
|             $a = array(); | |
|             for ($i = 0; $i < $len; $i++) { | |
|                 $a[] = game_substr( $s, $i, 1); | |
|             } | |
| 
 | |
|             for ($i = 0; $i < $len; $i++) { | |
|                 $this->setchar( $letters, $pos,  $a[ $i]); | |
|                 $pos += ( $rec->horizontal ? 1 : $cols); | |
| 
 | |
|                 $freqs1[ ++$count1] = $a[ $i]; | |
|                 if ($i + 1 < $len) { | |
|                     $freqs2[ ++$count2] = $a[ $i].$a[ $i + 1]; | |
|                 } | |
|             } | |
|         } | |
| 
 | |
|         $len = game_strlen( $letters); | |
|         $spaces = 0; | |
|         for ($i = 0; $i < $len; $i++) { | |
|             if (game_substr( $letters, $i, 1) == '.') { | |
|                 $spaces++; | |
|             } | |
|         } | |
| 
 | |
|         $step = 1; | |
|         while ($spaces) { | |
|             if ($step == 1) { | |
|                 $step = 2; | |
|                 $i = array_rand( $freqs1); | |
|                 $this->insertchar( $letters, $crossm->cols, $crossm->rows, $freqs1[ $i], $spaces); | |
|             } else { | |
|                 $step = 1; | |
|                 $i = array_rand( $freqs2); | |
|                 $this->insertchars( $letters, $crossm->cols, $crossm->rows, $freqs2[ $i], $spaces); | |
|             } | |
|         } | |
| 
 | |
|         $retletters = ""; | |
|         for ($row = 0; $row < $crossm->rows; $row++) { | |
|             $retletters .= game_substr( $letters, $cols * $row, ($cols - 1)); | |
|         } | |
| 
 | |
|         return $retletters; | |
|     } | |
| 
 | |
|     /** | |
|      * Displays the cryptex. | |
|      * | |
|      * @param $cols | |
|      * @param $rows | |
|      * @param $letters | |
|      * @param $mask | |
|      * @param boolean $showsolution | |
|      * @param $textdir | |
|      */ | |
|     public function displaycryptex( $cols, $rows, $letters, $mask, $showsolution, $textdir) { | |
|         echo "<table border=1 $textdir>"; | |
|         for ($row = 0; $row < $rows; $row++) { | |
|             echo "<tr>"; | |
|             for ($col = 0; $col < $cols; $col++) { | |
|                 $pos = $cols * $row + $col; | |
|                 $c = game_substr( $letters, $pos, 1); | |
|                 $m = game_substr( $mask, $pos, 1); | |
| 
 | |
|                 if ($showsolution and $m > '0') { | |
|                     echo "<td align=center><b><FONT color=red>".$c."</font></td>"; | |
|                 } else if ( $m == '1') { | |
|                     echo "<td align=center><b><FONT color=red>".$c."</font></td>"; | |
|                 } else { | |
|                     echo "<td align=center>".$c."</td>"; | |
|                 } | |
|             } | |
|             echo "</tr>\r\n"; | |
|         } | |
|         echo "</table>"; | |
|     } | |
| 
 | |
|     /** | |
|      * Inserts a char. | |
|      * | |
|      * @param $letters | |
|      * @param $cols | |
|      * @param $rows | |
|      * @param $char | |
|      * @param int &$spaces | |
|      */ | |
|     public function insertchar( &$letters, $cols, $rows, $char, &$spaces) { | |
|         $len = game_strlen( $letters); | |
|         for ($i = 0; $i < $len; $i++) { | |
|             if (game_substr( $letters, $i, 1) == '.') { | |
|                 $this->setchar( $letters, $i, $char); | |
|                 $spaces--; | |
|                 return; | |
|             } | |
|         } | |
|     } | |
| 
 | |
|     /** | |
|      * Inserts chars. | |
|      * | |
|      * @param $letters | |
|      * @param $cols | |
|      * @param $rows | |
|      * @param $char | |
|      * @param int &$spaces | |
|      */ | |
|     public function insertchars( &$letters, $cols, $rows, $char, &$spaces) { | |
|         $len = game_strlen( $letters); | |
|         for ($i = 0; $i < $len; $i++) { | |
|             if (game_substr( $letters, $i, 1) == '.'  and game_substr( $letters, $i + 1, 1) == '.' ) { | |
|                 $this->setchar( $letters, $i, game_substr( $char, 0, 1)); | |
|                 $this->setchar( $letters, $i + 1, game_substr( $char, 1, 1)); | |
|                 $spaces -= 2; | |
|                 return true; | |
|             } | |
|             if (game_substr( $letters, $i, 1) == '.' and game_substr( $letters, $i + $cols + 1, 1) == '.' ) { | |
|                 $this->setchar( $letters, $i, game_substr( $char, 0, 1)); | |
|                 $this->setchar( $letters, $i + $cols + 1, game_substr( $char, 1, 1)); | |
|                 $spaces -= 2; | |
|                 return true; | |
|             } | |
|         } | |
| 
 | |
|         return false; | |
|     } | |
| 
 | |
|     /** | |
|      * Gets the hash of a word. | |
|      * | |
|      * @param string $word | |
|      * | |
|      * @return the hash | |
|      */ | |
|     public function gethash( $word) { | |
|         $x = 37; | |
|         $len = count( game_strlen( $word)); | |
| 
 | |
|         for ($i = 0; $i < $len; $i++) { | |
|             $x = $x xor ord( game_substr( $word, $i, 1)); | |
|         } | |
| 
 | |
|         return $x; | |
|     } | |
| 
 | |
|     /** | |
|      * Loads the cryptex from database. | |
|      * | |
|      * @param array $crossm | |
|      * @param &$mask | |
|      * @param &$corrects | |
|      * @param &$language | |
|      * | |
|      * @return questions | |
|      */ | |
|     public function loadcryptex( $crossm, &$mask, &$corrects, &$language) { | |
|         global $DB; | |
| 
 | |
|         $questions = array(); | |
|         $corrects = array(); | |
| 
 | |
|         $mask = str_repeat( '0', $crossm->cols * $crossm->rows); | |
| 
 | |
|         if ($recs = $DB->get_records( 'game_queries', array( 'attemptid' => $crossm->id))) { | |
|             foreach ($recs as $rec) { | |
|                 if ($rec->questiontext == '') { | |
|                     $rec->questiontext = ' '; | |
|                 } | |
|                 $key = $this->gethash( $rec->questiontext).'-'.$rec->answertext.'-'.$rec->id; | |
|                 $questions[ $key] = $rec; | |
| 
 | |
|                 $word = $rec->answertext; | |
|                 $pos = $crossm->cols * ($rec->row - 1) + ($rec->col - 1); | |
|                 $len = game_strlen( $word); | |
|                 $found = ($rec->answertext == $rec->studentanswer); | |
| 
 | |
|                 for ($i = 0; $i < $len; $i++) { | |
|                     $c = ( $found ? '1' : '2'); | |
| 
 | |
|                     if (game_substr( $mask, $pos,  1) != '1') { | |
|                         game_setchar( $mask, $pos, $c); | |
|                     } | |
| 
 | |
|                     $pos += ($rec->horizontal ? 1 : $crossm->cols); | |
|                 } | |
| 
 | |
|                 if ($found) { | |
|                     $corrects[ $rec->id] = 1; | |
|                 } | |
| 
 | |
|                 if ($language == '') { | |
|                     $language = game_detectlanguage( $rec->answertext); | |
|                 } | |
|             } | |
|             ksort( $questions); | |
|         } | |
| 
 | |
|         return $questions; | |
|     } | |
| 
 | |
|     /** | |
|      * Calss the setwords of class Cross. | |
|      * | |
|      * @param $answers | |
|      * @param $maxcols | |
|      * @param $reps | |
|      * | |
|      * @return Cross::setwords | |
|      */ | |
|     public function setwords( $answers, $maxcols, $reps) { | |
|         return Cross::setwords( $answers, $maxcols, $reps); | |
|     } | |
| 
 | |
|     /** | |
|      * Calss the computedata of class Cross. | |
|      * | |
|      * @param &$crossm | |
|      * @param &$crossd | |
|      * @param &$letters | |
|      * @param $minwords | |
|      * @param $maxwords | |
|      * @param $mtimelimit | |
|      */ | |
|     public function computedata( &$crossm, &$crossd, &$letters, $minwords, $maxwords, $mtimelimit=3) { | |
|         if (!cross::computedata( $crossm, $crossd, $letters, $minwords, $maxwords, $mtimelimit)) { | |
|             return false; | |
|         } | |
| 
 | |
|         $letters = $this->computeletters( $crossm, $crossd); | |
| 
 | |
|         return true; | |
|     } | |
| }
 | |
| 
 |